mirror of
https://bitbucket.org/Ioncannon/project-meteor-server.git
synced 2025-04-20 11:47:48 +00:00
Finished writing most of this exe patcher.
This commit is contained in:
parent
8499eeff39
commit
ab3ca412b3
1 changed files with 315 additions and 34 deletions
|
@ -7,11 +7,278 @@ using System.Threading.Tasks;
|
|||
|
||||
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
|
||||
{
|
||||
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)
|
||||
|
@ -54,40 +321,50 @@ namespace Launcher_Editor
|
|||
count2++;
|
||||
}
|
||||
|
||||
offset += 4 + count2;
|
||||
return result;
|
||||
//offset += 4 + count2;
|
||||
}
|
||||
}
|
||||
|
||||
public static string FFXIVLoginStringDecode(byte[] data)
|
||||
{
|
||||
string result = "";
|
||||
uint key = (uint)data[0] << 8 | data[1];
|
||||
uint key2 = data[2];
|
||||
key = RotateRight(key, 1) & 0xFFFF;
|
||||
key -= 0x22AF;
|
||||
key2 = key2 ^ key;
|
||||
key = RotateRight(key, 1) & 0xFFFF;
|
||||
key -= 0x22AF;
|
||||
uint finalKey = key;
|
||||
key = data[3];
|
||||
uint count = (key2 & 0xFF) << 8;
|
||||
key = key ^ finalKey;
|
||||
key &= 0xFF;
|
||||
count |= key;
|
||||
|
||||
int count2 = 0;
|
||||
while (count != 0)
|
||||
Console.OutputEncoding = System.Text.Encoding.UTF8;
|
||||
while (true)
|
||||
{
|
||||
uint encrypted = data[4 + count2];
|
||||
finalKey = RotateRight(finalKey, 1) & 0xFFFF;
|
||||
finalKey -= 0x22AF;
|
||||
encrypted = encrypted ^ (finalKey & 0xFF);
|
||||
result += (char)encrypted;
|
||||
count--;
|
||||
count2++;
|
||||
}
|
||||
string result = "";
|
||||
uint key = (uint)data[0] << 8 | data[1];
|
||||
uint key2 = data[2];
|
||||
key = RotateRight(key, 1) & 0xFFFF;
|
||||
key -= 0x22AF;
|
||||
key &= 0xFFFF;
|
||||
key2 = key2 ^ key;
|
||||
key = RotateRight(key, 1) & 0xFFFF;
|
||||
key -= 0x22AF;
|
||||
key &= 0xFFFF;
|
||||
uint finalKey = key;
|
||||
key = data[3];
|
||||
uint count = (key2 & 0xFF) << 8;
|
||||
key = key ^ finalKey;
|
||||
key &= 0xFF;
|
||||
count |= key;
|
||||
|
||||
return result;
|
||||
int count2 = 0;
|
||||
while (count != 0)
|
||||
{
|
||||
uint encrypted = data[4 + count2];
|
||||
finalKey = RotateRight(finalKey, 1) & 0xFFFF;
|
||||
finalKey -= 0x22AF;
|
||||
finalKey &= 0xFFFF;
|
||||
encrypted = encrypted ^ (finalKey & 0xFF);
|
||||
|
||||
result += (char)encrypted;
|
||||
count--;
|
||||
count2++;
|
||||
}
|
||||
|
||||
return result;
|
||||
//offset += 4 + count2;
|
||||
}
|
||||
}
|
||||
|
||||
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));
|
||||
key += 0x22AF;
|
||||
key &= 0xFFFF;
|
||||
key = RotateLeft(key, 1) & 0xFFFF;
|
||||
key = RotateLeft(key, 1);
|
||||
key &= 0xFFFF;
|
||||
}
|
||||
|
||||
count = count ^ key;
|
||||
result[3] = (byte)(count & 0xFF);
|
||||
|
||||
key += 0x22AF & 0xFFFF;
|
||||
key = RotateLeft(key, 1) & 0xFFFF;
|
||||
key += 0x22AF;
|
||||
key &= 0xFFFF;
|
||||
key = RotateLeft(key, 1);
|
||||
key &= 0xFFFF;
|
||||
|
||||
result[2] = (byte)(key & 0xFF);
|
||||
|
||||
key += 0x22AF & 0xFFFF;
|
||||
key = RotateLeft(key, 1) & 0xFFFF;
|
||||
|
||||
key += 0x22AF;
|
||||
key &= 0xFFFF;
|
||||
key = RotateLeft(key, 1);
|
||||
key &= 0xFFFF;
|
||||
|
||||
result[1] = (byte)(key & 0xFF);
|
||||
result[0] = (byte)((key >> 8) & 0xFF);
|
||||
|
|
Loading…
Add table
Reference in a new issue