ref: 02ebdd61638a3e3afc0c29c9c265166a0d69448f
dir: /src/crop.c/
/* libSoX effect: Crop ends of audio (c) 2009 robs@users.sourceforge.net * * This library 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 library 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 library; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ /* This is W.I.P. hence marked SOX_EFF_ALPHA for now. * Need to change size_t/INT32_MAX to uint64_t/INT64_MAX (for LFS) and do * proper length change tracking through the effects chain. */ #include "sox_i.h" #include <string.h> typedef struct { int argc; struct {int flag; char * str; uint64_t at;} pos[2]; } priv_t; static int parse(sox_effect_t * effp, char * * argv, sox_rate_t rate) { priv_t * p = (priv_t *)effp->priv; char const * s, * q; size_t samples; int i; for (i = p->argc - 1; i == 0 || i == 1; --i) { if (argv) /* 1st parse only */ p->pos[i].str = lsx_strdup(argv[i]); s = p->pos[i].str; if (strchr("+-" + 1 - i, *s)) p->pos[i].flag = *s++; q = lsx_parsesamples(rate, s, &samples, 't'); p->pos[i].at = samples; if (!q || *q) break; } return i >= 0 ? lsx_usage(effp) : SOX_SUCCESS; } static int create(sox_effect_t * effp, int argc, char * * argv) { priv_t * p = (priv_t *)effp->priv; --argc, ++argv; p->argc = argc; return parse(effp, argv, 1e5); /* No rate yet; parse with dummy */ } static int start(sox_effect_t * effp) { priv_t * p = (priv_t *)effp->priv; int i; p->pos[1].at = ((uint64_t)(-1)) / 2 / effp->in_signal.channels; parse(effp, 0, effp->in_signal.rate); /* Re-parse now rate is known */ for (i = 0; i < 2; ++i) { p->pos[i].at *= effp->in_signal.channels; if (p->pos[i].flag == '-') { if (effp->in_signal.length == SOX_UNSPEC) { lsx_fail("cannot crop from end: audio length is not known"); return SOX_EOF; } if (p->pos[i].at > effp->in_signal.length) { lsx_fail("cannot crop that much from end: audio is too short"); return SOX_EOF; } p->pos[i].at = effp->in_signal.length - p->pos[i].at; } } if (p->pos[1].flag != '+') { if (p->pos[0].at > p->pos[1].at) { lsx_fail("start position must be less than stop position"); return SOX_EOF; } if (!(p->pos[1].at -= p->pos[0].at)) p->pos[0].at = 0; } if (effp->in_signal.length) { if (!p->pos[0].at && p->pos[1].at == effp->in_signal.length) return SOX_EFF_NULL; if (p->pos[0].at > effp->in_signal.length || (p->argc > 1 && p->pos[0].at + p->pos[1].at > effp->in_signal.length)) { lsx_fail("audio is too short"); return SOX_EOF; } effp->out_signal.length = p->argc == 2 ? p->pos[1].at : effp->in_signal.length - p->pos[0].at; } return SOX_SUCCESS; } static int flow(sox_effect_t * effp, sox_sample_t const * ibuf, sox_sample_t * obuf, size_t * isamp, size_t * osamp) { priv_t * p = (priv_t *)effp->priv; size_t skipped; p->pos[0].at -= skipped = min(p->pos[0].at, *isamp); *osamp = !p->pos[0].at * min(p->pos[1].at, min(*isamp - skipped, *osamp)); memcpy(obuf, ibuf + skipped, *osamp * sizeof(*obuf)); *isamp = skipped + *osamp; return (p->pos[1].at -= *osamp) ? SOX_SUCCESS : SOX_EOF; } static int stop(sox_effect_t * effp) { priv_t * p = (priv_t *)effp->priv; if (p->pos[0].at || (p->pos[1].at && p->argc == 2)) lsx_warn("input audio was too short to crop as requested"); return SOX_SUCCESS; } static int lsx_kill(sox_effect_t * effp) { priv_t * p = (priv_t *)effp->priv; free(p->pos[0].str); free(p->pos[1].str); return SOX_SUCCESS; } sox_effect_handler_t const * lsx_crop_effect_fn(void) { static sox_effect_handler_t handler = { "crop", "[-]before [[+|-]from]\n" " n\tposition relative to start\n" " -n\tposition relative to end\n" " +n\tposition relative to previous", SOX_EFF_MCHAN | /* SOX_EFF_LENGTH | */ SOX_EFF_MODIFY | SOX_EFF_ALPHA, create, start, flow, NULL, stop, lsx_kill, sizeof(priv_t) }; return &handler; } uint64_t sox_crop_get_start(sox_effect_t * effp) { return ((priv_t *)effp->priv)->pos[0].at; } void sox_crop_clear_start(sox_effect_t * effp) { ((priv_t *)effp->priv)->pos[0].at = 0; } #if 0 /*---------------------- emulation of the `trim' effect ----------------------*/ static int trim_getopts(sox_effect_t * effp, int argc, char * * argv) { priv_t * p = (priv_t *)effp->priv; p->pos[1].flag = '+'; return lsx_crop_effect_fn()->getopts(effp, argc, argv); } sox_effect_handler_t const * lsx_trim_effect_fn(void) { static sox_effect_handler_t handler; handler = *lsx_crop_effect_fn(); handler.name = "trim"; handler.usage = "start [length]"; handler.getopts = trim_getopts; return &handler; } #endif