8 #include "osdep/timer.h"
9 #include "input/input.h"
11 #include "libmpdemux/demuxer.h"
12 #include "stream_dvdnav.h"
13 #include "libvo/video_out.h"
14 #include "libavutil/common.h"
20 extern char *dvd_device
;
21 extern int dvd_chapter
;
22 extern int dvd_last_chapter
;
24 extern char *audio_lang
, *dvdsub_lang
;
26 static struct stream_priv_s
{
29 } stream_priv_dflts
= {
34 #define ST_OFF(f) M_ST_OFF(struct stream_priv_s,f)
36 static m_option_t stream_opts_fields
[] = {
37 {"filename", ST_OFF(device
), CONF_TYPE_STRING
, 0, 0, 0, NULL
},
38 {"hostname", ST_OFF(track
), CONF_TYPE_INT
, 0, 0, 0, NULL
},
39 { NULL
, NULL
, 0, 0, 0, 0, NULL
}
41 static struct m_struct_st stream_opts
= {
43 sizeof(struct stream_priv_s
),
48 int dvd_nav_still
=0; /* are we on a still picture? */
50 static int seek(stream_t
*s
, off_t newpos
);
52 static dvdnav_priv_t
* new_dvdnav_stream(char * filename
) {
59 if (!(priv
=calloc(1,sizeof(dvdnav_priv_t
))))
62 if (!(priv
->filename
=strdup(filename
))) {
67 if(dvdnav_open(&(priv
->dvdnav
),priv
->filename
)!=DVDNAV_STATUS_OK
)
79 if(1) //from vlc: if not used dvdnav from cvs will fail
84 dvdnav_get_next_block(priv
->dvdnav
,buf
,&event
,&len
);
85 dvdnav_sector_search(priv
->dvdnav
, 0, SEEK_SET
);
88 /* turn off dvdnav caching */
89 dvdnav_set_readahead_flag(priv
->dvdnav
, 0);
90 if(dvdnav_set_PGC_positioning_flag(priv
->dvdnav
, 1) != DVDNAV_STATUS_OK
)
91 mp_msg(MSGT_OPEN
,MSGL_ERR
,"stream_dvdnav, failed to set PGC positioning\n");
93 /* report the title?! */
94 if (dvdnav_get_title_string(priv
->dvdnav
,&title_str
)==DVDNAV_STATUS_OK
) {
95 mp_msg(MSGT_IDENTIFY
, MSGL_INFO
,"Title: '%s'\n",title_str
);
99 //dvdnav_event_clear(priv);
104 static void dvdnav_get_highlight (dvdnav_priv_t
*priv
, int display_mode
) {
105 pci_t
*pnavpci
= NULL
;
106 dvdnav_highlight_event_t
*hlev
= &(priv
->hlev
);
109 if (!priv
|| !priv
->dvdnav
)
112 pnavpci
= dvdnav_get_current_nav_pci (priv
->dvdnav
);
116 dvdnav_get_current_highlight (priv
->dvdnav
, &(hlev
->buttonN
));
117 hlev
->display
= display_mode
; /* show */
119 if (hlev
->buttonN
> 0 && pnavpci
->hli
.hl_gi
.btn_ns
> 0 && hlev
->display
) {
120 for (btnum
= 0; btnum
< pnavpci
->hli
.hl_gi
.btn_ns
; btnum
++) {
121 btni_t
*btni
= &(pnavpci
->hli
.btnit
[btnum
]);
123 if (hlev
->buttonN
== btnum
+ 1) {
124 hlev
->sx
= FFMIN (btni
->x_start
, btni
->x_end
);
125 hlev
->ex
= FFMAX (btni
->x_start
, btni
->x_end
);
126 hlev
->sy
= FFMIN (btni
->y_start
, btni
->y_end
);
127 hlev
->ey
= FFMAX (btni
->y_start
, btni
->y_end
);
129 hlev
->palette
= (btni
->btn_coln
== 0) ? 0 :
130 pnavpci
->hli
.btn_colit
.btn_coli
[btni
->btn_coln
- 1][0];
134 } else { /* hide button or no button */
135 hlev
->sx
= hlev
->ex
= 0;
136 hlev
->sy
= hlev
->ey
= 0;
137 hlev
->palette
= hlev
->buttonN
= 0;
141 static int dvdnav_stream_read(dvdnav_priv_t
* priv
, unsigned char *buf
, int *len
) {
142 int event
= DVDNAV_NOP
;
146 if (!priv
) return -1;
150 mp_msg(MSGT_OPEN
,MSGL_V
, "%s: got a stream_read while I should be asleep!\n",__FUNCTION__
);
155 if (dvdnav_get_next_block(priv
->dvdnav
,buf
,&event
,len
)!=DVDNAV_STATUS_OK
) {
156 mp_msg(MSGT_OPEN
,MSGL_V
, "Error getting next block from DVD %d (%s)\n",event
, dvdnav_err_to_string(priv
->dvdnav
) );
159 else if (event
!=DVDNAV_BLOCK_OK
) {
160 // need to handle certain events internally (like skipping stills)
162 case DVDNAV_NAV_PACKET
:
164 case DVDNAV_STILL_FRAME
: {
165 dvdnav_still_event_t
*still_event
= (dvdnav_still_event_t
*)(buf
);
166 //if (priv->started) dvd_nav_still=1;
168 dvdnav_still_skip(priv
->dvdnav
); // don't let dvdnav stall on this image
172 case DVDNAV_HIGHLIGHT
: {
173 dvdnav_get_highlight (priv
, 1);
176 case DVDNAV_CELL_CHANGE
: {
177 dvdnav_cell_change_event_t
*ev
= (dvdnav_cell_change_event_t
*)buf
;
179 priv
->duration
= ev
->pgc_length
/90;
182 case DVDNAV_SPU_CLUT_CHANGE
: {
183 memcpy(priv
->spu_clut
, buf
, 16*sizeof(unsigned int));
188 dvdnav_wait_skip(priv
->dvdnav
);
197 static void update_title_len(stream_t
*stream
) {
198 dvdnav_priv_t
*priv
= stream
->priv
;
199 dvdnav_status_t status
;
200 uint32_t pos
= 0, len
= 0;
202 status
= dvdnav_get_position(priv
->dvdnav
, &pos
, &len
);
203 if(status
== DVDNAV_STATUS_OK
&& len
) {
204 stream
->end_pos
= (off_t
) len
* 2048;
213 static int seek(stream_t
*s
, off_t newpos
) {
214 uint32_t pos
= 0, len
= 0, sector
= 0;
215 dvdnav_priv_t
*priv
= s
->priv
;
217 if(s
->end_pos
&& newpos
> s
->end_pos
)
219 sector
= newpos
/ 2048ULL;
220 if(dvdnav_sector_search(priv
->dvdnav
, (uint64_t) sector
, SEEK_SET
) != DVDNAV_STATUS_OK
)
228 mp_msg(MSGT_STREAM
,MSGL_INFO
,"dvdnav_stream, seeking to %"PRIu64
" failed: %s\n", newpos
, dvdnav_err_to_string(priv
->dvdnav
));
233 static void stream_dvdnav_close(stream_t
*s
) {
234 dvdnav_priv_t
*priv
= s
->priv
;
235 dvdnav_close(priv
->dvdnav
);
241 static int fill_buffer(stream_t
*s
, char *but
, int len
)
245 dvdnav_priv_t
* priv
=s
->priv
;
249 while(!len
) /* grab all event until DVDNAV_BLOCK_OK (len=2048), DVDNAV_STOP or DVDNAV_STILL_FRAME */
251 event
=dvdnav_stream_read(priv
, s
->buffer
, &len
);
252 if(event
==-1 || len
==-1)
254 mp_msg(MSGT_CPLAYER
,MSGL_ERR
, "DVDNAV stream read error!\n");
259 case DVDNAV_BLOCK_OK
:
260 case DVDNAV_NAV_PACKET
:
262 case DVDNAV_VTS_CHANGE
: {
263 int tit
= 0, part
= 0;
266 if(dvdnav_current_title_info(priv
->dvdnav
, &tit
, &part
) == DVDNAV_STATUS_OK
) {
267 mp_msg(MSGT_CPLAYER
,MSGL_V
, "\r\nDVDNAV, NEW TITLE %d\r\n", tit
);
268 dvdnav_get_highlight (priv
, 0);
269 if(priv
->title
> 0 && tit
!= priv
->title
)
274 case DVDNAV_CELL_CHANGE
: {
275 if(priv
->title
> 0 && dvd_last_chapter
> 0) {
277 if(dvdnav_current_title_info(priv
->dvdnav
, &tit
, &part
) == DVDNAV_STATUS_OK
&& part
> dvd_last_chapter
)
284 mp_msg(MSGT_STREAM
,MSGL_DBG2
,"DVDNAV fill_buffer len: %d\n",len
);
288 static int control(stream_t
*stream
, int cmd
, void* arg
) {
289 dvdnav_priv_t
* priv
=stream
->priv
;
294 case STREAM_CTRL_SEEK_TO_CHAPTER
:
296 int chap
= *((unsigned int *)arg
)+1;
298 if(chap
< 1 || dvdnav_current_title_info(priv
->dvdnav
, &tit
, &part
) != DVDNAV_STATUS_OK
)
300 if(dvdnav_part_play(priv
->dvdnav
, tit
, chap
) != DVDNAV_STATUS_OK
)
304 case STREAM_CTRL_GET_NUM_CHAPTERS
:
306 if(dvdnav_current_title_info(priv
->dvdnav
, &tit
, &part
) != DVDNAV_STATUS_OK
)
308 if(dvdnav_get_number_of_parts(priv
->dvdnav
, tit
, &part
) != DVDNAV_STATUS_OK
)
312 *((unsigned int *)arg
) = part
;
315 case STREAM_CTRL_GET_CURRENT_CHAPTER
:
317 if(dvdnav_current_title_info(priv
->dvdnav
, &tit
, &part
) != DVDNAV_STATUS_OK
)
319 *((unsigned int *)arg
) = part
- 1;
322 case STREAM_CTRL_GET_TIME_LENGTH
:
326 *((double *)arg
) = (double)priv
->duration
/ 1000.0;
333 return STREAM_UNSUPORTED
;
336 static int open_s(stream_t
*stream
,int mode
, void* opts
, int* file_format
) {
337 struct stream_priv_s
* p
= (struct stream_priv_s
*)opts
;
339 int event
,len
,tmplen
=0;
342 dvdnav_status_t status
;
344 if(p
->device
) filename
= p
->device
;
345 else if(dvd_device
) filename
= dvd_device
;
346 else filename
= DEFAULT_DVD_DEVICE
;
347 if(!(priv
=new_dvdnav_stream(filename
))) {
348 mp_msg(MSGT_OPEN
,MSGL_ERR
,MSGTR_CantOpenDVD
,filename
);
349 return STREAM_UNSUPORTED
;
353 if(dvd_chapter
> 0 && dvd_last_chapter
> 0 && dvd_chapter
> dvd_last_chapter
) {
354 mp_msg(MSGT_OPEN
,MSGL_FATAL
,"dvdnav_stream, invalid chapter range: %d > %d\n", dvd_chapter
, dvd_last_chapter
);
355 return STREAM_UNSUPORTED
;
357 priv
->title
= p
->track
;
358 if(dvdnav_title_play(priv
->dvdnav
, p
->track
) != DVDNAV_STATUS_OK
) {
359 mp_msg(MSGT_OPEN
,MSGL_FATAL
,"dvdnav_stream, couldn't select title %d, error '%s'\n", p
->track
, dvdnav_err_to_string(priv
->dvdnav
));
360 return STREAM_UNSUPORTED
;
363 dvdnav_part_play(priv
->dvdnav
, p
->track
, dvd_chapter
);
364 } else if(p
->track
== -1)
365 dvdnav_menu_call(priv
->dvdnav
, DVD_MENU_Root
);
367 mp_msg(MSGT_OPEN
,MSGL_INFO
,"dvdnav_stream, you didn't specify a track number (as in dvdnav://1), playing whole disc\n");
368 dvdnav_menu_call(priv
->dvdnav
, DVD_MENU_Title
);
371 dvdnav_angle_change(priv
->dvdnav
, dvd_angle
);
373 stream
->sector_size
= 2048;
374 stream
->flags
= STREAM_READ
| STREAM_SEEK
;
375 stream
->fill_buffer
= fill_buffer
;
377 stream
->control
= control
;
378 stream
->close
= stream_dvdnav_close
;
379 stream
->type
= STREAMTYPE_DVDNAV
;
380 stream
->priv
=(void*)priv
;
381 *file_format
= DEMUXER_TYPE_MPEG_PS
;
383 update_title_len(stream
);
385 mp_msg(MSGT_OPEN
,MSGL_ERR
, "INIT ERROR: %d, couldn't get init pos %s\r\n", status
, dvdnav_err_to_string(priv
->dvdnav
));
387 mp_msg(MSGT_OPEN
,MSGL_INFO
, "Remember to disable MPlayer's cache when playing dvdnav:// streams (adding -nocache to your command line)\r\n");
393 int mp_dvdnav_handle_input(stream_t
*stream
, int cmd
, int *button
) {
394 dvdnav_priv_t
* priv
=(dvdnav_priv_t
*)stream
->priv
;
395 dvdnav_t
*nav
= priv
->dvdnav
;
396 dvdnav_status_t status
=DVDNAV_STATUS_ERR
;
397 pci_t
*pci
= dvdnav_get_current_nav_pci(nav
);
400 if(cmd
!= MP_CMD_DVDNAV_SELECT
&& !pci
)
404 case MP_CMD_DVDNAV_UP
:
405 status
= dvdnav_upper_button_select(nav
, pci
);
407 case MP_CMD_DVDNAV_DOWN
:
408 status
= dvdnav_lower_button_select(nav
, pci
);
410 case MP_CMD_DVDNAV_LEFT
:
411 status
= dvdnav_left_button_select(nav
, pci
);
413 case MP_CMD_DVDNAV_RIGHT
:
414 status
= dvdnav_right_button_select(nav
, pci
);
416 case MP_CMD_DVDNAV_MENU
:
417 status
= dvdnav_menu_call(nav
,DVD_MENU_Root
);
420 case MP_CMD_DVDNAV_PREVMENU
: {
423 dvdnav_current_title_info(nav
, &title
, &part
);
425 if(dvdnav_menu_call(nav
, DVD_MENU_Part
) == DVDNAV_STATUS_OK
426 || dvdnav_menu_call(nav
, DVD_MENU_Title
) == DVDNAV_STATUS_OK
) {
431 if(dvdnav_menu_call(nav
, DVD_MENU_Root
) == DVDNAV_STATUS_OK
)
435 case MP_CMD_DVDNAV_SELECT
:
436 status
= dvdnav_button_activate(nav
, pci
);
437 if(status
== DVDNAV_STATUS_OK
) reset
= 1;
439 case MP_CMD_DVDNAV_MOUSECLICK
:
441 this is a workaround: in theory the simple dvdnav_lower_button_select()+dvdnav_button_activate()
442 should be enough (and generally it is), but there are cases when the calls to dvdnav_lower_button_select()
443 and friends fail! Hence we have to call dvdnav_mouse_activate(priv->mousex, priv->mousey) with
444 the coodinates saved by mp_dvdnav_update_mouse_pos().
445 This last call always works well
447 status
= dvdnav_mouse_activate(nav
, pci
, priv
->mousex
, priv
->mousey
);
450 mp_msg(MSGT_CPLAYER
, MSGL_V
, "Unknown DVDNAV cmd %d\n", cmd
);
454 if(status
== DVDNAV_STATUS_OK
)
455 dvdnav_get_current_highlight(nav
, button
);
460 void mp_dvdnav_update_mouse_pos(stream_t
*stream
, int32_t x
, int32_t y
, int* button
) {
461 dvdnav_priv_t
* priv
=(dvdnav_priv_t
*)stream
->priv
;
462 dvdnav_t
*nav
= priv
->dvdnav
;
463 dvdnav_status_t status
;
464 pci_t
*pci
= dvdnav_get_current_nav_pci(nav
);
468 status
= dvdnav_mouse_select(nav
, pci
, x
, y
);
469 if(status
== DVDNAV_STATUS_OK
) dvdnav_get_current_highlight(nav
, button
);
476 * \brief dvdnav_aid_from_lang() returns the audio id corresponding to the language code 'lang'
477 * \param stream: - stream pointer
478 * \param lang: 2-characters language code[s], eventually separated by spaces of commas
479 * \return -1 on error, current subtitle id if successful
481 int dvdnav_aid_from_lang(stream_t
*stream
, unsigned char *language
) {
482 dvdnav_priv_t
* priv
=(dvdnav_priv_t
*)stream
->priv
;
485 uint16_t lang
, lcode
;;
487 #ifdef DVDNAV_FORMAT_AC3
488 //this macro is defined only in libdvdnav-cvs
489 while(language
&& strlen(language
)>=2) {
490 lcode
= (language
[0] << 8) | (language
[1]);
491 for(k
=0; k
<32; k
++) {
492 lg
= dvdnav_get_audio_logical_stream(priv
->dvdnav
, k
);
493 if(lg
== 0xff) continue;
494 lang
= dvdnav_audio_stream_to_lang(priv
->dvdnav
, lg
);
495 if(lang
!= 0xFFFF && lang
== lcode
) {
496 format
= dvdnav_audio_stream_format(priv
->dvdnav
, lg
);
498 case DVDNAV_FORMAT_AC3
:
500 case DVDNAV_FORMAT_DTS
:
502 case DVDNAV_FORMAT_LPCM
:
504 case DVDNAV_FORMAT_MPEGAUDIO
:
512 while(language
[0]==',' || language
[0]==' ') ++language
;
519 * \brief dvdnav_lang_from_aid() assigns to buf the language corresponding to audio id 'aid'
520 * \param stream: - stream pointer
521 * \param sid: physical subtitle id
522 * \param buf: buffer to contain the 2-chars language string
523 * \return 0 on error, 1 if successful
525 int dvdnav_lang_from_aid(stream_t
*stream
, int aid
, unsigned char *buf
) {
528 dvdnav_priv_t
* priv
=(dvdnav_priv_t
*)stream
->priv
;
532 lg
= dvdnav_get_audio_logical_stream(priv
->dvdnav
, aid
& 0x7);
533 if(lg
== 0xff) return 0;
534 lang
= dvdnav_audio_stream_to_lang(priv
->dvdnav
, lg
);
535 if(lang
== 0xffff) return 0;
537 buf
[1] = lang
& 0xFF;
544 * \brief dvdnav_sid_from_lang() returns the subtitle id corresponding to the language code 'lang'
545 * \param stream: - stream pointer
546 * \param lang: 2-characters language code[s], eventually separated by spaces of commas
547 * \return -1 on error, current subtitle id if successful
549 int dvdnav_sid_from_lang(stream_t
*stream
, unsigned char *language
) {
550 dvdnav_priv_t
* priv
=(dvdnav_priv_t
*)stream
->priv
;
551 uint8_t format
, lg
, k
;
552 uint16_t lang
, lcode
;
554 while(language
&& strlen(language
)>=2) {
555 lcode
= (language
[0] << 8) | (language
[1]);
556 for(k
=0; k
<32; k
++) {
557 lg
= dvdnav_get_spu_logical_stream(priv
->dvdnav
, k
);
558 if(lg
== 0xff) continue;
559 lang
= dvdnav_spu_stream_to_lang(priv
->dvdnav
, lg
);
560 if(lang
!= 0xFFFF && lang
== lcode
) {
565 while(language
[0]==',' || language
[0]==' ') ++language
;
571 * \brief dvdnav_lang_from_sid() assigns to buf the language corresponding to subtitle id 'sid'
572 * \param stream: - stream pointer
573 * \param sid: physical subtitle id
574 * \param buf: buffer to contain the 2-chars language string
575 * \return 0 on error, 1 if successful
577 int dvdnav_lang_from_sid(stream_t
*stream
, int sid
, unsigned char *buf
) {
580 dvdnav_priv_t
*priv
=(dvdnav_priv_t
*)stream
->priv
;
581 if(sid
< 0) return 0;
582 lg
= dvdnav_get_spu_logical_stream(priv
->dvdnav
, sid
);
583 lang
= dvdnav_spu_stream_to_lang(priv
->dvdnav
, lg
);
584 if(lang
== 0xffff) return 0;
586 buf
[1] = lang
& 0xFF;
592 * \brief dvdnav_number_of_subs() returns the count of available subtitles
593 * \param stream: - stream pointer
594 * \return 0 on error, something meaningful otherwise
596 int dvdnav_number_of_subs(stream_t
*stream
) {
597 dvdnav_priv_t
* priv
=(dvdnav_priv_t
*)stream
->priv
;
600 for(k
=0; k
<32; k
++) {
601 lg
= dvdnav_get_spu_logical_stream(priv
->dvdnav
, k
);
602 if(lg
== 0xff) continue;
608 unsigned int *mp_dvdnav_get_spu_clut(stream_t
*stream
) {
609 dvdnav_priv_t
*priv
=(dvdnav_priv_t
*)stream
->priv
;
610 if(!priv
->spu_set
) return NULL
;
611 return priv
->spu_clut
;
614 void mp_dvdnav_get_highlight (stream_t
*stream
, nav_highlight_t
*hl
) {
615 dvdnav_priv_t
*priv
= (dvdnav_priv_t
*) stream
->priv
;
616 dvdnav_highlight_event_t hlev
= priv
->hlev
;
624 stream_info_t stream_info_dvdnav
= {
632 1 // Urls are an option string