2010-08-13 08:43:35 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2012-05-21 04:12:37 -07:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2010-08-13 08:43:35 -07:00
|
|
|
|
2011-10-10 22:50:08 -07:00
|
|
|
#include "mozilla/Util.h"
|
|
|
|
|
2010-08-27 08:49:02 -07:00
|
|
|
#include <windows.h>
|
2011-01-24 07:05:59 -08:00
|
|
|
#include <setupapi.h>
|
2010-08-13 08:43:35 -07:00
|
|
|
#include "gfxWindowsPlatform.h"
|
|
|
|
#include "GfxInfo.h"
|
2010-12-16 12:49:54 -08:00
|
|
|
#include "GfxInfoWebGL.h"
|
2010-08-27 08:49:02 -07:00
|
|
|
#include "nsUnicharUtils.h"
|
|
|
|
#include "mozilla/FunctionTimer.h"
|
2010-11-05 12:57:58 -07:00
|
|
|
#include "prenv.h"
|
|
|
|
#include "prprf.h"
|
2011-01-19 17:21:07 -08:00
|
|
|
#include "GfxDriverInfo.h"
|
2011-06-24 10:41:18 -07:00
|
|
|
#include "mozilla/Preferences.h"
|
2011-12-14 21:03:01 -08:00
|
|
|
#include "nsPrintfCString.h"
|
2010-08-13 08:43:35 -07:00
|
|
|
|
2011-05-23 09:54:47 -07:00
|
|
|
#if defined(MOZ_CRASHREPORTER)
|
2010-08-27 08:49:04 -07:00
|
|
|
#include "nsExceptionHandler.h"
|
|
|
|
#include "nsICrashReporter.h"
|
|
|
|
#define NS_CRASHREPORTER_CONTRACTID "@mozilla.org/toolkit/crash-reporter;1"
|
|
|
|
#endif
|
|
|
|
|
2011-10-10 22:50:08 -07:00
|
|
|
using namespace mozilla;
|
2010-08-13 08:43:35 -07:00
|
|
|
using namespace mozilla::widget;
|
|
|
|
|
2011-01-19 17:39:33 -08:00
|
|
|
#ifdef DEBUG
|
|
|
|
NS_IMPL_ISUPPORTS_INHERITED1(GfxInfo, GfxInfoBase, nsIGfxInfoDebug)
|
|
|
|
#endif
|
|
|
|
|
2011-01-20 20:31:03 -08:00
|
|
|
static const PRUint32 allWindowsVersions = 0xffffffff;
|
2011-01-28 18:12:25 -08:00
|
|
|
|
2011-01-20 20:31:03 -08:00
|
|
|
#define V(a,b,c,d) GFX_DRIVER_VERSION(a,b,c,d)
|
|
|
|
|
|
|
|
|
2011-01-19 17:37:24 -08:00
|
|
|
GfxInfo::GfxInfo()
|
2011-12-14 21:03:01 -08:00
|
|
|
: mWindowsVersion(0),
|
2011-10-01 19:16:19 -07:00
|
|
|
mHasDualGPU(false),
|
|
|
|
mIsGPU2Active(false)
|
2011-01-19 17:37:24 -08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-08-28 07:18:41 -07:00
|
|
|
/* GetD2DEnabled and GetDwriteEnabled shouldn't be called until after gfxPlatform initialization
|
|
|
|
* has occurred because they depend on it for information. (See bug 591561) */
|
2010-08-30 14:45:29 -07:00
|
|
|
nsresult
|
2011-09-28 23:19:26 -07:00
|
|
|
GfxInfo::GetD2DEnabled(bool *aEnabled)
|
2010-08-13 08:43:35 -07:00
|
|
|
{
|
|
|
|
*aEnabled = gfxWindowsPlatform::GetPlatform()->GetRenderMode() == gfxWindowsPlatform::RENDER_DIRECT2D;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2010-08-16 00:15:03 -07:00
|
|
|
|
2010-08-30 14:45:29 -07:00
|
|
|
nsresult
|
2011-09-28 23:19:26 -07:00
|
|
|
GfxInfo::GetDWriteEnabled(bool *aEnabled)
|
2010-08-16 00:15:03 -07:00
|
|
|
{
|
|
|
|
*aEnabled = gfxWindowsPlatform::GetPlatform()->DWriteEnabled();
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2010-08-27 08:49:02 -07:00
|
|
|
|
2011-06-24 10:41:18 -07:00
|
|
|
nsresult
|
2011-09-28 23:19:26 -07:00
|
|
|
GfxInfo::GetAzureEnabled(bool *aEnabled)
|
2011-06-24 10:41:18 -07:00
|
|
|
{
|
2011-10-01 19:16:19 -07:00
|
|
|
*aEnabled = false;
|
2011-06-24 10:41:18 -07:00
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
bool d2dEnabled =
|
2011-06-24 10:41:18 -07:00
|
|
|
gfxWindowsPlatform::GetPlatform()->GetRenderMode() == gfxWindowsPlatform::RENDER_DIRECT2D;
|
|
|
|
|
|
|
|
if (d2dEnabled) {
|
2011-09-28 23:19:26 -07:00
|
|
|
bool azure = false;
|
2011-06-24 10:41:18 -07:00
|
|
|
nsresult rv = mozilla::Preferences::GetBool("gfx.canvas.azure.enabled", &azure);
|
|
|
|
|
|
|
|
if (NS_SUCCEEDED(rv) && azure) {
|
2011-10-01 19:16:19 -07:00
|
|
|
*aEnabled = true;
|
2011-06-24 10:41:18 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-01-14 04:57:17 -08:00
|
|
|
/* readonly attribute DOMString DWriteVersion; */
|
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetDWriteVersion(nsAString & aDwriteVersion)
|
|
|
|
{
|
2011-04-21 12:36:49 -07:00
|
|
|
gfxWindowsPlatform::GetDLLVersion(L"dwrite.dll", aDwriteVersion);
|
2011-01-14 04:57:17 -08:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-05-10 17:30:20 -07:00
|
|
|
#define PIXEL_STRUCT_RGB 1
|
|
|
|
#define PIXEL_STRUCT_BGR 2
|
|
|
|
|
|
|
|
/* readonly attribute DOMString cleartypeParameters; */
|
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetCleartypeParameters(nsAString & aCleartypeParams)
|
|
|
|
{
|
|
|
|
nsTArray<ClearTypeParameterInfo> clearTypeParams;
|
|
|
|
|
|
|
|
gfxWindowsPlatform::GetPlatform()->GetCleartypeParams(clearTypeParams);
|
|
|
|
PRUint32 d, numDisplays = clearTypeParams.Length();
|
|
|
|
bool displayNames = (numDisplays > 1);
|
|
|
|
bool foundData = false;
|
|
|
|
nsString outStr;
|
|
|
|
WCHAR valStr[256];
|
|
|
|
|
|
|
|
for (d = 0; d < numDisplays; d++) {
|
|
|
|
ClearTypeParameterInfo& params = clearTypeParams[d];
|
|
|
|
|
|
|
|
if (displayNames) {
|
2011-10-10 22:50:08 -07:00
|
|
|
swprintf_s(valStr, ArrayLength(valStr),
|
2011-05-10 17:30:20 -07:00
|
|
|
L"%s [ ", params.displayName.get());
|
|
|
|
outStr.Append(valStr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.gamma >= 0) {
|
|
|
|
foundData = true;
|
2011-10-10 22:50:08 -07:00
|
|
|
swprintf_s(valStr, ArrayLength(valStr),
|
2011-05-10 17:30:20 -07:00
|
|
|
L"Gamma: %d ", params.gamma);
|
|
|
|
outStr.Append(valStr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.pixelStructure >= 0) {
|
|
|
|
foundData = true;
|
|
|
|
if (params.pixelStructure == PIXEL_STRUCT_RGB ||
|
|
|
|
params.pixelStructure == PIXEL_STRUCT_BGR)
|
|
|
|
{
|
2011-10-10 22:50:08 -07:00
|
|
|
swprintf_s(valStr, ArrayLength(valStr),
|
2011-05-10 17:30:20 -07:00
|
|
|
L"Pixel Structure: %s ",
|
|
|
|
(params.pixelStructure == PIXEL_STRUCT_RGB ?
|
|
|
|
L"RGB" : L"BGR"));
|
|
|
|
} else {
|
2011-10-10 22:50:08 -07:00
|
|
|
swprintf_s(valStr, ArrayLength(valStr),
|
2011-05-10 17:30:20 -07:00
|
|
|
L"Pixel Structure: %d ", params.pixelStructure);
|
|
|
|
}
|
|
|
|
outStr.Append(valStr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.clearTypeLevel >= 0) {
|
|
|
|
foundData = true;
|
2011-10-10 22:50:08 -07:00
|
|
|
swprintf_s(valStr, ArrayLength(valStr),
|
2011-05-10 17:30:20 -07:00
|
|
|
L"ClearType Level: %d ", params.clearTypeLevel);
|
|
|
|
outStr.Append(valStr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.enhancedContrast >= 0) {
|
|
|
|
foundData = true;
|
2011-10-10 22:50:08 -07:00
|
|
|
swprintf_s(valStr, ArrayLength(valStr),
|
2011-05-10 17:30:20 -07:00
|
|
|
L"Enhanced Contrast: %d ", params.enhancedContrast);
|
|
|
|
outStr.Append(valStr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (displayNames) {
|
|
|
|
outStr.Append(L"] ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (foundData) {
|
|
|
|
aCleartypeParams.Assign(outStr);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
2010-09-27 13:18:56 -07:00
|
|
|
static nsresult GetKeyValue(const WCHAR* keyLocation, const WCHAR* keyName, nsAString& destString, int type)
|
2010-08-27 08:49:02 -07:00
|
|
|
{
|
|
|
|
HKEY key;
|
|
|
|
DWORD dwcbData;
|
|
|
|
DWORD dValue;
|
2010-08-28 07:16:00 -07:00
|
|
|
DWORD resultType;
|
2010-08-27 08:49:02 -07:00
|
|
|
LONG result;
|
|
|
|
nsresult retval = NS_OK;
|
|
|
|
|
|
|
|
result = RegOpenKeyExW(HKEY_LOCAL_MACHINE, keyLocation, 0, KEY_QUERY_VALUE, &key);
|
|
|
|
if (result != ERROR_SUCCESS) {
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case REG_DWORD: {
|
|
|
|
// We only use this for vram size
|
|
|
|
dwcbData = sizeof(dValue);
|
2010-08-28 07:16:00 -07:00
|
|
|
result = RegQueryValueExW(key, keyName, NULL, &resultType, (LPBYTE)&dValue, &dwcbData);
|
|
|
|
if (result == ERROR_SUCCESS && resultType == REG_DWORD) {
|
|
|
|
dValue = dValue / 1024 / 1024;
|
2010-09-15 17:17:49 -07:00
|
|
|
destString.AppendInt(PRInt32(dValue));
|
2010-08-28 07:16:00 -07:00
|
|
|
} else {
|
2010-08-27 08:49:02 -07:00
|
|
|
retval = NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case REG_MULTI_SZ: {
|
|
|
|
// A chain of null-separated strings; we convert the nulls to spaces
|
2010-08-28 07:16:00 -07:00
|
|
|
WCHAR wCharValue[1024];
|
|
|
|
dwcbData = sizeof(wCharValue);
|
|
|
|
|
|
|
|
result = RegQueryValueExW(key, keyName, NULL, &resultType, (LPBYTE)wCharValue, &dwcbData);
|
|
|
|
if (result == ERROR_SUCCESS && resultType == REG_MULTI_SZ) {
|
|
|
|
// This bit here could probably be cleaner.
|
|
|
|
bool isValid = false;
|
|
|
|
|
|
|
|
DWORD strLen = dwcbData/sizeof(wCharValue[0]);
|
|
|
|
for (DWORD i = 0; i < strLen; i++) {
|
|
|
|
if (wCharValue[i] == '\0') {
|
|
|
|
if (i < strLen - 1 && wCharValue[i + 1] == '\0') {
|
|
|
|
isValid = true;
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
wCharValue[i] = ' ';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ensure wCharValue is null terminated
|
|
|
|
wCharValue[strLen-1] = '\0';
|
|
|
|
|
|
|
|
if (isValid)
|
|
|
|
destString = wCharValue;
|
|
|
|
|
|
|
|
} else {
|
2010-08-27 08:49:02 -07:00
|
|
|
retval = NS_ERROR_FAILURE;
|
|
|
|
}
|
2010-08-28 07:16:00 -07:00
|
|
|
|
2010-08-27 08:49:02 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
RegCloseKey(key);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The driver ID is a string like PCI\VEN_15AD&DEV_0405&SUBSYS_040515AD, possibly
|
|
|
|
// followed by &REV_XXXX. We uppercase the string, and strip the &REV_ part
|
|
|
|
// from it, if found.
|
2010-09-27 13:18:56 -07:00
|
|
|
static void normalizeDriverId(nsString& driverid) {
|
2010-08-27 08:49:02 -07:00
|
|
|
ToUpperCase(driverid);
|
|
|
|
PRInt32 rev = driverid.Find(NS_LITERAL_CSTRING("&REV_"));
|
|
|
|
if (rev != -1) {
|
|
|
|
driverid.Cut(rev, driverid.Length());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-20 16:01:11 -07:00
|
|
|
// The device ID is a string like PCI\VEN_15AD&DEV_0405&SUBSYS_040515AD
|
|
|
|
// this function is used to extract the id's out of it
|
|
|
|
PRUint32
|
|
|
|
ParseIDFromDeviceID(const nsAString &key, const char *prefix, int length)
|
|
|
|
{
|
|
|
|
nsAutoString id(key);
|
|
|
|
ToUpperCase(id);
|
|
|
|
PRInt32 start = id.Find(prefix);
|
|
|
|
if (start != -1) {
|
|
|
|
id.Cut(0, start + strlen(prefix));
|
|
|
|
id.Truncate(length);
|
|
|
|
}
|
|
|
|
nsresult err;
|
|
|
|
return id.ToInteger(&err, 16);
|
|
|
|
}
|
2010-08-27 08:49:02 -07:00
|
|
|
|
|
|
|
/* Other interesting places for info:
|
|
|
|
* IDXGIAdapter::GetDesc()
|
|
|
|
* IDirectDraw7::GetAvailableVidMem()
|
|
|
|
* e->GetAvailableTextureMem()
|
|
|
|
* */
|
|
|
|
|
|
|
|
#define DEVICE_KEY_PREFIX L"\\Registry\\Machine\\"
|
2011-01-19 17:33:51 -08:00
|
|
|
nsresult
|
2010-08-30 14:45:29 -07:00
|
|
|
GfxInfo::Init()
|
2010-08-27 08:49:02 -07:00
|
|
|
{
|
|
|
|
NS_TIME_FUNCTION;
|
|
|
|
|
2011-01-19 17:33:51 -08:00
|
|
|
nsresult rv = GfxInfoBase::Init();
|
|
|
|
|
2010-08-28 07:16:00 -07:00
|
|
|
DISPLAY_DEVICEW displayDevice;
|
|
|
|
displayDevice.cb = sizeof(displayDevice);
|
2010-08-27 08:49:02 -07:00
|
|
|
int deviceIndex = 0;
|
|
|
|
|
2010-11-25 12:44:37 -08:00
|
|
|
mDeviceKeyDebug = NS_LITERAL_STRING("PrimarySearch");
|
|
|
|
|
2010-08-28 07:16:00 -07:00
|
|
|
while (EnumDisplayDevicesW(NULL, deviceIndex, &displayDevice, 0)) {
|
2010-11-25 12:44:37 -08:00
|
|
|
if (displayDevice.StateFlags & DISPLAY_DEVICE_PRIMARY_DEVICE) {
|
|
|
|
mDeviceKeyDebug = NS_LITERAL_STRING("NullSearch");
|
2010-08-27 08:49:02 -07:00
|
|
|
break;
|
2010-11-25 12:44:37 -08:00
|
|
|
}
|
2010-08-27 08:49:02 -07:00
|
|
|
deviceIndex++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// make sure the string is NULL terminated
|
2011-10-10 22:50:08 -07:00
|
|
|
if (wcsnlen(displayDevice.DeviceKey, ArrayLength(displayDevice.DeviceKey))
|
|
|
|
== ArrayLength(displayDevice.DeviceKey)) {
|
2010-08-28 07:16:00 -07:00
|
|
|
// we did not find a NULL
|
2011-01-19 17:33:51 -08:00
|
|
|
return rv;
|
2010-08-27 08:49:02 -07:00
|
|
|
}
|
|
|
|
|
2010-11-25 12:44:37 -08:00
|
|
|
mDeviceKeyDebug = displayDevice.DeviceKey;
|
|
|
|
|
|
|
|
/* DeviceKey is "reserved" according to MSDN so we'll be careful with it */
|
|
|
|
/* check that DeviceKey begins with DEVICE_KEY_PREFIX */
|
|
|
|
/* some systems have a DeviceKey starting with \REGISTRY\Machine\ so we need to compare case insenstively */
|
2011-10-10 22:50:08 -07:00
|
|
|
if (_wcsnicmp(displayDevice.DeviceKey, DEVICE_KEY_PREFIX, ArrayLength(DEVICE_KEY_PREFIX)-1) != 0)
|
2011-01-19 17:33:51 -08:00
|
|
|
return rv;
|
2010-11-25 12:44:37 -08:00
|
|
|
|
2010-08-27 08:49:02 -07:00
|
|
|
// chop off DEVICE_KEY_PREFIX
|
2011-10-10 22:50:08 -07:00
|
|
|
mDeviceKey = displayDevice.DeviceKey + ArrayLength(DEVICE_KEY_PREFIX)-1;
|
2010-08-27 08:49:02 -07:00
|
|
|
|
2010-08-28 07:16:00 -07:00
|
|
|
mDeviceID = displayDevice.DeviceID;
|
|
|
|
mDeviceString = displayDevice.DeviceString;
|
2010-08-27 08:49:02 -07:00
|
|
|
|
2012-02-23 06:53:55 -08:00
|
|
|
/* create a device information set composed of the current display device */
|
|
|
|
HDEVINFO devinfo = SetupDiGetClassDevsW(NULL, mDeviceID.get(), NULL,
|
|
|
|
DIGCF_PRESENT | DIGCF_PROFILE | DIGCF_ALLCLASSES);
|
|
|
|
|
|
|
|
if (devinfo != INVALID_HANDLE_VALUE) {
|
|
|
|
HKEY key;
|
|
|
|
LONG result;
|
|
|
|
WCHAR value[255];
|
|
|
|
DWORD dwcbData;
|
|
|
|
SP_DEVINFO_DATA devinfoData;
|
|
|
|
DWORD memberIndex = 0;
|
|
|
|
|
|
|
|
devinfoData.cbSize = sizeof(devinfoData);
|
|
|
|
NS_NAMED_LITERAL_STRING(driverKeyPre, "System\\CurrentControlSet\\Control\\Class\\");
|
|
|
|
/* enumerate device information elements in the device information set */
|
|
|
|
while (SetupDiEnumDeviceInfo(devinfo, memberIndex++, &devinfoData)) {
|
|
|
|
/* get a string that identifies the device's driver key */
|
|
|
|
if (SetupDiGetDeviceRegistryPropertyW(devinfo,
|
|
|
|
&devinfoData,
|
|
|
|
SPDRP_DRIVER,
|
|
|
|
NULL,
|
|
|
|
(PBYTE)value,
|
|
|
|
sizeof(value),
|
|
|
|
NULL)) {
|
|
|
|
nsAutoString driverKey(driverKeyPre);
|
|
|
|
driverKey += value;
|
|
|
|
result = RegOpenKeyExW(HKEY_LOCAL_MACHINE, driverKey.BeginReading(), 0, KEY_QUERY_VALUE, &key);
|
|
|
|
if (result == ERROR_SUCCESS) {
|
|
|
|
/* we've found the driver we're looking for */
|
|
|
|
dwcbData = sizeof(value);
|
|
|
|
result = RegQueryValueExW(key, L"DriverVersion", NULL, NULL, (LPBYTE)value, &dwcbData);
|
|
|
|
if (result == ERROR_SUCCESS)
|
|
|
|
mDriverVersion = value;
|
|
|
|
dwcbData = sizeof(value);
|
|
|
|
result = RegQueryValueExW(key, L"DriverDate", NULL, NULL, (LPBYTE)value, &dwcbData);
|
|
|
|
if (result == ERROR_SUCCESS)
|
|
|
|
mDriverDate = value;
|
|
|
|
RegCloseKey(key);
|
|
|
|
break;
|
2011-11-09 07:16:23 -08:00
|
|
|
}
|
2010-08-27 08:49:02 -07:00
|
|
|
}
|
2012-02-23 06:53:55 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
SetupDiDestroyDeviceInfoList(devinfo);
|
|
|
|
}
|
2011-11-09 07:16:23 -08:00
|
|
|
|
2012-02-23 06:53:55 -08:00
|
|
|
mAdapterVendorID.AppendPrintf("0x%04x", ParseIDFromDeviceID(mDeviceID, "VEN_", 4));
|
|
|
|
mAdapterDeviceID.AppendPrintf("0x%04x", ParseIDFromDeviceID(mDeviceID, "&DEV_", 4));
|
|
|
|
mAdapterSubsysID = ParseIDFromDeviceID(mDeviceID, "&SUBSYS_", 8);
|
|
|
|
|
|
|
|
// We now check for second display adapter.
|
|
|
|
|
|
|
|
// Device interface class for display adapters.
|
|
|
|
CLSID GUID_DISPLAY_DEVICE_ARRIVAL;
|
|
|
|
HRESULT hresult = CLSIDFromString(L"{1CA05180-A699-450A-9A0C-DE4FBE3DDD89}",
|
|
|
|
&GUID_DISPLAY_DEVICE_ARRIVAL);
|
|
|
|
if (hresult == NOERROR) {
|
|
|
|
devinfo = SetupDiGetClassDevsW(&GUID_DISPLAY_DEVICE_ARRIVAL, NULL, NULL,
|
|
|
|
DIGCF_PRESENT | DIGCF_INTERFACEDEVICE);
|
|
|
|
|
|
|
|
if (devinfo != INVALID_HANDLE_VALUE) {
|
|
|
|
HKEY key;
|
|
|
|
LONG result;
|
|
|
|
WCHAR value[255];
|
|
|
|
DWORD dwcbData;
|
|
|
|
SP_DEVINFO_DATA devinfoData;
|
|
|
|
DWORD memberIndex = 0;
|
|
|
|
devinfoData.cbSize = sizeof(devinfoData);
|
|
|
|
|
|
|
|
nsAutoString adapterDriver2;
|
|
|
|
nsAutoString deviceID2;
|
|
|
|
nsAutoString driverVersion2;
|
|
|
|
nsAutoString driverDate2;
|
|
|
|
PRUint32 adapterVendorID2;
|
|
|
|
PRUint32 adapterDeviceID2;
|
|
|
|
|
|
|
|
NS_NAMED_LITERAL_STRING(driverKeyPre, "System\\CurrentControlSet\\Control\\Class\\");
|
|
|
|
/* enumerate device information elements in the device information set */
|
|
|
|
while (SetupDiEnumDeviceInfo(devinfo, memberIndex++, &devinfoData)) {
|
|
|
|
/* get a string that identifies the device's driver key */
|
|
|
|
if (SetupDiGetDeviceRegistryPropertyW(devinfo,
|
|
|
|
&devinfoData,
|
|
|
|
SPDRP_DRIVER,
|
|
|
|
NULL,
|
|
|
|
(PBYTE)value,
|
|
|
|
sizeof(value),
|
|
|
|
NULL)) {
|
|
|
|
nsAutoString driverKey2(driverKeyPre);
|
|
|
|
driverKey2 += value;
|
|
|
|
result = RegOpenKeyExW(HKEY_LOCAL_MACHINE, driverKey2.BeginReading(), 0, KEY_QUERY_VALUE, &key);
|
|
|
|
if (result == ERROR_SUCCESS) {
|
|
|
|
dwcbData = sizeof(value);
|
|
|
|
result = RegQueryValueExW(key, L"MatchingDeviceId", NULL, NULL, (LPBYTE)value, &dwcbData);
|
|
|
|
if (result != ERROR_SUCCESS) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
deviceID2 = value;
|
|
|
|
nsAutoString adapterVendorID2String;
|
|
|
|
nsAutoString adapterDeviceID2String;
|
|
|
|
adapterVendorID2 = ParseIDFromDeviceID(deviceID2, "VEN_", 4);
|
|
|
|
adapterVendorID2String.AppendPrintf("0x%04x", adapterVendorID2);
|
|
|
|
adapterDeviceID2 = ParseIDFromDeviceID(deviceID2, "&DEV_", 4);
|
|
|
|
adapterDeviceID2String.AppendPrintf("0x%04x", adapterDeviceID2);
|
|
|
|
if (mAdapterVendorID == adapterVendorID2String &&
|
|
|
|
mAdapterDeviceID == adapterDeviceID2String) {
|
|
|
|
RegCloseKey(key);
|
|
|
|
continue;
|
2012-02-08 13:04:21 -08:00
|
|
|
}
|
|
|
|
|
2012-02-23 06:53:55 -08:00
|
|
|
// If this device is missing driver information, it is unlikely to
|
|
|
|
// be a real display adapter.
|
|
|
|
if (NS_FAILED(GetKeyValue(driverKey2.BeginReading(), L"InstalledDisplayDrivers",
|
|
|
|
adapterDriver2, REG_MULTI_SZ))) {
|
|
|
|
RegCloseKey(key);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
dwcbData = sizeof(value);
|
|
|
|
result = RegQueryValueExW(key, L"DriverVersion", NULL, NULL, (LPBYTE)value, &dwcbData);
|
|
|
|
if (result != ERROR_SUCCESS) {
|
|
|
|
RegCloseKey(key);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
driverVersion2 = value;
|
|
|
|
dwcbData = sizeof(value);
|
|
|
|
result = RegQueryValueExW(key, L"DriverDate", NULL, NULL, (LPBYTE)value, &dwcbData);
|
|
|
|
if (result != ERROR_SUCCESS) {
|
|
|
|
RegCloseKey(key);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
driverDate2 = value;
|
|
|
|
dwcbData = sizeof(value);
|
|
|
|
result = RegQueryValueExW(key, L"Device Description", NULL, NULL, (LPBYTE)value, &dwcbData);
|
|
|
|
if (result != ERROR_SUCCESS) {
|
|
|
|
dwcbData = sizeof(value);
|
|
|
|
result = RegQueryValueExW(key, L"DriverDesc", NULL, NULL, (LPBYTE)value, &dwcbData);
|
|
|
|
}
|
|
|
|
RegCloseKey(key);
|
|
|
|
if (result == ERROR_SUCCESS) {
|
|
|
|
mHasDualGPU = true;
|
|
|
|
mDeviceString2 = value;
|
|
|
|
mDeviceID2 = deviceID2;
|
|
|
|
mDeviceKey2 = driverKey2;
|
|
|
|
mDriverVersion2 = driverVersion2;
|
|
|
|
mDriverDate2 = driverDate2;
|
|
|
|
mAdapterVendorID2.AppendPrintf("0x%04x", adapterVendorID2);
|
|
|
|
mAdapterDeviceID2.AppendPrintf("0x%04x", adapterDeviceID2);
|
|
|
|
mAdapterSubsysID2 = ParseIDFromDeviceID(mDeviceID2, "&SUBSYS_", 8);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2012-02-08 13:04:21 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-23 06:53:55 -08:00
|
|
|
SetupDiDestroyDeviceInfoList(devinfo);
|
2010-08-27 08:49:02 -07:00
|
|
|
}
|
2011-01-24 07:05:59 -08:00
|
|
|
}
|
2010-08-27 08:49:04 -07:00
|
|
|
|
2011-01-19 17:37:24 -08:00
|
|
|
const char *spoofedDriverVersionString = PR_GetEnv("MOZ_GFX_SPOOF_DRIVER_VERSION");
|
|
|
|
if (spoofedDriverVersionString) {
|
|
|
|
mDriverVersion.AssignASCII(spoofedDriverVersionString);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *spoofedVendor = PR_GetEnv("MOZ_GFX_SPOOF_VENDOR_ID");
|
|
|
|
if (spoofedVendor) {
|
2011-12-14 21:03:01 -08:00
|
|
|
mAdapterVendorID.AssignASCII(spoofedVendor);
|
2011-01-19 17:37:24 -08:00
|
|
|
}
|
|
|
|
|
2011-10-01 19:16:19 -07:00
|
|
|
mHasDriverVersionMismatch = false;
|
2011-12-14 21:03:01 -08:00
|
|
|
if (mAdapterVendorID == GfxDriverInfo::GetDeviceVendor(VendorIntel)) {
|
2011-01-20 20:31:03 -08:00
|
|
|
// we've had big crashers (bugs 590373 and 595364) apparently correlated
|
2011-04-21 12:36:49 -07:00
|
|
|
// with bad Intel driver installations where the DriverVersion reported
|
|
|
|
// by the registry was not the version of the DLL.
|
2011-09-28 23:19:26 -07:00
|
|
|
bool is64bitApp = sizeof(void*) == 8;
|
2011-01-28 02:35:40 -08:00
|
|
|
const PRUnichar *dllFileName = is64bitApp
|
|
|
|
? L"igd10umd64.dll"
|
|
|
|
: L"igd10umd32.dll";
|
2011-01-20 20:31:03 -08:00
|
|
|
nsString dllVersion;
|
2011-04-21 12:36:49 -07:00
|
|
|
gfxWindowsPlatform::GetDLLVersion((PRUnichar*)dllFileName, dllVersion);
|
2011-01-20 20:31:03 -08:00
|
|
|
|
|
|
|
PRUint64 dllNumericVersion = 0, driverNumericVersion = 0;
|
|
|
|
ParseDriverVersion(dllVersion, &dllNumericVersion);
|
|
|
|
ParseDriverVersion(mDriverVersion, &driverNumericVersion);
|
|
|
|
|
2011-04-21 12:36:49 -07:00
|
|
|
// if GetDLLVersion fails, it gives "0.0.0.0"
|
|
|
|
// so if GetDLLVersion failed, we get dllNumericVersion = 0
|
2011-01-20 20:31:03 -08:00
|
|
|
// so this test implicitly handles the case where GetDLLVersion failed
|
|
|
|
if (dllNumericVersion != driverNumericVersion)
|
2011-10-01 19:16:19 -07:00
|
|
|
mHasDriverVersionMismatch = true;
|
2011-01-20 20:31:03 -08:00
|
|
|
}
|
|
|
|
|
2011-01-19 17:37:24 -08:00
|
|
|
const char *spoofedDevice = PR_GetEnv("MOZ_GFX_SPOOF_DEVICE_ID");
|
|
|
|
if (spoofedDevice) {
|
2011-12-14 21:03:01 -08:00
|
|
|
mAdapterDeviceID.AssignASCII(spoofedDevice);
|
2011-01-19 17:37:24 -08:00
|
|
|
}
|
|
|
|
|
2011-01-19 17:39:33 -08:00
|
|
|
const char *spoofedWindowsVersion = PR_GetEnv("MOZ_GFX_SPOOF_WINDOWS_VERSION");
|
|
|
|
if (spoofedWindowsVersion) {
|
|
|
|
PR_sscanf(spoofedWindowsVersion, "%x", &mWindowsVersion);
|
|
|
|
} else {
|
|
|
|
mWindowsVersion = gfxWindowsPlatform::WindowsOSVersion();
|
|
|
|
}
|
|
|
|
|
2010-08-27 08:49:04 -07:00
|
|
|
AddCrashReportAnnotations();
|
2011-01-19 17:33:51 -08:00
|
|
|
|
|
|
|
return rv;
|
2010-08-27 08:49:02 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* readonly attribute DOMString adapterDescription; */
|
2010-08-30 14:45:29 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetAdapterDescription(nsAString & aAdapterDescription)
|
2010-08-27 08:49:02 -07:00
|
|
|
{
|
|
|
|
aAdapterDescription = mDeviceString;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-08-12 06:46:41 -07:00
|
|
|
/* readonly attribute DOMString adapterDescription2; */
|
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetAdapterDescription2(nsAString & aAdapterDescription)
|
|
|
|
{
|
|
|
|
aAdapterDescription = mDeviceString2;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-08-27 08:49:02 -07:00
|
|
|
/* readonly attribute DOMString adapterRAM; */
|
2010-08-30 14:45:29 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetAdapterRAM(nsAString & aAdapterRAM)
|
2010-08-27 08:49:02 -07:00
|
|
|
{
|
|
|
|
if (NS_FAILED(GetKeyValue(mDeviceKey.BeginReading(), L"HardwareInformation.MemorySize", aAdapterRAM, REG_DWORD)))
|
|
|
|
aAdapterRAM = L"Unknown";
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-08-12 06:46:41 -07:00
|
|
|
/* readonly attribute DOMString adapterRAM2; */
|
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetAdapterRAM2(nsAString & aAdapterRAM)
|
|
|
|
{
|
2011-11-09 07:16:23 -08:00
|
|
|
if (!mHasDualGPU) {
|
|
|
|
aAdapterRAM.AssignLiteral("");
|
|
|
|
} else if (NS_FAILED(GetKeyValue(mDeviceKey2.BeginReading(), L"HardwareInformation.MemorySize", aAdapterRAM, REG_DWORD))) {
|
2011-08-12 06:46:41 -07:00
|
|
|
aAdapterRAM = L"Unknown";
|
2011-11-09 07:16:23 -08:00
|
|
|
}
|
2011-08-12 06:46:41 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-08-27 08:49:02 -07:00
|
|
|
/* readonly attribute DOMString adapterDriver; */
|
2010-08-30 14:45:29 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetAdapterDriver(nsAString & aAdapterDriver)
|
2010-08-27 08:49:02 -07:00
|
|
|
{
|
|
|
|
if (NS_FAILED(GetKeyValue(mDeviceKey.BeginReading(), L"InstalledDisplayDrivers", aAdapterDriver, REG_MULTI_SZ)))
|
|
|
|
aAdapterDriver = L"Unknown";
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-08-12 06:46:41 -07:00
|
|
|
/* readonly attribute DOMString adapterDriver2; */
|
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetAdapterDriver2(nsAString & aAdapterDriver)
|
|
|
|
{
|
2011-11-09 07:16:23 -08:00
|
|
|
if (!mHasDualGPU) {
|
|
|
|
aAdapterDriver.AssignLiteral("");
|
|
|
|
} else if (NS_FAILED(GetKeyValue(mDeviceKey2.BeginReading(), L"InstalledDisplayDrivers", aAdapterDriver, REG_MULTI_SZ))) {
|
2011-08-12 06:46:41 -07:00
|
|
|
aAdapterDriver = L"Unknown";
|
2011-11-09 07:16:23 -08:00
|
|
|
}
|
2011-08-12 06:46:41 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-08-27 08:49:02 -07:00
|
|
|
/* readonly attribute DOMString adapterDriverVersion; */
|
2010-08-30 14:45:29 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetAdapterDriverVersion(nsAString & aAdapterDriverVersion)
|
2010-08-27 08:49:02 -07:00
|
|
|
{
|
|
|
|
aAdapterDriverVersion = mDriverVersion;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* readonly attribute DOMString adapterDriverDate; */
|
2010-08-30 14:45:29 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetAdapterDriverDate(nsAString & aAdapterDriverDate)
|
2010-08-27 08:49:02 -07:00
|
|
|
{
|
|
|
|
aAdapterDriverDate = mDriverDate;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-08-12 06:46:41 -07:00
|
|
|
/* readonly attribute DOMString adapterDriverVersion2; */
|
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetAdapterDriverVersion2(nsAString & aAdapterDriverVersion)
|
|
|
|
{
|
|
|
|
aAdapterDriverVersion = mDriverVersion2;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* readonly attribute DOMString adapterDriverDate2; */
|
|
|
|
NS_IMETHODIMP
|
|
|
|
GfxInfo::GetAdapterDriverDate2(nsAString & aAdapterDriverDate)
|
|
|
|
{
|
|
|
|
aAdapterDriverDate = mDriverDate2;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-12-14 21:03:01 -08:00
|
|
|
/* readonly attribute DOMString adapterVendorID; */
|
2010-08-30 14:45:29 -07:00
|
|
|
NS_IMETHODIMP
|
2011-12-14 21:03:01 -08:00
|
|
|
GfxInfo::GetAdapterVendorID(nsAString & aAdapterVendorID)
|
2010-08-27 08:49:02 -07:00
|
|
|
{
|
2011-12-14 21:03:01 -08:00
|
|
|
aAdapterVendorID = mAdapterVendorID;
|
2010-08-27 08:49:02 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-12-14 21:03:01 -08:00
|
|
|
/* readonly attribute DOMString adapterVendorID2; */
|
2011-08-12 06:46:41 -07:00
|
|
|
NS_IMETHODIMP
|
2011-12-14 21:03:01 -08:00
|
|
|
GfxInfo::GetAdapterVendorID2(nsAString & aAdapterVendorID)
|
2011-08-12 06:46:41 -07:00
|
|
|
{
|
2011-12-14 21:03:01 -08:00
|
|
|
aAdapterVendorID = mAdapterVendorID2;
|
2011-08-12 06:46:41 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-12-14 21:03:01 -08:00
|
|
|
/* readonly attribute DOMString adapterDeviceID; */
|
2010-08-30 14:45:29 -07:00
|
|
|
NS_IMETHODIMP
|
2011-12-14 21:03:01 -08:00
|
|
|
GfxInfo::GetAdapterDeviceID(nsAString & aAdapterDeviceID)
|
2010-08-27 08:49:02 -07:00
|
|
|
{
|
2011-12-14 21:03:01 -08:00
|
|
|
aAdapterDeviceID = mAdapterDeviceID;
|
2010-08-27 08:49:02 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
2010-08-27 08:49:04 -07:00
|
|
|
|
2011-12-14 21:03:01 -08:00
|
|
|
/* readonly attribute DOMString adapterDeviceID2; */
|
2011-08-12 06:46:41 -07:00
|
|
|
NS_IMETHODIMP
|
2011-12-14 21:03:01 -08:00
|
|
|
GfxInfo::GetAdapterDeviceID2(nsAString & aAdapterDeviceID)
|
2011-08-12 06:46:41 -07:00
|
|
|
{
|
2011-12-14 21:03:01 -08:00
|
|
|
aAdapterDeviceID = mAdapterDeviceID2;
|
2011-08-12 06:46:41 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* readonly attribute boolean isGPU2Active; */
|
|
|
|
NS_IMETHODIMP
|
2011-09-28 23:19:26 -07:00
|
|
|
GfxInfo::GetIsGPU2Active(bool* aIsGPU2Active)
|
2011-08-12 06:46:41 -07:00
|
|
|
{
|
|
|
|
*aIsGPU2Active = mIsGPU2Active;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-05-23 09:54:47 -07:00
|
|
|
#if defined(MOZ_CRASHREPORTER)
|
2011-02-01 08:30:25 -08:00
|
|
|
/* Cisco's VPN software can cause corruption of the floating point state.
|
|
|
|
* Make a note of this in our crash reports so that some weird crashes
|
|
|
|
* make more sense */
|
|
|
|
static void
|
|
|
|
CheckForCiscoVPN() {
|
|
|
|
LONG result;
|
|
|
|
HKEY key;
|
|
|
|
/* This will give false positives, but hopefully no false negatives */
|
|
|
|
result = RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"Software\\Cisco Systems\\VPN Client", 0, KEY_QUERY_VALUE, &key);
|
|
|
|
if (result == ERROR_SUCCESS) {
|
|
|
|
RegCloseKey(key);
|
|
|
|
CrashReporter::AppendAppNotesToCrashReport(NS_LITERAL_CSTRING("Cisco VPN\n"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-08-30 14:45:29 -07:00
|
|
|
void
|
|
|
|
GfxInfo::AddCrashReportAnnotations()
|
2010-08-27 08:49:04 -07:00
|
|
|
{
|
2011-05-23 09:54:47 -07:00
|
|
|
#if defined(MOZ_CRASHREPORTER)
|
2011-02-01 08:30:25 -08:00
|
|
|
CheckForCiscoVPN();
|
|
|
|
|
2011-12-14 21:03:01 -08:00
|
|
|
nsString deviceID, vendorID;
|
|
|
|
nsCString narrowDeviceID, narrowVendorID;
|
2011-01-20 20:31:03 -08:00
|
|
|
nsAutoString adapterDriverVersionString;
|
2010-08-27 08:49:04 -07:00
|
|
|
|
2011-12-14 21:03:01 -08:00
|
|
|
GetAdapterDeviceID(deviceID);
|
|
|
|
CopyUTF16toUTF8(deviceID, narrowDeviceID);
|
|
|
|
GetAdapterVendorID(vendorID);
|
|
|
|
CopyUTF16toUTF8(vendorID, narrowVendorID);
|
2011-01-20 20:31:03 -08:00
|
|
|
GetAdapterDriverVersion(adapterDriverVersionString);
|
2010-08-27 08:49:04 -07:00
|
|
|
|
|
|
|
CrashReporter::AnnotateCrashReport(NS_LITERAL_CSTRING("AdapterVendorID"),
|
2011-12-14 21:03:01 -08:00
|
|
|
narrowVendorID);
|
2010-08-27 08:49:04 -07:00
|
|
|
CrashReporter::AnnotateCrashReport(NS_LITERAL_CSTRING("AdapterDeviceID"),
|
2011-12-14 21:03:01 -08:00
|
|
|
narrowDeviceID);
|
2011-01-20 20:31:03 -08:00
|
|
|
|
2010-08-27 08:49:04 -07:00
|
|
|
/* Add an App Note for now so that we get the data immediately. These
|
|
|
|
* can go away after we store the above in the socorro db */
|
2010-08-30 11:05:30 -07:00
|
|
|
nsCAutoString note;
|
|
|
|
/* AppendPrintf only supports 32 character strings, mrghh. */
|
2011-12-14 21:03:01 -08:00
|
|
|
note.Append("AdapterVendorID: ");
|
|
|
|
note.Append(narrowVendorID);
|
|
|
|
note.Append(", AdapterDeviceID: ");
|
|
|
|
note.Append(narrowDeviceID);
|
|
|
|
note.AppendPrintf(", AdapterSubsysID: %08x, ", mAdapterSubsysID);
|
|
|
|
note.Append("AdapterDriverVersion: ");
|
2011-01-20 20:31:03 -08:00
|
|
|
note.Append(NS_LossyConvertUTF16toASCII(adapterDriverVersionString));
|
2010-11-25 12:44:37 -08:00
|
|
|
|
2011-12-14 21:03:01 -08:00
|
|
|
if (vendorID == GfxDriverInfo::GetDeviceVendor(VendorAll)) {
|
|
|
|
/* if we didn't find a valid vendorID lets append the mDeviceID string to try to find out why */
|
|
|
|
note.Append(", ");
|
|
|
|
note.AppendWithConversion(mDeviceID);
|
|
|
|
note.Append(", ");
|
|
|
|
note.AppendWithConversion(mDeviceKeyDebug);
|
|
|
|
LossyAppendUTF16toASCII(mDeviceKeyDebug, note);
|
2010-11-25 12:44:37 -08:00
|
|
|
}
|
|
|
|
note.Append("\n");
|
2010-08-30 11:05:30 -07:00
|
|
|
|
2011-08-12 06:46:41 -07:00
|
|
|
if (mHasDualGPU) {
|
2011-12-14 21:03:01 -08:00
|
|
|
nsString deviceID2, vendorID2;
|
2011-08-12 06:46:41 -07:00
|
|
|
nsAutoString adapterDriverVersionString2;
|
2011-12-14 21:03:01 -08:00
|
|
|
nsCString narrowDeviceID2, narrowVendorID2;
|
2011-08-12 06:46:41 -07:00
|
|
|
|
2011-12-05 01:10:27 -08:00
|
|
|
note.AppendLiteral("Has dual GPUs. GPU #2: ");
|
2011-12-14 21:03:01 -08:00
|
|
|
GetAdapterDeviceID2(deviceID2);
|
|
|
|
CopyUTF16toUTF8(deviceID2, narrowDeviceID2);
|
|
|
|
GetAdapterVendorID2(vendorID2);
|
|
|
|
CopyUTF16toUTF8(vendorID2, narrowVendorID2);
|
2011-08-12 06:46:41 -07:00
|
|
|
GetAdapterDriverVersion2(adapterDriverVersionString2);
|
2011-12-14 21:03:01 -08:00
|
|
|
note.Append("AdapterVendorID2: ");
|
|
|
|
note.Append(narrowVendorID2);
|
|
|
|
note.Append(", AdapterDeviceID2: ");
|
|
|
|
note.Append(narrowDeviceID2);
|
|
|
|
note.AppendPrintf(", AdapterSubsysID2: %08x, ", mAdapterSubsysID2);
|
2011-08-12 06:46:41 -07:00
|
|
|
note.AppendPrintf("AdapterDriverVersion2: ");
|
|
|
|
note.Append(NS_LossyConvertUTF16toASCII(adapterDriverVersionString2));
|
|
|
|
}
|
2010-08-30 11:05:30 -07:00
|
|
|
CrashReporter::AppendAppNotesToCrashReport(note);
|
2010-08-27 08:49:04 -07:00
|
|
|
|
|
|
|
#endif
|
|
|
|
}
|
2010-08-30 14:45:29 -07:00
|
|
|
|
2011-01-19 17:21:07 -08:00
|
|
|
static OperatingSystem
|
|
|
|
WindowsVersionToOperatingSystem(PRInt32 aWindowsVersion)
|
2010-08-30 14:45:29 -07:00
|
|
|
{
|
2011-01-19 17:21:07 -08:00
|
|
|
switch(aWindowsVersion) {
|
|
|
|
case gfxWindowsPlatform::kWindowsXP:
|
|
|
|
return DRIVER_OS_WINDOWS_XP;
|
|
|
|
case gfxWindowsPlatform::kWindowsServer2003:
|
|
|
|
return DRIVER_OS_WINDOWS_SERVER_2003;
|
|
|
|
case gfxWindowsPlatform::kWindowsVista:
|
|
|
|
return DRIVER_OS_WINDOWS_VISTA;
|
|
|
|
case gfxWindowsPlatform::kWindows7:
|
|
|
|
return DRIVER_OS_WINDOWS_7;
|
|
|
|
case gfxWindowsPlatform::kWindowsUnknown:
|
|
|
|
default:
|
|
|
|
return DRIVER_OS_UNKNOWN;
|
|
|
|
};
|
2010-08-30 14:45:29 -07:00
|
|
|
}
|
|
|
|
|
2011-12-14 21:02:59 -08:00
|
|
|
const nsTArray<GfxDriverInfo>&
|
2011-11-03 07:50:40 -07:00
|
|
|
GfxInfo::GetGfxDriverInfo()
|
|
|
|
{
|
2011-12-14 21:02:59 -08:00
|
|
|
if (!mDriverInfo->Length()) {
|
2011-12-14 21:03:01 -08:00
|
|
|
/*
|
2011-12-14 21:02:59 -08:00
|
|
|
* NVIDIA entries
|
|
|
|
*/
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST( DRIVER_OS_WINDOWS_XP,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorNVIDIA), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
GfxDriverInfo::allFeatures, nsIGfxInfo::FEATURE_BLOCKED_DRIVER_VERSION,
|
|
|
|
DRIVER_LESS_THAN, V(6,14,12,5721), "257.21" );
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST( DRIVER_OS_WINDOWS_VISTA,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorNVIDIA), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
GfxDriverInfo::allFeatures, nsIGfxInfo::FEATURE_BLOCKED_DRIVER_VERSION,
|
|
|
|
DRIVER_LESS_THAN, V(8,17,12,5721), "257.21" );
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST( DRIVER_OS_WINDOWS_7,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorNVIDIA), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
GfxDriverInfo::allFeatures, nsIGfxInfo::FEATURE_BLOCKED_DRIVER_VERSION,
|
|
|
|
DRIVER_LESS_THAN, V(8,17,12,5721), "257.21" );
|
|
|
|
|
|
|
|
/*
|
|
|
|
* AMD/ATI entries
|
|
|
|
*/
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST( DRIVER_OS_ALL,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorATI), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
GfxDriverInfo::allFeatures, nsIGfxInfo::FEATURE_BLOCKED_DRIVER_VERSION,
|
|
|
|
DRIVER_LESS_THAN, V(8,741,0,0), "10.6" );
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST( DRIVER_OS_ALL,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorAMD), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
GfxDriverInfo::allFeatures, nsIGfxInfo::FEATURE_BLOCKED_DRIVER_VERSION,
|
|
|
|
DRIVER_LESS_THAN, V(8,741,0,0), "10.6" );
|
|
|
|
|
|
|
|
/* OpenGL on any ATI/AMD hardware is discouraged
|
|
|
|
* See:
|
|
|
|
* bug 619773 - WebGL: Crash with blue screen : "NMI: Parity Check / Memory Parity Error"
|
|
|
|
* bugs 584403, 584404, 620924 - crashes in atioglxx
|
|
|
|
* + many complaints about incorrect rendering
|
|
|
|
*/
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST2( DRIVER_OS_ALL,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorATI), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
nsIGfxInfo::FEATURE_OPENGL_LAYERS, nsIGfxInfo::FEATURE_DISCOURAGED,
|
|
|
|
DRIVER_LESS_THAN, GfxDriverInfo::allDriverVersions );
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST2( DRIVER_OS_ALL,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorATI), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
nsIGfxInfo::FEATURE_WEBGL_OPENGL, nsIGfxInfo::FEATURE_DISCOURAGED,
|
|
|
|
DRIVER_LESS_THAN, GfxDriverInfo::allDriverVersions );
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST2( DRIVER_OS_ALL,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorAMD), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
nsIGfxInfo::FEATURE_OPENGL_LAYERS, nsIGfxInfo::FEATURE_DISCOURAGED,
|
|
|
|
DRIVER_LESS_THAN, GfxDriverInfo::allDriverVersions );
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST2( DRIVER_OS_ALL,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorAMD), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
nsIGfxInfo::FEATURE_WEBGL_OPENGL, nsIGfxInfo::FEATURE_DISCOURAGED,
|
|
|
|
DRIVER_LESS_THAN, GfxDriverInfo::allDriverVersions );
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Intel entries
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* implement the blocklist from bug 594877
|
|
|
|
* Block all features on any drivers before this, as there's a crash when a MS Hotfix is installed.
|
|
|
|
* The crash itself is Direct2D-related, but for safety we block all features.
|
|
|
|
*/
|
2011-12-14 21:03:01 -08:00
|
|
|
#define IMPLEMENT_INTEL_DRIVER_BLOCKLIST(winVer, devFamily, driverVer) \
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST2( winVer, \
|
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorIntel), (GfxDeviceFamily*) GfxDriverInfo::GetDeviceFamily(devFamily), \
|
|
|
|
GfxDriverInfo::allFeatures, nsIGfxInfo::FEATURE_BLOCKED_DRIVER_VERSION, \
|
2011-12-14 21:02:59 -08:00
|
|
|
DRIVER_LESS_THAN, driverVer )
|
|
|
|
|
2011-12-14 21:03:01 -08:00
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_XP, IntelGMA500, V(6,14,11,1018));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_XP, IntelGMA900, V(6,14,10,4764));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_XP, IntelGMA950, V(6,14,10,4926));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_XP, IntelGMA3150, V(6,14,10,5260));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_XP, IntelGMAX3000, V(6,14,10,5218));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_XP, IntelGMAX4500HD, V(6,14,10,5284));
|
|
|
|
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_VISTA, IntelGMA500, V(7,14,10,1006));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_VISTA, IntelGMA900, GfxDriverInfo::allDriverVersions);
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_VISTA, IntelGMA950, V(7,14,10,1504));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_VISTA, IntelGMA3150, V(7,14,10,2124));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_VISTA, IntelGMAX3000, V(7,15,10,1666));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_VISTA, IntelGMAX4500HD, V(8,15,10,2202));
|
|
|
|
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_7, IntelGMA500, V(5,0,0,2026));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_7, IntelGMA900, GfxDriverInfo::allDriverVersions);
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_7, IntelGMA950, V(8,15,10,1930));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_7, IntelGMA3150, V(8,14,10,2117));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_7, IntelGMAX3000, V(8,15,10,1930));
|
|
|
|
IMPLEMENT_INTEL_DRIVER_BLOCKLIST(DRIVER_OS_WINDOWS_7, IntelGMAX4500HD, V(8,15,10,2202));
|
|
|
|
|
|
|
|
/* OpenGL on any Intel hardware is discouraged */
|
2011-12-14 21:02:59 -08:00
|
|
|
APPEND_TO_DRIVER_BLOCKLIST2( DRIVER_OS_ALL,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorIntel), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
nsIGfxInfo::FEATURE_OPENGL_LAYERS, nsIGfxInfo::FEATURE_DISCOURAGED,
|
|
|
|
DRIVER_LESS_THAN, GfxDriverInfo::allDriverVersions );
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST2( DRIVER_OS_ALL,
|
2011-12-14 21:03:01 -08:00
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorIntel), GfxDriverInfo::allDevices,
|
2011-12-14 21:02:59 -08:00
|
|
|
nsIGfxInfo::FEATURE_WEBGL_OPENGL, nsIGfxInfo::FEATURE_DISCOURAGED,
|
|
|
|
DRIVER_LESS_THAN, GfxDriverInfo::allDriverVersions );
|
2011-12-14 21:03:01 -08:00
|
|
|
|
|
|
|
/* Disable D3D9 layers on NVIDIA 6100/6150/6200 series due to glitches
|
|
|
|
* whilst scrolling. See bugs: 612007, 644787 & 645872.
|
|
|
|
*/
|
|
|
|
APPEND_TO_DRIVER_BLOCKLIST2( DRIVER_OS_ALL,
|
|
|
|
(nsAString&) GfxDriverInfo::GetDeviceVendor(VendorNVIDIA), (GfxDeviceFamily*) GfxDriverInfo::GetDeviceFamily(NvidiaBlockD3D9Layers),
|
|
|
|
nsIGfxInfo::FEATURE_DIRECT3D_9_LAYERS, nsIGfxInfo::FEATURE_BLOCKED_DEVICE,
|
|
|
|
DRIVER_LESS_THAN, GfxDriverInfo::allDriverVersions );
|
2011-12-14 21:02:59 -08:00
|
|
|
}
|
|
|
|
return *mDriverInfo;
|
2011-11-03 07:50:40 -07:00
|
|
|
}
|
|
|
|
|
2010-10-06 21:40:08 -07:00
|
|
|
nsresult
|
2011-12-14 21:03:01 -08:00
|
|
|
GfxInfo::GetFeatureStatusImpl(PRInt32 aFeature,
|
2011-11-03 07:50:40 -07:00
|
|
|
PRInt32 *aStatus,
|
|
|
|
nsAString & aSuggestedDriverVersion,
|
2011-12-14 21:02:59 -08:00
|
|
|
const nsTArray<GfxDriverInfo>& aDriverInfo,
|
2011-11-03 07:50:40 -07:00
|
|
|
OperatingSystem* aOS /* = nsnull */)
|
2010-08-30 14:45:29 -07:00
|
|
|
{
|
2011-12-14 21:03:08 -08:00
|
|
|
NS_ENSURE_ARG_POINTER(aStatus);
|
2011-10-01 19:16:19 -07:00
|
|
|
aSuggestedDriverVersion.SetIsVoid(true);
|
2011-12-14 21:03:08 -08:00
|
|
|
OperatingSystem os = WindowsVersionToOperatingSystem(mWindowsVersion);
|
|
|
|
*aStatus = nsIGfxInfo::FEATURE_STATUS_UNKNOWN;
|
|
|
|
if (aOS)
|
|
|
|
*aOS = os;
|
2010-10-06 21:40:08 -07:00
|
|
|
|
2011-12-14 21:03:08 -08:00
|
|
|
// Don't evaluate special cases if we're checking the downloaded blocklist.
|
|
|
|
if (!aDriverInfo.Length()) {
|
|
|
|
nsAutoString adapterVendorID;
|
|
|
|
nsAutoString adapterDeviceID;
|
|
|
|
nsAutoString adapterDriverVersionString;
|
|
|
|
if (NS_FAILED(GetAdapterVendorID(adapterVendorID)) ||
|
|
|
|
NS_FAILED(GetAdapterDeviceID(adapterDeviceID)) ||
|
|
|
|
NS_FAILED(GetAdapterDriverVersion(adapterDriverVersionString)))
|
|
|
|
{
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
2011-12-14 21:02:59 -08:00
|
|
|
|
2012-02-21 11:49:06 -08:00
|
|
|
if (!adapterVendorID.Equals(GfxDriverInfo::GetDeviceVendor(VendorIntel), nsCaseInsensitiveStringComparator()) &&
|
|
|
|
!adapterVendorID.Equals(GfxDriverInfo::GetDeviceVendor(VendorNVIDIA), nsCaseInsensitiveStringComparator()) &&
|
|
|
|
!adapterVendorID.Equals(GfxDriverInfo::GetDeviceVendor(VendorAMD), nsCaseInsensitiveStringComparator()) &&
|
|
|
|
!adapterVendorID.Equals(GfxDriverInfo::GetDeviceVendor(VendorATI), nsCaseInsensitiveStringComparator()) &&
|
2011-12-14 21:03:08 -08:00
|
|
|
// FIXME - these special hex values are currently used in xpcshell tests introduced by
|
|
|
|
// bug 625160 patch 8/8. Maybe these tests need to be adjusted now that we're only whitelisting
|
|
|
|
// intel/ati/nvidia.
|
|
|
|
!adapterVendorID.LowerCaseEqualsLiteral("0xabcd") &&
|
|
|
|
!adapterVendorID.LowerCaseEqualsLiteral("0xdcba") &&
|
|
|
|
!adapterVendorID.LowerCaseEqualsLiteral("0xabab") &&
|
|
|
|
!adapterVendorID.LowerCaseEqualsLiteral("0xdcdc"))
|
|
|
|
{
|
|
|
|
*aStatus = FEATURE_BLOCKED_DEVICE;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2010-11-05 12:57:58 -07:00
|
|
|
|
2011-12-14 21:03:08 -08:00
|
|
|
PRUint64 driverVersion;
|
|
|
|
if (!ParseDriverVersion(adapterDriverVersionString, &driverVersion)) {
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
2011-03-01 17:01:17 -08:00
|
|
|
|
2011-12-14 21:03:08 -08:00
|
|
|
// special-case the WinXP test slaves: they have out-of-date drivers, but we still want to
|
|
|
|
// whitelist them, actually we do know that this combination of device and driver version
|
|
|
|
// works well.
|
|
|
|
if (mWindowsVersion == gfxWindowsPlatform::kWindowsXP &&
|
2012-02-21 11:49:06 -08:00
|
|
|
adapterVendorID.Equals(GfxDriverInfo::GetDeviceVendor(VendorNVIDIA), nsCaseInsensitiveStringComparator()) &&
|
2011-12-14 21:03:08 -08:00
|
|
|
adapterDeviceID.LowerCaseEqualsLiteral("0x0861") && // GeForce 9400
|
|
|
|
driverVersion == V(6,14,11,7756))
|
|
|
|
{
|
|
|
|
*aStatus = FEATURE_NO_INFO;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2011-01-19 17:21:07 -08:00
|
|
|
|
2011-12-14 21:03:08 -08:00
|
|
|
// ANGLE currently uses D3D10 <-> D3D9 interop, which crashes on Optimus
|
|
|
|
// machines.
|
|
|
|
if (aFeature == FEATURE_WEBGL_ANGLE &&
|
|
|
|
gfxWindowsPlatform::IsOptimus())
|
2011-01-20 20:31:03 -08:00
|
|
|
{
|
2011-12-14 21:03:08 -08:00
|
|
|
*aStatus = FEATURE_BLOCKED_DEVICE;
|
2011-01-20 20:31:03 -08:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-12-14 21:03:08 -08:00
|
|
|
// Windows Server 2003 should be just like Windows XP for present purpose, but still has a different version number.
|
|
|
|
// OTOH Windows Server 2008 R1 and R2 already have the same version numbers as Vista and Seven respectively
|
|
|
|
if (os == DRIVER_OS_WINDOWS_SERVER_2003)
|
|
|
|
os = DRIVER_OS_WINDOWS_XP;
|
|
|
|
|
|
|
|
if (mHasDriverVersionMismatch) {
|
|
|
|
if (aFeature == nsIGfxInfo::FEATURE_DIRECT3D_10_LAYERS ||
|
|
|
|
aFeature == nsIGfxInfo::FEATURE_DIRECT3D_10_1_LAYERS ||
|
|
|
|
aFeature == nsIGfxInfo::FEATURE_DIRECT2D)
|
|
|
|
{
|
|
|
|
*aStatus = nsIGfxInfo::FEATURE_BLOCKED_DRIVER_VERSION;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-10-06 21:40:08 -07:00
|
|
|
|
2011-11-03 07:50:40 -07:00
|
|
|
return GfxInfoBase::GetFeatureStatusImpl(aFeature, aStatus, aSuggestedDriverVersion, aDriverInfo, &os);
|
2010-08-30 14:45:29 -07:00
|
|
|
}
|
2011-01-19 17:39:33 -08:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
|
|
|
|
// Implement nsIGfxInfoDebug
|
|
|
|
|
2011-12-14 21:04:35 -08:00
|
|
|
/* void spoofVendorID (in DOMString aVendorID); */
|
|
|
|
NS_IMETHODIMP GfxInfo::SpoofVendorID(const nsAString & aVendorID)
|
2011-01-19 17:39:33 -08:00
|
|
|
{
|
2011-12-14 21:04:35 -08:00
|
|
|
mAdapterVendorID = aVendorID;
|
2011-01-19 17:39:33 -08:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* void spoofDeviceID (in unsigned long aDeviceID); */
|
2011-12-14 21:04:35 -08:00
|
|
|
NS_IMETHODIMP GfxInfo::SpoofDeviceID(const nsAString & aDeviceID)
|
2011-01-19 17:39:33 -08:00
|
|
|
{
|
2011-12-14 21:04:35 -08:00
|
|
|
mAdapterDeviceID = aDeviceID;
|
2011-01-19 17:39:33 -08:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* void spoofDriverVersion (in DOMString aDriverVersion); */
|
|
|
|
NS_IMETHODIMP GfxInfo::SpoofDriverVersion(const nsAString & aDriverVersion)
|
|
|
|
{
|
|
|
|
mDriverVersion = aDriverVersion;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* void spoofOSVersion (in unsigned long aVersion); */
|
|
|
|
NS_IMETHODIMP GfxInfo::SpoofOSVersion(PRUint32 aVersion)
|
|
|
|
{
|
|
|
|
mWindowsVersion = aVersion;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|