singe/vldp2/libvo/video_out_dx.c
2019-11-11 14:53:02 -06:00

520 lines
15 KiB
C

/*
* video_out_dx.c
* Copyright (C) 2000-2002 Michel Lespinasse <walken@zoy.org>
*
* Contributed by Gildas Bazin <gbazin@netcourrier.com>
*
* This file is part of mpeg2dec, a free MPEG-2 video stream decoder.
* See http://libmpeg2.sourceforge.net/ for updates.
*
* mpeg2dec is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* mpeg2dec is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include "config.h"
#ifdef LIBVO_DX
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <inttypes.h>
#include "video_out.h"
#include "convert.h"
#include <ddraw.h>
#include <initguid.h>
#define USE_OVERLAY_TRIPLE_BUFFERING 0
/*
* DirectDraw GUIDs.
* Defining them here allows us to get rid of the dxguid library during link.
*/
DEFINE_GUID (IID_IDirectDraw2, 0xB3A6F3E0,0x2B43,0x11CF,0xA2,0xDE,0x00,0xAA,0x00,0xB9,0x33,0x56);
DEFINE_GUID (IID_IDirectDrawSurface2, 0x57805885,0x6eec,0x11cf,0x94,0x41,0xa8,0x23,0x03,0xc1,0x0e,0x27);
#define FOURCC_YV12 0x32315659
typedef struct {
vo_instance_t vo;
int width;
int height;
HWND window;
RECT window_coords;
HINSTANCE hddraw_dll;
LPDIRECTDRAW2 ddraw;
LPDIRECTDRAWSURFACE2 display;
LPDIRECTDRAWCLIPPER clipper;
LPDIRECTDRAWSURFACE2 frame[3];
int index;
LPDIRECTDRAWSURFACE2 overlay;
uint8_t * yuv[3];
int stride;
} dx_instance_t;
static void update_overlay (dx_instance_t * instance)
{
DDOVERLAYFX ddofx;
DWORD dwFlags;
memset (&ddofx, 0, sizeof (DDOVERLAYFX));
ddofx.dwSize = sizeof (DDOVERLAYFX);
dwFlags = DDOVER_SHOW | DDOVER_KEYDESTOVERRIDE;
IDirectDrawSurface2_UpdateOverlay (instance->overlay, NULL,
instance->display,
&instance->window_coords,
dwFlags, &ddofx);
}
static long FAR PASCAL event_procedure (HWND hwnd, UINT message,
WPARAM wParam, LPARAM lParam)
{
RECT rect_window;
POINT point_window;
dx_instance_t * instance;
switch (message) {
case WM_WINDOWPOSCHANGED:
instance = (dx_instance_t *) GetWindowLong (hwnd, GWL_USERDATA);
/* update the window position and size */
point_window.x = 0;
point_window.y = 0;
ClientToScreen (hwnd, &point_window);
instance->window_coords.left = point_window.x;
instance->window_coords.top = point_window.y;
GetClientRect (hwnd, &rect_window);
instance->window_coords.right = rect_window.right + point_window.x;
instance->window_coords.bottom = rect_window.bottom + point_window.y;
/* update the overlay */
if (instance->overlay && instance->display)
update_overlay (instance);
return 0;
case WM_CLOSE: /* forbid the user to close the window */
return 0;
case WM_DESTROY: /* just destroy the window */
PostQuitMessage (0);
return 0;
}
return DefWindowProc (hwnd, message, wParam, lParam);
}
static void check_events (dx_instance_t * instance)
{
MSG msg;
while (PeekMessage (&msg, instance->window, 0, 0, PM_REMOVE)) {
TranslateMessage (&msg);
DispatchMessage (&msg);
}
}
static int create_window (dx_instance_t * instance)
{
RECT rect_window;
WNDCLASSEX wc;
wc.cbSize = sizeof (WNDCLASSEX);
wc.style = CS_DBLCLKS;
wc.lpfnWndProc = (WNDPROC) event_procedure;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = GetModuleHandle (NULL);
wc.hIcon = NULL;
wc.hCursor = LoadCursor (NULL, IDC_ARROW);
wc.hbrBackground = CreateSolidBrush (RGB (0, 0, 0));
wc.lpszMenuName = NULL;
wc.lpszClassName = "libvo_dx";
wc.hIconSm = NULL;
if (!RegisterClassEx (&wc)) {
fprintf (stderr, "Can not register window class\n");
return 1;
}
rect_window.top = 10;
rect_window.left = 10;
rect_window.right = rect_window.left + instance->width;
rect_window.bottom = rect_window.top + instance->height;
AdjustWindowRect (&rect_window, WS_OVERLAPPEDWINDOW|WS_SIZEBOX, 0);
instance->window = CreateWindow ("libvo_dx", "mpeg2dec",
WS_OVERLAPPEDWINDOW | WS_SIZEBOX,
CW_USEDEFAULT, 0,
rect_window.right - rect_window.left,
rect_window.bottom - rect_window.top,
NULL, NULL, GetModuleHandle (NULL), NULL);
if (instance->window == NULL) {
fprintf (stderr, "Can not create window\n");
return 1;
}
/* store a directx_instance pointer into the window local storage
* (for later use in event_handler).
* We need to use SetWindowLongPtr when it is available in mingw */
SetWindowLong (instance->window, GWL_USERDATA, (LONG) instance);
ShowWindow (instance->window, SW_SHOW);
return 0;
}
static LPDIRECTDRAWSURFACE2 alloc_surface (dx_instance_t * instance,
DDSURFACEDESC * ddsd)
{
LPDIRECTDRAWSURFACE surface;
LPDIRECTDRAWSURFACE2 surface2;
if (DD_OK != IDirectDraw2_CreateSurface (instance->ddraw, ddsd,
&surface, NULL) ||
DD_OK != IDirectDrawSurface_QueryInterface (surface,
&IID_IDirectDrawSurface2,
(LPVOID *) &surface2)) {
fprintf (stderr, "Can not create directDraw frame surface\n");
return NULL;
}
IDirectDrawSurface_Release (surface);
return surface2;
}
static int dx_init (dx_instance_t *instance)
{
HRESULT (WINAPI * OurDirectDrawCreate) (GUID *, LPDIRECTDRAW *,
IUnknown *);
LPDIRECTDRAW ddraw;
DDSURFACEDESC ddsd;
/* load direct draw DLL */
instance->hddraw_dll = LoadLibrary ("DDRAW.DLL");
if (instance->hddraw_dll == NULL) {
fprintf (stderr, "Can not load DDRAW.DLL\n");
return 1;
}
ddraw = NULL;
OurDirectDrawCreate = (void *) GetProcAddress (instance->hddraw_dll,
"DirectDrawCreate");
if (OurDirectDrawCreate == NULL ||
DD_OK != OurDirectDrawCreate (NULL, &ddraw, NULL) ||
DD_OK != IDirectDraw_QueryInterface (ddraw, &IID_IDirectDraw2,
(LPVOID *) &instance->ddraw) ||
DD_OK != IDirectDraw_SetCooperativeLevel (instance->ddraw,
instance->window,
DDSCL_NORMAL)) {
fprintf (stderr, "Can not initialize directDraw interface\n");
return 1;
}
IDirectDraw_Release (ddraw);
memset (&ddsd, 0, sizeof (DDSURFACEDESC));
ddsd.dwSize = sizeof (DDSURFACEDESC);
ddsd.dwFlags = DDSD_CAPS;
ddsd.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE;
instance->display = alloc_surface (instance, &ddsd);
if (instance->display == NULL) {
fprintf (stderr, "Can not create directDraw display surface\n");
return 1;
}
if (DD_OK != IDirectDraw2_CreateClipper (instance->ddraw, 0,
&instance->clipper, NULL) ||
DD_OK != IDirectDrawClipper_SetHWnd (instance->clipper, 0,
instance->window) ||
DD_OK != IDirectDrawSurface_SetClipper (instance->display,
instance->clipper)) {
fprintf (stderr, "Can not initialize directDraw clipper\n");
return 1;
}
return 0;
}
static int common_setup (dx_instance_t * instance, int width, int height)
{
instance->width = width;
instance->height = height;
instance->index = 0;
if (create_window (instance) || dx_init (instance))
return 1;
return 0;
}
static int dxrgb_setup (vo_instance_t * _instance, int width, int height,
vo_setup_result_t * result)
{
dx_instance_t * instance = (dx_instance_t *) _instance;
HDC hdc;
int bpp;
if (common_setup (instance, width, height))
return 1;
hdc = GetDC (NULL);
bpp = GetDeviceCaps (hdc, BITSPIXEL);
ReleaseDC (NULL, hdc);
result->convert = convert_rgb (CONVERT_RGB, bpp);
return 0;
}
static LPDIRECTDRAWSURFACE2 alloc_frame (dx_instance_t * instance)
{
DDSURFACEDESC ddsd;
LPDIRECTDRAWSURFACE2 surface;
memset (&ddsd, 0, sizeof (DDSURFACEDESC));
ddsd.dwSize = sizeof (DDSURFACEDESC);
ddsd.ddpfPixelFormat.dwSize = sizeof (DDPIXELFORMAT);
ddsd.dwFlags = DDSD_HEIGHT | DDSD_WIDTH | DDSD_CAPS;
ddsd.dwHeight = instance->height;
ddsd.dwWidth = instance->width;
ddsd.ddsCaps.dwCaps = DDSCAPS_OFFSCREENPLAIN | DDSCAPS_SYSTEMMEMORY;
surface = alloc_surface (instance, &ddsd);
if (surface == NULL)
fprintf (stderr, "Can not create directDraw frame surface\n");
return surface;
}
static void * surface_addr (LPDIRECTDRAWSURFACE2 surface, int * stride)
{
DDSURFACEDESC ddsd;
memset (&ddsd, 0, sizeof (DDSURFACEDESC));
ddsd.dwSize = sizeof (DDSURFACEDESC);
IDirectDrawSurface2_Lock (surface, NULL, &ddsd,
DDLOCK_NOSYSLOCK | DDLOCK_WAIT, NULL);
IDirectDrawSurface2_Unlock (surface, NULL);
*stride = ddsd.lPitch;
return ddsd.lpSurface;
}
static void dx_setup_fbuf (vo_instance_t * _instance,
uint8_t ** buf, void ** id)
{
dx_instance_t * instance = (dx_instance_t *) _instance;
int stride;
*id = instance->frame[instance->index++] = alloc_frame (instance);
buf[0] = surface_addr (*id, &stride);
buf[1] = NULL; buf[2] = NULL;
}
static void dxrgb_draw_frame (vo_instance_t * _instance,
uint8_t * const * buf, void * id)
{
dx_instance_t * instance = (dx_instance_t *) _instance;
LPDIRECTDRAWSURFACE2 surface = (LPDIRECTDRAWSURFACE2) id;
DDBLTFX ddbltfx;
check_events (instance);
memset (&ddbltfx, 0, sizeof (DDBLTFX));
ddbltfx.dwSize = sizeof (DDBLTFX);
ddbltfx.dwDDFX = DDBLTFX_NOTEARING;
if (DDERR_SURFACELOST ==
IDirectDrawSurface2_Blt (instance->display, &instance->window_coords,
surface, NULL, DDBLT_WAIT, &ddbltfx)) {
/* restore surface and try again */
IDirectDrawSurface2_Restore (instance->display);
IDirectDrawSurface2_Blt (instance->display, &instance->window_coords,
surface, NULL, DDBLT_WAIT, &ddbltfx);
}
}
static vo_instance_t * common_open (int (* setup) (vo_instance_t *, int, int,
vo_setup_result_t *),
void (* setup_fbuf) (vo_instance_t *,
uint8_t **, void **),
void (* draw) (vo_instance_t *,
uint8_t * const *,
void * id))
{
dx_instance_t * instance;
instance = malloc (sizeof (dx_instance_t));
if (instance == NULL)
return NULL;
memset (instance, 0, sizeof (dx_instance_t));
instance->vo.setup = setup;
instance->vo.setup_fbuf = setup_fbuf;
instance->vo.set_fbuf = NULL;
instance->vo.start_fbuf = NULL;
instance->vo.draw = draw;
instance->vo.discard = NULL;
instance->vo.close = NULL; //dx_close;
return (vo_instance_t *) instance;
}
vo_instance_t * vo_dxrgb_open (void)
{
return common_open (dxrgb_setup, dx_setup_fbuf, dxrgb_draw_frame);
}
static LPDIRECTDRAWSURFACE2 alloc_overlay (dx_instance_t * instance)
{
DDSURFACEDESC ddsd;
LPDIRECTDRAWSURFACE2 surface;
memset (&ddsd, 0, sizeof (DDSURFACEDESC));
ddsd.dwSize = sizeof (DDSURFACEDESC);
ddsd.ddpfPixelFormat.dwSize = sizeof (DDPIXELFORMAT);
ddsd.dwFlags = DDSD_HEIGHT | DDSD_WIDTH | DDSD_CAPS;
ddsd.dwHeight = instance->height;
ddsd.dwWidth = instance->width;
ddsd.ddpfPixelFormat.dwFlags = DDPF_FOURCC;
ddsd.ddpfPixelFormat.dwFourCC = FOURCC_YV12;
ddsd.dwFlags |= DDSD_PIXELFORMAT;
ddsd.ddsCaps.dwCaps = DDSCAPS_OVERLAY | DDSCAPS_VIDEOMEMORY;
#if USE_OVERLAY_TRIPLE_BUFFERING
ddsd.dwFlags |= DDSD_BACKBUFFERCOUNT;
ddsd.ddsCaps.dwCaps |= DDSCAPS_COMPLEX | DDSCAPS_FLIP;
#endif
ddsd.dwBackBufferCount = 2;
surface = alloc_surface (instance, &ddsd);
if (surface == NULL)
fprintf (stderr, "Can not create directDraw frame surface\n");
return surface;
}
static int dx_setup (vo_instance_t * _instance, int width, int height,
vo_setup_result_t * result)
{
dx_instance_t * instance = (dx_instance_t *) _instance;
LPDIRECTDRAWSURFACE2 surface;
DDSURFACEDESC ddsd;
if (common_setup (instance, width, height))
return 1;
instance->overlay = alloc_overlay (instance);
if (!instance->overlay)
return 1;
update_overlay (instance);
surface = instance->overlay;
/* Get the back buffer */
memset (&ddsd.ddsCaps, 0, sizeof (DDSCAPS));
ddsd.ddsCaps.dwCaps = DDSCAPS_BACKBUFFER;
if (DD_OK != IDirectDrawSurface2_GetAttachedSurface (instance->overlay,
&ddsd.ddsCaps,
&surface))
surface = instance->overlay;
instance->yuv[0] = surface_addr (surface, &instance->stride);
instance->yuv[2] = instance->yuv[0] + instance->stride * instance->height;
instance->yuv[1] =
instance->yuv[2] + (instance->stride * instance->height >> 2);
result->convert = NULL;
return 0;
}
static void copy_yuv_picture (dx_instance_t * instance,
uint8_t * const * buf, void * id)
{
uint8_t * dest[3];
int width, i;
dest[0] = instance->yuv[0];
dest[1] = instance->yuv[1];
dest[2] = instance->yuv[2];
width = instance->width;
for (i = 0; i < instance->height >> 1; i++) {
memcpy (dest[0], buf[0] + 2 * i * width, width);
dest[0] += instance->stride;
memcpy (dest[0], buf[0] + (2 * i + 1) * width, width);
dest[0] += instance->stride;
memcpy (dest[1], buf[1] + i * (width >> 1), width >> 1);
dest[1] += instance->stride >> 1;
memcpy (dest[2], buf[2] + i * (width >> 1), width >> 1);
dest[2] += instance->stride >> 1;
}
}
static void dx_draw_frame (vo_instance_t * _instance,
uint8_t * const * buf, void * id)
{
dx_instance_t * instance = (dx_instance_t *) _instance;
check_events (instance);
copy_yuv_picture (instance, buf, id);
if (DDERR_SURFACELOST ==
IDirectDrawSurface2_Flip (instance->overlay, NULL, DDFLIP_WAIT)) {
/* restore surfaces and try again */
IDirectDrawSurface2_Restore (instance->display);
IDirectDrawSurface2_Restore (instance->overlay);
IDirectDrawSurface2_Flip (instance->overlay, NULL, DDFLIP_WAIT);
}
}
vo_instance_t * vo_dx_open (void)
{
return common_open (dx_setup, NULL, dx_draw_frame);
}
#if 0
static void dx_close (vo_instance_t * _instance)
{
dx_instance_t * instance;
int i;
instance = (dx_instance_t *) _instance;
if (instance->using_overlay && instance->overlay) {
IDirectDrawSurface2_Release (instance->overlay);
instance->overlay = NULL;
} else
for (i = 0; i < 3; i++) {
if (instance->frame[i].p_surface != NULL)
IDirectDrawSurface2_Release (instance->frame[i].p_surface);
instance->frame[i].p_surface = NULL;
}
if (instance->clipper != NULL)
IDirectDrawClipper_Release (instance->clipper);
if (instance->display != NULL)
IDirectDrawSurface2_Release (instance->display);
if (instance->ddraw != NULL)
IDirectDraw2_Release (instance->ddraw);
if (instance->hddraw_dll != NULL)
FreeLibrary (instance->hddraw_dll);
if (instance->window != NULL)
DestroyWindow (instance->window);
}
#endif
#endif