Blame tools/wav_io.c

Packit 664db3
/* Copyright (C) 2002 Jean-Marc Valin 
Packit 664db3
   File: wav_io.c
Packit 664db3
   Routines to handle wav (RIFF) headers
Packit 664db3
Packit 664db3
   Redistribution and use in source and binary forms, with or without
Packit 664db3
   modification, are permitted provided that the following conditions
Packit 664db3
   are met:
Packit 664db3
   
Packit 664db3
   - Redistributions of source code must retain the above copyright
Packit 664db3
   notice, this list of conditions and the following disclaimer.
Packit 664db3
   
Packit 664db3
   - Redistributions in binary form must reproduce the above copyright
Packit 664db3
   notice, this list of conditions and the following disclaimer in the
Packit 664db3
   documentation and/or other materials provided with the distribution.
Packit 664db3
   
Packit 664db3
   - Neither the name of the Xiph.org Foundation nor the names of its
Packit 664db3
   contributors may be used to endorse or promote products derived from
Packit 664db3
   this software without specific prior written permission.
Packit 664db3
   
Packit 664db3
   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Packit 664db3
   ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Packit 664db3
   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
Packit 664db3
   A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR
Packit 664db3
   CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
Packit 664db3
   EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
Packit 664db3
   PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
Packit 664db3
   PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
Packit 664db3
   LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
Packit 664db3
   NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
Packit 664db3
   SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit 664db3
*/
Packit 664db3
Packit 664db3
#ifdef HAVE_CONFIG_H
Packit 664db3
# include "config.h"
Packit 664db3
#endif
Packit 664db3
Packit 664db3
#include <stdio.h>
Packit 664db3
#include <string.h>
Packit 664db3
#include "celt_types.h"
Packit 664db3
#include "wav_io.h"
Packit 664db3
Packit 664db3
Packit 664db3
int read_wav_header(FILE *file, int *rate, int *channels, int *format, celt_int32_t *size)
Packit 664db3
{
Packit 664db3
   char ch[5];
Packit 664db3
   celt_int32_t itmp;
Packit 664db3
   celt_int16_t stmp;
Packit 664db3
   celt_int32_t bpersec;
Packit 664db3
   celt_int16_t balign;
Packit 664db3
   int skip_bytes;
Packit 664db3
   int i;
Packit 664db3
Packit 664db3
   ch[4]=0;
Packit 664db3
#if 0
Packit 664db3
   fread(ch, 1, 4, file);
Packit 664db3
   if (strcmp(ch, "RIFF")!=0)
Packit 664db3
   {
Packit 664db3
      fseek(file, 0, SEEK_SET);
Packit 664db3
      return 0;
Packit 664db3
   }
Packit 664db3
Packit 664db3
   fread(&itmp, 4, 1, file);
Packit 664db3
   *size = le_int(itmp-36);
Packit 664db3
Packit 664db3
   fread(ch, 1, 4, file);
Packit 664db3
   if (strcmp(ch, "WAVE")!=0)
Packit 664db3
   {
Packit 664db3
      fprintf (stderr, "RIFF file is not a WAVE file\n");
Packit 664db3
      return -1;
Packit 664db3
   }
Packit 664db3
#endif
Packit 664db3
   fread(ch, 1, 4, file);
Packit 664db3
   while (strcmp(ch, "fmt ")!=0)
Packit 664db3
   {
Packit 664db3
      fread(&itmp, 4, 1, file);
Packit 664db3
      itmp = le_int(itmp);
Packit 664db3
      /*fprintf (stderr, "skip=%d\n", itmp);*/
Packit 664db3
      /*strange way of seeking, but it works even for pipes*/
Packit 664db3
      for (i=0;i
Packit 664db3
         fgetc(file);
Packit 664db3
      /*fseek(file, itmp, SEEK_CUR);*/
Packit 664db3
      fread(ch, 1, 4, file);
Packit 664db3
      if (feof(file))
Packit 664db3
      {
Packit 664db3
         fprintf (stderr, "Corrupted WAVE file: no \"fmt \"\n");
Packit 664db3
         return -1;
Packit 664db3
      }
Packit 664db3
   }
Packit 664db3
   /*if (strcmp(ch, "fmt ")!=0)
Packit 664db3
   {
Packit 664db3
      fprintf (stderr, "Corrupted WAVE file: no \"fmt \"\n");
Packit 664db3
      return -1;
Packit 664db3
      }*/
Packit 664db3
   
Packit 664db3
   fread(&itmp, 4, 1, file);
Packit 664db3
   itmp = le_int(itmp);
Packit 664db3
   skip_bytes=itmp-16;
Packit 664db3
   /*fprintf (stderr, "skip=%d\n", skip_bytes);*/
Packit 664db3
   
Packit 664db3
   fread(&stmp, 2, 1, file);
Packit 664db3
   stmp = le_short(stmp);
Packit 664db3
   if (stmp!=1)
Packit 664db3
   {
Packit 664db3
      fprintf (stderr, "Only PCM encoding is supported\n");
Packit 664db3
      return -1;
Packit 664db3
   }
Packit 664db3
Packit 664db3
   fread(&stmp, 2, 1, file);
Packit 664db3
   stmp = le_short(stmp);
Packit 664db3
   *channels = stmp;
Packit 664db3
   
Packit 664db3
   if (stmp>2)
Packit 664db3
   {
Packit 664db3
      fprintf (stderr, "Only mono and (intensity) stereo supported\n");
Packit 664db3
      return -1;
Packit 664db3
   }
Packit 664db3
Packit 664db3
   fread(&itmp, 4, 1, file);
Packit 664db3
   itmp = le_int(itmp);
Packit 664db3
   *rate = itmp;
Packit 664db3
Packit 664db3
   fread(&itmp, 4, 1, file);
Packit 664db3
   bpersec = le_int(itmp);
Packit 664db3
Packit 664db3
   fread(&stmp, 2, 1, file);
Packit 664db3
   balign = le_short(stmp);
Packit 664db3
Packit 664db3
   fread(&stmp, 2, 1, file);
Packit 664db3
   stmp = le_short(stmp);
Packit 664db3
   if (stmp!=16 && stmp!=8)
Packit 664db3
   {
Packit 664db3
      fprintf (stderr, "Only 8/16-bit linear supported\n");
Packit 664db3
      return -1;
Packit 664db3
   }
Packit 664db3
   *format=stmp;
Packit 664db3
Packit 664db3
   if (bpersec!=*rate**channels*stmp/8)
Packit 664db3
   {
Packit 664db3
      fprintf (stderr, "Corrupted header: ByteRate mismatch\n");
Packit 664db3
      return -1;
Packit 664db3
   }
Packit 664db3
Packit 664db3
   if (balign!=*channels*stmp/8)
Packit 664db3
   {
Packit 664db3
      fprintf (stderr, "Corrupted header: BlockAlign mismatch\n");
Packit 664db3
      return -1;
Packit 664db3
   }
Packit 664db3
Packit 664db3
   
Packit 664db3
   /*strange way of seeking, but it works even for pipes*/
Packit 664db3
   if (skip_bytes>0)
Packit 664db3
      for (i=0;i
Packit 664db3
         fgetc(file);
Packit 664db3
Packit 664db3
   /*fseek(file, skip_bytes, SEEK_CUR);*/
Packit 664db3
Packit 664db3
   fread(ch, 1, 4, file);
Packit 664db3
   while (strcmp(ch, "data")!=0)
Packit 664db3
   {
Packit 664db3
      fread(&itmp, 4, 1, file);
Packit 664db3
      itmp = le_int(itmp);
Packit 664db3
      /*strange way of seeking, but it works even for pipes*/
Packit 664db3
      for (i=0;i
Packit 664db3
         fgetc(file);
Packit 664db3
      /*fseek(file, itmp, SEEK_CUR);*/
Packit 664db3
      fread(ch, 1, 4, file);
Packit 664db3
      if (feof(file))
Packit 664db3
      {
Packit 664db3
         fprintf (stderr, "Corrupted WAVE file: no \"data\"\n");
Packit 664db3
         return -1;
Packit 664db3
      }
Packit 664db3
   }
Packit 664db3
Packit 664db3
   /*Ignore this for now*/
Packit 664db3
   fread(&itmp, 4, 1, file);
Packit 664db3
   itmp = le_int(itmp);
Packit 664db3
Packit 664db3
   *size=itmp;
Packit 664db3
Packit 664db3
   return 1;
Packit 664db3
}
Packit 664db3
Packit 664db3
Packit 664db3
Packit 664db3
void write_wav_header(FILE *file, int rate, int channels, int format, int size)
Packit 664db3
{
Packit 664db3
   char ch[5];
Packit 664db3
   celt_int32_t itmp;
Packit 664db3
   celt_int16_t stmp;
Packit 664db3
Packit 664db3
   ch[4]=0;
Packit 664db3
Packit 664db3
   fprintf (file, "RIFF");
Packit 664db3
Packit 664db3
   itmp = 0x7fffffff;
Packit 664db3
   fwrite(&itmp, 4, 1, file);
Packit 664db3
Packit 664db3
   fprintf (file, "WAVEfmt ");
Packit 664db3
Packit 664db3
   itmp = le_int(16);
Packit 664db3
   fwrite(&itmp, 4, 1, file);
Packit 664db3
Packit 664db3
   stmp = le_short(1);
Packit 664db3
   fwrite(&stmp, 2, 1, file);
Packit 664db3
Packit 664db3
   stmp = le_short(channels);
Packit 664db3
   fwrite(&stmp, 2, 1, file);
Packit 664db3
Packit 664db3
   itmp = le_int(rate);
Packit 664db3
   fwrite(&itmp, 4, 1, file);
Packit 664db3
Packit 664db3
   itmp = le_int(rate*channels*2);
Packit 664db3
   fwrite(&itmp, 4, 1, file);
Packit 664db3
Packit 664db3
   stmp = le_short(2*channels);
Packit 664db3
   fwrite(&stmp, 2, 1, file);
Packit 664db3
Packit 664db3
   stmp = le_short(16);
Packit 664db3
   fwrite(&stmp, 2, 1, file);
Packit 664db3
Packit 664db3
   fprintf (file, "data");
Packit 664db3
Packit 664db3
   itmp = le_int(0x7fffffff);
Packit 664db3
   fwrite(&itmp, 4, 1, file);
Packit 664db3
Packit 664db3
Packit 664db3
}