798 lines
29 KiB
798 lines
29 KiB
/*==LICENSE==* |
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools |
|
Copyright (C) 2011 Cyan Worlds, Inc. |
|
|
|
This program 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 3 of the License, or |
|
(at your option) any later version. |
|
|
|
This program 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, see <http://www.gnu.org/licenses/>. |
|
|
|
Additional permissions under GNU GPL version 3 section 7 |
|
|
|
If you modify this Program, or any covered work, by linking or |
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK, |
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent |
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK |
|
(or a modified version of those libraries), |
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA, |
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG |
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the |
|
licensors of this Program grant you additional |
|
permission to convey the resulting work. Corresponding Source for a |
|
non-source form of such a combination shall include the source code for |
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered |
|
work. |
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com |
|
or by snail mail at: |
|
Cyan Worlds, Inc. |
|
14617 N Newport Hwy |
|
Mead, WA 99021 |
|
|
|
*==LICENSE==*/ |
|
|
|
#include "plDXEnumerate.h" |
|
#include "plPipeline/hsG3DDeviceSelector.h" |
|
#include "hsGDirect3D.h" |
|
|
|
|
|
//// Local Typedefs /////////////////////////////////////////////////////////// |
|
|
|
typedef LPDIRECT3D9 (WINAPI * Direct3DCreateProc)( UINT sdkVersion ); |
|
|
|
const uint8_t hsGDirect3DTnLEnumerate::kNumDisplayFormats = 6; |
|
const D3DFORMAT hsGDirect3DTnLEnumerate::kDisplayFormats[] = |
|
{ |
|
D3DFMT_A1R5G5B5, |
|
D3DFMT_A2B10G10R10, |
|
D3DFMT_A8R8G8B8, |
|
D3DFMT_R5G6B5, |
|
D3DFMT_X1R5G5B5, |
|
D3DFMT_X8R8G8B8, |
|
}; |
|
|
|
HRESULT hsGDirect3DTnLEnumerate::SelectFromDevMode(const hsG3DDeviceRecord* devRec, const hsG3DDeviceMode* devMode) |
|
{ |
|
|
|
int i; |
|
for( i = 0; i < GetNumDrivers(); i++ ) |
|
{ |
|
if (devRec->GetDriverDesc().CompareI(GetDriver(i)->fAdapterInfo.Description) == 0) |
|
{ |
|
int j; |
|
for( j = 0; j < GetDriver(i)->fDevices.GetCount(); j++ ) |
|
{ |
|
if (devRec->GetDeviceDesc().CompareI(GetDriver(i)->fDevices[j].fStrName) == 0) |
|
{ |
|
SetCurrentDriver(GetDriver(i)); |
|
SetCurrentDevice(&GetDriver(i)->fDevices[j]); |
|
D3DEnum_SelectDefaultMode( |
|
devMode->GetWidth(), |
|
devMode->GetHeight(), |
|
devMode->GetColorDepth()); |
|
return false; |
|
} |
|
} |
|
} |
|
} |
|
plString errStr = plString::Format("Can't find requested device - %s:%s:%s:%s:%s", |
|
devRec->GetG3DDeviceTypeName(), |
|
devRec->GetDriverDesc().c_str(), |
|
devRec->GetDriverName().c_str(), |
|
devRec->GetDriverVersion().c_str(), |
|
devRec->GetDeviceDesc().c_str()); |
|
|
|
DWORD enumFlags = 0; |
|
int width = devMode->GetWidth(); |
|
int height = devMode->GetHeight(); |
|
int colorDepth = devMode->GetColorDepth(); |
|
// for a window, take whatever colordepth we can get. |
|
if( !colorDepth ) |
|
enumFlags |= D3DENUM_CANWINDOW; |
|
enumFlags |= D3DENUM_TNLHAL; |
|
|
|
D3DEnum_SelectDefaultDriver(enumFlags); |
|
|
|
// If we didn't get what we want, try for anything. |
|
if( !GetCurrentDriver() || !GetCurrentDevice() ) |
|
{ |
|
enumFlags = colorDepth ? 0 : D3DENUM_CANWINDOW; |
|
D3DEnum_SelectDefaultDriver(enumFlags); |
|
} |
|
if( !GetCurrentDriver() || !GetCurrentDevice() ) |
|
D3DEnum_SelectDefaultDriver(0); |
|
if( !GetCurrentDriver() || !GetCurrentDevice() ) |
|
{ |
|
if( !*GetEnumeErrorStr() ) |
|
SetEnumeErrorStr("Error finding device"); |
|
return true; |
|
} |
|
D3DEnum_SelectDefaultMode(width, height, colorDepth); |
|
if( !GetCurrentMode() ) |
|
{ |
|
if( !*GetEnumeErrorStr() ) |
|
SetEnumeErrorStr("Error finding mode"); |
|
return true; |
|
} |
|
|
|
return false; |
|
} |
|
|
|
HRESULT hsGDirect3DTnLEnumerate::D3DEnum_SelectDefaultMode(int width, int height, int depth) |
|
{ |
|
hsAssert(GetCurrentDriver() && GetCurrentDevice(), "Must have selected device already"); |
|
|
|
BOOL windowed = false; |
|
if (depth == 0) |
|
{ |
|
// Legacy code writes out 0 bit depth to mean windowed |
|
windowed = true; |
|
depth = 32; |
|
} |
|
|
|
D3DEnum_DeviceInfo* device = GetCurrentDevice(); |
|
int i; |
|
for( i = 0; i < device->fModes.GetCount(); i++ ) |
|
{ |
|
D3DEnum_ModeInfo* mode = &device->fModes[i]; |
|
if (mode->fWindowed != windowed) |
|
continue; |
|
|
|
if( depth ) |
|
{ |
|
if( width < mode->fDDmode.Width ) |
|
continue; |
|
if( height < mode->fDDmode.Height ) |
|
continue; |
|
} |
|
if( depth < mode->fBitDepth ) |
|
continue; |
|
|
|
if( GetCurrentMode() ) |
|
{ |
|
D3DEnum_ModeInfo* curMode = GetCurrentDriver()->fCurrentMode; |
|
if( depth ) |
|
{ |
|
if( curMode->fDDmode.Width > mode->fDDmode.Width ) |
|
continue; |
|
if( curMode->fDDmode.Height > mode->fDDmode.Height ) |
|
continue; |
|
} |
|
if( curMode->fBitDepth > mode->fBitDepth ) |
|
continue; |
|
|
|
} |
|
|
|
SetCurrentMode(mode); |
|
} |
|
return S_OK; |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
// Name: D3DEnum_SelectDefaultDriver() |
|
// Desc: Picks a default driver according to the passed in flags. |
|
//----------------------------------------------------------------------------- |
|
HRESULT hsGDirect3DTnLEnumerate::D3DEnum_SelectDefaultDriver( DWORD dwFlags ) |
|
{ |
|
|
|
|
|
// If a specific driver was requested, perform that search here |
|
if( dwFlags & D3DENUM_MASK ) |
|
{ |
|
int i; |
|
for( i = 0; i < fDrivers.GetCount(); i++ ) |
|
{ |
|
D3DEnum_DriverInfo* pDriver = &fDrivers[i]; |
|
int j; |
|
for( j = 0; j < pDriver->fDevices.GetCount(); j++ ) |
|
{ |
|
D3DEnum_DeviceInfo* pDevice = &pDriver->fDevices[j]; |
|
BOOL bFound = FALSE; |
|
|
|
if( pDevice->fDDType == D3DDEVTYPE_REF ) |
|
{ |
|
if( dwFlags & D3DENUM_REFERENCERAST ) |
|
bFound = TRUE; |
|
} |
|
else if( pDevice->fDDType == D3DDEVTYPE_HAL && |
|
pDevice->fDDCaps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT ) |
|
{ |
|
if( dwFlags & D3DENUM_TNLHAL ) |
|
bFound = TRUE; |
|
} |
|
else |
|
{ |
|
if( dwFlags & D3DENUM_CANWINDOW ) |
|
{ |
|
if( (pDriver == &fDrivers[0]) ) |
|
{ |
|
if( ( pDevice->fDDCaps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT ) |
|
^ !(dwFlags & D3DENUM_TNLHAL) ) |
|
bFound = TRUE; |
|
} |
|
} |
|
else |
|
if( dwFlags & D3DENUM_PRIMARYHAL ) |
|
{ |
|
if( pDriver == &fDrivers[0] ) |
|
bFound = TRUE; |
|
} |
|
else |
|
if( dwFlags & D3DENUM_SECONDARYHAL ) |
|
{ |
|
if( pDriver != &fDrivers[0] ) |
|
bFound = TRUE; |
|
} |
|
} |
|
|
|
if( bFound ) |
|
{ |
|
SetCurrentDriver(pDriver); |
|
SetCurrentDevice(pDevice); |
|
return S_OK; |
|
} |
|
} |
|
} |
|
return D3DENUMERR_NOTFOUND; |
|
} |
|
|
|
int i; |
|
for( i = 0; i < fDrivers.GetCount(); i++ ) |
|
{ |
|
D3DEnum_DriverInfo* pDriver = &fDrivers[i]; |
|
int j; |
|
for( j = 0; j < pDriver->fDevices.GetCount(); j++ ) |
|
{ |
|
D3DEnum_DeviceInfo* pDevice = &pDriver->fDevices[j]; |
|
if( !pDevice->fIsHardware ) |
|
continue; |
|
|
|
SetCurrentDriver(pDriver); |
|
SetCurrentDevice(pDevice); |
|
|
|
return S_OK; |
|
} |
|
} |
|
|
|
// No compatible devices were found. Return an error code |
|
return D3DENUMERR_NOCOMPATIBLEDEVICES; |
|
} |
|
|
|
|
|
//// Constructor ////////////////////////////////////////////////////////////// |
|
// |
|
// Inits the enumeration and builds our list of devices/whatever. |
|
|
|
hsGDirect3DTnLEnumerate::hsGDirect3DTnLEnumerate() |
|
{ |
|
memset( &fEnumeErrorStr[0], 0x00, sizeof(fEnumeErrorStr) ); |
|
|
|
fCurrentDriver = NULL; // The selected DD driver |
|
fDrivers.Reset(); // List of DD drivers |
|
|
|
// Create a D3D object to use |
|
IDirect3D9* pD3D = hsGDirect3D::GetDirect3D(); |
|
if (!pD3D) |
|
{ |
|
strcpy( fEnumeErrorStr, "Cannot load DirectX!" ); |
|
return; |
|
} |
|
|
|
/// Loop through the "adapters" (we don't call them drivers anymore) |
|
UINT iAdapter; |
|
for( iAdapter = 0; iAdapter < pD3D->GetAdapterCount(); iAdapter++ ) |
|
{ |
|
D3DEnum_DriverInfo* newDriver = fDrivers.Push(); |
|
ZeroMemory( newDriver, sizeof( *newDriver ) ); |
|
|
|
// Copy data to a device info structure |
|
D3DADAPTER_IDENTIFIER9 adapterInfo; |
|
pD3D->GetAdapterIdentifier( iAdapter, 0, &adapterInfo ); |
|
pD3D->GetAdapterDisplayMode( iAdapter, &newDriver->fDesktopMode ); |
|
|
|
memcpy( &newDriver->fAdapterInfo, &adapterInfo, sizeof( adapterInfo ) ); |
|
strncpy( newDriver->fStrName, adapterInfo.Driver, 39 ); |
|
strncpy( newDriver->fStrDesc, adapterInfo.Description, 39 ); |
|
newDriver->fGuid = adapterInfo.DeviceIdentifier; |
|
newDriver->fMemory = 16 * 1024 * 1024; /// Simulate 16 MB |
|
|
|
/// Do the mode and device enumeration for this adapter |
|
IEnumAdapterDevices( pD3D, iAdapter, newDriver ); |
|
} |
|
} |
|
|
|
//// IEnumAdapterDevices ////////////////////////////////////////////////////// |
|
// |
|
// DirectX: Enumerates all the modes for a given adapter, then using the |
|
// two faked modes for HAL and REF, attaches the modes to each "device" that |
|
// can support them. |
|
|
|
void hsGDirect3DTnLEnumerate::IEnumAdapterDevices( IDirect3D9 *pD3D, UINT iAdapter, D3DEnum_DriverInfo *drivInfo ) |
|
{ |
|
// A bit backwards from DX8... First we have to go through our list of formats and check for validity. |
|
// Then we can enum through the modes for each format. |
|
|
|
const DWORD numDeviceTypes = 2; |
|
const TCHAR* strDeviceDescs[] = { "HAL", "REF" }; |
|
const D3DDEVTYPE deviceTypes[] = { D3DDEVTYPE_HAL, D3DDEVTYPE_REF }; |
|
|
|
BOOL *formatWorks = new BOOL[kNumDisplayFormats + 1]; // One for each format |
|
DWORD *behavior = new DWORD[kNumDisplayFormats + 1]; |
|
UINT iDevice; |
|
for (iDevice = 0; iDevice < numDeviceTypes; iDevice++) |
|
{ |
|
D3DEnum_DeviceInfo *deviceInfo = drivInfo->fDevices.Push(); |
|
ZeroMemory(deviceInfo, sizeof(*deviceInfo)); |
|
|
|
pD3D->GetDeviceCaps(iAdapter, deviceTypes[iDevice], &deviceInfo->fDDCaps); |
|
strncpy(deviceInfo->fStrName, strDeviceDescs[iDevice], 39); |
|
deviceInfo->fDDType = deviceTypes[iDevice]; |
|
deviceInfo->fIsHardware = deviceInfo->fDDCaps.DevCaps & D3DDEVCAPS_HWRASTERIZATION; |
|
|
|
/// Loop through the formats, checking each against this device to see |
|
/// if it will work. If so, add all modes matching that format |
|
uint8_t iFormat; |
|
for (iFormat = 0; iFormat < kNumDisplayFormats + 1; iFormat++ ) |
|
{ |
|
// the desktop format gets to be first, everything else is nudged over one. |
|
D3DFORMAT currFormat = (iFormat == 0 ? drivInfo->fDesktopMode.Format : kDisplayFormats[iFormat - 1]); |
|
formatWorks[iFormat] = FALSE; |
|
|
|
int bitDepth = IGetDXBitDepth(currFormat); |
|
if (bitDepth == 0) |
|
continue; // Don't like this mode, skip it |
|
|
|
/// Can it be used as a render target? |
|
if (FAILED(pD3D->CheckDeviceType(iAdapter, deviceTypes[iDevice], |
|
currFormat, |
|
currFormat, |
|
FALSE))) |
|
continue; // Nope--skip it |
|
|
|
if (deviceInfo->fDDCaps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT) |
|
{ |
|
/// Confirm that HW vertex processing works on this device |
|
if (deviceInfo->fDDCaps.DevCaps & D3DDEVCAPS_PUREDEVICE) |
|
{ |
|
behavior[iFormat] = D3DCREATE_HARDWARE_VERTEXPROCESSING; |
|
if (SUCCEEDED(IConfirmDevice(&deviceInfo->fDDCaps, behavior[iFormat], |
|
currFormat))) |
|
{ |
|
formatWorks[iFormat] = TRUE; |
|
} |
|
} |
|
|
|
if (!formatWorks[iFormat]) |
|
{ |
|
/// HW vertex & Pure didn't work--just try HW vertex |
|
behavior[iFormat] = D3DCREATE_HARDWARE_VERTEXPROCESSING; |
|
if (SUCCEEDED(IConfirmDevice(&deviceInfo->fDDCaps, behavior[iFormat], |
|
currFormat))) |
|
{ |
|
formatWorks[iFormat] = TRUE; |
|
} |
|
} |
|
|
|
if (!formatWorks[iFormat]) |
|
{ |
|
/// HW vertex didn't work--can we do mixed? |
|
behavior[iFormat] = D3DCREATE_MIXED_VERTEXPROCESSING; |
|
|
|
if (SUCCEEDED(IConfirmDevice(&deviceInfo->fDDCaps, behavior[iFormat], |
|
currFormat))) |
|
{ |
|
formatWorks[iFormat] = TRUE; |
|
} |
|
} |
|
} |
|
|
|
if (!formatWorks[iFormat]) |
|
{ |
|
/// Egads. Try SW vertex processing |
|
behavior[iFormat] = D3DCREATE_SOFTWARE_VERTEXPROCESSING; |
|
|
|
if (SUCCEEDED(IConfirmDevice(&deviceInfo->fDDCaps, behavior[iFormat], |
|
currFormat))) |
|
{ |
|
formatWorks[iFormat] = TRUE; |
|
} |
|
} |
|
|
|
if (formatWorks[iFormat]) |
|
{ |
|
/// Now go through all the modes. If a given mode has a format that works, |
|
/// add it to the device |
|
UINT numAdapterModes = pD3D->GetAdapterModeCount(iAdapter, currFormat); |
|
DWORD iMode; |
|
for (iMode = 0; iMode < numAdapterModes; iMode++) |
|
{ |
|
// TODO: Check for modes that only differ by refresh rate and exclude duplicates. |
|
|
|
/// Get the mode attributes |
|
D3DDISPLAYMODE dispMode; |
|
pD3D->EnumAdapterModes(iAdapter, currFormat, iMode, &dispMode); |
|
{ |
|
/// Add it to our driver's global mode list |
|
D3DEnum_ModeInfo *modeInfo = drivInfo->fModes.Push(); |
|
ZeroMemory( modeInfo, sizeof( *modeInfo ) ); |
|
modeInfo->fDDmode = dispMode; |
|
sprintf( modeInfo->fStrDesc, TEXT( "%ld x %ld x %ld" ), dispMode.Width, dispMode.Height, bitDepth ); |
|
modeInfo->fBitDepth = bitDepth; |
|
|
|
// Add it to the device |
|
modeInfo->fDDBehavior = behavior[ iFormat ]; |
|
IFindDepthFormats( pD3D, iAdapter, deviceInfo->fDDType, modeInfo ); |
|
IFindFSAATypes( pD3D, iAdapter, deviceInfo->fDDType, modeInfo ); |
|
ICheckCubicRenderTargets( pD3D, iAdapter, deviceInfo->fDDType, modeInfo ); |
|
deviceInfo->fModes.Append( *modeInfo ); |
|
|
|
// Special check for the desktop, which we know is the first entry, because we put it there. |
|
if (iFormat == 0) |
|
{ |
|
/// Check if the device can window and/or is compatible with the desktop display mode |
|
deviceInfo->fCompatibleWithDesktop = TRUE; |
|
|
|
// As of DirectX 9, any device supports windowed mode |
|
//if (deviceInfo->fDDCaps.Caps2 & D3DCAPS2_CANRENDERWINDOWED) |
|
{ |
|
deviceInfo->fCanWindow = TRUE; |
|
|
|
/// Add a fake mode to represent windowed. Silly, but here for legacy |
|
D3DEnum_ModeInfo *pModeInfo = drivInfo->fModes.Push(); |
|
ZeroMemory(pModeInfo, sizeof(*pModeInfo)); |
|
pModeInfo->fDDmode = dispMode; |
|
pModeInfo->fDDBehavior = behavior[iFormat]; |
|
pModeInfo->fBitDepth = bitDepth; |
|
sprintf(pModeInfo->fStrDesc, TEXT("Windowed")); |
|
pModeInfo->fWindowed = true; |
|
|
|
IFindDepthFormats(pD3D, iAdapter, deviceInfo->fDDType, pModeInfo); |
|
IFindFSAATypes(pD3D, iAdapter, deviceInfo->fDDType, pModeInfo); |
|
ICheckCubicRenderTargets(pD3D, iAdapter, deviceInfo->fDDType, pModeInfo); |
|
deviceInfo->fModes.Append( *pModeInfo ); |
|
} |
|
} |
|
} |
|
} |
|
|
|
|
|
} |
|
} |
|
} |
|
|
|
delete [] formatWorks; |
|
delete [] behavior; |
|
} |
|
|
|
//// IFindDepthFormats //////////////////////////////////////////////////////// |
|
// DirectX: Given a device and mode, find ALL available depth/stencil |
|
// formats and add them to the mode info struct. |
|
|
|
bool hsGDirect3DTnLEnumerate::IFindDepthFormats( IDirect3D9 *pD3D, UINT iAdapter, D3DDEVTYPE deviceType, |
|
D3DEnum_ModeInfo *modeInfo ) |
|
{ |
|
D3DFORMAT formats[] = { D3DFMT_D16, D3DFMT_D24X8, D3DFMT_D32, |
|
D3DFMT_D15S1, D3DFMT_D24X4S4, D3DFMT_D24S8, D3DFMT_UNKNOWN }; |
|
|
|
/// Try 'em |
|
for( int i = 0; formats[ i ] != D3DFMT_UNKNOWN; i++ ) |
|
{ |
|
if( SUCCEEDED( pD3D->CheckDeviceFormat( iAdapter, deviceType, modeInfo->fDDmode.Format, |
|
D3DUSAGE_DEPTHSTENCIL, D3DRTYPE_SURFACE, |
|
formats[ i ] ) ) ) |
|
{ |
|
if( SUCCEEDED( pD3D->CheckDepthStencilMatch( iAdapter, deviceType, |
|
modeInfo->fDDmode.Format, modeInfo->fDDmode.Format, formats[ i ] ) ) ) |
|
{ |
|
modeInfo->fDepthFormats.Append( formats[ i ] ); |
|
} |
|
} |
|
} |
|
|
|
return( modeInfo->fDepthFormats.GetCount() > 0 ? true : false ); |
|
} |
|
|
|
//// IFindFSAATypes /////////////////////////////////////////////////////////// |
|
// DirectX: Given a device and mode, find ALL available multisample types |
|
// and add them to the mode info struct. |
|
|
|
bool hsGDirect3DTnLEnumerate::IFindFSAATypes( IDirect3D9 *pD3D, UINT iAdapter, D3DDEVTYPE deviceType, |
|
D3DEnum_ModeInfo *modeInfo ) |
|
{ |
|
/// Try 'em |
|
for (int type = 2; type <= 16; type++) |
|
{ |
|
if (SUCCEEDED(pD3D->CheckDeviceMultiSampleType(iAdapter, deviceType, modeInfo->fDDmode.Format, |
|
modeInfo->fWindowed ? TRUE : FALSE, |
|
(D3DMULTISAMPLE_TYPE)type, NULL))) |
|
{ |
|
modeInfo->fFSAATypes.Append((D3DMULTISAMPLE_TYPE)type); |
|
} |
|
} |
|
|
|
return (modeInfo->fFSAATypes.GetCount() > 0 ? true : false); |
|
} |
|
|
|
//// ICheckCubicRenderTargets ///////////////////////////////////////////////// |
|
|
|
bool hsGDirect3DTnLEnumerate::ICheckCubicRenderTargets( IDirect3D9 *pD3D, UINT iAdapter, D3DDEVTYPE deviceType, |
|
D3DEnum_ModeInfo *modeInfo ) |
|
{ |
|
if( SUCCEEDED( pD3D->CheckDeviceFormat( iAdapter, deviceType, modeInfo->fDDmode.Format, |
|
D3DUSAGE_RENDERTARGET, D3DRTYPE_CUBETEXTURE, |
|
modeInfo->fDDmode.Format ) ) ) |
|
{ |
|
modeInfo->fCanRenderToCubic = true; |
|
return true; |
|
} |
|
|
|
modeInfo->fCanRenderToCubic = false; |
|
return false; |
|
} |
|
|
|
//// IConfirmDevice /////////////////////////////////////////////////////////// |
|
// |
|
// Nice, encapsulated way of testing for specific caps on a particular device |
|
|
|
HRESULT hsGDirect3DTnLEnumerate::IConfirmDevice( D3DCAPS9* pCaps, DWORD dwBehavior, |
|
D3DFORMAT Format ) |
|
{ |
|
short bits; |
|
|
|
|
|
bits = IGetDXBitDepth( Format ); |
|
if( bits == 16 || bits == 24 || bits == 32 ) |
|
return S_OK; |
|
|
|
return E_FAIL; |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
// Name: ~hsGDirect3DTnLEnumerate() |
|
// Desc: |
|
//----------------------------------------------------------------------------- |
|
hsGDirect3DTnLEnumerate::~hsGDirect3DTnLEnumerate() |
|
{ |
|
D3DEnum_FreeResources(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
// Name: D3DEnum_FreeResources() |
|
// Desc: Frees all resources used for driver enumeration |
|
//----------------------------------------------------------------------------- |
|
VOID hsGDirect3DTnLEnumerate::D3DEnum_FreeResources() |
|
{ |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
// Name: SetEnumeErrorStr() |
|
// Desc: |
|
//----------------------------------------------------------------------------- |
|
void hsGDirect3DTnLEnumerate::SetEnumeErrorStr(const char* s) |
|
{ |
|
hsStrncpy(fEnumeErrorStr, s, 128); |
|
} |
|
|
|
//// IGetDXBitDepth ////////////////////////////////////////////////////////// |
|
// |
|
// From a D3DFORMAT enumeration, return the bit depth associated with it. |
|
// Copied from hsGDirect3DDevice to prevent inclusion of that class in |
|
// the RenderMenu project (for some reason, VC can't figure out we're only |
|
// calling one static function!) |
|
|
|
short hsGDirect3DTnLEnumerate::IGetDXBitDepth( D3DFORMAT format ) |
|
{ |
|
#define ReturnDepth(type, depth) if (format == type) return depth |
|
|
|
ReturnDepth(D3DFMT_UNKNOWN, 0); |
|
ReturnDepth(D3DFMT_A8R8G8B8, 32); |
|
ReturnDepth(D3DFMT_X8R8G8B8, 32); |
|
ReturnDepth(D3DFMT_R5G6B5, 16); |
|
ReturnDepth(D3DFMT_X1R5G5B5, 16); |
|
ReturnDepth(D3DFMT_A1R5G5B5, 16); |
|
|
|
// Supported by DX9, but we don't currently support it. Can add support if needed. |
|
//ReturnDepth(D3DFMT_A2B10G10R10, 32); |
|
|
|
// Unsupported translation format--return 0 |
|
return 0; |
|
} |
|
|
|
/////////////////////////////////////////////////////////////////////////////// |
|
//// Direct3D DeviceSelector Code /////////////////////////////////////////// |
|
/////////////////////////////////////////////////////////////////////////////// |
|
|
|
//// IGetD3DCardInfo ///////////////////////////////////////////////////////// |
|
// Given two enum structs, strips out and produces the vendor ID, device ID |
|
// and the driver name. Returns true if processed, false otherwise. |
|
|
|
bool hsG3DDeviceSelector::IGetD3DCardInfo( hsG3DDeviceRecord &record, // In |
|
void *driverInfo, |
|
void *deviceInfo, |
|
uint32_t *vendorID, uint32_t *deviceID, // Out |
|
char **driverString, char **descString ) |
|
{ |
|
D3DEnum_DriverInfo *driverD3DInfo = (D3DEnum_DriverInfo *)driverInfo; |
|
D3DEnum_DeviceInfo *deviceD3DInfo = (D3DEnum_DeviceInfo *)deviceInfo; |
|
|
|
D3DADAPTER_IDENTIFIER9 *adapterInfo; |
|
|
|
adapterInfo = &driverD3DInfo->fAdapterInfo; |
|
|
|
*vendorID = adapterInfo->VendorId; |
|
*deviceID = adapterInfo->DeviceId; |
|
*driverString = adapterInfo->Driver; |
|
*descString = adapterInfo->Description; |
|
|
|
return true; |
|
} |
|
|
|
//// ITryDirect3DTnL ////////////////////////////////////////////////////////// |
|
|
|
void hsG3DDeviceSelector::ITryDirect3DTnL(hsWinRef winRef) |
|
{ |
|
hsGDirect3DTnLEnumerate& d3dEnum = hsGDirect3D::EnumerateTnL(); |
|
|
|
int i; |
|
for( i = 0; i < d3dEnum.GetNumDrivers(); i++ ) |
|
{ |
|
ITryDirect3DTnLDriver(d3dEnum.GetDriver(i)); |
|
} |
|
} |
|
|
|
//// ITryDirect3DDriver /////////////////////////////////////////////////////// |
|
// |
|
// New DirectX Way |
|
|
|
void hsG3DDeviceSelector::ITryDirect3DTnLDriver(D3DEnum_DriverInfo* drivInfo) |
|
{ |
|
hsG3DDeviceRecord devRec; |
|
devRec.Clear(); |
|
devRec.SetG3DDeviceType( kDevTypeDirect3D ); |
|
|
|
devRec.SetDriverName( drivInfo->fAdapterInfo.Driver ); |
|
devRec.SetDriverDesc( drivInfo->fAdapterInfo.Description ); |
|
|
|
char buff[ 256 ]; |
|
sprintf( buff, "%d.%02d.%02d.%04d", |
|
HIWORD( drivInfo->fAdapterInfo.DriverVersion.u.HighPart ), |
|
LOWORD( drivInfo->fAdapterInfo.DriverVersion.u.HighPart ), |
|
HIWORD( drivInfo->fAdapterInfo.DriverVersion.u.LowPart ), |
|
LOWORD( drivInfo->fAdapterInfo.DriverVersion.u.LowPart ) ); |
|
|
|
|
|
devRec.SetDriverVersion(buff); |
|
|
|
devRec.SetMemoryBytes(drivInfo->fMemory); |
|
|
|
int i; |
|
for( i = 0; i < drivInfo->fDevices.GetCount(); i++ ) |
|
{ |
|
/// 9.6.2000 mcn - Changed here so we can do fudging here, rather |
|
/// than passing all the messy driver data to the function |
|
hsG3DDeviceRecord currDevRec = devRec; |
|
|
|
/// Done first now, so we can alter the D3D type later |
|
ITryDirect3DTnLDevice( &drivInfo->fDevices[i], currDevRec ); |
|
IFudgeDirectXDevice( currDevRec, (D3DEnum_DriverInfo *)drivInfo, (D3DEnum_DeviceInfo *)&drivInfo->fDevices[ i ] ); |
|
|
|
if( currDevRec.GetModes().GetCount() ) |
|
fRecords.Append( currDevRec ); |
|
} |
|
} |
|
|
|
//// ITryDirect3DTnLDevice //////////////////////////////////////////////////// |
|
// |
|
// New DirectX Way |
|
|
|
void hsG3DDeviceSelector::ITryDirect3DTnLDevice(D3DEnum_DeviceInfo* devInfo, hsG3DDeviceRecord& devRec) |
|
{ |
|
devRec.SetDeviceDesc(devInfo->fStrName); |
|
|
|
if( devInfo->fDDType == D3DDEVTYPE_REF ) |
|
devRec.SetG3DHALorHEL( kHHD3DRefDev ); |
|
else if( devInfo->fDDType == D3DDEVTYPE_HAL ) |
|
{ |
|
if( devInfo->fDDCaps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT ) |
|
{ |
|
devRec.SetG3DHALorHEL( kHHD3DTnLHalDev ); |
|
devRec.SetCap( kCapsHWTransform ); |
|
} |
|
else |
|
devRec.SetG3DHALorHEL( kHHD3DHALDev ); |
|
} |
|
|
|
if( devInfo->fDDCaps.TextureCaps & D3DPTEXTURECAPS_CUBEMAP ) |
|
devRec.SetCap( kCapsCubicTextures ); |
|
|
|
devRec.SetLayersAtOnce( devInfo->fDDCaps.MaxSimultaneousTextures ); |
|
|
|
if( devInfo->fDDCaps.TextureFilterCaps & D3DPTFILTERCAPS_MIPFLINEAR ) |
|
devRec.SetCap( kCapsMipmap ); |
|
if( devInfo->fDDCaps.TextureCaps & D3DPTEXTURECAPS_MIPCUBEMAP ) |
|
devRec.SetCap( kCapsCubicMipmap ); |
|
if( devInfo->fDDCaps.TextureCaps & D3DPTEXTURECAPS_PERSPECTIVE ) |
|
devRec.SetCap(kCapsPerspective); |
|
if( devInfo->fIsHardware ) |
|
devRec.SetCap( kCapsHardware ); |
|
if( devInfo->fDDCaps.RasterCaps & D3DPRASTERCAPS_FOGTABLE ) |
|
{ |
|
devRec.SetCap( kCapsFogLinear ); |
|
devRec.SetCap( kCapsFogExp ); |
|
devRec.SetCap( kCapsFogExp2 ); |
|
devRec.SetCap( kCapsPixelFog ); |
|
} |
|
else |
|
{ |
|
devRec.SetCap( kCapsFogLinear ); |
|
} |
|
if( devInfo->fDDCaps.RasterCaps & D3DPRASTERCAPS_FOGRANGE ) |
|
devRec.SetCap( kCapsFogRange ); |
|
|
|
if( devInfo->fDDCaps.MaxAnisotropy <= 1 ) |
|
devRec.SetMaxAnisotropicSamples( 0 ); |
|
else |
|
devRec.SetMaxAnisotropicSamples( (uint8_t)devInfo->fDDCaps.MaxAnisotropy ); |
|
|
|
if (D3DSHADER_VERSION_MAJOR(devInfo->fDDCaps.PixelShaderVersion) > 0) |
|
devRec.SetCap(kCapsPixelShader); |
|
|
|
/// Assume these by default |
|
devRec.SetCap( kCapsCompressTextures ); |
|
devRec.SetCap( kCapsDoesSmallTextures ); |
|
|
|
#if 1 // mf - want to leave this one off by default |
|
// if( devInfo->fCanAntialias ) |
|
// devRec.SetCap( kCapsAntiAlias ); |
|
#endif // mf - want to leave this one off by default |
|
|
|
hsG3DDeviceMode devMode; |
|
int i, j; |
|
|
|
const struct |
|
{ |
|
D3DFORMAT fmt; uint16_t depth; |
|
} depths[] = { { D3DFMT_D16, 0x0010 }, { D3DFMT_D24X8, 0x0018 }, { D3DFMT_D32, 0x0020 }, |
|
{ D3DFMT_D15S1, 0x010f }, { D3DFMT_D24X4S4, 0x0418 }, { D3DFMT_D24S8, 0x0818 }, { D3DFMT_UNKNOWN, 0 } }; |
|
|
|
for( i = 0; i < devInfo->fModes.GetCount(); i++ ) |
|
{ |
|
D3DEnum_ModeInfo* modeInfo = &devInfo->fModes[i]; |
|
|
|
devMode.Clear(); |
|
devMode.SetWidth( modeInfo->fDDmode.Width ); |
|
devMode.SetHeight( modeInfo->fDDmode.Height ); |
|
devMode.SetColorDepth( modeInfo->fBitDepth ); |
|
|
|
if( modeInfo->fCanRenderToCubic ) |
|
devMode.SetCanRenderToCubics( true ); |
|
else |
|
devMode.SetCanRenderToCubics( false ); |
|
|
|
for( j = 0; depths[ j ].depth != 0; j++ ) |
|
{ |
|
if( modeInfo->fDepthFormats.Find( depths[ j ].fmt ) != modeInfo->fDepthFormats.kMissingIndex ) |
|
devMode.AddZStencilDepth( depths[ j ].depth ); |
|
} |
|
|
|
for( j = 2; j <= 16; j++ ) |
|
{ |
|
if( modeInfo->fFSAATypes.Find( (D3DMULTISAMPLE_TYPE)j ) != modeInfo->fFSAATypes.kMissingIndex ) |
|
devMode.AddFSAAType( j ); |
|
} |
|
|
|
devRec.GetModes().Append( devMode ); |
|
} |
|
} |
|
|
|
|
|
|