SaveStates and Key Names
Monday, 31st March 2008
It's a good feeling when issue 1 is finally marked as Fixed - in this case it was another interrupt-related bug. The IFF1 flag was being used to mask non-maskable interrupts; I don't think it should and it hasn't seem to have broken anything just yet by making non-maskable interrupts truly non-maskable.
I have also changed the savestate format to something that will be a little more backwards compatible rather than a plain BinaryFormatter dump of the entire emulator state. The data is now saved in what is basically an INI file with C#-style attributes and CSS-style url() syntax for binary data.
[Type(BeeDevelopment.Sega8Bit.Hardware.VideoDisplayProcessor)] VideoRam=Dump(Video\VideoRam.bin) Address=49165 WaitingForSecond=False AccessMode=ColourRamWrite ReadBuffer=32 Registers=Dump(Video\Registers.bin) System=Ntsc SupportsMode4=True ; ... snip ...
Other changes include a faked fullscreen mode (ie, a maximised borderless window) and an option to retain the aspect ratio of the game, so games no longer appear stretched on widescreen monitors. The sound emulation is a bit better, but still a little noisy in certain games with spurious beeps or buzzes.
One minor problem was on the key configuration control panel. Converting members of the Keys enumeration into displayable strings can be done via .ToString(), but this results in names that do not match the user's keyboard layout (such as OemQuestion for /, OemTilde for # and even Oemcomma for , - note the lowercase c, all on a UK keyboard).
With a prod in the right direction from jpetrie, here's a snippet that can be used to convert Keys into friendly key name strings:
#region Converting Keys into human-readable strings. /// <summary> /// Converts a <see cref="Keys"/> value into a human-readable string describing the key. /// </summary> /// <param name="key">The <see cref="Keys"/> to convert.</param> /// <returns>A human-readable string describing the key.</returns> public static string GetKeyName(Keys key) { // Convert the virtual key code into a scancode (as required by GetKeyNameText). int Scancode = MapVirtualKey((int)key, MapVirtualKeyMode.MAPVK_VK_TO_VSC); // If that returned 0 (failure) just use the value returned by Keys.ToString(). if (Scancode == 0) return key.ToString(); // Certain keys end up being mapped to the number pad by the above function, // as their virtual key can be generated by the number pad too. // If it's one of the known number-pad duplicates, set the extended bit: switch (key) { case Keys.Insert: case Keys.Delete: case Keys.Home: case Keys.End: case Keys.PageUp: case Keys.PageDown: case Keys.Left: case Keys.Right: case Keys.Up: case Keys.Down: case Keys.NumLock: Scancode |= 0x100; break; } // Perform the conversion: StringBuilder KeyName = new StringBuilder("".PadRight(32)); if (GetKeyNameText((Scancode << 16), KeyName, KeyName.Length) != 0) { return KeyName.ToString(); } else { return key.ToString(); } } /// <summary> /// Retrieves a string that represents the name of a key. /// </summary> /// <param name="lParam">Specifies the second parameter of the keyboard message (such as <c>WM_KEYDOWN</c>) to be processed.</param> /// <param name="lpString">Pointer to a buffer that will receive the key name.</param> /// <param name="size">Specifies the maximum length, in TCHAR, of the key name, including the terminating null character. (This parameter should be equal to the size of the buffer pointed to by the lpString parameter).</param> /// <returns>The length of the returned string.</returns> [DllImport("user32.dll")] static extern int GetKeyNameText(int lParam, StringBuilder lpString, int size); /// <summary> /// Translates (maps) a virtual-key code into a scan code or character value, or translates a scan code into a virtual-key code. /// </summary> /// <param name="uCode">Specifies the virtual-key code or scan code for a key. How this value is interpreted depends on the value of the <paramref name="uMapType"/> parameter.</param> /// <param name="uMapType">Specifies the translation to perform. The value of this parameter depends on the value of the <paramref name="uCode"/> parameter.</param> /// <returns>Either a scan code, a virtual-key code, or a character value, depending on the value of <paramref="uCode"/> and <paramref="uMapType"/>. If there is no translation, the return value is zero.</returns> [DllImport("user32.dll")] static extern int MapVirtualKey(int uCode, MapVirtualKeyMode uMapType); enum MapVirtualKeyMode { /// <summary>uCode is a virtual-key code and is translated into a scan code. If it is a virtual-key code that does not distinguish between left- and right-hand keys, the left-hand scan code is returned. If there is no translation, the function returns 0.</summary> MAPVK_VK_TO_VSC = 0, /// <summary>uCode is a scan code and is translated into a virtual-key code that does not distinguish between left- and right-hand keys. If there is no translation, the function returns 0.</summary> MAPVK_VSC_TO_VK = 1, /// <summary>uCode is a virtual-key code and is translated into an unshifted character value in the low-order word of the return value. Dead keys (diacritics) are indicated by setting the top bit of the return value. If there is no translation, the function returns 0.</summary> MAPVK_VK_TO_CHAR = 2, /// <summary>uCode is a scan code and is translated into a virtual-key code that distinguishes between left- and right-hand keys. If there is no translation, the function returns 0.</summary> MAPVK_VSC_TO_VK_EX = 3, MAPVK_VK_TO_VSC_EX = 4, } #endregion
It uses P/Invoke and the Win32 API so is only suitable for use on Windows.