411 lines
11 KiB
C
411 lines
11 KiB
C
/*
|
|
** Copyright (C) 1999-2004 Erik de Castro Lopo <erikd@mega-nerd.com>
|
|
**
|
|
** This program is free software; you can redistribute it and/or modify
|
|
** it under the terms of the GNU Lesser General Public License as published by
|
|
** the Free Software Foundation; either version 2.1 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 Lesser General Public License for more details.
|
|
**
|
|
** You should have received a copy of the GNU Lesser 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.
|
|
*/
|
|
|
|
#include "sfconfig.h"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <ctype.h>
|
|
#include <stdarg.h>
|
|
|
|
#include "sndfile.h"
|
|
#include "sfendian.h"
|
|
#include "common.h"
|
|
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* Macros to handle big/little endian issues.
|
|
*/
|
|
|
|
#define FORM_MARKER (MAKE_MARKER ('F', 'O', 'R', 'M'))
|
|
#define SVX8_MARKER (MAKE_MARKER ('8', 'S', 'V', 'X'))
|
|
#define SV16_MARKER (MAKE_MARKER ('1', '6', 'S', 'V'))
|
|
#define VHDR_MARKER (MAKE_MARKER ('V', 'H', 'D', 'R'))
|
|
#define BODY_MARKER (MAKE_MARKER ('B', 'O', 'D', 'Y'))
|
|
|
|
#define ATAK_MARKER (MAKE_MARKER ('A', 'T', 'A', 'K'))
|
|
#define RLSE_MARKER (MAKE_MARKER ('R', 'L', 'S', 'E'))
|
|
|
|
#define c_MARKER (MAKE_MARKER ('(', 'c', ')', ' '))
|
|
#define NAME_MARKER (MAKE_MARKER ('N', 'A', 'M', 'E'))
|
|
#define AUTH_MARKER (MAKE_MARKER ('A', 'U', 'T', 'H'))
|
|
#define ANNO_MARKER (MAKE_MARKER ('A', 'N', 'N', 'O'))
|
|
#define CHAN_MARKER (MAKE_MARKER ('C', 'H', 'A', 'N'))
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* Typedefs for file chunks.
|
|
*/
|
|
|
|
typedef struct
|
|
{ unsigned int oneShotHiSamples, repeatHiSamples, samplesPerHiCycle ;
|
|
unsigned short samplesPerSec ;
|
|
unsigned char octave, compression ;
|
|
unsigned int volume ;
|
|
} VHDR_CHUNK ;
|
|
|
|
enum {
|
|
HAVE_FORM = 0x01,
|
|
|
|
HAVE_SVX = 0x02,
|
|
HAVE_VHDR = 0x04,
|
|
HAVE_BODY = 0x08
|
|
} ;
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* Private static functions.
|
|
*/
|
|
|
|
static int svx_close (SF_PRIVATE *psf) ;
|
|
static int svx_write_header (SF_PRIVATE *psf, int calc_length) ;
|
|
static int svx_read_header (SF_PRIVATE *psf) ;
|
|
|
|
/*------------------------------------------------------------------------------
|
|
** Public function.
|
|
*/
|
|
|
|
int
|
|
svx_open (SF_PRIVATE *psf)
|
|
{ int error ;
|
|
|
|
if (psf->mode == SFM_READ || (psf->mode == SFM_RDWR && psf->filelength > 0))
|
|
{ if ((error = svx_read_header (psf)))
|
|
return error ;
|
|
|
|
psf->endian = SF_ENDIAN_BIG ; /* All SVX files are big endian. */
|
|
|
|
psf->blockwidth = psf->sf.channels * psf->bytewidth ;
|
|
if (psf->blockwidth)
|
|
psf->sf.frames = psf->datalength / psf->blockwidth ;
|
|
|
|
psf_fseek (psf, psf->dataoffset, SEEK_SET) ;
|
|
} ;
|
|
|
|
if (psf->mode == SFM_WRITE || psf->mode == SFM_RDWR)
|
|
{ if (psf->is_pipe)
|
|
return SFE_NO_PIPE_WRITE ;
|
|
|
|
if ((psf->sf.format & SF_FORMAT_TYPEMASK) != SF_FORMAT_SVX)
|
|
return SFE_BAD_OPEN_FORMAT ;
|
|
|
|
psf->endian = psf->sf.format & SF_FORMAT_ENDMASK ;
|
|
|
|
if (psf->endian == SF_ENDIAN_LITTLE || (CPU_IS_LITTLE_ENDIAN && psf->endian == SF_ENDIAN_CPU))
|
|
return SFE_BAD_ENDIAN ;
|
|
|
|
psf->endian = SF_ENDIAN_BIG ; /* All SVX files are big endian. */
|
|
|
|
error = svx_write_header (psf, SF_FALSE) ;
|
|
if (error)
|
|
return error ;
|
|
|
|
psf->write_header = svx_write_header ;
|
|
} ;
|
|
|
|
psf->container_close = svx_close ;
|
|
|
|
if ((error = pcm_init (psf)))
|
|
return error ;
|
|
|
|
return 0 ;
|
|
} /* svx_open */
|
|
|
|
/*------------------------------------------------------------------------------
|
|
*/
|
|
|
|
static int
|
|
svx_read_header (SF_PRIVATE *psf)
|
|
{ VHDR_CHUNK vhdr ;
|
|
unsigned int FORMsize, vhdrsize, dword, marker ;
|
|
int filetype = 0, parsestage = 0, done = 0 ;
|
|
int bytecount = 0, channels ;
|
|
|
|
memset (&vhdr, 0, sizeof (vhdr)) ;
|
|
psf_binheader_readf (psf, "p", 0) ;
|
|
|
|
/* Set default number of channels. Currently can't handle stereo SVX files. */
|
|
psf->sf.channels = 1 ;
|
|
|
|
psf->sf.format = SF_FORMAT_SVX ;
|
|
|
|
while (! done)
|
|
{ psf_binheader_readf (psf, "m", &marker) ;
|
|
switch (marker)
|
|
{ case FORM_MARKER :
|
|
if (parsestage)
|
|
return SFE_SVX_NO_FORM ;
|
|
|
|
psf_binheader_readf (psf, "E4", &FORMsize) ;
|
|
|
|
if (FORMsize != psf->filelength - 2 * sizeof (dword))
|
|
{ dword = psf->filelength - 2 * sizeof (dword) ;
|
|
psf_log_printf (psf, "FORM : %d (should be %d)\n", FORMsize, dword) ;
|
|
FORMsize = dword ;
|
|
}
|
|
else
|
|
psf_log_printf (psf, "FORM : %d\n", FORMsize) ;
|
|
parsestage |= HAVE_FORM ;
|
|
break ;
|
|
|
|
case SVX8_MARKER :
|
|
case SV16_MARKER :
|
|
if (! (parsestage & HAVE_FORM))
|
|
return SFE_SVX_NO_FORM ;
|
|
filetype = marker ;
|
|
psf_log_printf (psf, " %M\n", marker) ;
|
|
parsestage |= HAVE_SVX ;
|
|
break ;
|
|
|
|
case VHDR_MARKER :
|
|
if (! (parsestage & (HAVE_FORM | HAVE_SVX)))
|
|
return SFE_SVX_NO_FORM ;
|
|
|
|
psf_binheader_readf (psf, "E4", &vhdrsize) ;
|
|
|
|
psf_log_printf (psf, " VHDR : %d\n", vhdrsize) ;
|
|
|
|
psf_binheader_readf (psf, "E4442114", &(vhdr.oneShotHiSamples), &(vhdr.repeatHiSamples),
|
|
&(vhdr.samplesPerHiCycle), &(vhdr.samplesPerSec), &(vhdr.octave), &(vhdr.compression),
|
|
&(vhdr.volume)) ;
|
|
|
|
psf_log_printf (psf, " OneShotHiSamples : %d\n", vhdr.oneShotHiSamples) ;
|
|
psf_log_printf (psf, " RepeatHiSamples : %d\n", vhdr.repeatHiSamples) ;
|
|
psf_log_printf (psf, " samplesPerHiCycle : %d\n", vhdr.samplesPerHiCycle) ;
|
|
psf_log_printf (psf, " Sample Rate : %d\n", vhdr.samplesPerSec) ;
|
|
psf_log_printf (psf, " Octave : %d\n", vhdr.octave) ;
|
|
|
|
psf_log_printf (psf, " Compression : %d => ", vhdr.compression) ;
|
|
|
|
switch (vhdr.compression)
|
|
{ case 0 : psf_log_printf (psf, "None.\n") ;
|
|
break ;
|
|
case 1 : psf_log_printf (psf, "Fibonacci delta\n") ;
|
|
break ;
|
|
case 2 : psf_log_printf (psf, "Exponential delta\n") ;
|
|
break ;
|
|
} ;
|
|
|
|
psf_log_printf (psf, " Volume : %d\n", vhdr.volume) ;
|
|
|
|
psf->sf.samplerate = vhdr.samplesPerSec ;
|
|
|
|
if (filetype == SVX8_MARKER)
|
|
{ psf->sf.format |= SF_FORMAT_PCM_S8 ;
|
|
psf->bytewidth = 1 ;
|
|
}
|
|
else if (filetype == SV16_MARKER)
|
|
{ psf->sf.format |= SF_FORMAT_PCM_16 ;
|
|
psf->bytewidth = 2 ;
|
|
} ;
|
|
|
|
parsestage |= HAVE_VHDR ;
|
|
break ;
|
|
|
|
case BODY_MARKER :
|
|
if (! (parsestage & HAVE_VHDR))
|
|
return SFE_SVX_NO_BODY ;
|
|
|
|
psf_binheader_readf (psf, "E4", &dword) ;
|
|
psf->datalength = dword ;
|
|
|
|
psf->dataoffset = psf_ftell (psf) ;
|
|
|
|
if (psf->datalength > psf->filelength - psf->dataoffset)
|
|
{ psf_log_printf (psf, " BODY : %D (should be %D)\n", psf->datalength, psf->filelength - psf->dataoffset) ;
|
|
psf->datalength = psf->filelength - psf->dataoffset ;
|
|
}
|
|
else
|
|
psf_log_printf (psf, " BODY : %D\n", psf->datalength) ;
|
|
|
|
parsestage |= HAVE_BODY ;
|
|
|
|
if (! psf->sf.seekable)
|
|
break ;
|
|
|
|
psf_fseek (psf, psf->datalength, SEEK_CUR) ;
|
|
break ;
|
|
|
|
case NAME_MARKER :
|
|
if (! (parsestage & HAVE_SVX))
|
|
return SFE_SVX_NO_FORM ;
|
|
|
|
psf_binheader_readf (psf, "E4", &dword) ;
|
|
|
|
psf_log_printf (psf, " %M : %d\n", marker, dword) ;
|
|
|
|
if (strlen (psf->filename) != dword)
|
|
{ if (dword > sizeof (psf->filename) - 1)
|
|
return SFE_SVX_BAD_NAME_LENGTH ;
|
|
|
|
psf_binheader_readf (psf, "b", psf->filename, dword) ;
|
|
psf->filename [dword] = 0 ;
|
|
}
|
|
else
|
|
psf_binheader_readf (psf, "j", dword) ;
|
|
break ;
|
|
|
|
case ANNO_MARKER :
|
|
if (! (parsestage & HAVE_SVX))
|
|
return SFE_SVX_NO_FORM ;
|
|
|
|
psf_binheader_readf (psf, "E4", &dword) ;
|
|
|
|
psf_log_printf (psf, " %M : %d\n", marker, dword) ;
|
|
|
|
psf_binheader_readf (psf, "j", dword) ;
|
|
break ;
|
|
|
|
case CHAN_MARKER :
|
|
if (! (parsestage & HAVE_SVX))
|
|
return SFE_SVX_NO_FORM ;
|
|
|
|
psf_binheader_readf (psf, "E4", &dword) ;
|
|
|
|
psf_log_printf (psf, " %M : %d\n", marker, dword) ;
|
|
|
|
bytecount += psf_binheader_readf (psf, "E4", &channels) ;
|
|
psf->sf.channels = channels ;
|
|
|
|
psf_log_printf (psf, " Channels : %d\n", channels) ;
|
|
|
|
psf_binheader_readf (psf, "j", dword - bytecount) ;
|
|
break ;
|
|
|
|
|
|
case AUTH_MARKER :
|
|
case c_MARKER :
|
|
if (! (parsestage & HAVE_SVX))
|
|
return SFE_SVX_NO_FORM ;
|
|
|
|
psf_binheader_readf (psf, "E4", &dword) ;
|
|
|
|
psf_log_printf (psf, " %M : %d\n", marker, dword) ;
|
|
|
|
psf_binheader_readf (psf, "j", dword) ;
|
|
break ;
|
|
|
|
default :
|
|
if (isprint ((marker >> 24) & 0xFF) && isprint ((marker >> 16) & 0xFF)
|
|
&& isprint ((marker >> 8) & 0xFF) && isprint (marker & 0xFF))
|
|
{ psf_binheader_readf (psf, "E4", &dword) ;
|
|
|
|
psf_log_printf (psf, "%M : %d (unknown marker)\n", marker, dword) ;
|
|
|
|
psf_binheader_readf (psf, "j", dword) ;
|
|
break ;
|
|
} ;
|
|
if ((dword = psf_ftell (psf)) & 0x03)
|
|
{ psf_log_printf (psf, " Unknown chunk marker at position %d. Resynching.\n", dword - 4) ;
|
|
|
|
psf_binheader_readf (psf, "j", -3) ;
|
|
break ;
|
|
} ;
|
|
psf_log_printf (psf, "*** Unknown chunk marker : %X. Exiting parser.\n", marker) ;
|
|
done = 1 ;
|
|
} ; /* switch (marker) */
|
|
|
|
if (! psf->sf.seekable && (parsestage & HAVE_BODY))
|
|
break ;
|
|
|
|
if (psf_ftell (psf) >= psf->filelength - SIGNED_SIZEOF (dword))
|
|
break ;
|
|
} ; /* while (1) */
|
|
|
|
if (vhdr.compression)
|
|
return SFE_SVX_BAD_COMP ;
|
|
|
|
if (psf->dataoffset <= 0)
|
|
return SFE_SVX_NO_DATA ;
|
|
|
|
return 0 ;
|
|
} /* svx_read_header */
|
|
|
|
static int
|
|
svx_close (SF_PRIVATE *psf)
|
|
{
|
|
if (psf->mode == SFM_WRITE || psf->mode == SFM_RDWR)
|
|
svx_write_header (psf, SF_TRUE) ;
|
|
|
|
return 0 ;
|
|
} /* svx_close */
|
|
|
|
static int
|
|
svx_write_header (SF_PRIVATE *psf, int calc_length)
|
|
{ static char annotation [] = "libsndfile by Erik de Castro Lopo\0\0\0" ;
|
|
sf_count_t current ;
|
|
|
|
current = psf_ftell (psf) ;
|
|
|
|
if (calc_length)
|
|
{ psf->filelength = psf_get_filelen (psf) ;
|
|
|
|
psf->datalength = psf->filelength - psf->dataoffset ;
|
|
|
|
if (psf->dataend)
|
|
psf->datalength -= psf->filelength - psf->dataend ;
|
|
|
|
psf->sf.frames = psf->datalength / (psf->bytewidth * psf->sf.channels) ;
|
|
} ;
|
|
|
|
psf->header [0] = 0 ;
|
|
psf->headindex = 0 ;
|
|
psf_fseek (psf, 0, SEEK_SET) ;
|
|
|
|
/* FORM marker and FORM size. */
|
|
psf_binheader_writef (psf, "Etm8", FORM_MARKER, (psf->filelength < 8) ?
|
|
psf->filelength * 0 : psf->filelength - 8) ;
|
|
|
|
psf_binheader_writef (psf, "m", (psf->bytewidth == 1) ? SVX8_MARKER : SV16_MARKER) ;
|
|
|
|
/* VHDR chunk. */
|
|
psf_binheader_writef (psf, "Em4", VHDR_MARKER, sizeof (VHDR_CHUNK)) ;
|
|
/* VHDR : oneShotHiSamples, repeatHiSamples, samplesPerHiCycle */
|
|
psf_binheader_writef (psf, "E444", psf->sf.frames, 0, 0) ;
|
|
/* VHDR : samplesPerSec, octave, compression */
|
|
psf_binheader_writef (psf, "E211", psf->sf.samplerate, 1, 0) ;
|
|
/* VHDR : volume */
|
|
psf_binheader_writef (psf, "E4", (psf->bytewidth == 1) ? 0xFF : 0xFFFF) ;
|
|
|
|
/* Filename and annotation strings. */
|
|
psf_binheader_writef (psf, "Emsms", NAME_MARKER, psf->filename, ANNO_MARKER, annotation) ;
|
|
|
|
/* BODY marker and size. */
|
|
psf_binheader_writef (psf, "Etm8", BODY_MARKER, (psf->datalength < 0) ?
|
|
psf->datalength * 0 : psf->datalength) ;
|
|
|
|
psf_fwrite (psf->header, psf->headindex, 1, psf) ;
|
|
|
|
if (psf->error)
|
|
return psf->error ;
|
|
|
|
psf->dataoffset = psf->headindex ;
|
|
|
|
if (current > 0)
|
|
psf_fseek (psf, current, SEEK_SET) ;
|
|
|
|
return psf->error ;
|
|
} /* svx_write_header */
|
|
|
|
|
|
/*
|
|
** Do not edit or modify anything in this comment block.
|
|
** The arch-tag line is a file identity tag for the GNU Arch
|
|
** revision control system.
|
|
**
|
|
** arch-tag: a80ab6fb-7d75-4d32-a6b0-0061a3f05d95
|
|
*/
|