1
Fork 0
mirror of https://bitbucket.org/Ioncannon/project-meteor-server.git synced 2025-04-20 19:57:46 +00:00

Finished writing most of this exe patcher.

This commit is contained in:
Filip Maj 2017-01-26 19:07:47 -05:00
parent 8499eeff39
commit ab3ca412b3

View file

@ -7,11 +7,278 @@ using System.Threading.Tasks;
namespace Launcher_Editor namespace Launcher_Editor
{ {
//ffxivboot.exe:
//Offset
//0x9663FC: Patch Server Port
//0x966404: Patch Server URL
//0x9663FC + 0x400000: Port Offset to search
//0x966404 + 0x400000: URL Offset to search
class Program class Program
{ {
static void Main(string[] args) static void Main(string[] args)
{ {
Console.WriteLine(FFXIVLoginStringDecodeBinary("C:\\Program Files (x86)\\SquareEnix\\FINAL FANTASY XIV\\ffxivlogin.exe")); //Console.WriteLine(FFXIVLoginStringDecodeBinary("C:\\Program Files (x86)\\SquareEnix\\FINAL FANTASY XIV\\ffxivlogin.exe"));
byte[] exeDataBoot = File.ReadAllBytes("C:\\Program Files (x86)\\SquareEnix\\FINAL FANTASY XIV\\ffxivboot - Copy.exe");
byte[] exeDataLogin = File.ReadAllBytes("C:\\Program Files (x86)\\SquareEnix\\FINAL FANTASY XIV\\ffxivlogin - Copy.exe");
string patchPortString = "54996";
string patchUrlString = "ver01.ffxiv.com";
string lobbyUrlString = "lobby01.ffxiv.com";
long patchPortStringOffset = 0;
long patchUrlStringOffset = 0;
long lobbyUrlStringOffset = 0;
long freeSpaceOffset = 0;
long loginUrlOffset = 0;
long freeSpaceInLoginOffset = 0;
Console.WriteLine("---Editing FFXIVBOOT.EXE---");
patchPortStringOffset = PrintSearch(exeDataBoot, patchPortString);
patchUrlStringOffset = PrintSearch(exeDataBoot, patchUrlString);
freeSpaceOffset = PrintFreeSpaceSearch(exeDataBoot);
Console.WriteLine("Writing \"{0}\" and updating offset to 0x{1:X}.", "80", freeSpaceOffset);
WriteNewString(exeDataBoot, patchPortStringOffset, "80", freeSpaceOffset);
Console.WriteLine("Writing \"{0}\" and updating offset to 0x{1:X}.", "127.0.0.1", freeSpaceOffset + 0x20);
WriteNewString(exeDataBoot, patchUrlStringOffset, "127.0.0.1", freeSpaceOffset + 0x20);
Console.WriteLine("---Editing FFXIVLOGIN.EXE---");
loginUrlOffset = PrintEncodedSearch(exeDataLogin, 0x739, "http://account.square-enix.com/account/content/ffxivlogin");
freeSpaceInLoginOffset = PrintFreeSpaceSearch(exeDataLogin);
Console.WriteLine("Writing encoded \"{0}\" and updating offset to 0x{1:X}.", "http://127.0.0.1/", freeSpaceInLoginOffset);
WriteNewStringEncoded(exeDataLogin, loginUrlOffset, 0x739, "http://127.0.0.1/", freeSpaceInLoginOffset);
File.WriteAllBytes("C:\\Users\\Filip\\Desktop\\ffxivboot_test.exe", exeDataBoot);
File.WriteAllBytes("C:\\Users\\Filip\\Desktop\\ffxivlogin_test.exe", exeDataLogin);
}
public static void WriteNewString(byte[] exeData, long offsetLocation, string newString, long freeSpaceLocation)
{
using (MemoryStream memStream = new MemoryStream(exeData))
{
using (BinaryWriter binaryWriter = new BinaryWriter(memStream))
{
binaryWriter.BaseStream.Seek(offsetLocation, SeekOrigin.Begin);
binaryWriter.Write((uint)freeSpaceLocation + 0x400000);
binaryWriter.BaseStream.Seek(freeSpaceLocation, SeekOrigin.Begin);
binaryWriter.Write(Encoding.ASCII.GetBytes(newString), 0, Encoding.ASCII.GetByteCount(newString) >= 0x20 ? 0x20 : Encoding.ASCII.GetByteCount(newString));
}
}
}
public static void WriteNewStringEncoded(byte[] exeData, long offsetLocation, uint key, string newString, long freeSpaceLocation)
{
byte[] encodedString = FFXIVLoginStringEncode(key, newString);
using (MemoryStream memStream = new MemoryStream(exeData))
{
using (BinaryWriter binaryWriter = new BinaryWriter(memStream))
{
//binaryWriter.BaseStream.Seek(offsetLocation, SeekOrigin.Begin);
//binaryWriter.Write((uint)freeSpaceLocation + 0x400000);
binaryWriter.BaseStream.Seek(offsetLocation, SeekOrigin.Begin);
binaryWriter.Write(encodedString);
}
}
}
public static long PrintSearch(byte[] exeData, string searchString)
{
Console.Write("Searching for string \"{0}\"...", searchString);
long offset = SearchForStringOffset(exeData, searchString);
if (offset != -1)
Console.WriteLine(" FOUND @ 0x{0:X}!", offset);
else
{
Console.WriteLine(" ERROR, could not find string.");
}
return offset;
}
public static long PrintEncodedSearch(byte[] exeData, uint key, string searchString)
{
Console.Write("Searching for encoded string \"{0}\"...", searchString);
long offset = SearchForEncodedStringOffset(exeData, key, searchString);
if (offset != -1)
Console.WriteLine(" FOUND @ 0x{0:X}!", offset);
else
{
Console.WriteLine(" ERROR, could not find string.");
}
return offset;
}
public static long PrintFreeSpaceSearch(byte[] exeData)
{
Console.Write("Searching for free space...");
long freeSpaceOffset = SearchForFreeSpace(exeData);
if (freeSpaceOffset != -1)
Console.WriteLine(" FOUND @ 0x{0:X}!", freeSpaceOffset);
else
{
Console.WriteLine(" ERROR, could not find free space.");
}
return freeSpaceOffset;
}
public static bool EditOffset(long offset, uint value)
{
return true;
}
public static long SearchForFreeSpace(byte[] exeData)
{
using (MemoryStream memStream = new MemoryStream(exeData))
{
using (BinaryReader binReader = new BinaryReader(memStream))
{
//Find the .data section header
long textSectionOffset = -1;
int strCheckoffset = 0;
while (binReader.BaseStream.Position + 4 < binReader.BaseStream.Length)
{
if (binReader.ReadByte() == ".text"[strCheckoffset])
{
if (strCheckoffset == 0)
textSectionOffset = binReader.BaseStream.Position - 1;
strCheckoffset++;
if (strCheckoffset == Encoding.ASCII.GetByteCount(".data"))
break;
}
else
{
strCheckoffset = 0;
textSectionOffset = -1;
}
}
//Read in the position and size
binReader.BaseStream.Seek(textSectionOffset, SeekOrigin.Begin);
binReader.ReadUInt64();
uint virtualSize = binReader.ReadUInt32();
uint address = binReader.ReadUInt32();
uint sizeOfRawData = binReader.ReadUInt32();
if (sizeOfRawData - virtualSize < 0x50)
return -1;
//Find a spot
binReader.BaseStream.Seek(address + sizeOfRawData, SeekOrigin.Begin);
while (binReader.BaseStream.Position >= address + virtualSize)
{
binReader.BaseStream.Seek(-0x50, SeekOrigin.Current);
long newPosition = binReader.BaseStream.Position;
bool foundNotZero = false;
for (int i = 0; i < 0x50; i++)
{
if (binReader.ReadByte() != 0)
{
foundNotZero = true;
break;
}
}
if (!foundNotZero)
return newPosition;
else
binReader.BaseStream.Seek(newPosition, SeekOrigin.Begin);
}
}
}
return -1;
}
public static long SearchForStringOffset(byte[] exeData, string testString)
{
testString += "\0";
using (MemoryStream memStream = new MemoryStream(exeData))
{
using (BinaryReader binReader = new BinaryReader(memStream))
{
long strOffset = -1;
int strCheckoffset = 0;
while (binReader.BaseStream.Position + 4 < binReader.BaseStream.Length)
{
if (binReader.ReadByte() == testString[strCheckoffset])
{
if (strCheckoffset == 0)
strOffset = binReader.BaseStream.Position-1;
strCheckoffset++;
if (strCheckoffset == Encoding.ASCII.GetByteCount(testString))
break;
}
else
{
strCheckoffset = 0;
strOffset = -1;
}
}
if (strOffset != -1)
{
strOffset += 0x400000;
binReader.BaseStream.Seek(0, SeekOrigin.Begin);
while (binReader.BaseStream.Position + 4 < binReader.BaseStream.Length)
{
if (binReader.ReadUInt32() == strOffset)
return binReader.BaseStream.Position - 0x4;
}
return -1;
}
else
return -1;
}
}
}
public static long SearchForEncodedStringOffset(byte[] exeData, uint testKey, string testString)
{
byte[] encoded = FFXIVLoginStringEncode(testKey, testString);
using (MemoryStream memStream = new MemoryStream(exeData))
{
using (BinaryReader binReader = new BinaryReader(memStream))
{
long strOffset = -1;
int strCheckoffset = 0;
while (binReader.BaseStream.Position + 4 < binReader.BaseStream.Length)
{
if (binReader.ReadByte() == encoded[strCheckoffset])
{
if (strCheckoffset == 0)
strOffset = binReader.BaseStream.Position - 1;
strCheckoffset++;
if (strCheckoffset == encoded.Length)
break;
}
else
{
strCheckoffset = 0;
strOffset = -1;
}
}
return strOffset;
}
}
} }
public static string FFXIVLoginStringDecodeBinary(string path) public static string FFXIVLoginStringDecodeBinary(string path)
@ -54,20 +321,26 @@ namespace Launcher_Editor
count2++; count2++;
} }
offset += 4 + count2; return result;
//offset += 4 + count2;
} }
} }
public static string FFXIVLoginStringDecode(byte[] data) public static string FFXIVLoginStringDecode(byte[] data)
{
Console.OutputEncoding = System.Text.Encoding.UTF8;
while (true)
{ {
string result = ""; string result = "";
uint key = (uint)data[0] << 8 | data[1]; uint key = (uint)data[0] << 8 | data[1];
uint key2 = data[2]; uint key2 = data[2];
key = RotateRight(key, 1) & 0xFFFF; key = RotateRight(key, 1) & 0xFFFF;
key -= 0x22AF; key -= 0x22AF;
key &= 0xFFFF;
key2 = key2 ^ key; key2 = key2 ^ key;
key = RotateRight(key, 1) & 0xFFFF; key = RotateRight(key, 1) & 0xFFFF;
key -= 0x22AF; key -= 0x22AF;
key &= 0xFFFF;
uint finalKey = key; uint finalKey = key;
key = data[3]; key = data[3];
uint count = (key2 & 0xFF) << 8; uint count = (key2 & 0xFF) << 8;
@ -81,13 +354,17 @@ namespace Launcher_Editor
uint encrypted = data[4 + count2]; uint encrypted = data[4 + count2];
finalKey = RotateRight(finalKey, 1) & 0xFFFF; finalKey = RotateRight(finalKey, 1) & 0xFFFF;
finalKey -= 0x22AF; finalKey -= 0x22AF;
finalKey &= 0xFFFF;
encrypted = encrypted ^ (finalKey & 0xFF); encrypted = encrypted ^ (finalKey & 0xFF);
result += (char)encrypted; result += (char)encrypted;
count--; count--;
count2++; count2++;
} }
return result; return result;
//offset += 4 + count2;
}
} }
public static byte[] FFXIVLoginStringEncode(uint key, string text) public static byte[] FFXIVLoginStringEncode(uint key, string text)
@ -102,20 +379,24 @@ namespace Launcher_Editor
result[result.Length - count - 1] = (byte)(asciiBytes[asciiBytes.Length - count - 1] ^ (key & 0xFF)); result[result.Length - count - 1] = (byte)(asciiBytes[asciiBytes.Length - count - 1] ^ (key & 0xFF));
key += 0x22AF; key += 0x22AF;
key &= 0xFFFF; key &= 0xFFFF;
key = RotateLeft(key, 1) & 0xFFFF; key = RotateLeft(key, 1);
key &= 0xFFFF;
} }
count = count ^ key; count = count ^ key;
result[3] = (byte)(count & 0xFF); result[3] = (byte)(count & 0xFF);
key += 0x22AF & 0xFFFF; key += 0x22AF;
key = RotateLeft(key, 1) & 0xFFFF; key &= 0xFFFF;
key = RotateLeft(key, 1);
key &= 0xFFFF;
result[2] = (byte)(key & 0xFF); result[2] = (byte)(key & 0xFF);
key += 0x22AF & 0xFFFF; key += 0x22AF;
key = RotateLeft(key, 1) & 0xFFFF; key &= 0xFFFF;
key = RotateLeft(key, 1);
key &= 0xFFFF;
result[1] = (byte)(key & 0xFF); result[1] = (byte)(key & 0xFF);
result[0] = (byte)((key >> 8) & 0xFF); result[0] = (byte)((key >> 8) & 0xFF);