Плохое качество звука при записи звука с Alsa - PullRequest
0 голосов
/ 02 мая 2020

Я пытаюсь записать звук с помощью alsa (библиотека asoundlib) и записать его в файл .pcm.

Вот мой код, основанный на https://gist.github.com/albanpeignier/104902 и Audio запись с использованием ALSA в формате PCM

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

int main(int argc, char *argv[])
{
  int i;
  int err;
  char *buffer; //?
  int buffer_frames = 128; //?
  unsigned int rate = 44100; // Sample rate of 44,1 kHz

  snd_pcm_t *capture_handle;

  snd_pcm_hw_params_t *hw_params;

  //Signed 16 bit Little Endian 
  snd_pcm_format_t format = SND_PCM_FORMAT_S16_LE;


  if ((err = snd_pcm_open(&capture_handle, argv[1], SND_PCM_STREAM_CAPTURE, 0)) < 0) {
    fprintf(stderr, "cannot open audio device %s (%s)\n", argv[1], snd_strerror(err));
    exit (1);
  }

  fprintf(stdout, "audio interface opened\n");

  if ((err = snd_pcm_hw_params_malloc(&hw_params)) < 0) {
    fprintf(stderr, "cannot allocate hardware parameter structure (%s)\n", snd_strerror(err));
    exit (1);
  }

  fprintf(stdout, "hw_params allocated\n");

  //First, we initialize the hwparams structure with the full configuration space of the soundcard.  
  if ((err = snd_pcm_hw_params_any(capture_handle, hw_params)) < 0) {
    fprintf(stderr, "cannot initialize hardware parameter structure (%s)\n", snd_strerror(err));
    exit (1);
  }

  fprintf(stdout, "hw_params initialized\n");

  if ((err = snd_pcm_hw_params_set_access(capture_handle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) {
    fprintf(stderr, "cannot set access type (%s)\n", snd_strerror(err));
    exit (1);
  }

  fprintf(stdout, "hw_params access setted\n");

  // Set sample format of Signed 16 bit Little Endian
  if ((err = snd_pcm_hw_params_set_format(capture_handle, hw_params, format)) < 0) {
    fprintf(stderr, "cannot set sample format (%s)\n", snd_strerror(err));
    exit (1);
  }

  fprintf(stdout, "hw_params format setted\n");

  // Set sample rate of 44,1 kHz
  if ((err = snd_pcm_hw_params_set_rate_near(capture_handle, hw_params, &rate, 0)) < 0) {
    fprintf(stderr, "cannot set sample rate (%s)\n", snd_strerror(err));
    exit (1);
  }

  fprintf(stdout, "hw_params rate setted\n");

  if ((err = snd_pcm_hw_params_set_channels(capture_handle, hw_params, 2)) < 0) {
    fprintf(stderr, "cannot set channel count (%s)\n", snd_strerror(err));
    exit (1);
  }

  fprintf(stdout, "hw_params channels setted\n");

  if ((err = snd_pcm_hw_params(capture_handle, hw_params)) < 0) {
    fprintf(stderr, "cannot set parameters (%s)\n", snd_strerror(err));
    exit (1);
  }

  fprintf(stdout, "hw_params setted\n");

  snd_pcm_hw_params_free(hw_params);

  fprintf(stdout, "hw_params freed\n");

  if ((err = snd_pcm_prepare(capture_handle)) < 0) {
    fprintf(stderr, "cannot prepare audio interface for use (%s)\n", snd_strerror(err));
    exit(1);
  }

  fprintf(stdout, "audio interface prepared\n");


  buffer = malloc(128 * snd_pcm_format_width(format) / 8 * 2);

  fprintf(stdout, "buffer allocated %d\n", 128 * snd_pcm_format_width(format) / 8 * 2);

  //open pcm file to write data to
  int fd = open("MicIn.pcm", O_CREAT | O_RDWR, 0666);

  i = 0;
  //for (i = 0; i < 10; ++i)
  while (++i) {
    if ((err = snd_pcm_readi(capture_handle, buffer, buffer_frames)) != buffer_frames) {
      fprintf(stderr, "%d: read from audio interface failed (%s)\n", err, snd_strerror(err));
      exit(1);
    }
    write(fd, buffer, 128 * snd_pcm_format_width(format) / 8 * 2);
    fprintf(stdout, "read %d done\n", i);
  }

  close(fd);
  free(buffer);

  fprintf(stdout, "buffer freed\n");

  snd_pcm_close(capture_handle);
  fprintf(stdout, "audio interface closed\n");

  exit (0);
}

Я пробовал следующее:

  1. ffplay -autoexit -f f32le -ac 1 -ar 44100 MicIn.pcm, который дал мне ужасное качество с некоторым белым шумом, но по крайней мере я мог услышать запись. Интересно, почему это работает, потому что f32le не тот формат, который я использовал в моем коде
  2. ffplay -autoexit -f s16le -ac 1 -ar 44100 MicIn.pcm, который не дает никакого звука, даже если он должен быть в правильном формате?
  3. попытался импортировать это в наглость но громкость была очень низкая

Может кто-нибудь мне помочь? Почему у меня плохое качество и почему я могу читать его только с f32le, который не в правильном формате?

...