2 #include <lib/base/eerror.h>
3 #include <lib/base/object.h>
4 #include <lib/base/ebase.h>
5 #include <lib/base/nconfig.h>
7 #include <lib/service/service.h>
8 #include <lib/base/init_num.h>
9 #include <lib/base/init.h>
10 #include <lib/gui/esubtitle.h>
11 #include <lib/gdi/gpixmap.h>
14 #include <netinet/in.h>
16 #error no byte order defined!
20 #include <dreamdvd/ddvdlib.h>
22 #include "servicedvd.h"
26 eServiceFactoryDVD::eServiceFactoryDVD()
28 ePtr<eServiceCenter> sc;
30 eServiceCenter::getPrivInstance(sc);
33 std::list<std::string> extensions;
34 extensions.push_back("iso");
35 sc->addServiceFactory(eServiceFactoryDVD::id, this, extensions);
39 eServiceFactoryDVD::~eServiceFactoryDVD()
41 ePtr<eServiceCenter> sc;
43 eServiceCenter::getPrivInstance(sc);
45 sc->removeServiceFactory(eServiceFactoryDVD::id);
48 DEFINE_REF(eServiceFactoryDVD)
51 RESULT eServiceFactoryDVD::play(const eServiceReference &ref, ePtr<iPlayableService> &ptr)
54 ptr = new eServiceDVD(ref.path.c_str());
58 RESULT eServiceFactoryDVD::record(const eServiceReference &/*ref*/, ePtr<iRecordableService> &ptr)
64 RESULT eServiceFactoryDVD::list(const eServiceReference &, ePtr<iListableService> &ptr)
71 RESULT eServiceFactoryDVD::info(const eServiceReference &/*ref*/, ePtr<iStaticServiceInformation> &ptr)
77 RESULT eServiceFactoryDVD::offlineOperations(const eServiceReference &, ePtr<iServiceOfflineOperations> &ptr)
85 DEFINE_REF(eServiceDVD);
87 eServiceDVD::eServiceDVD(const char *filename):
89 m_ddvdconfig(ddvd_create()),
95 int aspect = DDVD_16_9;
96 int policy = DDVD_PAN_SCAN;
101 m_sn = eSocketNotifier::create(eApp, ddvd_get_messagepipe_fd(m_ddvdconfig), eSocketNotifier::Read|eSocketNotifier::Priority|eSocketNotifier::Error|eSocketNotifier::Hungup);
102 eDebug("SERVICEDVD construct!");
104 ddvd_set_dvd_path(m_ddvdconfig, filename);
105 ddvd_set_ac3thru(m_ddvdconfig, 0);
106 ddvd_set_language(m_ddvdconfig, "de");
108 int fd = open("/proc/stb/video/aspect", O_RDONLY);
111 rd = read(fd, tmp, 255);
112 if (rd > 2 && !strncmp(tmp, "4:3", 3))
114 else if (rd > 4 && !strncmp(tmp, "16:10", 5))
119 fd = open("/proc/stb/video/policy", O_RDONLY);
122 rd = read(fd, tmp, 255);
123 if (rd > 6 && !strncmp(tmp, "bestfit", 3))
124 aspect = DDVD_JUSTSCALE;
125 else if (rd > 8 && !strncmp(tmp, "letterbox", 5))
126 aspect = DDVD_LETTERBOX;
130 ddvd_set_video(m_ddvdconfig, aspect, policy, DDVD_PAL /*unused*/);
132 CONNECT(m_sn->activated, eServiceDVD::gotMessage);
133 CONNECT(m_pump.recv_msg, eServiceDVD::gotThreadMessage);
134 strcpy(m_ddvd_titlestring,"");
139 void eServiceDVD::gotThreadMessage(const int &msg)
143 case 1: // thread stopped
145 m_event(this, evStopped);
150 void eServiceDVD::gotMessage(int /*what*/)
152 switch(ddvd_get_next_message(m_ddvdconfig,1))
154 case DDVD_COLORTABLE_UPDATE:
157 struct ddvd_color ctmp[4];
158 ddvd_get_last_colortable(ddvdconfig, ctmp);
162 rd1[252+i]=ctmp[i].red;
163 bl1[252+i]=ctmp[i].blue;
164 gn1[252+i]=ctmp[i].green;
165 tr1[252+i]=ctmp[i].trans;
168 if(ioctl(fb, FBIOPUTCMAP, &colormap) == -1)
170 printf("Framebuffer: <FBIOPUTCMAP failed>\n");
174 eDebug("no support for 8bpp framebuffer in dvdplayer yet!");
177 case DDVD_SCREEN_UPDATE:
178 eDebug("DVD_SCREEN_UPDATE!");
179 if (m_subtitle_widget)
180 m_subtitle_widget->setPixmap(m_pixmap, eRect(ePoint(0, 0), m_pixmap->size()));
182 case DDVD_SHOWOSD_STATE_PLAY:
184 eDebug("DVD_SHOWOSD_STATE_PLAY!");
186 m_event(this, evUser+1);
189 case DDVD_SHOWOSD_STATE_PAUSE:
191 eDebug("DVD_SHOWOSD_STATE_PAUSE!");
192 m_event(this, evUser+2);
195 case DDVD_SHOWOSD_STATE_FFWD:
197 eDebug("DVD_SHOWOSD_STATE_FFWD!");
198 m_event(this, evUser+3);
201 case DDVD_SHOWOSD_STATE_FBWD:
203 eDebug("DVD_SHOWOSD_STATE_FBWD!");
204 m_event(this, evUser+4);
207 case DDVD_SHOWOSD_STRING:
209 eDebug("DVD_SHOWOSD_STRING!");
210 m_event(this, evUser+5);
213 case DDVD_SHOWOSD_AUDIO:
215 eDebug("DVD_SHOWOSD_STRING!");
216 m_event(this, evUser+6);
219 case DDVD_SHOWOSD_SUBTITLE:
221 eDebug("DVD_SHOWOSD_SUBTITLE!");
222 m_event((iPlayableService*)this, evUpdatedInfo);
223 m_event(this, evUser+7);
226 case DDVD_EOF_REACHED:
227 eDebug("DVD_EOF_REACHED!");
228 m_event(this, evEOF);
230 case DDVD_SOF_REACHED:
231 eDebug("DVD_SOF_REACHED!");
232 m_event(this, evSOF);
234 case DDVD_SHOWOSD_TIME:
236 static struct ddvd_time last_info;
237 struct ddvd_time info;
238 // eDebug("DVD_SHOWOSD_TIME!");
239 ddvd_get_last_time(m_ddvdconfig, &info);
240 if ( info.pos_chapter != last_info.pos_chapter )
241 m_event(this, evUser+8); // chapterUpdated
242 if ( info.pos_title != last_info.pos_title )
243 m_event(this, evUser+9); // titleUpdated
244 memcpy(&last_info, &info, sizeof(struct ddvd_time));
247 case DDVD_SHOWOSD_TITLESTRING:
249 ddvd_get_title_string(m_ddvdconfig, m_ddvd_titlestring);
250 eDebug("DDVD_SHOWOSD_TITLESTRING: %s",m_ddvd_titlestring);
254 m_event(this, evStart);
257 case DDVD_MENU_OPENED:
258 eDebug("DVD_MENU_OPENED!");
260 m_event(this, evSeekableStatusChanged);
261 m_event(this, evUser+11);
263 case DDVD_MENU_CLOSED:
264 eDebug("DVD_MENU_CLOSED!");
266 m_event(this, evSeekableStatusChanged);
267 m_event(this, evUser+12);
274 eServiceDVD::~eServiceDVD()
276 eDebug("SERVICEDVD destruct!");
279 ddvd_close(m_ddvdconfig);
282 RESULT eServiceDVD::connectEvent(const Slot2<void,iPlayableService*,int> &event, ePtr<eConnection> &connection)
284 connection = new eConnection((iPlayableService*)this, m_event.connect(event));
288 RESULT eServiceDVD::start()
290 assert(m_state == stIdle);
292 eDebug("eServiceDVD starting");
293 // m_event(this, evStart);
297 RESULT eServiceDVD::stop()
299 assert(m_state != stIdle);
300 if (m_state == stStopped)
302 eDebug("DVD: stop %s", m_filename.c_str());
304 ddvd_send_key(m_ddvdconfig, DDVD_KEY_EXIT);
309 RESULT eServiceDVD::setTarget(int /*target*/)
314 RESULT eServiceDVD::pause(ePtr<iPauseableService> &ptr)
320 RESULT eServiceDVD::seek(ePtr<iSeekableService> &ptr)
326 RESULT eServiceDVD::subtitle(ePtr<iSubtitleOutput> &ptr)
332 RESULT eServiceDVD::keys(ePtr<iServiceKeys> &ptr)
339 RESULT eServiceDVD::setSlowMotion(int /*ratio*/)
344 RESULT eServiceDVD::setFastForward(int trick)
346 eDebug("setTrickmode(%d)", trick);
347 while (m_current_trick > trick && m_current_trick != -64)
349 ddvd_send_key(m_ddvdconfig, DDVD_KEY_FBWD);
350 if (m_current_trick == 0)
351 m_current_trick = -2;
352 else if (m_current_trick > 0)
354 m_current_trick /= 2;
355 if (abs(m_current_trick) == 1)
359 m_current_trick *= 2;
361 while (m_current_trick < trick && m_current_trick != 64)
363 ddvd_send_key(m_ddvdconfig, DDVD_KEY_FFWD);
364 if (m_current_trick == 0)
366 else if (m_current_trick < 0)
368 m_current_trick /= 2;
369 if (abs(m_current_trick) == 1)
373 m_current_trick *= 2;
378 RESULT eServiceDVD::pause()
380 eDebug("set pause!\n");
381 ddvd_send_key(m_ddvdconfig, DDVD_KEY_PAUSE);
385 RESULT eServiceDVD::unpause()
387 eDebug("set unpause!\n");
388 ddvd_send_key(m_ddvdconfig, DDVD_KEY_PLAY);
392 void eServiceDVD::thread()
394 eDebug("eServiceDVD dvd thread started");
396 ddvd_run(m_ddvdconfig);
399 void eServiceDVD::thread_finished()
401 eDebug("eServiceDVD dvd thread finished");
402 m_pump.send(1); // inform main thread
405 RESULT eServiceDVD::info(ePtr<iServiceInformation>&i)
411 RESULT eServiceDVD::getName(std::string &name)
413 if ( m_ddvd_titlestring[0] != '\0' )
414 name = m_ddvd_titlestring;
420 int eServiceDVD::getInfo(int w)
424 case sCurrentChapter:
426 struct ddvd_time info;
427 ddvd_get_last_time(m_ddvdconfig, &info);
428 return info.pos_chapter;
432 struct ddvd_time info;
433 ddvd_get_last_time(m_ddvdconfig, &info);
434 return info.end_chapter;
438 struct ddvd_time info;
439 ddvd_get_last_time(m_ddvdconfig, &info);
440 return info.pos_title;
444 struct ddvd_time info;
445 ddvd_get_last_time(m_ddvdconfig, &info);
446 return info.end_title;
448 case sTXTPID: // we abuse HAS_TELEXT icon in InfoBar to signalize subtitles status
452 ddvd_get_last_spu(m_ddvdconfig, &spu_id, &spu_lang);
457 return resIsPyObject;
463 std::string eServiceDVD::getInfoString(int w)
470 eDebug("unhandled getInfoString(%d)", w);
475 PyObject *eServiceDVD::getInfoObject(int w)
481 ePyObject tuple = PyTuple_New(3);
482 int audio_id,audio_type;
484 ddvd_get_last_audio(m_ddvdconfig, &audio_id, &audio_lang, &audio_type);
485 char audio_string[3]={audio_lang >> 8, audio_lang, 0};
486 PyTuple_SetItem(tuple, 0, PyInt_FromLong(audio_id+1));
487 PyTuple_SetItem(tuple, 1, PyString_FromString(audio_string));
491 PyTuple_SetItem(tuple, 2, PyString_FromString("MPEG"));
494 PyTuple_SetItem(tuple, 2, PyString_FromString("AC3"));
497 PyTuple_SetItem(tuple, 2, PyString_FromString("DTS"));
500 PyTuple_SetItem(tuple, 2, PyString_FromString("LPCM"));
503 PyTuple_SetItem(tuple, 2, PyString_FromString(""));
509 ePyObject tuple = PyTuple_New(2);
512 ddvd_get_last_spu(m_ddvdconfig, &spu_id, &spu_lang);
513 char spu_string[3]={spu_lang >> 8, spu_lang, 0};
516 PyTuple_SetItem(tuple, 0, PyInt_FromLong(0));
517 PyTuple_SetItem(tuple, 1, PyString_FromString(""));
521 PyTuple_SetItem(tuple, 0, PyInt_FromLong(spu_id+1));
522 PyTuple_SetItem(tuple, 1, PyString_FromString(spu_string));
527 eDebug("unhandled getInfoObject(%d)", w);
532 RESULT eServiceDVD::enableSubtitles(eWidget *parent, SWIG_PYOBJECT(ePyObject) /*entry*/)
534 if (m_subtitle_widget)
535 delete m_subtitle_widget;
537 m_subtitle_widget = new eSubtitleWidget(parent);
538 m_subtitle_widget->resize(parent->size());
540 eSize size = parent->size();
544 m_pixmap = new gPixmap(size, 32);
545 ddvd_set_lfb(m_ddvdconfig, (unsigned char *)m_pixmap->surface->data, size.width(), size.height(), 4, size.width()*4);
546 run(); // start the thread
549 m_subtitle_widget->setZPosition(-1);
550 m_subtitle_widget->show();
555 RESULT eServiceDVD::disableSubtitles(eWidget */*parent*/)
557 delete m_subtitle_widget;
558 m_subtitle_widget = 0;
562 PyObject *eServiceDVD::getSubtitleList()
564 eDebug("eServiceDVD::getSubtitleList nyi");
568 PyObject *eServiceDVD::getCachedSubtitle()
570 eDebug("eServiceDVD::getCachedSubtitle nyi");
574 RESULT eServiceDVD::getLength(pts_t &len)
576 // eDebug("eServiceDVD::getLength");
577 struct ddvd_time info;
578 ddvd_get_last_time(m_ddvdconfig, &info);
579 len = info.end_hours * 3600;
580 len += info.end_minutes * 60;
581 len += info.end_seconds;
586 RESULT eServiceDVD::seekTo(pts_t to)
588 eDebug("eServiceDVD::seekTo(%lld)",to);
591 eDebug("set_resume_pos: resume_info.title=%d, chapter=%d, block=%lu, audio_id=%d, audio_lock=%d, spu_id=%d, spu_lock=%d",m_resume_info.title,m_resume_info.chapter,m_resume_info.block,m_resume_info.audio_id, m_resume_info.audio_lock, m_resume_info.spu_id, m_resume_info.spu_lock);
592 ddvd_set_resume_pos(m_ddvdconfig, m_resume_info);
597 RESULT eServiceDVD::seekRelative(int direction, pts_t to)
599 int seconds = to / 90000;
600 seconds *= direction;
601 eDebug("seekRelative %d %d", direction, seconds);
602 ddvd_skip_seconds(m_ddvdconfig, seconds);
606 RESULT eServiceDVD::getPlayPosition(pts_t &pos)
608 struct ddvd_time info;
609 ddvd_get_last_time(m_ddvdconfig, &info);
610 pos = info.pos_hours * 3600;
611 pos += info.pos_minutes * 60;
612 pos += info.pos_seconds;
613 // eDebug("getPlayPosition %lld", pos);
618 RESULT eServiceDVD::seekTitle(int title)
620 eDebug("setTitle %d", title);
621 ddvd_set_title(m_ddvdconfig, title);
625 RESULT eServiceDVD::seekChapter(int chapter)
627 eDebug("setChapter %d", chapter);
629 ddvd_set_chapter(m_ddvdconfig, chapter);
633 RESULT eServiceDVD::setTrickmode(int /*trick*/)
638 RESULT eServiceDVD::isCurrentlySeekable()
640 return m_state == stRunning;
643 RESULT eServiceDVD::keyPressed(int key)
647 case iServiceKeys::keyLeft:
648 ddvd_send_key(m_ddvdconfig, DDVD_KEY_LEFT);
650 case iServiceKeys::keyRight:
651 ddvd_send_key(m_ddvdconfig, DDVD_KEY_RIGHT);
653 case iServiceKeys::keyUp:
654 ddvd_send_key(m_ddvdconfig, DDVD_KEY_UP);
656 case iServiceKeys::keyDown:
657 ddvd_send_key(m_ddvdconfig, DDVD_KEY_DOWN);
659 case iServiceKeys::keyOk:
660 ddvd_send_key(m_ddvdconfig, DDVD_KEY_OK);
662 case iServiceKeys::keyUser:
663 ddvd_send_key(m_ddvdconfig, DDVD_KEY_AUDIO);
665 case iServiceKeys::keyUser+1:
666 ddvd_send_key(m_ddvdconfig, DDVD_KEY_SUBTITLE);
668 case iServiceKeys::keyUser+2:
669 ddvd_send_key(m_ddvdconfig, DDVD_KEY_AUDIOMENU);
671 case iServiceKeys::keyUser+3:
672 ddvd_send_key(m_ddvdconfig, DDVD_KEY_NEXT_CHAPTER);
674 case iServiceKeys::keyUser+4:
675 ddvd_send_key(m_ddvdconfig, DDVD_KEY_PREV_CHAPTER);
677 case iServiceKeys::keyUser+5:
678 ddvd_send_key(m_ddvdconfig, DDVD_KEY_NEXT_TITLE);
680 case iServiceKeys::keyUser+6:
681 ddvd_send_key(m_ddvdconfig, DDVD_KEY_PREV_TITLE);
683 case iServiceKeys::keyUser+7:
684 ddvd_send_key(m_ddvdconfig, DDVD_KEY_MENU);
692 RESULT eServiceDVD::cueSheet(ePtr<iCueSheet> &ptr)
703 PyObject *eServiceDVD::getCutList()
705 ePyObject list = PyList_New(1);
706 ePyObject tuple = PyTuple_New(2);
707 PyTuple_SetItem(tuple, 0, PyLong_FromLongLong(m_cue_pts));
708 PyTuple_SetItem(tuple, 1, PyInt_FromLong(3));
709 PyList_SetItem(list, 0, tuple);
713 void eServiceDVD::setCutList(ePyObject /*list*/)
717 void eServiceDVD::setCutListEnable(int /*enable*/)
721 void eServiceDVD::loadCuesheet()
724 if ( m_ddvd_titlestring[0] != '\0' )
725 snprintf(filename, 128, "/home/root/dvd-%s.cuts", m_ddvd_titlestring);
727 snprintf(filename, 128, "%s/dvd.cuts", m_filename.c_str());
729 eDebug("eServiceDVD::loadCuesheet() filename=%s",filename);
731 FILE *f = fopen(filename, "rb");
735 unsigned long long where;
738 if (!fread(&where, sizeof(where), 1, f))
740 if (!fread(&what, sizeof(what), 1, f))
742 #if BYTE_ORDER == LITTLE_ENDIAN
743 where = bswap_64(where);
747 if (!fread(&m_resume_info, sizeof(struct ddvd_resume), 1, f))
749 if (!fread(&what, sizeof(what), 1, f))
760 eDebug("cutfile not found!");
764 m_event((iPlayableService*)this, evCuesheetChanged);
765 eDebug("eServiceDVD::loadCuesheet() pts=%lld",m_cue_pts);
769 void eServiceDVD::saveCuesheet()
771 eDebug("eServiceDVD::saveCuesheet()");
773 struct ddvd_resume resume_info;
774 ddvd_get_resume_pos(m_ddvdconfig, &resume_info);
776 if (resume_info.title)
778 struct ddvd_time info;
779 ddvd_get_last_time(m_ddvdconfig, &info);
781 pos = info.pos_hours * 3600;
782 pos += info.pos_minutes * 60;
783 pos += info.pos_seconds;
786 eDebug("ddvd_get_resume_pos resume_info.title=%d, chapter=%d, block=%lu, audio_id=%d, audio_lock=%d, spu_id=%d, spu_lock=%d (pts=%llu)",resume_info.title,resume_info.chapter,resume_info.block,resume_info.audio_id, resume_info.audio_lock, resume_info.spu_id, resume_info.spu_lock,m_cue_pts);
790 eDebug("we're in a menu or somewhere else funny. so save cuesheet with pts=0");
795 if ( m_ddvd_titlestring[0] != '\0' )
796 snprintf(filename, 128, "/home/root/dvd-%s.cuts", m_ddvd_titlestring);
798 snprintf(filename, 128, "%s/dvd.cuts", m_filename.c_str());
800 FILE *f = fopen(filename, "wb");
804 unsigned long long where;
807 #if BYTE_ORDER == BIG_ENDIAN
810 where = bswap_64(m_cue_pts);
813 fwrite(&where, sizeof(where), 1, f);
814 fwrite(&what, sizeof(what), 1, f);
817 fwrite(&resume_info, sizeof(struct ddvd_resume), 1, f);
818 fwrite(&what, sizeof(what), 1, f);
824 eAutoInitPtr<eServiceFactoryDVD> init_eServiceFactoryDVD(eAutoInitNumbers::service+1, "eServiceFactoryDVD");
829 Py_InitModule("servicedvd", NULL);