实时从数据创建 WAV 文件 - 需要帮助

Creating WAV file from data in real time - need assist

我对音频处理很陌生,我的C++编程水平也很基础。我必须创建录音功能,并以回调的形式从设备接收数据信号。我希望有人能用基本理论帮助我,因为我迷路了,我应该如何管理 WAV 内存。现在我的代码如下所示:

我目前使用的功能:

template <typename T>
void write(std::ofstream& stream, const T& t) {
  stream.write((const char*)&t, sizeof(T));
}

template <typename T>
void writeFormat(std::ofstream& stream) {
  write<short>(stream, 1);
}

template <>
void writeFormat<float>(std::ofstream& stream) {
  write<short>(stream, 3);
}

template <typename SampleType>
void writeWAVData(
  char const* outFile,
  SampleType* buf,
  size_t bufSize,
  int sampleRate,
  short channels)
{
  std::ofstream stream(outFile, std::ios::binary|ios::app|ios::ate);

  stream.write("RIFF", 4);                                        // Start writting RIFF
  write<int>(stream, 0);                                          // (file-size)-8 - FOR NOW IGNORED
  stream.write("WAVE", 4);                                        // File type

  stream.write("fmt ", 4);                                        // Start Writting format chunk "fmt"
  write<int>(stream, 16);                                         // Chunk Data Size 16 + extra format bytes
  writeFormat<SampleType>(stream);                                // Format (compression code)
  write<short>(stream, channels);                                 // Channels
  write<int>(stream, sampleRate);                                 // Sample Rate
  write<int>(stream, sampleRate * channels * sizeof(SampleType)); // Byterate (byte/per sec)
  write<short>(stream, channels * sizeof(SampleType));            // Frame size (block align)
  write<short>(stream, 8 * sizeof(SampleType));                   // Bits per sample 

  stream.write("data", 4);                                        // Start writting chunk for extra format bytes
  stream.write((const char*)&bufSize, 0);                         //  - FOR NOW IGNORED
  stream.write((const char*)buf, 0);                              //  - FOR NOW IGNORED
}

我想做的大体思路是:

  1. 创建 *.wav 文件。
  2. 写入header块。
  3. 只要用户不停止,就在文件末尾添加新数据。
  4. 当用户点击停止时,返回到 header 内存并使用正确的值编辑 "RIFF" 和 "data" 块下的字段。

第一个问题:所以我有缓冲区存储 256 个样本,所以在循环中,每 256 个我尝试将包含新数据的缓冲区写入文件,如下面的代码所示。阅读 *.wav documentaiton http://www.sonicspot.com/guide/wavefiles.html 后,我了解到 *.wav 文件在每个块的开头应该有 RIFF header。但是,如果我写入包含 256 个样本的缓冲区并在它之前放置 RIFF 块,那么如果我改为输入 512 个样本,会有什么不同?每个样本之前都需要 RIFF 块吗?

for (int j = 0 ; j < 256 ; j++)
{
    if (j == 255) 
        writeWAVData("mySound0.wav", (int*)bufferInfos[0].buffers[doubleBufferIndex], buffSize, 44100, 1);
}

其中:

第二个问题:关于通道参数,因为它有点混乱。我的信号由轮流接收的值组成:左平移、右平移、左平移、右平移...所以这意味着我有 2 个通道 - 左通道和右通道?

第三个问题:我的设备信号是 24 位的……我不应该只写每个样本 24 的位,或者它应该如何包含此信息。

RIFF header 每个文件在开头出现一次。您尝试包含多个。

通常,WAVE header 每个记录出现一次。 .WAV 格式允许多个,但这类似于专辑。

显然你确实有两个频道。

是的,如果每个样本有 24 位,那么您必须将其放入, 每个样本恰好写入 3 个字节。也就是说,

template <typename T>
void write(std::ofstream& stream, const T& t) {
  stream.write((const char*)&t, sizeof(T));
}

不太可能那样做。 sizeof(SampleType) 可能也不是 3。