Windows 8.1 DirectDraw и совместимость со старыми играми

4092
Caleb C

Я пытаюсь запустить старую игру (Nascar Heat 2002) на своем ноутбуке с Windows 8.1. Проблема, с которой я сталкиваюсь, заключается в том, что игра вылетает перед запуском, и в журналах сообщается, что видео-диск недоступен. Это файл журнала:

41.37.114: data directory: C:\Program Files (x86)\Hasbro Interactive\NASCAR Heat\Data\ 41.37.115: Config Dir: C:\Program Files (x86)\Hasbro Interactive\NASCAR Heat\ 41.37.274: ddraw: created directdraw with aticfx32.dll (AMD Radeon HD 8650G + HD 8600/8700M Dual Graphics) 41.37.274: ddraw: version 0.0.0.0 41.40.904: vid: 0 meg card (reported:0.523438) 41.40.904: vid: using AGP textures (1397), total: 0 41.40.905: unsupported: 0 megs of vram" 

Насколько я могу судить, версия DirectDraw с Windows 8.1 не совместима со старыми играми, такими как эта. Я пробовал использовать библиотеки WineD3D, среди других оболочек / хаков ddraw, но безрезультатно. Итак, мой вопрос заключается в следующем: есть ли способ заставить эмулируемое количество vram (на моей карте есть видеопамять) либо в Windows, либо в оболочке ddraw, чтобы эта игра обнаруживала его? Я обновил драйверы катализатора до последних версий и имею среду выполнения Microsoft DirectX 9.0c для конечных пользователей

2
Нет вопросов............? Xavierjazz 9 лет назад 0
Windows 8.1 полностью совместима со старыми играми DirectX, если установлена ​​версия DirectX, которую она использует. Ramhound 9 лет назад 0
Видите ли, дело в том, что у меня установлена ​​среда выполнения для конечного пользователя, которая должна исправить совместимость, но это не сработало Caleb C 9 лет назад 0

1 ответ на вопрос

6
beatcracker

I have a sneaking suspicion...

I can smell a video card detection routine gone bad from here. This has nothing to do with Windows and\or DirectDraw (well partially it does, but not the way you think). It's just an old game makes assumptions that are no longer valid. It's not uncommon. For example Oni game crashes on modern videocards:

This problem has been traced to the overflow of a particular text buffer - the one that lists the OpenGL extensions in the startup.txt file. When Oni was written, the OpenGL extension list dump was much shorter, and the developers did not allow for a larger dump. Modern graphics cards almost always cause this overflow.

We need to go deeper

I don't own Nascar Heat 2002, but I've downloaded a NASCAR Heat Demo and it's exhibits exactly the same problem. So I've unholstered my debugger and disassembler and spent an evening trying to figure out what's wrong with the game.

The game actually consists of two executables, communicating with each other via semaphore: the main executable (NASCAR Heat Demo.exe in my case), and the actual game engine (.\run\race.bin). The videocard detection routine is in the race.bin. On game launch the main executable copies race.bin to the Windows TEMP folder as heat.bin and runs it from there. If you try to rename race.bin to race.exe and run it, it searches for semaphore that should be created by the main executable, and if it's not found displays this message:

Sneaky user! Where is my canary!

After a disassembly and quick look at the string references, I've found a function call that prints vid: 0 meg card (reported:0.523438) message. It's actually a part of videocard memory size detection procedure, that in pseudocode looks like this (oversimplified):

RawVidMemSize = GetVidMemSizeFromDirectDraw() // Add 614400 bytes (600Kb - 640x480 mode?) to vidmem size (what for?!) RawVidMemSize = RawVidMemSize + 614400 if (RawVidMemSize < 2000000) { MemSize = 0 } else { if (RawVidMemSize < 4000000) { MemSize = 2 } if (RawVidMemSize < 8000000) { MemSize = 4 } if (RawVidMemSize < 12000000) { MemSize = 8 } if (RawVidMemSize < 16000000) { MemSize = 12 } if (RawVidMemSize < 32000000) { MemSize = 16 } if (RawVidMemSize < 64000000) { MemSize = 32 } if (RawVidMemSize > 64000000) { MemSize = 64 } } 

For ones interested, here is actual control flow of the function from the IDA with my comments. Full-size image onclick.

Videocard memory size detection

Now it's time to actually look at what's happens inside this procedure. I've used a classical break & enter trick (patched the first instruction at race.bin's entry point with int3), launched NASCAR Heat Demo.exe and waited for debugger to pop up. And that's when the things became clear.

Video memory size returned from GetVidMemSizeFromDirectDraw() is 0xFFFF0000 (4294901760 bytes = 4095MB) and it's has nothing to do with the real thing (should be 1Gb on my PC). It turns out that DirectDraw is not well suited for the modern videocard\PC architecture

With the growth of physical memories both RAM and VRAM, this API is also having problems coping since it returns 32-bit DWORD counts of the size in bytes.

and tends to report whatever it feels like:

You have a system with 1GB or greater of Video memory, and 4GB or greater of system memory (RAM).

You run the Direct-X Diagnostics tool, and it reports that you have an unexpectedly low amount of Approximate Total Memory on the display tab.

You may also see issues with some games or applications not allowing you to select the highest detail settings.

The API that DXDiag uses to approximate the system memory was not designed to handle systems in this configuration

On a system with 1GB of video memory, the following values are returned with the associated system memory:

╔═══════════════╦═══════════════════════════════════╗ ║ System Memory ║ Reported Approximate Total Memory ║ ╠═══════════════╬═══════════════════════════════════╣ ║ 4GB ║ 3496MB ║ ║ 6GB ║ 454MB ║ ║ 8GB ║ 1259MB ║ ╚═══════════════╩═══════════════════════════════════╝ 

So in my case it's just reports the value that almost fits into the 32-bit integer. And that's where the things go bad. Remember this line?

RawVidMemSize = RawVidMemSize + 614400 

It becomes this:

RawVidMemSize = 4294901760 + 614400 (= 4295516160) 

And 4295516160 is 548865 more than 32-bit value can handle (0xFFFFFFFF = 4294967295). Hence the integer overflow and the final result is 548864. So now, the game thinks that my vidmem size is whopping 536KB and refuses to run.

You can check this yourself in this online x86 Assembly emulator. Enter the code below, in right left corner click Windows and check Registers checkbox. Click the Step button and watch how 0xFFFF0000 in EAX register becomes 0x00086000 with Carry flag. If you click on register value it will toggle between hex and decimal representation of a number.

mov eax, 0xFFFF0000 add eax, 0x96000 

How do I fix it?

DirectDraw will probably never return a value more than 32-bit integer can handle1 (it's seems to be capped to fit regardless of actual memory size. So the easiest way to fix this problem is to remove RawVidMemSize = RawVidMemSize + 614400 operation from the code, so it wouldn't trigger the overflow. In executable it looks like this:

  • Assembly mnemonic: add eax, 96000h
  • Actual opcodes (hex): 0500600900

To remove it we need to replace it with NOP instructions (hex: 90). I already know the file offset, but it can be diferent in your executable. Fortunately, hex-string 0500600900 is unique in my race.bin and probably in yours. So get hex-editor (I recommend HxD: it's free, portable and easy to use) and open your bin file.

Do a hex-string search:

hex-string search

Once hex-string is found

hex-string found

Replace it with 90

Patch!

Save the file. HxD will automatically create backup of the file, wich you can restore if something goes wrong.

In my case this was enough and I was able to start the game. Here is how heat.log looks after the patch:

21.33.564: ddraw: created directdraw with aticfx32.dll (AMD Radeon HD 5800 Series)
21.33.564: ddraw: version 0.0.0.0
21.34.296: vid: 64 meg card (reported:4095.937500)
21.34.296: vid: using AGP textures (3231), total: 64
21.34.305: vid: triple buffer on

If your file by chance will contain several occurrences of 0500600900, replace the first one, then try to start game and if doesn't work, restore file from backup and try next. Don't replace everything at once, this is not a good idea.

It's also have been confirmed that the same bug exists in Viper Racing. Viper Racing uses slightly different (older?) version of the game engine than Nascar but the bug is the same: it too tries to add 614400 bytes to the video memory size. The values to search are different because in this case compiler decided not to use registers and just accessed variable from stack, i.e.:

  • Assembly mnemonic: add [esp+18h+var_14], 96000h
  • Actual opcodes (hex): 8144240400600900

Happy driving!

  1. This is one of those assumptions I've been talking about.

Похожие вопросы