ref: 86c9aebfb42aa5909ae2eed6b438f16411909789
dir: /frontend/audio.c/
/* ** FAAD - Freeware Advanced Audio Decoder ** Copyright (C) 2002 M. Bakker ** ** This program is free software; you can redistribute it and/or modify ** it under the terms of the GNU General Public License as published by ** the Free Software Foundation; either version 2 of the License, or ** (at your option) any later version. ** ** This program is distributed in the hope that it will be useful, ** but WITHOUT ANY WARRANTY; without even the implied warranty of ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ** GNU General Public License for more details. ** ** You should have received a copy of the GNU General Public License ** along with this program; if not, write to the Free Software ** Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. ** ** $Id: audio.c,v 1.5 2002/08/13 14:39:03 menno Exp $ **/ #ifdef _WIN32 #include <io.h> #endif #include <stdlib.h> #include <stdio.h> #include <fcntl.h> #include <faad.h> #include "audio.h" audio_file *open_audio_file(char *infile, int samplerate, int channels, int outputFormat, int fileType) { audio_file *aufile = malloc(sizeof(audio_file)); aufile->outputFormat = outputFormat; aufile->samplerate = samplerate; aufile->channels = channels; aufile->samples = 0; aufile->fileType = fileType; switch (outputFormat) { case FAAD_FMT_16BIT: aufile->bits_per_sample = 16; break; case FAAD_FMT_24BIT: aufile->bits_per_sample = 24; break; case FAAD_FMT_32BIT: aufile->bits_per_sample = 32; break; default: if (aufile) free(aufile); return NULL; } #ifdef _WIN32 if(infile[0] == '-') { setmode(fileno(stdout), O_BINARY); } #endif aufile->sndfile = fopen(infile, "wb"); if (aufile->sndfile == NULL) { if (aufile) free(aufile); return NULL; } if (aufile->fileType == OUTPUT_WAV) { write_wav_header(aufile->sndfile, aufile->samplerate, aufile->bits_per_sample, aufile->channels, aufile->samples); } return aufile; } int write_audio_file(audio_file *aufile, void *sample_buffer, int samples) { switch (aufile->outputFormat) { case FAAD_FMT_16BIT: return write_audio_16bit(aufile, sample_buffer, samples); case FAAD_FMT_24BIT: return write_audio_24bit(aufile, sample_buffer, samples); case FAAD_FMT_32BIT: return write_audio_32bit(aufile, sample_buffer, samples); default: return 0; } return 0; } void close_audio_file(audio_file *aufile) { if (aufile->fileType == OUTPUT_WAV) { fseek(aufile->sndfile, 0, SEEK_SET); write_wav_header(aufile->sndfile, aufile->samplerate, aufile->bits_per_sample, aufile->channels, aufile->samples); } fclose(aufile->sndfile); if (aufile) free(aufile); } static int write_wav_header(FILE *file, unsigned long samplerate, unsigned int bits_per_sample, unsigned int channels, unsigned long samples) { unsigned char header[44]; unsigned char* p = header; unsigned int bytes = (bits_per_sample + 7) / 8; float data_size = (float)bytes * samples; unsigned long word32; int ret; *p++ = 'R'; *p++ = 'I'; *p++ = 'F'; *p++ = 'F'; word32 = data_size + (44 - 8) < (float)MAXWAVESIZE ? (unsigned long)data_size + (44 - 8) : (unsigned long)MAXWAVESIZE; *p++ = (unsigned char)(word32 >> 0); *p++ = (unsigned char)(word32 >> 8); *p++ = (unsigned char)(word32 >> 16); *p++ = (unsigned char)(word32 >> 24); *p++ = 'W'; *p++ = 'A'; *p++ = 'V'; *p++ = 'E'; *p++ = 'f'; *p++ = 'm'; *p++ = 't'; *p++ = ' '; *p++ = 0x10; *p++ = 0x00; *p++ = 0x00; *p++ = 0x00; *p++ = 0x01; *p++ = 0x00; *p++ = (unsigned char)(channels >> 0); *p++ = (unsigned char)(channels >> 8); *p++ = (unsigned char)(samplerate >> 0); *p++ = (unsigned char)(samplerate >> 8); *p++ = (unsigned char)(samplerate >> 16); *p++ = (unsigned char)(samplerate >> 24); word32 *= bytes * channels; *p++ = (unsigned char)(word32 >> 0); *p++ = (unsigned char)(word32 >> 8); *p++ = (unsigned char)(word32 >> 16); *p++ = (unsigned char)(word32 >> 24); word32 = bytes * channels; *p++ = (unsigned char)(word32 >> 0); *p++ = (unsigned char)(word32 >> 8); *p++ = (unsigned char)(bits_per_sample >> 0); *p++ = (unsigned char)(bits_per_sample >> 8); *p++ = 'd'; *p++ = 'a'; *p++ = 't'; *p++ = 'a'; word32 = data_size < MAXWAVESIZE ? (unsigned long)data_size : (unsigned long)MAXWAVESIZE; *p++ = (unsigned char)(word32 >> 0); *p++ = (unsigned char)(word32 >> 8); *p++ = (unsigned char)(word32 >> 16); *p++ = (unsigned char)(word32 >> 24); ret = fwrite(header, sizeof(header), 1, file); return ret; } static int write_audio_16bit(audio_file *aufile, void *sample_buffer, unsigned int samples) { int ret; unsigned int i; short *sample_buffer16 = (short*)sample_buffer; char *data = malloc(samples*aufile->bits_per_sample*sizeof(char)/8); aufile->samples += samples; for (i = 0; i < samples; i++) { data[i*2] = sample_buffer16[i] & 0xFF; data[i*2+1] = (sample_buffer16[i] >> 8) & 0xFF; } ret = fwrite(data, samples, aufile->bits_per_sample/8, aufile->sndfile); if (data) free(data); return ret; } static int write_audio_24bit(audio_file *aufile, void *sample_buffer, unsigned int samples) { int ret; unsigned int i; long *sample_buffer24 = (long*)sample_buffer; char *data = malloc(samples*aufile->bits_per_sample*sizeof(char)/8); aufile->samples += samples; for (i = 0; i < samples; i++) { data[i*3] = sample_buffer24[i] & 0xFF; data[i*3+1] = (sample_buffer24[i] >> 8) & 0xFF; data[i*3+2] = (sample_buffer24[i] >> 16) & 0xFF; } ret = fwrite(data, samples, aufile->bits_per_sample/8, aufile->sndfile); if (data) free(data); return ret; } static int write_audio_32bit(audio_file *aufile, void *sample_buffer, unsigned int samples) { int ret; unsigned int i; long *sample_buffer32 = (long*)sample_buffer; char *data = malloc(samples*aufile->bits_per_sample*sizeof(char)/8); aufile->samples += samples; for (i = 0; i < samples; i++) { data[i*4] = sample_buffer32[i] & 0xFF; data[i*4+1] = (sample_buffer32[i] >> 8) & 0xFF; data[i*4+2] = (sample_buffer32[i] >> 16) & 0xFF; data[i*4+3] = (sample_buffer32[i] >> 24) & 0xFF; } ret = fwrite(data, samples, aufile->bits_per_sample/8, aufile->sndfile); if (data) free(data); return ret; }