yading@10
|
1 /*
|
yading@10
|
2 * Bytestream functions
|
yading@10
|
3 * copyright (c) 2006 Baptiste Coudurier <baptiste.coudurier@free.fr>
|
yading@10
|
4 * Copyright (c) 2012 Aneesh Dogra (lionaneesh) <lionaneesh@gmail.com>
|
yading@10
|
5 *
|
yading@10
|
6 * This file is part of FFmpeg.
|
yading@10
|
7 *
|
yading@10
|
8 * FFmpeg is free software; you can redistribute it and/or
|
yading@10
|
9 * modify it under the terms of the GNU Lesser General Public
|
yading@10
|
10 * License as published by the Free Software Foundation; either
|
yading@10
|
11 * version 2.1 of the License, or (at your option) any later version.
|
yading@10
|
12 *
|
yading@10
|
13 * FFmpeg is distributed in the hope that it will be useful,
|
yading@10
|
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
yading@10
|
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
yading@10
|
16 * Lesser General Public License for more details.
|
yading@10
|
17 *
|
yading@10
|
18 * You should have received a copy of the GNU Lesser General Public
|
yading@10
|
19 * License along with FFmpeg; if not, write to the Free Software
|
yading@10
|
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
yading@10
|
21 */
|
yading@10
|
22
|
yading@10
|
23 #ifndef AVCODEC_BYTESTREAM_H
|
yading@10
|
24 #define AVCODEC_BYTESTREAM_H
|
yading@10
|
25
|
yading@10
|
26 #include <stdint.h>
|
yading@10
|
27 #include <string.h>
|
yading@10
|
28
|
yading@10
|
29 #include "libavutil/common.h"
|
yading@10
|
30 #include "libavutil/intreadwrite.h"
|
yading@10
|
31
|
yading@10
|
32 typedef struct GetByteContext {
|
yading@10
|
33 const uint8_t *buffer, *buffer_end, *buffer_start;
|
yading@10
|
34 } GetByteContext;
|
yading@10
|
35
|
yading@10
|
36 typedef struct PutByteContext {
|
yading@10
|
37 uint8_t *buffer, *buffer_end, *buffer_start;
|
yading@10
|
38 int eof;
|
yading@10
|
39 } PutByteContext;
|
yading@10
|
40
|
yading@10
|
41 #define DEF(type, name, bytes, read, write) \
|
yading@10
|
42 static av_always_inline type bytestream_get_ ## name(const uint8_t **b) \
|
yading@10
|
43 { \
|
yading@10
|
44 (*b) += bytes; \
|
yading@10
|
45 return read(*b - bytes); \
|
yading@10
|
46 } \
|
yading@10
|
47 static av_always_inline void bytestream_put_ ## name(uint8_t **b, \
|
yading@10
|
48 const type value) \
|
yading@10
|
49 { \
|
yading@10
|
50 write(*b, value); \
|
yading@10
|
51 (*b) += bytes; \
|
yading@10
|
52 } \
|
yading@10
|
53 static av_always_inline void bytestream2_put_ ## name ## u(PutByteContext *p, \
|
yading@10
|
54 const type value) \
|
yading@10
|
55 { \
|
yading@10
|
56 bytestream_put_ ## name(&p->buffer, value); \
|
yading@10
|
57 } \
|
yading@10
|
58 static av_always_inline void bytestream2_put_ ## name(PutByteContext *p, \
|
yading@10
|
59 const type value) \
|
yading@10
|
60 { \
|
yading@10
|
61 if (!p->eof && (p->buffer_end - p->buffer >= bytes)) { \
|
yading@10
|
62 write(p->buffer, value); \
|
yading@10
|
63 p->buffer += bytes; \
|
yading@10
|
64 } else \
|
yading@10
|
65 p->eof = 1; \
|
yading@10
|
66 } \
|
yading@10
|
67 static av_always_inline type bytestream2_get_ ## name ## u(GetByteContext *g) \
|
yading@10
|
68 { \
|
yading@10
|
69 return bytestream_get_ ## name(&g->buffer); \
|
yading@10
|
70 } \
|
yading@10
|
71 static av_always_inline type bytestream2_get_ ## name(GetByteContext *g) \
|
yading@10
|
72 { \
|
yading@10
|
73 if (g->buffer_end - g->buffer < bytes) \
|
yading@10
|
74 return 0; \
|
yading@10
|
75 return bytestream2_get_ ## name ## u(g); \
|
yading@10
|
76 } \
|
yading@10
|
77 static av_always_inline type bytestream2_peek_ ## name(GetByteContext *g) \
|
yading@10
|
78 { \
|
yading@10
|
79 if (g->buffer_end - g->buffer < bytes) \
|
yading@10
|
80 return 0; \
|
yading@10
|
81 return read(g->buffer); \
|
yading@10
|
82 }
|
yading@10
|
83
|
yading@10
|
84 DEF(uint64_t, le64, 8, AV_RL64, AV_WL64)
|
yading@10
|
85 DEF(unsigned int, le32, 4, AV_RL32, AV_WL32)
|
yading@10
|
86 DEF(unsigned int, le24, 3, AV_RL24, AV_WL24)
|
yading@10
|
87 DEF(unsigned int, le16, 2, AV_RL16, AV_WL16)
|
yading@10
|
88 DEF(uint64_t, be64, 8, AV_RB64, AV_WB64)
|
yading@10
|
89 DEF(unsigned int, be32, 4, AV_RB32, AV_WB32)
|
yading@10
|
90 DEF(unsigned int, be24, 3, AV_RB24, AV_WB24)
|
yading@10
|
91 DEF(unsigned int, be16, 2, AV_RB16, AV_WB16)
|
yading@10
|
92 DEF(unsigned int, byte, 1, AV_RB8 , AV_WB8)
|
yading@10
|
93
|
yading@10
|
94 #if HAVE_BIGENDIAN
|
yading@10
|
95 # define bytestream2_get_ne16 bytestream2_get_be16
|
yading@10
|
96 # define bytestream2_get_ne24 bytestream2_get_be24
|
yading@10
|
97 # define bytestream2_get_ne32 bytestream2_get_be32
|
yading@10
|
98 # define bytestream2_get_ne64 bytestream2_get_be64
|
yading@10
|
99 # define bytestream2_get_ne16u bytestream2_get_be16u
|
yading@10
|
100 # define bytestream2_get_ne24u bytestream2_get_be24u
|
yading@10
|
101 # define bytestream2_get_ne32u bytestream2_get_be32u
|
yading@10
|
102 # define bytestream2_get_ne64u bytestream2_get_be64u
|
yading@10
|
103 # define bytestream2_put_ne16 bytestream2_put_be16
|
yading@10
|
104 # define bytestream2_put_ne24 bytestream2_put_be24
|
yading@10
|
105 # define bytestream2_put_ne32 bytestream2_put_be32
|
yading@10
|
106 # define bytestream2_put_ne64 bytestream2_put_be64
|
yading@10
|
107 # define bytestream2_peek_ne16 bytestream2_peek_be16
|
yading@10
|
108 # define bytestream2_peek_ne24 bytestream2_peek_be24
|
yading@10
|
109 # define bytestream2_peek_ne32 bytestream2_peek_be32
|
yading@10
|
110 # define bytestream2_peek_ne64 bytestream2_peek_be64
|
yading@10
|
111 #else
|
yading@10
|
112 # define bytestream2_get_ne16 bytestream2_get_le16
|
yading@10
|
113 # define bytestream2_get_ne24 bytestream2_get_le24
|
yading@10
|
114 # define bytestream2_get_ne32 bytestream2_get_le32
|
yading@10
|
115 # define bytestream2_get_ne64 bytestream2_get_le64
|
yading@10
|
116 # define bytestream2_get_ne16u bytestream2_get_le16u
|
yading@10
|
117 # define bytestream2_get_ne24u bytestream2_get_le24u
|
yading@10
|
118 # define bytestream2_get_ne32u bytestream2_get_le32u
|
yading@10
|
119 # define bytestream2_get_ne64u bytestream2_get_le64u
|
yading@10
|
120 # define bytestream2_put_ne16 bytestream2_put_le16
|
yading@10
|
121 # define bytestream2_put_ne24 bytestream2_put_le24
|
yading@10
|
122 # define bytestream2_put_ne32 bytestream2_put_le32
|
yading@10
|
123 # define bytestream2_put_ne64 bytestream2_put_le64
|
yading@10
|
124 # define bytestream2_peek_ne16 bytestream2_peek_le16
|
yading@10
|
125 # define bytestream2_peek_ne24 bytestream2_peek_le24
|
yading@10
|
126 # define bytestream2_peek_ne32 bytestream2_peek_le32
|
yading@10
|
127 # define bytestream2_peek_ne64 bytestream2_peek_le64
|
yading@10
|
128 #endif
|
yading@10
|
129
|
yading@10
|
130 static av_always_inline void bytestream2_init(GetByteContext *g,
|
yading@10
|
131 const uint8_t *buf,
|
yading@10
|
132 int buf_size)
|
yading@10
|
133 {
|
yading@10
|
134 g->buffer = buf;
|
yading@10
|
135 g->buffer_start = buf;
|
yading@10
|
136 g->buffer_end = buf + buf_size;
|
yading@10
|
137 }
|
yading@10
|
138
|
yading@10
|
139 static av_always_inline void bytestream2_init_writer(PutByteContext *p,
|
yading@10
|
140 uint8_t *buf,
|
yading@10
|
141 int buf_size)
|
yading@10
|
142 {
|
yading@10
|
143 p->buffer = buf;
|
yading@10
|
144 p->buffer_start = buf;
|
yading@10
|
145 p->buffer_end = buf + buf_size;
|
yading@10
|
146 p->eof = 0;
|
yading@10
|
147 }
|
yading@10
|
148
|
yading@10
|
149 static av_always_inline unsigned int bytestream2_get_bytes_left(GetByteContext *g)
|
yading@10
|
150 {
|
yading@10
|
151 return g->buffer_end - g->buffer;
|
yading@10
|
152 }
|
yading@10
|
153
|
yading@10
|
154 static av_always_inline unsigned int bytestream2_get_bytes_left_p(PutByteContext *p)
|
yading@10
|
155 {
|
yading@10
|
156 return p->buffer_end - p->buffer;
|
yading@10
|
157 }
|
yading@10
|
158
|
yading@10
|
159 static av_always_inline void bytestream2_skip(GetByteContext *g,
|
yading@10
|
160 unsigned int size)
|
yading@10
|
161 {
|
yading@10
|
162 g->buffer += FFMIN(g->buffer_end - g->buffer, size);
|
yading@10
|
163 }
|
yading@10
|
164
|
yading@10
|
165 static av_always_inline void bytestream2_skipu(GetByteContext *g,
|
yading@10
|
166 unsigned int size)
|
yading@10
|
167 {
|
yading@10
|
168 g->buffer += size;
|
yading@10
|
169 }
|
yading@10
|
170
|
yading@10
|
171 static av_always_inline void bytestream2_skip_p(PutByteContext *p,
|
yading@10
|
172 unsigned int size)
|
yading@10
|
173 {
|
yading@10
|
174 int size2;
|
yading@10
|
175 if (p->eof)
|
yading@10
|
176 return;
|
yading@10
|
177 size2 = FFMIN(p->buffer_end - p->buffer, size);
|
yading@10
|
178 if (size2 != size)
|
yading@10
|
179 p->eof = 1;
|
yading@10
|
180 p->buffer += size2;
|
yading@10
|
181 }
|
yading@10
|
182
|
yading@10
|
183 static av_always_inline int bytestream2_tell(GetByteContext *g)
|
yading@10
|
184 {
|
yading@10
|
185 return (int)(g->buffer - g->buffer_start);
|
yading@10
|
186 }
|
yading@10
|
187
|
yading@10
|
188 static av_always_inline int bytestream2_tell_p(PutByteContext *p)
|
yading@10
|
189 {
|
yading@10
|
190 return (int)(p->buffer - p->buffer_start);
|
yading@10
|
191 }
|
yading@10
|
192
|
yading@10
|
193 static av_always_inline int bytestream2_size(GetByteContext *g)
|
yading@10
|
194 {
|
yading@10
|
195 return (int)(g->buffer_end - g->buffer_start);
|
yading@10
|
196 }
|
yading@10
|
197
|
yading@10
|
198 static av_always_inline int bytestream2_size_p(PutByteContext *p)
|
yading@10
|
199 {
|
yading@10
|
200 return (int)(p->buffer_end - p->buffer_start);
|
yading@10
|
201 }
|
yading@10
|
202
|
yading@10
|
203 static av_always_inline int bytestream2_seek(GetByteContext *g,
|
yading@10
|
204 int offset,
|
yading@10
|
205 int whence)
|
yading@10
|
206 {
|
yading@10
|
207 switch (whence) {
|
yading@10
|
208 case SEEK_CUR:
|
yading@10
|
209 offset = av_clip(offset, -(g->buffer - g->buffer_start),
|
yading@10
|
210 g->buffer_end - g->buffer);
|
yading@10
|
211 g->buffer += offset;
|
yading@10
|
212 break;
|
yading@10
|
213 case SEEK_END:
|
yading@10
|
214 offset = av_clip(offset, -(g->buffer_end - g->buffer_start), 0);
|
yading@10
|
215 g->buffer = g->buffer_end + offset;
|
yading@10
|
216 break;
|
yading@10
|
217 case SEEK_SET:
|
yading@10
|
218 offset = av_clip(offset, 0, g->buffer_end - g->buffer_start);
|
yading@10
|
219 g->buffer = g->buffer_start + offset;
|
yading@10
|
220 break;
|
yading@10
|
221 default:
|
yading@10
|
222 return AVERROR(EINVAL);
|
yading@10
|
223 }
|
yading@10
|
224 return bytestream2_tell(g);
|
yading@10
|
225 }
|
yading@10
|
226
|
yading@10
|
227 static av_always_inline int bytestream2_seek_p(PutByteContext *p,
|
yading@10
|
228 int offset,
|
yading@10
|
229 int whence)
|
yading@10
|
230 {
|
yading@10
|
231 p->eof = 0;
|
yading@10
|
232 switch (whence) {
|
yading@10
|
233 case SEEK_CUR:
|
yading@10
|
234 if (p->buffer_end - p->buffer < offset)
|
yading@10
|
235 p->eof = 1;
|
yading@10
|
236 offset = av_clip(offset, -(p->buffer - p->buffer_start),
|
yading@10
|
237 p->buffer_end - p->buffer);
|
yading@10
|
238 p->buffer += offset;
|
yading@10
|
239 break;
|
yading@10
|
240 case SEEK_END:
|
yading@10
|
241 if (offset > 0)
|
yading@10
|
242 p->eof = 1;
|
yading@10
|
243 offset = av_clip(offset, -(p->buffer_end - p->buffer_start), 0);
|
yading@10
|
244 p->buffer = p->buffer_end + offset;
|
yading@10
|
245 break;
|
yading@10
|
246 case SEEK_SET:
|
yading@10
|
247 if (p->buffer_end - p->buffer_start < offset)
|
yading@10
|
248 p->eof = 1;
|
yading@10
|
249 offset = av_clip(offset, 0, p->buffer_end - p->buffer_start);
|
yading@10
|
250 p->buffer = p->buffer_start + offset;
|
yading@10
|
251 break;
|
yading@10
|
252 default:
|
yading@10
|
253 return AVERROR(EINVAL);
|
yading@10
|
254 }
|
yading@10
|
255 return bytestream2_tell_p(p);
|
yading@10
|
256 }
|
yading@10
|
257
|
yading@10
|
258 static av_always_inline unsigned int bytestream2_get_buffer(GetByteContext *g,
|
yading@10
|
259 uint8_t *dst,
|
yading@10
|
260 unsigned int size)
|
yading@10
|
261 {
|
yading@10
|
262 int size2 = FFMIN(g->buffer_end - g->buffer, size);
|
yading@10
|
263 memcpy(dst, g->buffer, size2);
|
yading@10
|
264 g->buffer += size2;
|
yading@10
|
265 return size2;
|
yading@10
|
266 }
|
yading@10
|
267
|
yading@10
|
268 static av_always_inline unsigned int bytestream2_get_bufferu(GetByteContext *g,
|
yading@10
|
269 uint8_t *dst,
|
yading@10
|
270 unsigned int size)
|
yading@10
|
271 {
|
yading@10
|
272 memcpy(dst, g->buffer, size);
|
yading@10
|
273 g->buffer += size;
|
yading@10
|
274 return size;
|
yading@10
|
275 }
|
yading@10
|
276
|
yading@10
|
277 static av_always_inline unsigned int bytestream2_put_buffer(PutByteContext *p,
|
yading@10
|
278 const uint8_t *src,
|
yading@10
|
279 unsigned int size)
|
yading@10
|
280 {
|
yading@10
|
281 int size2;
|
yading@10
|
282 if (p->eof)
|
yading@10
|
283 return 0;
|
yading@10
|
284 size2 = FFMIN(p->buffer_end - p->buffer, size);
|
yading@10
|
285 if (size2 != size)
|
yading@10
|
286 p->eof = 1;
|
yading@10
|
287 memcpy(p->buffer, src, size2);
|
yading@10
|
288 p->buffer += size2;
|
yading@10
|
289 return size2;
|
yading@10
|
290 }
|
yading@10
|
291
|
yading@10
|
292 static av_always_inline unsigned int bytestream2_put_bufferu(PutByteContext *p,
|
yading@10
|
293 const uint8_t *src,
|
yading@10
|
294 unsigned int size)
|
yading@10
|
295 {
|
yading@10
|
296 memcpy(p->buffer, src, size);
|
yading@10
|
297 p->buffer += size;
|
yading@10
|
298 return size;
|
yading@10
|
299 }
|
yading@10
|
300
|
yading@10
|
301 static av_always_inline void bytestream2_set_buffer(PutByteContext *p,
|
yading@10
|
302 const uint8_t c,
|
yading@10
|
303 unsigned int size)
|
yading@10
|
304 {
|
yading@10
|
305 int size2;
|
yading@10
|
306 if (p->eof)
|
yading@10
|
307 return;
|
yading@10
|
308 size2 = FFMIN(p->buffer_end - p->buffer, size);
|
yading@10
|
309 if (size2 != size)
|
yading@10
|
310 p->eof = 1;
|
yading@10
|
311 memset(p->buffer, c, size2);
|
yading@10
|
312 p->buffer += size2;
|
yading@10
|
313 }
|
yading@10
|
314
|
yading@10
|
315 static av_always_inline void bytestream2_set_bufferu(PutByteContext *p,
|
yading@10
|
316 const uint8_t c,
|
yading@10
|
317 unsigned int size)
|
yading@10
|
318 {
|
yading@10
|
319 memset(p->buffer, c, size);
|
yading@10
|
320 p->buffer += size;
|
yading@10
|
321 }
|
yading@10
|
322
|
yading@10
|
323 static av_always_inline unsigned int bytestream2_get_eof(PutByteContext *p)
|
yading@10
|
324 {
|
yading@10
|
325 return p->eof;
|
yading@10
|
326 }
|
yading@10
|
327
|
yading@10
|
328 static av_always_inline unsigned int bytestream_get_buffer(const uint8_t **b,
|
yading@10
|
329 uint8_t *dst,
|
yading@10
|
330 unsigned int size)
|
yading@10
|
331 {
|
yading@10
|
332 memcpy(dst, *b, size);
|
yading@10
|
333 (*b) += size;
|
yading@10
|
334 return size;
|
yading@10
|
335 }
|
yading@10
|
336
|
yading@10
|
337 static av_always_inline void bytestream_put_buffer(uint8_t **b,
|
yading@10
|
338 const uint8_t *src,
|
yading@10
|
339 unsigned int size)
|
yading@10
|
340 {
|
yading@10
|
341 memcpy(*b, src, size);
|
yading@10
|
342 (*b) += size;
|
yading@10
|
343 }
|
yading@10
|
344
|
yading@10
|
345 #endif /* AVCODEC_BYTESTREAM_H */
|