|
|
/*==========================================================================
* * Copyright (C) 1997 Microsoft Corporation. All Rights Reserved. * * File: factory.cpp * Content: DirectDraw Factory support * History: * Date By Reason * ==== == ====== * 24-feb-97 ralphl initial implementation * 25-feb-97 craige minor tweaks for dx checkin * 14-mar-97 jeffort version checking changed to support DX3 and above * 09-apr-97 jeffort version checking added for DX2 and NT * 30-apr-97 jeffort version >DX5 treated as DX5 * 10-jul-97 jeffort made OSVersion a static variable * 09-sep-97 mikear QI for IUnknown when aggregating ***************************************************************************/ #include "ddfactry.h"
//#defines for registry lookup
#define REGSTR_PATH_DDRAW "Software\\Microsoft\\DirectDraw"
#define REGSTR_VAL_DDRAW_OWNDC "OWNDC"
CDDFactory::CDDFactory(IUnknown *pUnkOuter) : m_cRef(1), m_pUnkOuter(pUnkOuter != 0 ? pUnkOuter : CAST_TO_IUNKNOWN(this)) { m_hDDrawDLL = NULL; DllAddRef(); }
STDAPI DirectDrawFactory_CreateInstance( IUnknown * pUnkOuter, REFIID riid, void ** ppv) { HRESULT hr; CDDFactory *pFactory = new CDDFactory(pUnkOuter);
if( !pFactory ) { hr = E_OUTOFMEMORY; } else { hr = pFactory->NonDelegatingQueryInterface(pUnkOuter ? IID_IUnknown : riid, ppv); pFactory->NonDelegatingRelease(); } return hr; }
STDMETHODIMP CDDFactory::NonDelegatingQueryInterface(REFIID riid, void ** ppv) { *ppv=NULL;
if (IID_IUnknown==riid) { *ppv=(INonDelegatingUnknown *)this; } else { if (IID_IDirectDrawFactory==riid) { *ppv=(IDirectDrawFactory *)this; } else { return E_NOINTERFACE; } }
((LPUNKNOWN)*ppv)->AddRef(); return NOERROR; }
STDMETHODIMP_(ULONG) CDDFactory::NonDelegatingAddRef() { return InterlockedIncrement(&m_cRef); }
STDMETHODIMP_(ULONG) CDDFactory::NonDelegatingRelease() { LONG lRefCount = InterlockedDecrement(&m_cRef); if( lRefCount ) { return lRefCount; } delete this; DllRelease(); return 0; }
// Standard IUnknown
STDMETHODIMP CDDFactory::QueryInterface(REFIID riid, void ** ppv) { return m_pUnkOuter->QueryInterface(riid, ppv); }
STDMETHODIMP_(ULONG) CDDFactory::AddRef(void) { return m_pUnkOuter->AddRef(); }
STDMETHODIMP_(ULONG) CDDFactory::Release(void) { return m_pUnkOuter->Release(); }
/*
* CDDFactory::CreateDirectDraw */ STDMETHODIMP CDDFactory::CreateDirectDraw( GUID * pGUID, HWND hWnd, DWORD dwCoopLevelFlags, DWORD dwReserved, IUnknown *pUnkOuter, IDirectDraw **ppDirectDraw ) { static OSVERSIONINFO osVer; CDirectDrawEx *pDirectDrawEx; HRESULT hr = S_OK; BOOL fDDrawDllVerFound = FALSE;
*ppDirectDraw = NULL; /*
* first, see if we can get at DirectDraw or not! */ if( m_hDDrawDLL == NULL ) { char path[_MAX_PATH]; char DllName[25]; DWORD dwRet;
// m_hDDrawDLL = LoadLibrary( "ddraw.dll" );
dwRet = GetProfileString("ddrawex","realdll","ddraw.dll",DllName,25); if( dwRet == 0 ) { return DDERR_GENERIC; } m_hDDrawDLL = LoadLibrary( DllName ); if( m_hDDrawDLL == NULL ) { return DDERR_LOADFAILED; }
/*
* get ddraw.dll version number */ if( GetModuleFileName( (HINSTANCE)m_hDDrawDLL, path, sizeof( path ) ) ) { int size; DWORD tmp; size = (int) GetFileVersionInfoSize( path, (LPDWORD) &tmp ); if( size != 0 ) { LPVOID vinfo; vinfo = (LPVOID) LocalAlloc( LPTR, size ); if(vinfo == NULL) { hr = DDERR_OUTOFMEMORY; goto CleanUp; }
if( GetFileVersionInfo( path, 0, size, vinfo ) ) { VS_FIXEDFILEINFO *ver=NULL; UINT cb;
if( VerQueryValue(vinfo, "\\", (LPVOID *)&ver, &cb) ) { if( ver != NULL ) { /*
* we only need the most significant dword, * the LS dword contains the build number only... * Hack: The '|5' forces recognition of DX6+ */ m_dwDDVerMS = ver->dwFileVersionMS | 5; fDDrawDllVerFound = TRUE; } } } LocalFree( vinfo ); } }
osVer.dwOSVersionInfoSize = sizeof (OSVERSIONINFO); if ( !fDDrawDllVerFound || !GetVersionEx(&osVer) ) { hr = DDERR_BADVERSIONINFO; goto CleanUp; }
//if we are on NT4.0 Gold, the DLL version will be 4.00
if (osVer.dwPlatformId == VER_PLATFORM_WIN32_NT && LOWORD( m_dwDDVerMS) == 0) { //boost the LOWORD up so we will not fail the next check below
m_dwDDVerMS += 3; }
/*
* don't work on anything but DX2 DX 3 or DX 5. */ if( !((HIWORD( m_dwDDVerMS ) >= 4) && (LOWORD( m_dwDDVerMS) >= 3)) ) { hr = DDERR_UNSUPPORTED; goto CleanUp; }
if( LOWORD( m_dwDDVerMS) > 5) { //we will assume that any version >= DX5 will support what we need, so we
//will mark anything greater than 5 as DX5
m_dwDDVerMS = 5; }
/*
* get the various entry points we need */ m_pDirectDrawCreate = (LPDIRECTDRAWCREATE) GetProcAddress((HINSTANCE) m_hDDrawDLL, "DirectDrawCreate" ); if( m_pDirectDrawCreate == NULL ) { hr = DDERR_BADPROCADDRESS; goto CleanUp; }
m_pDirectDrawEnumerateW = (LPDIRECTDRAWENUMW) GetProcAddress( (HINSTANCE)m_hDDrawDLL, "DirectDrawEnumerateW" ); if( m_pDirectDrawEnumerateW == NULL ) { hr = DDERR_BADPROCADDRESS; goto CleanUp; }
m_pDirectDrawEnumerateA = (LPDIRECTDRAWENUMA) GetProcAddress((HINSTANCE) m_hDDrawDLL, "DirectDrawEnumerateA" ); if( m_pDirectDrawEnumerateA == NULL ) { hr = DDERR_BADPROCADDRESS; goto CleanUp; }
} // m_hDDrawDLL = NULL
/*
* create and initialize the ddrawex object */ pDirectDrawEx = new CDirectDrawEx(pUnkOuter); if( !pDirectDrawEx ) { hr = DDERR_OUTOFMEMORY; goto CleanUp; } else { hr = pDirectDrawEx->Init( pGUID, hWnd, dwCoopLevelFlags, dwReserved, m_pDirectDrawCreate );
if( SUCCEEDED(hr) ) { hr = pDirectDrawEx->NonDelegatingQueryInterface(IID_IDirectDraw, (void **)ppDirectDraw); /*
* save the ddraw version number... */ if (osVer.dwPlatformId == VER_PLATFORM_WIN32_NT) { if( LOWORD( m_dwDDVerMS ) == 5 ) { DWORD type; DWORD value; DWORD cb; HKEY hkey;
//DX5 is busted with OwnDC for StretchBlt
//check a registry key to see if we are
//using dx5 style or dx3 style.
//default is dx3
pDirectDrawEx->m_dwDDVer = WINNT_DX5;
if( ERROR_SUCCESS == RegOpenKey( HKEY_LOCAL_MACHINE, REGSTR_PATH_DDRAW, &hkey ) ) { cb = sizeof( value ); if( ERROR_SUCCESS == RegQueryValueEx( hkey, REGSTR_VAL_DDRAW_OWNDC, NULL, &type, (CONST LPBYTE)&value, &cb ) ) { pDirectDrawEx->m_dwDDVer = WINNT_DX5; } RegCloseKey(hkey); } } else if (LOWORD (m_dwDDVerMS) == 4 ) { pDirectDrawEx->m_dwDDVer = WINNT_DX3; } else if (LOWORD (m_dwDDVerMS) == 3 ) { pDirectDrawEx->m_dwDDVer = WINNT_DX2; } //should never get here, alread checked above, but be conservative
else { hr = DDERR_UNSUPPORTED; goto CleanUp; } } else { if( LOWORD( m_dwDDVerMS ) == 5 ) { pDirectDrawEx->m_dwDDVer = WIN95_DX5; } else if (LOWORD (m_dwDDVerMS) == 4 ) { pDirectDrawEx->m_dwDDVer = WIN95_DX3; } else if (LOWORD (m_dwDDVerMS) == 3 ) { pDirectDrawEx->m_dwDDVer = WIN95_DX2; } //should never get here, alread checked above, but be conservative
else { hr = DDERR_UNSUPPORTED; goto CleanUp; } } } pDirectDrawEx->NonDelegatingRelease(); }
CleanUp:
if( hr != S_OK && m_hDDrawDLL != NULL ) { FreeLibrary((HINSTANCE) m_hDDrawDLL ); m_hDDrawDLL = NULL; }
return hr; } /* CDDFactory::CreateDirectDraw */
/*
* CDDFactory::DirectDrawEnumerate * * implements ddraw enumerate. */ STDMETHODIMP CDDFactory::DirectDrawEnumerate(LPDDENUMCALLBACK lpCallback, LPVOID lpContext) { #pragma message( REMIND( "DDFactory::DirectDrawEnumerate assumes ANSI" ))
return m_pDirectDrawEnumerateA(lpCallback, lpContext);
} /* CDDFactory::DirectDrawEnumerate */
|