2 Copyright 2004-2005 Chris Tallon
4 This file is part of VOMP.
6 VOMP is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2 of the License, or
9 (at your option) any later version.
11 VOMP is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with VOMP; if not, write to the Free Software
18 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
36 int VideoMVP::init(UCHAR tformat)
38 if (initted) return 0;
41 if ((fdVideo = open("/dev/vdec_dev", O_WRONLY)) < 0) return 0;
43 if (!setFormat(tformat)) { shutdown(); return 0; }
44 if (!setConnection(COMPOSITERGB)) { shutdown(); return 0; }
45 if (!setAspectRatio(ASPECT4X3,12,11)) { shutdown(); return 0; }
46 if (!setMode(NORMAL)) { shutdown(); return 0; }
47 if (!setSource()) { shutdown(); return 0; }
48 if (!attachFrameBuffer()) { shutdown(); return 0; }
52 if (format == PAL) setLetterboxBorder("38");
53 else setLetterboxBorder("31");
61 void VideoMVP::setLetterboxBorder(char* border)
63 FILE* fdlbox = fopen("/proc/lbox_border", "w");
65 fputs(border, fdlbox);
69 int VideoMVP::setTVsize(UCHAR ttvsize)
73 // Override the aspect ratio usage, temporarily use to set the video chip mode
74 if (!setAspectRatio(tvsize,parx,pary)) { shutdown(); return 0; }
76 if ((fdVideo = open("/dev/vdec_dev", O_WRONLY)) < 0) return 0;
77 if (!setSource()) { shutdown(); return 0; }
78 if (!attachFrameBuffer()) { shutdown(); return 0; }
80 // Reopening the fd causes the scart aspect line to go back to 4:3
81 // Set this again to the same as the tv screen size
82 if (!setAspectRatio(tvsize,parx,pary)) { shutdown(); return 0; }
84 // mode == LETTERBOX is invalid if the TV is widescreen
85 if (tvsize == ASPECT16X9) setMode(NORMAL);
90 int VideoMVP::setDefaultAspect()
92 return setAspectRatio(tvsize,parx,pary);
95 int VideoMVP::shutdown()
97 if (!initted) return 0;
103 int VideoMVP::checkSCART()
105 // Returns 3 for SCART Composite out
106 // Returns 3 for SCART S-Video out
107 // Returns 2 for SCART RGB out
108 // Returns 3 for SCART not plugged in
110 // So, as you can have RGB and composite out simultaneously,
111 // and it can't detect S-Video, what is the point of this?
114 if (ioctl(fdVideo, AV_CHK_SCART, &scart) != 0) return -10;
119 int VideoMVP::setFormat(UCHAR tformat)
121 if (!initted) return 0;
122 if ((tformat != PAL) && (tformat != NTSC)) return 0;
125 if (ioctl(fdVideo, AV_SET_VID_DISP_FMT, format) != 0) return 0;
141 int VideoMVP::setConnection(UCHAR tconnection)
143 if (!initted) return 0;
144 if ((tconnection != COMPOSITERGB) && (tconnection != SVIDEO)) return 0;
145 connection = tconnection;
147 if (ioctl(fdVideo, AV_SET_VID_OUTPUT, connection) != 0) return 0;
151 int VideoMVP::setAspectRatio(UCHAR taspectRatio, int tparx,int tpary)
153 if (!initted) return 0;
156 if ((taspectRatio != ASPECT4X3) && (taspectRatio != ASPECT16X9)) return 0;
157 aspectRatio = taspectRatio;
159 Log::getInstance()->log("Video", Log::DEBUG, "Setting aspect to %i", aspectRatio);
161 if (ioctl(fdVideo, AV_SET_VID_RATIO, aspectRatio) != 0) return 0;
165 int VideoMVP::setMode(UCHAR tmode)
167 if (!initted) return 0;
169 if ((tmode == LETTERBOX) && (tvsize == ASPECT16X9)) return 0; // invalid mode
171 if ((tmode != NORMAL) && (tmode != LETTERBOX) && (tmode != UNKNOWN2) && (tmode != QUARTER) && (tmode != EIGHTH)
172 && (tmode != ZOOM) && (tmode != UNKNOWN6)) return 0;
175 if (ioctl(fdVideo, AV_SET_VID_MODE, mode) != 0) return 0;
179 int VideoMVP::signalOff()
181 if (ioctl(fdVideo, AV_SET_VID_DENC, 0) != 0) return 0;
185 int VideoMVP::signalOn()
187 if (ioctl(fdVideo, AV_SET_VID_DENC, 1) != 0) return 0;
191 int VideoMVP::setSource()
193 if (!initted) return 0;
195 // What does this do...
196 if (ioctl(fdVideo, AV_SET_VID_SRC, 1) != 0) return 0;
200 int VideoMVP::setPosition(int x, int y)
202 if (!initted) return 0;
204 // vid_pos_regs_t pos_d;
208 vid_pos_regs_t pos_d;
210 memset(&pos_d, 0, sizeof(pos_d));
235 pos_d.y = 100; // Top left X
236 pos_d.x = 50; // Top left Y
244 if (ioctl(fdVideo, AV_SET_VID_POSITION, &pos_d) != 0) return 0;
250 if (!initted) return 0;
252 if (ioctl(fdVideo, AV_SET_VID_SYNC, 2) != 0) return 0;
258 if (!initted) return 0;
260 if (ioctl(fdVideo, AV_SET_VID_PLAY, 0) != 0) return 0;
266 if (!initted) return 0;
268 if (ioctl(fdVideo, AV_SET_VID_STOP, 0) != 0) return 0;
272 int VideoMVP::reset()
274 if (!initted) return 0;
276 if (ioctl(fdVideo, AV_SET_VID_RESET, 0x11) != 0) return 0;
280 int VideoMVP::pause()
282 if (!initted) return 0;
284 if (ioctl(fdVideo, AV_SET_VID_PAUSE, 0) != 0) return 0;
288 int VideoMVP::unPause() // FIXME get rid - same as play!!
290 if (!initted) return 0;
291 if (ioctl(fdVideo, AV_SET_VID_PLAY, 0) != 0) return 0;
295 int VideoMVP::fastForward()
297 if (!initted) return 0;
299 if (ioctl(fdVideo, AV_SET_VID_FFWD, 1) != 0) return 0;
303 int VideoMVP::unFastForward()
305 if (!initted) return 0;
307 // if (ioctl(fdVideo, AV_SET_VID_RESET, 0x11) != 0) return 0; // don't need this.
309 if (ioctl(fdVideo, AV_SET_VID_PLAY, 0) != 0) return 0;
313 int VideoMVP::attachFrameBuffer()
315 if (!initted) return 0;
317 if (ioctl(fdVideo, AV_SET_VID_FB, 0) != 0) return 0;
321 int VideoMVP::blank(void)
323 if (ioctl(fdVideo, AV_SET_VID_FB, 1) != 0) return 0;
324 if (ioctl(fdVideo, AV_SET_VID_FB, 0) != 0) return 0;
328 ULLONG VideoMVP::getCurrentTimestamp()
330 sync_data_t timestamps;
331 if (ioctl(fdVideo, AV_GET_VID_TIMESTAMPS, ×tamps) == 0)
333 // FIXME are these the right way around?
335 timestamps.stc = (timestamps.stc >> 31 ) | (timestamps.stc & 1);
336 timestamps.pts = (timestamps.pts >> 31 ) | (timestamps.pts & 1);
338 return timestamps.stc;
346 ULONG VideoMVP::timecodeToFrameNumber(ULLONG timecode)
348 if (format == PAL) return (ULONG)(((double)timecode / (double)90000) * (double)25);
349 else return (ULONG)(((double)timecode / (double)90000) * (double)30);
358 // return ioctl(fdVideo, AV_SET_VID_STC, &stc);
365 int VideoMVP::test2()
371 void VideoMVP::PrepareMediaSample(const MediaPacketList& mplist,UINT samplepos)
373 MediaPacketList::const_iterator iter = mplist.begin();
374 deliver_start = iter->pos_buffer + samplepos;
375 mediapacket_len[0] = deliver_length = iter->length;
377 while (++iter != mplist.end() &&
378 iter->pos_buffer == deliver_start + deliver_length)
380 deliver_length += iter->length;
381 mediapacket_len[deliver_count] = iter->length;
383 if (deliver_length >= WRITE_LENGTH ||
384 deliver_count == WRITE_PACKETS) break;
388 UINT VideoMVP::DeliverMediaSample(UCHAR* buffer, UINT* samplepos)
390 int written = ::write(fdVideo, buffer + deliver_start, deliver_length);
391 if (written == (int)deliver_length) { *samplepos = 0; return deliver_count;}
392 if (written <= 0) return 0;
393 // Handle a partial write. Is this possible? Should we bother?
395 while ((written -= mediapacket_len[i]) >= 0) i++;
396 *samplepos = mediapacket_len[i] + written;
400 void VideoMVP::ResetTimeOffsets()
404 bool VideoMVP::displayIFrame(const UCHAR* buffer, UINT length)
406 write(fdVideo, buffer, length);