floor1.c
上传用户:wstnjxml
上传日期:2014-04-03
资源大小:7248k
文件大小:14k
源码类别:

Windows CE

开发平台:

C/C++

  1. /********************************************************************
  2.  *                                                                  *
  3.  * THIS FILE IS PART OF THE OggVorbis 'TREMOR' CODEC SOURCE CODE.   *
  4.  *                                                                  *
  5.  * USE, DISTRIBUTION AND REPRODUCTION OF THIS LIBRARY SOURCE IS     *
  6.  * GOVERNED BY A BSD-STYLE SOURCE LICENSE INCLUDED WITH THIS SOURCE *
  7.  * IN 'COPYING'. PLEASE READ THESE TERMS BEFORE DISTRIBUTING.       *
  8.  *                                                                  *
  9.  * THE OggVorbis 'TREMOR' SOURCE CODE IS (C) COPYRIGHT 1994-2002    *
  10.  * BY THE Xiph.Org FOUNDATION http://www.xiph.org/                  *
  11.  *                                                                  *
  12.  ********************************************************************
  13.  function: floor backend 1 implementation
  14.  ********************************************************************/
  15. #include <stdlib.h>
  16. #include <string.h>
  17. #include "ogg.h"
  18. #include "ivorbiscodec.h"
  19. #include "codec_internal.h"
  20. #include "registry.h"
  21. #include "codebook.h"
  22. #include "misc.h"
  23. #define floor1_rangedB 140 /* floor 1 fixed at -140dB to 0dB range */
  24. typedef struct {
  25.   int forward_index[VIF_POSIT+2];
  26.   
  27.   int hineighbor[VIF_POSIT];
  28.   int loneighbor[VIF_POSIT];
  29.   int posts;
  30.   int n;
  31.   int quant_q;
  32.   vorbis_info_floor1 *vi;
  33. } vorbis_look_floor1;
  34. /***********************************************/
  35.  
  36. static void floor1_free_info(vorbis_info_floor *i){
  37.   vorbis_info_floor1 *info=(vorbis_info_floor1 *)i;
  38.   if(info){
  39.     memset(info,0,sizeof(*info));
  40.     _ogg_free(info);
  41.   }
  42. }
  43. static void floor1_free_look(vorbis_look_floor *i){
  44.   vorbis_look_floor1 *look=(vorbis_look_floor1 *)i;
  45.   if(look){
  46.     memset(look,0,sizeof(*look));
  47.     _ogg_free(look);
  48.   }
  49. }
  50. static int ilog(unsigned int v){
  51.   int ret=0;
  52.   while(v){
  53.     ret++;
  54.     v>>=1;
  55.   }
  56.   return(ret);
  57. }
  58. static vorbis_info_floor *floor1_unpack (vorbis_info *vi,oggpack_buffer *opb){
  59.   codec_setup_info     *ci=(codec_setup_info *)vi->codec_setup;
  60.   int j,k,count=0,maxclass=-1,rangebits;
  61.   vorbis_info_floor1 *info=(vorbis_info_floor1 *)_ogg_calloc(1,sizeof(*info));
  62.   /* read partitions */
  63.   info->partitions=oggpack_read(opb,5); /* only 0 to 31 legal */
  64.   for(j=0;j<info->partitions;j++){
  65.     info->partitionclass[j]=oggpack_read(opb,4); /* only 0 to 15 legal */
  66.     if(maxclass<info->partitionclass[j])maxclass=info->partitionclass[j];
  67.   }
  68.   /* read partition classes */
  69.   for(j=0;j<maxclass+1;j++){
  70.     info->class_dim[j]=oggpack_read(opb,3)+1; /* 1 to 8 */
  71.     info->class_subs[j]=oggpack_read(opb,2); /* 0,1,2,3 bits */
  72.     if(info->class_subs[j]<0)
  73.       goto err_out;
  74.     if(info->class_subs[j])info->class_book[j]=oggpack_read(opb,8);
  75.     if(info->class_book[j]<0 || info->class_book[j]>=ci->books)
  76.       goto err_out;
  77.     for(k=0;k<(1<<info->class_subs[j]);k++){
  78.       info->class_subbook[j][k]=oggpack_read(opb,8)-1;
  79.       if(info->class_subbook[j][k]<-1 || info->class_subbook[j][k]>=ci->books)
  80. goto err_out;
  81.     }
  82.   }
  83.   /* read the post list */
  84.   info->mult=oggpack_read(opb,2)+1;     /* only 1,2,3,4 legal now */ 
  85.   rangebits=oggpack_read(opb,4);
  86.   for(j=0,k=0;j<info->partitions;j++){
  87.     count+=info->class_dim[info->partitionclass[j]]; 
  88.     for(;k<count;k++){
  89.       int t=info->postlist[k+2]=oggpack_read(opb,rangebits);
  90.       if(t<0 || t>=(1<<rangebits))
  91. goto err_out;
  92.     }
  93.   }
  94.   info->postlist[0]=0;
  95.   info->postlist[1]=1<<rangebits;
  96.   return(info);
  97.   
  98.  err_out:
  99.   floor1_free_info(info);
  100.   return(NULL);
  101. }
  102. static int __cdecl icomp(const void *a,const void *b){
  103.   return(**(int **)a-**(int **)b);
  104. }
  105. static vorbis_look_floor *floor1_look(vorbis_dsp_state *vd,vorbis_info_mode *mi,
  106.                               vorbis_info_floor *in){
  107.   int *sortpointer[VIF_POSIT+2];
  108.   vorbis_info_floor1 *info=(vorbis_info_floor1 *)in;
  109.   vorbis_look_floor1 *look=(vorbis_look_floor1 *)_ogg_calloc(1,sizeof(*look));
  110.   int i,j,n=0;
  111.   look->vi=info;
  112.   look->n=info->postlist[1];
  113.  
  114.   /* we drop each position value in-between already decoded values,
  115.      and use linear interpolation to predict each new value past the
  116.      edges.  The positions are read in the order of the position
  117.      list... we precompute the bounding positions in the lookup.  Of
  118.      course, the neighbors can change (if a position is declined), but
  119.      this is an initial mapping */
  120.   for(i=0;i<info->partitions;i++)n+=info->class_dim[info->partitionclass[i]];
  121.   n+=2;
  122.   look->posts=n;
  123.   /* also store a sorted position index */
  124.   for(i=0;i<n;i++)sortpointer[i]=info->postlist+i;
  125.   qsort(sortpointer,n,sizeof(*sortpointer),icomp);
  126.   /* points from sort order back to range number */
  127.   for(i=0;i<n;i++)look->forward_index[i]=sortpointer[i]-info->postlist;
  128.   
  129.   /* quantize values to multiplier spec */
  130.   switch(info->mult){
  131.   case 1: /* 1024 -> 256 */
  132.     look->quant_q=256;
  133.     break;
  134.   case 2: /* 1024 -> 128 */
  135.     look->quant_q=128;
  136.     break;
  137.   case 3: /* 1024 -> 86 */
  138.     look->quant_q=86;
  139.     break;
  140.   case 4: /* 1024 -> 64 */
  141.     look->quant_q=64;
  142.     break;
  143.   }
  144.   /* discover our neighbors for decode where we don't use fit flags
  145.      (that would push the neighbors outward) */
  146.   for(i=0;i<n-2;i++){
  147.     int lo=0;
  148.     int hi=1;
  149.     int lx=0;
  150.     int hx=look->n;
  151.     int currentx=info->postlist[i+2];
  152.     for(j=0;j<i+2;j++){
  153.       int x=info->postlist[j];
  154.       if(x>lx && x<currentx){
  155. lo=j;
  156. lx=x;
  157.       }
  158.       if(x<hx && x>currentx){
  159. hi=j;
  160. hx=x;
  161.       }
  162.     }
  163.     look->loneighbor[i]=lo;
  164.     look->hineighbor[i]=hi;
  165.   }
  166.   return(look);
  167. }
  168. static int render_point(int x0,int x1,int y0,int y1,int x){
  169.   y0&=0x7fff; /* mask off flag */
  170.   y1&=0x7fff;
  171.     
  172.   {
  173.     int dy=y1-y0;
  174.     int adx=x1-x0;
  175.     int ady=abs(dy);
  176.     int err=ady*(x-x0);
  177.     
  178.     int off=err/adx;
  179.     if(dy<0)return(y0-off);
  180.     return(y0+off);
  181.   }
  182. }
  183. #ifdef _LOW_ACCURACY_
  184. #  define XdB(n) ((((n)>>8)+1)>>1)
  185. #else
  186. #  define XdB(n) (n)
  187. #endif
  188. static const ogg_int32_t FLOOR_fromdB_LOOKUP[256]={
  189.   XdB(0x000000e5), XdB(0x000000f4), XdB(0x00000103), XdB(0x00000114),
  190.   XdB(0x00000126), XdB(0x00000139), XdB(0x0000014e), XdB(0x00000163),
  191.   XdB(0x0000017a), XdB(0x00000193), XdB(0x000001ad), XdB(0x000001c9),
  192.   XdB(0x000001e7), XdB(0x00000206), XdB(0x00000228), XdB(0x0000024c),
  193.   XdB(0x00000272), XdB(0x0000029b), XdB(0x000002c6), XdB(0x000002f4),
  194.   XdB(0x00000326), XdB(0x0000035a), XdB(0x00000392), XdB(0x000003cd),
  195.   XdB(0x0000040c), XdB(0x00000450), XdB(0x00000497), XdB(0x000004e4),
  196.   XdB(0x00000535), XdB(0x0000058c), XdB(0x000005e8), XdB(0x0000064a),
  197.   XdB(0x000006b3), XdB(0x00000722), XdB(0x00000799), XdB(0x00000818),
  198.   XdB(0x0000089e), XdB(0x0000092e), XdB(0x000009c6), XdB(0x00000a69),
  199.   XdB(0x00000b16), XdB(0x00000bcf), XdB(0x00000c93), XdB(0x00000d64),
  200.   XdB(0x00000e43), XdB(0x00000f30), XdB(0x0000102d), XdB(0x0000113a),
  201.   XdB(0x00001258), XdB(0x0000138a), XdB(0x000014cf), XdB(0x00001629),
  202.   XdB(0x0000179a), XdB(0x00001922), XdB(0x00001ac4), XdB(0x00001c82),
  203.   XdB(0x00001e5c), XdB(0x00002055), XdB(0x0000226f), XdB(0x000024ac),
  204.   XdB(0x0000270e), XdB(0x00002997), XdB(0x00002c4b), XdB(0x00002f2c),
  205.   XdB(0x0000323d), XdB(0x00003581), XdB(0x000038fb), XdB(0x00003caf),
  206.   XdB(0x000040a0), XdB(0x000044d3), XdB(0x0000494c), XdB(0x00004e10),
  207.   XdB(0x00005323), XdB(0x0000588a), XdB(0x00005e4b), XdB(0x0000646b),
  208.   XdB(0x00006af2), XdB(0x000071e5), XdB(0x0000794c), XdB(0x0000812e),
  209.   XdB(0x00008993), XdB(0x00009283), XdB(0x00009c09), XdB(0x0000a62d),
  210.   XdB(0x0000b0f9), XdB(0x0000bc79), XdB(0x0000c8b9), XdB(0x0000d5c4),
  211.   XdB(0x0000e3a9), XdB(0x0000f274), XdB(0x00010235), XdB(0x000112fd),
  212.   XdB(0x000124dc), XdB(0x000137e4), XdB(0x00014c29), XdB(0x000161bf),
  213.   XdB(0x000178bc), XdB(0x00019137), XdB(0x0001ab4a), XdB(0x0001c70e),
  214.   XdB(0x0001e4a1), XdB(0x0002041f), XdB(0x000225aa), XdB(0x00024962),
  215.   XdB(0x00026f6d), XdB(0x000297f0), XdB(0x0002c316), XdB(0x0002f109),
  216.   XdB(0x000321f9), XdB(0x00035616), XdB(0x00038d97), XdB(0x0003c8b4),
  217.   XdB(0x000407a7), XdB(0x00044ab2), XdB(0x00049218), XdB(0x0004de23),
  218.   XdB(0x00052f1e), XdB(0x0005855c), XdB(0x0005e135), XdB(0x00064306),
  219.   XdB(0x0006ab33), XdB(0x00071a24), XdB(0x0007904b), XdB(0x00080e20),
  220.   XdB(0x00089422), XdB(0x000922da), XdB(0x0009bad8), XdB(0x000a5cb6),
  221.   XdB(0x000b091a), XdB(0x000bc0b1), XdB(0x000c8436), XdB(0x000d5471),
  222.   XdB(0x000e3233), XdB(0x000f1e5f), XdB(0x001019e4), XdB(0x001125c1),
  223.   XdB(0x00124306), XdB(0x001372d5), XdB(0x0014b663), XdB(0x00160ef7),
  224.   XdB(0x00177df0), XdB(0x001904c1), XdB(0x001aa4f9), XdB(0x001c603d),
  225.   XdB(0x001e384f), XdB(0x00202f0f), XdB(0x0022467a), XdB(0x002480b1),
  226.   XdB(0x0026dff7), XdB(0x002966b3), XdB(0x002c1776), XdB(0x002ef4fc),
  227.   XdB(0x0032022d), XdB(0x00354222), XdB(0x0038b828), XdB(0x003c67c2),
  228.   XdB(0x004054ae), XdB(0x004482e8), XdB(0x0048f6af), XdB(0x004db488),
  229.   XdB(0x0052c142), XdB(0x005821ff), XdB(0x005ddc33), XdB(0x0063f5b0),
  230.   XdB(0x006a74a7), XdB(0x00715faf), XdB(0x0078bdce), XdB(0x0080967f),
  231.   XdB(0x0088f1ba), XdB(0x0091d7f9), XdB(0x009b5247), XdB(0x00a56a41),
  232.   XdB(0x00b02a27), XdB(0x00bb9ce2), XdB(0x00c7ce12), XdB(0x00d4ca17),
  233.   XdB(0x00e29e20), XdB(0x00f15835), XdB(0x0101074b), XdB(0x0111bb4e),
  234.   XdB(0x01238531), XdB(0x01367704), XdB(0x014aa402), XdB(0x016020a7),
  235.   XdB(0x017702c3), XdB(0x018f6190), XdB(0x01a955cb), XdB(0x01c4f9cf),
  236.   XdB(0x01e269a8), XdB(0x0201c33b), XdB(0x0223265a), XdB(0x0246b4ea),
  237.   XdB(0x026c9302), XdB(0x0294e716), XdB(0x02bfda13), XdB(0x02ed9793),
  238.   XdB(0x031e4e09), XdB(0x03522ee4), XdB(0x03896ed0), XdB(0x03c445e2),
  239.   XdB(0x0402efd6), XdB(0x0445ac4b), XdB(0x048cbefc), XdB(0x04d87013),
  240.   XdB(0x05290c67), XdB(0x057ee5ca), XdB(0x05da5364), XdB(0x063bb204),
  241.   XdB(0x06a36485), XdB(0x0711d42b), XdB(0x0787710e), XdB(0x0804b299),
  242.   XdB(0x088a17ef), XdB(0x0918287e), XdB(0x09af747c), XdB(0x0a50957e),
  243.   XdB(0x0afc2f19), XdB(0x0bb2ef7f), XdB(0x0c759034), XdB(0x0d44d6ca),
  244.   XdB(0x0e2195bc), XdB(0x0f0cad0d), XdB(0x10070b62), XdB(0x1111aeea),
  245.   XdB(0x122da66c), XdB(0x135c120f), XdB(0x149e24d9), XdB(0x15f525b1),
  246.   XdB(0x176270e3), XdB(0x18e7794b), XdB(0x1a85c9ae), XdB(0x1c3f06d1),
  247.   XdB(0x1e14f07d), XdB(0x200963d7), XdB(0x221e5ccd), XdB(0x2455f870),
  248.   XdB(0x26b2770b), XdB(0x29363e2b), XdB(0x2be3db5c), XdB(0x2ebe06b6),
  249.   XdB(0x31c7a55b), XdB(0x3503ccd4), XdB(0x3875c5aa), XdB(0x3c210f44),
  250.   XdB(0x4009632b), XdB(0x4432b8cf), XdB(0x48a149bc), XdB(0x4d59959e),
  251.   XdB(0x52606733), XdB(0x57bad899), XdB(0x5d6e593a), XdB(0x6380b298),
  252.   XdB(0x69f80e9a), XdB(0x70dafda8), XdB(0x78307d76), XdB(0x7fffffff),
  253. };
  254.   
  255. static void render_line(int x0,int x1,int y0,int y1,ogg_int32_t *d){
  256.   int dy=y1-y0;
  257.   int adx=x1-x0;
  258.   int ady=abs(dy);
  259.   int base=dy/adx;
  260.   int sy=(dy<0?base-1:base+1);
  261.   int x=x0;
  262.   int y=y0;
  263.   int err=0;
  264.   ady-=abs(base*adx);
  265.   d[x]= MULT31_SHIFT15(d[x],FLOOR_fromdB_LOOKUP[y]);
  266.   while(++x<x1){
  267.     err=err+ady;
  268.     if(err>=adx){
  269.       err-=adx;
  270.       y+=sy;
  271.     }else{
  272.       y+=base;
  273.     }
  274.     d[x]= MULT31_SHIFT15(d[x],FLOOR_fromdB_LOOKUP[y]);
  275.   }
  276. }
  277. static void *floor1_inverse1(vorbis_block *vb,vorbis_look_floor *in){
  278.   vorbis_look_floor1 *look=(vorbis_look_floor1 *)in;
  279.   vorbis_info_floor1 *info=look->vi;
  280.   codec_setup_info   *ci=(codec_setup_info *)vb->vd->vi->codec_setup;
  281.   
  282.   int i,j,k;
  283.   codebook *books=ci->fullbooks;   
  284.   
  285.   /* unpack wrapped/predicted values from stream */
  286.   if(oggpack_read(&vb->opb,1)==1){
  287.     int *fit_value=(int *)_vorbis_block_alloc(vb,(look->posts)*sizeof(*fit_value));
  288.     
  289.     fit_value[0]=oggpack_read(&vb->opb,ilog(look->quant_q-1));
  290.     fit_value[1]=oggpack_read(&vb->opb,ilog(look->quant_q-1));
  291.     
  292.     /* partition by partition */
  293.     /* partition by partition */
  294.     for(i=0,j=2;i<info->partitions;i++){
  295.       int classv=info->partitionclass[i];
  296.       int cdim=info->class_dim[classv];
  297.       int csubbits=info->class_subs[classv];
  298.       int csub=1<<csubbits;
  299.       int cval=0;
  300.       /* decode the partition's first stage cascade value */
  301.       if(csubbits){
  302. cval=vorbis_book_decode(books+info->class_book[classv],&vb->opb);
  303. if(cval==-1)goto eop;
  304.       }
  305.       for(k=0;k<cdim;k++){
  306. int book=info->class_subbook[classv][cval&(csub-1)];
  307. cval>>=csubbits;
  308. if(book>=0){
  309.   if((fit_value[j+k]=vorbis_book_decode(books+book,&vb->opb))==-1)
  310.     goto eop;
  311. }else{
  312.   fit_value[j+k]=0;
  313. }
  314.       }
  315.       j+=cdim;
  316.     }
  317.     /* unwrap positive values and reconsitute via linear interpolation */
  318.     for(i=2;i<look->posts;i++){
  319.       int predicted=render_point(info->postlist[look->loneighbor[i-2]],
  320.  info->postlist[look->hineighbor[i-2]],
  321.  fit_value[look->loneighbor[i-2]],
  322.  fit_value[look->hineighbor[i-2]],
  323.  info->postlist[i]);
  324.       int hiroom=look->quant_q-predicted;
  325.       int loroom=predicted;
  326.       int room=(hiroom<loroom?hiroom:loroom)<<1;
  327.       int val=fit_value[i];
  328.       if(val){
  329. if(val>=room){
  330.   if(hiroom>loroom){
  331.     val = val-loroom;
  332.   }else{
  333.   val = -1-(val-hiroom);
  334.   }
  335. }else{
  336.   if(val&1){
  337.     val= -((val+1)>>1);
  338.   }else{
  339.     val>>=1;
  340.   }
  341. }
  342. fit_value[i]=val+predicted;
  343. fit_value[look->loneighbor[i-2]]&=0x7fff;
  344. fit_value[look->hineighbor[i-2]]&=0x7fff;
  345.       }else{
  346. fit_value[i]=predicted|0x8000;
  347.       }
  348.     }
  349.     return(fit_value);
  350.   }
  351.  eop:
  352.   return(NULL);
  353. }
  354. static int floor1_inverse2(vorbis_block *vb,vorbis_look_floor *in,void *memo,
  355.   ogg_int32_t *out){
  356.   vorbis_look_floor1 *look=(vorbis_look_floor1 *)in;
  357.   vorbis_info_floor1 *info=look->vi;
  358.   codec_setup_info   *ci=(codec_setup_info *)vb->vd->vi->codec_setup;
  359.   int                  n=ci->blocksizes[vb->W]/2;
  360.   int j;
  361.   if(memo){
  362.     /* render the lines */
  363.     int *fit_value=(int *)memo;
  364.     int hx=0;
  365.     int lx=0;
  366.     int ly=fit_value[0]*info->mult;
  367.     for(j=1;j<look->posts;j++){
  368.       int current=look->forward_index[j];
  369.       int hy=fit_value[current]&0x7fff;
  370.       if(hy==fit_value[current]){
  371. hy*=info->mult;
  372. hx=info->postlist[current];
  373. render_line(lx,hx,ly,hy,out);
  374. lx=hx;
  375. ly=hy;
  376.       }
  377.     }
  378.     for(j=hx;j<n;j++)out[j]*=ly; /* be certain */    
  379.     return(1);
  380.   }
  381.   memset(out,0,sizeof(*out)*n);
  382.   return(0);
  383. }
  384. /* export hooks */
  385. const vorbis_func_floor floor1_exportbundle={
  386.   &floor1_unpack,&floor1_look,&floor1_free_info,
  387.   &floor1_free_look,&floor1_inverse1,&floor1_inverse2
  388. };