ref: 7ef50e0130b8cdbf43f050f0de23e855e4a187f3
parent: d6185567e2757c9bd74defdfe7f227e59f2d6cd2
author: robs <robs>
date: Tue Jan 23 16:58:34 EST 2007
Can now use Ctrl-C to skip to next track when playing multiple files.
--- a/ChangeLog
+++ b/ChangeLog
@@ -22,8 +22,8 @@
o Allow the rate and number of channels of .au files to be overridden
by command-line arguments. (robs@users.sf.net)
o Add seek support for GSM data in WAV files. Rafal Maszkowski
- o Allow encoding quality to be specified (works for Ogg too, but not
- MP3).
+ o Allow encoding quality to be specified (FLAC & Ogg, but not
+ MP3 yet). (robs)
o Rename -b to -1, -w to -2, -l to -4, -d to -8, and mask to dither.
(robs)
o New options for specifying endianness (and separate options for
@@ -70,6 +70,8 @@
comment. (robs)
o Options to apply replay-gain on input; enabled by default
with `play'. (robs)
+ o Can now use Ctrl-C to skip to next track when playing multiple
+ files (e.g. play *.mp3); Ctrl-C twice to exit. (robs)
o Display (auto-)selected effects with -V. (robs)
o Added --interactive option to prompt to overwrite pre-existing
output file. (robs)
--- a/src/sox.c
+++ b/src/sox.c
@@ -29,6 +29,7 @@
#include <stdlib.h>
#include <signal.h>
#include <time.h>
+#include <sys/timeb.h>
#include <errno.h>
#ifdef HAVE_UNISTD_H
#include <unistd.h> /* for unlink() */
@@ -61,10 +62,11 @@
static st_bool interactive = st_false;
static st_globalinfo_t globalinfo = {st_false, 1};
static st_bool uservolume = st_false;
-typedef enum {RG_OFF, RG_TRACK, RG_ALBUM} rg_t;
-static rg_t replay_gain_mode;
+typedef enum {RG_OFF, RG_TRACK, RG_ALBUM} rg_t;
+static rg_t replay_gain_mode;
-static int user_abort = 0;
+static st_bool user_abort = st_false;
+static st_bool user_skip = st_false;
static int success = 0;
static st_option_t show_progress = ST_OPTION_DEFAULT;
@@ -174,7 +176,7 @@
}
/* Cleanup atexit() function, hence always called. */
-static void cleanup(void)
+static void cleanup(void)
{
size_t i;
ft_t ft = ofile;
@@ -190,7 +192,7 @@
if (!(ft->h->flags & ST_FILE_NOSTDIO)) {
struct stat st;
fstat(fileno(ft->fp), &st);
-
+
/* If we didn't succeed and we created an output file, remove it. */
if (!success && (st.st_mode & S_IFMT) == S_IFREG)
unlink(ft->filename);
@@ -202,12 +204,6 @@
}
}
-static void sigint(int s)
-{
- if (s == SIGINT || s == SIGTERM)
- user_abort = 1;
-}
-
static file_info_t make_file_info(void)
{
file_info_t fi = xcalloc(sizeof(*fi), 1);
@@ -282,7 +278,7 @@
rec = st_true;
}
- /* Loop over arguments and filenames, stop when an effect name is
+ /* Loop over arguments and filenames, stop when an effect name is
* found. */
while (optind < argc && !is_effect_name(argv[optind])) {
fi = make_file_info();
@@ -340,7 +336,7 @@
set_device(fo);
file_opts[0] = fo;
}
-
+
/* Make sure we got at least the required # of input filenames */
input_count = file_count ? file_count - 1 : 0;
if (input_count < (combine_method == SOX_CONCAT ? 1 : 2))
@@ -356,7 +352,7 @@
if (file_opts[i]->volume != HUGE_VAL)
usage("-v can only be given for an input file;\n"
"\tuse 'vol' to set the output file volume");
-
+
for (i = 0; i < input_count; i++) {
int j = input_count - 1 - i; /* Open in reverse order 'cos of rec (below) */
file_info_t fi = file_opts[j];
@@ -366,7 +362,7 @@
* this, and will override it, possibly causing clipping to occur. */
if (combine_method == SOX_MIX && !uservolume)
fi->volume = 1.0 / input_count;
-
+
if (rec && !j) { /* Set the recording sample rate & # of channels: */
if (input_count > 1) { /* Get them from the next input file: */
fi->signal.rate = file_desc[1]->signal.rate;
@@ -389,7 +385,7 @@
if (file_desc[j]->comment)
set_replay_gain(file_desc[j]->comment, fi);
}
-
+
/* Loop through the rest of the arguments looking for effects */
parse_effects(argc, argv);
@@ -531,7 +527,7 @@
case 5:
fi->signal.lua_script = xstrdup(optarg);
break;
-
+
case 6:
globalinfo.octave_plot_effect = st_true;
break;
@@ -690,7 +686,7 @@
"Endian Type : %s\n"
"Reverse Nibbles: %s\n"
"Reverse Bits : %s\n",
- f->signal.size == 1? "N/A" :
+ f->signal.size == 1? "N/A" :
f->signal.reverse_bytes != ST_IS_BIGENDIAN? "big" : "little",
no_yes[f->signal.reverse_nibbles],
no_yes[f->signal.reverse_bits]);
@@ -725,7 +721,21 @@
if (file_opts[f]->replay_gain != HUGE_VAL)
file_opts[f]->volume *= pow(10, file_opts[f]->replay_gain / 20);
}
-
+
+static void sigint(int s)
+{
+ static struct timeb then;
+ struct timeb now;
+ time_t secs;
+ ftime(&now);
+ secs = now.time - then.time;
+ if (show_progress && s == SIGINT && combine_method == SOX_CONCAT &&
+ (secs > 1 || 1000 * secs + now.millitm - then.millitm > 999))
+ user_skip = st_true;
+ else user_abort = st_true;
+ then = now;
+}
+
/*
* Process input file -> effect table -> output file one buffer at a time
*/
@@ -748,7 +758,7 @@
st_fail("Input files must have the same rate and # of channels");
exit(1);
}
-
+
{
st_loopinfo_t loops[ST_MAX_NLOOPS];
double factor;
@@ -755,7 +765,7 @@
int i;
file_info_t info = file_opts[file_count - 1];
char const *comment = NULL;
-
+
if (info->signal.rate == 0)
info->signal.rate = file_desc[0]->signal.rate;
if (info->signal.size == -1)
@@ -764,12 +774,12 @@
info->signal.encoding = file_desc[0]->signal.encoding;
if (info->signal.channels == 0)
info->signal.channels = file_desc[0]->signal.channels;
-
+
if (info->comment == NULL)
comment = file_desc[0]->comment ? file_desc[0]->comment : "Processed by SoX";
else if (*info->comment != '\0')
comment = info->comment;
-
+
/*
* copy loop info, resizing appropriately
* it's in samples, so # channels don't matter
@@ -776,7 +786,7 @@
* FIXME: This doesn't work for multi-file processing or
* effects that change file length.
*/
- factor = (double) info->signal.rate / (double)
+ factor = (double) info->signal.rate / (double)
file_desc[0]->signal.rate;
for (i = 0; i < ST_MAX_NLOOPS; i++) {
loops[i].start = file_desc[0]->loops[i].start * factor;
@@ -784,20 +794,20 @@
loops[i].count = file_desc[0]->loops[i].count;
loops[i].type = file_desc[0]->loops[i].type;
}
-
+
ofile = st_open_write(overwrite_permitted,
info->filename,
- &info->signal,
+ &info->signal,
info->filetype,
comment,
&file_desc[0]->instr,
loops);
-
+
if (!ofile)
/* st_open_write() will call st_warn for most errors.
* Rely on that printing something. */
exit(2);
-
+
/* When writing to an audio device, auto turn on the
* progress display to match behavior of ogg123,
* unless the user requested us not to display anything. */
@@ -807,7 +817,7 @@
report_file_info(file_count - 1);
}
-
+
/* Adjust the input rate for the speed effect */
for (f = 0; f < input_count; f++)
file_desc[f]->signal.rate = file_desc[f]->signal.rate * globalinfo.speed + .5;
@@ -830,7 +840,7 @@
/* Treat overall length the same as longest input file. */
if (file_desc[f]->length > input_samples)
input_samples = file_desc[f]->length;
-
+
if (combine_method == SOX_MERGE) {
alloc_size /= input_count;
file_desc[f]->signal.channels /= input_count;
@@ -843,7 +853,7 @@
input_samples = file_desc[current_input]->length;
progress_to_file(current_input);
}
-
+
/*
* Just like errno, we must set st_errno to known values before
* calling I/O operations.
@@ -850,15 +860,15 @@
*/
for (f = 0; f < file_count; f++)
file_desc[f]->st_errno = 0;
-
+
input_eff = 0;
input_eff_eof = 0;
-
+
/* mark chain as empty */
for(e = 1; e < neffects; e++)
efftab[e].odone = efftab[e].olen = 0;
-
- /* Run input data through effects and get more until olen == 0
+
+ /* Run input data through effects and get more until olen == 0
* (or ST_EOF) or user-abort.
*/
signal(SIGINT, sigint);
@@ -865,22 +875,27 @@
signal(SIGTERM, sigint);
do {
if (combine_method == SOX_CONCAT) {
- ilen[0] = st_read(file_desc[current_input], efftab[0].obuf,
- (st_ssize_t)ST_BUFSIZ);
- if (ilen[0] > ST_BUFSIZ) {
- st_warn("WARNING: Corrupt value of %d! Assuming 0 bytes read.", ilen);
- ilen[0] = 0;
+ if (user_skip) {
+ ilen[0] = ST_EOF;
+ user_skip = st_false;
+ fprintf(stderr, "\nSkipped.");
+ } else {
+ ilen[0] = st_read(file_desc[current_input], efftab[0].obuf,
+ (st_ssize_t)ST_BUFSIZ);
+ if (ilen[0] > ST_BUFSIZ) {
+ st_warn("WARNING: Corrupt value of %d! Assuming 0 bytes read.", ilen);
+ ilen[0] = 0;
+ }
+
+ if (ilen[0] == ST_EOF) {
+ efftab[0].olen = 0;
+ if (file_desc[current_input]->st_errno)
+ fprintf(stderr, file_desc[current_input]->st_errstr);
+ } else
+ efftab[0].olen = ilen[0];
+
+ read_samples += efftab[0].olen;
}
-
- if (ilen[0] == ST_EOF) {
- efftab[0].olen = 0;
- if (file_desc[current_input]->st_errno)
- fprintf(stderr, file_desc[current_input]->st_errstr);
- } else
- efftab[0].olen = ilen[0];
-
- read_samples += efftab[0].olen;
-
/* Some file handlers claim 0 bytes instead of returning
* ST_EOF. In either case, attempt to go to the next
* input file.
@@ -898,20 +913,20 @@
} else if (combine_method == SOX_MIX) {
for (f = 0; f < input_count; f++) {
ilen[f] = st_read(file_desc[f], ibuf[f], (st_ssize_t)ST_BUFSIZ);
-
+
if (ilen[f] == ST_EOF) {
ilen[f] = 0;
- if (file_desc[f]->st_errno)
+ if (file_desc[f]->st_errno)
fprintf(stderr, file_desc[f]->st_errstr);
}
-
+
/* Only count read samples for first file in mix */
if (f == 0)
read_samples += efftab[0].olen;
-
+
volumechange(ibuf[f], ilen[f], file_opts[f]);
}
-
+
/* FIXME: Should report if the size of the reads are not
* the same.
*/
@@ -919,9 +934,9 @@
for (f = 0; f < input_count; f++)
if ((st_size_t)ilen[f] > efftab[0].olen)
efftab[0].olen = ilen[f];
-
+
for (s = 0; s < efftab[0].olen; s++) {
- /* Mix audio by summing samples together.
+ /* Mix audio by summing samples together.
* Input side volume changes are performed above. */
for (f = 0; f < input_count; f++) {
if (f == 0)
@@ -947,7 +962,7 @@
efftab[0].olen = ilen[f];
volumechange(ibuf[f], ilen[f], file_opts[f]);
}
-
+
for (s = 0; s < efftab[0].olen; s++)
for (f = 0; f < input_count; f++)
efftab[0].obuf[s * input_count + f] =
@@ -1009,7 +1024,7 @@
file_desc[f]->h->names[0] : file_desc[f]->filename,
file_desc[f]->clips);
}
-
+
static void parse_effects(int argc, char **argv)
{
int argc_effect;
@@ -1032,7 +1047,7 @@
optind++; /* Skip past effect name */
e->globalinfo = &globalinfo;
getopts = e->h->getopts? e->h->getopts : st_effect_nothing_getopts;
- if (getopts(e, argc_effect, &argv[optind]) == ST_EOF)
+ if (getopts(e, argc_effect, &argv[optind]) == ST_EOF)
exit(2);
optind += argc_effect; /* Skip past the effect arguments */
@@ -1040,11 +1055,11 @@
}
static void add_effect(int * effects_mask)
-{
+{
struct st_effect * e = &efftab[neffects];
/* Copy format info to effect table */
- *effects_mask =
+ *effects_mask =
st_updateeffect(e, &file_desc[0]->signal, &ofile->signal, *effects_mask);
/* If this effect can't handle multiple channels then account for this. */
@@ -1059,7 +1074,7 @@
}
static void add_default_effect(char const * name, int * effects_mask)
-{
+{
struct st_effect * e = &efftab[neffects];
int (*getopts)(eff_t effp, int argc, char *argv[]);
@@ -1066,7 +1081,7 @@
/* Find effect and update initial pointers */
st_geteffect(e, name);
- /* Set up & give default opts for added effects */
+ /* Set up & give default opts for added effects */
e->globalinfo = &globalinfo;
getopts = e->h->getopts? e->h->getopts : st_effect_nothing_getopts;
if (getopts(e, 0, NULL) == ST_EOF)
@@ -1144,7 +1159,7 @@
int e, ret = ST_SUCCESS;
for (e = 1; e < neffects; e++) {
- int (*start)(eff_t effp) =
+ int (*start)(eff_t effp) =
efftab[e].h->start? efftab[e].h->start : st_effect_nothing;
efftab[e].clips = 0;
if ((ret = start(&efftab[e])) == ST_EOF)
@@ -1208,11 +1223,11 @@
*/
if (user_abort)
return ST_EOF;
-
- len = st_write(ofile,
+
+ len = st_write(ofile,
&efftab[neffects - 1].obuf[total],
efftab[neffects - 1].olen - total);
-
+
if (len != efftab[neffects - 1].olen - total || ofile->eof) {
st_warn("Error writing: %s", ofile->st_errstr);
return ST_EOF;
@@ -1223,7 +1238,7 @@
efftab[neffects-1].odone = efftab[neffects-1].olen = 0;
} else {
/* Make it look like everything was consumed */
- output_samples += (efftab[neffects-1].olen /
+ output_samples += (efftab[neffects-1].olen /
ofile->signal.channels);
efftab[neffects-1].odone = efftab[neffects-1].olen = 0;
}
@@ -1239,7 +1254,7 @@
*/
if (efftab[e].odone == efftab[e].olen)
efftab[e].odone = efftab[e].olen = 0;
-
+
if (efftab[e].odone < efftab[e].olen) {
/* Only mark that we have more data if a full
* frame that can be written.
@@ -1247,7 +1262,7 @@
* input buffer then the data will be lost and
* will cause stereo channels to be inversed.
*/
- if ((efftab[e].olen - efftab[e].odone) >=
+ if ((efftab[e].olen - efftab[e].odone) >=
ofile->signal.channels)
havedata = 1;
else
@@ -1275,7 +1290,7 @@
*/
while (input_eff < neffects) {
int rc = drain_effect(input_eff);
-
+
if (efftab[input_eff].olen == 0) {
input_eff++;
/* Assume next effect hasn't reached EOF yet. */
@@ -1329,18 +1344,18 @@
idone = efftab[e - 1].olen - efftab[e - 1].odone;
odone = ST_BUFSIZ - efftab[e].olen;
st_debug("pre %s idone=%d, odone=%d", efftab[e].name, idone, odone);
- st_debug("pre %s odone1=%d, olen1=%d odone=%d olen=%d", efftab[e].name, efftab[e-1].odone, efftab[e-1].olen, efftab[e].odone, efftab[e].olen);
+ st_debug("pre %s odone1=%d, olen1=%d odone=%d olen=%d", efftab[e].name, efftab[e-1].odone, efftab[e-1].olen, efftab[e].odone, efftab[e].olen);
effstatus = flow(&efftab[e],
&efftab[e - 1].obuf[efftab[e - 1].odone],
- &efftab[e].obuf[efftab[e].olen],
- (st_size_t *)&idone,
+ &efftab[e].obuf[efftab[e].olen],
+ (st_size_t *)&idone,
(st_size_t *)&odone);
efftab[e - 1].odone += idone;
/* Don't update efftab[e].odone as we didn't consume data */
- efftab[e].olen += odone;
- st_debug("post %s idone=%d, odone=%d", efftab[e].name, idone, odone);
+ efftab[e].olen += odone;
+ st_debug("post %s idone=%d, odone=%d", efftab[e].name, idone, odone);
st_debug("post %s odone1=%d, olen1=%d odone=%d olen=%d", efftab[e].name, efftab[e-1].odone, efftab[e-1].olen, efftab[e].odone, efftab[e].olen);
done = idone + odone;
@@ -1350,30 +1365,30 @@
*/
idone = efftab[e - 1].olen - efftab[e - 1].odone;
odone = ST_BUFSIZ - efftab[e].olen;
-
+
ibuf = &efftab[e - 1].obuf[efftab[e - 1].odone];
for (i = 0; i < idone; i += 2) {
ibufl[i / 2] = *ibuf++;
ibufr[i / 2] = *ibuf++;
}
-
+
/* left */
idonel = (idone + 1) / 2; /* odd-length logic */
odonel = odone / 2;
st_debug("pre %s idone=%d, odone=%d", efftab[e].name, idone, odone);
- st_debug("pre %s odone1=%d, olen1=%d odone=%d olen=%d", efftab[e].name, efftab[e - 1].odone, efftab[e - 1].olen, efftab[e].odone, efftab[e].olen);
-
+ st_debug("pre %s odone1=%d, olen1=%d odone=%d olen=%d", efftab[e].name, efftab[e - 1].odone, efftab[e - 1].olen, efftab[e].odone, efftab[e].olen);
+
effstatusl = flow(&efftab[e],
- ibufl, obufl, (st_size_t *)&idonel,
+ ibufl, obufl, (st_size_t *)&idonel,
(st_size_t *)&odonel);
-
+
/* right */
idoner = idone / 2; /* odd-length logic */
odoner = odone / 2;
effstatusr = flow(&efftabR[e],
- ibufr, obufr, (st_size_t *)&idoner,
+ ibufr, obufr, (st_size_t *)&idoner,
(st_size_t *)&odoner);
-
+
obuf = &efftab[e].obuf[efftab[e].olen];
/* This loop implies left and right effect will always output
* the same amount of data.
@@ -1385,11 +1400,11 @@
efftab[e-1].odone += idonel + idoner;
/* Don't zero efftab[e].odone since nothing has been consumed yet */
efftab[e].olen += odonel + odoner;
- st_debug("post %s idone=%d, odone=%d", efftab[e].name, idone, odone);
+ st_debug("post %s idone=%d, odone=%d", efftab[e].name, idone, odone);
st_debug("post %s odone1=%d, olen1=%d odone=%d olen=%d", efftab[e].name, efftab[e - 1].odone, efftab[e - 1].olen, efftab[e].odone, efftab[e].olen);
-
+
done = idonel + idoner + odonel + odoner;
-
+
if (effstatusl)
effstatus = effstatusl;
else
@@ -1416,7 +1431,7 @@
/* Try to prime the pump with some data */
while (input_eff < neffects) {
int rc = drain_effect(input_eff);
-
+
if (efftab[input_eff].olen == 0) {
input_eff++;
/* Assuming next effect hasn't reached EOF yet. */
@@ -1481,15 +1496,15 @@
for (e = 1; e < neffects; e++) {
st_size_t clips;
- int (*stop)(eff_t effp) =
+ int (*stop)(eff_t effp) =
efftab[e].h->stop? efftab[e].h->stop : st_effect_nothing;
- int (*delete)(eff_t effp) =
+ int (*delete)(eff_t effp) =
efftab[e].h->delete? efftab[e].h->delete : st_effect_nothing;
stop(&efftab[e]);
clips = efftab[e].clips;
delete(&efftab[e]);
-
+
if (efftabR[e].name) {
stop(&efftabR[e]);
clips += efftab[e].clips;
@@ -1540,7 +1555,7 @@
left_time = in_time - read_time;
if (left_time < 0)
left_time = 0;
-
+
completed = ((double)read_samples / (double)input_samples) * 100;
if (completed < 0)
completed = 0;
@@ -1559,7 +1574,6 @@
fprintf(stderr, "\rTime: %02i:%05.2f [%02i:%05.2f] of %02i:%05.2f (% 5.1f%%) Output Buffer:% 7.2f%c", read_min, read_sec, left_min, left_sec, in_min, in_sec, completed, out_size, unit);
}
-/* Adjust volume based on value specified by the -v option for this file. */
static void volumechange(st_sample_t * buf, st_ssize_t len, file_info_t fi)
{
if (fi->volume != 1)