继续修正数据包结构
This commit is contained in:
parent
ecc9ebe098
commit
d7edf84277
@ -321,49 +321,6 @@ namespace PSO2SERVER
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class NetInterface
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Interface status.
|
|
||||||
/// </summary>
|
|
||||||
public uint State { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Interface MAC address.
|
|
||||||
/// </summary>
|
|
||||||
public string Mac { get; set; } = new string('\0', 0x18); // 以字符串形式存储
|
|
||||||
|
|
||||||
public void ReadFromStream(PacketReader reader)
|
|
||||||
{
|
|
||||||
State = reader.ReadUInt32();
|
|
||||||
Mac = Encoding.ASCII.GetString(reader.ReadBytes(0x18)).TrimEnd('\0');
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string ToString()
|
|
||||||
{
|
|
||||||
return $"状态: {State}, MAC: {Mac}";
|
|
||||||
}
|
|
||||||
|
|
||||||
public void UpdateNetInterface(int id, int accountid, NetInterface updatedInterface)
|
|
||||||
{
|
|
||||||
using (var db = new ServerEf())
|
|
||||||
{
|
|
||||||
var existingInterface = db.AccountsNetInterFaces
|
|
||||||
.FirstOrDefault(x => x.AccountId == accountid && x.id == id);
|
|
||||||
|
|
||||||
if (existingInterface != null)
|
|
||||||
{
|
|
||||||
// 更新其他字段
|
|
||||||
existingInterface.State = (int)updatedInterface.State;
|
|
||||||
existingInterface.Mac = updatedInterface.Mac;
|
|
||||||
|
|
||||||
// 保存更改
|
|
||||||
db.SaveChanges();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public enum Language : uint
|
public enum Language : uint
|
||||||
{
|
{
|
||||||
Japanese = 0,
|
Japanese = 0,
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
|
using System.Net.Sockets;
|
||||||
using System.Runtime.InteropServices;
|
using System.Runtime.InteropServices;
|
||||||
|
|
||||||
namespace PSO2SERVER.Models
|
namespace PSO2SERVER.Models
|
||||||
@ -20,6 +21,10 @@ namespace PSO2SERVER.Models
|
|||||||
this.Flags2 = flags2;
|
this.Flags2 = flags2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PacketHeader(int size, byte type, byte subtype, PacketFlags flags1, PacketFlags flags2) : this(size, type, subtype, (byte)flags1, (byte)flags1)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
public PacketHeader(byte type, byte subtype) : this(type, subtype, (byte)0)
|
public PacketHeader(byte type, byte subtype) : this(type, subtype, (byte)0)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
@ -31,6 +36,24 @@ namespace PSO2SERVER.Models
|
|||||||
public PacketHeader(byte type, byte subtype, PacketFlags packetFlags) : this(type, subtype, (byte)packetFlags)
|
public PacketHeader(byte type, byte subtype, PacketFlags packetFlags) : this(type, subtype, (byte)packetFlags)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ToBytes 方法的实现
|
||||||
|
public byte[] ToBytes()
|
||||||
|
{
|
||||||
|
byte[] bytes = new byte[sizeof(UInt32) + sizeof(byte) * 4]; // 计算结构体的总大小
|
||||||
|
int index = 0;
|
||||||
|
|
||||||
|
// 将结构体每个字段转换为字节
|
||||||
|
Array.Copy(BitConverter.GetBytes(Size), 0, bytes, index, sizeof(UInt32));
|
||||||
|
index += sizeof(UInt32);
|
||||||
|
|
||||||
|
bytes[index++] = Type;
|
||||||
|
bytes[index++] = Subtype;
|
||||||
|
bytes[index++] = Flags1;
|
||||||
|
bytes[index++] = Flags2;
|
||||||
|
|
||||||
|
return bytes;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Packet flags.
|
/// Packet flags.
|
||||||
|
54
Server/Models/NetInterface.cs
Normal file
54
Server/Models/NetInterface.cs
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
using PSO2SERVER.Database;
|
||||||
|
using PSO2SERVER.Protocol;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace PSO2SERVER.Models
|
||||||
|
{
|
||||||
|
public class NetInterface
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Interface status.
|
||||||
|
/// </summary>
|
||||||
|
public uint State { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Interface MAC address.
|
||||||
|
/// </summary>
|
||||||
|
public string Mac { get; set; } = new string('\0', 0x18); // 以字符串形式存储
|
||||||
|
|
||||||
|
public void ReadFromStream(PacketReader reader)
|
||||||
|
{
|
||||||
|
State = reader.ReadUInt32();
|
||||||
|
Mac = Encoding.ASCII.GetString(reader.ReadBytes(0x18)).TrimEnd('\0');
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return $"状态: {State}, MAC: {Mac}";
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UpdateNetInterface(int id, int accountid, NetInterface updatedInterface)
|
||||||
|
{
|
||||||
|
using (var db = new ServerEf())
|
||||||
|
{
|
||||||
|
var existingInterface = db.AccountsNetInterFaces
|
||||||
|
.FirstOrDefault(x => x.AccountId == accountid && x.id == id);
|
||||||
|
|
||||||
|
if (existingInterface != null)
|
||||||
|
{
|
||||||
|
// 更新其他字段
|
||||||
|
existingInterface.State = (int)updatedInterface.State;
|
||||||
|
existingInterface.Mac = updatedInterface.Mac;
|
||||||
|
|
||||||
|
// 保存更改
|
||||||
|
db.SaveChanges();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
29
Server/Models/Orders.cs
Normal file
29
Server/Models/Orders.cs
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace PSO2SERVER.Models
|
||||||
|
{
|
||||||
|
public class Orders
|
||||||
|
{
|
||||||
|
public struct ClientOrder
|
||||||
|
{
|
||||||
|
public uint unk1;
|
||||||
|
public uint id;
|
||||||
|
public uint status;
|
||||||
|
public uint finish_date;
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct OrderStatus
|
||||||
|
{
|
||||||
|
public uint unk1;
|
||||||
|
public uint unk2;
|
||||||
|
public uint unk3;
|
||||||
|
public uint unk4;
|
||||||
|
public uint unk5;
|
||||||
|
public uint unk6;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -53,7 +53,7 @@ namespace PSO2SERVER.Models
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 从数据流中读取数据并填充到当前结构体
|
// 从数据流中读取数据并填充到当前结构体
|
||||||
public void ReadFromStream(PacketReader reader)
|
public void ReadObjectHeaderFromStream(PacketReader reader)
|
||||||
{
|
{
|
||||||
ID = reader.ReadUInt32(); // 读取对象ID
|
ID = reader.ReadUInt32(); // 读取对象ID
|
||||||
padding = reader.ReadUInt32(); // 读取填充部分
|
padding = reader.ReadUInt32(); // 读取填充部分
|
||||||
@ -62,7 +62,7 @@ namespace PSO2SERVER.Models
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 将当前结构体的数据写入到数据流
|
// 将当前结构体的数据写入到数据流
|
||||||
public void WriteToStream(PacketWriter writer)
|
public void WriteObjectHeaderToStream(PacketWriter writer)
|
||||||
{
|
{
|
||||||
writer.Write(ID); // 写入对象ID
|
writer.Write(ID); // 写入对象ID
|
||||||
writer.Write(padding); // 写入填充部分
|
writer.Write(padding); // 写入填充部分
|
||||||
|
207
Server/Models/RevealedRegions.cs
Normal file
207
Server/Models/RevealedRegions.cs
Normal file
@ -0,0 +1,207 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace PSO2SERVER.Models
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Revealed minimap regions
|
||||||
|
/// </summary>
|
||||||
|
public class RevealedRegions
|
||||||
|
{
|
||||||
|
// 内部实现:
|
||||||
|
// 该结构体由一个大小为[10]的字节数组组成,表示一个 8x10 的已揭示区域网格
|
||||||
|
// (即 10 列 8 行)。每一位(bit)表示一个区域是否已揭示。
|
||||||
|
// 位是从右到左索引的(即从 LSB 到 MSB),字节是从左到右索引的(arr[0]表示区域A1-8)。
|
||||||
|
// 获取该数组中的某个值可以通过如下方式:
|
||||||
|
//
|
||||||
|
// 示例代码:
|
||||||
|
// bool GetBit(byte[] arr, int row, int col)
|
||||||
|
// {
|
||||||
|
// // 确保行和列的范围有效
|
||||||
|
// if (row < 0 || row >= 8 || col < 0 || col >= 10)
|
||||||
|
// throw new ArgumentOutOfRangeException();
|
||||||
|
//
|
||||||
|
// // 计算对应位置的位偏移
|
||||||
|
// int offset = row * 10 + col;
|
||||||
|
// int byteOffset = offset / 8;
|
||||||
|
// int bitOffset = offset % 8;
|
||||||
|
//
|
||||||
|
// // 获取相应的位值
|
||||||
|
// return (arr[byteOffset] >> bitOffset & 1) == 1;
|
||||||
|
// }
|
||||||
|
|
||||||
|
// 使用一个长度为 10 的字节数组,表示 8 行 10 列的区域位图。
|
||||||
|
// Lsb0 表示最低有效位在字节的最右侧。
|
||||||
|
public byte[] zones = new byte[10];
|
||||||
|
|
||||||
|
// 默认构造函数
|
||||||
|
public RevealedRegions()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
// 构造函数 - 用指定数据初始化
|
||||||
|
public RevealedRegions(byte[] data)
|
||||||
|
{
|
||||||
|
// 如果传入的 unk2 长度小于 10,则填充剩余部分为 0
|
||||||
|
if (data.Length <= 10)
|
||||||
|
{
|
||||||
|
zones = data.Concat(new byte[10 - data.Length]).ToArray();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// 如果传入的 unk2 长度大于 10,则截取前 10 字节
|
||||||
|
zones = data.Take(10).ToArray();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 获取指定行列位置的位值
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="row">行索引(0 到 7)</param>
|
||||||
|
/// <param name="col">列索引(0 到 9)</param>
|
||||||
|
/// <returns>该位置的位值(true = 已揭示,false = 未揭示)</returns>
|
||||||
|
public bool GetBit(int row, int col)
|
||||||
|
{
|
||||||
|
// 确保行列值在有效范围内
|
||||||
|
if (row < 0 || row >= 8 || col < 0 || col >= 10)
|
||||||
|
{
|
||||||
|
throw new ArgumentOutOfRangeException();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 计算对应位置的偏移
|
||||||
|
int offset = row * 10 + col;
|
||||||
|
int byteOffset = offset / 8;
|
||||||
|
int bitOffset = offset % 8;
|
||||||
|
|
||||||
|
// 获取并返回该位置的位值
|
||||||
|
return (zones[byteOffset] >> bitOffset & 1) == 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 设置指定行列位置的位值
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="row">行索引(0 到 7)</param>
|
||||||
|
/// <param name="col">列索引(0 到 9)</param>
|
||||||
|
/// <param name="value">设置的值(true = 设置为已揭示,false = 设置为未揭示)</param>
|
||||||
|
public void SetBit(int row, int col, bool value)
|
||||||
|
{
|
||||||
|
// 确保行列值在有效范围内
|
||||||
|
if (row < 0 || row >= 8 || col < 0 || col >= 10)
|
||||||
|
{
|
||||||
|
throw new ArgumentOutOfRangeException();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 计算对应位置的偏移
|
||||||
|
int offset = row * 10 + col;
|
||||||
|
int byteOffset = offset / 8;
|
||||||
|
int bitOffset = offset % 8;
|
||||||
|
|
||||||
|
// 设置对应位置的位值
|
||||||
|
if (value)
|
||||||
|
{
|
||||||
|
// 设置为1
|
||||||
|
zones[byteOffset] |= (byte)(1 << bitOffset);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// 设置为0
|
||||||
|
zones[byteOffset] &= (byte)~(1 << bitOffset);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 读取数据
|
||||||
|
public static RevealedRegions Read(BinaryReader reader)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
byte[] data = reader.ReadBytes(10);
|
||||||
|
if (data.Length != 10)
|
||||||
|
throw new InvalidDataException("Failed to read 10 bytes for zones data.");
|
||||||
|
|
||||||
|
return new RevealedRegions(data);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
throw new PacketError("RevealedRegions", "zone_bits", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 写入数据
|
||||||
|
public void Write(BinaryWriter writer)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
writer.Write(zones);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
throw new PacketError("RevealedRegions", "zone_bits", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 索引操作,获取指定行的 10 位数据
|
||||||
|
public BitSlice GetRow(int index)
|
||||||
|
{
|
||||||
|
if (index < 0 || index >= 8)
|
||||||
|
throw new ArgumentOutOfRangeException();
|
||||||
|
|
||||||
|
byte[] row = new byte[10];
|
||||||
|
Array.Copy(zones, index * 10, row, 0, 10);
|
||||||
|
return new BitSlice(row);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 用于调试输出
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
sb.Append("[");
|
||||||
|
for (int i = 0; i < 8; i++)
|
||||||
|
{
|
||||||
|
sb.Append(GetRow(i).ToString());
|
||||||
|
if (i < 7) sb.Append(", ");
|
||||||
|
}
|
||||||
|
sb.Append("]");
|
||||||
|
return sb.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper class to represent a slice of bits
|
||||||
|
public class BitSlice
|
||||||
|
{
|
||||||
|
private byte[] data;
|
||||||
|
|
||||||
|
public BitSlice(byte[] data)
|
||||||
|
{
|
||||||
|
this.data = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
foreach (byte b in data)
|
||||||
|
{
|
||||||
|
sb.Append(Convert.ToString(b, 2).PadLeft(8, '0'));
|
||||||
|
}
|
||||||
|
return sb.ToString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Error handling class for packet-related errors
|
||||||
|
public class PacketError : Exception
|
||||||
|
{
|
||||||
|
public string PacketName { get; }
|
||||||
|
public string FieldName { get; }
|
||||||
|
public new Exception InnerException { get; }
|
||||||
|
|
||||||
|
public PacketError(string packetName, string fieldName, Exception innerException)
|
||||||
|
: base($"Error in packet '{packetName}', field '{fieldName}'", innerException)
|
||||||
|
{
|
||||||
|
PacketName = packetName;
|
||||||
|
FieldName = fieldName;
|
||||||
|
InnerException = innerException;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -9,7 +9,8 @@ namespace PSO2SERVER.Models
|
|||||||
Online,
|
Online,
|
||||||
Busy,
|
Busy,
|
||||||
Full,
|
Full,
|
||||||
Offline
|
Offline,
|
||||||
|
Undefined = 0xFFFF
|
||||||
}
|
}
|
||||||
|
|
||||||
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
|
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
|
||||||
|
@ -6,7 +6,7 @@ using System.Threading.Tasks;
|
|||||||
|
|
||||||
namespace PSO2SERVER.Models
|
namespace PSO2SERVER.Models
|
||||||
{
|
{
|
||||||
public class block
|
public class Block
|
||||||
{
|
{
|
||||||
public uint unk1 { get; set; }
|
public uint unk1 { get; set; }
|
||||||
public byte unk2 { get; set; }
|
public byte unk2 { get; set; }
|
||||||
|
@ -22,7 +22,7 @@ namespace PSO2SERVER.Protocol.Handlers
|
|||||||
for (uint i = 0; i < questCount; i++)
|
for (uint i = 0; i < questCount; i++)
|
||||||
{
|
{
|
||||||
var qobj = new ObjectHeader();
|
var qobj = new ObjectHeader();
|
||||||
qobj.ReadFromStream(reader);
|
qobj.ReadObjectHeaderFromStream(reader);
|
||||||
Quests.Add(qobj);
|
Quests.Add(qobj);
|
||||||
|
|
||||||
// 打印每个任务的 ID 和类型,方便调试
|
// 打印每个任务的 ID 和类型,方便调试
|
||||||
|
@ -204,20 +204,20 @@ namespace PSO2SERVER.Protocol.Handlers
|
|||||||
}
|
}
|
||||||
|
|
||||||
//TODO 方便GM测试
|
//TODO 方便GM测试
|
||||||
if (Password != user.Password)
|
//if (Password != user.Password)
|
||||||
{
|
//{
|
||||||
if (Password == "")
|
// if (Password == "")
|
||||||
{
|
// {
|
||||||
error = "密碼為空 #2.";
|
// error = "密碼為空 #2.";
|
||||||
user = null;
|
// user = null;
|
||||||
}
|
// }
|
||||||
else
|
// else
|
||||||
if (!BCrypt.Net.BCrypt.Verify(Password, user.Password))
|
// if (!BCrypt.Net.BCrypt.Verify(Password, user.Password))
|
||||||
{
|
// {
|
||||||
error = "密碼錯誤.";
|
// error = "密碼錯誤.";
|
||||||
user = null;
|
// user = null;
|
||||||
}
|
// }
|
||||||
}
|
//}
|
||||||
}
|
}
|
||||||
|
|
||||||
context.SendPacket(new LoginDataPacket("夢幻之星2", error, (user == null) ? (uint)0 : (uint)user.AccountId));
|
context.SendPacket(new LoginDataPacket("夢幻之星2", error, (user == null) ? (uint)0 : (uint)user.AccountId));
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using PSO2SERVER.Models;
|
using PSO2SERVER.Models;
|
||||||
using PSO2SERVER.Protocol.Packets;
|
using PSO2SERVER.Protocol.Packets;
|
||||||
|
|
||||||
@ -7,10 +8,38 @@ namespace PSO2SERVER.Protocol.Handlers
|
|||||||
[PacketHandlerAttr(0x11, 0x14)]
|
[PacketHandlerAttr(0x11, 0x14)]
|
||||||
public class BlockLogin : PacketHandler
|
public class BlockLogin : PacketHandler
|
||||||
{
|
{
|
||||||
|
// Player ID
|
||||||
|
public ulong PlayerId { get; set; }
|
||||||
|
|
||||||
|
// Unknown fields
|
||||||
|
public byte Unk1 { get; set; }
|
||||||
|
public byte Unk2 { get; set; }
|
||||||
|
public ushort Unk3 { get; set; }
|
||||||
|
public uint Unk4 { get; set; }
|
||||||
|
public uint Unk5 { get; set; }
|
||||||
|
|
||||||
|
// Version ID (32 bytes)
|
||||||
|
public byte[] VerId { get; set; } = new byte[0x20];
|
||||||
|
|
||||||
|
// Clients' active network interfaces
|
||||||
|
public List<NetInterface> Interfaces { get; set; }
|
||||||
|
|
||||||
|
// Login challenge
|
||||||
|
public uint Challenge { get; set; }
|
||||||
|
|
||||||
|
// Unknown data, fixed length (196 bytes)
|
||||||
|
public byte[] Unk6 { get; set; } = new byte[0xC4];
|
||||||
|
|
||||||
|
// Unknown field, 16 bytes
|
||||||
|
public byte[] Unk7 { get; set; } = new byte[0x10];
|
||||||
|
|
||||||
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
|
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
|
||||||
{
|
{
|
||||||
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
|
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
|
||||||
Logger.WriteHex(info, data);
|
Logger.WriteHex(info, data);
|
||||||
|
|
||||||
|
var reader = new PacketReader(data, position, size);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,29 @@
|
|||||||
|
using System;
|
||||||
|
using PSO2SERVER.Models;
|
||||||
|
using PSO2SERVER.Protocol.Packets;
|
||||||
|
|
||||||
|
namespace PSO2SERVER.Protocol.Handlers
|
||||||
|
{
|
||||||
|
[PacketHandlerAttr(0x1F, 0x01)]
|
||||||
|
public class TakenOrdersRequest : PacketHandler
|
||||||
|
{
|
||||||
|
public struct TakenOrdersRequestPacket
|
||||||
|
{
|
||||||
|
public uint unk1;
|
||||||
|
public uint unk2;
|
||||||
|
public uint unk3;
|
||||||
|
public uint unk4;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
|
||||||
|
{
|
||||||
|
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
|
||||||
|
Logger.WriteHex(info, data);
|
||||||
|
|
||||||
|
var reader = new PacketReader(data, position, size);
|
||||||
|
var packet = reader.ReadStruct<TakenOrdersRequestPacket>();
|
||||||
|
|
||||||
|
Logger.Write($"unk1 = {packet.unk1},unk2 = {packet.unk2},unk3 = {packet.unk3},unk4 = {packet.unk4}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,41 @@
|
|||||||
|
using System;
|
||||||
|
using PSO2SERVER.Models;
|
||||||
|
using PSO2SERVER.Protocol.Packets;
|
||||||
|
|
||||||
|
namespace PSO2SERVER.Protocol.Handlers
|
||||||
|
{
|
||||||
|
[PacketHandlerAttr(0x1F, 0x02)]
|
||||||
|
public class OrderListRequest : PacketHandler
|
||||||
|
{
|
||||||
|
public struct OrderListRequestPacket
|
||||||
|
{
|
||||||
|
public uint unk1;
|
||||||
|
public string source;
|
||||||
|
public uint unk3;
|
||||||
|
public uint unk4;
|
||||||
|
public uint unk5;
|
||||||
|
public uint unk6;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
|
||||||
|
{
|
||||||
|
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
|
||||||
|
Logger.WriteHex(info, data);
|
||||||
|
|
||||||
|
var reader = new PacketReader(data, position, size);
|
||||||
|
var packet = new OrderListRequestPacket
|
||||||
|
{
|
||||||
|
unk1 = reader.ReadUInt32(),
|
||||||
|
source = reader.ReadAscii(0x70B2, 0x9E),
|
||||||
|
unk3 = reader.ReadUInt32(),
|
||||||
|
unk4 = reader.ReadUInt32(),
|
||||||
|
unk5 = reader.ReadUInt32(),
|
||||||
|
unk6 = reader.ReadUInt32(),
|
||||||
|
};
|
||||||
|
|
||||||
|
Logger.Write($"unk1 = {packet.unk1},source = {packet.source},unk3 = {packet.unk3},unk4 = {packet.unk4},unk5 = {packet.unk5},unk6 = {packet.unk6}");
|
||||||
|
|
||||||
|
context.SendPacket(new OrderListPacket());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -7,10 +7,20 @@ namespace PSO2SERVER.Protocol.Handlers
|
|||||||
[PacketHandlerAttr(0x1F, 0x09)]
|
[PacketHandlerAttr(0x1F, 0x09)]
|
||||||
public class _1F_09_UNK : PacketHandler
|
public class _1F_09_UNK : PacketHandler
|
||||||
{
|
{
|
||||||
|
public uint unk1 { get; set; }
|
||||||
|
public string source { get; set; }
|
||||||
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
|
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
|
||||||
{
|
{
|
||||||
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
|
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
|
||||||
Logger.WriteHex(info, data);
|
Logger.WriteHex(info, data);
|
||||||
|
|
||||||
|
var reader = new PacketReader(data, position, size);
|
||||||
|
//unk1 = reader.ReadUInt32();
|
||||||
|
//source = reader.ReadAscii(0x1875, 0x1F);
|
||||||
|
|
||||||
|
//// 打印日志
|
||||||
|
//Logger.Write($"unk1: {unk1}");
|
||||||
|
//Logger.Write($"source: {source}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
27
Server/Protocol/Handlers/1F-ClientOrderHandler/1F-0F-UNK.cs
Normal file
27
Server/Protocol/Handlers/1F-ClientOrderHandler/1F-0F-UNK.cs
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
using System;
|
||||||
|
using PSO2SERVER.Models;
|
||||||
|
using PSO2SERVER.Protocol.Packets;
|
||||||
|
|
||||||
|
namespace PSO2SERVER.Protocol.Handlers
|
||||||
|
{
|
||||||
|
[PacketHandlerAttr(0x1F, 0x0F)]
|
||||||
|
public class _1F_0F_UNK : PacketHandler
|
||||||
|
{
|
||||||
|
public struct Unk1F0FPacket
|
||||||
|
{
|
||||||
|
public uint unk1;
|
||||||
|
public uint unk2;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void HandlePacket(Client context, byte flags, byte[] data, uint position, uint size)
|
||||||
|
{
|
||||||
|
var info = string.Format("[<--] 接收到的数据 (hex): {0} 字节", data.Length);
|
||||||
|
Logger.WriteHex(info, data);
|
||||||
|
|
||||||
|
var reader = new PacketReader(data, position, size);
|
||||||
|
var packet = reader.ReadStruct<Unk1F0FPacket>();
|
||||||
|
|
||||||
|
Logger.Write($"unk1 = {packet.unk1}, unk2 = {packet.unk2}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,5 @@
|
|||||||
using PSO2SERVER.Models;
|
using PSO2SERVER.Models;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
namespace PSO2SERVER.Protocol
|
namespace PSO2SERVER.Protocol
|
||||||
{
|
{
|
||||||
@ -6,5 +7,26 @@ namespace PSO2SERVER.Protocol
|
|||||||
{
|
{
|
||||||
public abstract byte[] Build();
|
public abstract byte[] Build();
|
||||||
public abstract PacketHeader GetHeader();
|
public abstract PacketHeader GetHeader();
|
||||||
|
|
||||||
|
// 新增:获取整体数据包字节表示
|
||||||
|
public byte[] GetPacketBytes()
|
||||||
|
{
|
||||||
|
var header = GetHeader(); // 获取包头
|
||||||
|
var packetBody = Build(); // 获取包体
|
||||||
|
|
||||||
|
// 如果包头的大小为0,则重新计算
|
||||||
|
if (header.Size == 0)
|
||||||
|
{
|
||||||
|
// 包头(包括固定的前四个字节)的大小加上包体的大小
|
||||||
|
header.Size = (uint)(header.ToBytes().Length + packetBody.Length);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将包头与包体合并
|
||||||
|
var totalPacket = new List<byte>();
|
||||||
|
totalPacket.AddRange(header.ToBytes()); // 包头转换为字节并加入
|
||||||
|
totalPacket.AddRange(packetBody); // 包体加入
|
||||||
|
|
||||||
|
return totalPacket.ToArray(); // 返回整个数据包字节数组
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -37,27 +37,6 @@ namespace PSO2SERVER.Protocol
|
|||||||
return (ReadUInt32() ^ xor) - sub;
|
return (ReadUInt32() ^ xor) - sub;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 示例方法:读取 ASCII 字符串
|
|
||||||
public string ReadAsciiForPosition(int start, int length)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (_position + length > _size)
|
|
||||||
{
|
|
||||||
throw new ArgumentOutOfRangeException(nameof(start), $"Reading beyond data boundary in {nameof(ReadAsciiForPosition)}");
|
|
||||||
}
|
|
||||||
|
|
||||||
var result = Encoding.ASCII.GetString(_data, _position, length);
|
|
||||||
_position += length;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Logger.WriteException(nameof(ReadAsciiForPosition), ex);
|
|
||||||
throw;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public string ReadAscii(uint xor, uint sub)
|
public string ReadAscii(uint xor, uint sub)
|
||||||
{
|
{
|
||||||
var magic = ReadMagic(xor, sub);
|
var magic = ReadMagic(xor, sub);
|
||||||
|
@ -174,6 +174,24 @@ namespace PSO2SERVER.Protocol
|
|||||||
Write(strArr);
|
Write(strArr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 新增的 WriteStructArray 方法
|
||||||
|
public unsafe void WriteStructArray<T>(T[] structures) where T : struct
|
||||||
|
{
|
||||||
|
int totalSize = Marshal.SizeOf(typeof(T)) * structures.Length;
|
||||||
|
var strArr = new byte[totalSize];
|
||||||
|
|
||||||
|
fixed (byte* ptr = strArr)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < structures.Length; i++)
|
||||||
|
{
|
||||||
|
byte* currentPtr = ptr + (i * Marshal.SizeOf(typeof(T)));
|
||||||
|
Marshal.StructureToPtr(structures[i], (IntPtr)currentPtr, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Write(strArr);
|
||||||
|
}
|
||||||
|
|
||||||
public byte[] ToArray()
|
public byte[] ToArray()
|
||||||
{
|
{
|
||||||
var ms = (MemoryStream) BaseStream;
|
var ms = (MemoryStream) BaseStream;
|
||||||
|
@ -8,32 +8,21 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
{
|
{
|
||||||
public class MinimapRevealPacket : Packet
|
public class MinimapRevealPacket : Packet
|
||||||
{
|
{
|
||||||
private ObjectHeader unk1 { get; set; }
|
/// World object where revealing was done.
|
||||||
|
private ObjectHeader world { get; set; }
|
||||||
|
/// Receivers party object (?).
|
||||||
private ObjectHeader party { get; set; }
|
private ObjectHeader party { get; set; }
|
||||||
private uint zone_id { get; set; }
|
private uint zone_id { get; set; }
|
||||||
private byte[] unk2 { get; set; } = new byte[10];
|
/// Bitset of revealed regions.
|
||||||
|
private RevealedRegions revealed_zones { get; set; } = new RevealedRegions();
|
||||||
|
|
||||||
// 构造函数,允许在创建时初始化字段
|
// 构造函数,允许在创建时初始化字段
|
||||||
public MinimapRevealPacket(ObjectHeader unk1, ObjectHeader party, uint zone_id, byte[] unk2)
|
public MinimapRevealPacket(ObjectHeader world, ObjectHeader party, uint zone_id, byte[] revealed_zones)
|
||||||
{
|
{
|
||||||
this.unk1 = unk1;
|
this.world = world;
|
||||||
this.party = party;
|
this.party = party;
|
||||||
this.zone_id = zone_id;
|
this.zone_id = zone_id;
|
||||||
|
this.revealed_zones = new RevealedRegions(revealed_zones);
|
||||||
// 如果传入的 unk2 长度小于 10,则填充剩余部分为 0
|
|
||||||
if (unk2.Length <= 10)
|
|
||||||
{
|
|
||||||
this.unk2 = unk2.Concat(new byte[10 - unk2.Length]).ToArray();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// 如果传入的 unk2 长度大于 10,则截取前 10 字节
|
|
||||||
this.unk2 = unk2.Take(10).ToArray();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public MinimapRevealPacket()
|
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#region implemented abstract members of Packet
|
#region implemented abstract members of Packet
|
||||||
@ -41,11 +30,10 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
public override byte[] Build()
|
public override byte[] Build()
|
||||||
{
|
{
|
||||||
var pkt = new PacketWriter();
|
var pkt = new PacketWriter();
|
||||||
pkt.WriteStruct(unk1);
|
pkt.WriteStruct(world);
|
||||||
pkt.WriteStruct(party);
|
pkt.WriteStruct(party);
|
||||||
pkt.Write(zone_id);
|
pkt.Write(zone_id);
|
||||||
for (var i = 0; i < 10; i++)
|
revealed_zones.Write(pkt);
|
||||||
pkt.Write(unk2[i]);
|
|
||||||
return pkt.ToArray();
|
return pkt.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,16 +1,11 @@
|
|||||||
using System;
|
using PSO2SERVER.Models;
|
||||||
using System.Collections.Generic;
|
using System;
|
||||||
using System.IO;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Text;
|
|
||||||
using Mysqlx;
|
|
||||||
using PSO2SERVER.Models;
|
|
||||||
|
|
||||||
namespace PSO2SERVER.Protocol.Packets
|
namespace PSO2SERVER.Protocol.Packets
|
||||||
{
|
{
|
||||||
class LoginDataPacket : Packet
|
class LoginDataPacket : Packet
|
||||||
{
|
{
|
||||||
public enum LoginStatus : UInt32
|
public enum LoginStatus : uint
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Login was successful.
|
/// Login was successful.
|
||||||
@ -31,7 +26,7 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
public LoginStatus Status;
|
public LoginStatus Status;
|
||||||
public string Error;
|
public string Error;
|
||||||
public ObjectHeader Player;
|
public ObjectHeader Player;
|
||||||
public string BlockName;
|
public string BlockName = new string(' ', 0x20);
|
||||||
public float Unk1;
|
public float Unk1;
|
||||||
public uint Unk2;
|
public uint Unk2;
|
||||||
public uint LevelCap;
|
public uint LevelCap;
|
||||||
@ -45,235 +40,130 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
public uint Unk11;
|
public uint Unk11;
|
||||||
public float Unk12;
|
public float Unk12;
|
||||||
public uint Unk13;
|
public uint Unk13;
|
||||||
public float[] Unk14; // Length: 10
|
public float[] Unk14 = new float[10]; // Length: 10
|
||||||
public float[] Unk15; // Length: 21
|
public float[] Unk15 = new float[21]; // Length: 21
|
||||||
public float Unk16;
|
public float Unk16;
|
||||||
public float Unk17;
|
public float Unk17;
|
||||||
public float[] Unk18; // Length: 9
|
public float[] Unk18 = new float[9]; // Length: 9
|
||||||
public uint[] Unk19; // Length: 2
|
public uint[] Unk19 = new uint[2]; // Length: 2
|
||||||
public uint Unk20;
|
public uint Unk20;
|
||||||
public uint Unk21;
|
public uint Unk21;
|
||||||
public float[] Unk22; // Length: 3
|
public float[] Unk22 = new float[3]; // Length: 3
|
||||||
public uint Unk23;
|
public uint Unk23;
|
||||||
public float Unk24;
|
public float Unk24;
|
||||||
public float Unk25;
|
public float Unk25;
|
||||||
public uint Unk26;
|
public uint Unk26;
|
||||||
public byte[] Unk27; // Length: 12
|
public byte[] Unk27 = new byte[12]; // Length: 12
|
||||||
public string Unk28;
|
public string Unk28 = new string(' ', 0x20);
|
||||||
public uint Unk29;
|
public uint Unk29;
|
||||||
public string Unk30;
|
public string Unk30 = new string(' ', 0x20);
|
||||||
public uint Unk31;
|
public uint Unk31;
|
||||||
|
|
||||||
private string ReadFixedString(PacketReader reader, int length)
|
|
||||||
{
|
|
||||||
byte[] bytes = reader.ReadBytes(length);
|
|
||||||
return Encoding.UTF8.GetString(bytes).TrimEnd('\0');
|
|
||||||
}
|
|
||||||
public void ReadFromStream(PacketReader reader)
|
|
||||||
{
|
|
||||||
Status = (LoginStatus)reader.ReadInt32();
|
|
||||||
Error = ReadFixedString(reader, 32); // 0x20 = 32
|
|
||||||
Player.ReadFromStream(reader);
|
|
||||||
BlockName = ReadFixedString(reader, 32);
|
|
||||||
Unk1 = reader.ReadSingle();
|
|
||||||
Unk2 = reader.ReadUInt32();
|
|
||||||
LevelCap = reader.ReadUInt32();
|
|
||||||
LevelCap2 = reader.ReadUInt32();
|
|
||||||
Unk5 = reader.ReadUInt32();
|
|
||||||
Unk6 = reader.ReadSingle();
|
|
||||||
Unk7 = reader.ReadSingle();
|
|
||||||
Unk8 = reader.ReadUInt32();
|
|
||||||
Unk9 = reader.ReadSingle();
|
|
||||||
Unk10 = reader.ReadSingle();
|
|
||||||
Unk11 = reader.ReadUInt32();
|
|
||||||
Unk12 = reader.ReadSingle();
|
|
||||||
Unk13 = reader.ReadUInt32();
|
|
||||||
|
|
||||||
Unk14 = new float[10];
|
|
||||||
for (int i = 0; i < Unk14.Length; i++)
|
|
||||||
{
|
|
||||||
Unk14[i] = reader.ReadSingle();
|
|
||||||
}
|
|
||||||
|
|
||||||
Unk15 = new float[21];
|
|
||||||
for (int i = 0; i < Unk15.Length; i++)
|
|
||||||
{
|
|
||||||
Unk15[i] = reader.ReadSingle();
|
|
||||||
}
|
|
||||||
|
|
||||||
Unk16 = reader.ReadSingle();
|
|
||||||
Unk17 = reader.ReadSingle();
|
|
||||||
|
|
||||||
Unk18 = new float[9];
|
|
||||||
for (int i = 0; i < Unk18.Length; i++)
|
|
||||||
{
|
|
||||||
Unk18[i] = reader.ReadSingle();
|
|
||||||
}
|
|
||||||
|
|
||||||
Unk19 = new uint[2];
|
|
||||||
for (int i = 0; i < Unk19.Length; i++)
|
|
||||||
{
|
|
||||||
Unk19[i] = reader.ReadUInt32();
|
|
||||||
}
|
|
||||||
|
|
||||||
Unk20 = reader.ReadUInt32();
|
|
||||||
Unk21 = reader.ReadUInt32();
|
|
||||||
|
|
||||||
Unk22 = new float[3];
|
|
||||||
for (int i = 0; i < Unk22.Length; i++)
|
|
||||||
{
|
|
||||||
Unk22[i] = reader.ReadSingle();
|
|
||||||
}
|
|
||||||
|
|
||||||
Unk23 = reader.ReadUInt32();
|
|
||||||
Unk24 = reader.ReadSingle();
|
|
||||||
Unk25 = reader.ReadSingle();
|
|
||||||
Unk26 = reader.ReadUInt32();
|
|
||||||
Unk27 = reader.ReadBytes(12);
|
|
||||||
Unk28 = ReadFixedString(reader, 32);
|
|
||||||
Unk29 = reader.ReadUInt32();
|
|
||||||
Unk30 = ReadFixedString(reader, 32);
|
|
||||||
Unk31 = reader.ReadUInt32();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void WriteFixedString(PacketWriter writer, string str, int length)
|
|
||||||
{
|
|
||||||
byte[] bytes = new byte[length];
|
|
||||||
byte[] strBytes = Encoding.UTF8.GetBytes(str);
|
|
||||||
Array.Copy(strBytes, bytes, Math.Min(strBytes.Length, length));
|
|
||||||
writer.Write(bytes);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void WriteToStream(PacketWriter writer)
|
|
||||||
{
|
|
||||||
writer.Write((int)Status);
|
|
||||||
writer.WriteUtf16(Error, 0x8BA4, 0xB6);
|
|
||||||
|
|
||||||
if (Player.ID == 0)
|
|
||||||
{
|
|
||||||
for (var i = 0; i < 0xEC; i++)
|
|
||||||
writer.Write((byte)0);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
//WriteFixedString(writer, Error, 32);
|
|
||||||
Player.WriteToStream(writer);
|
|
||||||
WriteFixedString(writer, BlockName, 32);
|
|
||||||
writer.Write(Unk1);
|
|
||||||
writer.Write(Unk2);
|
|
||||||
writer.Write(LevelCap);
|
|
||||||
writer.Write(LevelCap2);
|
|
||||||
writer.Write(Unk5);
|
|
||||||
writer.Write(Unk6);
|
|
||||||
writer.Write(Unk7);
|
|
||||||
writer.Write(Unk8);
|
|
||||||
writer.Write(Unk9);
|
|
||||||
writer.Write(Unk10);
|
|
||||||
writer.Write(Unk11);
|
|
||||||
writer.Write(Unk12);
|
|
||||||
writer.Write(Unk13);
|
|
||||||
|
|
||||||
foreach (var val in Unk14)
|
|
||||||
{
|
|
||||||
writer.Write(val);
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (var val in Unk15)
|
|
||||||
{
|
|
||||||
writer.Write(val);
|
|
||||||
}
|
|
||||||
|
|
||||||
writer.Write(Unk16);
|
|
||||||
writer.Write(Unk17);
|
|
||||||
|
|
||||||
foreach (var val in Unk18)
|
|
||||||
{
|
|
||||||
writer.Write(val);
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (var val in Unk19)
|
|
||||||
{
|
|
||||||
writer.Write(val);
|
|
||||||
}
|
|
||||||
|
|
||||||
writer.Write(Unk20);
|
|
||||||
writer.Write(Unk21);
|
|
||||||
|
|
||||||
foreach (var val in Unk22)
|
|
||||||
{
|
|
||||||
writer.Write(val);
|
|
||||||
}
|
|
||||||
|
|
||||||
writer.Write(Unk23);
|
|
||||||
writer.Write(Unk24);
|
|
||||||
writer.Write(Unk25);
|
|
||||||
writer.Write(Unk26);
|
|
||||||
writer.Write(Unk27);
|
|
||||||
WriteFixedString(writer, Unk28, 32);
|
|
||||||
writer.Write(Unk29);
|
|
||||||
WriteFixedString(writer, Unk30, 32);
|
|
||||||
writer.Write(Unk31);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public LoginDataPacket(string blockName, string error, uint userid)
|
public LoginDataPacket(string blockName, string error, uint userid)
|
||||||
{
|
{
|
||||||
Status = (userid == 0) ? LoginStatus.Failure : LoginStatus.Success;
|
Status = (userid == 0) ? LoginStatus.Failure : LoginStatus.Success;
|
||||||
Error = error;
|
Error = error;
|
||||||
Player.ID = userid;
|
Player = new ObjectHeader(userid, ObjectType.Player);
|
||||||
Player.ObjectType = ObjectType.Player;
|
|
||||||
BlockName = blockName;
|
BlockName = blockName;
|
||||||
|
Unk1 = 0x42700000;
|
||||||
|
Unk2 = 7;
|
||||||
|
LevelCap = 0x0A;
|
||||||
|
LevelCap2 = 1;
|
||||||
|
Unk5 = 0x41200000;
|
||||||
|
Unk6 = 0x40A00000;
|
||||||
|
Unk7 = 11;
|
||||||
|
Unk8 = 0x3F800000;
|
||||||
|
Unk9 = 0x42960000;
|
||||||
|
Unk10 = 40;
|
||||||
|
Unk11 = 0x41200000;
|
||||||
|
Unk12 = 1;
|
||||||
|
Unk13 = 1120403456;
|
||||||
|
|
||||||
|
//WHAT
|
||||||
|
for (int i = 0; i < Unk14.Length; i++)
|
||||||
|
{
|
||||||
|
Unk14[i] = 1065353216;
|
||||||
|
}
|
||||||
|
|
||||||
|
//ARE
|
||||||
|
for (int i = 0; i < Unk15.Length; i++)
|
||||||
|
{
|
||||||
|
Unk15[i] = 1120403456;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override byte[] Build()
|
public override byte[] Build()
|
||||||
{
|
{
|
||||||
var resp = new PacketWriter();
|
var pkt = new PacketWriter();
|
||||||
resp.Write((uint)Status); // Status flag: 0=success, 1=error
|
pkt.Write((uint)Status);
|
||||||
resp.WriteUtf16(Error, 0x8BA4, 0xB6);
|
pkt.WriteUtf16(Error, 0x8BA4, 0xB6);
|
||||||
|
|
||||||
if (Player.ID == 0)
|
if (Player.ID == 0)
|
||||||
{
|
{
|
||||||
for (var i = 0; i < 0xEC; i++)
|
for (var i = 0; i < 0xEC; i++)
|
||||||
resp.Write((byte)0);
|
pkt.Write((byte)0);
|
||||||
return resp.ToArray();
|
|
||||||
}
|
}
|
||||||
|
else
|
||||||
// TODO: Explore this data! Some if it seems really important. (May contain level cap setting + more)
|
|
||||||
|
|
||||||
resp.WriteStruct(Player);
|
|
||||||
resp.WriteFixedLengthUtf16(BlockName, 0x20); // This is right
|
|
||||||
// Set things to "default" values; Dunno these purposes yet.
|
|
||||||
resp.Write(0x42700000); //0
|
|
||||||
resp.Write(7); //4
|
|
||||||
resp.Write(0xA); //8 - Level Cap!
|
|
||||||
resp.Write(1); //C
|
|
||||||
resp.Write(0x41200000); //10
|
|
||||||
resp.Write(0x40A00000); //14
|
|
||||||
resp.Write(11); //18
|
|
||||||
resp.Write(0x3F800000); //1C (1 as a float)
|
|
||||||
resp.Write(0x42960000); //20
|
|
||||||
resp.Write(40); //24
|
|
||||||
resp.Write(0x41200000); //28
|
|
||||||
resp.Write(1); //2C?
|
|
||||||
resp.Write(1120403456); //30
|
|
||||||
|
|
||||||
//WHAT
|
|
||||||
for (int i = 0; i < 10; i++)
|
|
||||||
{
|
{
|
||||||
resp.Write(1065353216);
|
Player.WriteObjectHeaderToStream(pkt);
|
||||||
}
|
pkt.WriteFixedLengthUtf16(BlockName, 0x20);
|
||||||
//ARE
|
pkt.Write(Unk1);
|
||||||
for (int i = 0; i < 21; i++)
|
pkt.Write(Unk2);
|
||||||
|
pkt.Write(LevelCap);
|
||||||
|
pkt.Write(LevelCap2);
|
||||||
|
pkt.Write(Unk5);
|
||||||
|
pkt.Write(Unk6);
|
||||||
|
pkt.Write(Unk7);
|
||||||
|
pkt.Write(Unk8);
|
||||||
|
pkt.Write(Unk9);
|
||||||
|
pkt.Write(Unk10);
|
||||||
|
pkt.Write(Unk11);
|
||||||
|
pkt.Write(Unk12);
|
||||||
|
pkt.Write(Unk13);
|
||||||
|
|
||||||
|
foreach (var val in Unk14)
|
||||||
{
|
{
|
||||||
resp.Write(1120403456);
|
pkt.Write(val);
|
||||||
}
|
}
|
||||||
//THESE?
|
|
||||||
resp.Write(0x91A2B); //B0
|
|
||||||
resp.Write(0x91A2B); //B4
|
|
||||||
|
|
||||||
resp.WriteBytes(0, 12);
|
foreach (var val in Unk15)
|
||||||
|
{
|
||||||
|
pkt.Write(val);
|
||||||
|
}
|
||||||
|
|
||||||
return resp.ToArray();
|
pkt.Write(Unk16);
|
||||||
|
pkt.Write(Unk17);
|
||||||
|
|
||||||
|
foreach (var val in Unk18)
|
||||||
|
{
|
||||||
|
pkt.Write(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var val in Unk19)
|
||||||
|
{
|
||||||
|
pkt.Write(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
pkt.Write(Unk20);
|
||||||
|
pkt.Write(Unk21);
|
||||||
|
|
||||||
|
foreach (var val in Unk22)
|
||||||
|
{
|
||||||
|
pkt.Write(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
pkt.Write(Unk23);
|
||||||
|
pkt.Write(Unk24);
|
||||||
|
pkt.Write(Unk25);
|
||||||
|
pkt.Write(Unk26);
|
||||||
|
pkt.Write(Unk27);
|
||||||
|
pkt.WriteFixedLengthUtf16(Unk28, 0x20);
|
||||||
|
pkt.Write(Unk29);
|
||||||
|
pkt.WriteFixedLengthUtf16(Unk30, 0x20);
|
||||||
|
pkt.Write(Unk31);
|
||||||
|
}
|
||||||
|
|
||||||
|
return pkt.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
public override PacketHeader GetHeader()
|
public override PacketHeader GetHeader()
|
||||||
|
@ -15,9 +15,11 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
public byte[] data { get; set; } = new byte[0x11A];
|
public byte[] data { get; set; } = new byte[0x11A];
|
||||||
|
|
||||||
|
|
||||||
public BlockBalancePacket()
|
public BlockBalancePacket(string blockname, ushort port)
|
||||||
{
|
{
|
||||||
|
this.blockname = blockname;
|
||||||
ip = ServerApp.BindAddress.GetAddressBytes();
|
ip = ServerApp.BindAddress.GetAddressBytes();
|
||||||
|
this.port = port;
|
||||||
}
|
}
|
||||||
|
|
||||||
#region implemented abstract members of Packet
|
#region implemented abstract members of Packet
|
||||||
@ -28,8 +30,8 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
pkt.Write(unk1);
|
pkt.Write(unk1);
|
||||||
pkt.WriteFixedLengthUtf16(blockname, 0x20);
|
pkt.WriteFixedLengthUtf16(blockname, 0x20);
|
||||||
pkt.Write(ip);
|
pkt.Write(ip);
|
||||||
pkt.Write((UInt16)12205);
|
pkt.Write(port);
|
||||||
pkt.Write(new byte[0x90 - 0x6A]);
|
pkt.Write(new byte[0x11A]);
|
||||||
return pkt.ToArray();
|
return pkt.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3,14 +3,21 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
using static PSO2SERVER.Models.CharacterStruct;
|
||||||
|
|
||||||
namespace PSO2SERVER.Protocol.Packets
|
namespace PSO2SERVER.Protocol.Packets
|
||||||
{
|
{
|
||||||
public class ShipListPacket : Packet
|
public class ShipListPacket : Packet
|
||||||
{
|
{
|
||||||
|
public List<ShipEntry> shipEntries { get; set; } = new List<ShipEntry>();
|
||||||
|
public int timestamp { get; set; }
|
||||||
|
public uint unk { get; set; }
|
||||||
|
|
||||||
public ShipListPacket()
|
public ShipListPacket(List<ShipEntry> entries, int timestamp, uint unk)
|
||||||
{
|
{
|
||||||
|
shipEntries = entries;
|
||||||
|
this.timestamp = timestamp;
|
||||||
|
this.unk = unk;
|
||||||
}
|
}
|
||||||
|
|
||||||
#region implemented abstract members of Packet
|
#region implemented abstract members of Packet
|
||||||
@ -18,12 +25,19 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
public override byte[] Build()
|
public override byte[] Build()
|
||||||
{
|
{
|
||||||
var pkt = new PacketWriter();
|
var pkt = new PacketWriter();
|
||||||
|
pkt.WriteMagic((uint)shipEntries.Count, 0xE418, 0x51);
|
||||||
|
|
||||||
|
foreach (var entry in shipEntries)
|
||||||
|
pkt.WriteStruct(entry);
|
||||||
|
|
||||||
|
pkt.Write(timestamp);
|
||||||
|
pkt.Write(unk);
|
||||||
return pkt.ToArray();
|
return pkt.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
public override PacketHeader GetHeader()
|
public override PacketHeader GetHeader()
|
||||||
{
|
{
|
||||||
return new PacketHeader(0x11, 0x3D, PacketFlags.None);
|
return new PacketHeader(0x11, 0x3D, PacketFlags.PACKED);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
@ -3,11 +3,16 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
using static PSO2SERVER.Models.Orders;
|
||||||
|
|
||||||
namespace PSO2SERVER.Protocol.Packets
|
namespace PSO2SERVER.Protocol.Packets
|
||||||
{
|
{
|
||||||
public class OrderListPacket : Packet
|
public class OrderListPacket : Packet
|
||||||
{
|
{
|
||||||
|
public ObjectHeader user { get; set; } = new ObjectHeader();
|
||||||
|
public ClientOrder[] orders { get; set; } = new ClientOrder[100];
|
||||||
|
public uint unk1 { get; set; }
|
||||||
|
public uint unk2 { get; set; }
|
||||||
|
|
||||||
public OrderListPacket()
|
public OrderListPacket()
|
||||||
{
|
{
|
||||||
@ -18,6 +23,10 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
public override byte[] Build()
|
public override byte[] Build()
|
||||||
{
|
{
|
||||||
var pkt = new PacketWriter();
|
var pkt = new PacketWriter();
|
||||||
|
pkt.WriteObjectHeader(user);
|
||||||
|
pkt.WriteStructArray(orders);
|
||||||
|
pkt.Write(unk1);
|
||||||
|
pkt.Write(unk2);
|
||||||
return pkt.ToArray();
|
return pkt.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3,11 +3,18 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
using static PSO2SERVER.Models.Orders;
|
||||||
|
|
||||||
namespace PSO2SERVER.Protocol.Packets
|
namespace PSO2SERVER.Protocol.Packets
|
||||||
{
|
{
|
||||||
public class TakenOrdersPacket : Packet
|
public class TakenOrdersPacket : Packet
|
||||||
{
|
{
|
||||||
|
public ObjectHeader user { get; set; } = new ObjectHeader();
|
||||||
|
public ClientOrder[] orders { get; set; } = new ClientOrder[50];
|
||||||
|
public OrderStatus[] statues { get; set; } = new OrderStatus[50];
|
||||||
|
public uint unk1 { get; set; }
|
||||||
|
public uint unk2 { get; set; }
|
||||||
|
public uint unk3 { get; set; }
|
||||||
|
|
||||||
public TakenOrdersPacket()
|
public TakenOrdersPacket()
|
||||||
{
|
{
|
||||||
@ -18,6 +25,12 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
public override byte[] Build()
|
public override byte[] Build()
|
||||||
{
|
{
|
||||||
var pkt = new PacketWriter();
|
var pkt = new PacketWriter();
|
||||||
|
pkt.WriteObjectHeader(user);
|
||||||
|
pkt.WriteStructArray(orders);
|
||||||
|
pkt.WriteStructArray(statues);
|
||||||
|
pkt.Write(unk1);
|
||||||
|
pkt.Write(unk2);
|
||||||
|
pkt.Write(unk3);
|
||||||
return pkt.ToArray();
|
return pkt.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -8,6 +8,15 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
{
|
{
|
||||||
public class Unk230EPacket : Packet
|
public class Unk230EPacket : Packet
|
||||||
{
|
{
|
||||||
|
public struct Unk230EThing
|
||||||
|
{
|
||||||
|
public ushort unk1;
|
||||||
|
public ushort unk2;
|
||||||
|
public uint unk3;
|
||||||
|
public ObjectHeader unk4;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Unk230EThing> unk = new List<Unk230EThing>();
|
||||||
|
|
||||||
public Unk230EPacket()
|
public Unk230EPacket()
|
||||||
{
|
{
|
||||||
@ -18,6 +27,11 @@ namespace PSO2SERVER.Protocol.Packets
|
|||||||
public override byte[] Build()
|
public override byte[] Build()
|
||||||
{
|
{
|
||||||
var pkt = new PacketWriter();
|
var pkt = new PacketWriter();
|
||||||
|
pkt.WriteMagic((uint)unk.Count, 0xAC40, 0x99);
|
||||||
|
|
||||||
|
foreach (var entry in unk)
|
||||||
|
pkt.WriteStruct(entry);
|
||||||
|
|
||||||
return pkt.ToArray();
|
return pkt.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,6 +9,7 @@ using System.Threading;
|
|||||||
using PSO2SERVER.Models;
|
using PSO2SERVER.Models;
|
||||||
using PSO2SERVER.Protocol;
|
using PSO2SERVER.Protocol;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using PSO2SERVER.Protocol.Packets;
|
||||||
|
|
||||||
namespace PSO2SERVER
|
namespace PSO2SERVER
|
||||||
{
|
{
|
||||||
@ -78,7 +79,6 @@ namespace PSO2SERVER
|
|||||||
|
|
||||||
private async Task DoShipListAsync(Socket socket)
|
private async Task DoShipListAsync(Socket socket)
|
||||||
{
|
{
|
||||||
var writer = new PacketWriter();
|
|
||||||
var entries = new List<ShipEntry>();
|
var entries = new List<ShipEntry>();
|
||||||
|
|
||||||
for (var i = 1; i <= 10; i++)
|
for (var i = 1; i <= 10; i++)
|
||||||
@ -95,25 +95,10 @@ namespace PSO2SERVER
|
|||||||
entries.Add(entry);
|
entries.Add(entry);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Assuming header size: 8 bytes + (size of ShipEntry * number of entries) + 12 bytes
|
var shiplistpacket = new ShipListPacket(entries, (Int32)(DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1))).TotalSeconds, 1).GetPacketBytes();
|
||||||
int headerSize = 8;
|
|
||||||
int shipEntrySize = Marshal.SizeOf(typeof(ShipEntry));
|
|
||||||
int totalSize = headerSize + (shipEntrySize * entries.Count) + 12;
|
|
||||||
PacketHeader header = new PacketHeader(totalSize, 0x11, 0x3D, 0x04, 0x00);
|
|
||||||
|
|
||||||
writer.WriteStruct(header);
|
|
||||||
writer.WriteMagic((uint)entries.Count, 0xE418, 81);
|
|
||||||
|
|
||||||
foreach (var entry in entries)
|
|
||||||
writer.WriteStruct(entry);
|
|
||||||
|
|
||||||
writer.Write((Int32)(DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1))).TotalSeconds);
|
|
||||||
writer.Write(1);
|
|
||||||
|
|
||||||
var buffer = writer.ToArray();
|
|
||||||
|
|
||||||
await Task.Factory.FromAsync(
|
await Task.Factory.FromAsync(
|
||||||
(cb, state) => socket.BeginSend(buffer, 0, buffer.Length, SocketFlags.None, cb, state),
|
(cb, state) => socket.BeginSend(shiplistpacket, 0, shiplistpacket.Length, SocketFlags.None, cb, state),
|
||||||
socket.EndSend,
|
socket.EndSend,
|
||||||
null);
|
null);
|
||||||
socket.Close();
|
socket.Close();
|
||||||
@ -121,17 +106,10 @@ namespace PSO2SERVER
|
|||||||
|
|
||||||
private async Task DoBlockBalanceAsync(Socket socket)
|
private async Task DoBlockBalanceAsync(Socket socket)
|
||||||
{
|
{
|
||||||
var writer = new PacketWriter();
|
var balancepacket = new BlockBalancePacket("test", 12205).GetPacketBytes();
|
||||||
writer.WriteStruct(new PacketHeader(0x90, 0x11, 0x2C, 0x0, 0x0));
|
|
||||||
writer.Write(new byte[0x20]);
|
|
||||||
writer.Write(new byte[0x40]);
|
|
||||||
writer.Write(ServerApp.BindAddress.GetAddressBytes());
|
|
||||||
writer.Write((UInt16)12205);
|
|
||||||
writer.Write(new byte[0x11A]);
|
|
||||||
|
|
||||||
var buffer = writer.ToArray();
|
|
||||||
await Task.Factory.FromAsync(
|
await Task.Factory.FromAsync(
|
||||||
(cb, state) => socket.BeginSend(buffer, 0, buffer.Length, SocketFlags.None, cb, state),
|
(cb, state) => socket.BeginSend(balancepacket, 0, balancepacket.Length, SocketFlags.None, cb, state),
|
||||||
socket.EndSend,
|
socket.EndSend,
|
||||||
null);
|
null);
|
||||||
socket.Close();
|
socket.Close();
|
||||||
|
@ -165,14 +165,17 @@
|
|||||||
<Compile Include="ConsoleSystem.cs" />
|
<Compile Include="ConsoleSystem.cs" />
|
||||||
<Compile Include="Crypto\KeyLoader.cs" />
|
<Compile Include="Crypto\KeyLoader.cs" />
|
||||||
<Compile Include="Logger.cs" />
|
<Compile Include="Logger.cs" />
|
||||||
<Compile Include="Models\block.cs" />
|
<Compile Include="Models\Block.cs" />
|
||||||
<Compile Include="Models\Flags.cs" />
|
<Compile Include="Models\Flags.cs" />
|
||||||
<Compile Include="Models\Mission.cs" />
|
<Compile Include="Models\Mission.cs" />
|
||||||
|
<Compile Include="Models\NetInterface.cs" />
|
||||||
|
<Compile Include="Models\Orders.cs" />
|
||||||
<Compile Include="Models\PSO2Item.cs" />
|
<Compile Include="Models\PSO2Item.cs" />
|
||||||
<Compile Include="Models\PSOData.cs" />
|
<Compile Include="Models\PSOData.cs" />
|
||||||
<Compile Include="Models\PSOObject.cs" />
|
<Compile Include="Models\PSOObject.cs" />
|
||||||
<Compile Include="Models\PSOPalette.cs" />
|
<Compile Include="Models\PSOPalette.cs" />
|
||||||
<Compile Include="Models\Quest.cs" />
|
<Compile Include="Models\Quest.cs" />
|
||||||
|
<Compile Include="Models\RevealedRegions.cs" />
|
||||||
<Compile Include="Network\PortChecker.cs" />
|
<Compile Include="Network\PortChecker.cs" />
|
||||||
<Compile Include="Object\ObjectManager.cs" />
|
<Compile Include="Object\ObjectManager.cs" />
|
||||||
<Compile Include="Protocol\Handlers\0B-QuestHandler\0B-20-AcceptQuest.cs" />
|
<Compile Include="Protocol\Handlers\0B-QuestHandler\0B-20-AcceptQuest.cs" />
|
||||||
@ -186,7 +189,10 @@
|
|||||||
<Compile Include="Protocol\Handlers\1C-CharacterInfoHandler\1C-46-UNK.cs" />
|
<Compile Include="Protocol\Handlers\1C-CharacterInfoHandler\1C-46-UNK.cs" />
|
||||||
<Compile Include="Protocol\Handlers\1C-CharacterInfoHandler\1C-0E-UNK.cs" />
|
<Compile Include="Protocol\Handlers\1C-CharacterInfoHandler\1C-0E-UNK.cs" />
|
||||||
<Compile Include="Protocol\Handlers\1C-CharacterInfoHandler\1C-1E-CharacterInfoRequest.cs" />
|
<Compile Include="Protocol\Handlers\1C-CharacterInfoHandler\1C-1E-CharacterInfoRequest.cs" />
|
||||||
<Compile Include="Protocol\Handlers\1F-UNKHandler\1F-09-UNK1F09.cs" />
|
<Compile Include="Protocol\Handlers\1F-ClientOrderHandler\1F-02-OrderListRequest.cs" />
|
||||||
|
<Compile Include="Protocol\Handlers\1F-ClientOrderHandler\1F-01-TakenOrdersRequest.cs" />
|
||||||
|
<Compile Include="Protocol\Handlers\1F-ClientOrderHandler\1F-0F-UNK.cs" />
|
||||||
|
<Compile Include="Protocol\Handlers\1F-ClientOrderHandler\1F-09-UNK1F09.cs" />
|
||||||
<Compile Include="Protocol\Handlers\23-FlagHandler\23-0B-SkitItemAddRequest.cs" />
|
<Compile Include="Protocol\Handlers\23-FlagHandler\23-0B-SkitItemAddRequest.cs" />
|
||||||
<Compile Include="Protocol\Handlers\03-ServerHandler\03-03-InitialLoad.cs" />
|
<Compile Include="Protocol\Handlers\03-ServerHandler\03-03-InitialLoad.cs" />
|
||||||
<Compile Include="Protocol\Handlers\03-ServerHandler\03-0C-ServerPong.cs" />
|
<Compile Include="Protocol\Handlers\03-ServerHandler\03-0C-ServerPong.cs" />
|
||||||
|
Loading…
Reference in New Issue
Block a user