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"
47 #include "libavutil/avstring.h"
52 #define MAX_CHANNELS 8
53 #define CHANNEL_LINE_LEN 256
54 #define min(a, b) ((a) <= (b) ? (a) : (b))
57 //TODO: CAMBIARE list_ptr e da globale a per_priv
60 static struct stream_priv_s
72 #define ST_OFF(f) M_ST_OFF(struct stream_priv_s, f)
75 static const m_option_t stream_params
[] = {
76 {"prog", ST_OFF(prog
), CONF_TYPE_STRING
, 0, 0 ,0, NULL
},
77 {"card", ST_OFF(card
), CONF_TYPE_INT
, M_OPT_RANGE
, 1, 4, NULL
},
78 {"timeout",ST_OFF(timeout
), CONF_TYPE_INT
, M_OPT_RANGE
, 1, 30, NULL
},
79 {"file", ST_OFF(file
), CONF_TYPE_STRING
, 0, 0 ,0, NULL
},
81 {"hostname", ST_OFF(prog
), CONF_TYPE_STRING
, 0, 0, 0, NULL
},
82 {"username", ST_OFF(card
), CONF_TYPE_INT
, M_OPT_RANGE
, 1, 4, NULL
},
83 {NULL
, NULL
, 0, 0, 0, 0, NULL
}
86 static const struct m_struct_st stream_opts
= {
88 sizeof(struct stream_priv_s
),
95 const 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 {"timeout", &stream_defaults
.timeout
, CONF_TYPE_INT
, M_OPT_RANGE
, 1, 30, NULL
},
99 {"file", &stream_defaults
.file
, CONF_TYPE_STRING
, 0, 0 ,0, NULL
},
101 {NULL
, NULL
, 0, 0, 0, 0, NULL
}
107 int dvb_set_ts_filt(int fd
, uint16_t pid
, dmx_pes_type_t pestype
);
108 int dvb_demux_stop(int fd
);
109 int dvb_get_tuner_type(int fd
);
110 int dvb_open_devices(dvb_priv_t
*priv
, int n
, int demux_cnt
);
111 int dvb_fix_demuxes(dvb_priv_t
*priv
, int cnt
);
113 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
,
116 fe_code_rate_t LP_CodeRate
, fe_hierarchy_t hier
, int timeout
);
120 static dvb_channels_list
*dvb_get_channels(char *filename
, int type
)
122 dvb_channels_list
*list
;
124 char line
[CHANNEL_LINE_LEN
], *colon
;
126 int fields
, cnt
, pcnt
, k
;
128 dvb_channel_t
*ptr
, *tmp
, chn
;
129 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];
130 const char *cbl_conf
= "%d:%255[^:]:%d:%255[^:]:%255[^:]:%255[^:]:%255[^:]\n";
131 const char *sat_conf
= "%d:%c:%d:%d:%255[^:]:%255[^:]\n";
132 const char *ter_conf
= "%d:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]:%255[^:]\n";
133 const char *atsc_conf
= "%d:%255[^:]:%255[^:]:%255[^:]\n";
135 mp_msg(MSGT_DEMUX
, MSGL_V
, "CONFIG_READ FILE: %s, type: %d\n", filename
, type
);
136 if((f
=fopen(filename
, "r"))==NULL
)
138 mp_msg(MSGT_DEMUX
, MSGL_FATAL
, "CAN'T READ CONFIG FILE %s\n", filename
);
142 list
= malloc(sizeof(dvb_channels_list
));
146 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_GET_CHANNELS: couldn't malloc enough memory\n");
151 list
->NUM_CHANNELS
= 0;
152 list
->channels
= NULL
;
155 if( fgets(line
, CHANNEL_LINE_LEN
, f
) == NULL
)
158 if((line
[0] == '#') || (strlen(line
) == 0))
161 colon
= strchr(line
, ':');
167 ptr
->name
= malloc(k
+1);
170 av_strlcpy(ptr
->name
, line
, k
+1);
175 apid_str
[0] = vpid_str
[0] = 0;
178 if(type
== TUNER_TER
)
180 fields
= sscanf(&line
[k
], ter_conf
,
181 &ptr
->freq
, inv
, bw
, cr
, tmp_lcr
, mod
,
182 transm
, gi
, tmp_hier
, vpid_str
, apid_str
);
183 mp_msg(MSGT_DEMUX
, MSGL_V
,
184 "TER, NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d",
185 list
->NUM_CHANNELS
, fields
, ptr
->name
, ptr
->freq
);
187 else if(type
== TUNER_CBL
)
189 fields
= sscanf(&line
[k
], cbl_conf
,
190 &ptr
->freq
, inv
, &ptr
->srate
,
191 cr
, mod
, vpid_str
, apid_str
);
192 mp_msg(MSGT_DEMUX
, MSGL_V
,
193 "CBL, NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d, SRATE: %d",
194 list
->NUM_CHANNELS
, fields
, ptr
->name
, ptr
->freq
, ptr
->srate
);
197 else if(type
== TUNER_ATSC
)
199 fields
= sscanf(&line
[k
], atsc_conf
,
200 &ptr
->freq
, mod
, vpid_str
, apid_str
);
201 mp_msg(MSGT_DEMUX
, MSGL_V
,
202 "ATSC, NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d\n",
203 list
->NUM_CHANNELS
, fields
, ptr
->name
, ptr
->freq
);
208 fields
= sscanf(&line
[k
], sat_conf
,
209 &ptr
->freq
, &ptr
->pol
, &ptr
->diseqc
, &ptr
->srate
, vpid_str
, apid_str
);
210 ptr
->pol
= toupper(ptr
->pol
);
212 ptr
->srate
*= 1000UL;
214 ptr
->inv
= INVERSION_AUTO
;
216 if((ptr
->diseqc
> 4) || (ptr
->diseqc
< 0))
220 mp_msg(MSGT_DEMUX
, MSGL_V
,
221 "SAT, NUM: %d, NUM_FIELDS: %d, NAME: %s, FREQ: %d, SRATE: %d, POL: %c, DISEQC: %d",
222 list
->NUM_CHANNELS
, fields
, ptr
->name
, ptr
->freq
, ptr
->srate
, ptr
->pol
, ptr
->diseqc
);
227 pcnt
= sscanf(vpid_str
, "%d+%d+%d+%d+%d+%d+%d", &ptr
->pids
[0], &ptr
->pids
[1], &ptr
->pids
[2], &ptr
->pids
[3],
228 &ptr
->pids
[4], &ptr
->pids
[5], &ptr
->pids
[6]);
231 ptr
->pids_cnt
= pcnt
;
239 pcnt
= sscanf(apid_str
, "%d+%d+%d+%d+%d+%d+%d+%d", &ptr
->pids
[cnt
], &ptr
->pids
[cnt
+1], &ptr
->pids
[cnt
+2],
240 &ptr
->pids
[cnt
+3], &ptr
->pids
[cnt
+4], &ptr
->pids
[cnt
+5], &ptr
->pids
[cnt
+6], &ptr
->pids
[cnt
+7]);
243 ptr
->pids_cnt
+= pcnt
;
248 if((fields
< 2) || (ptr
->pids_cnt
<= 0) || (ptr
->freq
== 0) || (strlen(ptr
->name
) == 0))
252 for(cnt
= 0; cnt
< ptr
->pids_cnt
; cnt
++)
254 if(ptr
->pids
[cnt
] == 8192)
256 if(ptr
->pids
[cnt
] == 0)
266 ptr
->pids
[ptr
->pids_cnt
] = 0; //PID 0 is the PAT
269 mp_msg(MSGT_DEMUX
, MSGL_V
, " PIDS: ");
270 for(cnt
= 0; cnt
< ptr
->pids_cnt
; cnt
++)
271 mp_msg(MSGT_DEMUX
, MSGL_V
, " %d ", ptr
->pids
[cnt
]);
272 mp_msg(MSGT_DEMUX
, MSGL_V
, "\n");
274 if((type
== TUNER_TER
) || (type
== TUNER_CBL
))
276 if(! strcmp(inv
, "INVERSION_ON"))
277 ptr
->inv
= INVERSION_ON
;
278 else if(! strcmp(inv
, "INVERSION_OFF"))
279 ptr
->inv
= INVERSION_OFF
;
281 ptr
->inv
= INVERSION_AUTO
;
284 if(! strcmp(cr
, "FEC_1_2"))
286 else if(! strcmp(cr
, "FEC_2_3"))
288 else if(! strcmp(cr
, "FEC_3_4"))
290 else if(! strcmp(cr
, "FEC_4_5"))
292 else if(! strcmp(cr
, "FEC_6_7"))
294 else if(! strcmp(cr
, "FEC_8_9"))
296 else if(! strcmp(cr
, "FEC_5_6"))
298 else if(! strcmp(cr
, "FEC_7_8"))
300 else if(! strcmp(cr
, "FEC_NONE"))
302 else ptr
->cr
=FEC_AUTO
;
306 if((type
== TUNER_TER
) || (type
== TUNER_CBL
) || (type
== TUNER_ATSC
))
308 if(! strcmp(mod
, "QAM_128"))
310 else if(! strcmp(mod
, "QAM_256"))
312 else if(! strcmp(mod
, "QAM_64"))
314 else if(! strcmp(mod
, "QAM_32"))
316 else if(! strcmp(mod
, "QAM_16"))
319 else if(! strcmp(mod
, "VSB_8") || ! strcmp(mod
, "8VSB"))
321 else if(! strcmp(mod
, "VSB_16") || !strcmp(mod
, "16VSB"))
323 else if(! strcmp(mod
, "QAM_AUTO"))
329 if(type
== TUNER_TER
)
331 if(! strcmp(bw
, "BANDWIDTH_6_MHZ"))
332 ptr
->bw
= BANDWIDTH_6_MHZ
;
333 else if(! strcmp(bw
, "BANDWIDTH_7_MHZ"))
334 ptr
->bw
= BANDWIDTH_7_MHZ
;
335 else if(! strcmp(bw
, "BANDWIDTH_8_MHZ"))
336 ptr
->bw
= BANDWIDTH_8_MHZ
;
339 if(! strcmp(transm
, "TRANSMISSION_MODE_2K"))
340 ptr
->trans
= TRANSMISSION_MODE_2K
;
341 else if(! strcmp(transm
, "TRANSMISSION_MODE_8K"))
342 ptr
->trans
= TRANSMISSION_MODE_8K
;
343 else if(! strcmp(transm
, "TRANSMISSION_MODE_AUTO"))
344 ptr
->trans
= TRANSMISSION_MODE_AUTO
;
346 if(! strcmp(gi
, "GUARD_INTERVAL_1_32"))
347 ptr
->gi
= GUARD_INTERVAL_1_32
;
348 else if(! strcmp(gi
, "GUARD_INTERVAL_1_16"))
349 ptr
->gi
= GUARD_INTERVAL_1_16
;
350 else if(! strcmp(gi
, "GUARD_INTERVAL_1_8"))
351 ptr
->gi
= GUARD_INTERVAL_1_8
;
352 else if(! strcmp(gi
, "GUARD_INTERVAL_1_4"))
353 ptr
->gi
= GUARD_INTERVAL_1_4
;
354 else ptr
->gi
= GUARD_INTERVAL_AUTO
;
356 if(! strcmp(tmp_lcr
, "FEC_1_2"))
358 else if(! strcmp(tmp_lcr
, "FEC_2_3"))
360 else if(! strcmp(tmp_lcr
, "FEC_3_4"))
362 else if(! strcmp(tmp_lcr
, "FEC_4_5"))
364 else if(! strcmp(tmp_lcr
, "FEC_6_7"))
366 else if(! strcmp(tmp_lcr
, "FEC_8_9"))
368 else if(! strcmp(tmp_lcr
, "FEC_5_6"))
370 else if(! strcmp(tmp_lcr
, "FEC_7_8"))
372 else if(! strcmp(tmp_lcr
, "FEC_NONE"))
373 ptr
->cr_lp
=FEC_NONE
;
374 else ptr
->cr_lp
=FEC_AUTO
;
377 if(! strcmp(tmp_hier
, "HIERARCHY_1"))
378 ptr
->hier
= HIERARCHY_1
;
379 else if(! strcmp(tmp_hier
, "HIERARCHY_2"))
380 ptr
->hier
= HIERARCHY_2
;
381 else if(! strcmp(tmp_hier
, "HIERARCHY_4"))
382 ptr
->hier
= HIERARCHY_4
;
383 else if(! strcmp(tmp_hier
, "HIERARCHY_AUTO"))
384 ptr
->hier
= HIERARCHY_AUTO
;
385 else ptr
->hier
= HIERARCHY_NONE
;
388 tmp
= realloc(list
->channels
, sizeof(dvb_channel_t
) * (list
->NUM_CHANNELS
+ 1));
392 list
->channels
= tmp
;
393 memcpy(&(list
->channels
[list
->NUM_CHANNELS
]), ptr
, sizeof(dvb_channel_t
));
394 list
->NUM_CHANNELS
++;
395 if(sizeof(dvb_channel_t
) * list
->NUM_CHANNELS
>= 1024*1024)
397 mp_msg(MSGT_DEMUX
, MSGL_V
, "dvbin.c, > 1MB allocated for channels struct, dropping the rest of the file\r\n");
403 if(list
->NUM_CHANNELS
== 0)
405 if(list
->channels
!= NULL
)
406 free(list
->channels
);
415 void dvb_free_config(dvb_config_t
*config
)
419 for(i
=0; i
<config
->count
; i
++)
421 if(config
->cards
[i
].name
)
422 free(config
->cards
[i
].name
);
423 if(!config
->cards
[i
].list
)
425 if(config
->cards
[i
].list
->channels
)
427 for(j
=0; j
<config
->cards
[i
].list
->NUM_CHANNELS
; j
++)
429 if(config
->cards
[i
].list
->channels
[j
].name
)
430 free(config
->cards
[i
].list
->channels
[j
].name
);
432 free(config
->cards
[i
].list
->channels
);
434 free(config
->cards
[i
].list
);
439 static int dvb_streaming_read(stream_t
*stream
, char *buffer
, int size
)
441 struct pollfd pfds
[1];
442 int pos
=0, tries
, rk
, fd
;
443 dvb_priv_t
*priv
= (dvb_priv_t
*) stream
->priv
;
445 mp_msg(MSGT_DEMUX
, MSGL_DBG3
, "dvb_streaming_read(%d)\n", size
);
447 tries
= priv
->retry
+ 1;
453 pfds
[0].events
= POLLIN
| POLLPRI
;
456 if(poll(pfds
, 1, 500) <= 0)
458 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
);
465 if((rk
= read(fd
, &buffer
[pos
], rk
)) > 0)
468 mp_msg(MSGT_DEMUX
, MSGL_DBG3
, "ret (%d) bytes\n", pos
);
474 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_streaming_read, return %d bytes\n", pos
);
479 static void dvbin_close(stream_t
*stream
);
481 int dvb_set_channel(stream_t
*stream
, int card
, int n
)
483 dvb_channels_list
*new_list
;
484 dvb_channel_t
*channel
;
485 dvb_priv_t
*priv
= stream
->priv
;
487 dvb_config_t
*conf
= (dvb_config_t
*) priv
->config
;
491 if((card
< 0) || (card
> conf
->count
))
493 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_set_channel: INVALID CARD NUMBER: %d vs %d, abort\n", card
, conf
->count
);
497 devno
= conf
->cards
[card
].devno
;
498 new_list
= conf
->cards
[card
].list
;
499 if((n
> new_list
->NUM_CHANNELS
) || (n
< 0))
501 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_set_channel: INVALID CHANNEL NUMBER: %d, for card %d, abort\n", n
, card
);
504 channel
= &(new_list
->channels
[n
]);
506 if(priv
->is_on
) //the fds are already open and we have to stop the demuxers
508 for(i
= 0; i
< priv
->demux_fds_cnt
; i
++)
509 dvb_demux_stop(priv
->demux_fds
[i
]);
512 while(dvb_streaming_read(stream
, buf
, 4096) > 0); //empty both the stream's and driver's buffer
513 if(priv
->card
!= card
)
516 if(! dvb_open_devices(priv
, devno
, channel
->pids_cnt
))
518 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "DVB_SET_CHANNEL, COULDN'T OPEN DEVICES OF CARD: %d, EXIT\n", card
);
522 else //close all demux_fds with pos > pids required for the new channel or open other demux_fds if we have too few
524 if(! dvb_fix_demuxes(priv
, channel
->pids_cnt
))
530 if(! dvb_open_devices(priv
, devno
, channel
->pids_cnt
))
532 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "DVB_SET_CHANNEL2, COULDN'T OPEN DEVICES OF CARD: %d, EXIT\n", card
);
538 priv
->list
= new_list
;
540 new_list
->current
= n
;
541 stream
->fd
= priv
->dvr_fd
;
542 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_SET_CHANNEL: new channel name=%s, card: %d, channel %d\n", channel
->name
, card
, n
);
544 stream
->buf_pos
= stream
->buf_len
= 0;
548 if(channel
->freq
!= priv
->last_freq
)
549 if (! dvb_tune(priv
, channel
->freq
, channel
->pol
, channel
->srate
, channel
->diseqc
, channel
->tone
,
550 channel
->inv
, channel
->mod
, channel
->gi
, channel
->trans
, channel
->bw
, channel
->cr
, channel
->cr_lp
, channel
->hier
, priv
->timeout
))
553 priv
->last_freq
= channel
->freq
;
556 //sets demux filters and restart the stream
557 for(i
= 0; i
< channel
->pids_cnt
; i
++)
559 if(! dvb_set_ts_filt(priv
->demux_fds
[i
], channel
->pids
[i
], DMX_PES_OTHER
))
568 int dvb_step_channel(stream_t
*stream
, int dir
)
571 dvb_channels_list
*list
;
572 dvb_priv_t
*priv
= stream
->priv
;
574 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_STEP_CHANNEL dir %d\n", dir
);
578 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_step_channel: NULL priv_ptr, quit\n");
585 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_step_channel: NULL list_ptr, quit\n");
589 new_current
= (list
->NUM_CHANNELS
+ list
->current
+ (dir
== DVB_CHANNEL_HIGHER
? 1 : -1)) % list
->NUM_CHANNELS
;
591 return dvb_set_channel(stream
, priv
->card
, new_current
);
597 static void dvbin_close(stream_t
*stream
)
600 dvb_priv_t
*priv
= (dvb_priv_t
*) stream
->priv
;
602 for(i
= priv
->demux_fds_cnt
-1; i
>= 0; i
--)
604 priv
->demux_fds_cnt
--;
605 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVBIN_CLOSE, close(%d), fd=%d, COUNT=%d\n", i
, priv
->demux_fds
[i
], priv
->demux_fds_cnt
);
606 close(priv
->demux_fds
[i
]);
611 priv
->fe_fd
= priv
->sec_fd
= priv
->dvr_fd
= -1;
614 dvb_free_config(priv
->config
);
618 static int dvb_streaming_start(stream_t
*stream
, struct stream_priv_s
*opts
, int tuner_type
, char *progname
)
621 dvb_channel_t
*channel
= NULL
;
622 dvb_priv_t
*priv
= stream
->priv
;
624 mp_msg(MSGT_DEMUX
, MSGL_V
, "\r\ndvb_streaming_start(PROG: %s, CARD: %d, FILE: %s)\r\n",
625 opts
->prog
, opts
->card
, opts
->file
);
630 while((channel
== NULL
) && i
< priv
->list
->NUM_CHANNELS
)
632 if(! strcmp(priv
->list
->channels
[i
].name
, progname
))
633 channel
= &(priv
->list
->channels
[i
]);
640 priv
->list
->current
= i
-1;
641 mp_msg(MSGT_DEMUX
, MSGL_V
, "PROGRAM NUMBER %d: name=%s, freq=%u\n", i
-1, channel
->name
, channel
->freq
);
645 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "\n\nDVBIN: no such channel \"%s\"\n\n", progname
);
650 if(!dvb_set_channel(stream
, priv
->card
, priv
->list
->current
))
652 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "ERROR, COULDN'T SET CHANNEL %i: ", priv
->list
->current
);
657 mp_msg(MSGT_DEMUX
, MSGL_V
, "SUCCESSFUL EXIT from dvb_streaming_start\n");
665 static int dvb_open(stream_t
*stream
, int mode
, void *opts
, int *file_format
)
667 // I don't force the file format bacause, although it's almost always TS,
668 // there are some providers that stream an IP multicast with M$ Mpeg4 inside
669 struct stream_priv_s
* p
= (struct stream_priv_s
*)opts
;
672 int tuner_type
= 0, i
;
675 if(mode
!= STREAM_READ
)
676 return STREAM_UNSUPPORTED
;
678 stream
->priv
= calloc(1, sizeof(dvb_priv_t
));
679 if(stream
->priv
== NULL
)
682 priv
= (dvb_priv_t
*)stream
->priv
;
683 priv
->fe_fd
= priv
->sec_fd
= priv
->dvr_fd
= -1;
684 priv
->config
= dvb_get_config();
685 if(priv
->config
== NULL
)
688 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "DVB CONFIGURATION IS EMPTY, exit\n");
693 for(i
=0; i
<priv
->config
->count
; i
++)
695 if(priv
->config
->cards
[i
].devno
+1 == p
->card
)
705 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "NO CONFIGURATION FOUND FOR CARD N. %d, exit\n", p
->card
);
708 priv
->timeout
= p
->timeout
;
710 tuner_type
= priv
->config
->cards
[priv
->card
].type
;
715 mp_msg(MSGT_DEMUX
, MSGL_V
, "OPEN_DVB: UNKNOWN OR UNDETECTABLE TUNER TYPE, EXIT\n");
720 priv
->tuner_type
= tuner_type
;
722 mp_msg(MSGT_DEMUX
, MSGL_V
, "OPEN_DVB: prog=%s, card=%d, type=%d\n",
723 p
->prog
, priv
->card
+1, priv
->tuner_type
);
725 priv
->list
= priv
->config
->cards
[priv
->card
].list
;
727 if((! strcmp(p
->prog
, "")) && (priv
->list
!= NULL
))
728 progname
= priv
->list
->channels
[0].name
;
733 if(! dvb_streaming_start(stream
, p
, tuner_type
, progname
))
740 stream
->type
= STREAMTYPE_DVB
;
741 stream
->fill_buffer
= dvb_streaming_read
;
742 stream
->close
= dvbin_close
;
743 m_struct_free(&stream_opts
, opts
);
745 *file_format
= DEMUXER_TYPE_MPEG_TS
;
751 dvb_config_t
*dvb_get_config(void)
753 int i
, fd
, type
, size
;
754 char filename
[30], *conf_file
, *name
;
755 dvb_channels_list
*list
;
756 dvb_card_config_t
*cards
= NULL
, *tmp
;
757 dvb_config_t
*conf
= NULL
;
760 conf
= malloc(sizeof(dvb_config_t
));
767 for(i
=0; i
<MAX_CARDS
; i
++)
769 snprintf(filename
, sizeof(filename
), "/dev/dvb/adapter%d/frontend0", i
);
770 fd
= open(filename
, O_RDONLY
|O_NONBLOCK
);
773 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_CONFIG, can't open device %s, skipping\n", filename
);
777 type
= dvb_get_tuner_type(fd
);
779 if(type
!= TUNER_SAT
&& type
!= TUNER_TER
&& type
!= TUNER_CBL
&& type
!= TUNER_ATSC
)
781 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_CONFIG, can't detect tuner type of card %d, skipping\n", i
);
785 conf_file
= get_path("channels.conf");
789 conf_file
= get_path("channels.conf.ter");
792 conf_file
= get_path("channels.conf.cbl");
795 conf_file
= get_path("channels.conf.sat");
798 conf_file
= get_path("channels.conf.atsc");
802 if((access(conf_file
, F_OK
| R_OK
) != 0))
806 conf_file
= get_path("channels.conf");
807 if((access(conf_file
, F_OK
| R_OK
) != 0))
811 conf_file
= strdup(MPLAYER_CONFDIR
"/channels.conf");
815 list
= dvb_get_channels(conf_file
, type
);
821 size
= sizeof(dvb_card_config_t
) * (conf
->count
+ 1);
822 tmp
= realloc(conf
->cards
, size
);
826 fprintf(stderr
, "DVB_CONFIG, can't realloc %d bytes, skipping\n", size
);
834 fprintf(stderr
, "DVB_CONFIG, can't realloc 20 bytes, skipping\n");
839 conf
->cards
[conf
->count
].devno
= i
;
840 conf
->cards
[conf
->count
].list
= list
;
841 conf
->cards
[conf
->count
].type
= type
;
842 snprintf(name
, 20, "DVB-%c card n. %d", type
==TUNER_TER
? 'T' : (type
==TUNER_CBL
? 'C' : 'S'), conf
->count
+1);
843 conf
->cards
[conf
->count
].name
= name
;
858 const stream_info_t stream_info_dvb
= {
862 "based on the code from ??? (probably Arpi)",
866 1 // Urls are an option string