2020-09-30 16:53:55 +02:00

336 lines
15 KiB
Plaintext

This is an overview document describing how to create module patches
in Windows 95.
Legal aspects:
--------------
While we are obviously using this module-patching technology to
make already-shipping apps run or run better, it is possible
for us to make a mistake and wrongly patch a module. For example,
there might be two similar versions of an app which need to be
patched differently, but we only know about one of them. Due to the
obvious liability concerns, we need to obtain authorization from
the app vendor for each module patch we put into the registry.
The letter must specify the name of the module, the number of the
segment taking the patch, the bytes being replaced, and the new
bytes taking their place. The letter can be from any responsible
person at the vendor, eg. an engineer who is familiar with the code.
At the same time, talking to the vendor will help us find out how
many different versions there are of an app, so we can figure out
how to patch each one.
See brianrey if you have a module patch you want to check in.
It would be a good idea to get the ball rolling on getting the
letter as soon as possible after you know what the patch is, since
it can sometimes take a while to get these things taken care of.
We already get similar letters for app-hack bits.
How to check in a patch:
------------------------
Since the patch values go in the registry, and since the initial registry
contents are owned by setup, you must work with Andy Hill to get your
module patches checked in.
A common mistake when adding app-patches to msbase.inx is to leave out
field 4 (field 3 if you are a computer), which are flags. The reason
for the confusion is the flag that means "This is a binary regkey" is 01,
and the opcode for "Change" is also 01, so everything just shifts down
one step and nobody gets hurt. Except that the patch doesn't work.
eg: If you want to add a key whose value is <01,09,70,00,02,ff,76,eb,15>
WRONG
HKLM,"SYSTEM\...",Change,01,09,70,00,02,ff,76,eb,15
BETTER
HKLM,"SYSTEM\...",Change,1,01,09,70,00,02,ff,76,eb,15
How to make a patch:
--------------------
All definitions of the structures used below are in core\kernel\patch.asm.
The loader knows to look in the registry for patch data if the MCF_MODPATCH
bit is set in the [ModuleCompatibility] section in win.ini.
eg.
[ModuleCompatibility]
GENERIC=0x0002
All registry keys and values for module-patching are stored
under HKEY_LOCAL_MACHINE under REGSTR_PATH_APPPATCH:
(from dev\sdk\inc\regstr.h)
#define REGSTR_PATH_APPPATCH "System\\CurrentControlSet\\Control\\SessionManager\\AppPatches"
The actual patches are values stored in the registry under
REGSTR_PATH_APPPATCH\<mod_name>\<signature>\<segment_number>
where
mod_name = the module name
signature = a signature string identifying the module version
segment_number = the hex number of the segment receiving the patches
Signatures:
-----------
Signatures are strings of ANSI characters representing hex numbers
in nibble-swapped byte format (what you see in wdeb386 if you type
db). Blanks and commas are ignored, and may be included for readability.
The easiest way to create a signature is to run dev\tools\binw\gensig.
eg. If you want to patch segment 3 in foo.dll, run "gensig foo.exe 3"
and a tight signature will be written to stdout.
type-1 and -2 signatures:
A type-1 signature specifies a list of byte sequences with byte offsets
(type-2 has word offsets) which must all match for the signature to match.
A 0 byte-count marks the end of the list.
A match-any-file signature is "0100".
eg. signature = "01 02,00,4e,45 02,3e,0a,03 00" is a type-1 signature which
|| || || || || || || || || ||
type --------++ || || || || || || || || ||
|| || || || || || || || ||
byte count------++ || || || || || || || ||
offset-------------++ || || || || || || ||
match bytes-----------++-++ || || || || ||
|| || || || ||
byte count------------------++ || || || ||
offset-------------------------++ || || ||
match bytes-----------------------++-++ ||
||
terminating byte count------------------++
matches if the 2 bytes starting at offset 0 in the exe header match
4e,45 ("NE") and if the 2 bytes starting at offset 3e match 0a,03.
Note- DO NOT include a match on NE in your signature. This is just
for illustrative purposes. The windows version and size(s) of the
segment(s) being patched are good candidates.
type-3, -4 and -5 signatures:
Type-3, -4 and -5 signatures are similar to type-1 and -2 signatures,
except that the offsets are offsets in the module file. Type-3 takes
word offsets, type-4 takes 3-byte offsets, and type-5 takes dword offsets.
eg. signature = "03,03,67,05,c2,0a,00,00"
|| || || || || || || ||
type---------++ || || || || || || ||
|| || || || || || ||
byte count------++ || || || || || ||
file offset--------++-++ || || || ||
match bytes--------------++-++-++ ||
||
terminating byte count------------++
matches if the 3 bytes at offset 567h in the file match c2,0a,00.
type-6, -7 and -8 signatures:
Type-6, -7 and -8 signatures specify the file size of the matching
module. The number of bytes used to specify the matching file size
is 2, 3 or 4 respectively.
eg. signature = "06,d0,0c"
|| || ||
type---------++ || ||
file size-------++-++
matches if the file size is 0cd0h.
type-ff signatures:
Type-ff signatures are meta-signatures and consist of a list of the
other types of signatures. Each list element consists of a byte count
and a sub-signature. A 0 byte-count ends the list.
eg. signature = "ff 06,01,02,3e,0a,03,00 03,06,d0,0c 00"
|| || || || || || || || || || || || ||
type---------++ || || || || || || || || || || || ||
|| || || || || || || || || || || ||
byte count------++ || || || || || || || || || || ||
sub-type-----------++ || || || || || || || || || ||
sub-byte-count--------++ || || || || || || || || ||
hExe offset--------------++ || || || || || || || ||
match bytes-----------------++-++ || || || || || ||
sub-type terminator---------------++ || || || || ||
|| || || || ||
byte count---------------------------++ || || || ||
sub-type--------------------------------++ || || ||
file size----------------------------------++-++ ||
||
terminating byte count---------------------------++
matches if the 2 bytes at offset 3eh in the exe header match 0a,03
and the file size is 0cd0h.
Criteria for selecting a signature:
-----------------------------------
In terms of time required to validate a signature, the hExe types are
the fastest, the file size types are next-fastest, and the file data
types are the slowest. A signature specifying the expected Windows
version (hExe offset 3e) and the file size is probably sufficient
for most cases, since almost any code change changes the file size.
A signature which also requires a file match on the bytes being replaced
(if they are not fixed up and they are in a segment, they must be
somewhere in the file!) is very good, but might be overkill.
Since file match signatures hit the disk, they might noticeably increase
the load time of the module.
DO NOT match on the NE signature in the module header. This is
just a waste of bytes, since it always matches.
DO match on the file size unless you have a very good reason not to.
If going through the ifs, this is a very cheap test.
DO match on something, since we don't want to rely on just the
name of the module. We have signatures. Use them. If you have
no better idea (eg. there is only one version), match on the
windows version, the size(s) of the segment(s) being patched, and
the file size.
Patch data:
-----------
The patch values specify sequences of bytes to add or replace.
Add's must be placed after the end of the unpatched segment.
The segment is GlobalReAlloc'd as necessary to contain the new code.
Replace's must be within the limits of the unpatched segment,
and they must match the sequence of old bytes in the patch value.
Obviously, replaced bytes cannot contain fixups.
The contents of the value string are not used, except to be sent
to the debugger as a debugging aid when the value is loaded from
the registry. The contents of the value data must be in REG_BINARY
format.
Example:
--------
Note: The registry key show below will not work as shown, since it
has been broken into multiple lines for legibility. Join the lines
together and it works.
==================================================================
REGEDIT4
[HKEY_LOCAL_MACHINE\System\CurrentControlSet\control\SessionManager
\AppPatches\GENERIC
\ff 06,01,02,3e,0a,03,00 03,06,d0,0c 08,03,03,67,05,c2,0a,00,00 00
\1]
"Add"=hex:02,08,f0,03,03,c2,0a,00
"Replace"=hex:01,0b,67,00,03,c2,0a,00,e9,86,03
==================================================================
The Add value has data with a size of 8 bytes, saying that at offset
3f0h that the 3 bytes c2,0a,00 should be added.
"Add"=hex:02,08,f0,03,03,c2,0a,00
|| || || || || || || ||
type=Add--++ || || || || || || ||
total bytes--++ || || || || || ||
offset----------++-++ || || || ||
byte count------------++ || || ||
bytes to add-------------++-++-++
The Replace value has data with a size of 0bh bytes, saying that at
offset 67h that the 3 bytes c2,0a,00 should be replaced with e9,86,03.
"Replace"=hex:01,0b,67,00,03,c2,0a,00,e9,86,03
|| || || || || || || || || || ||
type=Replace--++ || || || || || || || || || ||
total bytes------++ || || || || || || || || ||
offset--------------++-++ || || || || || || ||
byte count----------------++ || || || || || ||
old bytes--------------------++-++-++ || || ||
new bytes-----------------------------++-++-++
The combined effect of these two changes is to replace a "retn a" at 67
with "jmp 3f0" and, at 3f0 to add "retn a". The app runs exactly as
before in this case, except for a small detour. When it reaches ip=67,
instead of doing "retn a", it does a "jmp 3f0" and then the "retn a".
Example for code segment 1 of module "INSTALL":
We want to change the message number they are checking against in a WndProc
from 181 to 402. So here's what ya gatta do....
1. Open a patch.ini file in your favorite editor
2. Run private\tools16\gensig.exe : gensig file.exe seg#
where seg# is the 1-based HEX value of the segment you want to patch. You
will see the correct segment number to use if you do a "vdmexts!lm sel"
dump for the selector of interest under ntsd (it will be under the "Seg"
heading).
gensig install.exe 1
ff06010242136500030600d600 <--- resulting hex string signature
3. Grab the resulting hex string & paste it in your patch.ini file with the
following format: RegistryPath\ModuleName\HexSignature\SegNum
\Registry\MACHINE\SYSTEM\CurrentControlSet\Control\Session Manager\AppPatches\INSTALL\ff06010242136500030600d600\1
4. Now the fun part. The patch bytes list needs to be done in groups of 4 bytes
because it will be converted to DWORDS later.
The first DWORD is the 1-Byte_ByteCount with a bunch of leading zeros:
0x0000000B Note: this must match the 1-Byte_ByteCount in the next DWORD
See "^^Note:" at the end of this section on how to
calculate this.
The second DWORD takes the following format:
2-Byte_CodeOffset 1-Byte_ByteCount 1-Byte_PatchCode
where the 2-Byte_CodeOffset is the same offset you would type to
disassemble the code under ntsd: u seg:off
For this example we want to alter 3 bytes at 1:305 so we need to write:
0x03050B01
\__/\/\/
\ \ \___ The 1-Byte_PatchCode (01 = "replace")
\ \____ The 1-Byte_ByteCount (# bytes in the entire patch string)
\______ The code offset
Next write a 1-Byte_PatchByteCount followed by the PatchBytes. For the 3
bytes we are replacing in this example:
03 2d 81 01 2d 02 04 00
\/ \/ \___/ \/ \___/ \/ ______
\ \ \ \ \ \___ Zero pad the DWORD |
\ \ \ \ \_______ The value "402" |
\ \ \ \__________ "Sub" instruction code |-- PatchBytes
\ \ \________________ The value "181" |
\ \___________________ "Sub" instruction ______|
\_____________________ 1-Byte_PatchByteCount
Now DWORD arrange the above 4-byte clusters:
0x01812d03 0x0004022d
Put this all together as follows:
Change1 = REG_BINARY 0x0000000B 0x03050B01 0x01812d03 0x0004022d
\/ \/
\________\__ These two will be the same
^^Note: the 1-Byte_ByteCount is the count of the bytes in all of the
DWORDS we built except the first one (0x0000000B in this
case) and doesn't count any of the padding zeros in the
last DWORD. See below:
0 bytes 4 bytes 4 bytes 3 bytes
Change1 = REG_BINARY 0x0000000B 0x03050B01 0x01812d03 0x0004022d
1-Byte_ByteCount = 4 + 4 + 3 = 0xB
5. Next add another registry path entry to the "ModuleCompatibility" section:
\Registry\MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\ModuleCompatibility
Followed by a REG_SZ entry for the Module:
INSTALL = REG_SZ "0x0002"
\____/
\____ use "0x0002" for WOW in general
and "0x8000" for RISC WOW only
6. The resulting patch.ini file should look like this:
\Registry\MACHINE\SYSTEM\CurrentControlSet\Control\Session Manager\AppPatches\INSTALL\ff06010242136500030600d600\1
Change1 = REG_BINARY 0x0000000B 0x03050B01 0x01812d03 0x0004022d
\Registry\MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\ModuleCompatibility
INSTALL = REG_SZ "0x0002"
7. Run "regini patch.ini" to update the registry
8. Install a checked krnl386 and run the app under ntsd to see any patching
messages when the module loads. If you have any problems set a bp at
PatchAppSegWorker in krnl386 (source: wow16\kernel31\patch.asm) and step
through what's going on. Chances are you have the byte order mixed up and
you can see how you will need to re-arrange the PatchBytes in patch.ini
to get what you really want.