/***************************************************************************** * 291.h: SMPTE 291 Ancillary Data Packet and Space Formatting ***************************************************************************** * Copyright (C) 2016 OpenHeadend * * Authors: Christophe Massiot * * 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 /* uint8_t, uint16_t, etc... */ #include /* 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 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