summaryrefslogtreecommitdiff
path: root/apps/dither.cc
blob: c85b296f81faa03037dd6b8f0dc696df3cb28241 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
// -------------------------------------------------------------------------
//
//    Copyright (C) 2010-2014 Fons Adriaensen <fons@linuxaudio.org>
//    
//    This program is free software; you can redistribute it and/or modify
//    it under the terms of the GNU General Public License as published by
//    the Free Software Foundation; either version 2 of the License, or
//    (at your option) any later version.
//
//    This program 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 General Public License for more details.
//
//    You should have received a copy of the GNU General Public License
//    along with this program; if not, write to the Free Software
//    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
//
// -------------------------------------------------------------------------


#include <string.h>
#include <math.h>
#include "dither.h"


float Dither::_div = 0;

#define SCALE 32768.0f
#define LIMIT 32767



Dither::Dither (void)
{
    reset ();
    _div = ldexpf (1.0f, 32);
}


void Dither::reset (void)
{
    memset (_err, 0, (SIZE + 4) * sizeof(float));
    _ind = SIZE - 1;
    _ran = 1234567;
}


void Dither::proc_rectangular (int nsam, const float *srce, int16_t *dest, int ds, int dd)
{
    float   v, r;
    int16_t k;

    while (nsam--)
    {
	r = genrand () - 0.5f;
        v = *srce * SCALE + r;
	k = lrintf (v);
	if      (k < -LIMIT) k = -LIMIT;
	else if (k >  LIMIT) k =  LIMIT;
        *dest = k;
        srce += ds;
        dest += dd;
    }
}


void Dither::proc_triangular (int nsam, const float *srce, int16_t *dest, int ds, int dd)
{
    float   v, r0, r1;
    int16_t k;

    r1 = *_err;
    while (nsam--)
    {
        r0 = genrand ();
        v = *srce * SCALE + r0 - r1;
	r1 = r0;
	k = lrintf (v);
	if      (k < -LIMIT) k = -LIMIT;
	else if (k >  LIMIT) k =  LIMIT;
        *dest = k;
        srce += ds;
        dest += dd;
    }
    *_err = r1;
}


void Dither::proc_lipschitz (int nsam, const float *srce, int16_t *dest, int ds, int dd)
{
    float   e, u, v, *p;
    int     i;
    int16_t k;

    i = _ind;
    while (nsam--)
    {
	p = _err + i;
        u = *srce * SCALE
	    - 2.033f * p [0]
	    + 2.165f * p [1]
	    - 1.959f * p [2]
	    + 1.590f * p [3]
	    - 0.615f * p [4];
	v = u + genrand () - genrand ();
	k = lrintf (v);
	e = k - u;
	if      (k < -LIMIT) k = -LIMIT;
	else if (k >  LIMIT) k =  LIMIT;
        *dest = k;
	if (--i < 0)
	{
	    _err [SIZE + 0] = _err [0];
	    _err [SIZE + 1] = _err [1];
	    _err [SIZE + 2] = _err [2];
	    _err [SIZE + 3] = _err [3];
	    i += SIZE;
	}
	_err [i] = e;
        srce += ds;
        dest += dd;
    }
    _ind = i;
}