cdrtools/cdrecord/audiosize.c
2025-06-15 04:19:58 +08:00

286 lines
7.2 KiB
C

/* @(#)audiosize.c 1.23 09/07/05 Copyright 1998-2009 J. Schilling */
#include <schily/mconfig.h>
#ifndef lint
static UConst char sccsid[] =
"@(#)audiosize.c 1.23 09/07/05 Copyright 1998-2009 J. Schilling";
#endif
/*
* Copyright (c) 1998-2009 J. Schilling
*
* First .vaw implementation made by Dave Platt <dplatt@iq.nc.com>
* Current .wav implementation with additional help from Heiko Eißfeld.
*/
/*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License, Version 1.0 only
* (the "License"). You may not use this file except in compliance
* with the License.
*
* See the file CDDL.Schily.txt in this distribution for details.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file CDDL.Schily.txt from this distribution.
*/
#include <schily/mconfig.h>
#include <schily/stat.h>
#include <schily/unistd.h>
#include <schily/standard.h>
#include <schily/utypes.h>
#include <schily/string.h>
#include <schily/intcvt.h>
#include <schily/schily.h>
#include <scg/scgcmd.h>
#include "auheader.h"
typedef struct {
Uchar magic[4];
Uchar hdr_size[4];
Uchar data_size[4];
Uchar encoding[4];
Uchar sample_rate[4];
Uchar channels[4];
} sun_au_t;
#define SUN_AU_MAGIC ".snd"
#define SUN_AU_UNKNOWN_LEN ((Uint)~0)
#define SUN_AU_ULAW8 1 /* American ISDN Telephonie */
#define SUN_AU_LINEAR8 2 /* Linear PCM 8 bit/channel */
#define SUN_AU_LINEAR16 3 /* Linear PCM 16 bit/channel */
#define SUN_AU_LINEAR24 4 /* Linear PCM 24 bit/channel */
#define SUN_AU_LINEAR32 5 /* Linear PCM 32 bit/channel */
#define SUN_AU_FLOAT 6 /* 32 bit IEEE floatingpoint */
#define SUN_AU_DOUBLE 7 /* 64 bit IEEE floatingpoint */
#define SUN_AU_G721 23 /* 4 bit CCITT G.721 ADPCM */
#define SUN_AU_G722 24 /* CCITT G.722 ADPCM */
#define SUN_AU_G723_3 25 /* 3 bit CCITT G.723 ADPCM */
#define SUN_AU_G723_5 26 /* 5 bit CCITT G.723 ADPCM */
#define SUN_AU_ALAW8 27 /* International ISDN Tel. */
typedef struct {
Uchar ckid[4];
Uchar cksize[4];
} chunk_t;
typedef struct {
Uchar wave[4];
} riff_chunk;
typedef struct {
Uchar fmt_tag[2];
Uchar channels[2];
Uchar sample_rate[4];
Uchar av_byte_rate[4];
Uchar block_size[2];
Uchar bits_per_sample[2];
} fmt_chunk;
#define WAV_RIFF_MAGIC "RIFF" /* Magic for file format */
#define WAV_WAVE_MAGIC "WAVE" /* Magic for Waveform Audio */
#define WAV_FMT_MAGIC "fmt " /* Start of Waveform format */
#define WAV_DATA_MAGIC "data" /* Start of data chunk */
#define WAV_FORMAT_PCM 0x0001 /* Linear PCM format */
#define WAV_FORMAT_ULAW 0x0101 /* American ISDN Telephonie */
#define WAV_FORMAT_ALAW 0x0102 /* International ISDN Tel. */
#define WAV_FORMAT_ADPCM 0x0103 /* ADPCM format */
#define le_a_to_u_short(a) ((unsigned short) \
((((unsigned char *)a)[0] & 0xFF) | \
(((unsigned char *)a)[1] << 8 & 0xFF00)))
#ifdef __STDC__
#define le_a_to_u_long(a) ((unsigned long) \
((((unsigned char *)a)[0] & 0xFF) | \
(((unsigned char *)a)[1] << 8 & 0xFF00) | \
(((unsigned char *)a)[2] << 16 & 0xFF0000) | \
(((unsigned char *)a)[3] << 24 & 0xFF000000UL)))
#else
#define le_a_to_u_long(a) ((unsigned long) \
((((unsigned char *)a)[0] & 0xFF) | \
(((unsigned char *)a)[1] << 8 & 0xFF00) | \
(((unsigned char *)a)[2] << 16 & 0xFF0000) | \
(((unsigned char *)a)[3] << 24 & 0xFF000000)))
#endif
EXPORT BOOL is_auname __PR((const char *name));
EXPORT off_t ausize __PR((int f));
EXPORT BOOL is_wavname __PR((const char *name));
EXPORT off_t wavsize __PR((int f));
EXPORT BOOL
is_auname(name)
const char *name;
{
const char *p;
if ((p = strrchr(name, '.')) == NULL)
return (FALSE);
return (streql(p, ".au"));
}
/*
* Read Sun audio header, leave file seek pointer past auheader.
*/
EXPORT off_t
ausize(f)
int f;
{
sun_au_t hdr;
struct stat sb;
mode_t mode;
off_t size;
Int32_t val;
long ret = AU_BAD_HEADER;
/*
* First check if a bad guy tries to call ausize()
* with an unappropriate file descriptor.
* return -1 in this case.
*/
if (isatty(f))
return (-1L);
if (fstat(f, &sb) < 0)
return (-1L);
mode = sb.st_mode & S_IFMT;
if (!S_ISREG(mode) && !S_ISBLK(mode) && !S_ISCHR(mode))
return (-1L);
lseek(f, (off_t)0L, SEEK_SET);
if (read(f, &hdr, sizeof (hdr)) != sizeof (hdr))
goto err;
if (strncmp((char *)hdr.magic, SUN_AU_MAGIC, 4) != 0)
goto err;
ret = AU_BAD_CODING;
val = a_to_u_4_byte(hdr.encoding);
if (val != SUN_AU_LINEAR16)
goto err;
val = a_to_u_4_byte(hdr.channels);
if (val != 2)
goto err;
val = a_to_u_4_byte(hdr.sample_rate);
if (val != 44100)
goto err;
size = (off_t)a_to_u_4_byte(hdr.hdr_size);
if (size < (off_t)sizeof (hdr) || size > 512)
goto err;
lseek(f, size, SEEK_SET);
/*
* Most .au files don't seem to honor the data_size field,
* so we use the whole file size without the header.
*/
size = sb.st_size - size;
return (size);
err:
lseek(f, (off_t)0L, SEEK_SET);
return ((off_t)ret);
}
EXPORT BOOL
is_wavname(name)
const char *name;
{
const char *p;
if ((p = strrchr(name, '.')) == NULL)
return (FALSE);
return (streql(p, ".wav") || streql(p, ".WAV"));
}
/*
* Read WAV header, leave file seek pointer past WAV header.
*/
EXPORT off_t
wavsize(f)
int f;
{
chunk_t chunk;
riff_chunk riff;
fmt_chunk fmt;
struct stat sb;
off_t cursor;
BOOL gotFormat;
mode_t mode;
off_t size;
long ret = AU_BAD_HEADER;
/*
* First check if a bad guy tries to call wavsize()
* with an unappropriate file descriptor.
* return -1 in this case.
*/
if (isatty(f))
return (-1L);
if (fstat(f, &sb) < 0)
return (-1L);
mode = sb.st_mode & S_IFMT;
if (!S_ISREG(mode) && !S_ISBLK(mode) && !S_ISCHR(mode))
return (-1L);
lseek(f, (off_t)0L, SEEK_SET);
cursor = (off_t)0;
gotFormat = FALSE;
for (;;) {
if (read(f, &chunk, sizeof (chunk)) != sizeof (chunk))
goto err;
size = (off_t)le_a_to_u_long(chunk.cksize);
if (strncmp((char *)chunk.ckid, WAV_RIFF_MAGIC, 4) == 0) {
/*
* We found (first) RIFF header. Check if a WAVE
* magic follows. Set up size to be able to skip
* past this header.
*/
if (read(f, &riff, sizeof (riff)) != sizeof (riff))
goto err;
if (strncmp((char *)riff.wave, WAV_WAVE_MAGIC, 4) != 0)
goto err;
size = (off_t)sizeof (riff);
} else if (strncmp((char *)chunk.ckid, WAV_FMT_MAGIC, 4) == 0) {
/*
* We found WAVE "fmt " header. Check size (if it is
* valid for a WAVE file) and coding whether it is
* useable for a CD.
*/
if (size < (off_t)sizeof (fmt)) goto err;
if (sizeof (fmt) != read(f, &fmt, sizeof (fmt))) goto err;
if (le_a_to_u_short(fmt.channels) != 2 ||
le_a_to_u_long(fmt.sample_rate) != 44100 ||
le_a_to_u_short(fmt.bits_per_sample) != 16) {
ret = AU_BAD_CODING;
goto err;
}
gotFormat = TRUE;
} else if (strncmp((char *)chunk.ckid, WAV_DATA_MAGIC, 4) == 0) {
/*
* We found WAVE "data" header. This contains the
* size value of the audio part.
*/
if (!gotFormat) {
ret = AU_BAD_CODING;
goto err;
}
if ((cursor + size + sizeof (chunk)) > sb.st_size)
size = sb.st_size - (cursor + sizeof (chunk));
return (size);
}
cursor += size + sizeof (chunk);
lseek(f, cursor, SEEK_SET); /* Skip over current chunk */
}
err:
lseek(f, (off_t)0L, SEEK_SET);
return (ret);
}