ref: 7e6e33818acc4623d440ed1d6371760cf84d6ad3
dir: /src/dither.c/
/* * libSoX dithering noise effect file. * * July 5, 1991 * Copyright 1991 Lance Norskog And Sundry Contributors * This source code is freely redistributable and may be used for * any purpose. This copyright notice must be maintained. * Lance Norskog And Sundry Contributors are not responsible for * the consequences of using this software. * * TODO: does triangular noise, could do local shaping * */ #include "sox_i.h" #include <stdlib.h> #include <math.h> typedef struct dither { double amount; } * dither_t; assert_static(sizeof(struct dither) <= SOX_MAX_EFFECT_PRIVSIZE, /* else */ dither_PRIVSIZE_too_big); static int getopts(sox_effect_t * effp, int n, char * * argv) { dither_t dither = (dither_t) effp->priv; if (n > 1) return sox_usage(effp); dither->amount = sqrt(2.); /* M_SQRT2 missing in some places */ /* Default to half a bit. */ if (n == 1) { double amount; char dummy; int scanned = sscanf(*argv, "%lf %c", &amount, &dummy); if (scanned == 1 && amount > 0) dither->amount *= amount; else return sox_usage(effp); } return SOX_SUCCESS; } static int start(sox_effect_t * effp) { dither_t dither = (dither_t) effp->priv; if (effp->out_signal.precision > 16) return SOX_EFF_NULL; /* Dithering not needed at >= 24 bits */ dither->amount *= 1 << (16 - effp->out_signal.precision); return SOX_SUCCESS; } static int flow(sox_effect_t * effp, const sox_sample_t * ibuf, sox_sample_t * obuf, sox_size_t * isamp, sox_size_t * osamp) { dither_t dither = (dither_t)effp->priv; sox_size_t len = min(*isamp, *osamp); *isamp = *osamp = len; while (len--) { /* 16 signed bits of triangular noise */ int tri16 = ((rand() % 32768) + (rand() % 32768)) - 32767; double l = *ibuf++ + tri16 * dither->amount; *obuf++ = SOX_ROUND_CLIP_COUNT(l, effp->clips); } return SOX_SUCCESS; } sox_effect_handler_t const * sox_dither_effect_fn(void) { static sox_effect_handler_t handler = { "dither", "[amount]", SOX_EFF_MCHAN | SOX_EFF_PREC, getopts, start, flow, 0, 0, 0 }; return &handler; }