switch_ratio may not work with some filter chains
[mplayer/greg.git] / libvo / vo_dga.c
blob30bf24f37be17b00af3e63d104e5392c8a697208
1 /*
3 * X11 DGA Interface
5 * Copyright ( C ) 2001, Andreas Ackermann. All Rights Reserved.
7 * <acki@acki-netz.de>
9 * Sourceforge username: acki2
13 #include <stdio.h>
14 #include <stdlib.h>
15 #include <string.h>
16 #include <errno.h>
18 #include "config.h"
19 #include "video_out.h"
20 #include "video_out_internal.h"
21 #include "aspect.h"
22 #include "x11_common.h"
23 #include "fastmemcpy.h"
24 #include "../mp_msg.h"
26 #include <X11/Xlib.h>
27 #include <X11/extensions/xf86dga.h>
29 #ifdef HAVE_XF86VM
30 #include <X11/extensions/xf86vmode.h>
31 #endif
33 static vo_info_t info = {
34 #ifdef HAVE_DGA2
35 "DGA ( Direct Graphic Access V2.0 )",
36 #else
37 #ifdef HAVE_XF86VM
38 "DGA ( Direct Graphic Access V1.0+XF86VidModeExt. )",
39 #else
40 "DGA ( Direct Graphic Access V1.0 )",
41 #endif
42 #endif
43 "dga",
44 "Andreas Ackermann <acki@acki-netz.de>",
48 LIBVO_EXTERN(dga)
49 //------------------------------------------------------------------
50 //#define BITSPP (vo_dga_modes[vo_dga_active_mode].vdm_bitspp)
51 //#define BYTESPP (vo_dga_modes[vo_dga_active_mode].vdm_bytespp)
52 #define VO_DGA_INVALID_RES 100000
53 #define HW_MODE (vo_dga_modes[vo_dga_hw_mode])
54 #define SRC_MODE (vo_dga_modes[vo_dga_src_mode])
55 struct vd_modes
57 int vdm_mplayer_depth;
58 int vdm_supported;
59 int vdm_depth;
60 int vdm_bitspp;
61 int vdm_bytespp;
62 int vdm_rmask;
63 int vdm_gmask;
64 int vdm_bmask;
65 int vdm_hw_mode;
68 //------------------------------------------------------------------
70 static struct vd_modes vo_dga_modes[] = {
71 // these entries describe HW modes
72 // however, we use the same entries to tell mplayer what we support
73 // so the last two values describe, which HW mode to use and which conversion
74 // function to use for a mode that is not supported by HW
76 {0, 0, 0, 0, 0, 0, 0, 0, 0,},
77 {15, 0, 15, 16, 2, 0x7c00, 0x03e0, 0x001f, 2,},
78 {16, 0, 16, 16, 2, 0xf800, 0x07e0, 0x001f, 2,},
79 {24, 0, 24, 24, 3, 0xff0000, 0x00ff00, 0x0000ff, 4},
80 {32, 0, 24, 32, 4, 0x00ff0000, 0x0000ff00, 0x000000ff, 4}
83 static int vo_dga_mode_num =
84 sizeof(vo_dga_modes) / sizeof(struct vd_modes);
86 // enable a HW mode (by description)
87 static int vd_EnableMode(int depth, int bitspp,
88 int rmask, int gmask, int bmask)
90 int i;
92 for (i = 1; i < vo_dga_mode_num; i++)
94 if (vo_dga_modes[i].vdm_depth == depth &&
95 vo_dga_modes[i].vdm_bitspp == bitspp &&
96 vo_dga_modes[i].vdm_rmask == rmask &&
97 vo_dga_modes[i].vdm_gmask == gmask &&
98 vo_dga_modes[i].vdm_bmask == bmask)
100 vo_dga_modes[i].vdm_supported = 1;
101 vo_dga_modes[i].vdm_hw_mode = i;
102 return i;
105 return 0;
108 static int vd_ModeEqual(int depth, int bitspp,
109 int rmask, int gmask, int bmask, int index)
111 return ((vo_dga_modes[index].vdm_depth == depth &&
112 vo_dga_modes[index].vdm_bitspp == bitspp &&
113 vo_dga_modes[index].vdm_rmask == rmask &&
114 vo_dga_modes[index].vdm_gmask == gmask &&
115 vo_dga_modes[index].vdm_bmask == bmask) ? 1 : 0);
119 // enable a HW mode (mplayer_depth decides which)
120 static int vd_ValidateMode(int mplayer_depth)
122 int i;
124 if (mplayer_depth == 0)
125 return 0;
126 for (i = 1; i < vo_dga_mode_num; i++)
128 if (vo_dga_modes[i].vdm_mplayer_depth == mplayer_depth)
130 vo_dga_modes[i].vdm_supported = 1;
131 vo_dga_modes[i].vdm_hw_mode = i;
132 return i;
135 return 0;
138 // do we support this mode? (not important whether native or conversion)
139 static int vd_ModeValid(int mplayer_depth)
141 int i;
143 if (mplayer_depth == 0)
144 return 0;
145 for (i = 1; i < vo_dga_mode_num; i++)
147 if (vo_dga_modes[i].vdm_mplayer_depth == mplayer_depth &&
148 vo_dga_modes[i].vdm_supported != 0)
150 return i;
153 return 0;
156 static char *vd_GetModeString(int index)
159 #define VO_DGA_MAX_STRING_LEN 100
160 static char stringbuf[VO_DGA_MAX_STRING_LEN];
162 stringbuf[VO_DGA_MAX_STRING_LEN - 1] = 0;
163 snprintf(stringbuf, VO_DGA_MAX_STRING_LEN - 2,
164 "depth=%d, bpp=%d, r=%06x, g=%06x, b=%06x, %s (-bpp %d)",
165 vo_dga_modes[index].vdm_depth,
166 vo_dga_modes[index].vdm_bitspp,
167 vo_dga_modes[index].vdm_rmask,
168 vo_dga_modes[index].vdm_gmask,
169 vo_dga_modes[index].vdm_bmask,
170 vo_dga_modes[index].
171 vdm_supported ? "native" : "not supported",
172 vo_dga_modes[index].vdm_mplayer_depth);
173 return stringbuf;
176 //-----------------------------------------------------------------
178 #if defined(HAVE_XF86VM) && !defined(HAVE_DGA2)
179 static XF86VidModeModeInfo **vo_dga_vidmodes = NULL;
180 #endif
183 static int vo_dga_src_format;
184 static int vo_dga_width; // bytes per line in framebuffer
185 static int vo_dga_vp_width; // visible pixels per line in
187 // framebuffer
188 static int vo_dga_vp_height; // visible lines in framebuffer
189 static int vo_dga_is_running = 0;
190 static int vo_dga_src_width; // width of video in pixels
191 static int vo_dga_src_height; // height of video in pixels
192 static int vo_dga_src_offset = 0; // offset in src
193 static int vo_dga_vp_offset = 0; // offset in dest
194 static int vo_dga_bytes_per_line; // bytes per line to copy
195 static int vo_dga_vp_skip; // dto. for dest
196 static int vo_dga_lines; // num of lines to copy
197 static int vo_dga_hw_mode = 0; // index in mode list that is actually
199 // used by framebuffer
200 static int vo_dga_src_mode = 0; // index in mode list that is used by
202 // codec
203 static int vo_dga_XServer_mode = 0; // index in mode list for resolution
205 #ifdef HAVE_DGA2
206 static XDGAMode *vo_modelines;
207 static int vo_modecount;
208 #endif
210 #define MAX_NR_VIDEO_BUFFERS 3
212 #define CURRENT_VIDEO_BUFFER \
213 (vo_dga_video_buffer[vo_dga_current_video_buffer])
215 static int vo_dga_nr_video_buffers; // Total number of frame buffers.
216 static int vo_dga_current_video_buffer; // Buffer available for rendering.
218 static struct video_buffer
220 int y;
221 uint8_t *data;
222 } vo_dga_video_buffer[MAX_NR_VIDEO_BUFFERS];
224 /* saved src and dst dimensions for SwScaler */
225 static unsigned int scale_srcW = 0,
226 scale_dstW = 0, scale_srcH = 0, scale_dstH = 0;
228 //---------------------------------------------------------
230 static void draw_alpha(int x0, int y0, int w, int h, unsigned char *src,
231 unsigned char *srca, int stride)
234 char *d;
235 unsigned int offset;
236 unsigned int buffer_stride;
238 offset = vo_dga_width * y0 + x0;
239 buffer_stride = vo_dga_width;
240 d = CURRENT_VIDEO_BUFFER.data + vo_dga_vp_offset;
242 switch (HW_MODE.vdm_mplayer_depth)
245 case 32:
246 vo_draw_alpha_rgb32(w, h, src, srca, stride, d + 4 * offset,
247 4 * buffer_stride);
248 break;
249 case 24:
250 vo_draw_alpha_rgb24(w, h, src, srca, stride, d + 3 * offset,
251 3 * buffer_stride);
252 break;
253 case 15:
254 vo_draw_alpha_rgb15(w, h, src, srca, stride, d + 2 * offset,
255 2 * buffer_stride);
256 break;
257 case 16:
258 vo_draw_alpha_rgb16(w, h, src, srca, stride, d + 2 * offset,
259 2 * buffer_stride);
260 break;
265 //---------------------------------------------------------
270 // quick & dirty - for debugging only
271 #if 0
272 static void fillblock(char *strt, int yoff, int lines, int val)
274 char *i;
276 for (i = strt + yoff * vo_dga_width * HW_MODE.vdm_bytespp;
277 i < strt + (lines + yoff) * vo_dga_width * HW_MODE.vdm_bytespp;)
279 *i++ = val;
282 #endif
284 //---------------------------------------------------------
286 static uint32_t draw_frame(uint8_t * src[])
289 int vp_skip = vo_dga_vp_skip;
290 int numlines = vo_dga_lines;
292 char *s, *d;
294 s = *src;
295 d = CURRENT_VIDEO_BUFFER.data + vo_dga_vp_offset;
297 mem2agpcpy_pic(d, s,
298 vo_dga_bytes_per_line,
299 numlines,
300 vo_dga_bytes_per_line + vo_dga_vp_skip,
301 vo_dga_bytes_per_line);
303 // DBG-COde
305 #if 0
306 d = CURRENT_VIDEO_BUFFER.data + vo_dga_vp_offset;
307 fillblock(d, 0, 10, 0x800000ff);
308 fillblock(d, 10, 10, 0x8000ff00);
309 fillblock(d, 20, 10, 0x80ff0000);
310 fillblock(d, 30, 10, 0xff0000ff);
311 fillblock(d, 40, 10, 0x800000ff);
312 fillblock(d, 50, 10, 0x0f0000ff);
313 #endif
314 return 0;
317 //---------------------------------------------------------
319 static void check_events(void)
321 vo_x11_check_events(mDisplay);
324 //---------------------------------------------------------
326 #include "sub.h"
328 static void draw_osd(void)
330 vo_draw_text(vo_dga_src_width, vo_dga_src_height, draw_alpha);
333 static void switch_video_buffers(void)
335 vo_dga_current_video_buffer =
336 (vo_dga_current_video_buffer + 1) % vo_dga_nr_video_buffers;
339 static void flip_page(void)
341 if (1 < vo_dga_nr_video_buffers)
343 #ifdef HAVE_DGA2
344 XDGASetViewport(mDisplay, mScreen,
345 0, CURRENT_VIDEO_BUFFER.y, XDGAFlipRetrace);
346 #else
347 XF86DGASetViewPort(mDisplay, mScreen, 0, CURRENT_VIDEO_BUFFER.y);
348 #endif
349 switch_video_buffers();
353 //---------------------------------------------------------
355 static uint32_t draw_slice(uint8_t * src[], int stride[],
356 int w, int h, int x, int y)
358 return 0;
361 //---------------------------------------------------------
363 static uint32_t query_format(uint32_t format)
366 if ((format & IMGFMT_BGR_MASK) == IMGFMT_BGR &&
367 vd_ModeValid(format & 0xff))
369 return VFCAP_CSP_SUPPORTED | VFCAP_CSP_SUPPORTED_BY_HW | VFCAP_OSD;
372 return 0;
375 //---------------------------------------------------------
377 static void uninit(void)
380 #ifdef HAVE_DGA2
381 XDGADevice *dgadevice;
382 #endif
384 if (!vo_config_count)
385 return;
387 if (vo_dga_is_running)
389 vo_dga_is_running = 0;
390 mp_msg(MSGT_VO, MSGL_V, "vo_dga: in uninit\n");
391 if (vo_grabpointer)
392 XUngrabPointer(mDisplay, CurrentTime);
393 XUngrabKeyboard(mDisplay, CurrentTime);
394 #ifdef HAVE_DGA2
395 XDGACloseFramebuffer(mDisplay, mScreen);
396 dgadevice = XDGASetMode(mDisplay, mScreen, 0);
397 if (dgadevice != NULL)
399 XFree(dgadevice);
401 #else
402 XF86DGADirectVideo(mDisplay, mScreen, 0);
403 // first disable DirectVideo and then switch mode back!
404 #ifdef HAVE_XF86VM
405 if (vo_dga_vidmodes != NULL)
407 int screen;
409 screen = XDefaultScreen(mDisplay);
410 mp_msg(MSGT_VO, MSGL_V,
411 "vo_dga: VidModeExt: Switching back..\n");
412 // seems some graphics adaptors need this more than once ...
413 XF86VidModeSwitchToMode(mDisplay, screen, vo_dga_vidmodes[0]);
414 XF86VidModeSwitchToMode(mDisplay, screen, vo_dga_vidmodes[0]);
415 XF86VidModeSwitchToMode(mDisplay, screen, vo_dga_vidmodes[0]);
416 XF86VidModeSwitchToMode(mDisplay, screen, vo_dga_vidmodes[0]);
417 XFree(vo_dga_vidmodes);
419 #endif
420 #endif
422 vo_x11_uninit();
426 //----------------------------------------------------------
427 // TODO: check for larger maxy value
428 // (useful for double buffering!!!)
430 static int check_res(int num, int x, int y, int bpp,
431 int new_x, int new_y, int new_vbi, int new_maxy,
432 int *old_x, int *old_y, int *old_vbi, int *old_maxy)
435 mp_msg(MSGT_VO, MSGL_V,
436 "vo_dga: (%3d) Trying %4d x %4d @ %3d Hz @ depth %2d ..", num,
437 new_x, new_y, new_vbi, bpp);
438 mp_msg(MSGT_VO, MSGL_V, "(old: %dx%d@%d).", *old_x, *old_y, *old_vbi);
439 if ((new_x >= x) && (new_y >= y) && (
440 // prefer a better resolution either in X or in Y
441 // as long as the other dimension is at least the same
443 // hmm ... MAYBE it would be more clever to focus on the
444 // x-resolution; I had 712x400 and 640x480 and the movie
445 // was 640x360; 640x480 would be the 'right thing' here
446 // but since 712x400 was queried first I got this one.
447 // I think there should be a cmd-line switch to let the
448 // user choose the mode he likes ... (acki2)
449 (((new_x < *old_x) &&
450 !(new_y > *old_y)) ||
451 ((new_y < *old_y) &&
452 !(new_x > *old_x)))
453 // but if we get an identical resolution choose
454 // the one with the lower refreshrate (saves bandwidth !!!)
455 // as long as it's above 50 Hz (acki2 on 30/3/2001)
457 (((new_x == *old_x) &&
458 (new_y == *old_y) &&
459 ((new_vbi >= *old_vbi
460 && *old_vbi < 50)
461 || (*old_vbi >= 50
462 && new_vbi < *old_vbi
463 && new_vbi >= 50))) ||
464 // if everything is equal, then use the mode with the lower
465 // stride
466 ((new_x == *old_x) &&
467 (new_y == *old_y) &&
468 (new_vbi == *old_vbi) &&
469 (new_maxy > *old_maxy)))))
471 *old_x = new_x;
472 *old_y = new_y;
473 *old_maxy = new_maxy;
474 *old_vbi = new_vbi;
475 mp_msg(MSGT_VO, MSGL_V, ".ok!!\n");
476 return 1;
477 } else
479 mp_msg(MSGT_VO, MSGL_V, ".no\n");
480 return 0;
486 //---------------------------------------------------------
488 static void init_video_buffers(uint8_t * buffer_base,
489 int view_port_height,
490 int bytes_per_scanline,
491 int max_view_port_y,
492 int use_multiple_buffers)
494 int bytes_per_buffer = view_port_height * bytes_per_scanline;
495 int i;
497 if (use_multiple_buffers)
498 vo_dga_nr_video_buffers = max_view_port_y / view_port_height;
499 else
500 vo_dga_nr_video_buffers = 1;
502 if (vo_dga_nr_video_buffers > MAX_NR_VIDEO_BUFFERS)
503 vo_dga_nr_video_buffers = MAX_NR_VIDEO_BUFFERS;
504 if (vo_dga_nr_video_buffers <= 0)
505 vo_dga_nr_video_buffers = 1;
507 vo_dga_current_video_buffer = 0;
509 for (i = 0; i < vo_dga_nr_video_buffers; i++)
511 vo_dga_video_buffer[i].y = i * view_port_height;
512 vo_dga_video_buffer[i].data = buffer_base + i * bytes_per_buffer;
514 // Clear video buffer.
515 memset(vo_dga_video_buffer[i].data, 0, bytes_per_buffer);
519 static uint32_t config(uint32_t width, uint32_t height,
520 uint32_t d_width, uint32_t d_height,
521 uint32_t flags, char *title, uint32_t format)
524 int x_off, y_off;
525 int wanted_width, wanted_height;
527 static unsigned char *vo_dga_base;
528 static int prev_width, prev_height;
530 #ifdef HAVE_DGA2
531 // needed to change DGA video mode
532 int mX = VO_DGA_INVALID_RES, mY = VO_DGA_INVALID_RES, mVBI =
533 100000, mMaxY = 0, i, j = 0;
534 int dga_modenum;
535 XDGAMode *modeline;
536 XDGADevice *dgadevice;
537 #else
538 #ifdef HAVE_XF86VM
539 unsigned int vm_event, vm_error;
540 unsigned int vm_ver, vm_rev;
541 int i, j = 0, have_vm = 0;
542 int mX = VO_DGA_INVALID_RES, mY = VO_DGA_INVALID_RES, mVBI =
543 100000, mMaxY = 0, dga_modenum;
544 #endif
545 int bank, ram;
546 #endif
548 vo_dga_src_format = format;
550 wanted_width = d_width;
551 wanted_height = d_height;
553 if (!wanted_height)
554 wanted_height = height;
555 if (!wanted_width)
556 wanted_width = width;
558 if (!vo_dbpp)
560 if ((format & IMGFMT_BGR_MASK) == IMGFMT_BGR)
562 vo_dga_src_mode = vd_ModeValid(format & 0xff);
564 } else
566 vo_dga_src_mode = vd_ModeValid(vo_dbpp);
568 vo_dga_hw_mode = SRC_MODE.vdm_hw_mode;
570 if (!vo_dga_src_mode)
572 mp_msg(MSGT_VO, MSGL_ERR, "vo_dga: unsupported video format!\n");
573 return 1;
576 vo_dga_vp_width = vo_screenwidth;
577 vo_dga_vp_height = vo_screenheight;
579 mp_msg(MSGT_VO, MSGL_V, "vo_dga: XServer res: %dx%d\n",
580 vo_dga_vp_width, vo_dga_vp_height);
582 // choose a suitable mode ...
584 #ifdef HAVE_DGA2
585 // Code to change the video mode added by Michael Graffam
586 // mgraffam@idsi.net
588 mp_msg(MSGT_VO, MSGL_V, "vo_dga: vo_modelines=%p, vo_modecount=%d\n",
589 vo_modelines, vo_modecount);
591 if (vo_modelines == NULL)
593 mp_msg(MSGT_VO, MSGL_ERR, "vo_dga: can't get modelines\n");
594 return 1;
597 mp_msg(MSGT_VO, MSGL_INFO,
598 "vo_dga: DGA 2.0 available :-) Can switch resolution AND depth!\n");
599 for (i = 0; i < vo_modecount; i++)
601 if (vd_ModeEqual(vo_modelines[i].depth,
602 vo_modelines[i].bitsPerPixel,
603 vo_modelines[i].redMask,
604 vo_modelines[i].greenMask,
605 vo_modelines[i].blueMask, vo_dga_hw_mode))
608 mp_msg(MSGT_VO, MSGL_V, "maxy: %4d, depth: %2d, %4dx%4d, ",
609 vo_modelines[i].maxViewportY, vo_modelines[i].depth,
610 vo_modelines[i].imageWidth,
611 vo_modelines[i].imageHeight);
612 if (check_res
613 (i, wanted_width, wanted_height, vo_modelines[i].depth,
614 vo_modelines[i].viewportWidth,
615 vo_modelines[i].viewportHeight,
616 (unsigned) vo_modelines[i].verticalRefresh,
617 vo_modelines[i].maxViewportY, &mX, &mY, &mVBI, &mMaxY))
618 j = i;
621 mp_msg(MSGT_VO, MSGL_INFO,
622 "vo_dga: Selected hardware mode %4d x %4d @ %3d Hz @ depth %2d, bitspp %2d.\n",
623 mX, mY, mVBI, HW_MODE.vdm_depth, HW_MODE.vdm_bitspp);
624 mp_msg(MSGT_VO, MSGL_INFO,
625 "vo_dga: Video parameters by codec: %3d x %3d, depth %2d, bitspp %2d.\n",
626 width, height, SRC_MODE.vdm_depth, SRC_MODE.vdm_bitspp);
627 vo_dga_vp_width = mX;
628 vo_dga_vp_height = mY;
630 if ((flags & 0x04) || (flags & 0x01))
631 { /* -zoom or -fs */
632 scale_dstW = (d_width + 7) & ~7;
633 scale_dstH = d_height;
634 scale_srcW = width;
635 scale_srcH = height;
636 aspect_save_screenres(mX, mY);
637 aspect_save_orig(scale_srcW, scale_srcH);
638 aspect_save_prescale(scale_dstW, scale_dstH);
639 if (flags & 0x01) /* -fs */
640 aspect(&scale_dstW, &scale_dstH, A_ZOOM);
641 else if (flags & 0x04) /* -fs */
642 aspect(&scale_dstW, &scale_dstH, A_NOZOOM);
643 mp_msg(MSGT_VO, MSGL_INFO,
644 "vo_dga: Aspect corrected size for SwScaler: %4d x %4d.\n",
645 scale_dstW, scale_dstH);
646 /* XXX this is a hack, but I'm lazy ;-) :: atmos */
647 width = scale_dstW;
648 height = scale_dstH;
651 vo_dga_width = vo_modelines[j].bytesPerScanline / HW_MODE.vdm_bytespp;
652 dga_modenum = vo_modelines[j].num;
653 modeline = vo_modelines + j;
655 #else
657 #ifdef HAVE_XF86VM
659 mp_msg(MSGT_VO, MSGL_INFO,
660 "vo_dga: DGA 1.0 compatibility code: Using XF86VidMode for mode switching!\n");
662 if (XF86VidModeQueryExtension(mDisplay, &vm_event, &vm_error))
664 XF86VidModeQueryVersion(mDisplay, &vm_ver, &vm_rev);
665 mp_msg(MSGT_VO, MSGL_INFO,
666 "vo_dga: XF86VidMode Extension v%i.%i\n", vm_ver, vm_rev);
667 have_vm = 1;
668 } else
670 mp_msg(MSGT_VO, MSGL_ERR,
671 "vo_dga: XF86VidMode Extension not available.\n");
674 #define GET_VREFRESH(dotclk, x, y)( (((dotclk)/(x))*1000)/(y) )
676 if (have_vm)
678 int modecount;
680 XF86VidModeGetAllModeLines(mDisplay, mScreen, &modecount,
681 &vo_dga_vidmodes);
683 if (vo_dga_vidmodes != NULL)
685 for (i = 0; i < modecount; i++)
687 if (check_res(i, wanted_width, wanted_height,
688 vo_dga_modes[vo_dga_hw_mode].vdm_depth,
689 vo_dga_vidmodes[i]->hdisplay,
690 vo_dga_vidmodes[i]->vdisplay,
691 GET_VREFRESH(vo_dga_vidmodes[i]->dotclock,
692 vo_dga_vidmodes[i]->htotal,
693 vo_dga_vidmodes[i]->vtotal),
694 0, &mX, &mY, &mVBI, &mMaxY))
695 j = i;
698 mp_msg(MSGT_VO, MSGL_INFO,
699 "vo_dga: Selected video mode %4d x %4d @ %3d Hz @ depth %2d, bitspp %2d, video %3d x %3d.\n",
700 mX, mY, mVBI,
701 vo_dga_modes[vo_dga_hw_mode].vdm_depth,
702 vo_dga_modes[vo_dga_hw_mode].vdm_bitspp, width, height);
703 } else
705 mp_msg(MSGT_VO, MSGL_INFO,
706 "vo_dga: XF86VidMode returned no screens - using current resolution.\n");
708 dga_modenum = j;
709 vo_dga_vp_width = mX;
710 vo_dga_vp_height = mY;
713 #else
714 mp_msg(MSGT_VO, MSGL_INFO,
715 "vo_dga: Only have DGA 1.0 extension and no XF86VidMode :-(\n");
716 mp_msg(MSGT_VO, MSGL_INFO,
717 " Thus, resolution switching is NOT possible.\n");
719 #endif
720 #endif
722 vo_dga_src_width = width;
723 vo_dga_src_height = height;
725 if (vo_dga_src_width > vo_dga_vp_width ||
726 vo_dga_src_height > vo_dga_vp_height)
728 mp_msg(MSGT_VO, MSGL_ERR,
729 "vo_dga: Sorry, video larger than viewport is not yet supported!\n");
730 // ugly, do something nicer in the future ...
731 #ifndef HAVE_DGA2
732 #ifdef HAVE_XF86VM
733 if (vo_dga_vidmodes)
735 XFree(vo_dga_vidmodes);
736 vo_dga_vidmodes = NULL;
738 #endif
739 #endif
740 return 1;
743 if (vo_dga_vp_width == VO_DGA_INVALID_RES)
745 mp_msg(MSGT_VO, MSGL_ERR,
746 "vo_dga: Something is wrong with your DGA. There doesn't seem to be a\n"
747 " single suitable mode!\n"
748 " Please file a bug report (see DOCS/HTML/en/bugreports.html)\n");
749 #ifndef HAVE_DGA2
750 #ifdef HAVE_XF86VM
751 if (vo_dga_vidmodes)
753 XFree(vo_dga_vidmodes);
754 vo_dga_vidmodes = NULL;
756 #endif
757 #endif
758 return 1;
760 // now let's start the DGA thing
762 if (!vo_config_count || width != prev_width || height != prev_height)
764 #ifdef HAVE_DGA2
766 if (!XDGAOpenFramebuffer(mDisplay, mScreen))
768 mp_msg(MSGT_VO, MSGL_ERR,
769 "vo_dga: Framebuffer mapping failed!!!\n");
770 return 1;
773 dgadevice = XDGASetMode(mDisplay, mScreen, dga_modenum);
774 XDGASync(mDisplay, mScreen);
776 vo_dga_base = dgadevice->data;
777 XFree(dgadevice);
779 XDGASetViewport(mDisplay, mScreen, 0, 0, XDGAFlipRetrace);
781 #else
783 #ifdef HAVE_XF86VM
784 if (have_vm)
786 XF86VidModeLockModeSwitch(mDisplay, mScreen, 0);
787 // Two calls are needed to switch modes on my ATI Rage 128. Why?
788 // for riva128 one call is enough!
789 XF86VidModeSwitchToMode(mDisplay, mScreen,
790 vo_dga_vidmodes[dga_modenum]);
791 XF86VidModeSwitchToMode(mDisplay, mScreen,
792 vo_dga_vidmodes[dga_modenum]);
794 #endif
796 XF86DGAGetViewPortSize(mDisplay, mScreen,
797 &vo_dga_vp_width, &vo_dga_vp_height);
799 XF86DGAGetVideo(mDisplay, mScreen,
800 (char **) &vo_dga_base, &vo_dga_width, &bank,
801 &ram);
803 XF86DGADirectVideo(mDisplay, mScreen,
804 XF86DGADirectGraphics | XF86DGADirectMouse |
805 XF86DGADirectKeyb);
807 XF86DGASetViewPort(mDisplay, mScreen, 0, 0);
809 #endif
811 // do some more checkings here ...
813 mp_msg(MSGT_VO, MSGL_V,
814 "vo_dga: bytes/line: %d, screen res: %dx%d, depth: %d, base: %08x, bpp: %d\n",
815 vo_dga_width, vo_dga_vp_width, vo_dga_vp_height,
816 HW_MODE.vdm_bytespp, vo_dga_base, HW_MODE.vdm_bitspp);
818 x_off = (vo_dga_vp_width - vo_dga_src_width) >> 1;
819 y_off = (vo_dga_vp_height - vo_dga_src_height) >> 1;
821 vo_dga_bytes_per_line = vo_dga_src_width * HW_MODE.vdm_bytespp;
822 vo_dga_lines = vo_dga_src_height;
824 vo_dga_src_offset = 0;
825 vo_dga_vp_offset =
826 (y_off * vo_dga_width + x_off) * HW_MODE.vdm_bytespp;
828 vo_dga_vp_skip = (vo_dga_width - vo_dga_src_width) * HW_MODE.vdm_bytespp; // todo
830 mp_msg(MSGT_VO, MSGL_V, "vo_dga: vp_off=%d, vp_skip=%d, bpl=%d\n",
831 vo_dga_vp_offset, vo_dga_vp_skip, vo_dga_bytes_per_line);
834 XGrabKeyboard(mDisplay, DefaultRootWindow(mDisplay), True,
835 GrabModeAsync, GrabModeAsync, CurrentTime);
836 if (vo_grabpointer)
837 XGrabPointer(mDisplay, DefaultRootWindow(mDisplay), True,
838 ButtonPressMask, GrabModeAsync, GrabModeAsync,
839 None, None, CurrentTime);
841 if (!vo_config_count || width != prev_width || height != prev_height)
843 init_video_buffers(vo_dga_base,
844 vo_dga_vp_height,
845 vo_dga_width * HW_MODE.vdm_bytespp,
846 #ifdef HAVE_DGA2
847 modeline->maxViewportY,
848 #else
849 vo_dga_vp_height,
850 #endif
851 vo_doublebuffering);
852 prev_width = width;
853 prev_height = height;
856 mp_msg(MSGT_VO, MSGL_V, "vo_dga: Using %d frame buffer%s.\n",
857 vo_dga_nr_video_buffers,
858 vo_dga_nr_video_buffers == 1 ? "" : "s");
860 vo_dga_is_running = 1;
861 return 0;
864 static int dga_depths_init = 0;
866 static uint32_t preinit(const char *arg)
868 if (arg)
870 mp_msg(MSGT_VO, MSGL_INFO, "vo_dga: Unknown subdevice: %s\n", arg);
871 return ENOSYS;
874 if (!vo_init())
875 return -1; // Can't open X11
877 if (dga_depths_init == 0)
878 { // FIXME!?
879 int i;
881 vo_dga_XServer_mode = vd_ValidateMode(vo_depthonscreen);
883 if (vo_dga_XServer_mode == 0)
885 #ifndef HAVE_DGA2
886 mp_msg(MSGT_VO, MSGL_ERR,
887 "vo_dga: Your X-Server is not running in a ");
888 mp_msg(MSGT_VO, MSGL_ERR,
889 "resolution supported by DGA driver!\n");
890 #endif
891 } //else{
892 // mp_msg(MSGT_VO, MSGL_INFO, "vo_dga: X running at: %s\n",
893 // vd_GetModeString(vo_dga_XServer_mode));
894 //}
896 #ifdef HAVE_DGA2
897 vo_modelines = XDGAQueryModes(mDisplay, mScreen, &vo_modecount);
898 if (vo_modelines)
900 for (i = 0; i < vo_modecount; i++)
902 mp_msg(MSGT_VO, MSGL_V,
903 "vo_dga: (%03d) depth=%d, bpp=%d, r=%08x, g=%08x, b=%08x, %d x %d\n",
904 i, vo_modelines[i].depth,
905 vo_modelines[i].bitsPerPixel,
906 vo_modelines[i].redMask, vo_modelines[i].greenMask,
907 vo_modelines[i].blueMask,
908 vo_modelines[i].viewportWidth,
909 vo_modelines[i].viewportHeight);
910 vd_EnableMode(vo_modelines[i].depth,
911 vo_modelines[i].bitsPerPixel,
912 vo_modelines[i].redMask,
913 vo_modelines[i].greenMask,
914 vo_modelines[i].blueMask);
917 #endif
918 dga_depths_init = 1;
920 if (!vo_dga_modes[1].vdm_supported
921 && vo_dga_modes[2].vdm_supported)
923 vo_dga_modes[1].vdm_supported = 1;
926 if (!vo_dga_modes[3].vdm_supported
927 && vo_dga_modes[4].vdm_supported)
929 vo_dga_modes[3].vdm_supported = 1;
932 for (i = 1; i < vo_dga_mode_num; i++)
934 mp_msg(MSGT_VO, MSGL_INFO, "vo_dga: Mode: %s",
935 vd_GetModeString(i));
936 if (vo_dbpp && vo_dbpp != vo_dga_modes[i].vdm_mplayer_depth)
938 vo_dga_modes[i].vdm_supported = 0;
939 mp_msg(MSGT_VO, MSGL_INFO, " ...disabled by -bpp %d",
940 vo_dbpp);
942 mp_msg(MSGT_VO, MSGL_INFO, "\n");
946 return 0;
949 static uint32_t get_image(mp_image_t * mpi)
951 if (!IMGFMT_IS_BGR(mpi->imgfmt) ||
952 (IMGFMT_BGR_DEPTH(mpi->imgfmt) !=
953 vo_dga_modes[vo_dga_hw_mode].vdm_mplayer_depth)
954 || (mpi->type == MP_IMGTYPE_STATIC && vo_dga_nr_video_buffers > 1)
955 || (mpi->type == MP_IMGTYPE_IP && vo_dga_nr_video_buffers < 2)
956 || (mpi->type == MP_IMGTYPE_IPB))
957 return (VO_FALSE);
959 if ((mpi->flags & MP_IMGFLAG_ACCEPT_STRIDE) ||
960 (mpi->flags & MP_IMGFLAG_ACCEPT_WIDTH &&
961 ((vo_dga_bytes_per_line + vo_dga_vp_skip) % (mpi->bpp / 8)) == 0)
962 || (mpi->width * (mpi->bpp / 8) ==
963 (vo_dga_bytes_per_line + vo_dga_vp_skip)))
966 mpi->planes[0] = CURRENT_VIDEO_BUFFER.data + vo_dga_vp_offset;
967 mpi->stride[0] = vo_dga_bytes_per_line + vo_dga_vp_skip;
968 mpi->width =
969 (vo_dga_bytes_per_line + vo_dga_vp_skip) / (mpi->bpp / 8);
970 mpi->flags |= MP_IMGFLAG_DIRECT;
971 return (VO_TRUE);
974 return (VO_FALSE);
977 static uint32_t control(uint32_t request, void *data, ...)
979 switch (request)
981 case VOCTRL_GET_IMAGE:
982 return get_image(data);
983 case VOCTRL_QUERY_FORMAT:
984 return query_format(*((uint32_t *) data));
986 return VO_NOTIMPL;
989 //---------------------------------------------------------