在 C 中,如何将 1 通道 PCM 转换为 2 通道 PCM?

In C, how do I convert 1-channel PCM into 2-channel PCM?

我正在使用 libao 和 libsndfile 来读取和播放音频。我想通过将一个通道复制为两个来将单声道流转换为立体声流。此测试代码将正确播放立体声剪辑,但会非常快速和高音播放单声道剪辑。此外,我在 free(output); 呼叫中收到 "double free or corruption"。我做错了什么?

/* Converting a 1-channel stream to 2-channels
 * compile with "gcc -o stereoize stereoize.c -lao -lsndfile"
 *
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <limits.h>
#include <ao/ao.h>
#include <sndfile.h>

#define BUFFSIZE 512

void stereoize(short *, short *, size_t);
void playfile(FILE *);

int main(int argc, char *argv[])
{
    FILE *fp;

    if (argc != 2) {
        printf("usage: %s <filename>\n", argv[0]);
    exit(1);
    }

    fp = fopen(argv[1], "rb");
    if (fp == NULL) {
        printf("Cannot open %s.\n", argv[1]);
    exit(2);
    }

    playfile(fp);
    fclose(fp);
    printf("Finished.\n");

    return 0;
}

void playfile(FILE *fp)
{
    int default_driver;
    int frames_read;
    int count;
    int toread;
    short *buffer;
    short *output;

    ao_device *device;
    ao_sample_format format;

    SNDFILE     *sndfile;
    SF_INFO     sf_info;


    ao_initialize();
    default_driver = ao_default_driver_id();

    sf_info.format = 0;

    sndfile = sf_open_fd(fileno(fp), SFM_READ, &sf_info, 0);

    memset(&format, 0, sizeof(ao_sample_format));

    format.byte_format = AO_FMT_NATIVE;
    format.bits = 16;
    format.rate = sf_info.samplerate;
    //    format.channels = sf_info.channels;
    format.channels = 2;

    printf("Channels: %d\n", sf_info.channels);
    printf("Samplerate: %d\n", sf_info.samplerate);

    if (sf_info.channels > 2 || sf_info.channels < 1) {
        printf("Sorry.  Only 1 or 2 channels, please.\n");
        exit(1);
    }
    device = ao_open_live(default_driver, &format, NULL);
    if (device == NULL) {
        printf("Error opening sound device.\n");
        exit(1);
    }

    buffer = malloc(BUFFSIZE * sf_info.channels * sizeof(short));
    output = malloc(BUFFSIZE * 2 * sizeof(short));

    frames_read = 0;
    toread = sf_info.frames * sf_info.channels;

    while (toread > 0) {
    if (toread < BUFFSIZE * sf_info.channels)
        count = toread;
    else
        count = BUFFSIZE * sf_info.channels;

        frames_read = sf_read_short(sndfile, buffer, count);

    if (sf_info.channels == 1)
        stereoize(output, buffer, count);
    else
        memcpy(output, buffer, count * sizeof(short));

    if (sf_info.channels == 1)
            ao_play(device, (char *)output, 2 * frames_read * sizeof(short));
    else
            ao_play(device, (char *)output, frames_read * sizeof(short));

    toread = toread - frames_read;
    }

    free(buffer);
    free(output);
    ao_close(device);
    sf_close(sndfile);
    ao_shutdown();

    return;
}

void stereoize(short *outbuf, short *inbuf, size_t length)
{
    int count;
    int outcount;

    outcount = 0;
    for (count = 0; count < length; count++) {
        outbuf[outcount] = outbuf[outcount+1] = inbuf[count];
        outcount += 2;
    }
}

语句 stereoize(output, buffer, count * sizeof(short)); 很可能是导致问题的原因。 stereoize 函数需要样本数作为 length 参数,但您传递的是总块大小(以字节为单位)。改成stereoize(output, buffer, count);,看看能不能解决问题。 另请查看@Joachim Pileborg 的评论。