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.
 
 
 

189 lines
5.5 KiB

/*****************************************************************************
* 291.h: SMPTE 291 Ancillary Data Packet and Space Formatting
*****************************************************************************
* Copyright (C) 2016 OpenHeadend
*
* Authors: Christophe Massiot <cmassiot@openheadend.tv>
*
* 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:
* - SMPTE 291
* - SMPTE RP-291
*/
#ifndef __BITSTREAM_SMPTE_291_H__
#define __BITSTREAM_SMPTE_291_H__
#include <stdint.h> /* uint8_t, uint16_t, etc... */
#include <stdbool.h> /* bool */
#ifdef __cplusplus
extern "C"
{
#endif
/*****************************************************************************
* SMPTE 291 ancillary data
*****************************************************************************/
#define S291_HEADER_SIZE 6
#define S291_FOOTER_SIZE 1
#define S291_ADF1 0x0000
#define S291_ADF2 0x03ff
#define S291_ADF3 0x03ff
/* SMPTE 2016 */
#define S291_AFD_DID 0x41
#define S291_AFD_SDID 0x05
#define S291_PAN_SCAN_DID 0x41
#define S291_PAN_SCAN_SDID 0x06
/* SMPTE 2010 */
#define S291_SCTE104_DID 0x41
#define S291_SCTE104_SDID 0x07
/* OP-47 / SMPTE RDD-8 */
#define S291_OP47SDP_DID 0x43
#define S291_OP47SDP_SDID 0x02
#define S291_OP47MP_DID 0x43
#define S291_OP47MP_SDID 0x03
/* SMPTE 12M */
#define S291_ATC_DID 0x60
#define S291_ATC_SDID 0x60
/* SMPTE 334 */
#define S291_CEA708_DID 0x61
#define S291_CEA708_SDID 0x01
#define S291_CEA608_DID 0x61
#define S291_CEA608_SDID 0x02
/* SMPTE RP-207 */
#define S291_PROGDESC_DID 0x62
#define S291_PROGDESC_SDID 0x01
/* SMPTE 334 */
#define S291_DATABROADCAST_DID 0x62
#define S291_DATABROADCAST_SDID 0x02
/* SMPTE RP-208 (legacy) */
#define S291_SMPTEVBI_DID 0x62
#define S291_SMPTEVBI_SDID 0x03
/* SMPTE 299 */
#define S291_HD_AUDIO_GROUP1_DID 0xe7
#define S291_HD_AUDIO_GROUP2_DID 0xe6
#define S291_HD_AUDIO_GROUP3_DID 0xe5
#define S291_HD_AUDIO_GROUP4_DID 0xe4
#define S291_HD_AUDIOCONTROL_GROUP1_DID 0xe3
#define S291_HD_AUDIOCONTROL_GROUP2_DID 0xe2
#define S291_HD_AUDIOCONTROL_GROUP3_DID 0xe1
#define S291_HD_AUDIOCONTROL_GROUP4_DID 0xe0
/* SMPTE 352 */
#define S291_PAYLOADID_DID 0x41
#define S291_PAYLOADID_SDID 0x01
/* SMPTE 272 */
#define S291_SD_AUDIO_GROUP1_DID 0xff
#define S291_SD_AUDIO_GROUP2_DID 0xfd
#define S291_SD_AUDIO_GROUP3_DID 0xfb
#define S291_SD_AUDIO_GROUP4_DID 0xf9
#define S291_SD_AUDIOCONTROL_GROUP1_DID 0xef
#define S291_SD_AUDIOCONTROL_GROUP2_DID 0xee
#define S291_SD_AUDIOCONTROL_GROUP3_DID 0xed
#define S291_SD_AUDIOCONTROL_GROUP4_DID 0xec
static inline uint16_t s291_parity(uint8_t i_val)
{
uint8_t i_parity =
(((i_val * 0x0101010101010101ULL) & 0x8040201008040201ULL) % 0x1FF) & 1;
return (uint16_t)i_parity << 8 | ((uint16_t)(i_parity ^ 1) << 9);
}
static inline void s291_set_did(uint16_t *p_s291, uint8_t i_did)
{
p_s291[3] = i_did | s291_parity(i_did);
}
static inline uint8_t s291_get_did(const uint16_t *p_s291)
{
return p_s291[3] & 0xff;
}
static inline void s291_set_sdid(uint16_t *p_s291, uint8_t i_sdid)
{
p_s291[4] = i_sdid | s291_parity(i_sdid);
}
static inline uint8_t s291_get_sdid(const uint16_t *p_s291)
{
return p_s291[4] & 0xff;
}
#define s291_set_dbn s291_set_sdid
#define s291_get_dbn s291_get_sdid
static inline void s291_set_dc(uint16_t *p_s291, uint8_t i_dc)
{
p_s291[5] = i_dc | s291_parity(i_dc);
}
static inline uint8_t s291_get_dc(const uint16_t *p_s291)
{
return p_s291[5] & 0xff;
}
static inline uint16_t *s291_get_udw(const uint16_t *p_s291)
{
return (uint16_t *)&p_s291[6];
}
static inline uint16_t s291_compute_cs(const uint16_t *p_s291)
{
uint16_t i_cs = 0;
uint8_t i_dc = s291_get_dc(p_s291);
unsigned int i;
for (i = 3; i < i_dc + S291_HEADER_SIZE; i++) {
i_cs += p_s291[i] & 0x1ff;
i_cs &= 0x1ff;
}
return i_cs | (~i_cs & 0x100) << 1;
}
static inline void s291_set_cs(uint16_t *p_s291)
{
p_s291[S291_HEADER_SIZE + s291_get_dc(p_s291)] = s291_compute_cs(p_s291);
}
static inline bool s291_check_cs(const uint16_t *p_s291)
{
return p_s291[S291_HEADER_SIZE + s291_get_dc(p_s291)] ==
s291_compute_cs(p_s291);
}
#ifdef __cplusplus
}
#endif
#endif