// input: ALSA // assuming stereo #define CHANNELS_COUNT 2 #define SAMPLE_RATE 44100 static void initialize_audio_parameters(snd_pcm_t** handle, struct audio_data* audio, snd_pcm_uframes_t* frames) { // alsa: open device to capture audio int err = snd_pcm_open(handle, audio->source, SND_PCM_STREAM_CAPTURE, 0); if (err < 0) { fprintf(stderr, "error opening stream: %s\n", snd_strerror(err)); exit(EXIT_FAILURE); } #ifdef DEBUG else printf("open stream successful\n"); #endif snd_pcm_hw_params_t* params; snd_pcm_hw_params_alloca(¶ms); // assembling params snd_pcm_hw_params_any(*handle, params); // setting defaults or something // interleaved mode right left right left snd_pcm_hw_params_set_access(*handle, params, SND_PCM_ACCESS_RW_INTERLEAVED); // trying to set 16bit snd_pcm_hw_params_set_format(*handle, params, SND_PCM_FORMAT_S16_LE); snd_pcm_hw_params_set_channels(*handle, params, CHANNELS_COUNT); unsigned int sample_rate = SAMPLE_RATE; // trying our rate snd_pcm_hw_params_set_rate_near(*handle, params, &sample_rate, NULL); // number of frames pr read snd_pcm_hw_params_set_period_size_near(*handle, params, frames, NULL); err = snd_pcm_hw_params(*handle, params); // attempting to set params if (err < 0) { fprintf(stderr, "unable to set hw parameters: %s\n", snd_strerror(err)); exit(EXIT_FAILURE); } if ((err = snd_pcm_prepare (*handle)) < 0) { fprintf (stderr, "cannot prepare audio interface for use (%s)\n", snd_strerror (err)); exit (EXIT_FAILURE); } // getting actual format snd_pcm_hw_params_get_format(params, (snd_pcm_format_t*)&sample_rate); // converting result to number of bits if (sample_rate <= 5) audio->format = 16; else if (sample_rate <= 9 ) audio->format = 24; else audio->format = 32; snd_pcm_hw_params_get_rate(params, &audio->rate, NULL); snd_pcm_hw_params_get_period_size(params, frames, NULL); // snd_pcm_hw_params_get_period_time(params, &sample_rate, &dir); } static int get_certain_frame(signed char* buffer, int buffer_index, int adjustment) { // using the 10 upper bits this would give me a vert res of 1024, enough... int temp = buffer[buffer_index + adjustment - 1] << 2; int lo = buffer[buffer_index + adjustment - 2] >> 6; if (lo < 0) lo = abs(lo) + 1; if (temp >= 0) temp += lo; else temp -= lo; return temp; } static void fill_audio_outs(struct audio_data* audio, signed char* buffer, const int size) { int radj = audio->format / 4; // adjustments for interleaved int ladj = audio->format / 8; static int audio_out_buffer_index = 0; // sorting out one channel and only biggest octet for (int buffer_index = 0; buffer_index < size; buffer_index += ladj * 2) { // first channel int tempr = get_certain_frame(buffer, buffer_index, radj); // second channel int templ = get_certain_frame(buffer, buffer_index, ladj); // mono: adding channels and storing it in the buffer if (audio->channels == 1) audio->audio_out_l[audio_out_buffer_index] = (templ + tempr) / 2; else { // stereo storing channels in buffer audio->audio_out_l[audio_out_buffer_index] = templ; audio->audio_out_r[audio_out_buffer_index] = tempr; } ++audio_out_buffer_index; audio_out_buffer_index %= 2048; } } #define FRAMES_NUMBER 256 void* input_alsa(void* data) { int err; struct audio_data* audio = (struct audio_data*)data; snd_pcm_t* handle; snd_pcm_uframes_t frames = FRAMES_NUMBER; int16_t buf[FRAMES_NUMBER * 2]; initialize_audio_parameters(&handle, audio, &frames); // frames * bits/8 * channels const int size = frames * (audio->format / 8) * CHANNELS_COUNT; signed char* buffer = malloc(size); int n = 0; while (1) { switch (audio->format) { case 16: err = snd_pcm_readi(handle, buf, frames); for (int i = 0; i < FRAMES_NUMBER * 2; i += 2) { if (audio->channels == 1) audio->audio_out_l[n] = (buf[i] + buf[i + 1]) / 2; //stereo storing channels in buffer if (audio->channels == 2) { audio->audio_out_l[n] = buf[i]; audio->audio_out_r[n] = buf[i + 1]; } n++; if (n == 2048 - 1)n = 0; } break; default: err = snd_pcm_readi(handle, buffer, frames); fill_audio_outs(audio, buffer, size); break; } if (err == -EPIPE) { /* EPIPE means overrun */ #ifdef DEBUG fprintf(stderr, "overrun occurred\n"); #endif snd_pcm_prepare(handle); } else if (err < 0) { #ifdef DEBUG fprintf(stderr, "error from read: %s\n", snd_strerror(err)); #endif } else if (err != (int)frames) { #ifdef DEBUG fprintf(stderr, "short read, read %d %d frames\n", err, (int)frames); #endif } if (audio->terminate == 1) { free(buffer); snd_pcm_close(handle); return NULL; } } }