mptsd reads mpegts streams from udp/multicast or http and combines them into one multiple program stream that is suitable for outputting to DVB-C modulator. Tested with Dektec DTE-3114 Quad QAM Modulator and used in production in small DVB-C networks. https://georgi.unixsol.org/programs/mptsd/
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.

data.c 9.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389
  1. #include <stdlib.h>
  2. #include <math.h>
  3. #include <ctype.h>
  4. #include <unistd.h>
  5. #include <string.h>
  6. #include <sys/time.h>
  7. #include <regex.h>
  8. #include <sys/types.h>
  9. #include <sys/stat.h>
  10. #include <fcntl.h>
  11. #include "libfuncs/io.h"
  12. #include "libfuncs/log.h"
  13. #include "libfuncs/list.h"
  14. #include "libfuncs/asyncdns.h"
  15. #include "libtsfuncs/tsfuncs.h"
  16. #include "data.h"
  17. #include "config.h"
  18. #include "output.h"
  19. extern CONFIG *config;
  20. channel_source get_sproto(char *url) {
  21. return strncmp(url, "http", 4)==0 ? tcp_sock : udp_sock;
  22. }
  23. CHANSRC *chansrc_init(char *url) {
  24. if (!url)
  25. return NULL;
  26. regex_t re;
  27. regmatch_t res[5];
  28. regcomp(&re, "^([a-z]+)://([^:/?]+):?([0-9]*)/?(.*)", REG_EXTENDED);
  29. if (regexec(&re,url,5,res,0)==0) {
  30. char *data = strdup(url);
  31. char *proto, *host, *port, *path;
  32. int iport;
  33. proto= data+res[1].rm_so; data[res[1].rm_eo]=0;
  34. host = data+res[2].rm_so; data[res[2].rm_eo]=0;
  35. port = data+res[3].rm_so; data[res[3].rm_eo]=0;
  36. path = data+res[4].rm_so; data[res[4].rm_eo]=0;
  37. iport = atoi(port);
  38. /* Setup */
  39. CHANSRC *src = calloc(1, sizeof(CHANSRC));
  40. src->proto = strdup(proto);
  41. src->sproto= get_sproto(url);
  42. src->host = strdup(host);
  43. src->port = iport ? iport : 80;
  44. src->path = strdup(path);
  45. FREE(data);
  46. regfree(&re);
  47. return src;
  48. }
  49. regfree(&re);
  50. return NULL;
  51. }
  52. void chansrc_free(CHANSRC **purl) {
  53. CHANSRC *url = *purl;
  54. if (url) {
  55. FREE(url->proto);
  56. FREE(url->host);
  57. FREE(url->path);
  58. FREE(*purl);
  59. }
  60. };
  61. void chansrc_add(CHANNEL *c, char *src) {
  62. if (c->num_src >= MAX_CHANNEL_SOURCES-1)
  63. return;
  64. c->sources[c->num_src] = strdup(src);
  65. if (c->num_src == 0) /* Set default source to first one */
  66. c->source = c->sources[c->num_src];
  67. c->num_src++;
  68. }
  69. void chansrc_next(CHANNEL *c) {
  70. if (c->num_src <= 1)
  71. return;
  72. // uint8_t old_src = c->curr_src;
  73. c->curr_src++;
  74. if (c->curr_src >= MAX_CHANNEL_SOURCES-1 || c->sources[c->curr_src] == NULL)
  75. c->curr_src = 0;
  76. c->source = c->sources[c->curr_src];
  77. // LOGf("CHAN : Switch source | Channel: %s OldSrc: %d %s NewSrc: %d %s\n", c->name, old_src, c->sources[old_src], c->curr_src, c->source);
  78. }
  79. void chansrc_set(CHANNEL *c, uint8_t src_id) {
  80. if (src_id >= MAX_CHANNEL_SOURCES-1 || c->sources[src_id] == NULL)
  81. return;
  82. // uint8_t old_src = c->curr_src;
  83. c->curr_src = src_id;
  84. c->source = c->sources[c->curr_src];
  85. // LOGf("CHAN : Set source | Channel: %s OldSrc: %d %s NewSrc: %d %s\n", c->name, old_src, c->sources[old_src], c->curr_src, c->source);
  86. }
  87. CHANNEL *channel_new(int service_id, int is_radio, char *id, char *name, char *source) {
  88. CHANNEL *c = calloc(1, sizeof(CHANNEL));
  89. c->service_id = service_id;
  90. c->radio = is_radio;
  91. c->base_pid = service_id * 32; // The first pid is saved for PMT
  92. c->pmt_pid = c->base_pid; // The first pid is saved for PMT
  93. c->id = strdup(id);
  94. c->name = strdup(name);
  95. chansrc_add(c, source);
  96. return c;
  97. }
  98. void channel_free_epg(CHANNEL *c) {
  99. epg_free(&c->epg_now);
  100. epg_free(&c->epg_next);
  101. ts_eit_free(&c->eit_now);
  102. ts_eit_free(&c->eit_next);
  103. }
  104. void channel_free(CHANNEL **pc) {
  105. CHANNEL *c = *pc;
  106. if (c) {
  107. channel_free_epg(c);
  108. FREE(c->id);
  109. FREE(c->name);
  110. int i;
  111. for (i=c->num_src-1; i>=0; i--) {
  112. FREE(c->sources[i]);
  113. }
  114. c->source = NULL;
  115. FREE(*pc);
  116. }
  117. }
  118. EPG_ENTRY *epg_new(time_t start, int duration, char *encoding, char *event, char *short_desc, char *long_desc) {
  119. EPG_ENTRY *e;
  120. if (!event)
  121. return NULL;
  122. e = calloc(1, sizeof(EPG_ENTRY));
  123. e->event_id = (start / 60) &~ 0xffff0000;
  124. e->start = start;
  125. e->duration = duration;
  126. if (encoding && strcmp(encoding, "iso-8859-5")==0) {
  127. e->event = init_dvb_string_iso_8859_5(event);
  128. e->short_desc = init_dvb_string_iso_8859_5(short_desc);
  129. e->long_desc = init_dvb_string_iso_8859_5(long_desc);
  130. } else { // Default is utf-8
  131. e->event = init_dvb_string_utf8(event);
  132. e->short_desc = init_dvb_string_utf8(short_desc);
  133. e->long_desc = init_dvb_string_utf8(long_desc);
  134. }
  135. return e;
  136. }
  137. void epg_free(EPG_ENTRY **pe) {
  138. EPG_ENTRY *e = *pe;
  139. if (e) {
  140. FREE(e->event);
  141. FREE(e->short_desc);
  142. FREE(e->long_desc);
  143. FREE(*pe);
  144. }
  145. }
  146. // Return 1 if they are different
  147. // Return 0 if they are the same
  148. int epg_changed(EPG_ENTRY *a, EPG_ENTRY *b) {
  149. if (!a && b) return 1;
  150. if (!b && a) return 1;
  151. if (!a && !b) return 0;
  152. if (a->event_id != b->event_id) return 1;
  153. if (a->start != b->start) return 1;
  154. if (a->duration != b->duration) return 1;
  155. if (xstrcmp(a->event, b->event) != 0) return 1;
  156. if (xstrcmp(a->short_desc, b->short_desc) != 0) return 1;
  157. if (xstrcmp(a->long_desc, b->long_desc) != 0) return 1;
  158. return 0;
  159. }
  160. void input_stream_alloc(INPUT *input) {
  161. input->stream.pidref = pidref_init(64, input->channel->base_pid);
  162. input->stream.pat = ts_pat_alloc();
  163. input->stream.pmt = ts_pmt_alloc();
  164. input->stream.last_pat = ts_pat_alloc();
  165. input->stream.last_pmt = ts_pmt_alloc();
  166. }
  167. void input_stream_free(INPUT *input) {
  168. ts_pmt_free(&input->stream.pmt);
  169. ts_pmt_free(&input->stream.pmt_rewritten);
  170. ts_pmt_free(&input->stream.last_pmt);
  171. ts_pat_free(&input->stream.pat);
  172. ts_pat_free(&input->stream.pat_rewritten);
  173. ts_pat_free(&input->stream.last_pat);
  174. pidref_free(&input->stream.pidref);
  175. input->stream.nit_pid = 0;
  176. input->stream.pmt_pid = 0;
  177. input->stream.pcr_pid = 0;
  178. input->stream.input_pcr = 0;
  179. }
  180. void input_stream_reset(INPUT *input) {
  181. input_stream_free(input);
  182. input_stream_alloc(input);
  183. }
  184. INPUT * input_new(const char *name, CHANNEL *channel) {
  185. char *tmp;
  186. INPUT *r = calloc(1, sizeof(INPUT));
  187. r->name = strdup(name);
  188. r->sock = -1;
  189. r->channel = channel;
  190. if (config->write_input_file) {
  191. asprintf(&tmp, "mptsd-input-%s.ts", channel->id);
  192. r->ifd = open(tmp, O_CREAT | O_WRONLY | O_TRUNC, 0644);
  193. FREE(tmp);
  194. }
  195. r->buf = cbuf_init(1428 * 1316, channel->id); // ~ 10000 x 188
  196. input_stream_alloc(r);
  197. return r;
  198. }
  199. void input_free(INPUT **pinput) {
  200. INPUT *r = *pinput;
  201. if (!r)
  202. return;
  203. if (r->sock > -1)
  204. shutdown_fd(&(r->sock));
  205. if (r->freechannel)
  206. channel_free(&r->channel);
  207. if (r->ifd)
  208. close(r->ifd);
  209. input_stream_free(r);
  210. cbuf_free(&r->buf);
  211. FREE(r->name);
  212. FREE(*pinput);
  213. }
  214. OUTPUT *output_new() {
  215. OUTPUT *o = calloc(1, sizeof(OUTPUT));
  216. o->obuf_ms = 100;
  217. o->psibuf = cbuf_init(50 * 1316, "psi");
  218. if (!o->psibuf) {
  219. LOGf("ERROR: Can't allocate PSI input buffer\n");
  220. exit(1);
  221. }
  222. cbuf_poison(o->psibuf, 'Y');
  223. return o;
  224. }
  225. void output_open_file(OUTPUT *o) {
  226. o->ofd = open("mptsd-output.ts", O_CREAT | O_WRONLY | O_TRUNC, 0644);
  227. }
  228. void obuf_reset(OBUF *ob) {
  229. int i;
  230. memset(ob->buf, 0xff, ob->size);
  231. for (i=0; i<ob->size; i+=188) {
  232. ob->buf[i+0] = 0x47;
  233. ob->buf[i+1] = 0x1f;
  234. ob->buf[i+2] = 0xff;
  235. ob->buf[i+3] = 0x00;
  236. }
  237. ob->written = 0;
  238. ob->status = obuf_empty;
  239. }
  240. void output_buffer_alloc(OUTPUT *o, double output_bitrate) {
  241. if (!output_bitrate) {
  242. LOGf("No output bitrate, can't determine buffer!\n");
  243. exit(1);
  244. }
  245. o->output_bitrate = output_bitrate;
  246. long pps = ceil((double)output_bitrate / (FRAME_PACKET_SIZE * 8)); // Packets per second
  247. long ppms = ceil((double)pps / ((double)1000 / o->obuf_ms)); // Packets per o->buffer_ms miliseconds
  248. long obuf_size = ppms * 1316;
  249. o->obuf[0].size = obuf_size;
  250. o->obuf[0].status = obuf_empty;
  251. o->obuf[0].buf = malloc(o->obuf[0].size);
  252. obuf_reset(&o->obuf[0]);
  253. o->obuf[1].size = obuf_size;
  254. o->obuf[1].status = obuf_empty;
  255. o->obuf[1].buf = malloc(o->obuf[0].size);
  256. obuf_reset(&o->obuf[1]);
  257. LOGf("\tOutput buf size : %ld * 2 = %ld\n", obuf_size, obuf_size * 2);
  258. LOGf("\tOutput buf packets: %ld (188 bytes)\n", obuf_size / 188);
  259. LOGf("\tOutput buf frames : %ld (1316 bytes)\n", obuf_size / 1316);
  260. LOGf("\tOutput buf ms : %u ms\n", o->obuf_ms);
  261. }
  262. void output_free(OUTPUT **po) {
  263. OUTPUT *o = *po;
  264. if (!o)
  265. return;
  266. if (o->out_sock > -1)
  267. shutdown_fd(&(o->out_sock));
  268. if (o->ofd)
  269. close(o->ofd);
  270. cbuf_free(&o->psibuf);
  271. FREE(o->obuf[0].buf);
  272. FREE(o->obuf[1].buf);
  273. output_psi_free(o);
  274. FREE(*po);
  275. }
  276. NIT *nit_new(uint16_t ts_id, char *freq, char *mod, char *symbol_rate) {
  277. char tmp[9];
  278. unsigned i, pos;
  279. NIT *n = calloc(1, sizeof(NIT));
  280. if (strlen(freq) != 9 || strlen(symbol_rate) != 8)
  281. return NULL;
  282. n->freq = strdup(freq);
  283. n->modulation = strdup(mod);
  284. n->symbol_rate = strdup(symbol_rate);
  285. n->ts_id = ts_id;
  286. n->_modulation =
  287. strcmp(mod, "16-QAM") == 0 ? 0x01 :
  288. strcmp(mod, "32-QAM") == 0 ? 0x02 :
  289. strcmp(mod, "64-QAM") == 0 ? 0x03 :
  290. strcmp(mod, "128-QAM") == 0 ? 0x04 :
  291. strcmp(mod, "256-QAM") == 0 ? 0x05 : 0x00;
  292. memset(tmp, 0, sizeof(tmp));
  293. pos = 0;
  294. for (i=0;i<strlen(freq);i++) {
  295. if (isdigit(freq[i])) {
  296. tmp[pos] = freq[i];
  297. pos++;
  298. }
  299. }
  300. n->_freq = strtol(tmp, NULL, 16);
  301. memset(tmp, 0, sizeof(tmp));
  302. pos = 0;
  303. for (i=0;i<strlen(symbol_rate);i++) {
  304. if (isdigit(symbol_rate[i])) {
  305. tmp[pos] = symbol_rate[i];
  306. pos++;
  307. }
  308. }
  309. n->_symbol_rate = strtol(tmp, NULL, 16);
  310. return n;
  311. }
  312. void nit_free(NIT **pn) {
  313. NIT *n = *pn;
  314. if (n) {
  315. FREE(n->freq);
  316. FREE(n->modulation);
  317. FREE(n->symbol_rate);
  318. FREE(*pn);
  319. }
  320. }
  321. void proxy_log(INPUT *r, char *msg) {
  322. LOGf("INPUT : [%-12s] %s fd: %d src: %s\n", r->channel->id, msg, r->sock, r->channel->source);
  323. }
  324. void proxy_close(LIST *inputs, INPUT **input) {
  325. proxy_log(*input, "Stop");
  326. // If there are no clients left, no "Timeout" messages will be logged
  327. list_del_entry(inputs, *input);
  328. input_free(input);
  329. }