0

我正在尝试使用下面的程序播放此声音,但声音有点快并且会跳过。声音文件的采样率为 11025 Hz,立体声,采样大小为 16 位。由于参数无效,问题似乎是snd_pcm_hw_params_set_buffer_size()and snd_pcm_hw_params_set_period_size()(两者都是我正在研究的函数),但我不知道为什么它们无效并且将它们注释掉并不能解决跳过效果。我究竟做错了什么?

#include <stdlib.h>
#include <stdint.h>
#include <alsa/asoundlib.h>

#define STEREO              2
#define BITS            16 / 8
#define FRAMEBUFFERSIZE   512 // in samples
#define PERIODS             2
#define SAMPLERATE      11025

void snd_error_checker(int error, char *function_name)
{
    if (error)
    {
        printf("Error (%s): %s\n", function_name, snd_strerror(error));
        // exit(EXIT_FAILURE);
    }
}

int main(void)
{
    snd_pcm_t *handle;
    uint32_t channels                   = STEREO;
    uint32_t sample_size                = BITS; // 16 bit
    uint32_t frame_size                 = sample_size * channels;
    snd_pcm_uframes_t frames            = FRAMEBUFFERSIZE / frame_size;
    snd_pcm_uframes_t frames_per_period = frames / PERIODS;
    int32_t dir = 0; // No idea what this does.
    snd_pcm_hw_params_t *params;
    int16_t *buffer;
    FILE *wav;
    int32_t size;
    int error;

    error = snd_pcm_open(&handle, "default", SND_PCM_STREAM_PLAYBACK, 0);
    snd_error_checker(error, "snd_pcm_open()");
    snd_pcm_hw_params_alloca(&params);
    error = snd_pcm_hw_params_any(handle, params);
    snd_error_checker(error, "snd_pcm_hw_params_any()");
    error = snd_pcm_hw_params_set_buffer_size(handle, params, frames);
    snd_error_checker(error, "snd_pcm_hw_params_set_buffer_size()");
    error = snd_pcm_hw_params_set_period_size(handle, params, frames_per_period, dir);
    snd_error_checker(error, "snd_pcm_hw_params_set_period_size()");
    error = snd_pcm_hw_params_set_rate(handle, params, SAMPLERATE, dir);
    snd_error_checker(error, "snd_pcm_hw_params_set_rate()");
    error = snd_pcm_hw_params_set_access(handle, params, SND_PCM_ACCESS_RW_INTERLEAVED);
    snd_error_checker(error, "snd_pcm_hw_params_set_access()");
    error = snd_pcm_hw_params_set_channels(handle, params, STEREO);
    snd_error_checker(error, "snd_pcm_hw_params_set_channels()");
    error = snd_pcm_hw_params_set_format(handle, params, SND_PCM_FORMAT_S16_LE);
    snd_error_checker(error, "snd_pcm_hw_params_set_format()");
    error = snd_pcm_hw_params(handle, params);
    snd_error_checker(error, "snd_pcm_hw_params()");

    // Insert samples to framebuffer.
    wav = fopen("pcm1611s.wav", "r");
    fseek(wav, 0L, SEEK_END);
    size = ftell(wav);
    fseek(wav, 0L, SEEK_SET);
    buffer = malloc(size);
    fread(buffer, sizeof(int16_t), size, wav);
    fclose(wav);
    size /= sizeof(int16_t);

    // Set ptrbuffer 46 bytes ahead to skip the header.
    for (int16_t *ptrbuffer = buffer + 46; size > ptrbuffer - buffer; ptrbuffer += FRAMEBUFFERSIZE * STEREO * BITS)
    {
        error = snd_pcm_writei(handle, ptrbuffer, frames);
        if (error)
        {
            snd_pcm_recover(handle, error, 0);
        }
    }
    snd_pcm_drain(handle);
    snd_pcm_close(handle);
    exit(EXIT_SUCCESS);
}

4

2 回答 2

1
#define FRAMEBUFFERSIZE   512 // in samples

评论在说谎;该值用作字节计数。

uint32_t sample_size                = BITS; // 16 bit

评论在说谎;该值实际上以字节为单位。(并且BITS命名错误。)

fread(buffer, sizeof(int16_t), size, wav);

size以字节为单位,但您告诉fread()读取 16 位字。(这实际上并没有伤害,因为它在文件末尾停止读取。但您应该检查错误。)

int16_t *ptrbuffer = ...

您告诉编译器ptrbuffer指向 16 位值。

ptrbuffer += FRAMEBUFFERSIZE * STEREO * BITS

您是在告诉编译器跳过 2048 个 16 位值,即超过 4096 个字节。您实际上想要遍历缓冲区中的所有样本,即frames * STEREO.

error = snd_pcm_writei(...);
if (error)

snd_pcm_writei()成功返回正数;错误是负面的。

于 2020-06-24T20:37:04.413 回答
1

正确播放需要考虑三点

  1. 检查的返回值snd_pcm_writei,将给出一个清晰的想法,如果有任何错误是怎么回事。参考alsa 项目

  2. 在循环中读取文件while()并给予可能会导致在运行不足(错误)snd_pcm_writei的情况下跳过音频,因为这是一个耗时的调用。-EPIPEfread

  3. 检查所有库调用的返回值并确保一切都成功。还要确保在驱动程序中正确设置这些值。比如snd_pcm_hw_params_set_rate用 验证后snd_pcm_hw_params_get_rate,如果你的驱动不支持11025KHz的采样率,我们可以用返回值查出来,再用snd_pcm_hw_params_get_rate

于 2020-06-24T11:47:03.160 回答