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 extensions.push_back("img");
36 sc->addServiceFactory(eServiceFactoryDVD::id, this, extensions);
40 eServiceFactoryDVD::~eServiceFactoryDVD()
42 ePtr<eServiceCenter> sc;
44 eServiceCenter::getPrivInstance(sc);
46 sc->removeServiceFactory(eServiceFactoryDVD::id);
49 DEFINE_REF(eServiceFactoryDVD)
52 RESULT eServiceFactoryDVD::play(const eServiceReference &ref, ePtr<iPlayableService> &ptr)
55 ptr = new eServiceDVD(ref);
59 RESULT eServiceFactoryDVD::record(const eServiceReference &/*ref*/, ePtr<iRecordableService> &ptr)
65 RESULT eServiceFactoryDVD::list(const eServiceReference &, ePtr<iListableService> &ptr)
72 RESULT eServiceFactoryDVD::info(const eServiceReference &/*ref*/, ePtr<iStaticServiceInformation> &ptr)
78 RESULT eServiceFactoryDVD::offlineOperations(const eServiceReference &, ePtr<iServiceOfflineOperations> &ptr)
86 DEFINE_REF(eServiceDVD);
88 eServiceDVD::eServiceDVD(eServiceReference ref):
89 m_ref(ref), m_ddvdconfig(ddvd_create()), m_subtitle_widget(0), m_state(stIdle),
90 m_current_trick(0), m_pump(eApp, 1), m_width(-1), m_height(-1),
91 m_aspect(-1), m_framerate(-1), m_progressive(-1)
93 int aspect = DDVD_16_9;
94 int policy = DDVD_PAN_SCAN;
95 int policy2 = DDVD_PAN_SCAN;
100 m_sn = eSocketNotifier::create(eApp, ddvd_get_messagepipe_fd(m_ddvdconfig), eSocketNotifier::Read|eSocketNotifier::Priority|eSocketNotifier::Error|eSocketNotifier::Hungup);
101 eDebug("SERVICEDVD construct!");
103 ddvd_set_dvd_path(m_ddvdconfig, ref.path.c_str());
104 ddvd_set_ac3thru(m_ddvdconfig, 0);
106 std::string ddvd_language;
107 if (!ePythonConfigQuery::getConfigValue("config.osd.language", ddvd_language))
108 ddvd_set_language(m_ddvdconfig, (ddvd_language.substr(0,2)).c_str());
110 int fd = open("/proc/stb/video/aspect", O_RDONLY);
113 rd = read(fd, tmp, 255);
114 if (rd > 2 && !strncmp(tmp, "4:3", 3))
116 else if (rd > 4 && !strncmp(tmp, "16:10", 5))
121 fd = open("/proc/stb/video/policy", O_RDONLY);
124 rd = read(fd, tmp, 255);
125 if (rd > 6 && !strncmp(tmp, "bestfit", 7))
126 policy = DDVD_JUSTSCALE;
127 else if (rd > 8 && !strncmp(tmp, "letterbox", 9))
128 policy = DDVD_LETTERBOX;
132 #ifdef DDVD_SUPPORTS_16_10_SCALING
133 fd = open("/proc/stb/video/policy2", O_RDONLY);
136 rd = read(fd, tmp, 255);
137 if (rd > 6 && !strncmp(tmp, "bestfit", 7))
138 policy2 = DDVD_JUSTSCALE;
139 else if (rd > 8 && !strncmp(tmp, "letterbox", 9))
140 policy2 = DDVD_LETTERBOX;
143 ddvd_set_video_ex(m_ddvdconfig, aspect, policy, policy2, DDVD_PAL /*unused*/);
145 ddvd_set_video(m_ddvdconfig, aspect, policy, DDVD_PAL /*unused*/);
146 #warning please update libdreamdvd for 16:10 scaling support!
149 CONNECT(m_sn->activated, eServiceDVD::gotMessage);
150 CONNECT(m_pump.recv_msg, eServiceDVD::gotThreadMessage);
151 strcpy(m_ddvd_titlestring,"");
156 void eServiceDVD::gotThreadMessage(const int &msg)
160 case 1: // thread stopped
162 m_event(this, evStopped);
167 void eServiceDVD::gotMessage(int /*what*/)
169 switch(ddvd_get_next_message(m_ddvdconfig,1))
171 case DDVD_COLORTABLE_UPDATE:
174 struct ddvd_color ctmp[4];
175 ddvd_get_last_colortable(ddvdconfig, ctmp);
179 rd1[252+i]=ctmp[i].red;
180 bl1[252+i]=ctmp[i].blue;
181 gn1[252+i]=ctmp[i].green;
182 tr1[252+i]=ctmp[i].trans;
185 if(ioctl(fb, FBIOPUTCMAP, &colormap) == -1)
187 printf("Framebuffer: <FBIOPUTCMAP failed>\n");
191 eDebug("no support for 8bpp framebuffer in dvdplayer yet!");
194 case DDVD_SCREEN_UPDATE:
195 eDebug("DVD_SCREEN_UPDATE!");
196 if (m_subtitle_widget) {
198 ddvd_get_last_blit_area(m_ddvdconfig, &x1, &x2, &y1, &y2);
200 int x_offset = 0, y_offset = 0, width = 720, height = 576;
202 #ifdef DDVD_SUPPORTS_GET_BLIT_DESTINATION
203 ddvd_get_blit_destination(m_ddvdconfig, &x_offset, &y_offset, &width, &height);
204 eDebug("values got from ddvd: %d %d %d %d", x_offset, y_offset, width, height);
205 y_offset = -y_offset;
206 width -= x_offset * 2;
207 height -= y_offset * 2;
209 eRect dest(x_offset, y_offset, width, height);
211 if (dest.width() && dest.height())
212 m_subtitle_widget->setPixmap(m_pixmap, eRect(x1, y1, (x2-x1)+1, (y2-y1)+1), dest);
215 case DDVD_SHOWOSD_STATE_PLAY:
217 eDebug("DVD_SHOWOSD_STATE_PLAY!");
219 m_event(this, evUser+1);
222 case DDVD_SHOWOSD_STATE_PAUSE:
224 eDebug("DVD_SHOWOSD_STATE_PAUSE!");
225 m_event(this, evUser+2);
228 case DDVD_SHOWOSD_STATE_FFWD:
230 eDebug("DVD_SHOWOSD_STATE_FFWD!");
231 m_event(this, evUser+3);
234 case DDVD_SHOWOSD_STATE_FBWD:
236 eDebug("DVD_SHOWOSD_STATE_FBWD!");
237 m_event(this, evUser+4);
240 case DDVD_SHOWOSD_STRING:
242 eDebug("DVD_SHOWOSD_STRING!");
243 m_event(this, evUser+5);
246 case DDVD_SHOWOSD_AUDIO:
248 eDebug("DVD_SHOWOSD_STRING!");
249 m_event(this, evUser+6);
252 case DDVD_SHOWOSD_SUBTITLE:
254 eDebug("DVD_SHOWOSD_SUBTITLE!");
255 m_event((iPlayableService*)this, evUpdatedInfo);
256 m_event(this, evUser+7);
259 case DDVD_EOF_REACHED:
260 eDebug("DVD_EOF_REACHED!");
261 m_event(this, evEOF);
263 case DDVD_SOF_REACHED:
264 eDebug("DVD_SOF_REACHED!");
265 m_event(this, evSOF);
267 case DDVD_SHOWOSD_ANGLE:
270 ddvd_get_angle_info(m_ddvdconfig, ¤t, &num);
271 eDebug("DVD_ANGLE_INFO: %d / %d", current, num);
272 m_event(this, evUser+13);
275 case DDVD_SHOWOSD_TIME:
277 static struct ddvd_time last_info;
278 struct ddvd_time info;
279 // eDebug("DVD_SHOWOSD_TIME!");
280 ddvd_get_last_time(m_ddvdconfig, &info);
281 if ( info.pos_chapter != last_info.pos_chapter )
282 m_event(this, evUser+8); // chapterUpdated
283 if ( info.pos_title != last_info.pos_title )
284 m_event(this, evUser+9); // titleUpdated
285 memcpy(&last_info, &info, sizeof(struct ddvd_time));
288 case DDVD_SHOWOSD_TITLESTRING:
290 ddvd_get_title_string(m_ddvdconfig, m_ddvd_titlestring);
291 eDebug("DDVD_SHOWOSD_TITLESTRING: %s",m_ddvd_titlestring);
295 m_event(this, evStart);
298 case DDVD_MENU_OPENED:
299 eDebug("DVD_MENU_OPENED!");
301 m_event(this, evSeekableStatusChanged);
302 m_event(this, evUser+11);
304 case DDVD_MENU_CLOSED:
305 eDebug("DVD_MENU_CLOSED!");
307 m_event(this, evSeekableStatusChanged);
308 m_event(this, evUser+12);
310 #ifdef DDVD_SUPPORTS_PICTURE_INFO
311 case DDVD_SIZE_CHANGED:
313 int changed = m_width != -1 && m_height != -1 && m_aspect != -1;
314 ddvd_get_last_size(m_ddvdconfig, &m_width, &m_height, &m_aspect);
316 m_event((iPlayableService*)this, evVideoSizeChanged);
319 case DDVD_PROGRESSIVE_CHANGED:
321 int changed = m_progressive != -1;
322 ddvd_get_last_progressive(m_ddvdconfig, &m_progressive);
324 m_event((iPlayableService*)this, evVideoProgressiveChanged);
327 case DDVD_FRAMERATE_CHANGED:
329 int changed = m_framerate != -1;
330 ddvd_get_last_framerate(m_ddvdconfig, &m_framerate);
332 m_event((iPlayableService*)this, evVideoFramerateChanged);
341 eServiceDVD::~eServiceDVD()
343 eDebug("SERVICEDVD destruct!");
346 ddvd_close(m_ddvdconfig);
350 RESULT eServiceDVD::connectEvent(const Slot2<void,iPlayableService*,int> &event, ePtr<eConnection> &connection)
352 connection = new eConnection((iPlayableService*)this, m_event.connect(event));
356 RESULT eServiceDVD::start()
358 ASSERT(m_state == stIdle);
360 eDebug("eServiceDVD starting");
361 // m_event(this, evStart);
365 RESULT eServiceDVD::stop()
367 ASSERT(m_state != stIdle);
368 if (m_state == stStopped)
370 eDebug("DVD: stop %s", m_ref.path.c_str());
372 ddvd_send_key(m_ddvdconfig, DDVD_KEY_EXIT);
377 RESULT eServiceDVD::setTarget(int /*target*/)
382 RESULT eServiceDVD::pause(ePtr<iPauseableService> &ptr)
388 RESULT eServiceDVD::seek(ePtr<iSeekableService> &ptr)
394 RESULT eServiceDVD::subtitle(ePtr<iSubtitleOutput> &ptr)
400 RESULT eServiceDVD::keys(ePtr<iServiceKeys> &ptr)
407 RESULT eServiceDVD::setSlowMotion(int /*ratio*/)
412 RESULT eServiceDVD::setFastForward(int trick)
414 eDebug("setTrickmode(%d)", trick);
415 while (m_current_trick > trick && m_current_trick != -64)
417 ddvd_send_key(m_ddvdconfig, DDVD_KEY_FBWD);
418 if (m_current_trick == 0)
419 m_current_trick = -2;
420 else if (m_current_trick > 0)
422 m_current_trick /= 2;
423 if (abs(m_current_trick) == 1)
427 m_current_trick *= 2;
429 while (m_current_trick < trick && m_current_trick != 64)
431 ddvd_send_key(m_ddvdconfig, DDVD_KEY_FFWD);
432 if (m_current_trick == 0)
434 else if (m_current_trick < 0)
436 m_current_trick /= 2;
437 if (abs(m_current_trick) == 1)
441 m_current_trick *= 2;
446 RESULT eServiceDVD::pause()
448 eDebug("set pause!\n");
449 ddvd_send_key(m_ddvdconfig, DDVD_KEY_PAUSE);
453 RESULT eServiceDVD::unpause()
455 eDebug("set unpause!\n");
456 ddvd_send_key(m_ddvdconfig, DDVD_KEY_PLAY);
460 void eServiceDVD::thread()
462 eDebug("eServiceDVD dvd thread started");
464 ddvd_run(m_ddvdconfig);
467 void eServiceDVD::thread_finished()
469 eDebug("eServiceDVD dvd thread finished");
470 m_pump.send(1); // inform main thread
473 RESULT eServiceDVD::info(ePtr<iServiceInformation>&i)
479 RESULT eServiceDVD::getName(std::string &name)
481 if ( m_ddvd_titlestring[0] != '\0' )
482 name = m_ddvd_titlestring;
484 if ( !m_ref.name.empty() )
491 int eServiceDVD::getInfo(int w)
495 case sCurrentChapter:
497 struct ddvd_time info;
498 ddvd_get_last_time(m_ddvdconfig, &info);
499 return info.pos_chapter;
503 struct ddvd_time info;
504 ddvd_get_last_time(m_ddvdconfig, &info);
505 return info.end_chapter;
509 struct ddvd_time info;
510 ddvd_get_last_time(m_ddvdconfig, &info);
511 return info.pos_title;
515 struct ddvd_time info;
516 ddvd_get_last_time(m_ddvdconfig, &info);
517 return info.end_title;
519 case sTXTPID: // we abuse HAS_TELEXT icon in InfoBar to signalize subtitles status
523 ddvd_get_last_spu(m_ddvdconfig, &spu_id, &spu_lang);
529 return resIsPyObject;
530 #ifdef DDVD_SUPPORTS_PICTURE_INFO
538 return m_progressive;
547 std::string eServiceDVD::getInfoString(int w)
552 return m_ref.toString();
554 eDebug("unhandled getInfoString(%d)", w);
559 PyObject *eServiceDVD::getInfoObject(int w)
565 ePyObject tuple = PyTuple_New(3);
566 int audio_id,audio_type;
568 ddvd_get_last_audio(m_ddvdconfig, &audio_id, &audio_lang, &audio_type);
569 char audio_string[3]={audio_lang >> 8, audio_lang, 0};
570 PyTuple_SetItem(tuple, 0, PyInt_FromLong(audio_id+1));
571 PyTuple_SetItem(tuple, 1, PyString_FromString(audio_string));
575 PyTuple_SetItem(tuple, 2, PyString_FromString("MPEG"));
578 PyTuple_SetItem(tuple, 2, PyString_FromString("AC3"));
581 PyTuple_SetItem(tuple, 2, PyString_FromString("DTS"));
584 PyTuple_SetItem(tuple, 2, PyString_FromString("LPCM"));
587 PyTuple_SetItem(tuple, 2, PyString_FromString(""));
593 ePyObject tuple = PyTuple_New(2);
596 ddvd_get_last_spu(m_ddvdconfig, &spu_id, &spu_lang);
597 char spu_string[3]={spu_lang >> 8, spu_lang, 0};
600 PyTuple_SetItem(tuple, 0, PyInt_FromLong(0));
601 PyTuple_SetItem(tuple, 1, PyString_FromString(""));
605 PyTuple_SetItem(tuple, 0, PyInt_FromLong(spu_id+1));
606 PyTuple_SetItem(tuple, 1, PyString_FromString(spu_string));
612 ePyObject tuple = PyTuple_New(2);
614 ddvd_get_angle_info(m_ddvdconfig, ¤t, &num);
615 PyTuple_SetItem(tuple, 0, PyInt_FromLong(current));
616 PyTuple_SetItem(tuple, 1, PyInt_FromLong(num));
621 eDebug("unhandled getInfoObject(%d)", w);
626 RESULT eServiceDVD::enableSubtitles(eWidget *parent, SWIG_PYOBJECT(ePyObject) /*entry*/)
628 delete m_subtitle_widget;
630 m_subtitle_widget = new eSubtitleWidget(parent);
631 m_subtitle_widget->resize(parent->size());
633 eSize size = eSize(720, 576);
637 m_pixmap = new gPixmap(size, 32, 1); /* allocate accel surface (if possible) */
638 #ifdef DDVD_SUPPORTS_GET_BLIT_DESTINATION
639 ddvd_set_lfb_ex(m_ddvdconfig, (unsigned char *)m_pixmap->surface->data, size.width(), size.height(), 4, size.width()*4, 1);
641 ddvd_set_lfb(m_ddvdconfig, (unsigned char *)m_pixmap->surface->data, size.width(), size.height(), 4, size.width()*4);
642 #warning please update libdreamdvd for fast scaling
644 run(); // start the thread
647 m_subtitle_widget->setZPosition(-1);
648 m_subtitle_widget->show();
653 RESULT eServiceDVD::disableSubtitles(eWidget */*parent*/)
655 delete m_subtitle_widget;
656 m_subtitle_widget = 0;
660 PyObject *eServiceDVD::getSubtitleList()
662 eDebug("eServiceDVD::getSubtitleList nyi");
666 PyObject *eServiceDVD::getCachedSubtitle()
668 eDebug("eServiceDVD::getCachedSubtitle nyi");
672 RESULT eServiceDVD::getLength(pts_t &len)
674 // eDebug("eServiceDVD::getLength");
675 struct ddvd_time info;
676 ddvd_get_last_time(m_ddvdconfig, &info);
677 len = info.end_hours * 3600;
678 len += info.end_minutes * 60;
679 len += info.end_seconds;
684 RESULT eServiceDVD::seekTo(pts_t to)
686 eDebug("eServiceDVD::seekTo(%lld)",to);
689 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);
690 ddvd_set_resume_pos(m_ddvdconfig, m_resume_info);
695 RESULT eServiceDVD::seekRelative(int direction, pts_t to)
697 int seconds = to / 90000;
698 seconds *= direction;
699 eDebug("seekRelative %d %d", direction, seconds);
700 ddvd_skip_seconds(m_ddvdconfig, seconds);
704 RESULT eServiceDVD::getPlayPosition(pts_t &pos)
706 struct ddvd_time info;
707 ddvd_get_last_time(m_ddvdconfig, &info);
708 pos = info.pos_hours * 3600;
709 pos += info.pos_minutes * 60;
710 pos += info.pos_seconds;
711 // eDebug("getPlayPosition %lld", pos);
716 RESULT eServiceDVD::seekTitle(int title)
718 eDebug("setTitle %d", title);
719 ddvd_set_title(m_ddvdconfig, title);
723 RESULT eServiceDVD::seekChapter(int chapter)
725 eDebug("setChapter %d", chapter);
727 ddvd_set_chapter(m_ddvdconfig, chapter);
731 RESULT eServiceDVD::setTrickmode(int /*trick*/)
736 RESULT eServiceDVD::isCurrentlySeekable()
738 return m_state == stRunning ? 3 : 0;
741 RESULT eServiceDVD::keyPressed(int key)
745 case iServiceKeys::keyLeft:
746 ddvd_send_key(m_ddvdconfig, DDVD_KEY_LEFT);
748 case iServiceKeys::keyRight:
749 ddvd_send_key(m_ddvdconfig, DDVD_KEY_RIGHT);
751 case iServiceKeys::keyUp:
752 ddvd_send_key(m_ddvdconfig, DDVD_KEY_UP);
754 case iServiceKeys::keyDown:
755 ddvd_send_key(m_ddvdconfig, DDVD_KEY_DOWN);
757 case iServiceKeys::keyOk:
758 ddvd_send_key(m_ddvdconfig, DDVD_KEY_OK);
760 case iServiceKeys::keyUser:
761 ddvd_send_key(m_ddvdconfig, DDVD_KEY_AUDIO);
763 case iServiceKeys::keyUser+1:
764 ddvd_send_key(m_ddvdconfig, DDVD_KEY_SUBTITLE);
766 case iServiceKeys::keyUser+2:
767 ddvd_send_key(m_ddvdconfig, DDVD_KEY_AUDIOMENU);
769 case iServiceKeys::keyUser+3:
770 ddvd_send_key(m_ddvdconfig, DDVD_KEY_NEXT_CHAPTER);
772 case iServiceKeys::keyUser+4:
773 ddvd_send_key(m_ddvdconfig, DDVD_KEY_PREV_CHAPTER);
775 case iServiceKeys::keyUser+5:
776 ddvd_send_key(m_ddvdconfig, DDVD_KEY_NEXT_TITLE);
778 case iServiceKeys::keyUser+6:
779 ddvd_send_key(m_ddvdconfig, DDVD_KEY_PREV_TITLE);
781 case iServiceKeys::keyUser+7:
782 ddvd_send_key(m_ddvdconfig, DDVD_KEY_MENU);
784 case iServiceKeys::keyUser+8:
785 ddvd_send_key(m_ddvdconfig, DDVD_KEY_ANGLE);
793 RESULT eServiceDVD::cueSheet(ePtr<iCueSheet> &ptr)
804 PyObject *eServiceDVD::getCutList()
806 ePyObject list = PyList_New(1);
807 ePyObject tuple = PyTuple_New(2);
808 PyTuple_SetItem(tuple, 0, PyLong_FromLongLong(m_cue_pts));
809 PyTuple_SetItem(tuple, 1, PyInt_FromLong(3));
810 PyList_SetItem(list, 0, tuple);
814 void eServiceDVD::setCutList(ePyObject /*list*/)
818 void eServiceDVD::setCutListEnable(int /*enable*/)
822 void eServiceDVD::loadCuesheet()
825 if ( m_ddvd_titlestring[0] != '\0' )
826 snprintf(filename, 128, "/home/root/dvd-%s.cuts", m_ddvd_titlestring);
828 snprintf(filename, 128, "%s/dvd.cuts", m_ref.path.c_str());
830 eDebug("eServiceDVD::loadCuesheet() filename=%s",filename);
832 FILE *f = fopen(filename, "rb");
836 unsigned long long where;
839 if (!fread(&where, sizeof(where), 1, f))
841 if (!fread(&what, sizeof(what), 1, f))
843 #if BYTE_ORDER == LITTLE_ENDIAN
844 where = bswap_64(where);
848 if (!fread(&m_resume_info, sizeof(struct ddvd_resume), 1, f))
850 if (!fread(&what, sizeof(what), 1, f))
861 eDebug("cutfile not found!");
865 m_event((iPlayableService*)this, evCuesheetChanged);
866 eDebug("eServiceDVD::loadCuesheet() pts=%lld",m_cue_pts);
870 void eServiceDVD::saveCuesheet()
872 eDebug("eServiceDVD::saveCuesheet()");
874 struct ddvd_resume resume_info;
875 ddvd_get_resume_pos(m_ddvdconfig, &resume_info);
877 if (resume_info.title)
879 struct ddvd_time info;
880 ddvd_get_last_time(m_ddvdconfig, &info);
882 pos = info.pos_hours * 3600;
883 pos += info.pos_minutes * 60;
884 pos += info.pos_seconds;
887 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);
891 eDebug("we're in a menu or somewhere else funny. so save cuesheet with pts=0");
896 if ( m_ddvd_titlestring[0] != '\0' )
897 snprintf(filename, 128, "/home/root/dvd-%s.cuts", m_ddvd_titlestring);
899 snprintf(filename, 128, "%s/dvd.cuts", m_ref.path.c_str());
901 FILE *f = fopen(filename, "wb");
905 unsigned long long where;
908 #if BYTE_ORDER == BIG_ENDIAN
911 where = bswap_64(m_cue_pts);
914 fwrite(&where, sizeof(where), 1, f);
915 fwrite(&what, sizeof(what), 1, f);
918 fwrite(&resume_info, sizeof(struct ddvd_resume), 1, f);
919 fwrite(&what, sizeof(what), 1, f);
925 eAutoInitPtr<eServiceFactoryDVD> init_eServiceFactoryDVD(eAutoInitNumbers::service+1, "eServiceFactoryDVD");
930 Py_InitModule("servicedvd", NULL);