ALSA принимает 1024 буфера перед прерыванием

Я пытаюсь заставить мой интерфейс ALSA работать с минимально возможной задержкой. Документация ALSA обо всех кадрах / периодах / буферах очень запутанная, поэтому я спрашиваю здесь.

Приведенный ниже код заставляет ALSA фактически читать в 1024 раза больше размера буфера (я установил размер буфера в 1024 байта). writeAudio Функция занимает 1024 * 1024 байта, прежде чем она замедляется. Я хотел бы сохранить эти кадры? считать как можно ниже, чтобы я мог отслеживать время воспроизведения в самом приложении. Если я попытаюсь установить размер периода на 2 с помощью snd_pcm_hw_params_set_periods (Я полагаю, что это замедлит чтение после того, как 2 * 1024 байта будет записано в буфер.) Однако это изменение в коде не меняет поведение; плеер по-прежнему буферизует 1024 * 1024 байта до того, как буферизация замедлится до скорости воспроизведения звука из динамиков.

TLDR; Размер буферизации 1024 * 1024 байта — это слишком много для меня, как уменьшить его? Ниже мой код. И да, я играю без знака 8 бит с моно выходом.

int32_t ALSAPlayer::initPlayer(ALSAConfig cfg)
{
std::cout << "=== INITIALIZING ALSA ===" << std::endl;

if(!cfg.channels || !cfg.rate)
{
std::cout << "ERROR: player config was bad" << std::endl;
return -1;
}

m_channels = cfg.channels;

m_rate = cfg.rate;

m_frames = 1024;

uint32_t tmp;
uint32_t buff_size;

int dir = 0;

/* Open the PCM device in playback mode */
if ((pcm = snd_pcm_open(&pcm_handle, PCM_DEVICE, SND_PCM_STREAM_PLAYBACK, 0)) < 0)
{
printf("ERROR: Can't open \"%s\" PCM device. %s\n", PCM_DEVICE, snd_strerror(pcm));
}
snd_pcm_hw_params_alloca(&params);

snd_pcm_hw_params_any(pcm_handle, params);

if ((pcm = snd_pcm_hw_params_set_access(pcm_handle, params, SND_PCM_ACCESS_RW_INTERLEAVED)) < 0)
{
printf("ERROR: Can't set interleaved mode. %s\n", snd_strerror(pcm));
}

if ((pcm = snd_pcm_hw_params_set_format(pcm_handle, params, SND_PCM_FORMAT_S8)) < 0)
{
printf("ERROR: Can't set format. %s\n", snd_strerror(pcm));
}

if ((pcm = snd_pcm_hw_params_set_channels(pcm_handle, params, m_channels)) < 0)
{
printf("ERROR: Can't set channels number. %s\n", snd_strerror(pcm));
}

if ((pcm = snd_pcm_hw_params_set_rate_near(pcm_handle, params, &m_rate, &dir)) < 0)
{
printf("ERROR: Can't set rate. %s\n", snd_strerror(pcm));
}

// force the ALSA interface to use exactly *m_frames* number of frames

snd_pcm_hw_params_set_period_size(pcm_handle, params, m_frames, dir);

/* Write parameters */
if ((pcm = snd_pcm_hw_params(pcm_handle, params)) < 0)
{
printf("ERROR: Can't set harware parameters. %s\n", snd_strerror(pcm));
}

std::cout << "ALSA output device name:        " << snd_pcm_name(pcm_handle) << std::endl;
std::cout << "ALSA output device state:       " << snd_pcm_state_name(snd_pcm_state(pcm_handle)) << std::endl;

snd_pcm_hw_params_get_channels(params, &tmp);
std::cout << "ALSA output device channels:    " << tmp << std::endl;

snd_pcm_hw_params_get_rate(params, &tmp, 0);
std::cout << "ALSA output device rate:        " << tmp << std::endl;

snd_pcm_hw_params_get_period_size(params, &m_frames, &dir);

buff_size = m_frames * m_channels;

std::cout << "ALSA output device frames size: " << m_frames << std::endl;

std::cout << "ALSA output device buffer size: " << buff_size << "(should be 1024)" << std::endl;

return 0;
}

int ALSAPlayer::writeAudio(byte* buffer, uint32_t buffSize)
{
int pcmRetVal;
if(buffSize == 0)
{
snd_pcm_drain(pcm_handle);
snd_pcm_close(pcm_handle);
return -1;
}

if((pcmRetVal = snd_pcm_writei(pcm_handle, buffer, m_frames)) == -EPIPE)
{
snd_pcm_prepare(pcm_handle);
}
else if(pcm < 0)
{
std::cout << "ERROR: could not write to audio interface" << std::endl;
}
return 0;
}

0

Решение

(Один кадр не обязательно один байт; пожалуйста, не путайте их.)

Этот код не устанавливает размер буфера.

snd_pcm_hw_params_set_periods() устанавливает количество периодов.
snd_pcm_hw_params_set_period_size() установил бы размер периода.

Чтобы иметь 2048-кадровый буфер с двумя периодами по 1024 кадра каждый, установите количество периодов равным 2, а размер периодов — 1024.

Вы должны проверить все вызовы функций на наличие ошибок, в том числе snd_pcm_hw_params_set_period_size(),

1

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]