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"))
327 ptr
->inv
= INVERSION_AUTO
;
331 if(type
== TUNER_TER
)
333 if(! strcmp(bw
, "BANDWIDTH_6_MHZ"))
334 ptr
->bw
= BANDWIDTH_6_MHZ
;
335 else if(! strcmp(bw
, "BANDWIDTH_7_MHZ"))
336 ptr
->bw
= BANDWIDTH_7_MHZ
;
337 else if(! strcmp(bw
, "BANDWIDTH_8_MHZ"))
338 ptr
->bw
= BANDWIDTH_8_MHZ
;
341 if(! strcmp(transm
, "TRANSMISSION_MODE_2K"))
342 ptr
->trans
= TRANSMISSION_MODE_2K
;
343 else if(! strcmp(transm
, "TRANSMISSION_MODE_8K"))
344 ptr
->trans
= TRANSMISSION_MODE_8K
;
347 if(! strcmp(gi
, "GUARD_INTERVAL_1_32"))
348 ptr
->gi
= GUARD_INTERVAL_1_32
;
349 else if(! strcmp(gi
, "GUARD_INTERVAL_1_16"))
350 ptr
->gi
= GUARD_INTERVAL_1_16
;
351 else if(! strcmp(gi
, "GUARD_INTERVAL_1_8"))
352 ptr
->gi
= GUARD_INTERVAL_1_8
;
353 else ptr
->gi
= GUARD_INTERVAL_1_4
;
355 if(! strcmp(tmp_lcr
, "FEC_1_2"))
357 else if(! strcmp(tmp_lcr
, "FEC_2_3"))
359 else if(! strcmp(tmp_lcr
, "FEC_3_4"))
361 #ifdef CONFIG_DVB_HEAD
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"))
369 else if(! strcmp(tmp_lcr
, "FEC_5_6"))
371 else if(! strcmp(tmp_lcr
, "FEC_7_8"))
373 else if(! strcmp(tmp_lcr
, "FEC_NONE"))
374 ptr
->cr_lp
=FEC_NONE
;
375 else ptr
->cr_lp
=FEC_AUTO
;
378 if(! strcmp(tmp_hier
, "HIERARCHY_1"))
379 ptr
->hier
= HIERARCHY_1
;
380 else if(! strcmp(tmp_hier
, "HIERARCHY_2"))
381 ptr
->hier
= HIERARCHY_2
;
382 else if(! strcmp(tmp_hier
, "HIERARCHY_4"))
383 ptr
->hier
= HIERARCHY_4
;
384 #ifdef CONFIG_DVB_HEAD
385 else if(! strcmp(tmp_hier
, "HIERARCHY_AUTO"))
386 ptr
->hier
= HIERARCHY_AUTO
;
388 else ptr
->hier
= HIERARCHY_NONE
;
391 tmp
= realloc(list
->channels
, sizeof(dvb_channel_t
) * (list
->NUM_CHANNELS
+ 1));
395 list
->channels
= tmp
;
396 memcpy(&(list
->channels
[list
->NUM_CHANNELS
]), ptr
, sizeof(dvb_channel_t
));
397 list
->NUM_CHANNELS
++;
398 if(sizeof(dvb_channel_t
) * list
->NUM_CHANNELS
>= 1024*1024)
400 mp_msg(MSGT_DEMUX
, MSGL_V
, "dvbin.c, > 1MB allocated for channels struct, dropping the rest of the file\r\n");
406 if(list
->NUM_CHANNELS
== 0)
408 if(list
->channels
!= NULL
)
409 free(list
->channels
);
418 void dvb_free_config(dvb_config_t
*config
)
422 for(i
=0; i
<config
->count
; i
++)
424 if(config
->cards
[i
].name
)
425 free(config
->cards
[i
].name
);
426 if(!config
->cards
[i
].list
)
428 if(config
->cards
[i
].list
->channels
)
430 for(j
=0; j
<config
->cards
[i
].list
->NUM_CHANNELS
; j
++)
432 if(config
->cards
[i
].list
->channels
[j
].name
)
433 free(config
->cards
[i
].list
->channels
[j
].name
);
435 free(config
->cards
[i
].list
->channels
);
437 free(config
->cards
[i
].list
);
442 static int dvb_streaming_read(stream_t
*stream
, char *buffer
, int size
)
444 struct pollfd pfds
[1];
445 int pos
=0, tries
, rk
, fd
;
446 dvb_priv_t
*priv
= (dvb_priv_t
*) stream
->priv
;
448 mp_msg(MSGT_DEMUX
, MSGL_DBG3
, "dvb_streaming_read(%d)\n", size
);
450 tries
= priv
->retry
+ 1;
456 pfds
[0].events
= POLLIN
| POLLPRI
;
459 if(poll(pfds
, 1, 500) <= 0)
461 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
);
468 if((rk
= read(fd
, &buffer
[pos
], rk
)) > 0)
471 mp_msg(MSGT_DEMUX
, MSGL_DBG3
, "ret (%d) bytes\n", pos
);
477 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_streaming_read, return %d bytes\n", pos
);
482 static void dvbin_close(stream_t
*stream
);
484 int dvb_set_channel(stream_t
*stream
, int card
, int n
)
486 dvb_channels_list
*new_list
;
487 dvb_channel_t
*channel
;
488 dvb_priv_t
*priv
= stream
->priv
;
490 dvb_config_t
*conf
= (dvb_config_t
*) priv
->config
;
494 if((card
< 0) || (card
> conf
->count
))
496 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_set_channel: INVALID CARD NUMBER: %d vs %d, abort\n", card
, conf
->count
);
500 devno
= conf
->cards
[card
].devno
;
501 new_list
= conf
->cards
[card
].list
;
502 if((n
> new_list
->NUM_CHANNELS
) || (n
< 0))
504 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_set_channel: INVALID CHANNEL NUMBER: %d, for card %d, abort\n", n
, card
);
507 channel
= &(new_list
->channels
[n
]);
509 if(priv
->is_on
) //the fds are already open and we have to stop the demuxers
511 for(i
= 0; i
< priv
->demux_fds_cnt
; i
++)
512 dvb_demux_stop(priv
->demux_fds
[i
]);
515 while(dvb_streaming_read(stream
, buf
, 4096) > 0); //empty both the stream's and driver's buffer
516 if(priv
->card
!= card
)
519 if(! dvb_open_devices(priv
, devno
, channel
->pids_cnt
))
521 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "DVB_SET_CHANNEL, COULDN'T OPEN DEVICES OF CARD: %d, EXIT\n", card
);
525 else //close all demux_fds with pos > pids required for the new channel or open other demux_fds if we have too few
527 if(! dvb_fix_demuxes(priv
, channel
->pids_cnt
))
533 if(! dvb_open_devices(priv
, devno
, channel
->pids_cnt
))
535 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "DVB_SET_CHANNEL2, COULDN'T OPEN DEVICES OF CARD: %d, EXIT\n", card
);
541 priv
->list
= new_list
;
543 new_list
->current
= n
;
544 stream
->fd
= priv
->dvr_fd
;
545 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_SET_CHANNEL: new channel name=%s, card: %d, channel %d\n", channel
->name
, card
, n
);
548 stream_reset(stream
);
551 if(channel
->freq
!= priv
->last_freq
)
552 if (! dvb_tune(priv
, channel
->freq
, channel
->pol
, channel
->srate
, channel
->diseqc
, channel
->tone
,
553 channel
->inv
, channel
->mod
, channel
->gi
, channel
->trans
, channel
->bw
, channel
->cr
, channel
->cr_lp
, channel
->hier
, priv
->timeout
))
556 priv
->last_freq
= channel
->freq
;
559 //sets demux filters and restart the stream
560 for(i
= 0; i
< channel
->pids_cnt
; i
++)
562 if(! dvb_set_ts_filt(priv
->demux_fds
[i
], channel
->pids
[i
], DMX_PES_OTHER
))
571 int dvb_step_channel(stream_t
*stream
, int dir
)
574 dvb_channels_list
*list
;
575 dvb_priv_t
*priv
= stream
->priv
;
577 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_STEP_CHANNEL dir %d\n", dir
);
581 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_step_channel: NULL priv_ptr, quit\n");
588 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "dvb_step_channel: NULL list_ptr, quit\n");
592 new_current
= (list
->NUM_CHANNELS
+ list
->current
+ (dir
== DVB_CHANNEL_HIGHER
? 1 : -1)) % list
->NUM_CHANNELS
;
594 return dvb_set_channel(stream
, priv
->card
, new_current
);
600 static void dvbin_close(stream_t
*stream
)
603 dvb_priv_t
*priv
= (dvb_priv_t
*) stream
->priv
;
605 for(i
= priv
->demux_fds_cnt
-1; i
>= 0; i
--)
607 priv
->demux_fds_cnt
--;
608 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVBIN_CLOSE, close(%d), fd=%d, COUNT=%d\n", i
, priv
->demux_fds
[i
], priv
->demux_fds_cnt
);
609 close(priv
->demux_fds
[i
]);
614 #ifndef CONFIG_DVB_HEAD
617 priv
->fe_fd
= priv
->sec_fd
= priv
->dvr_fd
= -1;
620 dvb_free_config(priv
->config
);
624 static int dvb_streaming_start(stream_t
*stream
, struct stream_priv_s
*opts
, int tuner_type
, char *progname
)
627 dvb_channel_t
*channel
= NULL
;
628 dvb_priv_t
*priv
= stream
->priv
;
630 mp_msg(MSGT_DEMUX
, MSGL_V
, "\r\ndvb_streaming_start(PROG: %s, CARD: %d, FILE: %s)\r\n",
631 opts
->prog
, opts
->card
, opts
->file
);
636 while((channel
== NULL
) && i
< priv
->list
->NUM_CHANNELS
)
638 if(! strcmp(priv
->list
->channels
[i
].name
, progname
))
639 channel
= &(priv
->list
->channels
[i
]);
646 priv
->list
->current
= i
-1;
647 mp_msg(MSGT_DEMUX
, MSGL_V
, "PROGRAM NUMBER %d: name=%s, freq=%u\n", i
-1, channel
->name
, channel
->freq
);
651 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "\n\nDVBIN: no such channel \"%s\"\n\n", progname
);
656 if(!dvb_set_channel(stream
, priv
->card
, priv
->list
->current
))
658 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "ERROR, COULDN'T SET CHANNEL %i: ", priv
->list
->current
);
663 mp_msg(MSGT_DEMUX
, MSGL_V
, "SUCCESSFUL EXIT from dvb_streaming_start\n");
671 static int dvb_open(stream_t
*stream
, int mode
, void *opts
, int *file_format
)
673 // I don't force the file format bacause, although it's almost always TS,
674 // there are some providers that stream an IP multicast with M$ Mpeg4 inside
675 struct stream_priv_s
* p
= (struct stream_priv_s
*)opts
;
678 int tuner_type
= 0, i
;
681 if(mode
!= STREAM_READ
)
682 return STREAM_UNSUPPORTED
;
684 stream
->priv
= calloc(1, sizeof(dvb_priv_t
));
685 if(stream
->priv
== NULL
)
688 priv
= (dvb_priv_t
*)stream
->priv
;
689 priv
->fe_fd
= priv
->sec_fd
= priv
->dvr_fd
= -1;
690 priv
->config
= dvb_get_config();
691 if(priv
->config
== NULL
)
694 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "DVB CONFIGURATION IS EMPTY, exit\n");
699 for(i
=0; i
<priv
->config
->count
; i
++)
701 if(priv
->config
->cards
[i
].devno
+1 == p
->card
)
711 mp_msg(MSGT_DEMUX
, MSGL_ERR
, "NO CONFIGURATION FOUND FOR CARD N. %d, exit\n", p
->card
);
714 priv
->timeout
= p
->timeout
;
716 tuner_type
= priv
->config
->cards
[priv
->card
].type
;
721 mp_msg(MSGT_DEMUX
, MSGL_V
, "OPEN_DVB: UNKNOWN OR UNDETECTABLE TUNER TYPE, EXIT\n");
726 priv
->tuner_type
= tuner_type
;
728 mp_msg(MSGT_DEMUX
, MSGL_V
, "OPEN_DVB: prog=%s, card=%d, type=%d\n",
729 p
->prog
, priv
->card
+1, priv
->tuner_type
);
731 priv
->list
= priv
->config
->cards
[priv
->card
].list
;
733 if((! strcmp(p
->prog
, "")) && (priv
->list
!= NULL
))
734 progname
= priv
->list
->channels
[0].name
;
739 if(! dvb_streaming_start(stream
, p
, tuner_type
, progname
))
746 stream
->type
= STREAMTYPE_DVB
;
747 stream
->fill_buffer
= dvb_streaming_read
;
748 stream
->close
= dvbin_close
;
749 m_struct_free(&stream_opts
, opts
);
751 *file_format
= DEMUXER_TYPE_MPEG_TS
;
757 dvb_config_t
*dvb_get_config(void)
759 int i
, fd
, type
, size
;
760 char filename
[30], *conf_file
, *name
;
761 dvb_channels_list
*list
;
762 dvb_card_config_t
*cards
= NULL
, *tmp
;
763 dvb_config_t
*conf
= NULL
;
766 conf
= malloc(sizeof(dvb_config_t
));
773 for(i
=0; i
<MAX_CARDS
; i
++)
775 snprintf(filename
, sizeof(filename
), "/dev/dvb/adapter%d/frontend0", i
);
776 fd
= open(filename
, O_RDONLY
|O_NONBLOCK
);
779 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_CONFIG, can't open device %s, skipping\n", filename
);
783 type
= dvb_get_tuner_type(fd
);
785 if(type
!= TUNER_SAT
&& type
!= TUNER_TER
&& type
!= TUNER_CBL
&& type
!= TUNER_ATSC
)
787 mp_msg(MSGT_DEMUX
, MSGL_V
, "DVB_CONFIG, can't detect tuner type of card %d, skipping\n", i
);
794 conf_file
= get_path("channels.conf.ter");
797 conf_file
= get_path("channels.conf.cbl");
800 conf_file
= get_path("channels.conf.sat");
803 conf_file
= get_path("channels.conf.atsc");
807 if((access(conf_file
, F_OK
| R_OK
) != 0))
811 conf_file
= get_path("channels.conf");
812 if((access(conf_file
, F_OK
| R_OK
) != 0))
816 conf_file
= strdup(MPLAYER_CONFDIR
"/channels.conf");
820 list
= dvb_get_channels(conf_file
, type
);
826 size
= sizeof(dvb_card_config_t
) * (conf
->count
+ 1);
827 tmp
= realloc(conf
->cards
, size
);
831 fprintf(stderr
, "DVB_CONFIG, can't realloc %d bytes, skipping\n", size
);
839 fprintf(stderr
, "DVB_CONFIG, can't realloc 20 bytes, skipping\n");
844 conf
->cards
[conf
->count
].devno
= i
;
845 conf
->cards
[conf
->count
].list
= list
;
846 conf
->cards
[conf
->count
].type
= type
;
847 snprintf(name
, 20, "DVB-%c card n. %d", type
==TUNER_TER
? 'T' : (type
==TUNER_CBL
? 'C' : 'S'), conf
->count
+1);
848 conf
->cards
[conf
->count
].name
= name
;
863 const stream_info_t stream_info_dvb
= {
867 "based on the code from ??? (probably Arpi)",
871 1 // Urls are an option string