annotate src/libvorbis-1.3.3/lib/bitrate.c @ 86:98c1576536ae

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