using System; using System.Collections.Generic; using System.Runtime.InteropServices; using System.Diagnostics; using FFXIVClassic_Map_Server.common; using System.IO; namespace FFXIVClassic_Map_Server.packets { [StructLayout(LayoutKind.Sequential)] public struct BasePacketHeader { public byte isAuthenticated; public byte isCompressed; public ushort connectionType; public ushort packetSize; public ushort numSubpackets; public ulong timestamp; //Miliseconds } public class BasePacket{ public const int TYPE_ZONE = 1; public const int TYPE_CHAT = 2; public const int BASEPACKET_SIZE = 0x10; public BasePacketHeader header; public byte[] data; //Loads a sniffed packet from a file public unsafe BasePacket(String path) { byte[] bytes = File.ReadAllBytes(path); if (bytes.Length < BASEPACKET_SIZE) throw new OverflowException("Packet Error: Packet was too small"); fixed (byte* pdata = &bytes[0]) { header = (BasePacketHeader)Marshal.PtrToStructure(new IntPtr(pdata), typeof(BasePacketHeader)); } if (bytes.Length < header.packetSize) throw new OverflowException("Packet Error: Packet size didn't equal given size"); int packetSize = header.packetSize; if (packetSize - BASEPACKET_SIZE != 0) { data = new byte[packetSize - BASEPACKET_SIZE]; Array.Copy(bytes, BASEPACKET_SIZE, data, 0, packetSize - BASEPACKET_SIZE); } else data = new byte[0]; } //Loads a sniffed packet from a byte array public unsafe BasePacket(byte[] bytes) { if (bytes.Length < BASEPACKET_SIZE) throw new OverflowException("Packet Error: Packet was too small"); fixed (byte* pdata = &bytes[0]) { header = (BasePacketHeader)Marshal.PtrToStructure(new IntPtr(pdata), typeof(BasePacketHeader)); } if (bytes.Length < header.packetSize) throw new OverflowException("Packet Error: Packet size didn't equal given size"); int packetSize = header.packetSize; data = new byte[packetSize - BASEPACKET_SIZE]; Array.Copy(bytes, BASEPACKET_SIZE, data, 0, packetSize - BASEPACKET_SIZE); } public unsafe BasePacket(byte[] bytes, ref int offset) { if (bytes.Length < offset + BASEPACKET_SIZE) throw new OverflowException("Packet Error: Packet was too small"); fixed (byte* pdata = &bytes[offset]) { header = (BasePacketHeader)Marshal.PtrToStructure(new IntPtr(pdata), typeof(BasePacketHeader)); } int packetSize = header.packetSize; if (bytes.Length < offset + header.packetSize) throw new OverflowException("Packet Error: Packet size didn't equal given size"); data = new byte[packetSize - BASEPACKET_SIZE]; Array.Copy(bytes, offset + BASEPACKET_SIZE, data, 0, packetSize - BASEPACKET_SIZE); offset += packetSize; } public BasePacket(BasePacketHeader header, byte[] data) { this.header = header; this.data = data; } public List getSubpackets() { List subpackets = new List(header.numSubpackets); int offset = 0; while (offset < data.Length) subpackets.Add(new SubPacket(data, ref offset)); return subpackets; } public unsafe static BasePacketHeader getHeader(byte[] bytes) { BasePacketHeader header; if (bytes.Length < BASEPACKET_SIZE) throw new OverflowException("Packet Error: Packet was too small"); fixed (byte* pdata = &bytes[0]) { header = (BasePacketHeader)Marshal.PtrToStructure(new IntPtr(pdata), typeof(BasePacketHeader)); } return header; } public byte[] getHeaderBytes() { int size = Marshal.SizeOf(header); byte[] arr = new byte[size]; IntPtr ptr = Marshal.AllocHGlobal(size); Marshal.StructureToPtr(header, ptr, true); Marshal.Copy(ptr, arr, 0, size); Marshal.FreeHGlobal(ptr); return arr; } public byte[] getPacketBytes() { byte[] outBytes = new byte[header.packetSize]; Array.Copy(getHeaderBytes(), 0, outBytes, 0, BASEPACKET_SIZE); Array.Copy(data, 0, outBytes, BASEPACKET_SIZE, data.Length); return outBytes; } //Replaces all instances of the sniffed actorID with the given one public void replaceActorID(uint actorID) { using (MemoryStream mem = new MemoryStream(data)) { using (BinaryWriter binWriter = new BinaryWriter(mem)) { using (BinaryReader binreader = new BinaryReader(mem)) { while (binreader.BaseStream.Position + 4 < data.Length) { uint read = binreader.ReadUInt32(); if (read == 0x029B2941 || read == 0x02977DC7 || read == 0x0297D2C8 || read == 0x0230d573 || read == 0x23317df || read == 0x23344a3 || read == 0x1730bdb || read == 0x6c) //Original ID { binWriter.BaseStream.Seek(binreader.BaseStream.Position - 0x4, SeekOrigin.Begin); binWriter.Write(actorID); } } } } } } //Replaces all instances of the sniffed actorID with the given one public void replaceActorID(uint fromActorID, uint actorID) { using (MemoryStream mem = new MemoryStream(data)) { using (BinaryWriter binWriter = new BinaryWriter(mem)) { using (BinaryReader binreader = new BinaryReader(mem)) { while (binreader.BaseStream.Position + 4 < data.Length) { uint read = binreader.ReadUInt32(); if (read == fromActorID) //Original ID { binWriter.BaseStream.Seek(binreader.BaseStream.Position - 0x4, SeekOrigin.Begin); binWriter.Write(actorID); } } } } } } #region Utility Functions public static BasePacket createPacket(List subpackets, bool isAuthed, bool isCompressed) { //Create Header BasePacketHeader header = new BasePacketHeader(); byte[] data = null; header.isAuthenticated = isAuthed?(byte)1:(byte)0; header.isCompressed = isCompressed?(byte)1:(byte)0; header.numSubpackets = (ushort)subpackets.Count; header.packetSize = BASEPACKET_SIZE; header.timestamp = Utils.MilisUnixTimeStampUTC(); //Get packet size foreach (SubPacket subpacket in subpackets) header.packetSize += subpacket.header.subpacketSize; data = new byte[header.packetSize-0x10]; //Add Subpackets int offset = 0; foreach (SubPacket subpacket in subpackets) { byte[] subpacketData = subpacket.getBytes(); Array.Copy(subpacketData, 0, data, offset, subpacketData.Length); offset += (ushort)subpacketData.Length; } Debug.Assert(data != null && offset == data.Length && header.packetSize == 0x10 + offset); BasePacket packet = new BasePacket(header, data); return packet; } public static BasePacket createPacket(SubPacket subpacket, bool isAuthed, bool isCompressed) { //Create Header BasePacketHeader header = new BasePacketHeader(); byte[] data = null; header.isAuthenticated = isAuthed ? (byte)1 : (byte)0; header.isCompressed = isCompressed ? (byte)1 : (byte)0; header.numSubpackets = (ushort)1; header.packetSize = BASEPACKET_SIZE; header.timestamp = Utils.MilisUnixTimeStampUTC(); //Get packet size header.packetSize += subpacket.header.subpacketSize; data = new byte[header.packetSize - 0x10]; //Add Subpackets byte[] subpacketData = subpacket.getBytes(); Array.Copy(subpacketData, 0, data, 0, subpacketData.Length); Debug.Assert(data != null); BasePacket packet = new BasePacket(header, data); return packet; } public static BasePacket createPacket(byte[] data, bool isAuthed, bool isCompressed) { Debug.Assert(data != null); //Create Header BasePacketHeader header = new BasePacketHeader(); header.isAuthenticated = isAuthed ? (byte)1 : (byte)0; header.isCompressed = isCompressed ? (byte)1 : (byte)0; header.numSubpackets = (ushort)1; header.packetSize = BASEPACKET_SIZE; header.timestamp = Utils.MilisUnixTimeStampUTC(); //Get packet size header.packetSize += (ushort)data.Length; BasePacket packet = new BasePacket(header, data); return packet; } public static unsafe void encryptPacket(Blowfish blowfish, BasePacket packet) { byte[] data = packet.data; int size = packet.header.packetSize; int offset = 0; while (offset < data.Length) { if (data.Length < offset + SubPacket.SUBPACKET_SIZE) throw new OverflowException("Packet Error: Subpacket was too small"); SubPacketHeader header; fixed (byte* pdata = &data[offset]) { header = (SubPacketHeader)Marshal.PtrToStructure(new IntPtr(pdata), typeof(SubPacketHeader)); } if (data.Length < offset + header.subpacketSize) throw new OverflowException("Packet Error: Subpacket size didn't equal subpacket data"); blowfish.Encipher(data, offset + 0x10, header.subpacketSize-0x10); offset += header.subpacketSize; } } public static unsafe void decryptPacket(Blowfish blowfish, ref BasePacket packet) { byte[] data = packet.data; int size = packet.header.packetSize; int offset = 0; while (offset < data.Length) { if (data.Length < offset + SubPacket.SUBPACKET_SIZE) throw new OverflowException("Packet Error: Subpacket was too small"); SubPacketHeader header; fixed (byte* pdata = &data[offset]) { header = (SubPacketHeader)Marshal.PtrToStructure(new IntPtr(pdata), typeof(SubPacketHeader)); } if (data.Length < offset + header.subpacketSize) throw new OverflowException("Packet Error: Subpacket size didn't equal subpacket data"); blowfish.Decipher(data, offset + 0x10, header.subpacketSize-0x10); offset += header.subpacketSize; } } #endregion public void debugPrintPacket() { #if DEBUG Console.BackgroundColor = ConsoleColor.DarkYellow; Log.Debug(String.Format("IsAuth: {0} IsEncrypted: {1}, Size: 0x{2:X}, NumSubpackets: {3}{4}{5}", header.isAuthenticated, header.isCompressed, header.packetSize, header.numSubpackets, Environment.NewLine, Utils.ByteArrayToHex(getHeaderBytes()))); foreach (SubPacket sub in getSubpackets()) { sub.debugPrintSubPacket(); } Console.BackgroundColor = ConsoleColor.Black; #endif } } }