Mercurial > mplayer.hg
annotate libmpdemux/dvbin.c @ 11882:3d605c60ba8e
Sync
author | rtognimp |
---|---|
date | Thu, 29 Jan 2004 21:24:17 +0000 |
parents | d158978a3d3c |
children | 6fbcdac049c6 |
rev | line source |
---|---|
9610 | 1 /* |
2 | |
3 dvbstream | |
4 (C) Dave Chapman <dave@dchapman.com> 2001, 2002. | |
5 | |
6 The latest version can be found at http://www.linuxstb.org/dvbstream | |
7 | |
8 Copyright notice: | |
9 | |
10 This program is free software; you can redistribute it and/or modify | |
11 it under the terms of the GNU General Public License as published by | |
12 the Free Software Foundation; either version 2 of the License, or | |
13 (at your option) any later version. | |
14 | |
15 This program is distributed in the hope that it will be useful, | |
16 but WITHOUT ANY WARRANTY; without even the implied warranty of | |
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
18 GNU General Public License for more details. | |
19 | |
20 You should have received a copy of the GNU General Public License | |
21 along with this program; if not, write to the Free Software | |
22 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | |
23 | |
24 */ | |
25 | |
10560 | 26 #include "config.h" |
9610 | 27 #include <stdio.h> |
28 #include <stdlib.h> | |
29 #include <string.h> | |
30 #include <ctype.h> | |
31 #include <sys/ioctl.h> | |
32 #include <sys/time.h> | |
33 #include <sys/poll.h> | |
10560 | 34 #include <unistd.h> |
9610 | 35 #include <fcntl.h> |
36 #include <string.h> | |
37 #include <errno.h> | |
38 #include <fcntl.h> | |
39 | |
40 #include "stream.h" | |
41 #include "demuxer.h" | |
10560 | 42 #include "help_mp.h" |
43 #include "../m_option.h" | |
44 #include "../m_struct.h" | |
9610 | 45 |
46 #include "dvbin.h" | |
47 | |
48 | |
49 #define MAX_CHANNELS 8 | |
10560 | 50 #define CHANNEL_LINE_LEN 256 |
9610 | 51 #define min(a, b) ((a) <= (b) ? (a) : (b)) |
52 | |
10560 | 53 |
54 //TODO: CAMBIARE list_ptr e da globale a per_priv | |
55 | |
56 | |
57 static struct stream_priv_s | |
58 { | |
59 char *prog; | |
60 int card; | |
61 char *type; | |
62 int vid, aid; | |
63 char *file; | |
64 } | |
65 stream_defaults = | |
66 { | |
67 "", 1, "", 0, 0, "channels.conf" | |
68 }; | |
69 | |
70 #define ST_OFF(f) M_ST_OFF(struct stream_priv_s, f) | |
9610 | 71 |
10560 | 72 /// URL definition |
73 static m_option_t stream_params[] = { | |
74 {"prog", ST_OFF(prog), CONF_TYPE_STRING, 0, 0 ,0, NULL}, | |
75 {"card", ST_OFF(card), CONF_TYPE_INT, M_OPT_RANGE, 1, 4, NULL}, | |
76 {"type", ST_OFF(type), CONF_TYPE_STRING, 0, 0 ,0, NULL}, | |
77 {"vid", ST_OFF(vid), CONF_TYPE_INT, 0, 0 ,0, NULL}, | |
78 {"aid", ST_OFF(aid), CONF_TYPE_INT, 0, 0 ,0, NULL}, | |
79 {"file", ST_OFF(file), CONF_TYPE_STRING, 0, 0 ,0, NULL}, | |
80 | |
81 {"hostname", ST_OFF(prog), CONF_TYPE_STRING, 0, 0, 0, NULL }, | |
82 {"filename", ST_OFF(card), CONF_TYPE_INT, M_OPT_RANGE, 1, 4, NULL}, | |
83 {NULL, NULL, 0, 0, 0, 0, NULL} | |
84 }; | |
85 | |
86 static struct m_struct_st stream_opts = { | |
87 "dvbin", | |
88 sizeof(struct stream_priv_s), | |
89 &stream_defaults, | |
90 stream_params | |
9610 | 91 }; |
92 | |
93 | |
10560 | 94 |
95 m_option_t dvbin_opts_conf[] = { | |
96 {"prog", &stream_defaults.prog, CONF_TYPE_STRING, 0, 0 ,0, NULL}, | |
97 {"card", &stream_defaults.card, CONF_TYPE_INT, M_OPT_RANGE, 1, 4, NULL}, | |
98 {"type", &stream_defaults.type, CONF_TYPE_STRING, 0, 0 ,0, NULL}, | |
99 {"vid", &stream_defaults.vid, CONF_TYPE_INT, 0, 0 ,0, NULL}, | |
100 {"aid", &stream_defaults.aid, CONF_TYPE_INT, 0, 0 ,0, NULL}, | |
101 {"file", &stream_defaults.file, CONF_TYPE_STRING, 0, 0 ,0, NULL}, | |
9610 | 102 |
10560 | 103 {NULL, NULL, 0, 0, 0, 0, NULL} |
104 }; | |
105 | |
106 | |
107 | |
108 | |
109 extern int dvb_set_ts_filt(int fd, uint16_t pid, dmx_pes_type_t pestype); | |
110 extern int dvb_demux_stop(int fd); | |
111 extern int dvb_get_tuner_type(dvb_priv_t *priv); | |
112 | |
9610 | 113 extern int dvb_tune(dvb_priv_t *priv, int freq, char pol, int srate, int diseqc, int tone, |
114 fe_spectral_inversion_t specInv, fe_modulation_t modulation, fe_guard_interval_t guardInterval, | |
115 fe_transmit_mode_t TransmissionMode, fe_bandwidth_t bandWidth, fe_code_rate_t HP_CodeRate); | |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
116 extern char *dvb_dvrdev[4], *dvb_demuxdev[4], *dvb_frontenddev[4]; |
10560 | 117 |
118 dvb_channels_list *dvb_list_ptr = NULL; | |
9610 | 119 |
120 | |
10560 | 121 static dvb_channels_list *dvb_get_channels(char *filename, int type) |
9610 | 122 { |
123 dvb_channels_list *list; | |
124 FILE *f; | |
10560 | 125 uint8_t line[CHANNEL_LINE_LEN]; |
126 | |
9610 | 127 int fields, row_count; |
128 dvb_channel_t *ptr; | |
129 char *tmp_lcr, *tmp_hier, *inv, *bw, *cr, *mod, *transm, *gi; | |
11352 | 130 const char *cbl_conf = "%a[^:]:%d:%a[^:]:%d:%a[^:]:%a[^:]:%d:%d\n"; |
9610 | 131 const char *sat_conf = "%a[^:]:%d:%c:%d:%d:%d:%d:%d:%d:%d\n"; |
132 const char *ter_conf = "%a[^:]:%d:%a[^:]:%a[^:]:%a[^:]:%a[^:]:%a[^:]:%a[^:]:%a[^:]:%a[^:]:%d:%d\n"; | |
133 | |
10560 | 134 if(type != TUNER_SAT && type != TUNER_TER && type != TUNER_CBL) |
135 { | |
136 mp_msg(MSGT_DEMUX, MSGL_V, "DVB_GET_CHANNELS: wrong tuner type, exit\n"); | |
137 return 0; | |
138 } | |
139 | |
9610 | 140 list = malloc(sizeof(dvb_channels_list)); |
141 if(list == NULL) | |
142 { | |
143 mp_msg(MSGT_DEMUX, MSGL_V, "DVB_GET_CHANNELS: couldn't malloc enough memory\n"); | |
144 return NULL; | |
145 } | |
146 | |
10560 | 147 bzero(list, sizeof(dvb_channels_list)); |
148 mp_msg(MSGT_DEMUX, MSGL_V, "CONFIG_READ FILE: %s, type: %d\n", filename, type); | |
9610 | 149 if((f=fopen(filename, "r"))==NULL) |
150 { | |
151 mp_msg(MSGT_DEMUX, MSGL_FATAL, "CAN'T READ CONFIG FILE %s\n", filename); | |
152 return NULL; | |
153 } | |
154 | |
155 list->NUM_CHANNELS = 0; | |
156 row_count = 0; | |
157 while(! feof(f) && row_count < 512) | |
158 { | |
10708 | 159 if( fgets(line, CHANNEL_LINE_LEN, f) == NULL ) |
160 continue; | |
9610 | 161 |
10708 | 162 if((line[0] == '#') || (strlen(line) == 0)) |
10560 | 163 continue; |
9610 | 164 |
10560 | 165 ptr = &(list->channels[list->NUM_CHANNELS]); |
9610 | 166 |
10560 | 167 if(type == TUNER_TER) |
9610 | 168 { |
169 fields = sscanf(line, ter_conf, | |
10560 | 170 &ptr->name, &ptr->freq, &inv, &bw, &cr, &tmp_lcr, &mod, |
9610 | 171 &transm, &gi, &tmp_hier, &ptr->vpid, &ptr->apid1); |
10708 | 172 /* |
173 mp_msg(MSGT_DEMUX, MSGL_V, | |
174 "NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d, VPID: %d, APID1: %d\n", | |
175 list->NUM_CHANNELS, fields, ptr->name, ptr->freq, ptr->vpid, ptr->apid1); | |
176 */ | |
10560 | 177 } |
178 else if(type == TUNER_CBL) | |
179 { | |
180 fields = sscanf(line, cbl_conf, | |
181 &ptr->name, &ptr->freq, &inv, &ptr->srate, | |
182 &cr, &mod, &ptr->vpid, &ptr->apid1); | |
10708 | 183 /* |
184 mp_msg(MSGT_DEMUX, MSGL_V, | |
185 "NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d, SRATE: %d, VPID: %d, APID1: %d\n", | |
186 list->NUM_CHANNELS, fields, ptr->name, ptr->freq, ptr->srate, ptr->vpid, ptr->apid1); | |
187 */ | |
10560 | 188 } |
189 else //SATELLITE | |
190 { | |
191 fields = sscanf(line, sat_conf, | |
192 &ptr->name, &ptr->freq, &ptr->pol, &ptr->diseqc, &ptr->srate, &ptr->vpid, &ptr->apid1, | |
193 &ptr->tpid, &ptr->ca, &ptr->progid); | |
194 ptr->pol = toupper(ptr->pol); | |
195 ptr->freq *= 1000UL; | |
196 ptr->srate *= 1000UL; | |
197 ptr->tone = -1; | |
198 mp_msg(MSGT_DEMUX, MSGL_V, | |
10708 | 199 "NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d, SRATE: %d, POL: %c, DISEQC: %d, TONE: %d, VPID: %d, APID1: %d, APID2: %d, TPID: %d, PROGID: %d\n", |
200 list->NUM_CHANNELS, fields, ptr->name, ptr->freq, ptr->srate, ptr->pol, ptr->diseqc, ptr->tone, ptr->vpid, ptr->apid1, ptr->apid2, ptr->tpid, ptr->progid); | |
10560 | 201 } |
9610 | 202 |
10560 | 203 |
10708 | 204 if(((ptr->vpid <= 0) && (ptr->apid1 <=0)) || (ptr->freq == 0)) |
205 continue; | |
206 | |
207 | |
10560 | 208 if((type == TUNER_TER) || (type == TUNER_CBL)) |
209 { | |
9610 | 210 if(! strcmp(inv, "INVERSION_ON")) |
211 ptr->inv = INVERSION_ON; | |
212 else if(! strcmp(inv, "INVERSION_OFF")) | |
213 ptr->inv = INVERSION_OFF; | |
214 else | |
215 ptr->inv = INVERSION_AUTO; | |
216 | |
217 | |
218 if(! strcmp(cr, "FEC_1_2")) | |
219 ptr->cr =FEC_1_2; | |
220 else if(! strcmp(cr, "FEC_2_3")) | |
221 ptr->cr =FEC_2_3; | |
222 else if(! strcmp(cr, "FEC_3_4")) | |
223 ptr->cr =FEC_3_4; | |
224 #ifdef HAVE_DVB_HEAD | |
225 else if(! strcmp(cr, "FEC_4_5")) | |
226 ptr->cr =FEC_4_5; | |
227 else if(! strcmp(cr, "FEC_6_7")) | |
228 ptr->cr =FEC_6_7; | |
229 else if(! strcmp(cr, "FEC_8_9")) | |
230 ptr->cr =FEC_8_9; | |
231 #endif | |
232 else if(! strcmp(cr, "FEC_5_6")) | |
233 ptr->cr =FEC_5_6; | |
234 else if(! strcmp(cr, "FEC_7_8")) | |
235 ptr->cr =FEC_7_8; | |
236 else if(! strcmp(cr, "FEC_NONE")) | |
237 ptr->cr =FEC_NONE; | |
238 else ptr->cr =FEC_AUTO; | |
239 | |
240 if(! strcmp(mod, "QAM_128")) | |
241 ptr->mod = QAM_128; | |
242 else if(! strcmp(mod, "QAM_256")) | |
243 ptr->mod = QAM_256; | |
244 else if(! strcmp(mod, "QAM_64")) | |
245 ptr->mod = QAM_64; | |
246 else if(! strcmp(mod, "QAM_32")) | |
247 ptr->mod = QAM_32; | |
248 else if(! strcmp(mod, "QAM_16")) | |
249 ptr->mod = QAM_16; | |
10560 | 250 //else ptr->mod = QPSK; |
251 } | |
252 | |
253 | |
254 if(type == TUNER_TER) | |
255 { | |
256 if(! strcmp(bw, "BANDWIDTH_6_MHZ")) | |
257 ptr->bw = BANDWIDTH_6_MHZ; | |
258 else if(! strcmp(bw, "BANDWIDTH_7_MHZ")) | |
259 ptr->bw = BANDWIDTH_7_MHZ; | |
260 else if(! strcmp(bw, "BANDWIDTH_8_MHZ")) | |
261 ptr->bw = BANDWIDTH_8_MHZ; | |
9610 | 262 |
263 | |
264 if(! strcmp(transm, "TRANSMISSION_MODE_2K")) | |
265 ptr->trans = TRANSMISSION_MODE_2K; | |
266 else if(! strcmp(transm, "TRANSMISSION_MODE_8K")) | |
267 ptr->trans = TRANSMISSION_MODE_8K; | |
268 | |
10560 | 269 |
9610 | 270 if(! strcmp(gi, "GUARD_INTERVAL_1_32")) |
271 ptr->gi = GUARD_INTERVAL_1_32; | |
272 else if(! strcmp(gi, "GUARD_INTERVAL_1_16")) | |
273 ptr->gi = GUARD_INTERVAL_1_16; | |
274 else if(! strcmp(gi, "GUARD_INTERVAL_1_8")) | |
275 ptr->gi = GUARD_INTERVAL_1_8; | |
276 else ptr->gi = GUARD_INTERVAL_1_4; | |
277 } | |
278 | |
279 list->NUM_CHANNELS++; | |
280 row_count++; | |
281 } | |
282 | |
283 fclose(f); | |
10560 | 284 list->current = 0; |
9610 | 285 return list; |
286 } | |
287 | |
288 | |
289 | |
10560 | 290 static int dvb_streaming_read(stream_t *stream, char *buffer, int size) |
9610 | 291 { |
292 struct pollfd pfds[1]; | |
10560 | 293 int pos=0, tries, rk; |
294 int fd = stream->fd; | |
295 dvb_priv_t *priv = (dvb_priv_t *) stream->priv; | |
9610 | 296 |
10560 | 297 mp_msg(MSGT_DEMUX, MSGL_V, "dvb_streaming_read(%d)\n", size); |
9610 | 298 |
11352 | 299 tries = priv->retry + 1; |
300 | |
9610 | 301 while(pos < size) |
302 { | |
10560 | 303 pfds[0].fd = fd; |
304 pfds[0].events = POLLIN | POLLPRI; | |
9610 | 305 |
10560 | 306 poll(pfds, 1, 500); |
307 rk = size - pos; | |
308 if((rk = read(fd, &buffer[pos], rk)) > 0) | |
309 { | |
310 pos += rk; | |
311 mp_msg(MSGT_DEMUX, MSGL_V, "ret (%d) bytes\n", pos); | |
312 } | |
313 else | |
314 { | |
315 mp_msg(MSGT_DEMUX, MSGL_ERR, "dvb_streaming_read, attempt N. %d failed with errno %d when reading %d bytes\n", tries, errno, size-pos); | |
316 if(--tries > 0) | |
317 { | |
318 errno = 0; | |
319 //reset_demuxers(priv); | |
320 continue; | |
321 } | |
322 else | |
323 { | |
324 errno = 0; | |
325 break; | |
326 } | |
327 } | |
328 } | |
9610 | 329 |
10560 | 330 if(! pos) |
331 mp_msg(MSGT_DEMUX, MSGL_ERR, "dvb_streaming_read, return %d bytes\n", pos); | |
9610 | 332 |
333 return pos; | |
334 } | |
335 | |
336 | |
10560 | 337 static int reset_demuxers(dvb_priv_t *priv) |
9610 | 338 { |
10560 | 339 dvb_channel_t *channel; |
340 dvb_channels_list *list = priv->list; | |
341 | |
342 channel = &(list->channels[list->current]); | |
343 | |
344 if(priv->is_on) //the fds are already open and we have to stop the demuxers | |
345 { | |
346 dvb_demux_stop(priv->demux_fd[0]); | |
347 dvb_demux_stop(priv->demux_fd[1]); | |
348 } | |
349 | |
350 if(channel->vpid) | |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
351 if(! dvb_set_ts_filt(priv->demux_fd[0], channel->vpid, DMX_PES_OTHER)) |
10560 | 352 return 0; |
353 | |
354 if(channel->apid1) | |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
355 if(! dvb_set_ts_filt(priv->demux_fd[1], channel->apid1, DMX_PES_OTHER)) |
10560 | 356 return 0; |
357 | |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
358 return 1; |
10560 | 359 } |
360 | |
361 | |
362 int dvb_set_channel(dvb_priv_t *priv, int n) | |
363 { | |
364 dvb_channels_list *list; | |
365 dvb_channel_t *channel; | |
366 int do_tuning; | |
367 stream_t *stream = (stream_t*) priv->stream; | |
368 char buf[4096]; | |
9610 | 369 |
10560 | 370 if(priv->is_on) //the fds are already open and we have to stop the demuxers |
371 { | |
372 dvb_demux_stop(priv->demux_fd[0]); | |
373 dvb_demux_stop(priv->demux_fd[1]); | |
374 priv->retry = 0; | |
11352 | 375 while(dvb_streaming_read(stream, buf, 4096) > 0); //empty both the stream's and driver's buffer |
10560 | 376 } |
377 | |
11352 | 378 priv->retry = 5; |
10560 | 379 mp_msg(MSGT_DEMUX, MSGL_V, "DVB_SET_CHANNEL: channel %d\n", n); |
380 list = priv->list; | |
381 if(list == NULL) | |
382 { | |
383 mp_msg(MSGT_DEMUX, MSGL_ERR, "dvb_set_channel: LIST NULL PTR, quit\n"); | |
384 return 0; | |
385 } | |
386 | |
387 if((n > list->NUM_CHANNELS) || (n < 0)) | |
388 { | |
389 mp_msg(MSGT_DEMUX, MSGL_ERR, "dvb_set_channel: INVALID CHANNEL NUMBER: %d, abort\n", n); | |
390 return 0; | |
391 } | |
9610 | 392 |
10560 | 393 list->current = n; |
394 channel = &(list->channels[list->current]); | |
395 mp_msg(MSGT_DEMUX, MSGL_V, "DVB_SET_CHANNEL: new channel name=%s\n", channel->name); | |
396 | |
397 switch(priv->tuner_type) | |
398 { | |
399 case TUNER_SAT: | |
400 sprintf(priv->new_tuning, "%d|%09d|%09d|%d|%c", priv->card, channel->freq, channel->srate, channel->diseqc, channel->pol); | |
401 break; | |
9610 | 402 |
10560 | 403 case TUNER_TER: |
404 sprintf(priv->new_tuning, "%d|%09d|%d|%d|%d|%d|%d|%d", priv->card, channel->freq, channel->inv, | |
405 channel->bw, channel->cr, channel->mod, channel->trans, channel->gi); | |
406 break; | |
407 | |
408 case TUNER_CBL: | |
409 sprintf(priv->new_tuning, "%d|%09d|%d|%d|%d|%d", priv->card, channel->freq, channel->inv, channel->srate, | |
410 channel->cr, channel->mod); | |
411 break; | |
412 } | |
413 | |
414 | |
9610 | 415 |
10560 | 416 if(strcmp(priv->prev_tuning, priv->new_tuning)) |
417 { | |
418 mp_msg(MSGT_DEMUX, MSGL_V, "DIFFERENT TUNING THAN THE PREVIOUS: %s -> %s\n", priv->prev_tuning, priv->new_tuning); | |
419 strcpy(priv->prev_tuning, priv->new_tuning); | |
420 do_tuning = 1; | |
421 } | |
422 else | |
423 { | |
11352 | 424 mp_msg(MSGT_DEMUX, MSGL_V, "SAME TUNING PARAMETERS, NO TUNING\n"); |
10560 | 425 do_tuning = 0; |
426 } | |
427 | |
428 stream->eof=1; | |
429 stream_reset(stream); | |
430 | |
9610 | 431 |
10560 | 432 if(do_tuning) |
433 if (! dvb_tune(priv, channel->freq, channel->pol, channel->srate, channel->diseqc, channel->tone, | |
434 channel->inv, channel->mod, channel->gi, channel->trans, channel->bw, channel->cr)) | |
435 return 0; | |
436 | |
437 | |
438 priv->is_on = 1; | |
439 | |
440 //sets demux filters and restart the stream | |
441 if(channel->vpid) | |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
442 if(! dvb_set_ts_filt(priv->demux_fd[0], channel->vpid, DMX_PES_OTHER)) |
10560 | 443 return 0; |
444 | |
445 if(channel->apid1) | |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
446 if(! dvb_set_ts_filt(priv->demux_fd[1], channel->apid1, DMX_PES_OTHER)) |
10560 | 447 return 0; |
448 | |
449 return 1; | |
450 } | |
451 | |
452 | |
9610 | 453 |
10560 | 454 int dvb_step_channel(dvb_priv_t *priv, int dir) |
455 { | |
456 int new_current; | |
457 dvb_channels_list *list; | |
458 | |
459 mp_msg(MSGT_DEMUX, MSGL_V, "DVB_STEP_CHANNEL dir %d\n", dir); | |
9610 | 460 |
10560 | 461 if(priv == NULL) |
462 { | |
463 mp_msg(MSGT_DEMUX, MSGL_ERR, "dvb_step_channel: NULL priv_ptr, quit\n"); | |
464 return 0; | |
465 } | |
9610 | 466 |
10560 | 467 list = priv->list; |
468 if(list == NULL) | |
469 { | |
470 mp_msg(MSGT_DEMUX, MSGL_ERR, "dvb_step_channel: NULL list_ptr, quit\n"); | |
471 return 0; | |
472 } | |
9610 | 473 |
10560 | 474 |
475 if(dir == DVB_CHANNEL_HIGHER) | |
476 { | |
477 if(list->current == list->NUM_CHANNELS-1) | |
478 return 0; | |
9610 | 479 |
10560 | 480 new_current = list->current + 1; |
481 } | |
482 else | |
483 { | |
484 if(list->current == 0) | |
485 return 0; | |
9610 | 486 |
10560 | 487 new_current = list->current - 1; |
488 } | |
489 | |
490 return dvb_set_channel(priv, new_current); | |
9610 | 491 } |
492 | |
493 | |
10560 | 494 |
495 | |
9610 | 496 extern char *get_path(char *); |
497 | |
10560 | 498 static void dvbin_close(stream_t *stream) |
9610 | 499 { |
10560 | 500 dvb_priv_t *priv = (dvb_priv_t *) stream->priv; |
9610 | 501 |
10560 | 502 close(priv->dvr_fd); |
503 close(priv->demux_fd[0]); | |
504 close(priv->demux_fd[1]); | |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
505 |
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
506 close(priv->fe_fd); |
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
507 #ifdef HAVE_DVB |
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
508 close(priv->sec_fd); |
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
509 #endif |
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
510 |
10560 | 511 priv->is_on = 0; |
512 priv->stream = NULL; | |
513 if(dvb_list_ptr) | |
514 free(dvb_list_ptr); | |
515 | |
516 dvb_list_ptr = NULL; | |
517 } | |
9610 | 518 |
519 | |
10560 | 520 static int dvb_streaming_start(dvb_priv_t *priv, struct stream_priv_s *opts, int tuner_type) |
521 { | |
522 int pids[MAX_CHANNELS], pestypes[MAX_CHANNELS], npids = 0, i; | |
523 dvb_channel_t *channel = NULL; | |
524 stream_t *stream = (stream_t*) priv->stream; | |
9610 | 525 |
526 | |
527 mp_msg(MSGT_DEMUX, MSGL_INFO, "code taken from dvbstream for mplayer v0.4pre1 - (C) Dave Chapman 2001\n"); | |
528 mp_msg(MSGT_DEMUX, MSGL_INFO, "Released under the GPL.\n"); | |
529 mp_msg(MSGT_DEMUX, MSGL_INFO, "Latest version available from http://www.linuxstb.org/\n"); | |
10560 | 530 mp_msg(MSGT_DEMUX, MSGL_V, "PROG: %s, CARD: %d, VID: %d, AID: %d, TYPE: %s, FILE: %s\n", |
531 opts->prog, opts->card, opts->vid, opts->aid, opts->type, opts->file); | |
9610 | 532 |
10560 | 533 priv->is_on = 0; |
9610 | 534 |
10560 | 535 if(strlen(opts->prog)) |
9610 | 536 { |
10560 | 537 if(dvb_list_ptr != NULL) |
538 { | |
539 i = 0; | |
540 while((channel == NULL) && i < dvb_list_ptr->NUM_CHANNELS) | |
541 { | |
542 if(! strcmp(dvb_list_ptr->channels[i].name, opts->prog)) | |
543 channel = &(dvb_list_ptr->channels[i]); | |
9610 | 544 |
10560 | 545 i++; |
546 } | |
9610 | 547 |
10560 | 548 if(channel != NULL) |
549 { | |
550 dvb_list_ptr->current = i-1; | |
551 mp_msg(MSGT_DEMUX, MSGL_V, "PROGRAM NUMBER %d: name=%s, vid=%d, aid=%d, freq=%lu, srate=%lu, pol=%c, diseqc: %d, tone: %d\n", i-1, | |
552 channel->name, channel->vpid, channel->apid1, | |
553 channel->freq, channel->srate, channel->pol, channel->diseqc, channel->tone); | |
554 } | |
555 else if(opts->prog) | |
556 { | |
557 mp_msg(MSGT_DEMUX, MSGL_ERR, "\n\nDVBIN: no such channel \"%s\"\n\n", opts->prog); | |
558 return 0; | |
559 } | |
560 } | |
561 else | |
562 { | |
563 mp_msg(MSGT_DEMUX, MSGL_ERR, "DVBIN: chanel %s requested, but no channel list supplied %s\n", opts->prog); | |
564 return 0; | |
565 } | |
9610 | 566 } |
567 | |
568 | |
10560 | 569 |
570 if(opts->vid > 0) | |
9610 | 571 { |
10560 | 572 pids[npids] = opts->vid; |
9610 | 573 } |
574 else if(channel != NULL) | |
575 { | |
10560 | 576 pids[npids] = channel->vpid; |
9610 | 577 } |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
578 pestypes[npids] = DMX_PES_OTHER; |
9610 | 579 npids++; |
580 | |
10560 | 581 if(opts->aid > 0) |
9610 | 582 { |
10560 | 583 pids[npids] = opts->aid; |
9610 | 584 } |
585 else if(channel != NULL) | |
586 { | |
10560 | 587 pids[npids] = channel->apid1; |
9610 | 588 } |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
589 pestypes[npids] = DMX_PES_OTHER; |
9610 | 590 npids++; |
591 | |
592 | |
593 | |
10560 | 594 priv->demux_fd[0] = open(dvb_demuxdev[priv->card], O_RDWR); |
595 if(priv->demux_fd[0] < 0) | |
9610 | 596 { |
10560 | 597 mp_msg(MSGT_DEMUX, MSGL_ERR, "ERROR OPENING DEMUX 0: %d\n", errno); |
598 return 0; | |
9610 | 599 } |
600 | |
10560 | 601 priv->demux_fd[1] = open(dvb_demuxdev[priv->card], O_RDWR); |
602 if(priv->demux_fd[1] < 0) | |
9610 | 603 { |
10560 | 604 mp_msg(MSGT_DEMUX, MSGL_ERR, "ERROR OPENING DEMUX 1: %d\n", errno); |
605 return 0; | |
9610 | 606 } |
607 | |
608 | |
10560 | 609 priv->dvr_fd = open(dvb_dvrdev[priv->card], O_RDONLY| O_NONBLOCK); |
610 if(priv->dvr_fd < 0) | |
9610 | 611 { |
10560 | 612 mp_msg(MSGT_DEMUX, MSGL_ERR, "ERROR OPENING DVR DEVICE %s: %d\n", dvb_dvrdev[priv->card], errno); |
613 return 0; | |
9610 | 614 } |
615 | |
616 | |
10560 | 617 strcpy(priv->prev_tuning, ""); |
618 if(!dvb_set_channel(priv, dvb_list_ptr->current)) | |
619 { | |
620 mp_msg(MSGT_DEMUX, MSGL_ERR, "ERROR, COULDN'T SET CHANNEL %i: ", dvb_list_ptr->current); | |
621 dvbin_close(stream); | |
622 return 0; | |
623 } | |
624 | |
9610 | 625 stream->fd = priv->dvr_fd; |
626 | |
10560 | 627 mp_msg(MSGT_DEMUX, MSGL_V, "SUCCESSFUL EXIT from dvb_streaming_start\n"); |
9610 | 628 |
629 return 1; | |
630 } | |
631 | |
632 | |
10560 | 633 |
634 | |
635 static int dvb_open(stream_t *stream, int mode, void *opts, int *file_format) | |
9610 | 636 { |
10560 | 637 // I don't force the file format bacause, although it's almost always TS, |
638 // there are some providers that stream an IP multicast with M$ Mpeg4 inside | |
639 struct stream_priv_s* p = (struct stream_priv_s*)opts; | |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
640 char *filename; |
10560 | 641 dvb_priv_t *priv; |
11352 | 642 int tuner_type = 0; |
10560 | 643 |
644 | |
645 if(mode != STREAM_READ) | |
646 return STREAM_UNSUPORTED; | |
647 | |
648 stream->priv = (dvb_priv_t*) malloc(sizeof(dvb_priv_t)); | |
649 if(stream->priv == NULL) | |
650 return STREAM_ERROR; | |
651 | |
652 priv = (dvb_priv_t *)stream->priv; | |
653 priv->stream = stream; | |
654 | |
655 priv->card = p->card - 1; | |
11872
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
656 if(! dvb_open_fe(priv)) |
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
657 { |
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
658 mp_msg(MSGT_DEMUX, MSGL_ERR, "ERROR OPENING FRONTEND DEVICE %s: %d\n", dvb_frontenddev[priv->card], errno); |
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
659 return STREAM_ERROR; |
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
660 } |
d158978a3d3c
Compliance with the DVB power management specification (doesn't close
attila
parents:
11352
diff
changeset
|
661 |
10560 | 662 if(!strncmp(p->type, "CBL", 3)) |
663 { | |
664 tuner_type = TUNER_CBL; | |
665 } | |
666 else if(!strncmp(p->type, "TER", 3)) | |
667 { | |
668 tuner_type = TUNER_TER; | |
669 } | |
670 else if(!strncmp(p->type, "SAT", 3)) | |
671 { | |
672 tuner_type = TUNER_SAT; | |
673 } | |
674 else | |
675 { | |
676 int t = dvb_get_tuner_type(priv); | |
677 | |
678 if((t==TUNER_SAT) || (t==TUNER_TER) || (t==TUNER_CBL)) | |
679 { | |
680 tuner_type = t; | |
681 } | |
682 } | |
683 | |
11352 | 684 if(tuner_type == 0) |
685 { | |
686 mp_msg(MSGT_DEMUX, MSGL_V, "OPEN_DVB: UNKNOWN OR UNDETECTABLE TUNER TYPE, EXIT\n"); | |
687 return STREAM_ERROR; | |
688 } | |
689 | |
690 | |
10560 | 691 priv->tuner_type = tuner_type; |
692 | |
693 mp_msg(MSGT_DEMUX, MSGL_V, "OPEN_DVB: prog=%s, card=%d, type=%d, vid=%d, aid=%d, file=%s\n", | |
694 p->prog, priv->card+1, priv->tuner_type, p->vid, p->aid, p->file); | |
695 | |
696 if(dvb_list_ptr == NULL) | |
697 { | |
698 filename = get_path(p->file); | |
699 if(filename) | |
700 { | |
701 if((dvb_list_ptr = dvb_get_channels(filename, tuner_type)) == NULL) | |
702 mp_msg(MSGT_DEMUX, MSGL_ERR, "EMPTY CHANNELS LIST FROM FILE %s!\n", filename); | |
11352 | 703 priv->list = dvb_list_ptr; |
10560 | 704 } |
705 else | |
706 { | |
11352 | 707 priv->list = dvb_list_ptr = NULL; |
10560 | 708 mp_msg(MSGT_DEMUX, MSGL_WARN, "NO CHANNELS FILE FOUND!\n"); |
709 } | |
710 } | |
711 else | |
712 priv->list = dvb_list_ptr; | |
713 | |
714 | |
11352 | 715 if(priv->list == NULL) |
716 { | |
717 mp_msg(MSGT_DEMUX, MSGL_ERR, "NO CHANNELS AVAILABLE, EXIT!\n"); | |
718 return STREAM_ERROR; | |
719 } | |
720 | |
10560 | 721 if(! strcmp(p->prog, "")) |
722 { | |
723 if(dvb_list_ptr != NULL) | |
724 { | |
725 dvb_channel_t *channel; | |
726 | |
727 channel = &(dvb_list_ptr->channels[dvb_list_ptr->current]); | |
728 p->prog = channel->name; | |
729 } | |
730 } | |
731 | |
732 | |
733 if(! dvb_streaming_start(priv, p, tuner_type)) | |
734 { | |
735 free(stream->priv); | |
736 stream->priv = NULL; | |
737 return STREAM_ERROR; | |
738 } | |
739 | |
740 stream->type = STREAMTYPE_DVB; | |
741 stream->fill_buffer = dvb_streaming_read; | |
742 stream->close = dvbin_close; | |
743 m_struct_free(&stream_opts, opts); | |
744 | |
745 return STREAM_OK; | |
9610 | 746 } |
10560 | 747 |
748 | |
749 | |
750 stream_info_t stream_info_dvb = { | |
751 "Dvb Input", | |
752 "dvbin", | |
753 "Nico", | |
754 "based on the code from ??? (probably Arpi)", | |
755 dvb_open, | |
756 { "dvb", NULL }, | |
757 &stream_opts, | |
758 1 // Urls are an option string | |
759 }; | |
760 | |
761 | |
762 | |
763 |