cannam@86: /******************************************************************** cannam@86: * * cannam@86: * THIS FILE IS PART OF THE OggVorbis SOFTWARE CODEC SOURCE CODE. * cannam@86: * USE, DISTRIBUTION AND REPRODUCTION OF THIS LIBRARY SOURCE IS * cannam@86: * GOVERNED BY A BSD-STYLE SOURCE LICENSE INCLUDED WITH THIS SOURCE * cannam@86: * IN 'COPYING'. PLEASE READ THESE TERMS BEFORE DISTRIBUTING. * cannam@86: * * cannam@86: * THE OggVorbis SOURCE CODE IS (C) COPYRIGHT 1994-2009 * cannam@86: * by the Xiph.Org Foundation http://www.xiph.org/ * cannam@86: * * cannam@86: ******************************************************************** cannam@86: cannam@86: function: bitrate tracking and management cannam@86: last mod: $Id: bitrate.c 16227 2009-07-08 06:58:46Z xiphmont $ cannam@86: cannam@86: ********************************************************************/ cannam@86: cannam@86: #include cannam@86: #include cannam@86: #include cannam@86: #include cannam@86: #include "vorbis/codec.h" cannam@86: #include "codec_internal.h" cannam@86: #include "os.h" cannam@86: #include "misc.h" cannam@86: #include "bitrate.h" cannam@86: cannam@86: /* compute bitrate tracking setup */ cannam@86: void vorbis_bitrate_init(vorbis_info *vi,bitrate_manager_state *bm){ cannam@86: codec_setup_info *ci=vi->codec_setup; cannam@86: bitrate_manager_info *bi=&ci->bi; cannam@86: cannam@86: memset(bm,0,sizeof(*bm)); cannam@86: cannam@86: if(bi && (bi->reservoir_bits>0)){ cannam@86: long ratesamples=vi->rate; cannam@86: int halfsamples=ci->blocksizes[0]>>1; cannam@86: cannam@86: bm->short_per_long=ci->blocksizes[1]/ci->blocksizes[0]; cannam@86: bm->managed=1; cannam@86: cannam@86: bm->avg_bitsper= rint(1.*bi->avg_rate*halfsamples/ratesamples); cannam@86: bm->min_bitsper= rint(1.*bi->min_rate*halfsamples/ratesamples); cannam@86: bm->max_bitsper= rint(1.*bi->max_rate*halfsamples/ratesamples); cannam@86: cannam@86: bm->avgfloat=PACKETBLOBS/2; cannam@86: cannam@86: /* not a necessary fix, but one that leads to a more balanced cannam@86: typical initialization */ cannam@86: { cannam@86: long desired_fill=bi->reservoir_bits*bi->reservoir_bias; cannam@86: bm->minmax_reservoir=desired_fill; cannam@86: bm->avg_reservoir=desired_fill; cannam@86: } cannam@86: cannam@86: } cannam@86: } cannam@86: cannam@86: void vorbis_bitrate_clear(bitrate_manager_state *bm){ cannam@86: memset(bm,0,sizeof(*bm)); cannam@86: return; cannam@86: } cannam@86: cannam@86: int vorbis_bitrate_managed(vorbis_block *vb){ cannam@86: vorbis_dsp_state *vd=vb->vd; cannam@86: private_state *b=vd->backend_state; cannam@86: bitrate_manager_state *bm=&b->bms; cannam@86: cannam@86: if(bm && bm->managed)return(1); cannam@86: return(0); cannam@86: } cannam@86: cannam@86: /* finish taking in the block we just processed */ cannam@86: int vorbis_bitrate_addblock(vorbis_block *vb){ cannam@86: vorbis_block_internal *vbi=vb->internal; cannam@86: vorbis_dsp_state *vd=vb->vd; cannam@86: private_state *b=vd->backend_state; cannam@86: bitrate_manager_state *bm=&b->bms; cannam@86: vorbis_info *vi=vd->vi; cannam@86: codec_setup_info *ci=vi->codec_setup; cannam@86: bitrate_manager_info *bi=&ci->bi; cannam@86: cannam@86: int choice=rint(bm->avgfloat); cannam@86: long this_bits=oggpack_bytes(vbi->packetblob[choice])*8; cannam@86: long min_target_bits=(vb->W?bm->min_bitsper*bm->short_per_long:bm->min_bitsper); cannam@86: long max_target_bits=(vb->W?bm->max_bitsper*bm->short_per_long:bm->max_bitsper); cannam@86: int samples=ci->blocksizes[vb->W]>>1; cannam@86: long desired_fill=bi->reservoir_bits*bi->reservoir_bias; cannam@86: if(!bm->managed){ cannam@86: /* not a bitrate managed stream, but for API simplicity, we'll cannam@86: buffer the packet to keep the code path clean */ cannam@86: cannam@86: if(bm->vb)return(-1); /* one has been submitted without cannam@86: being claimed */ cannam@86: bm->vb=vb; cannam@86: return(0); cannam@86: } cannam@86: cannam@86: bm->vb=vb; cannam@86: cannam@86: /* look ahead for avg floater */ cannam@86: if(bm->avg_bitsper>0){ cannam@86: double slew=0.; cannam@86: long avg_target_bits=(vb->W?bm->avg_bitsper*bm->short_per_long:bm->avg_bitsper); cannam@86: double slewlimit= 15./bi->slew_damp; cannam@86: cannam@86: /* choosing a new floater: cannam@86: if we're over target, we slew down cannam@86: if we're under target, we slew up cannam@86: cannam@86: choose slew as follows: look through packetblobs of this frame cannam@86: and set slew as the first in the appropriate direction that cannam@86: gives us the slew we want. This may mean no slew if delta is cannam@86: already favorable. cannam@86: cannam@86: Then limit slew to slew max */ cannam@86: cannam@86: if(bm->avg_reservoir+(this_bits-avg_target_bits)>desired_fill){ cannam@86: while(choice>0 && this_bits>avg_target_bits && cannam@86: bm->avg_reservoir+(this_bits-avg_target_bits)>desired_fill){ cannam@86: choice--; cannam@86: this_bits=oggpack_bytes(vbi->packetblob[choice])*8; cannam@86: } cannam@86: }else if(bm->avg_reservoir+(this_bits-avg_target_bits)avg_reservoir+(this_bits-avg_target_bits)packetblob[choice])*8; cannam@86: } cannam@86: } cannam@86: cannam@86: slew=rint(choice-bm->avgfloat)/samples*vi->rate; cannam@86: if(slew<-slewlimit)slew=-slewlimit; cannam@86: if(slew>slewlimit)slew=slewlimit; cannam@86: choice=rint(bm->avgfloat+= slew/vi->rate*samples); cannam@86: this_bits=oggpack_bytes(vbi->packetblob[choice])*8; cannam@86: } cannam@86: cannam@86: cannam@86: cannam@86: /* enforce min(if used) on the current floater (if used) */ cannam@86: if(bm->min_bitsper>0){ cannam@86: /* do we need to force the bitrate up? */ cannam@86: if(this_bitsminmax_reservoir-(min_target_bits-this_bits)<0){ cannam@86: choice++; cannam@86: if(choice>=PACKETBLOBS)break; cannam@86: this_bits=oggpack_bytes(vbi->packetblob[choice])*8; cannam@86: } cannam@86: } cannam@86: } cannam@86: cannam@86: /* enforce max (if used) on the current floater (if used) */ cannam@86: if(bm->max_bitsper>0){ cannam@86: /* do we need to force the bitrate down? */ cannam@86: if(this_bits>max_target_bits){ cannam@86: while(bm->minmax_reservoir+(this_bits-max_target_bits)>bi->reservoir_bits){ cannam@86: choice--; cannam@86: if(choice<0)break; cannam@86: this_bits=oggpack_bytes(vbi->packetblob[choice])*8; cannam@86: } cannam@86: } cannam@86: } cannam@86: cannam@86: /* Choice of packetblobs now made based on floater, and min/max cannam@86: requirements. Now boundary check extreme choices */ cannam@86: cannam@86: if(choice<0){ cannam@86: /* choosing a smaller packetblob is insufficient to trim bitrate. cannam@86: frame will need to be truncated */ cannam@86: long maxsize=(max_target_bits+(bi->reservoir_bits-bm->minmax_reservoir))/8; cannam@86: bm->choice=choice=0; cannam@86: cannam@86: if(oggpack_bytes(vbi->packetblob[choice])>maxsize){ cannam@86: cannam@86: oggpack_writetrunc(vbi->packetblob[choice],maxsize*8); cannam@86: this_bits=oggpack_bytes(vbi->packetblob[choice])*8; cannam@86: } cannam@86: }else{ cannam@86: long minsize=(min_target_bits-bm->minmax_reservoir+7)/8; cannam@86: if(choice>=PACKETBLOBS) cannam@86: choice=PACKETBLOBS-1; cannam@86: cannam@86: bm->choice=choice; cannam@86: cannam@86: /* prop up bitrate according to demand. pad this frame out with zeroes */ cannam@86: minsize-=oggpack_bytes(vbi->packetblob[choice]); cannam@86: while(minsize-->0)oggpack_write(vbi->packetblob[choice],0,8); cannam@86: this_bits=oggpack_bytes(vbi->packetblob[choice])*8; cannam@86: cannam@86: } cannam@86: cannam@86: /* now we have the final packet and the final packet size. Update statistics */ cannam@86: /* min and max reservoir */ cannam@86: if(bm->min_bitsper>0 || bm->max_bitsper>0){ cannam@86: cannam@86: if(max_target_bits>0 && this_bits>max_target_bits){ cannam@86: bm->minmax_reservoir+=(this_bits-max_target_bits); cannam@86: }else if(min_target_bits>0 && this_bitsminmax_reservoir+=(this_bits-min_target_bits); cannam@86: }else{ cannam@86: /* inbetween; we want to take reservoir toward but not past desired_fill */ cannam@86: if(bm->minmax_reservoir>desired_fill){ cannam@86: if(max_target_bits>0){ /* logical bulletproofing against initialization state */ cannam@86: bm->minmax_reservoir+=(this_bits-max_target_bits); cannam@86: if(bm->minmax_reservoirminmax_reservoir=desired_fill; cannam@86: }else{ cannam@86: bm->minmax_reservoir=desired_fill; cannam@86: } cannam@86: }else{ cannam@86: if(min_target_bits>0){ /* logical bulletproofing against initialization state */ cannam@86: bm->minmax_reservoir+=(this_bits-min_target_bits); cannam@86: if(bm->minmax_reservoir>desired_fill)bm->minmax_reservoir=desired_fill; cannam@86: }else{ cannam@86: bm->minmax_reservoir=desired_fill; cannam@86: } cannam@86: } cannam@86: } cannam@86: } cannam@86: cannam@86: /* avg reservoir */ cannam@86: if(bm->avg_bitsper>0){ cannam@86: long avg_target_bits=(vb->W?bm->avg_bitsper*bm->short_per_long:bm->avg_bitsper); cannam@86: bm->avg_reservoir+=this_bits-avg_target_bits; cannam@86: } cannam@86: cannam@86: return(0); cannam@86: } cannam@86: cannam@86: int vorbis_bitrate_flushpacket(vorbis_dsp_state *vd,ogg_packet *op){ cannam@86: private_state *b=vd->backend_state; cannam@86: bitrate_manager_state *bm=&b->bms; cannam@86: vorbis_block *vb=bm->vb; cannam@86: int choice=PACKETBLOBS/2; cannam@86: if(!vb)return 0; cannam@86: cannam@86: if(op){ cannam@86: vorbis_block_internal *vbi=vb->internal; cannam@86: cannam@86: if(vorbis_bitrate_managed(vb)) cannam@86: choice=bm->choice; cannam@86: cannam@86: op->packet=oggpack_get_buffer(vbi->packetblob[choice]); cannam@86: op->bytes=oggpack_bytes(vbi->packetblob[choice]); cannam@86: op->b_o_s=0; cannam@86: op->e_o_s=vb->eofflag; cannam@86: op->granulepos=vb->granulepos; cannam@86: op->packetno=vb->sequence; /* for sake of completeness */ cannam@86: } cannam@86: cannam@86: bm->vb=0; cannam@86: return(1); cannam@86: }