/* * VGA hardware emulation * * Copyright 1998 Ove Kåven (with some help from Marcus Meissner) * */ #include #include "winbase.h" #include "wincon.h" #include "miscemu.h" #include "vga.h" #include "ddraw.h" #include "services.h" #include "debugtools.h" DEFAULT_DEBUG_CHANNEL(ddraw); static IDirectDraw *lpddraw = NULL; static IDirectDrawSurface *lpddsurf; static IDirectDrawPalette *lpddpal; static DDSURFACEDESC sdesc; static LONG vga_polling,vga_refresh; static HANDLE poll_timer; typedef HRESULT WINAPI (*DirectDrawCreateProc)(LPGUID,LPDIRECTDRAW *,LPUNKNOWN); static DirectDrawCreateProc pDirectDrawCreate; static void VGA_DeinstallTimer(void) { if (poll_timer) { SERVICE_Delete( poll_timer ); poll_timer = 0; } } static void VGA_InstallTimer(unsigned Rate) { VGA_DeinstallTimer(); if (!poll_timer) poll_timer = SERVICE_AddTimer( Rate, VGA_Poll, 0 ); } HANDLE VGA_AlphaConsole(void) { /* this assumes that no Win32 redirection has taken place, but then again, * only 16-bit apps are likely to use this part of Wine... */ return GetStdHandle(STD_OUTPUT_HANDLE); } char*VGA_AlphaBuffer(void) { return DOSMEM_MapDosToLinear(0xb8000); } /*** GRAPHICS MODE ***/ int VGA_SetMode(unsigned Xres,unsigned Yres,unsigned Depth) { if (lpddraw) VGA_Exit(); if (!lpddraw) { if (!pDirectDrawCreate) { HMODULE hmod = LoadLibraryA( "ddraw.dll" ); if (hmod) pDirectDrawCreate = (DirectDrawCreateProc)GetProcAddress( hmod, "DirectDrawCreate" ); } if (pDirectDrawCreate) pDirectDrawCreate(NULL,&lpddraw,NULL); if (!lpddraw) { ERR("DirectDraw is not available\n"); return 1; } if (IDirectDraw_SetDisplayMode(lpddraw,Xres,Yres,Depth)) { ERR("DirectDraw does not support requested display mode\n"); IDirectDraw_Release(lpddraw); lpddraw=NULL; return 1; } IDirectDraw_CreatePalette(lpddraw,DDPCAPS_8BIT,NULL,&lpddpal,NULL); memset(&sdesc,0,sizeof(sdesc)); sdesc.dwSize=sizeof(sdesc); sdesc.dwFlags = DDSD_CAPS; sdesc.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE; if (IDirectDraw_CreateSurface(lpddraw,&sdesc,&lpddsurf,NULL)||(!lpddsurf)) { ERR("DirectDraw surface is not available\n"); IDirectDraw_Release(lpddraw); lpddraw=NULL; return 1; } FIXME("no default palette entries\n"); IDirectDrawSurface_SetPalette(lpddsurf,lpddpal); vga_refresh=0; /* poll every 20ms (50fps should provide adequate responsiveness) */ VGA_InstallTimer(20); } return 0; } int VGA_GetMode(unsigned*Height,unsigned*Width,unsigned*Depth) { if (!lpddraw) return 1; if (!lpddsurf) return 1; if (Height) *Height=sdesc.dwHeight; if (Width) *Width=sdesc.dwWidth; if (Depth) *Depth=sdesc.ddpfPixelFormat.u1.dwRGBBitCount; return 0; } void VGA_Exit(void) { if (lpddraw) { VGA_DeinstallTimer(); IDirectDrawSurface_SetPalette(lpddsurf,NULL); IDirectDrawSurface_Release(lpddsurf); lpddsurf=NULL; IDirectDrawPalette_Release(lpddpal); lpddpal=NULL; IDirectDraw_Release(lpddraw); lpddraw=NULL; } } void VGA_SetPalette(PALETTEENTRY*pal,int start,int len) { if (!lpddraw) return; IDirectDrawPalette_SetEntries(lpddpal,0,start,len,pal); } void VGA_SetQuadPalette(RGBQUAD*color,int start,int len) { PALETTEENTRY pal[256]; int c; if (!lpddraw) return; for (c=0; c=0) *dat = attr; dat++; } } /*** CONTROL ***/ void CALLBACK VGA_Poll( ULONG_PTR arg ) { char *dat; unsigned int Pitch,Height,Width,Y,X; char *surf; if (!InterlockedExchangeAdd(&vga_polling, 1)) { /* FIXME: optimize by doing this only if the data has actually changed * (in a way similar to DIBSection, perhaps) */ if (lpddraw) { /* graphics mode */ surf = VGA_Lock(&Pitch,&Height,&Width,NULL); if (!surf) return; dat = DOSMEM_MapDosToLinear(0xa0000); /* copy from virtual VGA frame buffer to DirectDraw surface */ for (Y=0; Y