shithub: sox

ref: 92ba81db75c5b673ce4d1d561a97d3296ebad4df
dir: /src/sndrtool.c/

View raw version
/*
 * Sounder/Sndtool format handler: W V Neisius, February 1992
 *
 * June 28, 93: force output to mono.
 * 
 * March 3, 1999 - cbagwell@sprynet.com
 *   Forced extra comment fields to zero.
 */

#include "sox_i.h"

#include <math.h>
#include <string.h>
#include <stdio.h>
#include <errno.h>

/* Private data used by writer */
typedef struct sndpriv {
        sox_size_t nsamples;
        sox_size_t dataStart;
} *snd_t;

static void sndtwriteheader(sox_format_t * ft, sox_size_t nsamples)
{
    char name_buf[97];

    /* sndtool header */
    sox_writes(ft, "SOUND"); /* magic */
    sox_writeb(ft, 0x1a);
    sox_writew (ft,0);  /* hGSound */
    sox_writedw (ft,nsamples);
    sox_writedw (ft,0);
    sox_writedw (ft,nsamples);
    sox_writew (ft, ft->signal.rate);
    sox_writew (ft,0);
    sox_writew (ft,10);
    sox_writew (ft,4);
    memset (name_buf, 0, 96);
    sprintf (name_buf,"%.62s - File created by SoX",ft->filename);
    sox_writebuf(ft, name_buf, 96);
}

static int sox_sndseek(sox_format_t * ft, sox_size_t offset) 
{
    sox_size_t new_offset, channel_block, alignment;
    snd_t snd = (snd_t ) ft->priv;

    new_offset = offset * ft->signal.size;
    /* Make sure request aligns to a channel block (ie left+right) */
    channel_block = ft->signal.channels * ft->signal.size;
    alignment = new_offset % channel_block;
    /* Most common mistaken is to compute something like
     * "skip everthing upto and including this sample" so
     * advance to next sample block in this case.
     */
    if (alignment != 0)
        new_offset += (channel_block - alignment);
    new_offset += snd->dataStart;

    return sox_seeki(ft, (sox_ssize_t)new_offset, SEEK_SET);
}

static int sox_sndtstartread(sox_format_t * ft)
{
        snd_t snd = (snd_t ) ft->priv;

        char buf[97];

        unsigned short rate;
        int rc;

        /* Needed for rawread() */
        rc = sox_rawstartread(ft);
        if (rc)
            return rc;

        rate = 0;

        /* determine file type */
        /* if first 5 bytes == SOUND then this is probably a sndtool sound */
        /* if first word (16 bits) == 0 
         and second word is between 4000 & 25000 then this is sounder sound */
        /* otherwise, its probably raw, not handled here */

        if (sox_readbuf(ft, buf, 2) != 2)
        {
                sox_fail_errno(ft,errno,"SND: unexpected EOF");
                return(SOX_EOF);
        }
        if (strncmp(buf,"\0\0",2) == 0)
        {
        /* sounder */
        sox_readw(ft, &rate);
        if (rate < 4000 || rate > 25000 )
        {
                sox_fail_errno(ft,SOX_EFMT,"SND: sample rate out of range");
                return(SOX_EOF);
        }
        sox_seeki(ft, 4, SEEK_CUR);
        }
        else
        {
        /* sndtool ? */
        sox_readbuf(ft, &buf[2], 6);
        if (strncmp(buf,"SOUND",5))
        {
                sox_fail_errno(ft,SOX_EFMT,"SND: unrecognized SND format");
                return(SOX_EOF);
        }
        sox_seeki(ft, 12, SEEK_CUR);
        sox_readw(ft, &rate);
        sox_seeki(ft, 6, SEEK_CUR);
        if (sox_reads(ft, buf, 96) == SOX_EOF)
        {
                sox_fail_errno(ft,SOX_EHDR,"SND: unexpected EOF in SND header");
                return(SOX_EOF);
        }
        sox_debug("%s",buf);
        }

        ft->signal.channels = 1;
        ft->signal.rate = rate;
        ft->signal.encoding = SOX_ENCODING_UNSIGNED;
        ft->signal.size = SOX_SIZE_BYTE;

        snd->dataStart = sox_tell(ft);
        ft->length = sox_filelength(ft) - snd->dataStart;

        return (SOX_SUCCESS);
}

static int sox_sndtstartwrite(sox_format_t * ft)
{
        snd_t p = (snd_t ) ft->priv;
        int rc;

        /* Needed for rawwrite() */
        rc = sox_rawstartwrite(ft);
        if (rc)
            return rc;

        /* write header */
        ft->signal.channels = 1;
        ft->signal.encoding = SOX_ENCODING_UNSIGNED;
        ft->signal.size = SOX_SIZE_BYTE;
        p->nsamples = 0;
        sndtwriteheader(ft, 0);

        return(SOX_SUCCESS);
}

static sox_size_t sox_sndtwrite(sox_format_t * ft, const sox_ssample_t *buf, sox_size_t len)
{
        snd_t p = (snd_t ) ft->priv;
        p->nsamples += len;
        return sox_rawwrite(ft, buf, len);
}

static int sox_sndtstopwrite(sox_format_t * ft)
{
        snd_t p = (snd_t ) ft->priv;
        int rc;

        /* Flush remaining buffer out */
        rc = sox_rawstopwrite(ft);
        if (rc)
            return rc;

        /* fixup file sizes in header */
        if (sox_seeki(ft, 0, 0) != 0){
                sox_fail_errno(ft,errno,"can't rewind output file to rewrite SND header");
                return SOX_EOF;
        }
                
        sndtwriteheader(ft, p->nsamples);
                

        return(SOX_SUCCESS);
}

/* Sndtool Sound File */
static const char *sndtnames[] = {
  "sndt",
  NULL
};

static const sox_format_handler_t sox_snd_format = {
  sndtnames,
  SOX_FILE_SEEK | SOX_FILE_LIT_END,
  sox_sndtstartread,
  sox_rawread,
  sox_rawstopread,
  sox_sndtstartwrite,
  sox_sndtwrite,
  sox_sndtstopwrite,
  sox_sndseek
};

const sox_format_handler_t *sox_sndrtool_format_fn(void);

const sox_format_handler_t *sox_sndrtool_format_fn(void)
{
    return &sox_snd_format;
}