#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/file.h>
#include <sys/param.h>
#include <Audio.h>
#include <AudioFile.h>
#include <AudioPipe.h>
#include <AudioRawPipe.h>
#include <AudioLib.h>
#include <AudioTypePcm.h>
#include <AudioTypeG72X.h>
#include <AudioTypeChannel.h>
#include <AudioTypeMux.h>
#include <AudioTypeSampleRate.h>
#include <convert.h>
#define CVTMAXTIME ((double)5.0)
#define CVTMAXBUF (64 * 1024)
struct conv_list {
struct conv_list *next;
unsigned bufcnt;
AudioTypeConvert* conv;
AudioHdr hdr;
char *desc;
};
int
verify_conversion(
AudioHdr ihdr,
AudioHdr ohdr)
{
char *enc1;
char *enc2;
if (((ihdr.encoding != ULAW) &&
(ihdr.encoding != ALAW) &&
(ihdr.encoding != LINEAR) &&
(ihdr.encoding != FLOAT) &&
(ihdr.encoding != G721) &&
(ihdr.encoding != G723)) ||
((ohdr.encoding != ULAW) &&
(ohdr.encoding != ALAW) &&
(ohdr.encoding != LINEAR) &&
(ohdr.encoding != FLOAT) &&
(ohdr.encoding != G721) &&
(ohdr.encoding != G723))) {
enc1 = ihdr.EncodingString();
enc2 = ohdr.EncodingString();
Err(MGET("can't convert from %s to %s\n"), enc1, enc2);
delete enc1;
delete enc2;
return (-1);
}
return (0);
}
int
noop_conversion(
AudioHdr ihdr,
AudioHdr ohdr,
format_type i_fmt,
format_type o_fmt,
off_t i_offset,
off_t )
{
if ((ihdr == ohdr) &&
(i_fmt == o_fmt) &&
(i_offset == 0)) {
return (1);
}
return (0);
}
struct conv_list
*get_last_conv(
struct conv_list *list)
{
struct conv_list *lp;
for (lp = list; lp != NULL; lp = lp->next) {
if (lp->next == NULL)
break;
}
return (lp);
}
void
free_conv_list(
struct conv_list *&list)
{
unsigned int i;
unsigned int bufs;
struct conv_list *tlp;
AudioTypeConvert* conv;
while (list != NULL) {
bufs = list->bufcnt;
conv = list->conv;
for (i = 0; i < bufs; i++) {
if (list[i].desc != NULL)
free(list[i].desc);
if ((list[i].conv != NULL) &&
((i == 0) || (list[i].conv != conv)))
delete(list[i].conv);
}
tlp = list->next;
free((char *)list);
list = tlp;
}
}
void
append_conv_list(
struct conv_list *&list,
AudioHdr tohdr,
unsigned int bufs,
AudioTypeConvert* conv,
char *desc)
{
unsigned int i;
struct conv_list *lp;
struct conv_list *nlp;
Boolean B;
nlp = new struct conv_list[bufs];
if (nlp == NULL) {
Err(MGET("out of memory\n"));
exit(1);
}
B = tohdr.Validate();
for (i = 0; i < bufs; i++) {
nlp[i].next = NULL;
nlp[i].hdr = tohdr;
B = nlp[i].hdr.Validate();
nlp[i].bufcnt = bufs;
nlp[i].conv = conv;
if (desc && *desc) {
nlp[i].desc = strdup(desc);
} else {
nlp[i].desc = NULL;
}
}
if (list == NULL) {
list = nlp;
} else {
lp = get_last_conv(list);
lp->next = nlp;
}
}
AudioError
add_mux_convert(
struct conv_list *&list,
AudioHdr& ihdr,
unsigned int& bufs)
{
AudioTypeConvert* conv;
unsigned int n;
char *msg;
conv = new AudioTypeMux;
if (!conv->CanConvert(ihdr)) {
error: delete conv;
return (AUDIO_ERR_FORMATLOCK);
}
if (bufs == 1) {
n = ihdr.channels;
ihdr.channels = 1;
msg = MGET("Split multi-channel data");
} else {
ihdr.channels = bufs;
n = 1;
bufs = 1;
msg = MGET("Interleave multi-channel data");
}
if (!conv->CanConvert(ihdr))
goto error;
append_conv_list(list, ihdr, bufs, conv, msg);
bufs = n;
return (AUDIO_SUCCESS);
}
AudioError
add_pcm_convert(
struct conv_list *&list,
AudioHdr& ihdr,
AudioEncoding tofmt,
unsigned int unitsz,
unsigned int& bufs)
{
AudioTypeConvert* conv;
char msg[BUFSIZ];
char *infmt;
char *outfmt;
AudioError err;
conv = new AudioTypePcm;
if (!conv->CanConvert(ihdr)) {
error: delete conv;
return (AUDIO_ERR_FORMATLOCK);
}
infmt = ihdr.EncodingString();
ihdr.encoding = tofmt;
ihdr.bytes_per_unit = unitsz;
ihdr.samples_per_unit = 1;
if (!conv->CanConvert(ihdr))
goto error;
outfmt = ihdr.EncodingString();
sprintf(msg, MGET("Convert %s to %s"), infmt, outfmt);
delete infmt;
delete outfmt;
append_conv_list(list, ihdr, bufs, conv, msg);
return (AUDIO_SUCCESS);
}
AudioError
add_channel_convert(
struct conv_list *&list,
AudioHdr& ihdr,
unsigned int tochans,
unsigned int& bufs)
{
AudioTypeConvert* conv;
char msg[BUFSIZ];
char *inchans;
char *outchans;
AudioError err;
if (((ihdr.channels != 1) && (tochans != 1)) || (bufs != 1))
return (AUDIO_ERR_FORMATLOCK);
conv = new AudioTypeChannel;
if (!conv->CanConvert(ihdr) || (ihdr.channels != 1)) {
if (err = add_pcm_convert(list, ihdr, LINEAR, 2, bufs)) {
delete conv;
return (err);
}
if (!conv->CanConvert(ihdr)) {
error: delete conv;
return (AUDIO_ERR_FORMATLOCK);
}
}
inchans = ihdr.ChannelString();
ihdr.channels = tochans;
if (!conv->CanConvert(ihdr))
goto error;
outchans = ihdr.ChannelString();
sprintf(msg, MGET("Convert %s to %s"), inchans, outchans);
delete inchans;
delete outchans;
append_conv_list(list, ihdr, bufs, conv, msg);
return (AUDIO_SUCCESS);
}
AudioError
add_compress(
struct conv_list *&list,
AudioHdr& ihdr,
AudioEncoding tofmt,
unsigned int unitsz,
unsigned int& bufs)
{
AudioTypeConvert* conv;
char msg[BUFSIZ];
char *infmt;
char *outfmt;
struct conv_list *lp;
int i;
AudioError err;
if ((tofmt != G721) && (tofmt != G723))
return (AUDIO_ERR_FORMATLOCK);
conv = new AudioTypeG72X;
if (!conv->CanConvert(ihdr)) {
if (err = add_pcm_convert(list, ihdr, LINEAR, 2, bufs)) {
delete conv;
return (err);
}
if (!conv->CanConvert(ihdr)) {
error: delete conv;
return (AUDIO_ERR_FORMATLOCK);
}
}
infmt = ihdr.EncodingString();
ihdr.encoding = tofmt;
switch (tofmt) {
case G721:
ihdr.bytes_per_unit = unitsz;
ihdr.samples_per_unit = 2;
break;
case G723:
ihdr.bytes_per_unit = unitsz;
ihdr.samples_per_unit = 8;
break;
}
if (!conv->CanConvert(ihdr))
goto error;
outfmt = ihdr.EncodingString();
sprintf(msg, MGET("Convert %s to %s"), infmt, outfmt);
delete infmt;
delete outfmt;
append_conv_list(list, ihdr, bufs, NULL, msg);
lp = get_last_conv(list);
for (i = 0; i < bufs; i++) {
if (i == 0)
lp[i].conv = conv;
else
lp[i].conv = new AudioTypeG72X;
}
return (AUDIO_SUCCESS);
}
AudioError
add_decompress(
struct conv_list *&list,
AudioHdr& ihdr,
AudioEncoding tofmt,
unsigned int unitsz,
unsigned int& bufs)
{
AudioTypeConvert* conv;
char msg[BUFSIZ];
char *infmt;
char *outfmt;
struct conv_list *lp;
int i;
AudioError err;
if ((ihdr.encoding != G721) && (ihdr.encoding != G723))
return (AUDIO_ERR_FORMATLOCK);
conv = new AudioTypeG72X;
if (!conv->CanConvert(ihdr)) {
error: delete conv;
return (AUDIO_ERR_FORMATLOCK);
}
infmt = ihdr.EncodingString();
ihdr.encoding = tofmt;
ihdr.bytes_per_unit = unitsz;
ihdr.samples_per_unit = 1;
if (!conv->CanConvert(ihdr)) {
ihdr.encoding = LINEAR;
ihdr.bytes_per_unit = 2;
if (!conv->CanConvert(ihdr))
goto error;
}
outfmt = ihdr.EncodingString();
sprintf(msg, MGET("Convert %s to %s"), infmt, outfmt);
delete infmt;
delete outfmt;
append_conv_list(list, ihdr, bufs, NULL, msg);
lp = get_last_conv(list);
for (i = 0; i < bufs; i++) {
if (i == 0)
lp[i].conv = conv;
else
lp[i].conv = new AudioTypeG72X;
}
return (AUDIO_SUCCESS);
}
AudioError
add_rate_convert(
struct conv_list *&list,
AudioHdr& ihdr,
unsigned int torate,
unsigned int& bufs)
{
AudioTypeConvert* conv;
unsigned int fromrate;
char msg[BUFSIZ];
char *inrate;
char *outrate;
struct conv_list *lp;
int i;
AudioError err;
fromrate = ihdr.sample_rate;
conv = new AudioTypeSampleRate(fromrate, torate);
if (!conv->CanConvert(ihdr)) {
if (err = add_pcm_convert(list, ihdr, LINEAR, 2, bufs)) {
delete conv;
return (err);
}
if (!conv->CanConvert(ihdr)) {
error: delete conv;
return (AUDIO_ERR_FORMATLOCK);
}
}
inrate = ihdr.RateString();
ihdr.sample_rate = torate;
if (!conv->CanConvert(ihdr))
goto error;
outrate = ihdr.RateString();
sprintf(msg, MGET("Convert %s to %s"), inrate, outrate);
delete inrate;
delete outrate;
append_conv_list(list, ihdr, bufs, NULL, msg);
lp = get_last_conv(list);
for (i = 0; i < bufs; i++) {
if (i == 0)
lp[i].conv = conv;
else
lp[i].conv = new AudioTypeSampleRate(fromrate, torate);
}
return (AUDIO_SUCCESS);
}
Boolean
pcmtype(
AudioHdr& hdr)
{
if (hdr.samples_per_unit != 1)
return (FALSE);
switch (hdr.encoding) {
case LINEAR:
case FLOAT:
case ULAW:
case ALAW:
return (TRUE);
}
return (FALSE);
}
#define IS_PCM(ihp) (pcmtype(ihp))
#define IS_MONO(ihp) (ihp.channels == 1)
#define RATE_CONV(ihp, ohp) (ihp.sample_rate != ohp.sample_rate)
#define ENC_CONV(ihp, ohp) ((ihp.encoding != ohp.encoding) || \
(ihp.samples_per_unit != \
ohp.samples_per_unit) || \
(ihp.bytes_per_unit != ohp.bytes_per_unit))
#define CHAN_CONV(ihp, ohp) (ihp.channels != ohp.channels)
AudioError
build_conversion_list(
struct conv_list *&list,
AudioStream* ifp,
AudioStream* ofp)
{
AudioHdr ihdr;
AudioHdr ohdr;
unsigned int bufs;
AudioError err;
ihdr = ifp->GetHeader();
ohdr = ofp->GetHeader();
bufs = 1;
while (((ihdr != ohdr) || (bufs != 1)) && !err) {
if (IS_MONO(ohdr)) {
if (!IS_MONO(ihdr)) {
if (IS_PCM(ihdr)) {
err = add_channel_convert(list,
ihdr, 1, bufs);
} else {
err = add_mux_convert(list, ihdr, bufs);
}
continue;
} else if (bufs != 1) {
if (IS_PCM(ihdr)) {
err = add_mux_convert(list, ihdr, bufs);
} else {
err = add_decompress(list, ihdr,
ohdr.encoding, ohdr.bytes_per_unit,
bufs);
}
continue;
}
} else if (ihdr.channels != 1) {
if (RATE_CONV(ihdr, ohdr) ||
(ENC_CONV(ihdr, ohdr) &&
(!IS_PCM(ihdr) || !IS_PCM(ohdr)))) {
err = add_mux_convert(list, ihdr, bufs);
continue;
}
}
if (RATE_CONV(ihdr, ohdr)) {
if (!IS_PCM(ihdr)) {
err = add_decompress(list, ihdr,
ohdr.encoding, ohdr.bytes_per_unit,
bufs);
} else {
err = add_rate_convert(list, ihdr,
ohdr.sample_rate, bufs);
}
continue;
}
if (ENC_CONV(ihdr, ohdr)) {
if (!IS_PCM(ihdr)) {
err = add_decompress(list, ihdr,
ohdr.encoding, ohdr.bytes_per_unit,
bufs);
} else if (IS_PCM(ohdr)) {
err = add_pcm_convert(list, ihdr,
ohdr.encoding, ohdr.bytes_per_unit,
bufs);
} else {
err = add_compress(list, ihdr,
ohdr.encoding, ohdr.bytes_per_unit,
bufs);
}
continue;
}
if (bufs > 1) {
err = add_mux_convert(list, ihdr, bufs);
continue;
}
if (!IS_MONO(ohdr)) {
err = add_channel_convert(list,
ihdr, ohdr.channels, bufs);
continue;
}
return (AUDIO_ERR_FORMATLOCK);
}
return (err);
}
int
do_convert(
AudioStream* ifp,
AudioStream* ofp)
{
struct conv_list *list = NULL;
struct conv_list *lp;
AudioBuffer* obuf;
AudioBuffer** multibuf;
AudioError err;
AudioHdr ihdr;
AudioHdr ohdr;
Double pos = 0.0;
size_t len;
unsigned int i;
Double cvtlen;
char *msg1;
char *msg2;
ihdr = ifp->GetHeader();
ohdr = ofp->GetHeader();
if ((err = build_conversion_list(list, ifp, ofp)) != AUDIO_SUCCESS) {
free_conv_list(list);
msg1 = ohdr.FormatString();
Err(MGET("Cannot convert %s to %s\n"), ifp->GetName(), msg1);
delete msg1;
return (-1);
}
if ((ohdr.sample_rate < 8000) || (ohdr.sample_rate > 48000)) {
msg1 = ohdr.RateString();
Err(MGET("Warning: converting %s to %s\n"),
ifp->GetName(), msg1);
delete msg1;
}
if (ohdr.channels > 2) {
msg1 = ohdr.ChannelString();
Err(MGET("Warning: converting %s to %s\n"),
ifp->GetName(), msg1);
delete msg1;
}
if (Debug) {
msg1 = ihdr.FormatString();
msg2 = ohdr.FormatString();
Err(MGET("Converting %s:\n\t\tfrom: %s\n\t\tto: %s\n"),
ifp->GetName(), msg1, msg2);
delete msg1;
delete msg2;
for (lp = list; lp; lp = lp->next) {
(void) fprintf(stderr, MGET("\t%s %s\n"), lp->desc,
(lp->bufcnt == 1) ? "" : MGET("(multi-channel)"));
}
}
cvtlen = ihdr.Bytes_to_Time(CVTMAXBUF);
if (cvtlen > CVTMAXTIME)
cvtlen = CVTMAXTIME;
if (cvtlen > ohdr.Bytes_to_Time(CVTMAXBUF * 4))
cvtlen = ohdr.Bytes_to_Time(CVTMAXBUF * 4);
if (!(obuf = new AudioBuffer(cvtlen, MGET("Audio Convert Buffer")))) {
Err(MGET("Can't create conversion buffer\n"));
exit(1);
}
while (1) {
len = (size_t)ihdr.Time_to_Bytes(cvtlen);
if ((err = obuf->SetHeader(ihdr)) != AUDIO_SUCCESS) {
Err(MGET("Can't set buffer header: %s\n"), err.msg());
return (-1);
}
if (obuf->GetSize() < cvtlen)
obuf->SetSize(0.);
obuf->SetSize(cvtlen);
if (err = ifp->ReadData(obuf->GetAddress(), len, pos))
break;
obuf->SetLength(ihdr.Bytes_to_Time(len));
for (lp = list; lp; lp = lp->next) {
if (lp->conv) {
if (lp->bufcnt == 1) {
err = lp->conv->Convert(obuf, lp->hdr);
} else {
multibuf = (AudioBuffer**)obuf;
for (i = 0; i < lp->bufcnt; i++) {
err = lp[i].conv->Convert(
multibuf[i], lp[i].hdr);
if (err)
break;
}
}
if (err) {
Err(MGET(
"Conversion failed: %s (%s)\n"),
lp->desc ? lp->desc : MGET("???"),
err.msg());
return (-1);
}
}
}
if ((err = write_output(obuf, ofp)) != AUDIO_SUCCESS) {
Err(MGET("Error writing to output file %s (%s)\n"),
ofp->GetName(), err.msg());
return (-1);
}
}
obuf->SetLength(0.0);
for (lp = list; lp; lp = lp->next) {
if (lp->conv) {
if (lp->bufcnt == 1) {
err = lp->conv->Convert(obuf, lp->hdr);
if (!err)
err = lp->conv->Flush(obuf);
} else {
multibuf = (AudioBuffer**)obuf;
for (i = 0; i < lp->bufcnt; i++) {
err = lp[i].conv->Convert(
multibuf[i], lp[i].hdr);
if (!err) {
err = lp[i].conv->Flush(
multibuf[i]);
}
if (err)
break;
}
}
if (err) {
Err(MGET(
"Warning: Flush of final bytes failed: "
"%s (%s)\n"),
lp->desc ? lp->desc : MGET("???"),
err.msg());
break;
}
}
}
if (obuf->GetLength() > 0.0) {
if ((err = write_output(obuf, ofp)) != AUDIO_SUCCESS) {
Err(MGET("Warning: Final write to %s failed (%s)\n"),
ofp->GetName(), err.msg());
}
}
delete obuf;
free_conv_list(list);
return (0);
}