/* * $Logfile: /DescentIII/Main/ddgr_win32/ddgrWin32DX.cpp $ * $Revision: 2 $ * $Date: 9/12/97 4:13p $ * $Author: Samir $ * * DirectX implementation of DDGR library. * * $Log: /DescentIII/Main/ddgr_win32/ddgrWin32DX.cpp $ * * 2 9/12/97 4:13p Samir * Added some private data access functions and more DirectX * functionality. * * $NoKeywords: $ */ #include "DDAccess.h" // Allow this module full DD access #include "ddgrWin32.h" #include "ddgrWin32DX.h" #include "Application.h" #include "mono.h" #include "pserror.h" tDDGRDXInternalData DDGR_DX_lib_data; // Initializes DX subsystem bool ddgr_dx_Init(oeApplication *app) { DX_DATA(hPrimaryWnd) = (HWND)((oeWin32Application *)app)->m_hWnd; DX_DATA(lpDD) = NULL; // Initialize DirectDraw system if we're going fullscreen, and GDIX system HRESULT hres; hres = DirectDrawCreate(NULL, &DX_DATA(lpDD), NULL); if (hres != DD_OK) { ddgr_PushError("Failure to initialize DirectDraw driver (%d)", LOWORD(hres)); return false; } hres = DX_DATA(lpDD)->SetCooperativeLevel(DX_DATA(hPrimaryWnd), DDSCL_ALLOWREBOOT | DDSCL_EXCLUSIVE | DDSCL_FULLSCREEN); if (hres != DD_OK) { ddgr_PushError("Failed to set DirectDraw fullscreen access level (%d)", LOWORD(hres)); return false; } DX_DATA(vidrefs) = 0; // reset video reference count. DX_DATA(init) = true; mprintf((0, "DX system initialized.\n")); return true; } // Closes DX subsystem void ddgr_dx_Close() { ASSERT(DX_DATA(vidrefs) == 0); if (DX_DATA(lpDD)) { DX_DATA(lpDD)->RestoreDisplayMode(); DX_DATA(lpDD)->Release(); } DX_DATA(lpDD) = NULL; DX_DATA(hPrimaryWnd) = NULL; DX_DATA(init) = false; mprintf((0, "DX system closed.\n")); } // Initializes the display for use with the DX subsystem. bool ddgr_dx_surf_InitVideo(ddgr_surface *sf) { tDXSurface *dxsf; DDSURFACEDESC ddsd; // we need to create a primary surface from the display mode we set, and possibly a backbuffer if // requested. if (DX_DATA(vidrefs) == 0) { HRESULT hres; DDSCAPS ddscaps; hres = DX_DATA(lpDD)->SetDisplayMode(sf->w, sf->h, sf->bpp); if (hres != DD_OK) { ddgr_PushError("Unable to set DirectDraw display mode (%d)\n", LOWORD(hres)); return false; } memset(&ddsd, 0, sizeof(ddsd)); ddsd.dwSize = sizeof(ddsd); ddsd.dwFlags = DDSD_CAPS | DDSD_BACKBUFFERCOUNT; ddsd.dwBackBufferCount = 1; ddsd.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE | DDSCAPS_FLIP | DDSCAPS_COMPLEX; hres = DX_DATA(lpDD)->CreateSurface(&ddsd, &DX_DATA(vidsurf.lpdds), NULL); if (hres != DD_OK) { ddgr_PushError("Unable to capture DirectDraw display surface (%d)\n", LOWORD(hres)); return false; } ddscaps.dwCaps = DDSCAPS_BACKBUFFER; hres = DX_DATA(vidsurf.lpdds)->GetAttachedSurface(&ddscaps, &DX_DATA(vidsurf.lpddsback)); if (hres != DD_OK) { ddgr_PushError("Unable to capture DirectDraw display back surface (%d)\n", LOWORD(hres)); return false; } } DX_DATA(vidrefs)++; // increment vidref count dxsf = new tDXSurface; dxsf->lpdds = DX_DATA(vidsurf.lpdds); // this means that we are capturing the screen, not creating a screen. // get the screen's width and height and bitdepth if (sf->flags == 0) { memset(&ddsd, 0, sizeof(ddsd)); ddsd.dwSize = sizeof(ddsd); dxsf->lpdds->GetSurfaceDesc(&ddsd); sf->w = ddsd.dwWidth; sf->h = ddsd.dwHeight; sf->bpp = ddsd.ddpfPixelFormat.dwRGBBitCount; } // we create a dummy surface for the screen's back buffer, or our accessable buffer // specifying SURFFLAG_BACKBUFFER ensures that we create a GDI bitmap of size requested // in 'sf' and it will be our renderable bitmap, or OUR primary render buffer if (sf->flags & SURFFLAG_BACKBUFFER) { // set our front buffer to this newly created surface and delete the gdi bitmap object dxsf->lpddsback = DX_DATA(vidsurf.lpddsback); dxsf->backbuffer = true; } else { dxsf->lpddsback = NULL; dxsf->backbuffer = false; } sf->obj = (void *)dxsf; return true; } // Deinitializes the video surface void ddgr_dx_surf_CloseVideo(ddgr_surface *sf) { tDXSurface *bm = (tDXSurface *)sf->obj; ASSERT(DX_DATA(vidrefs) > 0); DX_DATA(vidrefs)--; if (DX_DATA(vidrefs) == 0) { DX_DATA(vidsurf.lpdds)->Release(); DX_DATA(vidsurf.lpdds) = NULL; DX_DATA(vidsurf.lpddsback) = NULL; } delete bm; } // flips the buffers in a surface. really only useful for video screens bool ddgr_dx_surf_FlipVideo(ddgr_surface *sf) { tDXSurface *bm = (tDXSurface *)sf->obj; HRESULT hres; if (bm->backbuffer && (sf->flags & SURFFLAG_BACKBUFFER)) { hres = bm->lpdds->Flip(NULL, 0); if (hres == DDERR_WASSTILLDRAWING || hres == DD_OK) return true; else return false; } return true; } /* input: sf->name is optional. sf->w, sf->h, sf->bpp are mandatory sf->type and sf->flags are mandatory. output: sf->obj = surface object. sf->locks = 0 */ bool ddgr_dx_surf_Create(ddgr_surface *sf) { return true; } void ddgr_dx_surf_Destroy(ddgr_surface *sf) {} /* retrieves a pointer to surface memory. allowed to lock one surface multiple times. ptr is the returned pointer to surface memory. used to unlock surface also rowsize is the size in bytes of one row of memory. */ bool ddgr_dx_surf_Lock(ddgr_surface *sf, void **ptr, int *rowsize) { HRESULT hres; DDSURFACEDESC ddsd; tDXSurface *bm = (tDXSurface *)sf->obj; memset(&ddsd, 0, sizeof(ddsd)); ddsd.dwSize = sizeof(ddsd); if (bm->backbuffer) hres = bm->lpddsback->Lock(NULL, &ddsd, DDLOCK_WAIT, NULL); else hres = bm->lpdds->Lock(NULL, &ddsd, DDLOCK_WAIT, NULL); if (hres != DD_OK) { ddgr_PushError("Unable to set lock DirectDraw surface(%d)\n", LOWORD(hres)); return false; } *rowsize = (int)ddsd.lPitch; *ptr = ddsd.lpSurface; return true; } bool ddgr_dx_surf_Unlock(ddgr_surface *sf, void *ptr) { HRESULT hres; tDXSurface *bm = (tDXSurface *)sf->obj; if (bm->backbuffer) hres = bm->lpddsback->Unlock(ptr); else hres = bm->lpdds->Unlock(ptr); if (hres != DD_OK) { ddgr_PushError("Unable to unlock DirectDraw surface(%d)\n", LOWORD(hres)); return false; } return true; } // attaches an OS handle to a surface void ddgr_dx_surf_AttachHandle(ddgr_surface *sf, unsigned handle) {} // The only graphic primatives // clear // blt. void ddgr_dx_surf_Clear(ddgr_surface *dsf, ddgr_color col, int l, int t, int w, int h) { HRESULT hres; DDBLTFX ddbltfx; tDXSurface *dbm = (tDXSurface *)dsf->obj; LPDIRECTDRAWSURFACE lpdds; RECT rect; lpdds = (dbm->backbuffer) ? dbm->lpddsback : dbm->lpdds; memset(&ddbltfx, 0, sizeof(DDBLTFX)); ddbltfx.dwSize = sizeof(ddbltfx); ddbltfx.dwFillColor = RGB(DDGR_COLOR_RED(col), DDGR_COLOR_GREEN(col), DDGR_COLOR_BLUE(col)); SetRect(&rect, l, t, l + w - 1, t + h - 1); hres = lpdds->Blt(&rect, // dest rect NULL, // src surface NULL, // src rect DDBLT_COLORFILL | DDBLT_WAIT, &ddbltfx); } bool ddgr_dx_surf_Blt(ddgr_surface *dsf, int dx, int dy, ddgr_surface *ssf, int sx, int sy, int sw, int sh) { HRESULT hres; tDXSurface *sbm = (tDXSurface *)ssf->obj; tDXSurface *dbm = (tDXSurface *)dsf->obj; LPDIRECTDRAWSURFACE lpddsd; LPDIRECTDRAWSURFACE lpddss; RECT srect, drect; if (ssf->flags & SURFFLAG_BACKBUFFER) lpddss = sbm->lpddsback; else lpddss = sbm->lpdds; if (dsf->flags & SURFFLAG_BACKBUFFER) lpddsd = dbm->lpddsback; else lpddsd = dbm->lpdds; SetRect(&srect, sx, sy, sx + sw - 1, sy + sh - 1); SetRect(&drect, dx, dy, dx + sw - 1, dy + sh + 1); hres = lpddsd->Blt(&drect, lpddss, &srect, DDBLT_WAIT, NULL); if (hres != DD_OK) { ddgr_PushError("Unable to blt surface %s to DirectDraw surface %s(%d)\n", ssf->name, dsf->name, LOWORD(hres)); return false; } return true; }