1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354
|
/*
* Data definitions
* Copyright (C) 2011 Unix Solutions Ltd.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License (COPYING file) for more details.
*
*/
#ifndef DATA_H
#define DATA_H
#include <pthread.h>
#include <limits.h>
#include <stdbool.h>
#include <openssl/aes.h>
#include <openssl/des.h>
#include <openssl/md5.h>
#include "libfuncs/libfuncs.h"
#include "libtsfuncs/tsfuncs.h"
// 7 * 188
#define FRAME_SIZE 1316
// How much seconds to assume the key is valid
#define KEY_VALID_TIME 30
#define EMM_QUEUE_HARD_LIMIT 10000
#define EMM_QUEUE_SOFT_LIMIT 1000
#define ECM_QUEUE_HARD_LIMIT 10
#define ECM_QUEUE_SOFT_LIMIT 3
// 64k should be enough for everybody
#define THREAD_STACK_SIZE (64 * 1024)
struct notify {
pthread_t thread; /* Thread handle */
QUEUE *notifications; /* Notification queue */
char ident[512]; /* tsdecrypt ident (set by -i) */
char program[512]; /* What program to exec */
};
#define CODEWORD_LENGTH 16
#define BISSKEY_LENGTH 6
typedef void csakey_t;
struct key {
uint8_t cw[CODEWORD_LENGTH];
csakey_t *csakey;
int is_valid_cw;
time_t ts; // At what time the key is set
struct timeval ts_keyset; // At what time the key is set
};
// 4 auth header, 20 header size, 256 max data size, 16 potential padding
#define CAMD35_HDR_LEN (20)
#define CAMD35_BUF_LEN (4 + CAMD35_HDR_LEN + 256 + 16)
// When this limit is reached invalid_cw flag is set.
#define ECM_RECV_ERRORS_LIMIT 10
// When this limit is reached camd_reconnect is called.
#define EMM_RECV_ERRORS_LIMIT 100
struct camd;
struct ts;
enum msg_type { EMM_MSG, ECM_MSG };
struct camd_msg {
enum msg_type type;
uint16_t ca_id;
uint16_t service_id;
uint8_t data_len;
uint8_t data[255];
struct ts *ts;
};
enum camd_proto {
CAMD_CS378X,
CAMD_NEWCAMD,
};
struct camd_ops {
char *ident;
enum camd_proto proto;
int (*connect)(struct camd *c);
void (*disconnect)(struct camd *c);
int (*reconnect)(struct camd *c);
int (*do_emm)(struct camd *c, struct camd_msg *msg);
int (*do_ecm)(struct camd *c, struct camd_msg *msg);
int (*get_cw)(struct camd *c, uint16_t *ca_id, uint16_t *idx, uint8_t *cw);
};
struct cs378x {
// cs378x private data
uint8_t buf[CAMD35_BUF_LEN];
AES_KEY aes_encrypt_key;
AES_KEY aes_decrypt_key;
uint32_t auth_token;
uint16_t msg_id;
};
#define DESKEY_LENGTH 28
#define NEWCAMD_MSG_SIZE 400
#define NEWCAMD_MAXPROV 32
typedef struct {
DES_key_schedule ks1;
DES_key_schedule ks2;
uint8_t des_key[16];
} triple_des_t;
struct newcamd {
// newcamd private data
uint8_t buf[NEWCAMD_MSG_SIZE];
char hex_des_key[DESKEY_LENGTH + 1];
uint8_t bin_des_key[DESKEY_LENGTH / 2]; // Decoded des_key
triple_des_t td_key;
uint16_t msg_id;
// Initialized from CARD INFO command
int caid;
uint8_t ua[8];
uint8_t num_of_provs;
uint8_t provs_ident[NEWCAMD_MAXPROV][3];
uint8_t provs_id[NEWCAMD_MAXPROV][8];
uint8_t prov_ident_manual;
char *crypt_passwd;
};
struct camd {
int server_fd;
char *hostname;
char *service;
char *user;
char *pass;
unsigned int ecm_recv_errors; // Error counter, reset on successful send/recv
unsigned int emm_recv_errors; // Error counter, reset on successful send/recv
unsigned int no_reconnect;
unsigned int check_emm_errors;
struct key *key;
unsigned int constant_codeword; // The codeword is set on the command line once, no ecm processing is done.
pthread_t thread;
QUEUE *req_queue;
QUEUE *ecm_queue;
QUEUE *emm_queue;
struct camd_ops ops;
struct cs378x cs378x;
struct newcamd newcamd;
};
enum io_type {
FILE_IO,
NET_IO,
WTF_IO
};
struct io {
int fd;
enum io_type type;
char *fname;
char *hostname;
char *service;
// Used only for output
int ttl;
int tos;
struct in_addr intf;
int v6_if_index;
// Used for input
struct in_addr isrc;
};
struct packet_buf {
int64_t time;
uint8_t data[188];
};
#define MAX_FILTERS 16
#define MAX_FILTER_NAME 32
#define MAX_FILTER_LEN 16
enum filter_action {
FILTER_NO_MATCH = 0,
FILTER_ACCEPT_ALL = 1,
FILTER_REJECT_ALL = 2,
FILTER_ACCEPT = 3,
FILTER_REJECT = 4,
};
enum filter_type {
FILTER_TYPE_DATA = 0, // Compare data at offset X
FILTER_TYPE_MASK = 1, // Compare data + mask
FILTER_TYPE_LENGTH = 2, // Compare section length /EMM[2]/)
};
struct filter {
enum filter_action action;
enum filter_type type;
uint8_t offset; // Offset into EMM
uint8_t filter_len; // Filter length
uint8_t data[MAX_FILTER_LEN]; // Data | Matched bytes
uint8_t mask[MAX_FILTER_LEN]; // Mask bytes
char name[MAX_FILTER_NAME]; // Filter name (default: NO_NAME)
};
struct chid {
int seen;
uint16_t chid;
};
#define MAX_PIDS 8192
struct ts {
// Stream handling
struct ts_pat *pat, *curpat;
struct ts_pat *genpat;
uint8_t genpat_cc;
struct ts_cat *cat, *curcat;
struct ts_pmt *pmt, *curpmt;
struct ts_sdt *sdt, *cursdt;
struct ts_privsec *emm, *last_emm;
struct ts_privsec *ecm, *last_ecm;
struct ts_privsec *tmp_emm;
struct ts_privsec *tmp_ecm;
uint16_t pmt_pid;
uint16_t service_id;
uint16_t forced_service_id;
uint16_t emm_caid, emm_pid;
uint16_t ecm_caid, ecm_pid;
uint16_t forced_caid;
uint16_t forced_emm_pid;
uint16_t forced_ecm_pid;
pidmap_t pidmap;
pidmap_t cc; // Continuity counters
pidmap_t pid_seen;
// Stats
unsigned int emm_input_count;
unsigned int emm_seen_count;
unsigned int emm_processed_count;
unsigned int emm_skipped_count;
unsigned int emm_report_interval;
time_t emm_last_report;
unsigned int ecm_seen_count;
unsigned int ecm_processed_count;
unsigned int ecm_duplicate_count;
unsigned int ecm_report_interval;
time_t ecm_last_report;
unsigned int cw_warn_sec;
time_t cw_last_warn;
time_t cw_next_warn;
struct timeval ecm_change_time;
unsigned int pid_report;
unsigned int pid_stats[MAX_PIDS];
// CAMD handling
struct key key;
struct camd camd;
// Config
char *ident;
char *syslog_host;
int syslog_port;
int syslog_active;
int syslog_remote;
char *pidfile;
enum CA_system req_CA_sys;
bool output_stream; // Decode and output the decoded stream
bool process_ecm; // Process ECM packets (and send them to CAMD)
bool process_emm; // Process EMM packets (and send them to CAMD)
int pid_filter;
int eit_passthrough;
int tdt_passthrough;
int nit_passthrough;
uint8_t irdeto_ecm_idx;
uint16_t irdeto_ecm_chid;
enum {
IRDETO_FILTER_IDX,
IRDETO_FILTER_CHID,
} irdeto_ecm_filter_type;
struct chid irdeto_chid[0xff];
uint8_t irdeto_max_chids;
int ecm_cw_log;
int rtp_input;
int rtp_output;
uint32_t rtp_ssrc;
uint16_t rtp_seqnum;
struct io input;
struct io output;
FILE *input_dump_file;
char *input_dump_filename;
int debug_level;
int ts_discont;
int camd_stop;
int is_cw_error;
int no_output_on_error;
int threaded;
pthread_attr_t thread_attr;
int decode_stop;
pthread_t decode_thread;
CBUF *decode_buf;
int write_stop;
pthread_t write_thread;
CBUF *write_buf;
struct notify *notify;
char *notify_program;
unsigned int input_buffer_time;
LIST *input_buffer;
int emm_filters_num;
struct filter emm_filters[MAX_FILTERS];
};
void data_init(struct ts *ts);
void data_free(struct ts *ts);
#endif
|