2020-09-30 16:53:55 +02:00

90 lines
3.0 KiB
C

#include <stdio.h>
#include <memory.h>
#define BUF 220 // input buffer size for test main
#define OUT2(o,i,t0,t1,t2,t3,t4,t5,t6,t7,t8,t9,t10,t11,t12,t13) \
t = ( (int)in[i]*t0 + (int)in[i+1]*t1 + \
(int)in[i+2]*t2 + (int)in[i+3]*t3 + \
(int)in[i+4]*t4 + (int)in[i+5]*t5 + \
(int)in[i+6]*t6 + (int)in[i+7]*t7 + \
(int)in[i+8]*t8 + (int)in[i+9]*t9 + \
(int)in[i+8]*t10 + (int)in[i+9]*t11 + \
(int)in[i+8]*t12 + (int)in[i+9]*t13 ) >> 10; \
if (t < -32768) out[o] = -32768; else if (t > 32767) out[o]=32767; else out[o] = t;
//--------------------------------------------------------
void segment11to8(short *in, short *out)
{
int t;
OUT2( 0, 0, 7, -20, 12, 42,-140, 238,
745, 238,-140, 42, 12, -20, 7, 0);
OUT2( 1, 1, 4, -4, -18, 62, -85, 0,
654, 510, -99, -27, 49, -26, 3, 1);
OUT2( 2, 2, 0, 6, -26, 38, 0,-126,
422, 704, 70,-110, 61, -10, -9, 5);
OUT2( 3, 4, 6, -15, 0, 55,-129, 150,
734, 330,-140, 24, 25, -24, 7, 0);
OUT2( 4, 5, 2, 0, -24, 58, -56, -57,
589, 589, -57, -56, 58, -24, 0, 2);
OUT2( 5, 7, 7, -24, 25, 24,-140, 330,
734, 150,-129, 55, 0, -15, 6, 0);
OUT2( 6, 8, 5, -9, -10, 61,-110, 70,
704, 422,-126, 0, 38, -26, 6, 0);
OUT2( 7, 9, 1, 3, -26, 49, -27, -99,
510, 654, 0, -85, 62, -18, -4, 4);
}
//--------------------------------------------------------
void convert11to8(short *in, short *out, short *prev, int len)
{
/*
Convert a buffer from 11KHz to 8KHz.
Note: len is number of shorts in input buffer, which MUST
be a multiple of 11 and at least 44.
How the overhang works: The filter kernel for 1 section of
11 input samples requires KERNEL (=25) samples of the input. So we use 14
samples of overhang from the previous frame, which means the
beginning of this frame looks like:
ppppppppppp ppp01234567 89abcdefghi 19.... 30.... / 41 42 43
X X x x
So we first have to do two special segments (the ones starting
at X) then we do the rest (the x's) in a loop. For the example
length=44 shown above, we'll do up & including 44-25=19, stopping on the
last x shown. Then we save 30-43 in the overhang buffer so that
30 is the first group done on the next frame.
*/
#define OVERHANG2 14
#define KERNEL2 25
int i,k;
short tmp[KERNEL2+11];
// Convert the first two segments, where segment= 11 samples of input
memcpy(tmp,prev,sizeof(short)*OVERHANG2);
memcpy(tmp+OVERHANG2,in,sizeof(short)*(KERNEL2+11-OVERHANG2));
segment11to8(tmp,out);
segment11to8(tmp+11,out+8);
// Loop through the remaining segments
k = 16;
for (i=22-OVERHANG2; i<=len-KERNEL2; i+=11)
{
segment11to8(in+i,out+k);
k += 8;
}
// Save OVERHANG2 samples for next time
memcpy(prev,in+len-OVERHANG2,sizeof(short)*OVERHANG2);
}