#ifndef _GF_DVB_MPE_DEV_H_
#define _GF_DVB_MPE_DEV_H_
#include <gpac/dvb_mpe.h>
#include <gpac/internal/reedsolomon.h>
#ifndef GPAC_DISABLE_MPEG2TS
typedef struct
{
        u32 id;
        u32 processing_order;
        u32 number_of_descriptor;
        GF_List * descriptors;
} GF_M2TS_INT;
typedef struct
{
        u32 tag;
        u32 length;
        u32 network_id;
        u32 original_network_id;
        u32 ts_id;
        u32 service_id;
        u32 component_tag;
} GF_M2TS_LOC_DSCPTR_IP_STREAM;
typedef struct descriptor_TimeSliceFec
{
        Bool time_slicing;
        u8 mpe_fec;
        u8 frame_size;
        u8 max_burst_duration;
        u8 max_average_rate;
        u8 time_slice_fec_id;
        u8 * id_selector;
} GF_M2TS_DesTimeSliceFec;
typedef struct
{
        u16     network_id;
        u16 original_network_id;
        u16 transport_stream_id;
        u16 service_id;
        u8 component_tag;
} GF_M2TS_DesLocation;
typedef struct {
        u8 type; 
        u32 address_mask;
        u8 address[4];
        u8 slash_mask;
        u32 rx_port[10];   
} GF_M2TS_IP_Target;
typedef struct
{
        GF_List *targets; 
        u32 PID;
        Bool stream_info_gathered;
        
        GF_M2TS_DesLocation location;
        GF_M2TS_DesTimeSliceFec time_slice_fec;
} GF_M2TS_IP_Stream;
typedef struct __gf_dvb_mpe_ip_platform
{
        
        u32 id;
        u32 processing_order;
        u32 number_of_descriptor;
        u8 *name; 
        u8 *provider_name; 
        
        GF_M2TS_DesLocation *location;
        GF_List * ip_streams;
        Bool all_info_gathered;
        GF_List *socket_struct;
} GF_M2TS_IP_PLATFORM;
typedef struct
{
        char *data;                        
        u32 u32_version;                   
        u32 u32_hdr_length;                
        u32 u32_total_length;              
        u32 u32_payload_size;              
        u32 u32_id_nb;                     
        u32 u32_flag;                      
        u32 u32_frag_offset;               
        u32 u32_TTL;                       
        u32 u32_protocol;                  
        u32 u32_crc;
        u8 u8_tx_adr[4];                   
        u8 u8_rx_adr[4];                   
        u32 u32_size_option;               
        u32 u32_padding;                   
        u32 u32_sum;
        
        u32 u32_tx_udp_port;                
        u32 u32_rx_udp_port;                
        u32 u32_udp_data_size;
        u32 u32_udp_chksm;
} GF_M2TS_IP_Packet;
#define MPE_ADT_COLS 191
#define MPE_RS_COLS NPAR
typedef struct mpe_error_holes
{
        u32 offset;
        u32 length;
} MPE_Error_Holes;
typedef struct mpe_fec_frame
{
        u32 rows;
        u32 col_adt ;
        u32 col_rs ;
        u8 *p_adt; 
        u8 *p_rs;  
        u32 *p_error_adt;
        u32 *p_error_rs ;
        u32 capacity_total;
        u32 current_offset_adt ;
        u32 current_offset_rs;
        u32 initialized ;
        u8  ADT_done;
        u32 PID;
        GF_List *mpe_holes;
        
} MPE_FEC_FRAME;
void gf_m2ts_process_int(GF_M2TS_Demuxer *ts, GF_M2TS_SECTION_ES *ip_table, unsigned char *data, u32 data_size, u32 table_id);
void section_DSMCC_INT(GF_M2TS_IP_PLATFORM* ip_platform, u8 *data, u32 data_size);
u32  platform_descriptorDSMCC_INT_UNT  (GF_M2TS_IP_PLATFORM* ip_platform,u8 *data);
u32 dsmcc_pto_platform_descriptor_loop (GF_M2TS_IP_PLATFORM* ip_platform,u8 *data);
u32  descriptorDSMCC_INT_UNT  (GF_M2TS_IP_Stream *ip_str,u8 *data);
void descriptorDSMCC_target_IP_address ( GF_M2TS_IP_Stream *ip_str, u8 *data);
u32 dsmcc_pto_descriptor_loop (GF_M2TS_IP_Stream *ip_str, u8 *data);
void descriptorTime_slice_fec_identifier(GF_M2TS_IP_Stream *ip_str, u8 *data);
void gf_m2ts_target_ip( GF_M2TS_IP_Stream* ip_str, u8 *data);
void descriptorLocation(GF_M2TS_IP_Stream *ip_str , u8 *data);
void gf_ip_platform_descriptor(GF_M2TS_IP_PLATFORM* ip_platform, u8 *data);
void gf_ip_platform_provider_descriptor(GF_M2TS_IP_PLATFORM* ip_platform,u8 *data);
void gf_m2ts_ip_platform_init(GF_M2TS_IP_PLATFORM * ip_platform);
u32 gf_m2ts_ipdatagram_reader(u8 *datagram, GF_M2TS_IP_Packet *ip_packet, u32 offset);
void gf_m2ts_process_ipdatagram(MPE_FEC_FRAME *mff,GF_M2TS_Demuxer *ts);
Bool gf_m2ts_compare_ip(u8 rx_ip_address[4], u8 ip_address_bootstrap[4]);
struct _sock_entry
{
        u32 ipv4_addr;
        u16 port;
        GF_Socket *sock;
        Bool bind_failure;
};
struct tag_m2ts_section_mpe
{
        ABSTRACT_ES
        GF_M2TS_SectionFilter *sec;
        
        GF_M2TS_IP_Stream *ip_platform;
        MPE_FEC_FRAME *mff;
};
void gf_m2ts_process_mpe(GF_M2TS_Demuxer *ts, GF_M2TS_SECTION_MPE *mpe, unsigned char *data, u32 data_size, u8 table_id);
void gf_m2ts_gather_ipdatagram_information(MPE_FEC_FRAME *mff,GF_M2TS_Demuxer *ts);
void socket_simu(GF_M2TS_IP_Packet *ip_packet, GF_M2TS_Demuxer *ts, Bool yield);
void gf_m2ts_mpe_send_datagram(GF_M2TS_Demuxer *ts, u32 pid, unsigned char *data, u32 data_size);
u32 init_frame(MPE_FEC_FRAME * mff, u32 rows);
void getRowFromADT(MPE_FEC_FRAME * mff,u32 index, u8 * adt_row);
void getRowFromRS(MPE_FEC_FRAME * mff,u32 index, u8 * rs_row);
void setRowRS(MPE_FEC_FRAME * mff,u32 index, u8 * p_rs);
void getErrorPositions(MPE_FEC_FRAME * mff, u32 row, u32 * errPositions);
void setColRS( MPE_FEC_FRAME * mff, u32 offset, u8 * pds, u32 length );
void getColRS(MPE_FEC_FRAME * mff, u32 offset, u8 * pds, u32 length);
void setIpDatagram(MPE_FEC_FRAME * mff,u32 offset, u8 * dgram, u32 length );
void setErrorIndicator(u32 * data , u32 offset1, u32 length  );
void resetMFF(MPE_FEC_FRAME * mff) ;
u32  getErrasurePositions( MPE_FEC_FRAME *mff , u32 row, u32 *errasures);
void decode_fec(MPE_FEC_FRAME * mff);
typedef enum {
        MPEG, DVB_SI,
        DSMCC_STREAM, DSMCC_CAROUSEL, DSMCC_INT_UNT, MHP_AIT, TVA_RNT
} DTAG_SCOPE;
void descriptor_PRIVATE (u8 *b, DTAG_SCOPE tag_scope, GF_List * descriptors );
#endif 
#endif