annotate src/libvorbis-1.3.3/lib/bitrate.c @ 56:af97cad61ff0

Add updated build of PortAudio for OSX
author Chris Cannam <cannam@all-day-breakfast.com>
date Tue, 03 Jan 2017 15:10:52 +0000
parents 05aa0afa9217
children
rev   line source
Chris@1 1 /********************************************************************
Chris@1 2 * *
Chris@1 3 * THIS FILE IS PART OF THE OggVorbis SOFTWARE CODEC SOURCE CODE. *
Chris@1 4 * USE, DISTRIBUTION AND REPRODUCTION OF THIS LIBRARY SOURCE IS *
Chris@1 5 * GOVERNED BY A BSD-STYLE SOURCE LICENSE INCLUDED WITH THIS SOURCE *
Chris@1 6 * IN 'COPYING'. PLEASE READ THESE TERMS BEFORE DISTRIBUTING. *
Chris@1 7 * *
Chris@1 8 * THE OggVorbis SOURCE CODE IS (C) COPYRIGHT 1994-2009 *
Chris@1 9 * by the Xiph.Org Foundation http://www.xiph.org/ *
Chris@1 10 * *
Chris@1 11 ********************************************************************
Chris@1 12
Chris@1 13 function: bitrate tracking and management
Chris@1 14 last mod: $Id: bitrate.c 16227 2009-07-08 06:58:46Z xiphmont $
Chris@1 15
Chris@1 16 ********************************************************************/
Chris@1 17
Chris@1 18 #include <stdlib.h>
Chris@1 19 #include <string.h>
Chris@1 20 #include <math.h>
Chris@1 21 #include <ogg/ogg.h>
Chris@1 22 #include "vorbis/codec.h"
Chris@1 23 #include "codec_internal.h"
Chris@1 24 #include "os.h"
Chris@1 25 #include "misc.h"
Chris@1 26 #include "bitrate.h"
Chris@1 27
Chris@1 28 /* compute bitrate tracking setup */
Chris@1 29 void vorbis_bitrate_init(vorbis_info *vi,bitrate_manager_state *bm){
Chris@1 30 codec_setup_info *ci=vi->codec_setup;
Chris@1 31 bitrate_manager_info *bi=&ci->bi;
Chris@1 32
Chris@1 33 memset(bm,0,sizeof(*bm));
Chris@1 34
Chris@1 35 if(bi && (bi->reservoir_bits>0)){
Chris@1 36 long ratesamples=vi->rate;
Chris@1 37 int halfsamples=ci->blocksizes[0]>>1;
Chris@1 38
Chris@1 39 bm->short_per_long=ci->blocksizes[1]/ci->blocksizes[0];
Chris@1 40 bm->managed=1;
Chris@1 41
Chris@1 42 bm->avg_bitsper= rint(1.*bi->avg_rate*halfsamples/ratesamples);
Chris@1 43 bm->min_bitsper= rint(1.*bi->min_rate*halfsamples/ratesamples);
Chris@1 44 bm->max_bitsper= rint(1.*bi->max_rate*halfsamples/ratesamples);
Chris@1 45
Chris@1 46 bm->avgfloat=PACKETBLOBS/2;
Chris@1 47
Chris@1 48 /* not a necessary fix, but one that leads to a more balanced
Chris@1 49 typical initialization */
Chris@1 50 {
Chris@1 51 long desired_fill=bi->reservoir_bits*bi->reservoir_bias;
Chris@1 52 bm->minmax_reservoir=desired_fill;
Chris@1 53 bm->avg_reservoir=desired_fill;
Chris@1 54 }
Chris@1 55
Chris@1 56 }
Chris@1 57 }
Chris@1 58
Chris@1 59 void vorbis_bitrate_clear(bitrate_manager_state *bm){
Chris@1 60 memset(bm,0,sizeof(*bm));
Chris@1 61 return;
Chris@1 62 }
Chris@1 63
Chris@1 64 int vorbis_bitrate_managed(vorbis_block *vb){
Chris@1 65 vorbis_dsp_state *vd=vb->vd;
Chris@1 66 private_state *b=vd->backend_state;
Chris@1 67 bitrate_manager_state *bm=&b->bms;
Chris@1 68
Chris@1 69 if(bm && bm->managed)return(1);
Chris@1 70 return(0);
Chris@1 71 }
Chris@1 72
Chris@1 73 /* finish taking in the block we just processed */
Chris@1 74 int vorbis_bitrate_addblock(vorbis_block *vb){
Chris@1 75 vorbis_block_internal *vbi=vb->internal;
Chris@1 76 vorbis_dsp_state *vd=vb->vd;
Chris@1 77 private_state *b=vd->backend_state;
Chris@1 78 bitrate_manager_state *bm=&b->bms;
Chris@1 79 vorbis_info *vi=vd->vi;
Chris@1 80 codec_setup_info *ci=vi->codec_setup;
Chris@1 81 bitrate_manager_info *bi=&ci->bi;
Chris@1 82
Chris@1 83 int choice=rint(bm->avgfloat);
Chris@1 84 long this_bits=oggpack_bytes(vbi->packetblob[choice])*8;
Chris@1 85 long min_target_bits=(vb->W?bm->min_bitsper*bm->short_per_long:bm->min_bitsper);
Chris@1 86 long max_target_bits=(vb->W?bm->max_bitsper*bm->short_per_long:bm->max_bitsper);
Chris@1 87 int samples=ci->blocksizes[vb->W]>>1;
Chris@1 88 long desired_fill=bi->reservoir_bits*bi->reservoir_bias;
Chris@1 89 if(!bm->managed){
Chris@1 90 /* not a bitrate managed stream, but for API simplicity, we'll
Chris@1 91 buffer the packet to keep the code path clean */
Chris@1 92
Chris@1 93 if(bm->vb)return(-1); /* one has been submitted without
Chris@1 94 being claimed */
Chris@1 95 bm->vb=vb;
Chris@1 96 return(0);
Chris@1 97 }
Chris@1 98
Chris@1 99 bm->vb=vb;
Chris@1 100
Chris@1 101 /* look ahead for avg floater */
Chris@1 102 if(bm->avg_bitsper>0){
Chris@1 103 double slew=0.;
Chris@1 104 long avg_target_bits=(vb->W?bm->avg_bitsper*bm->short_per_long:bm->avg_bitsper);
Chris@1 105 double slewlimit= 15./bi->slew_damp;
Chris@1 106
Chris@1 107 /* choosing a new floater:
Chris@1 108 if we're over target, we slew down
Chris@1 109 if we're under target, we slew up
Chris@1 110
Chris@1 111 choose slew as follows: look through packetblobs of this frame
Chris@1 112 and set slew as the first in the appropriate direction that
Chris@1 113 gives us the slew we want. This may mean no slew if delta is
Chris@1 114 already favorable.
Chris@1 115
Chris@1 116 Then limit slew to slew max */
Chris@1 117
Chris@1 118 if(bm->avg_reservoir+(this_bits-avg_target_bits)>desired_fill){
Chris@1 119 while(choice>0 && this_bits>avg_target_bits &&
Chris@1 120 bm->avg_reservoir+(this_bits-avg_target_bits)>desired_fill){
Chris@1 121 choice--;
Chris@1 122 this_bits=oggpack_bytes(vbi->packetblob[choice])*8;
Chris@1 123 }
Chris@1 124 }else if(bm->avg_reservoir+(this_bits-avg_target_bits)<desired_fill){
Chris@1 125 while(choice+1<PACKETBLOBS && this_bits<avg_target_bits &&
Chris@1 126 bm->avg_reservoir+(this_bits-avg_target_bits)<desired_fill){
Chris@1 127 choice++;
Chris@1 128 this_bits=oggpack_bytes(vbi->packetblob[choice])*8;
Chris@1 129 }
Chris@1 130 }
Chris@1 131
Chris@1 132 slew=rint(choice-bm->avgfloat)/samples*vi->rate;
Chris@1 133 if(slew<-slewlimit)slew=-slewlimit;
Chris@1 134 if(slew>slewlimit)slew=slewlimit;
Chris@1 135 choice=rint(bm->avgfloat+= slew/vi->rate*samples);
Chris@1 136 this_bits=oggpack_bytes(vbi->packetblob[choice])*8;
Chris@1 137 }
Chris@1 138
Chris@1 139
Chris@1 140
Chris@1 141 /* enforce min(if used) on the current floater (if used) */
Chris@1 142 if(bm->min_bitsper>0){
Chris@1 143 /* do we need to force the bitrate up? */
Chris@1 144 if(this_bits<min_target_bits){
Chris@1 145 while(bm->minmax_reservoir-(min_target_bits-this_bits)<0){
Chris@1 146 choice++;
Chris@1 147 if(choice>=PACKETBLOBS)break;
Chris@1 148 this_bits=oggpack_bytes(vbi->packetblob[choice])*8;
Chris@1 149 }
Chris@1 150 }
Chris@1 151 }
Chris@1 152
Chris@1 153 /* enforce max (if used) on the current floater (if used) */
Chris@1 154 if(bm->max_bitsper>0){
Chris@1 155 /* do we need to force the bitrate down? */
Chris@1 156 if(this_bits>max_target_bits){
Chris@1 157 while(bm->minmax_reservoir+(this_bits-max_target_bits)>bi->reservoir_bits){
Chris@1 158 choice--;
Chris@1 159 if(choice<0)break;
Chris@1 160 this_bits=oggpack_bytes(vbi->packetblob[choice])*8;
Chris@1 161 }
Chris@1 162 }
Chris@1 163 }
Chris@1 164
Chris@1 165 /* Choice of packetblobs now made based on floater, and min/max
Chris@1 166 requirements. Now boundary check extreme choices */
Chris@1 167
Chris@1 168 if(choice<0){
Chris@1 169 /* choosing a smaller packetblob is insufficient to trim bitrate.
Chris@1 170 frame will need to be truncated */
Chris@1 171 long maxsize=(max_target_bits+(bi->reservoir_bits-bm->minmax_reservoir))/8;
Chris@1 172 bm->choice=choice=0;
Chris@1 173
Chris@1 174 if(oggpack_bytes(vbi->packetblob[choice])>maxsize){
Chris@1 175
Chris@1 176 oggpack_writetrunc(vbi->packetblob[choice],maxsize*8);
Chris@1 177 this_bits=oggpack_bytes(vbi->packetblob[choice])*8;
Chris@1 178 }
Chris@1 179 }else{
Chris@1 180 long minsize=(min_target_bits-bm->minmax_reservoir+7)/8;
Chris@1 181 if(choice>=PACKETBLOBS)
Chris@1 182 choice=PACKETBLOBS-1;
Chris@1 183
Chris@1 184 bm->choice=choice;
Chris@1 185
Chris@1 186 /* prop up bitrate according to demand. pad this frame out with zeroes */
Chris@1 187 minsize-=oggpack_bytes(vbi->packetblob[choice]);
Chris@1 188 while(minsize-->0)oggpack_write(vbi->packetblob[choice],0,8);
Chris@1 189 this_bits=oggpack_bytes(vbi->packetblob[choice])*8;
Chris@1 190
Chris@1 191 }
Chris@1 192
Chris@1 193 /* now we have the final packet and the final packet size. Update statistics */
Chris@1 194 /* min and max reservoir */
Chris@1 195 if(bm->min_bitsper>0 || bm->max_bitsper>0){
Chris@1 196
Chris@1 197 if(max_target_bits>0 && this_bits>max_target_bits){
Chris@1 198 bm->minmax_reservoir+=(this_bits-max_target_bits);
Chris@1 199 }else if(min_target_bits>0 && this_bits<min_target_bits){
Chris@1 200 bm->minmax_reservoir+=(this_bits-min_target_bits);
Chris@1 201 }else{
Chris@1 202 /* inbetween; we want to take reservoir toward but not past desired_fill */
Chris@1 203 if(bm->minmax_reservoir>desired_fill){
Chris@1 204 if(max_target_bits>0){ /* logical bulletproofing against initialization state */
Chris@1 205 bm->minmax_reservoir+=(this_bits-max_target_bits);
Chris@1 206 if(bm->minmax_reservoir<desired_fill)bm->minmax_reservoir=desired_fill;
Chris@1 207 }else{
Chris@1 208 bm->minmax_reservoir=desired_fill;
Chris@1 209 }
Chris@1 210 }else{
Chris@1 211 if(min_target_bits>0){ /* logical bulletproofing against initialization state */
Chris@1 212 bm->minmax_reservoir+=(this_bits-min_target_bits);
Chris@1 213 if(bm->minmax_reservoir>desired_fill)bm->minmax_reservoir=desired_fill;
Chris@1 214 }else{
Chris@1 215 bm->minmax_reservoir=desired_fill;
Chris@1 216 }
Chris@1 217 }
Chris@1 218 }
Chris@1 219 }
Chris@1 220
Chris@1 221 /* avg reservoir */
Chris@1 222 if(bm->avg_bitsper>0){
Chris@1 223 long avg_target_bits=(vb->W?bm->avg_bitsper*bm->short_per_long:bm->avg_bitsper);
Chris@1 224 bm->avg_reservoir+=this_bits-avg_target_bits;
Chris@1 225 }
Chris@1 226
Chris@1 227 return(0);
Chris@1 228 }
Chris@1 229
Chris@1 230 int vorbis_bitrate_flushpacket(vorbis_dsp_state *vd,ogg_packet *op){
Chris@1 231 private_state *b=vd->backend_state;
Chris@1 232 bitrate_manager_state *bm=&b->bms;
Chris@1 233 vorbis_block *vb=bm->vb;
Chris@1 234 int choice=PACKETBLOBS/2;
Chris@1 235 if(!vb)return 0;
Chris@1 236
Chris@1 237 if(op){
Chris@1 238 vorbis_block_internal *vbi=vb->internal;
Chris@1 239
Chris@1 240 if(vorbis_bitrate_managed(vb))
Chris@1 241 choice=bm->choice;
Chris@1 242
Chris@1 243 op->packet=oggpack_get_buffer(vbi->packetblob[choice]);
Chris@1 244 op->bytes=oggpack_bytes(vbi->packetblob[choice]);
Chris@1 245 op->b_o_s=0;
Chris@1 246 op->e_o_s=vb->eofflag;
Chris@1 247 op->granulepos=vb->granulepos;
Chris@1 248 op->packetno=vb->sequence; /* for sake of completeness */
Chris@1 249 }
Chris@1 250
Chris@1 251 bm->vb=0;
Chris@1 252 return(1);
Chris@1 253 }