Are you modifying those on a byte-by-byte level? If so, and you're doing replacing rather than inserting, then my macros could work for you. However, they don't allow for insertion (because that would break the executable by messing up later offsets). But you could probably tweak it slightly to do that if you need it to.
Here's the macros:
DEFINE_PATCH_MACRO ~Q_Engine_Patcher~
BEGIN
SET i = 0
SET Q_SaveLoop = Q_Loop
WHILE (i < 0x10001) BEGIN //search 0x10000 bytes from the starting point for our search pattern.
SET offset = "Q_Starting_Offset" + "%i%"
READ_ASCII "%offset%" pattern_current (%searchlength%)
PATCH_IF ("%pattern_current%" STRING_EQUAL "%searchpattern%") THEN BEGIN
INNER_PATCH_SAVE hexoffsetstring ~0x000000~
BEGIN
SET hexoffset = offset + Q_Replace_Offset
SET byte = hexoffset / 1048576
PATCH_IF byte < 10 BEGIN WRITE_BYTE 0x2 byte + 48 END
PATCH_IF byte > 9 BEGIN WRITE_BYTE 0x2 byte + 55 END
SET hexoffset = hexoffset - (byte * 1048576)
SET byte = hexoffset / 65536
PATCH_IF byte < 10 BEGIN WRITE_BYTE 0x3 byte + 48 END
PATCH_IF byte > 9 BEGIN WRITE_BYTE 0x3 byte + 55 END
SET hexoffset = hexoffset - (byte * 65536)
SET byte = hexoffset / 4096
PATCH_IF byte < 10 BEGIN WRITE_BYTE 0x4 byte + 48 END
PATCH_IF byte > 9 BEGIN WRITE_BYTE 0x4 byte + 55 END
SET hexoffset = hexoffset - (byte * 4096)
SET byte = hexoffset / 256
PATCH_IF byte < 10 BEGIN WRITE_BYTE 0x5 byte + 48 END
PATCH_IF byte > 9 BEGIN WRITE_BYTE 0x5 byte + 55 END
SET hexoffset = hexoffset - (byte * 256)
SET byte = hexoffset / 16
PATCH_IF byte < 10 BEGIN WRITE_BYTE 0x6 byte + 48 END
PATCH_IF byte > 9 BEGIN WRITE_BYTE 0x6 byte + 55 END
SET byte = hexoffset - (byte * 16)
PATCH_IF byte < 10 BEGIN WRITE_BYTE 0x7 byte + 48 END
PATCH_IF byte > 9 BEGIN WRITE_BYTE 0x7 byte + 55 END
END
WRITE_EVALUATED_ASCII ("%offset%" + "Q_Replace_Offset") "%replacepattern%" (%replacelength%)
PATCH_IF Q_Loop = 0 BEGIN SET i = 0x10001 END
ELSE BEGIN SET Q_Loop = Q_Loop - 1 END
PATCH_PRINT ~%Q_Patch_Name%~
PATCH_PRINT ~Engine Patched at Offset: %hexoffsetstring%~
END
SET i += 1
PATCH_IF ("%i%" = 0x10001) AND ( Q_SaveLoop = 0 OR Q_Loop > 0 ) BEGIN
INNER_ACTION BEGIN
FAIL ~No pattern for %Q_Patch_Name% found.~
END
END
END
END
DEFINE_PATCH_MACRO ~Q_Pattern_Maker~
BEGIN
PATCH_IF searchlength > 0
BEGIN
FOR (i = 0; i < searchlength; i += 1)
BEGIN
INNER_PATCH ~%searchbytes%~
BEGIN
READ_BYTE ((i * 3) + 0) byte1
READ_BYTE ((i * 3) + 1) byte2
END
INNER_PATCH_SAVE searchpattern ~%searchpattern%~
BEGIN
PATCH_IF byte1 >= 48 AND byte1 <= 57 THEN BEGIN SET byte1 = (byte1 - 48) * 16 END ELSE BEGIN SET byte1 =(byte1 - 55) * 16 END
PATCH_IF byte2 >= 48 AND byte2 <= 57 THEN BEGIN SET byte2 = (byte2 - 48) END ELSE BEGIN SET byte2 =(byte2 - 55) END
SET byte = byte1 + byte2
WRITE_BYTE i byte
END
END
END
PATCH_IF replacelength > 0
BEGIN
FOR (i = 0; i < replacelength; i += 1)
BEGIN
INNER_PATCH ~%replacebytes%~
BEGIN
READ_BYTE ((i * 3) + 0) byte1
READ_BYTE ((i * 3) + 1) byte2
END
INNER_PATCH_SAVE replacepattern ~%replacepattern%~
BEGIN
PATCH_IF byte1 >= 48 AND byte1 <= 57 THEN BEGIN SET byte1 = (byte1 - 48) * 16 END ELSE BEGIN SET byte1 =(byte1 - 55) * 16 END
PATCH_IF byte2 >= 48 AND byte2 <= 57 THEN BEGIN SET byte2 = (byte2 - 48) END ELSE BEGIN SET byte2 =(byte2 - 55) END
SET byte = byte1 + byte2
WRITE_BYTE i byte
END
END
END
END
(Note that a big chunk of that first macro does nothing more than convert a decimal offset to hexadecimal for printout in the
weidu log).
And here's a sample call to the macros:
SPRINT "Q_Patch_Name" ~4d. Make Morale functions use Maximum Morale + roll over fixes. Patch 9.~
// search: 88 82 27 07 00 00 8B 45 08 33 C9 8A 88 27 07 00 00
// replace: ?? ?? 5C 05 ?? ?? ?? ?? ?? 0F BE 88 5C 05 00 ?? 90
SET Q_Starting_Offset = 0xB9D00
SET Q_Replace_Offset = 2
SET searchlength = 17
SPRINT searchpattern ~12345678901234567~
SPRINT searchbytes ~88 82 27 07 00 00 8B 45 08 33 C9 8A 88 27 07 00 00~
SET replacelength = 15
SPRINT replacepattern ~123456789012345~
SPRINT replacebytes ~5C 05 00 00 8B 45 08 0F BE 88 5C 05 00 00 90~
LAUNCH_PATCH_MACRO Q_Pattern_Maker
LAUNCH_PATCH_MACRO Q_Engine_Patcher
Note that the searchpattern and replacepattern are just strings of searchlength and replacelength lengths, respectively.
Q_ReplaceOffset is how many bytes (positive or negative) from where the searchbytes are found to place the replacebytes.
Q_StartingOffset is what byte location to start looking for the searchpattern in, so you don't have to search the whole file every time.
Summary: The Q_Pattern_Maker macro takes the searchbytes and replacebytes strings, and converts them to an actual sequence of the bytes in question (by replacing the bytes of searchpattern and replacepattern one byte at a time). Q_Engine_Patcher then looks for the searchpattern starting at the Q_StartingOffset, and writes the replacepattern at the offset where it found the searchpattern, plus the value of Q_Replace_Offset (which can be and often is negative).
Very simple to make it do different patterns based on whether you're using the 2CD or 4CD executable (which -are- different).
Qwinn
Edited by Qwinn, 07 October 2008 - 06:05 AM.