190 lines
5.7 KiB
C#
190 lines
5.7 KiB
C#
using System;
|
||
using System.Collections.Generic;
|
||
using System.IO;
|
||
using System.Runtime.InteropServices;
|
||
using System.Text;
|
||
using PSO2SERVER.Models;
|
||
|
||
namespace PSO2SERVER.Protocol
|
||
{
|
||
public class PacketWriter : BinaryWriter
|
||
{
|
||
public PacketWriter()
|
||
: base(new MemoryStream())
|
||
{
|
||
}
|
||
|
||
public PacketWriter(Stream s)
|
||
: base(s)
|
||
{
|
||
}
|
||
|
||
public void WriteMagic(uint magic, uint xor, uint sub)
|
||
{
|
||
var encoded = (magic + sub) ^ xor;
|
||
Write(encoded);
|
||
}
|
||
|
||
public void WriteAscii(string str, uint xor, uint sub)
|
||
{
|
||
if((str == null) || (str == "") || (str.Length == 0))
|
||
{
|
||
|
||
WriteMagic(0, xor, sub);
|
||
//if (str.Length == 0)
|
||
//{
|
||
//}
|
||
}
|
||
else
|
||
{
|
||
// Magic, followed by string, followed by null terminator,
|
||
// followed by padding characters if needed.
|
||
var charCount = (uint) str.Length;
|
||
var padding = 4 - (charCount & 3);
|
||
|
||
WriteMagic(charCount + 1, xor, sub);
|
||
Write(Encoding.ASCII.GetBytes(str));
|
||
for (var i = 0; i < padding; i++)
|
||
Write((byte) 0);
|
||
}
|
||
}
|
||
|
||
public void WriteAscii(AsciiString str, uint xor, uint sub)
|
||
{
|
||
if (str == null || str.Length == 0)
|
||
{
|
||
// 如果 AsciiString 为空或长度为 0,写入 magic,长度为 0
|
||
WriteMagic(0, xor, sub);
|
||
}
|
||
else
|
||
{
|
||
// Magic, followed by string, followed by null terminator,
|
||
// followed by padding characters if needed.
|
||
var charCount = (uint)str.Length;
|
||
var padding = (4 - (charCount + 1) % 4) % 4; // +1 用于计算 null 终结符的空间
|
||
|
||
// 写入 Magic 值(charCount + 1:包括 null 终结符)
|
||
WriteMagic(charCount + 1, xor, sub);
|
||
|
||
// 写入字符串的字节数据
|
||
Write(str.ToBytes());
|
||
|
||
// 写入 null 终结符
|
||
Write((byte)0);
|
||
|
||
// 填充 null 字节以保证 4 字节对齐
|
||
for (var i = 0; i < padding; i++)
|
||
{
|
||
Write((byte)0);
|
||
}
|
||
}
|
||
}
|
||
|
||
internal void WritePosition(PSOLocation location)
|
||
{
|
||
Write(Helper.FloatToHalfPrecision(location.RotX));
|
||
Write(Helper.FloatToHalfPrecision(location.RotY));
|
||
Write(Helper.FloatToHalfPrecision(location.RotZ));
|
||
Write(Helper.FloatToHalfPrecision(location.RotW));
|
||
Write(Helper.FloatToHalfPrecision(location.PosX));
|
||
Write(Helper.FloatToHalfPrecision(location.PosY));
|
||
Write(Helper.FloatToHalfPrecision(location.PosZ));
|
||
}
|
||
|
||
public void WriteUtf16(string str, uint xor, uint sub)
|
||
{
|
||
if ((str == null) || (str == "") || (str.Length == 0))
|
||
{
|
||
WriteMagic(0, xor, sub);
|
||
}
|
||
else
|
||
{
|
||
// Magic, followed by string, followed by null terminator,
|
||
// followed by a padding character if needed.
|
||
var charCount = (uint) str.Length + 1;
|
||
var padding = (charCount & 1);
|
||
|
||
WriteMagic(charCount, xor, sub);
|
||
Write(Encoding.GetEncoding("UTF-16").GetBytes(str));
|
||
Write((ushort) 0);
|
||
if (padding != 0)
|
||
Write((ushort) 0);
|
||
}
|
||
}
|
||
|
||
public void WriteFixedLengthASCII(string str, int charCount)
|
||
{
|
||
var writeAmount = Math.Min(str.Length, charCount);
|
||
var paddingAmount = charCount - writeAmount;
|
||
|
||
if (writeAmount > 0)
|
||
{
|
||
var chopped = writeAmount != str.Length ? str.Substring(0, writeAmount) : str;
|
||
|
||
Write(Encoding.GetEncoding("ASCII").GetBytes(chopped));
|
||
}
|
||
|
||
if (paddingAmount > 0)
|
||
{
|
||
for (var i = 0; i < paddingAmount; i++)
|
||
Write((byte) 0);
|
||
}
|
||
}
|
||
|
||
public void WriteFixedLengthUtf16(string str, int charCount)
|
||
{
|
||
var writeAmount = Math.Min(str.Length, charCount);
|
||
var paddingAmount = charCount - writeAmount;
|
||
|
||
if (writeAmount > 0)
|
||
{
|
||
var chopped = writeAmount != str.Length ? str.Substring(0, writeAmount) : str;
|
||
|
||
Write(Encoding.GetEncoding("UTF-16").GetBytes(chopped));
|
||
}
|
||
|
||
if (paddingAmount > 0)
|
||
{
|
||
for (var i = 0; i < paddingAmount; i++)
|
||
Write((ushort) 0);
|
||
}
|
||
}
|
||
|
||
public void WriteAccountHeader(uint AccountId)
|
||
{
|
||
Write(AccountId);
|
||
Write((uint) 0);
|
||
Write((ushort)ObjectType.Player);
|
||
Write((ushort) 0);
|
||
}
|
||
|
||
public unsafe void WriteStruct<T>(T structure) where T : struct
|
||
{
|
||
int size = Marshal.SizeOf(structure);
|
||
//Logger.Write($"Writing {size} bytes of structure: {structure}");
|
||
|
||
var strArr = new byte[size];
|
||
|
||
fixed (byte* ptr = strArr)
|
||
{
|
||
Marshal.StructureToPtr(structure, (IntPtr)ptr, false);
|
||
}
|
||
|
||
Write(strArr);
|
||
}
|
||
|
||
public byte[] ToArray()
|
||
{
|
||
var ms = (MemoryStream) BaseStream;
|
||
return ms.ToArray();
|
||
}
|
||
|
||
internal void WriteBytes(byte b, uint count)
|
||
{
|
||
for(int i = 0; i < count; i++)
|
||
{
|
||
Write(b);
|
||
}
|
||
}
|
||
}
|
||
} |