Implement OCSP stapling in Windows BoringSSL port.
[chromium-blink-merge.git] / third_party / speex / libspeex / ltp.c
blob0129c95f1598a3225f899d908b557de99e2111f9
1 /* Copyright (C) 2002-2006 Jean-Marc Valin
2 File: ltp.c
3 Long-Term Prediction functions
5 Redistribution and use in source and binary forms, with or without
6 modification, are permitted provided that the following conditions
7 are met:
9 - Redistributions of source code must retain the above copyright
10 notice, this list of conditions and the following disclaimer.
12 - Redistributions in binary form must reproduce the above copyright
13 notice, this list of conditions and the following disclaimer in the
14 documentation and/or other materials provided with the distribution.
16 - Neither the name of the Xiph.org Foundation nor the names of its
17 contributors may be used to endorse or promote products derived from
18 this software without specific prior written permission.
20 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21 ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22 LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
23 A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR
24 CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
25 EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
26 PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
27 PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
28 LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
29 NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
30 SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
33 #ifdef HAVE_CONFIG_H
34 #include "config.h"
35 #endif
37 #include <math.h>
38 #include "ltp.h"
39 #include "stack_alloc.h"
40 #include "filters.h"
41 #include <speex/speex_bits.h>
42 #include "math_approx.h"
43 #include "os_support.h"
45 #ifndef NULL
46 #define NULL 0
47 #endif
50 #ifdef _USE_SSE
51 #include "ltp_sse.h"
52 #elif defined (ARM4_ASM) || defined(ARM5E_ASM)
53 #include "ltp_arm4.h"
54 #elif defined (BFIN_ASM)
55 #include "ltp_bfin.h"
56 #endif
58 #ifndef OVERRIDE_INNER_PROD
59 spx_word32_t inner_prod(const spx_word16_t *x, const spx_word16_t *y, int len)
61 spx_word32_t sum=0;
62 len >>= 2;
63 while(len--)
65 spx_word32_t part=0;
66 part = MAC16_16(part,*x++,*y++);
67 part = MAC16_16(part,*x++,*y++);
68 part = MAC16_16(part,*x++,*y++);
69 part = MAC16_16(part,*x++,*y++);
70 /* HINT: If you had a 40-bit accumulator, you could shift only at the end */
71 sum = ADD32(sum,SHR32(part,6));
73 return sum;
75 #endif
77 #ifndef OVERRIDE_PITCH_XCORR
78 #if 0 /* HINT: Enable this for machines with enough registers (i.e. not x86) */
79 void pitch_xcorr(const spx_word16_t *_x, const spx_word16_t *_y, spx_word32_t *corr, int len, int nb_pitch, char *stack)
81 int i,j;
82 for (i=0;i<nb_pitch;i+=4)
84 /* Compute correlation*/
85 /*corr[nb_pitch-1-i]=inner_prod(x, _y+i, len);*/
86 spx_word32_t sum1=0;
87 spx_word32_t sum2=0;
88 spx_word32_t sum3=0;
89 spx_word32_t sum4=0;
90 const spx_word16_t *y = _y+i;
91 const spx_word16_t *x = _x;
92 spx_word16_t y0, y1, y2, y3;
93 /*y0=y[0];y1=y[1];y2=y[2];y3=y[3];*/
94 y0=*y++;
95 y1=*y++;
96 y2=*y++;
97 y3=*y++;
98 for (j=0;j<len;j+=4)
100 spx_word32_t part1;
101 spx_word32_t part2;
102 spx_word32_t part3;
103 spx_word32_t part4;
104 part1 = MULT16_16(*x,y0);
105 part2 = MULT16_16(*x,y1);
106 part3 = MULT16_16(*x,y2);
107 part4 = MULT16_16(*x,y3);
108 x++;
109 y0=*y++;
110 part1 = MAC16_16(part1,*x,y1);
111 part2 = MAC16_16(part2,*x,y2);
112 part3 = MAC16_16(part3,*x,y3);
113 part4 = MAC16_16(part4,*x,y0);
114 x++;
115 y1=*y++;
116 part1 = MAC16_16(part1,*x,y2);
117 part2 = MAC16_16(part2,*x,y3);
118 part3 = MAC16_16(part3,*x,y0);
119 part4 = MAC16_16(part4,*x,y1);
120 x++;
121 y2=*y++;
122 part1 = MAC16_16(part1,*x,y3);
123 part2 = MAC16_16(part2,*x,y0);
124 part3 = MAC16_16(part3,*x,y1);
125 part4 = MAC16_16(part4,*x,y2);
126 x++;
127 y3=*y++;
129 sum1 = ADD32(sum1,SHR32(part1,6));
130 sum2 = ADD32(sum2,SHR32(part2,6));
131 sum3 = ADD32(sum3,SHR32(part3,6));
132 sum4 = ADD32(sum4,SHR32(part4,6));
134 corr[nb_pitch-1-i]=sum1;
135 corr[nb_pitch-2-i]=sum2;
136 corr[nb_pitch-3-i]=sum3;
137 corr[nb_pitch-4-i]=sum4;
141 #else
142 void pitch_xcorr(const spx_word16_t *_x, const spx_word16_t *_y, spx_word32_t *corr, int len, int nb_pitch, char *stack)
144 int i;
145 for (i=0;i<nb_pitch;i++)
147 /* Compute correlation*/
148 corr[nb_pitch-1-i]=inner_prod(_x, _y+i, len);
152 #endif
153 #endif
155 #ifndef OVERRIDE_COMPUTE_PITCH_ERROR
156 static inline spx_word32_t compute_pitch_error(spx_word16_t *C, spx_word16_t *g, spx_word16_t pitch_control)
158 spx_word32_t sum = 0;
159 sum = ADD32(sum,MULT16_16(MULT16_16_16(g[0],pitch_control),C[0]));
160 sum = ADD32(sum,MULT16_16(MULT16_16_16(g[1],pitch_control),C[1]));
161 sum = ADD32(sum,MULT16_16(MULT16_16_16(g[2],pitch_control),C[2]));
162 sum = SUB32(sum,MULT16_16(MULT16_16_16(g[0],g[1]),C[3]));
163 sum = SUB32(sum,MULT16_16(MULT16_16_16(g[2],g[1]),C[4]));
164 sum = SUB32(sum,MULT16_16(MULT16_16_16(g[2],g[0]),C[5]));
165 sum = SUB32(sum,MULT16_16(MULT16_16_16(g[0],g[0]),C[6]));
166 sum = SUB32(sum,MULT16_16(MULT16_16_16(g[1],g[1]),C[7]));
167 sum = SUB32(sum,MULT16_16(MULT16_16_16(g[2],g[2]),C[8]));
168 return sum;
170 #endif
172 #ifndef OVERRIDE_OPEN_LOOP_NBEST_PITCH
173 void open_loop_nbest_pitch(spx_word16_t *sw, int start, int end, int len, int *pitch, spx_word16_t *gain, int N, char *stack)
175 int i,j,k;
176 VARDECL(spx_word32_t *best_score);
177 VARDECL(spx_word32_t *best_ener);
178 spx_word32_t e0;
179 VARDECL(spx_word32_t *corr);
180 #ifdef FIXED_POINT
181 /* In fixed-point, we need only one (temporary) array of 32-bit values and two (corr16, ener16)
182 arrays for (normalized) 16-bit values */
183 VARDECL(spx_word16_t *corr16);
184 VARDECL(spx_word16_t *ener16);
185 spx_word32_t *energy;
186 int cshift=0, eshift=0;
187 int scaledown = 0;
188 ALLOC(corr16, end-start+1, spx_word16_t);
189 ALLOC(ener16, end-start+1, spx_word16_t);
190 ALLOC(corr, end-start+1, spx_word32_t);
191 energy = corr;
192 #else
193 /* In floating-point, we need to float arrays and no normalized copies */
194 VARDECL(spx_word32_t *energy);
195 spx_word16_t *corr16;
196 spx_word16_t *ener16;
197 ALLOC(energy, end-start+2, spx_word32_t);
198 ALLOC(corr, end-start+1, spx_word32_t);
199 corr16 = corr;
200 ener16 = energy;
201 #endif
203 ALLOC(best_score, N, spx_word32_t);
204 ALLOC(best_ener, N, spx_word32_t);
205 for (i=0;i<N;i++)
207 best_score[i]=-1;
208 best_ener[i]=0;
209 pitch[i]=start;
212 #ifdef FIXED_POINT
213 for (i=-end;i<len;i++)
215 if (ABS16(sw[i])>16383)
217 scaledown=1;
218 break;
221 /* If the weighted input is close to saturation, then we scale it down */
222 if (scaledown)
224 for (i=-end;i<len;i++)
226 sw[i]=SHR16(sw[i],1);
229 #endif
230 energy[0]=inner_prod(sw-start, sw-start, len);
231 e0=inner_prod(sw, sw, len);
232 for (i=start;i<end;i++)
234 /* Update energy for next pitch*/
235 energy[i-start+1] = SUB32(ADD32(energy[i-start],SHR32(MULT16_16(sw[-i-1],sw[-i-1]),6)), SHR32(MULT16_16(sw[-i+len-1],sw[-i+len-1]),6));
236 if (energy[i-start+1] < 0)
237 energy[i-start+1] = 0;
240 #ifdef FIXED_POINT
241 eshift = normalize16(energy, ener16, 32766, end-start+1);
242 #endif
244 /* In fixed-point, this actually overrites the energy array (aliased to corr) */
245 pitch_xcorr(sw, sw-end, corr, len, end-start+1, stack);
247 #ifdef FIXED_POINT
248 /* Normalize to 180 so we can square it and it still fits in 16 bits */
249 cshift = normalize16(corr, corr16, 180, end-start+1);
250 /* If we scaled weighted input down, we need to scale it up again (OK, so we've just lost the LSB, who cares?) */
251 if (scaledown)
253 for (i=-end;i<len;i++)
255 sw[i]=SHL16(sw[i],1);
258 #endif
260 /* Search for the best pitch prediction gain */
261 for (i=start;i<=end;i++)
263 spx_word16_t tmp = MULT16_16_16(corr16[i-start],corr16[i-start]);
264 /* Instead of dividing the tmp by the energy, we multiply on the other side */
265 if (MULT16_16(tmp,best_ener[N-1])>MULT16_16(best_score[N-1],ADD16(1,ener16[i-start])))
267 /* We can safely put it last and then check */
268 best_score[N-1]=tmp;
269 best_ener[N-1]=ener16[i-start]+1;
270 pitch[N-1]=i;
271 /* Check if it comes in front of others */
272 for (j=0;j<N-1;j++)
274 if (MULT16_16(tmp,best_ener[j])>MULT16_16(best_score[j],ADD16(1,ener16[i-start])))
276 for (k=N-1;k>j;k--)
278 best_score[k]=best_score[k-1];
279 best_ener[k]=best_ener[k-1];
280 pitch[k]=pitch[k-1];
282 best_score[j]=tmp;
283 best_ener[j]=ener16[i-start]+1;
284 pitch[j]=i;
285 break;
291 /* Compute open-loop gain if necessary */
292 if (gain)
294 for (j=0;j<N;j++)
296 spx_word16_t g;
297 i=pitch[j];
298 g = DIV32(SHL32(EXTEND32(corr16[i-start]),cshift), 10+SHR32(MULT16_16(spx_sqrt(e0),spx_sqrt(SHL32(EXTEND32(ener16[i-start]),eshift))),6));
299 /* FIXME: g = max(g,corr/energy) */
300 if (g<0)
301 g = 0;
302 gain[j]=g;
308 #endif
310 #ifndef OVERRIDE_PITCH_GAIN_SEARCH_3TAP_VQ
311 static int pitch_gain_search_3tap_vq(
312 const signed char *gain_cdbk,
313 int gain_cdbk_size,
314 spx_word16_t *C16,
315 spx_word16_t max_gain
318 const signed char *ptr=gain_cdbk;
319 int best_cdbk=0;
320 spx_word32_t best_sum=-VERY_LARGE32;
321 spx_word32_t sum=0;
322 spx_word16_t g[3];
323 spx_word16_t pitch_control=64;
324 spx_word16_t gain_sum;
325 int i;
327 for (i=0;i<gain_cdbk_size;i++) {
329 ptr = gain_cdbk+4*i;
330 g[0]=ADD16((spx_word16_t)ptr[0],32);
331 g[1]=ADD16((spx_word16_t)ptr[1],32);
332 g[2]=ADD16((spx_word16_t)ptr[2],32);
333 gain_sum = (spx_word16_t)ptr[3];
335 sum = compute_pitch_error(C16, g, pitch_control);
337 if (sum>best_sum && gain_sum<=max_gain) {
338 best_sum=sum;
339 best_cdbk=i;
343 return best_cdbk;
345 #endif
347 /** Finds the best quantized 3-tap pitch predictor by analysis by synthesis */
348 static spx_word32_t pitch_gain_search_3tap(
349 const spx_word16_t target[], /* Target vector */
350 const spx_coef_t ak[], /* LPCs for this subframe */
351 const spx_coef_t awk1[], /* Weighted LPCs #1 for this subframe */
352 const spx_coef_t awk2[], /* Weighted LPCs #2 for this subframe */
353 spx_sig_t exc[], /* Excitation */
354 const signed char *gain_cdbk,
355 int gain_cdbk_size,
356 int pitch, /* Pitch value */
357 int p, /* Number of LPC coeffs */
358 int nsf, /* Number of samples in subframe */
359 SpeexBits *bits,
360 char *stack,
361 const spx_word16_t *exc2,
362 const spx_word16_t *r,
363 spx_word16_t *new_target,
364 int *cdbk_index,
365 int plc_tuning,
366 spx_word32_t cumul_gain,
367 int scaledown
370 int i,j;
371 VARDECL(spx_word16_t *tmp1);
372 VARDECL(spx_word16_t *e);
373 spx_word16_t *x[3];
374 spx_word32_t corr[3];
375 spx_word32_t A[3][3];
376 spx_word16_t gain[3];
377 spx_word32_t err;
378 spx_word16_t max_gain=128;
379 int best_cdbk=0;
381 ALLOC(tmp1, 3*nsf, spx_word16_t);
382 ALLOC(e, nsf, spx_word16_t);
384 if (cumul_gain > 262144)
385 max_gain = 31;
387 x[0]=tmp1;
388 x[1]=tmp1+nsf;
389 x[2]=tmp1+2*nsf;
391 for (j=0;j<nsf;j++)
392 new_target[j] = target[j];
395 VARDECL(spx_mem_t *mm);
396 int pp=pitch-1;
397 ALLOC(mm, p, spx_mem_t);
398 for (j=0;j<nsf;j++)
400 if (j-pp<0)
401 e[j]=exc2[j-pp];
402 else if (j-pp-pitch<0)
403 e[j]=exc2[j-pp-pitch];
404 else
405 e[j]=0;
407 #ifdef FIXED_POINT
408 /* Scale target and excitation down if needed (avoiding overflow) */
409 if (scaledown)
411 for (j=0;j<nsf;j++)
412 e[j] = SHR16(e[j],1);
413 for (j=0;j<nsf;j++)
414 new_target[j] = SHR16(new_target[j],1);
416 #endif
417 for (j=0;j<p;j++)
418 mm[j] = 0;
419 iir_mem16(e, ak, e, nsf, p, mm, stack);
420 for (j=0;j<p;j++)
421 mm[j] = 0;
422 filter_mem16(e, awk1, awk2, e, nsf, p, mm, stack);
423 for (j=0;j<nsf;j++)
424 x[2][j] = e[j];
426 for (i=1;i>=0;i--)
428 spx_word16_t e0=exc2[-pitch-1+i];
429 #ifdef FIXED_POINT
430 /* Scale excitation down if needed (avoiding overflow) */
431 if (scaledown)
432 e0 = SHR16(e0,1);
433 #endif
434 x[i][0]=MULT16_16_Q14(r[0], e0);
435 for (j=0;j<nsf-1;j++)
436 x[i][j+1]=ADD32(x[i+1][j],MULT16_16_P14(r[j+1], e0));
439 for (i=0;i<3;i++)
440 corr[i]=inner_prod(x[i],new_target,nsf);
441 for (i=0;i<3;i++)
442 for (j=0;j<=i;j++)
443 A[i][j]=A[j][i]=inner_prod(x[i],x[j],nsf);
446 spx_word32_t C[9];
447 #ifdef FIXED_POINT
448 spx_word16_t C16[9];
449 #else
450 spx_word16_t *C16=C;
451 #endif
452 C[0]=corr[2];
453 C[1]=corr[1];
454 C[2]=corr[0];
455 C[3]=A[1][2];
456 C[4]=A[0][1];
457 C[5]=A[0][2];
458 C[6]=A[2][2];
459 C[7]=A[1][1];
460 C[8]=A[0][0];
462 /*plc_tuning *= 2;*/
463 if (plc_tuning<2)
464 plc_tuning=2;
465 if (plc_tuning>30)
466 plc_tuning=30;
467 #ifdef FIXED_POINT
468 C[0] = SHL32(C[0],1);
469 C[1] = SHL32(C[1],1);
470 C[2] = SHL32(C[2],1);
471 C[3] = SHL32(C[3],1);
472 C[4] = SHL32(C[4],1);
473 C[5] = SHL32(C[5],1);
474 C[6] = MAC16_32_Q15(C[6],MULT16_16_16(plc_tuning,655),C[6]);
475 C[7] = MAC16_32_Q15(C[7],MULT16_16_16(plc_tuning,655),C[7]);
476 C[8] = MAC16_32_Q15(C[8],MULT16_16_16(plc_tuning,655),C[8]);
477 normalize16(C, C16, 32767, 9);
478 #else
479 C[6]*=.5*(1+.02*plc_tuning);
480 C[7]*=.5*(1+.02*plc_tuning);
481 C[8]*=.5*(1+.02*plc_tuning);
482 #endif
484 best_cdbk = pitch_gain_search_3tap_vq(gain_cdbk, gain_cdbk_size, C16, max_gain);
486 #ifdef FIXED_POINT
487 gain[0] = ADD16(32,(spx_word16_t)gain_cdbk[best_cdbk*4]);
488 gain[1] = ADD16(32,(spx_word16_t)gain_cdbk[best_cdbk*4+1]);
489 gain[2] = ADD16(32,(spx_word16_t)gain_cdbk[best_cdbk*4+2]);
490 /*printf ("%d %d %d %d\n",gain[0],gain[1],gain[2], best_cdbk);*/
491 #else
492 gain[0] = 0.015625*gain_cdbk[best_cdbk*4] + .5;
493 gain[1] = 0.015625*gain_cdbk[best_cdbk*4+1]+ .5;
494 gain[2] = 0.015625*gain_cdbk[best_cdbk*4+2]+ .5;
495 #endif
496 *cdbk_index=best_cdbk;
499 SPEEX_MEMSET(exc, 0, nsf);
500 for (i=0;i<3;i++)
502 int j;
503 int tmp1, tmp3;
504 int pp=pitch+1-i;
505 tmp1=nsf;
506 if (tmp1>pp)
507 tmp1=pp;
508 for (j=0;j<tmp1;j++)
509 exc[j]=MAC16_16(exc[j],SHL16(gain[2-i],7),exc2[j-pp]);
510 tmp3=nsf;
511 if (tmp3>pp+pitch)
512 tmp3=pp+pitch;
513 for (j=tmp1;j<tmp3;j++)
514 exc[j]=MAC16_16(exc[j],SHL16(gain[2-i],7),exc2[j-pp-pitch]);
516 for (i=0;i<nsf;i++)
518 spx_word32_t tmp = ADD32(ADD32(MULT16_16(gain[0],x[2][i]),MULT16_16(gain[1],x[1][i])),
519 MULT16_16(gain[2],x[0][i]));
520 new_target[i] = SUB16(new_target[i], EXTRACT16(PSHR32(tmp,6)));
522 err = inner_prod(new_target, new_target, nsf);
524 return err;
527 /** Finds the best quantized 3-tap pitch predictor by analysis by synthesis */
528 int pitch_search_3tap(
529 spx_word16_t target[], /* Target vector */
530 spx_word16_t *sw,
531 spx_coef_t ak[], /* LPCs for this subframe */
532 spx_coef_t awk1[], /* Weighted LPCs #1 for this subframe */
533 spx_coef_t awk2[], /* Weighted LPCs #2 for this subframe */
534 spx_sig_t exc[], /* Excitation */
535 const void *par,
536 int start, /* Smallest pitch value allowed */
537 int end, /* Largest pitch value allowed */
538 spx_word16_t pitch_coef, /* Voicing (pitch) coefficient */
539 int p, /* Number of LPC coeffs */
540 int nsf, /* Number of samples in subframe */
541 SpeexBits *bits,
542 char *stack,
543 spx_word16_t *exc2,
544 spx_word16_t *r,
545 int complexity,
546 int cdbk_offset,
547 int plc_tuning,
548 spx_word32_t *cumul_gain
551 int i;
552 int cdbk_index, pitch=0, best_gain_index=0;
553 VARDECL(spx_sig_t *best_exc);
554 VARDECL(spx_word16_t *new_target);
555 VARDECL(spx_word16_t *best_target);
556 int best_pitch=0;
557 spx_word32_t err, best_err=-1;
558 int N;
559 const ltp_params *params;
560 const signed char *gain_cdbk;
561 int gain_cdbk_size;
562 int scaledown=0;
564 VARDECL(int *nbest);
566 params = (const ltp_params*) par;
567 gain_cdbk_size = 1<<params->gain_bits;
568 gain_cdbk = params->gain_cdbk + 4*gain_cdbk_size*cdbk_offset;
570 N=complexity;
571 if (N>10)
572 N=10;
573 if (N<1)
574 N=1;
576 ALLOC(nbest, N, int);
577 params = (const ltp_params*) par;
579 if (end<start)
581 speex_bits_pack(bits, 0, params->pitch_bits);
582 speex_bits_pack(bits, 0, params->gain_bits);
583 SPEEX_MEMSET(exc, 0, nsf);
584 return start;
587 #ifdef FIXED_POINT
588 /* Check if we need to scale everything down in the pitch search to avoid overflows */
589 for (i=0;i<nsf;i++)
591 if (ABS16(target[i])>16383)
593 scaledown=1;
594 break;
597 for (i=-end;i<nsf;i++)
599 if (ABS16(exc2[i])>16383)
601 scaledown=1;
602 break;
605 #endif
606 if (N>end-start+1)
607 N=end-start+1;
608 if (end != start)
609 open_loop_nbest_pitch(sw, start, end, nsf, nbest, NULL, N, stack);
610 else
611 nbest[0] = start;
613 ALLOC(best_exc, nsf, spx_sig_t);
614 ALLOC(new_target, nsf, spx_word16_t);
615 ALLOC(best_target, nsf, spx_word16_t);
617 for (i=0;i<N;i++)
619 pitch=nbest[i];
620 SPEEX_MEMSET(exc, 0, nsf);
621 err=pitch_gain_search_3tap(target, ak, awk1, awk2, exc, gain_cdbk, gain_cdbk_size, pitch, p, nsf,
622 bits, stack, exc2, r, new_target, &cdbk_index, plc_tuning, *cumul_gain, scaledown);
623 if (err<best_err || best_err<0)
625 SPEEX_COPY(best_exc, exc, nsf);
626 SPEEX_COPY(best_target, new_target, nsf);
627 best_err=err;
628 best_pitch=pitch;
629 best_gain_index=cdbk_index;
632 /*printf ("pitch: %d %d\n", best_pitch, best_gain_index);*/
633 speex_bits_pack(bits, best_pitch-start, params->pitch_bits);
634 speex_bits_pack(bits, best_gain_index, params->gain_bits);
635 #ifdef FIXED_POINT
636 *cumul_gain = MULT16_32_Q13(SHL16(params->gain_cdbk[4*best_gain_index+3],8), MAX32(1024,*cumul_gain));
637 #else
638 *cumul_gain = 0.03125*MAX32(1024,*cumul_gain)*params->gain_cdbk[4*best_gain_index+3];
639 #endif
640 /*printf ("%f\n", cumul_gain);*/
641 /*printf ("encode pitch: %d %d\n", best_pitch, best_gain_index);*/
642 SPEEX_COPY(exc, best_exc, nsf);
643 SPEEX_COPY(target, best_target, nsf);
644 #ifdef FIXED_POINT
645 /* Scale target back up if needed */
646 if (scaledown)
648 for (i=0;i<nsf;i++)
649 target[i]=SHL16(target[i],1);
651 #endif
652 return pitch;
655 void pitch_unquant_3tap(
656 spx_word16_t exc[], /* Input excitation */
657 spx_word32_t exc_out[], /* Output excitation */
658 int start, /* Smallest pitch value allowed */
659 int end, /* Largest pitch value allowed */
660 spx_word16_t pitch_coef, /* Voicing (pitch) coefficient */
661 const void *par,
662 int nsf, /* Number of samples in subframe */
663 int *pitch_val,
664 spx_word16_t *gain_val,
665 SpeexBits *bits,
666 char *stack,
667 int count_lost,
668 int subframe_offset,
669 spx_word16_t last_pitch_gain,
670 int cdbk_offset
673 int i;
674 int pitch;
675 int gain_index;
676 spx_word16_t gain[3];
677 const signed char *gain_cdbk;
678 int gain_cdbk_size;
679 const ltp_params *params;
681 params = (const ltp_params*) par;
682 gain_cdbk_size = 1<<params->gain_bits;
683 gain_cdbk = params->gain_cdbk + 4*gain_cdbk_size*cdbk_offset;
685 pitch = speex_bits_unpack_unsigned(bits, params->pitch_bits);
686 pitch += start;
687 gain_index = speex_bits_unpack_unsigned(bits, params->gain_bits);
688 /*printf ("decode pitch: %d %d\n", pitch, gain_index);*/
689 #ifdef FIXED_POINT
690 gain[0] = ADD16(32,(spx_word16_t)gain_cdbk[gain_index*4]);
691 gain[1] = ADD16(32,(spx_word16_t)gain_cdbk[gain_index*4+1]);
692 gain[2] = ADD16(32,(spx_word16_t)gain_cdbk[gain_index*4+2]);
693 #else
694 gain[0] = 0.015625*gain_cdbk[gain_index*4]+.5;
695 gain[1] = 0.015625*gain_cdbk[gain_index*4+1]+.5;
696 gain[2] = 0.015625*gain_cdbk[gain_index*4+2]+.5;
697 #endif
699 if (count_lost && pitch > subframe_offset)
701 spx_word16_t gain_sum;
702 if (1) {
703 #ifdef FIXED_POINT
704 spx_word16_t tmp = count_lost < 4 ? last_pitch_gain : SHR16(last_pitch_gain,1);
705 if (tmp>62)
706 tmp=62;
707 #else
708 spx_word16_t tmp = count_lost < 4 ? last_pitch_gain : 0.5 * last_pitch_gain;
709 if (tmp>.95)
710 tmp=.95;
711 #endif
712 gain_sum = gain_3tap_to_1tap(gain);
714 if (gain_sum > tmp)
716 spx_word16_t fact = DIV32_16(SHL32(EXTEND32(tmp),14),gain_sum);
717 for (i=0;i<3;i++)
718 gain[i]=MULT16_16_Q14(fact,gain[i]);
725 *pitch_val = pitch;
726 gain_val[0]=gain[0];
727 gain_val[1]=gain[1];
728 gain_val[2]=gain[2];
729 gain[0] = SHL16(gain[0],7);
730 gain[1] = SHL16(gain[1],7);
731 gain[2] = SHL16(gain[2],7);
732 SPEEX_MEMSET(exc_out, 0, nsf);
733 for (i=0;i<3;i++)
735 int j;
736 int tmp1, tmp3;
737 int pp=pitch+1-i;
738 tmp1=nsf;
739 if (tmp1>pp)
740 tmp1=pp;
741 for (j=0;j<tmp1;j++)
742 exc_out[j]=MAC16_16(exc_out[j],gain[2-i],exc[j-pp]);
743 tmp3=nsf;
744 if (tmp3>pp+pitch)
745 tmp3=pp+pitch;
746 for (j=tmp1;j<tmp3;j++)
747 exc_out[j]=MAC16_16(exc_out[j],gain[2-i],exc[j-pp-pitch]);
749 /*for (i=0;i<nsf;i++)
750 exc[i]=PSHR32(exc32[i],13);*/
754 /** Forced pitch delay and gain */
755 int forced_pitch_quant(
756 spx_word16_t target[], /* Target vector */
757 spx_word16_t *sw,
758 spx_coef_t ak[], /* LPCs for this subframe */
759 spx_coef_t awk1[], /* Weighted LPCs #1 for this subframe */
760 spx_coef_t awk2[], /* Weighted LPCs #2 for this subframe */
761 spx_sig_t exc[], /* Excitation */
762 const void *par,
763 int start, /* Smallest pitch value allowed */
764 int end, /* Largest pitch value allowed */
765 spx_word16_t pitch_coef, /* Voicing (pitch) coefficient */
766 int p, /* Number of LPC coeffs */
767 int nsf, /* Number of samples in subframe */
768 SpeexBits *bits,
769 char *stack,
770 spx_word16_t *exc2,
771 spx_word16_t *r,
772 int complexity,
773 int cdbk_offset,
774 int plc_tuning,
775 spx_word32_t *cumul_gain
778 int i;
779 VARDECL(spx_word16_t *res);
780 ALLOC(res, nsf, spx_word16_t);
781 #ifdef FIXED_POINT
782 if (pitch_coef>63)
783 pitch_coef=63;
784 #else
785 if (pitch_coef>.99)
786 pitch_coef=.99;
787 #endif
788 for (i=0;i<nsf&&i<start;i++)
790 exc[i]=MULT16_16(SHL16(pitch_coef, 7),exc2[i-start]);
792 for (;i<nsf;i++)
794 exc[i]=MULT16_32_Q15(SHL16(pitch_coef, 9),exc[i-start]);
796 for (i=0;i<nsf;i++)
797 res[i] = EXTRACT16(PSHR32(exc[i], SIG_SHIFT-1));
798 syn_percep_zero16(res, ak, awk1, awk2, res, nsf, p, stack);
799 for (i=0;i<nsf;i++)
800 target[i]=EXTRACT16(SATURATE(SUB32(EXTEND32(target[i]),EXTEND32(res[i])),32700));
801 return start;
804 /** Unquantize forced pitch delay and gain */
805 void forced_pitch_unquant(
806 spx_word16_t exc[], /* Input excitation */
807 spx_word32_t exc_out[], /* Output excitation */
808 int start, /* Smallest pitch value allowed */
809 int end, /* Largest pitch value allowed */
810 spx_word16_t pitch_coef, /* Voicing (pitch) coefficient */
811 const void *par,
812 int nsf, /* Number of samples in subframe */
813 int *pitch_val,
814 spx_word16_t *gain_val,
815 SpeexBits *bits,
816 char *stack,
817 int count_lost,
818 int subframe_offset,
819 spx_word16_t last_pitch_gain,
820 int cdbk_offset
823 int i;
824 #ifdef FIXED_POINT
825 if (pitch_coef>63)
826 pitch_coef=63;
827 #else
828 if (pitch_coef>.99)
829 pitch_coef=.99;
830 #endif
831 for (i=0;i<nsf;i++)
833 exc_out[i]=MULT16_16(exc[i-start],SHL16(pitch_coef,7));
834 exc[i] = EXTRACT16(PSHR32(exc_out[i],13));
836 *pitch_val = start;
837 gain_val[0]=gain_val[2]=0;
838 gain_val[1] = pitch_coef;