/***************************************************************************** * ci.h: CENELEC EN 50 221 Common Interface Specification ***************************************************************************** * Copyright (C) 2010 VideoLAN * * 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: * - CENELEC EN 50 221 (1997) (Common Interface Specification) */ #ifndef __BITSTREAM_DVB_CI_H__ #define __BITSTREAM_DVB_CI_H__ #include #include #ifdef __cplusplus extern "C" { #endif /***************************************************************************** * Conditional Access Program Map Table ***************************************************************************** * Implementation note: the first field here is ca_pmt_list_management, * since the previous length field is variable size and is generally put * afterwards. *****************************************************************************/ #define CAPMT_HEADER_SIZE 6 #define CAPMT_ES_SIZE 5 #define CAPMTI_HEADER_SIZE 3 /* Max theoritical size is PSI_MAX_SIZE + ~100 */ #define CAPMT_DECLARE PSI_PRIVATE_DECLARE #define capmt_allocate psi_private_allocate static inline void capmt_init(uint8_t *p_capmt) { p_capmt[3] = 0xc1; p_capmt[4] = 0xf0; } static inline void capmt_set_listmanagement(uint8_t *p_capmt, uint8_t i_mgt) { p_capmt[0] = i_mgt; } static inline void capmt_set_program(uint8_t *p_capmt, uint16_t i_program) { p_capmt[1] = i_program >> 8; p_capmt[2] = i_program & 0xff; } static inline void capmt_set_version(uint8_t *p_capmt, uint8_t i_version) { p_capmt[3] &= ~0x3e; p_capmt[3] |= i_version << 1; } static inline uint8_t *capmt_get_infos(uint8_t *p_capmt) { return &p_capmt[4]; } static inline uint16_t capmt_get_infoslength(const uint8_t *p_capmt) { return ((p_capmt[4] & 0xf) << 8) | p_capmt[5]; } static inline void capmtn_init(uint8_t *p_capmt_n) { p_capmt_n[1] = 0xe0; } static inline void capmtn_set_streamtype(uint8_t *p_capmt_n, uint8_t i_stream_type) { p_capmt_n[0] = i_stream_type; } static inline void capmtn_set_pid(uint8_t *p_capmt_n, uint16_t i_pid) { p_capmt_n[1] &= ~0x1f; p_capmt_n[1] |= i_pid >> 8; p_capmt_n[2] = i_pid & 0xff; } static inline uint16_t capmtn_get_infoslength(const uint8_t *p_capmt_n) { return ((p_capmt_n[3] & 0xf) << 8) | p_capmt_n[4]; } static inline uint8_t *capmtn_get_infos(uint8_t *p_capmt_n) { return &p_capmt_n[3]; } static inline uint8_t *capmt_get_es(uint8_t *p_capmt, uint8_t n) { uint8_t *p_capmt_n = p_capmt + CAPMT_HEADER_SIZE + capmt_get_infoslength(p_capmt); while (n) { p_capmt_n += CAPMT_ES_SIZE + capmtn_get_infoslength(p_capmt_n); n--; } return p_capmt_n; } static inline void capmti_init(uint8_t *p_infos) { p_infos[0] = 0xf0; } static inline void capmti_set_length(uint8_t *p_infos, uint16_t i_length) { p_infos[0] &= ~0xf; p_infos[0] |= i_length >> 8; p_infos[1] = i_length & 0xff; } static inline uint16_t capmti_get_length(const uint8_t *p_infos) { return ((p_infos[0] & 0xf) << 8) | p_infos[1]; } static inline void capmti_set_cmd(uint8_t *p_infos, uint8_t i_cmd) { p_infos[2] = i_cmd; } static inline uint8_t *capmti_get_info(uint8_t *p_infos, uint16_t n) { uint8_t *p_info = p_infos + CAPMTI_HEADER_SIZE; uint16_t i_infos_size = capmti_get_length(p_infos) + DESCS_HEADER_SIZE; while (n) { if (p_info + DESC_HEADER_SIZE - p_infos > i_infos_size) return NULL; p_info += DESC_HEADER_SIZE + desc_get_length(p_info); n--; } if (p_info - p_infos >= i_infos_size) return NULL; return p_info; } #ifdef __cplusplus } #endif #endif