| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212 |
- /**********************************************************************************************/
- /* The MIT License */
- /* */
- /* Copyright 2016-2017 Twitch Interactive, Inc. or its affiliates. All Rights Reserved. */
- /* */
- /* Permission is hereby granted, free of charge, to any person obtaining a copy */
- /* of this software and associated documentation files (the "Software"), to deal */
- /* in the Software without restriction, including without limitation the rights */
- /* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell */
- /* copies of the Software, and to permit persons to whom the Software is */
- /* furnished to do so, subject to the following conditions: */
- /* */
- /* The above copyright notice and this permission notice shall be included in */
- /* all copies or substantial portions of the Software. */
- /* */
- /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR */
- /* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, */
- /* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE */
- /* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER */
- /* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, */
- /* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN */
- /* THE SOFTWARE. */
- /**********************************************************************************************/
- #ifndef LIBCAPTION_MPEG_H
- #define LIBCAPTION_MPEG_H
- #ifdef __cplusplus
- extern "C" {
- #endif
- #include "caption.h"
- #include "cea708.h"
- #include "scc.h"
- #include <float.h>
- #include <stddef.h>
- ////////////////////////////////////////////////////////////////////////////////
- #define STREAM_TYPE_H262 0x02
- #define STREAM_TYPE_H264 0x1B
- #define STREAM_TYPE_H265 0x24
- #define H262_SEI_PACKET 0xB2
- #define H264_SEI_PACKET 0x06
- #define H265_SEI_PACKET 0x27 // There is also 0x28
- #define MAX_NALU_SIZE (6 * 1024 * 1024)
- #define MAX_REFRENCE_FRAMES 64
- typedef struct {
- size_t size;
- uint8_t data[MAX_NALU_SIZE + 1];
- double dts, cts;
- libcaption_stauts_t status;
- // Priority queue for out of order frame processing
- // Should probablly be a linked list
- size_t front;
- size_t latent;
- cea708_t cea708[MAX_REFRENCE_FRAMES];
- } mpeg_bitstream_t;
- void mpeg_bitstream_init(mpeg_bitstream_t* packet);
- ////////////////////////////////////////////////////////////////////////////////
- // TODO make convenience functions for flv/mp4
- /*! \brief
- \param
- */
- size_t mpeg_bitstream_parse(mpeg_bitstream_t* packet, caption_frame_t* frame, const uint8_t* data, size_t size, unsigned stream_type, double dts, double cts);
- /*! \brief
- \param
- */
- static inline libcaption_stauts_t mpeg_bitstream_status(mpeg_bitstream_t* packet) { return packet->status; }
- /*! \brief
- Flushes latent packets caused by out or order frames.
- Returns number of latent frames remaining, 0 when complete;
- \param
- */
- size_t mpeg_bitstream_flush(mpeg_bitstream_t* packet, caption_frame_t* frame);
- ////////////////////////////////////////////////////////////////////////////////
- typedef enum {
- sei_type_buffering_period = 0,
- sei_type_pic_timing = 1,
- sei_type_pan_scan_rect = 2,
- sei_type_filler_payload = 3,
- sei_type_user_data_registered_itu_t_t35 = 4,
- sei_type_user_data_unregistered = 5,
- sei_type_recovery_point = 6,
- sei_type_dec_ref_pic_marking_repetition = 7,
- sei_type_spare_pic = 8,
- sei_type_scene_info = 9,
- sei_type_sub_seq_info = 10,
- sei_type_sub_seq_layer_characteristics = 11,
- sei_type_sub_seq_characteristics = 12,
- sei_type_full_frame_freeze = 13,
- sei_type_full_frame_freeze_release = 14,
- sei_type_full_frame_snapshot = 15,
- sei_type_progressive_refinement_segment_start = 16,
- sei_type_progressive_refinement_segment_end = 17,
- sei_type_motion_constrained_slice_group_set = 18,
- sei_type_film_grain_characteristics = 19,
- sei_type_deblocking_filter_display_preference = 20,
- sei_type_stereo_video_info = 21,
- } sei_msgtype_t;
- ////////////////////////////////////////////////////////////////////////////////
- typedef struct _sei_message_t {
- size_t size;
- sei_msgtype_t type;
- struct _sei_message_t* next;
- } sei_message_t;
- typedef struct {
- double timestamp;
- sei_message_t* head;
- sei_message_t* tail;
- } sei_t;
- /*! \brief
- \param
- */
- void sei_init(sei_t* sei, double timestamp);
- /*! \brief
- \param
- */
- void sei_free(sei_t* sei);
- /*! \brief
- \param
- */
- void sei_cat(sei_t* to, sei_t* from, int itu_t_t35);
- /*! \brief
- \param
- */
- void sei_message_append(sei_t* sei, sei_message_t* msg);
- /*! \brief
- \param
- */
- libcaption_stauts_t sei_parse(sei_t* sei, const uint8_t* data, size_t size, double timestamp);
- /*! \brief
- \param
- */
- static inline sei_message_t* sei_message_head(sei_t* sei) { return sei->head; }
- /*! \brief
- \param
- */
- static inline sei_message_t* sei_message_tail(sei_t* sei) { return sei->tail; }
- /*! \brief
- \param
- */
- sei_message_t* sei_message_next(sei_message_t* msg);
- /*! \brief
- \param
- */
- sei_msgtype_t sei_message_type(sei_message_t* msg);
- /*! \brief
- \param
- */
- size_t sei_message_size(sei_message_t* msg);
- /*! \brief
- \param
- */
- uint8_t* sei_message_data(sei_message_t* msg);
- /*! \brief
- \param
- */
- sei_message_t* sei_message_new(sei_msgtype_t type, uint8_t* data, size_t size);
- /*! \brief
- \param
- */
- static inline sei_message_t* sei_message_copy(sei_message_t* msg)
- {
- return sei_message_new(sei_message_type(msg), sei_message_data(msg), sei_message_size(msg));
- }
- /**
- Free message and all accoiated data. Messaged added to sei_t by using sei_append_message MUST NOT be freed
- These messages will be freed by calling sei_free()
- */
- /*! \brief
- \param
- */
- void sei_message_free(sei_message_t* msg);
- ////////////////////////////////////////////////////////////////////////////////
- /*! \brief
- \param
- */
- size_t sei_render_size(sei_t* sei);
- /*! \brief
- \param
- */
- size_t sei_render(sei_t* sei, uint8_t* data);
- /*! \brief
- \param
- */
- void sei_dump(sei_t* sei);
- /*! \brief
- \param
- */
- void sei_dump_messages(sei_message_t* head, double timestamp);
- ////////////////////////////////////////////////////////////////////////////////
- /*! \brief
- \param
- */
- libcaption_stauts_t sei_from_scc(sei_t* sei, const scc_t* scc);
- /*! \brief
- \param
- */
- libcaption_stauts_t sei_from_caption_frame(sei_t* sei, caption_frame_t* frame);
- /*! \brief
- \param
- */
- libcaption_stauts_t sei_from_caption_clear(sei_t* sei);
- /*! \brief
- \param
- */
- libcaption_stauts_t sei_to_caption_frame(sei_t* sei, caption_frame_t* frame);
- ////////////////////////////////////////////////////////////////////////////////
- #ifdef __cplusplus
- }
- #endif
- #endif
|