You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

202 lines
5.3KB

  1. #ifndef _hls_parser_h_
  2. #define _hls_parser_h_
  3. #include <stdint.h>
  4. #include <stddef.h>
  5. #ifdef __cplusplus
  6. extern "C" {
  7. #endif
  8. enum
  9. {
  10. HLS_M3U8_UNKNOWN = 0,
  11. HLS_M3U8_PLAYLIST,
  12. HLS_M3U8_MASTER,
  13. };
  14. enum
  15. {
  16. HLS_PLAYLIST_TYPE_LIVE = 0,
  17. HLS_PLAYLIST_TYPE_EVENT,
  18. HLS_PLAYLIST_TYPE_VOD,
  19. };
  20. enum
  21. {
  22. HLS_MEDIA_UNKNOWN = 0,
  23. HLS_MEDIA_AUDIO,
  24. HLS_MEDIA_VIDEO,
  25. HLS_MEDIA_SUBTITLE,
  26. HLS_MEDIA_CLOSED_CAPTIONS,
  27. };
  28. // #EXT-X-MEDIA
  29. struct hls_media_t
  30. {
  31. char* type; // audio/video/subtitle/closed-captions
  32. char* uri; // Media Playlist file, If the TYPE is CLOSED-CAPTIONS, the URI attribute MUST NOT be present.
  33. char* group_id;
  34. char* language; // RFC5645
  35. char* assoc_language;
  36. char* name;
  37. char* instream_id; // CC1, SERVICE1, for HLS_MEDIA_CLOSED_CAPTIONS
  38. char* characteristics; // public.accessibility.describes-video
  39. char* channels; // for HLS_MEDIA_AUDIO
  40. int autoselect; // 1-YES, 0-NO
  41. int is_default; // 1-YES, 0-NO
  42. int forced; // 1-YES, 0-NO, SUBTITLES only
  43. };
  44. // #EXT-X-STREAM-INF
  45. struct hls_variant_t
  46. {
  47. uint32_t bandwidth; // the peak segment bit rate of the Variant Stream
  48. uint32_t average_bandwidth; // the average segment bit rate of the Variant Stream
  49. int width, height; // resolution
  50. double fps; // frame-rate
  51. char* hdcp_level; // TYPE-0
  52. char* uri;
  53. char* codecs; // RFC6381, mp4a.40.2,avc1.4d401e
  54. char* video_range; // SDR/PQ
  55. char* audio; // GROUP-ID of EXT-X-MEDIA
  56. char* video; // GROUP-ID of EXT-X-MEDIA
  57. char* subtitle; // GROUP-ID of EXT-X-MEDIA
  58. char* closed_captions; // GROUP-ID of EXT-X-MEDIA
  59. struct hls_variant_t* i_frame_stream_inf; // EXT-X-I-FRAME-STREAM-INF
  60. };
  61. struct hls_segment_t
  62. {
  63. double duration;
  64. char* uri;
  65. char* title;
  66. int64_t bytes; // EXT-X-BYTERANGE, -1 if don't exist
  67. int64_t offset; // EXT-X-BYTERANGE, -1 if don't exist
  68. // 1. file format
  69. // 2. number, type, and identifiers of tracks
  70. // 3. timestamp sequence
  71. // 4. encoding parameters
  72. // 5. encoding sequence
  73. int discontinuity; // EXT-X-DISCONTINUITY
  74. struct {
  75. char* method; // NONE, AES-128, and SAMPLE-AES
  76. char* uri;
  77. char* keyformat; // identity
  78. char* keyformatversions; // "1", "1/2", "1/2/5"
  79. uint8_t iv[16]; // 128bits
  80. } key; // EXT-X-KEY
  81. // It applies to every Media Segment that appears after it in the
  82. // Playlist until the next EXT-X-MAP tag or until the end of the Playlist.
  83. struct {
  84. char* uri;
  85. int64_t bytes; // BYTERANGE, -1 if don't exist
  86. int64_t offset; // BYTERANGE, -1 if don't exist
  87. } map; // EXT-X-MAP
  88. char* program_date_time; // EXT-X-PROGRAM-DATE-TIME
  89. struct {
  90. char* id;
  91. char* cls;
  92. char* start_date;
  93. char* end_date;
  94. double duration; // decimal-floating-point number of seconds
  95. double planned_duration; // decimal-floating-point number of seconds
  96. char* x_client_attribute; // TODO
  97. int end_on_next; // 1-YES, 0-NO
  98. } daterange; // EXT-X-DATERANGE
  99. };
  100. // If the Media Playlist contains the EXT-X-MEDIA-SEQUENCE tag, the
  101. // client SHOULD assume that each Media Segment in it will become
  102. // unavailable at the time that the Playlist file was loaded plus the
  103. // duration of the Playlist file.
  104. struct hls_playlist_t
  105. {
  106. int version;
  107. uint64_t target_duration; // EXT-X-TARGETDURATION, seconds
  108. uint64_t media_sequence; // EXT-X-MEDIA-SEQUENCE, base from 0
  109. uint64_t discontinuity_sequence; // EXT-X-DISCONTINUITY-SEQUENCE, decimal-integer
  110. int endlist; // EXT-X-ENDLIST, 1-endlist, 0-don't have endlist
  111. int type; // EXT-X-PLAYLIST-TYPE, 0-LIVE, 1-EVENT, 2-VOD
  112. int i_frames_only; // EXT-X-I-FRAMES-ONLY
  113. int independent_segments; // EXT-X-INDEPENDENT-SEGMENTS
  114. double start_time_offset; // EXT-X-START, seconds
  115. int start_precise; // EXT-X-START
  116. struct hls_segment_t* segments;
  117. size_t count;
  118. };
  119. struct hls_master_t
  120. {
  121. int version;
  122. size_t variant_count;
  123. struct hls_variant_t *variants;
  124. // renditions: audio, video, subtitle, close_captions
  125. size_t media_count;
  126. struct hls_media_t* medias;
  127. struct {
  128. char* data_id;
  129. char* value;
  130. char* uri;
  131. char* language;
  132. } *session_data;
  133. size_t session_data_count;
  134. struct {
  135. char* method; // NONE, AES-128, and SAMPLE-AES
  136. char* uri;
  137. char* keyformat; // identity
  138. char* keyformatversions; // "1", "1/2", "1/2/5"
  139. uint8_t iv[16]; // 128bits
  140. } *session_key; // EXT-X-SESSION-KEY
  141. size_t session_key_count;
  142. int independent_segments; // EXT-X-INDEPENDENT-SEGMENTS
  143. double start_time_offset; // EXT-X-START
  144. int start_precise; // EXT-X-START
  145. };
  146. /// Probe m3u8 content type(HLS_M3U8_MASTER/HLS_M3U8_PLAYLIST)
  147. /// @return 1-master playlist, 2-media playlist, other-unknown
  148. int hls_parser_probe(const char* m3u8, size_t len);
  149. /// Parse m3u8 master playlist
  150. /// @param[out] master m3u8 master playlist(free by hls_master_free)
  151. /// @return 0-ok, other-error
  152. int hls_master_parse(struct hls_master_t** master, const char* m3u8, size_t len);
  153. int hls_master_free(struct hls_master_t** master);
  154. int hls_master_best_variant(const struct hls_master_t* master);
  155. int hls_master_rendition(const struct hls_master_t* master, int variant, int type, const char* name);
  156. /// Parse m3u8 media playlist
  157. /// @param[out] playlist m3u8 media playlist(free by hls_playlist_free)
  158. /// @return 0-ok, other-error
  159. int hls_playlist_parse(struct hls_playlist_t** playlist, const char* m3u8, size_t len);
  160. int hls_playlist_free(struct hls_playlist_t** playlist);
  161. /// @return total duration in MS
  162. int64_t hls_playlist_duration(const struct hls_playlist_t* playlist);
  163. #ifdef __cplusplus
  164. }
  165. #endif
  166. #endif /* !_hls_parser_h_ */