ref: aed66866cc63b01f8d66e1f381ca7a26c514887d
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.6 2002/08/13 19:16:07 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:
case FAAD_FMT_16BIT_DITHER:
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:
case FAAD_FMT_16BIT_DITHER:
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;
}