1 /* GStreamer byte reader
2 *
3 * Copyright (C) 2008 Sebastian Dröge <sebastian.droege@collabora.co.uk>.
4 * Copyright (C) 2009,2014 Tim-Philipp Müller <tim centricular net>
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Library General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Library General Public License for more details.
15 *
16 * You should have received a copy of the GNU Library General Public
17 * License along with this library; if not, write to the
18 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
19 * Boston, MA 02110-1301, USA.
20 */
21
22 #ifdef HAVE_CONFIG_H
23 #include "config.h"
24 #endif
25
26 #define GST_BYTE_READER_DISABLE_INLINES
27 #include "gstbytereader.h"
28
29 #include <string.h>
30
31 /**
32 * SECTION:gstbytereader
33 * @title: GstByteReader
34 * @short_description: Reads different integer, string and floating point
35 * types from a memory buffer
36 *
37 * #GstByteReader provides a byte reader that can read different integer and
38 * floating point types from a memory buffer. It provides functions for reading
39 * signed/unsigned, little/big endian integers of 8, 16, 24, 32 and 64 bits
40 * and functions for reading little/big endian floating points numbers of
41 * 32 and 64 bits. It also provides functions to read NUL-terminated strings
42 * in various character encodings.
43 */
44
45 /**
46 * gst_byte_reader_new: (skip)
47 * @data: (in) (transfer none) (array length=size): data from which the
48 * #GstByteReader should read
49 * @size: Size of @data in bytes
50 *
51 * Create a new #GstByteReader instance, which will read from @data.
52 *
53 * Free-function: gst_byte_reader_free
54 *
55 * Returns: (transfer full): a new #GstByteReader instance
56 */
57 GstByteReader *
gst_byte_reader_new(const guint8 * data,guint size)58 gst_byte_reader_new (const guint8 * data, guint size)
59 {
60 GstByteReader *ret = g_slice_new0 (GstByteReader);
61
62 ret->data = data;
63 ret->size = size;
64
65 return ret;
66 }
67
68 /**
69 * gst_byte_reader_free:
70 * @reader: (in) (transfer full): a #GstByteReader instance
71 *
72 * Frees a #GstByteReader instance, which was previously allocated by
73 * gst_byte_reader_new().
74 */
75 void
gst_byte_reader_free(GstByteReader * reader)76 gst_byte_reader_free (GstByteReader * reader)
77 {
78 g_return_if_fail (reader != NULL);
79
80 g_slice_free (GstByteReader, reader);
81 }
82
83 /**
84 * gst_byte_reader_init:
85 * @reader: a #GstByteReader instance
86 * @data: (in) (transfer none) (array length=size): data from which
87 * the #GstByteReader should read
88 * @size: Size of @data in bytes
89 *
90 * Initializes a #GstByteReader instance to read from @data. This function
91 * can be called on already initialized instances.
92 */
93 void
gst_byte_reader_init(GstByteReader * reader,const guint8 * data,guint size)94 gst_byte_reader_init (GstByteReader * reader, const guint8 * data, guint size)
95 {
96 g_return_if_fail (reader != NULL);
97
98 reader->data = data;
99 reader->size = size;
100 reader->byte = 0;
101 }
102
103 /**
104 * gst_byte_reader_peek_sub_reader: (skip)
105 * @reader: an existing and initialized #GstByteReader instance
106 * @sub_reader: a #GstByteReader instance to initialize as sub-reader
107 * @size: size of @sub_reader in bytes
108 *
109 * Initializes a #GstByteReader sub-reader instance to contain @size bytes of
110 * data from the current position of @reader. This is useful to read chunked
111 * formats and make sure that one doesn't read beyond the size of the sub-chunk.
112 *
113 * Unlike gst_byte_reader_get_sub_reader(), this function does not modify the
114 * current position of @reader.
115 *
116 * Returns: FALSE on error or if @reader does not contain @size more bytes from
117 * the current position, and otherwise TRUE
118 *
119 * Since: 1.6
120 */
121 gboolean
gst_byte_reader_peek_sub_reader(GstByteReader * reader,GstByteReader * sub_reader,guint size)122 gst_byte_reader_peek_sub_reader (GstByteReader * reader,
123 GstByteReader * sub_reader, guint size)
124 {
125 return _gst_byte_reader_peek_sub_reader_inline (reader, sub_reader, size);
126 }
127
128 /**
129 * gst_byte_reader_get_sub_reader: (skip)
130 * @reader: an existing and initialized #GstByteReader instance
131 * @sub_reader: a #GstByteReader instance to initialize as sub-reader
132 * @size: size of @sub_reader in bytes
133 *
134 * Initializes a #GstByteReader sub-reader instance to contain @size bytes of
135 * data from the current position of @reader. This is useful to read chunked
136 * formats and make sure that one doesn't read beyond the size of the sub-chunk.
137 *
138 * Unlike gst_byte_reader_peek_sub_reader(), this function also modifies the
139 * position of @reader and moves it forward by @size bytes.
140 *
141 * Returns: FALSE on error or if @reader does not contain @size more bytes from
142 * the current position, and otherwise TRUE
143 *
144 * Since: 1.6
145 */
146 gboolean
gst_byte_reader_get_sub_reader(GstByteReader * reader,GstByteReader * sub_reader,guint size)147 gst_byte_reader_get_sub_reader (GstByteReader * reader,
148 GstByteReader * sub_reader, guint size)
149 {
150 return _gst_byte_reader_get_sub_reader_inline (reader, sub_reader, size);
151 }
152
153 /**
154 * gst_byte_reader_set_pos:
155 * @reader: a #GstByteReader instance
156 * @pos: The new position in bytes
157 *
158 * Sets the new position of a #GstByteReader instance to @pos in bytes.
159 *
160 * Returns: %TRUE if the position could be set successfully, %FALSE
161 * otherwise.
162 */
163 gboolean
gst_byte_reader_set_pos(GstByteReader * reader,guint pos)164 gst_byte_reader_set_pos (GstByteReader * reader, guint pos)
165 {
166 g_return_val_if_fail (reader != NULL, FALSE);
167
168 if (pos > reader->size)
169 return FALSE;
170
171 reader->byte = pos;
172
173 return TRUE;
174 }
175
176 /**
177 * gst_byte_reader_get_pos:
178 * @reader: a #GstByteReader instance
179 *
180 * Returns the current position of a #GstByteReader instance in bytes.
181 *
182 * Returns: The current position of @reader in bytes.
183 */
184 guint
gst_byte_reader_get_pos(const GstByteReader * reader)185 gst_byte_reader_get_pos (const GstByteReader * reader)
186 {
187 return _gst_byte_reader_get_pos_inline (reader);
188 }
189
190 /**
191 * gst_byte_reader_get_remaining:
192 * @reader: a #GstByteReader instance
193 *
194 * Returns the remaining number of bytes of a #GstByteReader instance.
195 *
196 * Returns: The remaining number of bytes of @reader instance.
197 */
198 guint
gst_byte_reader_get_remaining(const GstByteReader * reader)199 gst_byte_reader_get_remaining (const GstByteReader * reader)
200 {
201 return _gst_byte_reader_get_remaining_inline (reader);
202 }
203
204 /**
205 * gst_byte_reader_get_size:
206 * @reader: a #GstByteReader instance
207 *
208 * Returns the total number of bytes of a #GstByteReader instance.
209 *
210 * Returns: The total number of bytes of @reader instance.
211 */
212 guint
gst_byte_reader_get_size(const GstByteReader * reader)213 gst_byte_reader_get_size (const GstByteReader * reader)
214 {
215 return _gst_byte_reader_get_size_inline (reader);
216 }
217
218 #define gst_byte_reader_get_remaining _gst_byte_reader_get_remaining_inline
219 #define gst_byte_reader_get_size _gst_byte_reader_get_size_inline
220
221 /**
222 * gst_byte_reader_skip:
223 * @reader: a #GstByteReader instance
224 * @nbytes: the number of bytes to skip
225 *
226 * Skips @nbytes bytes of the #GstByteReader instance.
227 *
228 * Returns: %TRUE if @nbytes bytes could be skipped, %FALSE otherwise.
229 */
230 gboolean
gst_byte_reader_skip(GstByteReader * reader,guint nbytes)231 gst_byte_reader_skip (GstByteReader * reader, guint nbytes)
232 {
233 return _gst_byte_reader_skip_inline (reader, nbytes);
234 }
235
236 /**
237 * gst_byte_reader_get_uint8:
238 * @reader: a #GstByteReader instance
239 * @val: (out): Pointer to a #guint8 to store the result
240 *
241 * Read an unsigned 8 bit integer into @val and update the current position.
242 *
243 * Returns: %TRUE if successful, %FALSE otherwise.
244 */
245
246 /**
247 * gst_byte_reader_get_int8:
248 * @reader: a #GstByteReader instance
249 * @val: (out): Pointer to a #gint8 to store the result
250 *
251 * Read a signed 8 bit integer into @val and update the current position.
252 *
253 * Returns: %TRUE if successful, %FALSE otherwise.
254 */
255
256 /**
257 * gst_byte_reader_peek_uint8:
258 * @reader: a #GstByteReader instance
259 * @val: (out): Pointer to a #guint8 to store the result
260 *
261 * Read an unsigned 8 bit integer into @val but keep the current position.
262 *
263 * Returns: %TRUE if successful, %FALSE otherwise.
264 */
265
266 /**
267 * gst_byte_reader_peek_int8:
268 * @reader: a #GstByteReader instance
269 * @val: (out): Pointer to a #gint8 to store the result
270 *
271 * Read a signed 8 bit integer into @val but keep the current position.
272 *
273 * Returns: %TRUE if successful, %FALSE otherwise.
274 */
275
276 /**
277 * gst_byte_reader_get_uint16_le:
278 * @reader: a #GstByteReader instance
279 * @val: (out): Pointer to a #guint16 to store the result
280 *
281 * Read an unsigned 16 bit little endian integer into @val
282 * and update the current position.
283 *
284 * Returns: %TRUE if successful, %FALSE otherwise.
285 */
286
287 /**
288 * gst_byte_reader_get_int16_le:
289 * @reader: a #GstByteReader instance
290 * @val: (out): Pointer to a #gint16 to store the result
291 *
292 * Read a signed 16 bit little endian integer into @val
293 * and update the current position.
294 *
295 * Returns: %TRUE if successful, %FALSE otherwise.
296 */
297
298 /**
299 * gst_byte_reader_peek_uint16_le:
300 * @reader: a #GstByteReader instance
301 * @val: (out): Pointer to a #guint16 to store the result
302 *
303 * Read an unsigned 16 bit little endian integer into @val
304 * but keep the current position.
305 *
306 * Returns: %TRUE if successful, %FALSE otherwise.
307 */
308
309 /**
310 * gst_byte_reader_peek_int16_le:
311 * @reader: a #GstByteReader instance
312 * @val: (out): Pointer to a #gint16 to store the result
313 *
314 * Read a signed 16 bit little endian integer into @val
315 * but keep the current position.
316 *
317 * Returns: %TRUE if successful, %FALSE otherwise.
318 */
319
320 /**
321 * gst_byte_reader_get_uint16_be:
322 * @reader: a #GstByteReader instance
323 * @val: (out): Pointer to a #guint16 to store the result
324 *
325 * Read an unsigned 16 bit big endian integer into @val
326 * and update the current position.
327 *
328 * Returns: %TRUE if successful, %FALSE otherwise.
329 */
330
331 /**
332 * gst_byte_reader_get_int16_be:
333 * @reader: a #GstByteReader instance
334 * @val: (out): Pointer to a #gint16 to store the result
335 *
336 * Read a signed 16 bit big endian integer into @val
337 * and update the current position.
338 *
339 * Returns: %TRUE if successful, %FALSE otherwise.
340 */
341
342 /**
343 * gst_byte_reader_peek_uint16_be:
344 * @reader: a #GstByteReader instance
345 * @val: (out): Pointer to a #guint16 to store the result
346 *
347 * Read an unsigned 16 bit big endian integer into @val
348 * but keep the current position.
349 *
350 * Returns: %TRUE if successful, %FALSE otherwise.
351 */
352
353 /**
354 * gst_byte_reader_peek_int16_be:
355 * @reader: a #GstByteReader instance
356 * @val: (out): Pointer to a #gint16 to store the result
357 *
358 * Read a signed 16 bit big endian integer into @val
359 * but keep the current position.
360 *
361 * Returns: %TRUE if successful, %FALSE otherwise.
362 */
363
364 /**
365 * gst_byte_reader_get_uint24_le:
366 * @reader: a #GstByteReader instance
367 * @val: (out): Pointer to a #guint32 to store the result
368 *
369 * Read an unsigned 24 bit little endian integer into @val
370 * and update the current position.
371 *
372 * Returns: %TRUE if successful, %FALSE otherwise.
373 */
374
375 /**
376 * gst_byte_reader_get_int24_le:
377 * @reader: a #GstByteReader instance
378 * @val: (out): Pointer to a #gint32 to store the result
379 *
380 * Read a signed 24 bit little endian integer into @val
381 * and update the current position.
382 *
383 * Returns: %TRUE if successful, %FALSE otherwise.
384 */
385
386 /**
387 * gst_byte_reader_peek_uint24_le:
388 * @reader: a #GstByteReader instance
389 * @val: (out): Pointer to a #guint32 to store the result
390 *
391 * Read an unsigned 24 bit little endian integer into @val
392 * but keep the current position.
393 *
394 * Returns: %TRUE if successful, %FALSE otherwise.
395 */
396
397 /**
398 * gst_byte_reader_peek_int24_le:
399 * @reader: a #GstByteReader instance
400 * @val: (out): Pointer to a #gint32 to store the result
401 *
402 * Read a signed 24 bit little endian integer into @val
403 * but keep the current position.
404 *
405 * Returns: %TRUE if successful, %FALSE otherwise.
406 */
407
408 /**
409 * gst_byte_reader_get_uint24_be:
410 * @reader: a #GstByteReader instance
411 * @val: (out): Pointer to a #guint32 to store the result
412 *
413 * Read an unsigned 24 bit big endian integer into @val
414 * and update the current position.
415 *
416 * Returns: %TRUE if successful, %FALSE otherwise.
417 */
418
419 /**
420 * gst_byte_reader_get_int24_be:
421 * @reader: a #GstByteReader instance
422 * @val: (out): Pointer to a #gint32 to store the result
423 *
424 * Read a signed 24 bit big endian integer into @val
425 * and update the current position.
426 *
427 * Returns: %TRUE if successful, %FALSE otherwise.
428 */
429
430 /**
431 * gst_byte_reader_peek_uint24_be:
432 * @reader: a #GstByteReader instance
433 * @val: (out): Pointer to a #guint32 to store the result
434 *
435 * Read an unsigned 24 bit big endian integer into @val
436 * but keep the current position.
437 *
438 * Returns: %TRUE if successful, %FALSE otherwise.
439 */
440
441 /**
442 * gst_byte_reader_peek_int24_be:
443 * @reader: a #GstByteReader instance
444 * @val: (out): Pointer to a #gint32 to store the result
445 *
446 * Read a signed 24 bit big endian integer into @val
447 * but keep the current position.
448 *
449 * Returns: %TRUE if successful, %FALSE otherwise.
450 */
451
452
453 /**
454 * gst_byte_reader_get_uint32_le:
455 * @reader: a #GstByteReader instance
456 * @val: (out): Pointer to a #guint32 to store the result
457 *
458 * Read an unsigned 32 bit little endian integer into @val
459 * and update the current position.
460 *
461 * Returns: %TRUE if successful, %FALSE otherwise.
462 */
463
464 /**
465 * gst_byte_reader_get_int32_le:
466 * @reader: a #GstByteReader instance
467 * @val: (out): Pointer to a #gint32 to store the result
468 *
469 * Read a signed 32 bit little endian integer into @val
470 * and update the current position.
471 *
472 * Returns: %TRUE if successful, %FALSE otherwise.
473 */
474
475 /**
476 * gst_byte_reader_peek_uint32_le:
477 * @reader: a #GstByteReader instance
478 * @val: (out): Pointer to a #guint32 to store the result
479 *
480 * Read an unsigned 32 bit little endian integer into @val
481 * but keep the current position.
482 *
483 * Returns: %TRUE if successful, %FALSE otherwise.
484 */
485
486 /**
487 * gst_byte_reader_peek_int32_le:
488 * @reader: a #GstByteReader instance
489 * @val: (out): Pointer to a #gint32 to store the result
490 *
491 * Read a signed 32 bit little endian integer into @val
492 * but keep the current position.
493 *
494 * Returns: %TRUE if successful, %FALSE otherwise.
495 */
496
497 /**
498 * gst_byte_reader_get_uint32_be:
499 * @reader: a #GstByteReader instance
500 * @val: (out): Pointer to a #guint32 to store the result
501 *
502 * Read an unsigned 32 bit big endian integer into @val
503 * and update the current position.
504 *
505 * Returns: %TRUE if successful, %FALSE otherwise.
506 */
507
508 /**
509 * gst_byte_reader_get_int32_be:
510 * @reader: a #GstByteReader instance
511 * @val: (out): Pointer to a #gint32 to store the result
512 *
513 * Read a signed 32 bit big endian integer into @val
514 * and update the current position.
515 *
516 * Returns: %TRUE if successful, %FALSE otherwise.
517 */
518
519 /**
520 * gst_byte_reader_peek_uint32_be:
521 * @reader: a #GstByteReader instance
522 * @val: (out): Pointer to a #guint32 to store the result
523 *
524 * Read an unsigned 32 bit big endian integer into @val
525 * but keep the current position.
526 *
527 * Returns: %TRUE if successful, %FALSE otherwise.
528 */
529
530 /**
531 * gst_byte_reader_peek_int32_be:
532 * @reader: a #GstByteReader instance
533 * @val: (out): Pointer to a #gint32 to store the result
534 *
535 * Read a signed 32 bit big endian integer into @val
536 * but keep the current position.
537 *
538 * Returns: %TRUE if successful, %FALSE otherwise.
539 */
540
541 /**
542 * gst_byte_reader_get_uint64_le:
543 * @reader: a #GstByteReader instance
544 * @val: (out): Pointer to a #guint64 to store the result
545 *
546 * Read an unsigned 64 bit little endian integer into @val
547 * and update the current position.
548 *
549 * Returns: %TRUE if successful, %FALSE otherwise.
550 */
551
552 /**
553 * gst_byte_reader_get_int64_le:
554 * @reader: a #GstByteReader instance
555 * @val: (out): Pointer to a #gint64 to store the result
556 *
557 * Read a signed 64 bit little endian integer into @val
558 * and update the current position.
559 *
560 * Returns: %TRUE if successful, %FALSE otherwise.
561 */
562
563 /**
564 * gst_byte_reader_peek_uint64_le:
565 * @reader: a #GstByteReader instance
566 * @val: (out): Pointer to a #guint64 to store the result
567 *
568 * Read an unsigned 64 bit little endian integer into @val
569 * but keep the current position.
570 *
571 * Returns: %TRUE if successful, %FALSE otherwise.
572 */
573
574 /**
575 * gst_byte_reader_peek_int64_le:
576 * @reader: a #GstByteReader instance
577 * @val: (out): Pointer to a #gint64 to store the result
578 *
579 * Read a signed 64 bit little endian integer into @val
580 * but keep the current position.
581 *
582 * Returns: %TRUE if successful, %FALSE otherwise.
583 */
584
585 /**
586 * gst_byte_reader_get_uint64_be:
587 * @reader: a #GstByteReader instance
588 * @val: (out): Pointer to a #guint64 to store the result
589 *
590 * Read an unsigned 64 bit big endian integer into @val
591 * and update the current position.
592 *
593 * Returns: %TRUE if successful, %FALSE otherwise.
594 */
595
596 /**
597 * gst_byte_reader_get_int64_be:
598 * @reader: a #GstByteReader instance
599 * @val: (out): Pointer to a #gint64 to store the result
600 *
601 * Read a signed 64 bit big endian integer into @val
602 * and update the current position.
603 *
604 * Returns: %TRUE if successful, %FALSE otherwise.
605 */
606
607 /**
608 * gst_byte_reader_peek_uint64_be:
609 * @reader: a #GstByteReader instance
610 * @val: (out): Pointer to a #guint64 to store the result
611 *
612 * Read an unsigned 64 bit big endian integer into @val
613 * but keep the current position.
614 *
615 * Returns: %TRUE if successful, %FALSE otherwise.
616 */
617
618 /**
619 * gst_byte_reader_peek_int64_be:
620 * @reader: a #GstByteReader instance
621 * @val: (out): Pointer to a #gint64 to store the result
622 *
623 * Read a signed 64 bit big endian integer into @val
624 * but keep the current position.
625 *
626 * Returns: %TRUE if successful, %FALSE otherwise.
627 */
628
629 #define GST_BYTE_READER_PEEK_GET(bits,type,name) \
630 gboolean \
631 gst_byte_reader_get_##name (GstByteReader * reader, type * val) \
632 { \
633 return _gst_byte_reader_get_##name##_inline (reader, val); \
634 } \
635 \
636 gboolean \
637 gst_byte_reader_peek_##name (const GstByteReader * reader, type * val) \
638 { \
639 return _gst_byte_reader_peek_##name##_inline (reader, val); \
640 }
641
642 /* *INDENT-OFF* */
643
644 GST_BYTE_READER_PEEK_GET(8,guint8,uint8)
645 GST_BYTE_READER_PEEK_GET(8,gint8,int8)
646
647 GST_BYTE_READER_PEEK_GET(16,guint16,uint16_le)
648 GST_BYTE_READER_PEEK_GET(16,guint16,uint16_be)
649 GST_BYTE_READER_PEEK_GET(16,gint16,int16_le)
650 GST_BYTE_READER_PEEK_GET(16,gint16,int16_be)
651
652 GST_BYTE_READER_PEEK_GET(24,guint32,uint24_le)
653 GST_BYTE_READER_PEEK_GET(24,guint32,uint24_be)
654 GST_BYTE_READER_PEEK_GET(24,gint32,int24_le)
655 GST_BYTE_READER_PEEK_GET(24,gint32,int24_be)
656
657 GST_BYTE_READER_PEEK_GET(32,guint32,uint32_le)
658 GST_BYTE_READER_PEEK_GET(32,guint32,uint32_be)
659 GST_BYTE_READER_PEEK_GET(32,gint32,int32_le)
660 GST_BYTE_READER_PEEK_GET(32,gint32,int32_be)
661
662 GST_BYTE_READER_PEEK_GET(64,guint64,uint64_le)
663 GST_BYTE_READER_PEEK_GET(64,guint64,uint64_be)
664 GST_BYTE_READER_PEEK_GET(64,gint64,int64_le)
665 GST_BYTE_READER_PEEK_GET(64,gint64,int64_be)
666
667 /**
668 * gst_byte_reader_get_float32_le:
669 * @reader: a #GstByteReader instance
670 * @val: (out): Pointer to a #gfloat to store the result
671 *
672 * Read a 32 bit little endian floating point value into @val
673 * and update the current position.
674 *
675 * Returns: %TRUE if successful, %FALSE otherwise.
676 */
677
678 /**
679 * gst_byte_reader_peek_float32_le:
680 * @reader: a #GstByteReader instance
681 * @val: (out): Pointer to a #gfloat to store the result
682 *
683 * Read a 32 bit little endian floating point value into @val
684 * but keep the current position.
685 *
686 * Returns: %TRUE if successful, %FALSE otherwise.
687 */
688
689 /**
690 * gst_byte_reader_get_float32_be:
691 * @reader: a #GstByteReader instance
692 * @val: (out): Pointer to a #gfloat to store the result
693 *
694 * Read a 32 bit big endian floating point value into @val
695 * and update the current position.
696 *
697 * Returns: %TRUE if successful, %FALSE otherwise.
698 */
699
700 /**
701 * gst_byte_reader_peek_float32_be:
702 * @reader: a #GstByteReader instance
703 * @val: (out): Pointer to a #gfloat to store the result
704 *
705 * Read a 32 bit big endian floating point value into @val
706 * but keep the current position.
707 *
708 * Returns: %TRUE if successful, %FALSE otherwise.
709 */
710
711 /**
712 * gst_byte_reader_get_float64_le:
713 * @reader: a #GstByteReader instance
714 * @val: (out): Pointer to a #gdouble to store the result
715 *
716 * Read a 64 bit little endian floating point value into @val
717 * and update the current position.
718 *
719 * Returns: %TRUE if successful, %FALSE otherwise.
720 */
721
722 /**
723 * gst_byte_reader_peek_float64_le:
724 * @reader: a #GstByteReader instance
725 * @val: (out): Pointer to a #gdouble to store the result
726 *
727 * Read a 64 bit little endian floating point value into @val
728 * but keep the current position.
729 *
730 * Returns: %TRUE if successful, %FALSE otherwise.
731 */
732
733 /**
734 * gst_byte_reader_get_float64_be:
735 * @reader: a #GstByteReader instance
736 * @val: (out): Pointer to a #gdouble to store the result
737 *
738 * Read a 64 bit big endian floating point value into @val
739 * and update the current position.
740 *
741 * Returns: %TRUE if successful, %FALSE otherwise.
742 */
743
744 /**
745 * gst_byte_reader_peek_float64_be:
746 * @reader: a #GstByteReader instance
747 * @val: (out): Pointer to a #gdouble to store the result
748 *
749 * Read a 64 bit big endian floating point value into @val
750 * but keep the current position.
751 *
752 * Returns: %TRUE if successful, %FALSE otherwise.
753 */
754
755 GST_BYTE_READER_PEEK_GET(32,gfloat,float32_le)
756 GST_BYTE_READER_PEEK_GET(32,gfloat,float32_be)
757 GST_BYTE_READER_PEEK_GET(64,gdouble,float64_le)
758 GST_BYTE_READER_PEEK_GET(64,gdouble,float64_be)
759
760 /* *INDENT-ON* */
761
762 /**
763 * gst_byte_reader_get_data:
764 * @reader: a #GstByteReader instance
765 * @size: Size in bytes
766 * @val: (out) (transfer none) (array length=size): address of a
767 * #guint8 pointer variable in which to store the result
768 *
769 * Returns a constant pointer to the current data
770 * position if at least @size bytes are left and
771 * updates the current position.
772 *
773 * Returns: %TRUE if successful, %FALSE otherwise.
774 */
775 gboolean
gst_byte_reader_get_data(GstByteReader * reader,guint size,const guint8 ** val)776 gst_byte_reader_get_data (GstByteReader * reader, guint size,
777 const guint8 ** val)
778 {
779 return _gst_byte_reader_get_data_inline (reader, size, val);
780 }
781
782 /**
783 * gst_byte_reader_peek_data:
784 * @reader: a #GstByteReader instance
785 * @size: Size in bytes
786 * @val: (out) (transfer none) (array length=size): address of a
787 * #guint8 pointer variable in which to store the result
788 *
789 * Returns a constant pointer to the current data
790 * position if at least @size bytes are left and
791 * keeps the current position.
792 *
793 * Returns: %TRUE if successful, %FALSE otherwise.
794 */
795 gboolean
gst_byte_reader_peek_data(const GstByteReader * reader,guint size,const guint8 ** val)796 gst_byte_reader_peek_data (const GstByteReader * reader, guint size,
797 const guint8 ** val)
798 {
799 return _gst_byte_reader_peek_data_inline (reader, size, val);
800 }
801
802 /**
803 * gst_byte_reader_dup_data:
804 * @reader: a #GstByteReader instance
805 * @size: Size in bytes
806 * @val: (out) (transfer full) (array length=size): address of a
807 * #guint8 pointer variable in which to store the result
808 *
809 * Free-function: g_free
810 *
811 * Returns a newly-allocated copy of the current data
812 * position if at least @size bytes are left and
813 * updates the current position. Free with g_free() when no longer needed.
814 *
815 * Returns: %TRUE if successful, %FALSE otherwise.
816 */
817 gboolean
gst_byte_reader_dup_data(GstByteReader * reader,guint size,guint8 ** val)818 gst_byte_reader_dup_data (GstByteReader * reader, guint size, guint8 ** val)
819 {
820 return _gst_byte_reader_dup_data_inline (reader, size, val);
821 }
822
823 /* Special optimized scan for mask 0xffffff00 and pattern 0x00000100 */
824 static inline gint
_scan_for_start_code(const guint8 * data,guint size)825 _scan_for_start_code (const guint8 * data, guint size)
826 {
827 guint8 *pdata = (guint8 *) data;
828 guint8 *pend = (guint8 *) (data + size - 4);
829
830 while (pdata <= pend) {
831 if (pdata[2] > 1) {
832 pdata += 3;
833 } else if (pdata[1]) {
834 pdata += 2;
835 } else if (pdata[0] || pdata[2] != 1) {
836 pdata++;
837 } else {
838 return (pdata - data);
839 }
840 }
841
842 /* nothing found */
843 return -1;
844 }
845
846 static inline guint
_masked_scan_uint32_peek(const GstByteReader * reader,guint32 mask,guint32 pattern,guint offset,guint size,guint32 * value)847 _masked_scan_uint32_peek (const GstByteReader * reader,
848 guint32 mask, guint32 pattern, guint offset, guint size, guint32 * value)
849 {
850 const guint8 *data;
851 guint32 state;
852 guint i;
853
854 g_return_val_if_fail (size > 0, -1);
855 g_return_val_if_fail ((guint64) offset + size <= reader->size - reader->byte,
856 -1);
857
858 /* we can't find the pattern with less than 4 bytes */
859 if (G_UNLIKELY (size < 4))
860 return -1;
861
862 data = reader->data + reader->byte + offset;
863
864 /* Handle special case found in MPEG and H264 */
865 if ((pattern == 0x00000100) && (mask == 0xffffff00)) {
866 gint ret = _scan_for_start_code (data, size);
867
868 if (ret == -1)
869 return ret;
870
871 if (value != NULL)
872 *value = (1 << 8) | data[ret + 3];
873
874 return ret + offset;
875 }
876
877 /* set the state to something that does not match */
878 state = ~pattern;
879
880 /* now find data */
881 for (i = 0; i < size; i++) {
882 /* throw away one byte and move in the next byte */
883 state = ((state << 8) | data[i]);
884 if (G_UNLIKELY ((state & mask) == pattern)) {
885 /* we have a match but we need to have skipped at
886 * least 4 bytes to fill the state. */
887 if (G_LIKELY (i >= 3)) {
888 if (value)
889 *value = state;
890 return offset + i - 3;
891 }
892 }
893 }
894
895 /* nothing found */
896 return -1;
897 }
898
899
900 /**
901 * gst_byte_reader_masked_scan_uint32:
902 * @reader: a #GstByteReader
903 * @mask: mask to apply to data before matching against @pattern
904 * @pattern: pattern to match (after mask is applied)
905 * @offset: offset from which to start scanning, relative to the current
906 * position
907 * @size: number of bytes to scan from offset
908 *
909 * Scan for pattern @pattern with applied mask @mask in the byte reader data,
910 * starting from offset @offset relative to the current position.
911 *
912 * The bytes in @pattern and @mask are interpreted left-to-right, regardless
913 * of endianness. All four bytes of the pattern must be present in the
914 * byte reader data for it to match, even if the first or last bytes are masked
915 * out.
916 *
917 * It is an error to call this function without making sure that there is
918 * enough data (offset+size bytes) in the byte reader.
919 *
920 * Returns: offset of the first match, or -1 if no match was found.
921 *
922 * Example:
923 * |[
924 * // Assume the reader contains 0x00 0x01 0x02 ... 0xfe 0xff
925 *
926 * gst_byte_reader_masked_scan_uint32 (reader, 0xffffffff, 0x00010203, 0, 256);
927 * // -> returns 0
928 * gst_byte_reader_masked_scan_uint32 (reader, 0xffffffff, 0x00010203, 1, 255);
929 * // -> returns -1
930 * gst_byte_reader_masked_scan_uint32 (reader, 0xffffffff, 0x01020304, 1, 255);
931 * // -> returns 1
932 * gst_byte_reader_masked_scan_uint32 (reader, 0xffff, 0x0001, 0, 256);
933 * // -> returns -1
934 * gst_byte_reader_masked_scan_uint32 (reader, 0xffff, 0x0203, 0, 256);
935 * // -> returns 0
936 * gst_byte_reader_masked_scan_uint32 (reader, 0xffff0000, 0x02030000, 0, 256);
937 * // -> returns 2
938 * gst_byte_reader_masked_scan_uint32 (reader, 0xffff0000, 0x02030000, 0, 4);
939 * // -> returns -1
940 * ]|
941 */
942 guint
gst_byte_reader_masked_scan_uint32(const GstByteReader * reader,guint32 mask,guint32 pattern,guint offset,guint size)943 gst_byte_reader_masked_scan_uint32 (const GstByteReader * reader, guint32 mask,
944 guint32 pattern, guint offset, guint size)
945 {
946 return _masked_scan_uint32_peek (reader, mask, pattern, offset, size, NULL);
947 }
948
949 /**
950 * gst_byte_reader_masked_scan_uint32_peek:
951 * @reader: a #GstByteReader
952 * @mask: mask to apply to data before matching against @pattern
953 * @pattern: pattern to match (after mask is applied)
954 * @offset: offset from which to start scanning, relative to the current
955 * position
956 * @size: number of bytes to scan from offset
957 * @value: (out): pointer to uint32 to return matching data
958 *
959 * Scan for pattern @pattern with applied mask @mask in the byte reader data,
960 * starting from offset @offset relative to the current position.
961 *
962 * The bytes in @pattern and @mask are interpreted left-to-right, regardless
963 * of endianness. All four bytes of the pattern must be present in the
964 * byte reader data for it to match, even if the first or last bytes are masked
965 * out.
966 *
967 * It is an error to call this function without making sure that there is
968 * enough data (offset+size bytes) in the byte reader.
969 *
970 * Returns: offset of the first match, or -1 if no match was found.
971 *
972 * Since: 1.6
973 */
974 guint
gst_byte_reader_masked_scan_uint32_peek(const GstByteReader * reader,guint32 mask,guint32 pattern,guint offset,guint size,guint32 * value)975 gst_byte_reader_masked_scan_uint32_peek (const GstByteReader * reader,
976 guint32 mask, guint32 pattern, guint offset, guint size, guint32 * value)
977 {
978 return _masked_scan_uint32_peek (reader, mask, pattern, offset, size, value);
979 }
980
981 #define GST_BYTE_READER_SCAN_STRING(bits) \
982 static guint \
983 gst_byte_reader_scan_string_utf##bits (const GstByteReader * reader) \
984 { \
985 guint len, off, max_len; \
986 \
987 max_len = (reader->size - reader->byte) / sizeof (guint##bits); \
988 \
989 /* need at least a single NUL terminator */ \
990 if (max_len < 1) \
991 return 0; \
992 \
993 len = 0; \
994 off = reader->byte; \
995 /* endianness does not matter if we are looking for a NUL terminator */ \
996 while (GST_READ_UINT##bits##_LE (&reader->data[off]) != 0) { \
997 ++len; \
998 off += sizeof (guint##bits); \
999 /* have we reached the end without finding a NUL terminator? */ \
1000 if (len == max_len) \
1001 return 0; \
1002 } \
1003 /* return size in bytes including the NUL terminator (hence the +1) */ \
1004 return (len + 1) * sizeof (guint##bits); \
1005 }
1006
1007 #define GST_READ_UINT8_LE GST_READ_UINT8
1008 GST_BYTE_READER_SCAN_STRING (8);
1009 #undef GST_READ_UINT8_LE
1010 GST_BYTE_READER_SCAN_STRING (16);
1011 GST_BYTE_READER_SCAN_STRING (32);
1012
1013 #define GST_BYTE_READER_SKIP_STRING(bits) \
1014 gboolean \
1015 gst_byte_reader_skip_string_utf##bits (GstByteReader * reader) \
1016 { \
1017 guint size; /* size in bytes including the terminator */ \
1018 \
1019 g_return_val_if_fail (reader != NULL, FALSE); \
1020 \
1021 size = gst_byte_reader_scan_string_utf##bits (reader); \
1022 reader->byte += size; \
1023 return (size > 0); \
1024 }
1025
1026 /**
1027 * gst_byte_reader_skip_string:
1028 * @reader: a #GstByteReader instance
1029 *
1030 * Skips a NUL-terminated string in the #GstByteReader instance, advancing
1031 * the current position to the byte after the string. This will work for
1032 * any NUL-terminated string with a character width of 8 bits, so ASCII,
1033 * UTF-8, ISO-8859-N etc.
1034 *
1035 * This function will fail if no NUL-terminator was found in in the data.
1036 *
1037 * Returns: %TRUE if a string could be skipped, %FALSE otherwise.
1038 */
1039 /**
1040 * gst_byte_reader_skip_string_utf8:
1041 * @reader: a #GstByteReader instance
1042 *
1043 * Skips a NUL-terminated string in the #GstByteReader instance, advancing
1044 * the current position to the byte after the string. This will work for
1045 * any NUL-terminated string with a character width of 8 bits, so ASCII,
1046 * UTF-8, ISO-8859-N etc. No input checking for valid UTF-8 is done.
1047 *
1048 * This function will fail if no NUL-terminator was found in in the data.
1049 *
1050 * Returns: %TRUE if a string could be skipped, %FALSE otherwise.
1051 */
1052 GST_BYTE_READER_SKIP_STRING (8);
1053
1054 /**
1055 * gst_byte_reader_skip_string_utf16:
1056 * @reader: a #GstByteReader instance
1057 *
1058 * Skips a NUL-terminated UTF-16 string in the #GstByteReader instance,
1059 * advancing the current position to the byte after the string.
1060 *
1061 * No input checking for valid UTF-16 is done.
1062 *
1063 * This function will fail if no NUL-terminator was found in in the data.
1064 *
1065 * Returns: %TRUE if a string could be skipped, %FALSE otherwise.
1066 */
1067 GST_BYTE_READER_SKIP_STRING (16);
1068
1069 /**
1070 * gst_byte_reader_skip_string_utf32:
1071 * @reader: a #GstByteReader instance
1072 *
1073 * Skips a NUL-terminated UTF-32 string in the #GstByteReader instance,
1074 * advancing the current position to the byte after the string.
1075 *
1076 * No input checking for valid UTF-32 is done.
1077 *
1078 * This function will fail if no NUL-terminator was found in in the data.
1079 *
1080 * Returns: %TRUE if a string could be skipped, %FALSE otherwise.
1081 */
1082 GST_BYTE_READER_SKIP_STRING (32);
1083
1084 /**
1085 * gst_byte_reader_peek_string:
1086 * @reader: a #GstByteReader instance
1087 * @str: (out) (transfer none) (array zero-terminated=1): address of a
1088 * #gchar pointer variable in which to store the result
1089 *
1090 * Returns a constant pointer to the current data position if there is
1091 * a NUL-terminated string in the data (this could be just a NUL terminator).
1092 * The current position will be maintained. This will work for any
1093 * NUL-terminated string with a character width of 8 bits, so ASCII,
1094 * UTF-8, ISO-8859-N etc.
1095 *
1096 * This function will fail if no NUL-terminator was found in in the data.
1097 *
1098 * Returns: %TRUE if a string could be skipped, %FALSE otherwise.
1099 */
1100 /**
1101 * gst_byte_reader_peek_string_utf8:
1102 * @reader: a #GstByteReader instance
1103 * @str: (out) (transfer none) (array zero-terminated=1): address of a
1104 * #gchar pointer variable in which to store the result
1105 *
1106 * Returns a constant pointer to the current data position if there is
1107 * a NUL-terminated string in the data (this could be just a NUL terminator).
1108 * The current position will be maintained. This will work for any
1109 * NUL-terminated string with a character width of 8 bits, so ASCII,
1110 * UTF-8, ISO-8859-N etc.
1111 *
1112 * No input checking for valid UTF-8 is done.
1113 *
1114 * This function will fail if no NUL-terminator was found in in the data.
1115 *
1116 * Returns: %TRUE if a string could be skipped, %FALSE otherwise.
1117 */
1118 gboolean
gst_byte_reader_peek_string_utf8(const GstByteReader * reader,const gchar ** str)1119 gst_byte_reader_peek_string_utf8 (const GstByteReader * reader,
1120 const gchar ** str)
1121 {
1122 g_return_val_if_fail (reader != NULL, FALSE);
1123 g_return_val_if_fail (str != NULL, FALSE);
1124
1125 if (gst_byte_reader_scan_string_utf8 (reader) > 0) {
1126 *str = (const gchar *) (reader->data + reader->byte);
1127 } else {
1128 *str = NULL;
1129 }
1130 return (*str != NULL);
1131 }
1132
1133 /**
1134 * gst_byte_reader_get_string_utf8:
1135 * @reader: a #GstByteReader instance
1136 * @str: (out) (transfer none) (array zero-terminated=1): address of a
1137 * #gchar pointer variable in which to store the result
1138 *
1139 * Returns a constant pointer to the current data position if there is
1140 * a NUL-terminated string in the data (this could be just a NUL terminator),
1141 * advancing the current position to the byte after the string. This will work
1142 * for any NUL-terminated string with a character width of 8 bits, so ASCII,
1143 * UTF-8, ISO-8859-N etc.
1144 *
1145 * No input checking for valid UTF-8 is done.
1146 *
1147 * This function will fail if no NUL-terminator was found in in the data.
1148 *
1149 * Returns: %TRUE if a string could be found, %FALSE otherwise.
1150 */
1151 gboolean
gst_byte_reader_get_string_utf8(GstByteReader * reader,const gchar ** str)1152 gst_byte_reader_get_string_utf8 (GstByteReader * reader, const gchar ** str)
1153 {
1154 guint size; /* size in bytes including the terminator */
1155
1156 g_return_val_if_fail (reader != NULL, FALSE);
1157 g_return_val_if_fail (str != NULL, FALSE);
1158
1159 size = gst_byte_reader_scan_string_utf8 (reader);
1160 if (size == 0) {
1161 *str = NULL;
1162 return FALSE;
1163 }
1164
1165 *str = (const gchar *) (reader->data + reader->byte);
1166 reader->byte += size;
1167 return TRUE;
1168 }
1169
1170 #define GST_BYTE_READER_DUP_STRING(bits,type) \
1171 gboolean \
1172 gst_byte_reader_dup_string_utf##bits (GstByteReader * reader, type ** str) \
1173 { \
1174 guint size; /* size in bytes including the terminator */ \
1175 \
1176 g_return_val_if_fail (reader != NULL, FALSE); \
1177 g_return_val_if_fail (str != NULL, FALSE); \
1178 \
1179 size = gst_byte_reader_scan_string_utf##bits (reader); \
1180 if (size == 0) { \
1181 *str = NULL; \
1182 return FALSE; \
1183 } \
1184 *str = g_memdup (reader->data + reader->byte, size); \
1185 reader->byte += size; \
1186 return TRUE; \
1187 }
1188
1189 /**
1190 * gst_byte_reader_dup_string_utf8:
1191 * @reader: a #GstByteReader instance
1192 * @str: (out) (transfer full) (array zero-terminated=1): address of a
1193 * #gchar pointer variable in which to store the result
1194 *
1195 * Free-function: g_free
1196 *
1197 * FIXME:Reads (copies) a NUL-terminated string in the #GstByteReader instance,
1198 * advancing the current position to the byte after the string. This will work
1199 * for any NUL-terminated string with a character width of 8 bits, so ASCII,
1200 * UTF-8, ISO-8859-N etc. No input checking for valid UTF-8 is done.
1201 *
1202 * This function will fail if no NUL-terminator was found in in the data.
1203 *
1204 * Returns: %TRUE if a string could be read into @str, %FALSE otherwise. The
1205 * string put into @str must be freed with g_free() when no longer needed.
1206 */
1207 GST_BYTE_READER_DUP_STRING (8, gchar);
1208
1209 /**
1210 * gst_byte_reader_dup_string_utf16:
1211 * @reader: a #GstByteReader instance
1212 * @str: (out) (transfer full) (array zero-terminated=1): address of a
1213 * #guint16 pointer variable in which to store the result
1214 *
1215 * Free-function: g_free
1216 *
1217 * Returns a newly-allocated copy of the current data position if there is
1218 * a NUL-terminated UTF-16 string in the data (this could be an empty string
1219 * as well), and advances the current position.
1220 *
1221 * No input checking for valid UTF-16 is done. This function is endianness
1222 * agnostic - you should not assume the UTF-16 characters are in host
1223 * endianness.
1224 *
1225 * This function will fail if no NUL-terminator was found in in the data.
1226 *
1227 * Note: there is no peek or get variant of this function to ensure correct
1228 * byte alignment of the UTF-16 string.
1229 *
1230 * Returns: %TRUE if a string could be read, %FALSE otherwise. The
1231 * string put into @str must be freed with g_free() when no longer needed.
1232 */
1233 GST_BYTE_READER_DUP_STRING (16, guint16);
1234
1235 /**
1236 * gst_byte_reader_dup_string_utf32:
1237 * @reader: a #GstByteReader instance
1238 * @str: (out) (transfer full) (array zero-terminated=1): address of a
1239 * #guint32 pointer variable in which to store the result
1240 *
1241 * Free-function: g_free
1242 *
1243 * Returns a newly-allocated copy of the current data position if there is
1244 * a NUL-terminated UTF-32 string in the data (this could be an empty string
1245 * as well), and advances the current position.
1246 *
1247 * No input checking for valid UTF-32 is done. This function is endianness
1248 * agnostic - you should not assume the UTF-32 characters are in host
1249 * endianness.
1250 *
1251 * This function will fail if no NUL-terminator was found in in the data.
1252 *
1253 * Note: there is no peek or get variant of this function to ensure correct
1254 * byte alignment of the UTF-32 string.
1255 *
1256 * Returns: %TRUE if a string could be read, %FALSE otherwise. The
1257 * string put into @str must be freed with g_free() when no longer needed.
1258 */
1259 GST_BYTE_READER_DUP_STRING (32, guint32);
1260