336 lines
15 KiB
Plaintext
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.
|