annotate rangecoder.h @ 3990:746a60ba3177 libavcodec

enable CMOV_IS_FAST as its faster or equal speed on every cpu (duron, athlon, PM, P3) from which ive seen benchmarks, it might be slower on P4 but noone has posted benchmarks ...
author michael
date Wed, 11 Oct 2006 12:23:40 +0000
parents c8c591fe26f8
children 176ac8353f48
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
2334
25448d0bc924 range coder
michael
parents:
diff changeset
1 /*
25448d0bc924 range coder
michael
parents:
diff changeset
2 * Range coder
25448d0bc924 range coder
michael
parents:
diff changeset
3 * Copyright (c) 2004 Michael Niedermayer <michaelni@gmx.at>
25448d0bc924 range coder
michael
parents:
diff changeset
4 *
3947
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3036
diff changeset
5 * This file is part of FFmpeg.
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3036
diff changeset
6 *
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3036
diff changeset
7 * FFmpeg is free software; you can redistribute it and/or
2334
25448d0bc924 range coder
michael
parents:
diff changeset
8 * modify it under the terms of the GNU Lesser General Public
25448d0bc924 range coder
michael
parents:
diff changeset
9 * License as published by the Free Software Foundation; either
3947
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3036
diff changeset
10 * version 2.1 of the License, or (at your option) any later version.
2334
25448d0bc924 range coder
michael
parents:
diff changeset
11 *
3947
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3036
diff changeset
12 * FFmpeg is distributed in the hope that it will be useful,
2334
25448d0bc924 range coder
michael
parents:
diff changeset
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
25448d0bc924 range coder
michael
parents:
diff changeset
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
25448d0bc924 range coder
michael
parents:
diff changeset
15 * Lesser General Public License for more details.
25448d0bc924 range coder
michael
parents:
diff changeset
16 *
25448d0bc924 range coder
michael
parents:
diff changeset
17 * You should have received a copy of the GNU Lesser General Public
3947
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3036
diff changeset
18 * License along with FFmpeg; if not, write to the Free Software
3036
0b546eab515d Update licensing information: The FSF changed postal address.
diego
parents: 2967
diff changeset
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
2334
25448d0bc924 range coder
michael
parents:
diff changeset
20 *
25448d0bc924 range coder
michael
parents:
diff changeset
21 */
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2522
diff changeset
22
2334
25448d0bc924 range coder
michael
parents:
diff changeset
23 /**
25448d0bc924 range coder
michael
parents:
diff changeset
24 * @file rangecoder.h
25448d0bc924 range coder
michael
parents:
diff changeset
25 * Range coder.
25448d0bc924 range coder
michael
parents:
diff changeset
26 */
25448d0bc924 range coder
michael
parents:
diff changeset
27
25448d0bc924 range coder
michael
parents:
diff changeset
28 typedef struct RangeCoder{
25448d0bc924 range coder
michael
parents:
diff changeset
29 int low;
25448d0bc924 range coder
michael
parents:
diff changeset
30 int range;
25448d0bc924 range coder
michael
parents:
diff changeset
31 int outstanding_count;
25448d0bc924 range coder
michael
parents:
diff changeset
32 int outstanding_byte;
25448d0bc924 range coder
michael
parents:
diff changeset
33 uint8_t zero_state[256];
25448d0bc924 range coder
michael
parents:
diff changeset
34 uint8_t one_state[256];
25448d0bc924 range coder
michael
parents:
diff changeset
35 uint8_t *bytestream_start;
25448d0bc924 range coder
michael
parents:
diff changeset
36 uint8_t *bytestream;
25448d0bc924 range coder
michael
parents:
diff changeset
37 uint8_t *bytestream_end;
25448d0bc924 range coder
michael
parents:
diff changeset
38 }RangeCoder;
25448d0bc924 range coder
michael
parents:
diff changeset
39
25448d0bc924 range coder
michael
parents:
diff changeset
40 void ff_init_range_encoder(RangeCoder *c, uint8_t *buf, int buf_size);
25448d0bc924 range coder
michael
parents:
diff changeset
41 void ff_init_range_decoder(RangeCoder *c, const uint8_t *buf, int buf_size);
25448d0bc924 range coder
michael
parents:
diff changeset
42 int ff_rac_terminate(RangeCoder *c);
25448d0bc924 range coder
michael
parents:
diff changeset
43 void ff_build_rac_states(RangeCoder *c, int factor, int max_p);
25448d0bc924 range coder
michael
parents:
diff changeset
44
25448d0bc924 range coder
michael
parents:
diff changeset
45 static inline void renorm_encoder(RangeCoder *c){
25448d0bc924 range coder
michael
parents:
diff changeset
46 //FIXME optimize
25448d0bc924 range coder
michael
parents:
diff changeset
47 while(c->range < 0x100){
25448d0bc924 range coder
michael
parents:
diff changeset
48 if(c->outstanding_byte < 0){
25448d0bc924 range coder
michael
parents:
diff changeset
49 c->outstanding_byte= c->low>>8;
25448d0bc924 range coder
michael
parents:
diff changeset
50 }else if(c->low <= 0xFF00){
25448d0bc924 range coder
michael
parents:
diff changeset
51 *c->bytestream++ = c->outstanding_byte;
25448d0bc924 range coder
michael
parents:
diff changeset
52 for(;c->outstanding_count; c->outstanding_count--)
25448d0bc924 range coder
michael
parents:
diff changeset
53 *c->bytestream++ = 0xFF;
25448d0bc924 range coder
michael
parents:
diff changeset
54 c->outstanding_byte= c->low>>8;
25448d0bc924 range coder
michael
parents:
diff changeset
55 }else if(c->low >= 0x10000){
25448d0bc924 range coder
michael
parents:
diff changeset
56 *c->bytestream++ = c->outstanding_byte + 1;
25448d0bc924 range coder
michael
parents:
diff changeset
57 for(;c->outstanding_count; c->outstanding_count--)
25448d0bc924 range coder
michael
parents:
diff changeset
58 *c->bytestream++ = 0x00;
25448d0bc924 range coder
michael
parents:
diff changeset
59 c->outstanding_byte= (c->low>>8) & 0xFF;
25448d0bc924 range coder
michael
parents:
diff changeset
60 }else{
25448d0bc924 range coder
michael
parents:
diff changeset
61 c->outstanding_count++;
25448d0bc924 range coder
michael
parents:
diff changeset
62 }
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2522
diff changeset
63
2334
25448d0bc924 range coder
michael
parents:
diff changeset
64 c->low = (c->low & 0xFF)<<8;
25448d0bc924 range coder
michael
parents:
diff changeset
65 c->range <<= 8;
25448d0bc924 range coder
michael
parents:
diff changeset
66 }
25448d0bc924 range coder
michael
parents:
diff changeset
67 }
25448d0bc924 range coder
michael
parents:
diff changeset
68
25448d0bc924 range coder
michael
parents:
diff changeset
69 static inline void put_rac(RangeCoder *c, uint8_t * const state, int bit){
25448d0bc924 range coder
michael
parents:
diff changeset
70 int range1= (c->range * (*state)) >> 8;
25448d0bc924 range coder
michael
parents:
diff changeset
71
25448d0bc924 range coder
michael
parents:
diff changeset
72 assert(*state);
25448d0bc924 range coder
michael
parents:
diff changeset
73 assert(range1 < c->range);
25448d0bc924 range coder
michael
parents:
diff changeset
74 assert(range1 > 0);
25448d0bc924 range coder
michael
parents:
diff changeset
75 if(!bit){
25448d0bc924 range coder
michael
parents:
diff changeset
76 c->range -= range1;
25448d0bc924 range coder
michael
parents:
diff changeset
77 *state= c->zero_state[*state];
25448d0bc924 range coder
michael
parents:
diff changeset
78 }else{
25448d0bc924 range coder
michael
parents:
diff changeset
79 c->low += c->range - range1;
25448d0bc924 range coder
michael
parents:
diff changeset
80 c->range = range1;
25448d0bc924 range coder
michael
parents:
diff changeset
81 *state= c->one_state[*state];
25448d0bc924 range coder
michael
parents:
diff changeset
82 }
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2522
diff changeset
83
2334
25448d0bc924 range coder
michael
parents:
diff changeset
84 renorm_encoder(c);
25448d0bc924 range coder
michael
parents:
diff changeset
85 }
25448d0bc924 range coder
michael
parents:
diff changeset
86
25448d0bc924 range coder
michael
parents:
diff changeset
87 static inline void refill(RangeCoder *c){
25448d0bc924 range coder
michael
parents:
diff changeset
88 if(c->range < 0x100){
25448d0bc924 range coder
michael
parents:
diff changeset
89 c->range <<= 8;
25448d0bc924 range coder
michael
parents:
diff changeset
90 c->low <<= 8;
25448d0bc924 range coder
michael
parents:
diff changeset
91 if(c->bytestream < c->bytestream_end)
25448d0bc924 range coder
michael
parents:
diff changeset
92 c->low+= c->bytestream[0];
25448d0bc924 range coder
michael
parents:
diff changeset
93 c->bytestream++;
25448d0bc924 range coder
michael
parents:
diff changeset
94 }
25448d0bc924 range coder
michael
parents:
diff changeset
95 }
25448d0bc924 range coder
michael
parents:
diff changeset
96
25448d0bc924 range coder
michael
parents:
diff changeset
97 static inline int get_rac(RangeCoder *c, uint8_t * const state){
25448d0bc924 range coder
michael
parents:
diff changeset
98 int range1= (c->range * (*state)) >> 8;
2522
e25782262d7d kill warnings patch by (Mns Rullgrd <mru inprovide com>)
michael
parents: 2337
diff changeset
99 int attribute_unused one_mask;
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2522
diff changeset
100
2334
25448d0bc924 range coder
michael
parents:
diff changeset
101 c->range -= range1;
25448d0bc924 range coder
michael
parents:
diff changeset
102 #if 1
25448d0bc924 range coder
michael
parents:
diff changeset
103 if(c->low < c->range){
25448d0bc924 range coder
michael
parents:
diff changeset
104 *state= c->zero_state[*state];
25448d0bc924 range coder
michael
parents:
diff changeset
105 refill(c);
25448d0bc924 range coder
michael
parents:
diff changeset
106 return 0;
25448d0bc924 range coder
michael
parents:
diff changeset
107 }else{
25448d0bc924 range coder
michael
parents:
diff changeset
108 c->low -= c->range;
25448d0bc924 range coder
michael
parents:
diff changeset
109 *state= c->one_state[*state];
25448d0bc924 range coder
michael
parents:
diff changeset
110 c->range = range1;
25448d0bc924 range coder
michael
parents:
diff changeset
111 refill(c);
25448d0bc924 range coder
michael
parents:
diff changeset
112 return 1;
25448d0bc924 range coder
michael
parents:
diff changeset
113 }
25448d0bc924 range coder
michael
parents:
diff changeset
114 #else
25448d0bc924 range coder
michael
parents:
diff changeset
115 one_mask= (c->range - c->low-1)>>31;
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2522
diff changeset
116
2334
25448d0bc924 range coder
michael
parents:
diff changeset
117 c->low -= c->range & one_mask;
25448d0bc924 range coder
michael
parents:
diff changeset
118 c->range += (range1 - c->range) & one_mask;
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2522
diff changeset
119
2334
25448d0bc924 range coder
michael
parents:
diff changeset
120 *state= c->zero_state[(*state) + (256&one_mask)];
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2522
diff changeset
121
2334
25448d0bc924 range coder
michael
parents:
diff changeset
122 refill(c);
25448d0bc924 range coder
michael
parents:
diff changeset
123
25448d0bc924 range coder
michael
parents:
diff changeset
124 return one_mask&1;
25448d0bc924 range coder
michael
parents:
diff changeset
125 #endif
25448d0bc924 range coder
michael
parents:
diff changeset
126 }
25448d0bc924 range coder
michael
parents:
diff changeset
127