Bitstream by VideoLAN
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.
 
 
 

207 lines
6.1 KiB

/*****************************************************************************
* pes.h: ISO/IEC 13818-1 Packetized Elementary Stream
*****************************************************************************
* Copyright (C) 2010 VideoLAN
*
* Authors: Christophe Massiot <massiot@via.ecp.fr>
*
* 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.
*****************************************************************************/
/*
* Normative references:
* - ISO/IEC 13818-1:2007(E) (MPEG-2 systems)
*/
#ifndef __BITSTREAM_MPEG_PES_H__
#define __BITSTREAM_MPEG_PES_H__
#include <stdint.h> /* uint8_t, uint16_t, etc... */
#include <stdbool.h> /* bool */
#include <string.h> /* memset */
#ifdef __cplusplus
extern "C"
{
#endif
/*****************************************************************************
* PES header
*****************************************************************************/
#define PES_HEADER_SIZE 6
#define PES_HEADER_SIZE_NOPTS 9
#define PES_HEADER_SIZE_PTS 14
#define PES_HEADER_SIZE_PTSDTS 19
#define PES_HEADER_OPTIONAL_SIZE 3
#define PES_HEADER_TS_SIZE 5
#define PES_STREAM_ID_MIN 0xbc
#define PES_STREAM_ID_PSM 0xbc
#define PES_STREAM_ID_PRIVATE_1 0xbd
#define PES_STREAM_ID_PADDING 0xbe
#define PES_STREAM_ID_PRIVATE_2 0xbf
#define PES_STREAM_ID_AUDIO_MPEG 0xc0 /* and following */
#define PES_STREAM_ID_VIDEO_MPEG 0xe0 /* and following */
#define PES_STREAM_ID_ECM 0xf0
#define PES_STREAM_ID_EMM 0xf1
#define PES_STREAM_ID_DSMCC 0xf2
#define PES_STREAM_ID_MHEG 0xf3
#define PES_STREAM_ID_H222_1_E 0xf8
#define PES_STREAM_ID_PSD 0xff
static inline void pes_init(uint8_t *p_pes)
{
p_pes[0] = 0x0;
p_pes[1] = 0x0;
p_pes[2] = 0x1;
}
static inline void pes_set_streamid(uint8_t *p_pes, uint8_t i_stream_id)
{
p_pes[3] = i_stream_id;
}
static inline uint8_t pes_get_streamid(const uint8_t *p_pes)
{
return p_pes[3];
}
static inline void pes_set_length(uint8_t *p_pes, uint16_t i_length)
{
p_pes[4] = i_length >> 8;
p_pes[5] = i_length & 0xff;
}
static inline uint16_t pes_get_length(const uint8_t *p_pes)
{
return (p_pes[4] << 8) | p_pes[5];
}
static inline void pes_set_headerlength(uint8_t *p_pes, uint8_t i_length)
{
p_pes[6] = 0x80;
p_pes[7] = 0x0;
p_pes[8] = i_length;
if ( i_length > 0 )
memset( &p_pes[9], 0xff, i_length ); /* stuffing */
}
static inline uint8_t pes_get_headerlength(const uint8_t *p_pes)
{
return p_pes[8];
}
static inline void pes_set_dataalignment(uint8_t *p_pes)
{
p_pes[6] |= 0x4;
}
static inline bool pes_get_dataalignment(const uint8_t *p_pes)
{
return !!(p_pes[6] & 0x4);
}
static inline bool pes_has_pts(const uint8_t *p_pes)
{
return !!(p_pes[7] & 0x80);
}
static inline bool pes_has_dts(const uint8_t *p_pes)
{
return (p_pes[7] & 0xc0) == 0xc0;
}
static inline void pes_set_pts(uint8_t *p_pes, uint64_t i_pts)
{
p_pes[7] |= 0x80;
if (p_pes[8] < 5)
p_pes[8] = 5;
uint8_t marker = pes_has_dts(p_pes) ? 0x30 : 0x20;
p_pes[9] = marker | 0x1 | ((i_pts >> 29) & 0xe);
p_pes[10] = (i_pts >> 22) & 0xff;
p_pes[11] = 0x1 | ((i_pts >> 14) & 0xfe);
p_pes[12] = (i_pts >> 7) & 0xff;
p_pes[13] = 0x1 | ((i_pts << 1) & 0xfe);
}
static inline bool pes_validate_pts(const uint8_t *p_pes)
{
return ((p_pes[9] & 0xe1) == 0x21)
&& (p_pes[11] & 0x1) && (p_pes[13] & 0x1);
}
static inline uint64_t pes_get_pts(const uint8_t *p_pes)
{
return (((uint64_t)p_pes[9] & 0xe)) << 29 | (p_pes[10] << 22) |
((p_pes[11] & 0xfe) << 14) | (p_pes[12] << 7) |
((p_pes[13] & 0xfe) >> 1);
}
static inline void pes_set_dts(uint8_t *p_pes, uint64_t i_dts)
{
p_pes[7] |= 0x40;
if (p_pes[8] < 10)
p_pes[8] = 10;
p_pes[9] &= 0x0f;
p_pes[9] |= 0x30;
p_pes[14] = 0x11 | ((i_dts >> 29) & 0xe);
p_pes[15] = (i_dts >> 22) & 0xff;
p_pes[16] = 0x1 | ((i_dts >> 14) & 0xfe);
p_pes[17] = (i_dts >> 7) & 0xff;
p_pes[18] = 0x1 | ((i_dts << 1) & 0xfe);
}
static inline bool pes_validate_dts(const uint8_t *p_pes)
{
return (p_pes[9] & 0x10) && ((p_pes[14] & 0xf1) == 0x11)
&& (p_pes[16] & 0x1) && (p_pes[18] & 0x1);
}
static inline uint64_t pes_get_dts(const uint8_t *p_pes)
{
return (((uint64_t)p_pes[14] & 0xe)) << 29 | (p_pes[15] << 22) |
((p_pes[16] & 0xfe) << 14) | (p_pes[17] << 7) |
((p_pes[18] & 0xfe) >> 1);
}
static inline bool pes_validate(const uint8_t *p_pes)
{
return (p_pes[0] == 0x0 && p_pes[1] == 0x0 && p_pes[2] == 0x1
&& p_pes[3] >= PES_STREAM_ID_MIN);
}
static inline bool pes_validate_header(const uint8_t *p_pes)
{
return ((p_pes[6] & 0xc0) == 0x80);
}
/*****************************************************************************
* PES payload
*****************************************************************************/
static inline uint8_t *pes_payload(uint8_t *p_pes)
{
return p_pes + PES_HEADER_SIZE + PES_HEADER_OPTIONAL_SIZE + pes_get_headerlength(p_pes);
}
#ifdef __cplusplus
}
#endif
#endif