|  | /* tinycap.c | 
|  | ** | 
|  | ** Copyright 2011, The Android Open Source Project | 
|  | ** | 
|  | ** Redistribution and use in source and binary forms, with or without | 
|  | ** modification, are permitted provided that the following conditions are met: | 
|  | **     * Redistributions of source code must retain the above copyright | 
|  | **       notice, this list of conditions and the following disclaimer. | 
|  | **     * Redistributions in binary form must reproduce the above copyright | 
|  | **       notice, this list of conditions and the following disclaimer in the | 
|  | **       documentation and/or other materials provided with the distribution. | 
|  | **     * Neither the name of The Android Open Source Project nor the names of | 
|  | **       its contributors may be used to endorse or promote products derived | 
|  | **       from this software without specific prior written permission. | 
|  | ** | 
|  | ** THIS SOFTWARE IS PROVIDED BY The Android Open Source Project ``AS IS'' AND | 
|  | ** ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
|  | ** IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | 
|  | ** ARE DISCLAIMED. IN NO EVENT SHALL The Android Open Source Project BE LIABLE | 
|  | ** FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
|  | ** DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
|  | ** SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
|  | ** CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT | 
|  | ** LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY | 
|  | ** OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH | 
|  | ** DAMAGE. | 
|  | */ | 
|  |  | 
|  | #include <tinyalsa/asoundlib.h> | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  | #include <stdint.h> | 
|  | #include <signal.h> | 
|  | #include <string.h> | 
|  |  | 
|  | #define ID_RIFF 0x46464952 | 
|  | #define ID_WAVE 0x45564157 | 
|  | #define ID_FMT  0x20746d66 | 
|  | #define ID_DATA 0x61746164 | 
|  |  | 
|  | #define FORMAT_PCM 1 | 
|  |  | 
|  | struct wav_header { | 
|  | uint32_t riff_id; | 
|  | uint32_t riff_sz; | 
|  | uint32_t riff_fmt; | 
|  | uint32_t fmt_id; | 
|  | uint32_t fmt_sz; | 
|  | uint16_t audio_format; | 
|  | uint16_t num_channels; | 
|  | uint32_t sample_rate; | 
|  | uint32_t byte_rate; | 
|  | uint16_t block_align; | 
|  | uint16_t bits_per_sample; | 
|  | uint32_t data_id; | 
|  | uint32_t data_sz; | 
|  | }; | 
|  |  | 
|  | int capturing = 1; | 
|  |  | 
|  | unsigned int capture_sample(FILE *file, unsigned int card, unsigned int device, | 
|  | unsigned int channels, unsigned int rate, | 
|  | enum pcm_format format, unsigned int period_size, | 
|  | unsigned int period_count); | 
|  |  | 
|  | void sigint_handler(int sig) // __unused) | 
|  | { | 
|  | capturing = 0; | 
|  | } | 
|  |  | 
|  | int main(int argc, char **argv) | 
|  | { | 
|  | FILE *file; | 
|  | struct wav_header header; | 
|  | unsigned int card = 0; | 
|  | unsigned int device = 0; | 
|  | unsigned int channels = 2; | 
|  | unsigned int rate = 44100; | 
|  | unsigned int bits = 16; | 
|  | unsigned int frames; | 
|  | unsigned int period_size = 1024; | 
|  | unsigned int period_count = 4; | 
|  | enum pcm_format format; | 
|  |  | 
|  | if (argc < 2) { | 
|  | fprintf(stderr, "Usage: %s file.wav [-D card] [-d device] [-c channels] " | 
|  | "[-r rate] [-b bits] [-p period_size] [-n n_periods]\n", argv[0]); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | file = fopen(argv[1], "wb"); | 
|  | if (!file) { | 
|  | fprintf(stderr, "Unable to create file '%s'\n", argv[1]); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | /* parse command line arguments */ | 
|  | argv += 2; | 
|  | while (*argv) { | 
|  | if (strcmp(*argv, "-d") == 0) { | 
|  | argv++; | 
|  | if (*argv) | 
|  | device = atoi(*argv); | 
|  | } else if (strcmp(*argv, "-c") == 0) { | 
|  | argv++; | 
|  | if (*argv) | 
|  | channels = atoi(*argv); | 
|  | } else if (strcmp(*argv, "-r") == 0) { | 
|  | argv++; | 
|  | if (*argv) | 
|  | rate = atoi(*argv); | 
|  | } else if (strcmp(*argv, "-b") == 0) { | 
|  | argv++; | 
|  | if (*argv) | 
|  | bits = atoi(*argv); | 
|  | } else if (strcmp(*argv, "-D") == 0) { | 
|  | argv++; | 
|  | if (*argv) | 
|  | card = atoi(*argv); | 
|  | } else if (strcmp(*argv, "-p") == 0) { | 
|  | argv++; | 
|  | if (*argv) | 
|  | period_size = atoi(*argv); | 
|  | } else if (strcmp(*argv, "-n") == 0) { | 
|  | argv++; | 
|  | if (*argv) | 
|  | period_count = atoi(*argv); | 
|  | } | 
|  | if (*argv) | 
|  | argv++; | 
|  | } | 
|  |  | 
|  | header.riff_id = ID_RIFF; | 
|  | header.riff_sz = 0; | 
|  | header.riff_fmt = ID_WAVE; | 
|  | header.fmt_id = ID_FMT; | 
|  | header.fmt_sz = 16; | 
|  | header.audio_format = FORMAT_PCM; | 
|  | header.num_channels = channels; | 
|  | header.sample_rate = rate; | 
|  |  | 
|  | switch (bits) { | 
|  | case 32: | 
|  | format = PCM_FORMAT_S32_LE; | 
|  | break; | 
|  | case 24: | 
|  | format = PCM_FORMAT_S24_LE; | 
|  | break; | 
|  | case 16: | 
|  | format = PCM_FORMAT_S16_LE; | 
|  | break; | 
|  | default: | 
|  | fprintf(stderr, "%d bits is not supported.\n", bits); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | header.bits_per_sample = pcm_format_to_bits(format); | 
|  | header.byte_rate = (header.bits_per_sample / 8) * channels * rate; | 
|  | header.block_align = channels * (header.bits_per_sample / 8); | 
|  | header.data_id = ID_DATA; | 
|  |  | 
|  | /* leave enough room for header */ | 
|  | fseek(file, sizeof(struct wav_header), SEEK_SET); | 
|  |  | 
|  | /* install signal handler and begin capturing */ | 
|  | signal(SIGINT, sigint_handler); | 
|  | frames = capture_sample(file, card, device, header.num_channels, | 
|  | header.sample_rate, format, | 
|  | period_size, period_count); | 
|  | printf("Captured %d frames\n", frames); | 
|  |  | 
|  | /* write header now all information is known */ | 
|  | header.data_sz = frames * header.block_align; | 
|  | header.riff_sz = header.data_sz + sizeof(header) - 8; | 
|  | fseek(file, 0, SEEK_SET); | 
|  | fwrite(&header, sizeof(struct wav_header), 1, file); | 
|  |  | 
|  | fclose(file); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | unsigned int capture_sample(FILE *file, unsigned int card, unsigned int device, | 
|  | unsigned int channels, unsigned int rate, | 
|  | enum pcm_format format, unsigned int period_size, | 
|  | unsigned int period_count) | 
|  | { | 
|  | struct pcm_config config; | 
|  | struct pcm *pcm; | 
|  | char *buffer; | 
|  | unsigned int size; | 
|  | unsigned int bytes_read = 0; | 
|  |  | 
|  | memset(&config, 0, sizeof(config)); | 
|  | config.channels = channels; | 
|  | config.rate = rate; | 
|  | config.period_size = period_size; | 
|  | config.period_count = period_count; | 
|  | config.format = format; | 
|  | config.start_threshold = 0; | 
|  | config.stop_threshold = 0; | 
|  | config.silence_threshold = 0; | 
|  |  | 
|  | pcm = pcm_open(card, device, PCM_IN, &config); | 
|  | if (!pcm || !pcm_is_ready(pcm)) { | 
|  | fprintf(stderr, "Unable to open PCM device (%s)\n", | 
|  | pcm_get_error(pcm)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | size = pcm_frames_to_bytes(pcm, pcm_get_buffer_size(pcm)); | 
|  | buffer = malloc(size); | 
|  | if (!buffer) { | 
|  | fprintf(stderr, "Unable to allocate %d bytes\n", size); | 
|  | free(buffer); | 
|  | pcm_close(pcm); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | printf("Capturing sample: %u ch, %u hz, %u bit\n", channels, rate, | 
|  | pcm_format_to_bits(format)); | 
|  |  | 
|  | while (capturing && !pcm_read(pcm, buffer, size)) { | 
|  | if (fwrite(buffer, 1, size, file) != size) { | 
|  | fprintf(stderr,"Error capturing sample\n"); | 
|  | break; | 
|  | } | 
|  | bytes_read += size; | 
|  | } | 
|  |  | 
|  | free(buffer); | 
|  | pcm_close(pcm); | 
|  | return pcm_bytes_to_frames(pcm, bytes_read); | 
|  | } | 
|  |  |