4 (C) Dave Chapman <dave@dchapman.com> 2001, 2002.
6 The latest version can be found at http://www.linuxstb.org/dvbstream
8 Modified for use with MPlayer, for details see the changelog at
9 http://svn.mplayerhq.hu/mplayer/trunk/
14 This program is free software; you can redistribute it and/or modify
15 it under the terms of the GNU General Public License as published by
16 the Free Software Foundation; either version 2 of the License, or
17 (at your option) any later version.
19 This program is distributed in the hope that it will be useful,
20 but WITHOUT ANY WARRANTY; without even the implied warranty of
21 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22 GNU General Public License for more details.
24 You should have received a copy of the GNU General Public License
25 along with this program; if not, write to the Free Software
26 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
35 #include <sys/ioctl.h>
43 #include "libmpdemux/demuxer.h"
48 #include "libavutil/avstring.h"
53 #define MAX_CHANNELS 8
54 #define CHANNEL_LINE_LEN 256
55 #define min(a, b) ((a) <= (b) ? (a) : (b))
58 //TODO: CAMBIARE list_ptr e da globale a per_priv
61 static struct stream_priv_s
73 #define ST_OFF(f) M_ST_OFF(struct stream_priv_s, f)
76 static const m_option_t stream_params
[] = {
77 {"prog", ST_OFF(prog
), CONF_TYPE_STRING
, 0, 0 ,0, NULL
},
78 {"card", ST_OFF(card
), CONF_TYPE_INT
, M_OPT_RANGE
, 1, 4, NULL
},
79 {"timeout",ST_OFF(timeout
), CONF_TYPE_INT
, M_OPT_RANGE
, 1, 30, NULL
},
80 {"file", ST_OFF(file
), CONF_TYPE_STRING
, 0, 0 ,0, NULL
},
82 {"hostname", ST_OFF(prog
), CONF_TYPE_STRING
, 0, 0, 0, NULL
},
83 {"username", ST_OFF(card
), CONF_TYPE_INT
, M_OPT_RANGE
, 1, 4, NULL
},
84 {NULL
, NULL
, 0, 0, 0, 0, NULL
}
87 static const struct m_struct_st stream_opts
= {
89 sizeof(struct stream_priv_s
),
96 const m_option_t dvbin_opts_conf
[] = {
97 {"prog", &stream_defaults
.prog
, CONF_TYPE_STRING
, 0, 0 ,0, NULL
},
98 {"card", &stream_defaults
.card
, CONF_TYPE_INT
, M_OPT_RANGE
, 1, 4, NULL
},
99 {"timeout", &stream_defaults
.timeout
, CONF_TYPE_INT
, M_OPT_RANGE
, 1, 30, NULL
},
100 {"file", &stream_defaults
.file
, CONF_TYPE_STRING
, 0, 0 ,0, NULL
},
102 {NULL
, NULL
, 0, 0, 0, 0, NULL
}
108 int dvb_set_ts_filt(int fd
, uint16_t pid
, dmx_pes_type_t pestype
);
109 int dvb_demux_stop(int fd
);
110 int dvb_get_tuner_type(int fd
);
111 int dvb_open_devices(dvb_priv_t
*priv
, int n
, int demux_cnt
);
112 int dvb_fix_demuxes(dvb_priv_t
*priv
, int cnt
);
114 int dvb_tune(dvb_priv_t
*priv
, int freq
, char pol
, int srate
, int diseqc
, int tone
,
115 fe_spectral_inversion_t specInv
, fe_modulation_t modulation
, fe_guard_interval_t guardInterval
,
116 fe_transmit_mode_t TransmissionMode
, fe_bandwidth_t bandWidth
, fe_code_rate_t HP_CodeRate
,
117 fe_code_rate_t LP_CodeRate
, fe_hierarchy_t hier
, int timeout
);
121 static dvb_channels_list
*dvb_get_channels(char *filename
, int type
)
123 dvb_channels_list
*list
;
125 char line
[CHANNEL_LINE_LEN
], *colon
;
127 int fields
, cnt
, pcnt
, k
;
129 dvb_channel_t
*ptr
, *tmp
, chn
;
130 char tmp_lcr
[256], tmp_hier
[256], inv
[256], bw
[256], cr
[256], mod
[256], transm
[256], gi
[256], vpid_str
[256], apid_str
[256];
131 const char *cbl_conf
= "%d:%255[^:]:%d:%255[^:]:%255[^:]:%255[^:]:%255[^:]\n";
132 const char *sat_conf
= "%d:%c:%d:%d:%255[^:]:%255[^:]\n";
133 const char *ter_conf
= "%d:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]\n";
134 const char *atsc_conf
= "%d:%255[^:]:%255[^:]:%255[^:]\n";
136 mp_msg(MSGT_DEMUX
, MSGL_V
, "CONFIG_READ FILE: %s, type: %d\n", filename
, type
);
137 if((f
=fopen(filename
, "r"))==NULL
)
139 mp_msg(MSGT_DEMUX
, MSGL_FATAL
, "CAN'T READ CONFIG FILE %s\n", filename
);
143 list
= malloc(sizeof(dvb_channels_list
));
147 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_GET_CHANNELS: couldn't malloc enough memory\n");
152 list
->NUM_CHANNELS
= 0;
153 list
->channels
= NULL
;
156 if( fgets(line
, CHANNEL_LINE_LEN
, f
) == NULL
)
159 if((line
[0] == '#') || (strlen(line
) == 0))
162 colon
= strchr(line
, ':');
168 ptr
->name
= malloc(k
+1);
171 av_strlcpy(ptr
->name
, line
, k
+1);
176 apid_str
[0] = vpid_str
[0] = 0;
179 if(type
== TUNER_TER
)
181 fields
= sscanf(&line
[k
], ter_conf
,
182 &ptr
->freq
, inv
, bw
, cr
, tmp_lcr
, mod
,
183 transm
, gi
, tmp_hier
, vpid_str
, apid_str
);
184 mp_msg(MSGT_DEMUX
, MSGL_V
,
185 "TER, NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d",
186 list
->NUM_CHANNELS
, fields
, ptr
->name
, ptr
->freq
);
188 else if(type
== TUNER_CBL
)
190 fields
= sscanf(&line
[k
], cbl_conf
,
191 &ptr
->freq
, inv
, &ptr
->srate
,
192 cr
, mod
, vpid_str
, apid_str
);
193 mp_msg(MSGT_DEMUX
, MSGL_V
,
194 "CBL, NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d, SRATE: %d",
195 list
->NUM_CHANNELS
, fields
, ptr
->name
, ptr
->freq
, ptr
->srate
);
198 else if(type
== TUNER_ATSC
)
200 fields
= sscanf(&line
[k
], atsc_conf
,
201 &ptr
->freq
, mod
, vpid_str
, apid_str
);
202 mp_msg(MSGT_DEMUX
, MSGL_V
,
203 "ATSC, NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d\n",
204 list
->NUM_CHANNELS
, fields
, ptr
->name
, ptr
->freq
);
209 fields
= sscanf(&line
[k
], sat_conf
,
210 &ptr
->freq
, &ptr
->pol
, &ptr
->diseqc
, &ptr
->srate
, vpid_str
, apid_str
);
211 ptr
->pol
= toupper(ptr
->pol
);
213 ptr
->srate
*= 1000UL;
215 ptr
->inv
= INVERSION_AUTO
;
217 if((ptr
->diseqc
> 4) || (ptr
->diseqc
< 0))
221 mp_msg(MSGT_DEMUX
, MSGL_V
,
222 "SAT, NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d, SRATE: %d, POL: %c, DISEQC: %d",
223 list
->NUM_CHANNELS
, fields
, ptr
->name
, ptr
->freq
, ptr
->srate
, ptr
->pol
, ptr
->diseqc
);
228 pcnt
= sscanf(vpid_str
, "%d+%d+%d+%d+%d+%d+%d", &ptr
->pids
[0], &ptr
->pids
[1], &ptr
->pids
[2], &ptr
->pids
[3],
229 &ptr
->pids
[4], &ptr
->pids
[5], &ptr
->pids
[6]);
232 ptr
->pids_cnt
= pcnt
;
240 pcnt
= sscanf(apid_str
, "%d+%d+%d+%d+%d+%d+%d+%d", &ptr
->pids
[cnt
], &ptr
->pids
[cnt
+1], &ptr
->pids
[cnt
+2],
241 &ptr
->pids
[cnt
+3], &ptr
->pids
[cnt
+4], &ptr
->pids
[cnt
+5], &ptr
->pids
[cnt
+6], &ptr
->pids
[cnt
+7]);
244 ptr
->pids_cnt
+= pcnt
;
249 if((fields
< 2) || (ptr
->pids_cnt
<= 0) || (ptr
->freq
== 0) || (strlen(ptr
->name
) == 0))
253 for(cnt
= 0; cnt
< ptr
->pids_cnt
; cnt
++)
255 if(ptr
->pids
[cnt
] == 8192)
257 if(ptr
->pids
[cnt
] == 0)
267 ptr
->pids
[ptr
->pids_cnt
] = 0; //PID 0 is the PAT
270 mp_msg(MSGT_DEMUX
, MSGL_V
, " PIDS: ");
271 for(cnt
= 0; cnt
< ptr
->pids_cnt
; cnt
++)
272 mp_msg(MSGT_DEMUX
, MSGL_V
, " %d ", ptr
->pids
[cnt
]);
273 mp_msg(MSGT_DEMUX
, MSGL_V
, "\n");
275 if((type
== TUNER_TER
) || (type
== TUNER_CBL
))
277 if(! strcmp(inv
, "INVERSION_ON"))
278 ptr
->inv
= INVERSION_ON
;
279 else if(! strcmp(inv
, "INVERSION_OFF"))
280 ptr
->inv
= INVERSION_OFF
;
282 ptr
->inv
= INVERSION_AUTO
;
285 if(! strcmp(cr
, "FEC_1_2"))
287 else if(! strcmp(cr
, "FEC_2_3"))
289 else if(! strcmp(cr
, "FEC_3_4"))
291 #ifdef CONFIG_DVB_HEAD
292 else if(! strcmp(cr
, "FEC_4_5"))
294 else if(! strcmp(cr
, "FEC_6_7"))
296 else if(! strcmp(cr
, "FEC_8_9"))
299 else if(! strcmp(cr
, "FEC_5_6"))
301 else if(! strcmp(cr
, "FEC_7_8"))
303 else if(! strcmp(cr
, "FEC_NONE"))
305 else ptr
->cr
=FEC_AUTO
;
309 if((type
== TUNER_TER
) || (type
== TUNER_CBL
) || (type
== TUNER_ATSC
))
311 if(! strcmp(mod
, "QAM_128"))
313 else if(! strcmp(mod
, "QAM_256"))
315 else if(! strcmp(mod
, "QAM_64"))
317 else if(! strcmp(mod
, "QAM_32"))
319 else if(! strcmp(mod
, "QAM_16"))
322 else if(! strcmp(mod
, "VSB_8") || ! strcmp(mod
, "8VSB"))
324 else if(! strcmp(mod
, "VSB_16") || !strcmp(mod
, "16VSB"))
326 else if(! strcmp(mod
, "QAM_AUTO"))
332 if(type
== TUNER_TER
)
334 if(! strcmp(bw
, "BANDWIDTH_6_MHZ"))
335 ptr
->bw
= BANDWIDTH_6_MHZ
;
336 else if(! strcmp(bw
, "BANDWIDTH_7_MHZ"))
337 ptr
->bw
= BANDWIDTH_7_MHZ
;
338 else if(! strcmp(bw
, "BANDWIDTH_8_MHZ"))
339 ptr
->bw
= BANDWIDTH_8_MHZ
;
342 if(! strcmp(transm
, "TRANSMISSION_MODE_2K"))
343 ptr
->trans
= TRANSMISSION_MODE_2K
;
344 else if(! strcmp(transm
, "TRANSMISSION_MODE_8K"))
345 ptr
->trans
= TRANSMISSION_MODE_8K
;
346 else if(! strcmp(transm
, "TRANSMISSION_MODE_AUTO"))
347 ptr
->trans
= TRANSMISSION_MODE_AUTO
;
349 if(! strcmp(gi
, "GUARD_INTERVAL_1_32"))
350 ptr
->gi
= GUARD_INTERVAL_1_32
;
351 else if(! strcmp(gi
, "GUARD_INTERVAL_1_16"))
352 ptr
->gi
= GUARD_INTERVAL_1_16
;
353 else if(! strcmp(gi
, "GUARD_INTERVAL_1_8"))
354 ptr
->gi
= GUARD_INTERVAL_1_8
;
355 else if(! strcmp(gi
, "GUARD_INTERVAL_1_4"))
356 ptr
->gi
= GUARD_INTERVAL_1_4
;
357 else ptr
->gi
= GUARD_INTERVAL_AUTO
;
359 if(! strcmp(tmp_lcr
, "FEC_1_2"))
361 else if(! strcmp(tmp_lcr
, "FEC_2_3"))
363 else if(! strcmp(tmp_lcr
, "FEC_3_4"))
365 #ifdef CONFIG_DVB_HEAD
366 else if(! strcmp(tmp_lcr
, "FEC_4_5"))
368 else if(! strcmp(tmp_lcr
, "FEC_6_7"))
370 else if(! strcmp(tmp_lcr
, "FEC_8_9"))
373 else if(! strcmp(tmp_lcr
, "FEC_5_6"))
375 else if(! strcmp(tmp_lcr
, "FEC_7_8"))
377 else if(! strcmp(tmp_lcr
, "FEC_NONE"))
378 ptr
->cr_lp
=FEC_NONE
;
379 else ptr
->cr_lp
=FEC_AUTO
;
382 if(! strcmp(tmp_hier
, "HIERARCHY_1"))
383 ptr
->hier
= HIERARCHY_1
;
384 else if(! strcmp(tmp_hier
, "HIERARCHY_2"))
385 ptr
->hier
= HIERARCHY_2
;
386 else if(! strcmp(tmp_hier
, "HIERARCHY_4"))
387 ptr
->hier
= HIERARCHY_4
;
388 #ifdef CONFIG_DVB_HEAD
389 else if(! strcmp(tmp_hier
, "HIERARCHY_AUTO"))
390 ptr
->hier
= HIERARCHY_AUTO
;
392 else ptr
->hier
= HIERARCHY_NONE
;
395 tmp
= realloc(list
->channels
, sizeof(dvb_channel_t
) * (list
->NUM_CHANNELS
+ 1));
399 list
->channels
= tmp
;
400 memcpy(&(list
->channels
[list
->NUM_CHANNELS
]), ptr
, sizeof(dvb_channel_t
));
401 list
->NUM_CHANNELS
++;
402 if(sizeof(dvb_channel_t
) * list
->NUM_CHANNELS
>= 1024*1024)
404 mp_msg(MSGT_DEMUX
, MSGL_V
, "dvbin.c, > 1MB allocated for channels struct, dropping the rest of the file\r\n");
410 if(list
->NUM_CHANNELS
== 0)
412 if(list
->channels
!= NULL
)
413 free(list
->channels
);
422 void dvb_free_config(dvb_config_t
*config
)
426 for(i
=0; i
<config
->count
; i
++)
428 if(config
->cards
[i
].name
)
429 free(config
->cards
[i
].name
);
430 if(!config
->cards
[i
].list
)
432 if(config
->cards
[i
].list
->channels
)
434 for(j
=0; j
<config
->cards
[i
].list
->NUM_CHANNELS
; j
++)
436 if(config
->cards
[i
].list
->channels
[j
].name
)
437 free(config
->cards
[i
].list
->channels
[j
].name
);
439 free(config
->cards
[i
].list
->channels
);
441 free(config
->cards
[i
].list
);
446 static int dvb_streaming_read(stream_t
*stream
, char *buffer
, int size
)
448 struct pollfd pfds
[1];
449 int pos
=0, tries
, rk
, fd
;
450 dvb_priv_t
*priv
= (dvb_priv_t
*) stream
->priv
;
452 mp_msg(MSGT_DEMUX
, MSGL_DBG3
, "dvb_streaming_read(%d)\n", size
);
454 tries
= priv
->retry
+ 1;
460 pfds
[0].events
= POLLIN
| POLLPRI
;
463 if(poll(pfds
, 1, 500) <= 0)
465 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
);
472 if((rk
= read(fd
, &buffer
[pos
], rk
)) > 0)
475 mp_msg(MSGT_DEMUX
, MSGL_DBG3
, "ret (%d) bytes\n", pos
);
481 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_streaming_read, return %d bytes\n", pos
);
486 static void dvbin_close(stream_t
*stream
);
488 int dvb_set_channel(stream_t
*stream
, int card
, int n
)
490 dvb_channels_list
*new_list
;
491 dvb_channel_t
*channel
;
492 dvb_priv_t
*priv
= stream
->priv
;
494 dvb_config_t
*conf
= (dvb_config_t
*) priv
->config
;
498 if((card
< 0) || (card
> conf
->count
))
500 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_set_channel: INVALID CARD NUMBER: %d vs %d, abort\n", card
, conf
->count
);
504 devno
= conf
->cards
[card
].devno
;
505 new_list
= conf
->cards
[card
].list
;
506 if((n
> new_list
->NUM_CHANNELS
) || (n
< 0))
508 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_set_channel: INVALID CHANNEL NUMBER: %d, for card %d, abort\n", n
, card
);
511 channel
= &(new_list
->channels
[n
]);
513 if(priv
->is_on
) //the fds are already open and we have to stop the demuxers
515 for(i
= 0; i
< priv
->demux_fds_cnt
; i
++)
516 dvb_demux_stop(priv
->demux_fds
[i
]);
519 while(dvb_streaming_read(stream
, buf
, 4096) > 0); //empty both the stream's and driver's buffer
520 if(priv
->card
!= card
)
523 if(! dvb_open_devices(priv
, devno
, channel
->pids_cnt
))
525 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "DVB_SET_CHANNEL, COULDN'T OPEN DEVICES OF CARD: %d, EXIT\n", card
);
529 else //close all demux_fds with pos > pids required for the new channel or open other demux_fds if we have too few
531 if(! dvb_fix_demuxes(priv
, channel
->pids_cnt
))
537 if(! dvb_open_devices(priv
, devno
, channel
->pids_cnt
))
539 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "DVB_SET_CHANNEL2, COULDN'T OPEN DEVICES OF CARD: %d, EXIT\n", card
);
545 priv
->list
= new_list
;
547 new_list
->current
= n
;
548 stream
->fd
= priv
->dvr_fd
;
549 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_SET_CHANNEL: new channel name=%s, card: %d, channel %d\n", channel
->name
, card
, n
);
552 stream_reset(stream
);
555 if(channel
->freq
!= priv
->last_freq
)
556 if (! dvb_tune(priv
, channel
->freq
, channel
->pol
, channel
->srate
, channel
->diseqc
, channel
->tone
,
557 channel
->inv
, channel
->mod
, channel
->gi
, channel
->trans
, channel
->bw
, channel
->cr
, channel
->cr_lp
, channel
->hier
, priv
->timeout
))
560 priv
->last_freq
= channel
->freq
;
563 //sets demux filters and restart the stream
564 for(i
= 0; i
< channel
->pids_cnt
; i
++)
566 if(! dvb_set_ts_filt(priv
->demux_fds
[i
], channel
->pids
[i
], DMX_PES_OTHER
))
575 int dvb_step_channel(stream_t
*stream
, int dir
)
578 dvb_channels_list
*list
;
579 dvb_priv_t
*priv
= stream
->priv
;
581 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_STEP_CHANNEL dir %d\n", dir
);
585 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_step_channel: NULL priv_ptr, quit\n");
592 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_step_channel: NULL list_ptr, quit\n");
596 new_current
= (list
->NUM_CHANNELS
+ list
->current
+ (dir
== DVB_CHANNEL_HIGHER
? 1 : -1)) % list
->NUM_CHANNELS
;
598 return dvb_set_channel(stream
, priv
->card
, new_current
);
604 static void dvbin_close(stream_t
*stream
)
607 dvb_priv_t
*priv
= (dvb_priv_t
*) stream
->priv
;
609 for(i
= priv
->demux_fds_cnt
-1; i
>= 0; i
--)
611 priv
->demux_fds_cnt
--;
612 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVBIN_CLOSE, close(%d), fd=%d, COUNT=%d\n", i
, priv
->demux_fds
[i
], priv
->demux_fds_cnt
);
613 close(priv
->demux_fds
[i
]);
618 #ifndef CONFIG_DVB_HEAD
621 priv
->fe_fd
= priv
->sec_fd
= priv
->dvr_fd
= -1;
624 dvb_free_config(priv
->config
);
628 static int dvb_streaming_start(stream_t
*stream
, struct stream_priv_s
*opts
, int tuner_type
, char *progname
)
631 dvb_channel_t
*channel
= NULL
;
632 dvb_priv_t
*priv
= stream
->priv
;
634 mp_msg(MSGT_DEMUX
, MSGL_V
, "\r\ndvb_streaming_start(PROG: %s, CARD: %d, FILE: %s)\r\n",
635 opts
->prog
, opts
->card
, opts
->file
);
640 while((channel
== NULL
) && i
< priv
->list
->NUM_CHANNELS
)
642 if(! strcmp(priv
->list
->channels
[i
].name
, progname
))
643 channel
= &(priv
->list
->channels
[i
]);
650 priv
->list
->current
= i
-1;
651 mp_msg(MSGT_DEMUX
, MSGL_V
, "PROGRAM NUMBER %d: name=%s, freq=%u\n", i
-1, channel
->name
, channel
->freq
);
655 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "\n\nDVBIN: no such channel \"%s\"\n\n", progname
);
660 if(!dvb_set_channel(stream
, priv
->card
, priv
->list
->current
))
662 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "ERROR, COULDN'T SET CHANNEL %i: ", priv
->list
->current
);
667 mp_msg(MSGT_DEMUX
, MSGL_V
, "SUCCESSFUL EXIT from dvb_streaming_start\n");
675 static int dvb_open(stream_t
*stream
, int mode
, void *opts
, int *file_format
)
677 // I don't force the file format bacause, although it's almost always TS,
678 // there are some providers that stream an IP multicast with M$ Mpeg4 inside
679 struct stream_priv_s
* p
= (struct stream_priv_s
*)opts
;
682 int tuner_type
= 0, i
;
685 if(mode
!= STREAM_READ
)
686 return STREAM_UNSUPPORTED
;
688 stream
->priv
= calloc(1, sizeof(dvb_priv_t
));
689 if(stream
->priv
== NULL
)
692 priv
= (dvb_priv_t
*)stream
->priv
;
693 priv
->fe_fd
= priv
->sec_fd
= priv
->dvr_fd
= -1;
694 priv
->config
= dvb_get_config();
695 if(priv
->config
== NULL
)
698 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "DVB CONFIGURATION IS EMPTY, exit\n");
703 for(i
=0; i
<priv
->config
->count
; i
++)
705 if(priv
->config
->cards
[i
].devno
+1 == p
->card
)
715 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "NO CONFIGURATION FOUND FOR CARD N. %d, exit\n", p
->card
);
718 priv
->timeout
= p
->timeout
;
720 tuner_type
= priv
->config
->cards
[priv
->card
].type
;
725 mp_msg(MSGT_DEMUX
, MSGL_V
, "OPEN_DVB: UNKNOWN OR UNDETECTABLE TUNER TYPE, EXIT\n");
730 priv
->tuner_type
= tuner_type
;
732 mp_msg(MSGT_DEMUX
, MSGL_V
, "OPEN_DVB: prog=%s, card=%d, type=%d\n",
733 p
->prog
, priv
->card
+1, priv
->tuner_type
);
735 priv
->list
= priv
->config
->cards
[priv
->card
].list
;
737 if((! strcmp(p
->prog
, "")) && (priv
->list
!= NULL
))
738 progname
= priv
->list
->channels
[0].name
;
743 if(! dvb_streaming_start(stream
, p
, tuner_type
, progname
))
750 stream
->type
= STREAMTYPE_DVB
;
751 stream
->fill_buffer
= dvb_streaming_read
;
752 stream
->close
= dvbin_close
;
753 m_struct_free(&stream_opts
, opts
);
755 *file_format
= DEMUXER_TYPE_MPEG_TS
;
761 dvb_config_t
*dvb_get_config(void)
763 int i
, fd
, type
, size
;
764 char filename
[30], *conf_file
, *name
;
765 dvb_channels_list
*list
;
766 dvb_card_config_t
*cards
= NULL
, *tmp
;
767 dvb_config_t
*conf
= NULL
;
770 conf
= malloc(sizeof(dvb_config_t
));
777 for(i
=0; i
<MAX_CARDS
; i
++)
779 snprintf(filename
, sizeof(filename
), "/dev/dvb/adapter%d/frontend0", i
);
780 fd
= open(filename
, O_RDONLY
|O_NONBLOCK
);
783 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_CONFIG, can't open device %s, skipping\n", filename
);
787 type
= dvb_get_tuner_type(fd
);
789 if(type
!= TUNER_SAT
&& type
!= TUNER_TER
&& type
!= TUNER_CBL
&& type
!= TUNER_ATSC
)
791 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_CONFIG, can't detect tuner type of card %d, skipping\n", i
);
798 conf_file
= get_path("channels.conf.ter");
801 conf_file
= get_path("channels.conf.cbl");
804 conf_file
= get_path("channels.conf.sat");
807 conf_file
= get_path("channels.conf.atsc");
811 if((access(conf_file
, F_OK
| R_OK
) != 0))
815 conf_file
= get_path("channels.conf");
816 if((access(conf_file
, F_OK
| R_OK
) != 0))
820 conf_file
= strdup(MPLAYER_CONFDIR
"/channels.conf");
824 list
= dvb_get_channels(conf_file
, type
);
830 size
= sizeof(dvb_card_config_t
) * (conf
->count
+ 1);
831 tmp
= realloc(conf
->cards
, size
);
835 fprintf(stderr
, "DVB_CONFIG, can't realloc %d bytes, skipping\n", size
);
843 fprintf(stderr
, "DVB_CONFIG, can't realloc 20 bytes, skipping\n");
848 conf
->cards
[conf
->count
].devno
= i
;
849 conf
->cards
[conf
->count
].list
= list
;
850 conf
->cards
[conf
->count
].type
= type
;
851 snprintf(name
, 20, "DVB-%c card n. %d", type
==TUNER_TER
? 'T' : (type
==TUNER_CBL
? 'C' : 'S'), conf
->count
+1);
852 conf
->cards
[conf
->count
].name
= name
;
867 const stream_info_t stream_info_dvb
= {
871 "based on the code from ??? (probably Arpi)",
875 1 // Urls are an option string