gpt4 book ai didi

从 PCM 数据创建 .wav 文件

转载 作者:行者123 更新时间:2023-11-30 14:44:42 40 4
gpt4 key购买 nike

我正在尝试从 SPH0645LM4H-B https://www.knowles.com/docs/default-source/model-downloads/sph0645lm4h-b-datasheet-rev-c.pdf?sfvrsn=4 读取数据并将其转换为 .wav 格式,以便我可以验证我的传感器是否正常工作。到目前为止,我已经成功连接传感器并通过 i2s-dma 传输读取数据样本。

此外,我发现大多数主流计算机声音文件格式都包含一个 header (包含长度等),后跟 16 位补码 PCM https://www.dsprelated.com/freebooks/mdft/Number_Systems_Digital_Audio.html

为了将PCM数据转换为波形格式,我引用了converting PCM to wav file

现在,传感器数据格式为I2S、24 位、2 的补码、MSB 在前。数据精度为18位;未使用的位为零。我已将传感器数据转换为32位(MSB是符号位)

我已使用以下链接(以及其中提到的链接)中的代码来创建 .wav 文件:- Append .pcm audio raw data into wav (in C)

//.wav文件头数据

struct wavfile
{
char id[4]; // should always contain "RIFF"
int totallength; // total file length minus 8
char wavefmt[8]; // should be "WAVEfmt "
int format; // 16 for PCM format
short pcm; // WAVE_FORMAT_IEEE_FLOAT (3).
short channels; // channels
int frequency; // sampling frequency, 16000 in this case
int bytes_per_second;
short bytes_by_capture;
short bits_per_sample;
char data[4]; // should always contain "data"
int bytes_in_data;
};

//写入头文件输出.wav文件

void write_wav_header(char* name, int samples, int channels)
{
struct wavfile filler;
FILE *pFile;
strcpy(filler.id, "RIFF");
filler.totallength = (samples * channels) + sizeof(struct wavfile) - 8;
strcpy(filler.wavefmt, "WAVEfmt ");
filler.format = 16;
filler.pcm = 3; // WAVE_FORMAT_IEEE_FLOAT (3).
filler.channels = channels;
filler.frequency = 32000;
filler.bits_per_sample = 32;
filler.bytes_per_second = filler.channels * filler.frequency * filler.bits_per_sample/8;
filler.bytes_by_capture = filler.channels*filler.bits_per_sample/8;
filler.bytes_in_data = samples * filler.channels * filler.bits_per_sample/8;
strcpy(filler.data, "data");
pFile = fopen(name, "wb");
fwrite(&filler, 1, sizeof(filler), pFile);
fclose(pFile);
}

//将音频传感器数据附加到此 .wav 文件

void write_pcm_data_to_file(char* inFile, char* outFile)
{
char buffer[SAMPLE_SIZE];
size_t n;
FILE *fin,*fout;
fin = fopen(inFile,"r");
fout = fopen(outFile,"a");
while((n = fread(buffer, 1, sizeof(buffer), fin)) > 0)
{
if(n != fwrite(buffer, 1, n, fout))
{
perror("fwrite");
exit(1);
}
}
fclose(fin);
fclose(fout);
}

这是生成的 .wav 文件在十六进制编辑器中的外观:- .wav file in hex editor我可以看到 wav header 值设置正确,后面是来自音频传感器的 PCM 数据。

但是,当我播放 .wav 文件时,我无法播放输入音频。相反,我听到了持续的声音。我尝试更改输入音频(播放不同的音乐、歌曲),但生成的 .wav 文件播放相同的恒定音调。

为了将麦克风播放的输入音乐重新创建为 wav 文件,我应该做什么修改?

提前致谢。

编辑:-

根据 42LeapsOfFaith 的回答,为了将我的样本转换为十六进制,我使用了以下代码:-

hexValue = strtoll(sample, NULL, 16);

我转换了缓冲区中的每个值,然后将其写入我的 .wav 文件中。现在我的 .wav 文件看起来像 modified .wav file in hex editor

但是,即使是这个 wav 文件也无法播放音频。对于将麦克风播放的输入音乐重新创建为 wav 文件,还有其他建议吗?

非常感谢您的帮助

最佳答案

对于初学者,您将 PCM 数据以 ascii-hex 形式存储在文件中。我希望这会有所帮助。

这是一个你可以使用的“黑客”...

char c[2];
while((n = fread(&c[0], 2, 1, fin)) > 0)
{
if (c[0] > 0x39) c[0] -= 7;
c[0] &= 0x0F;
if (c[1] > 0x39) c[1] -= 7;
c[1] &= 0x0F;
c[1] |= c[0] << 4;
if(1 != fwrite(&c[1], 1, 1, fout))

关于从 PCM 数据创建 .wav 文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53417630/

40 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com