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