|
30 | 30 | // Ensure structs are packed to 1-byte alignment for network protocol compatibility |
31 | 31 | #pragma pack(push, 1) |
32 | 32 |
|
| 33 | +// Network packet field type definitions |
| 34 | +typedef UnsignedByte NetPacketFieldType; |
| 35 | + |
| 36 | +namespace NetPacketFieldTypes { |
| 37 | + constexpr const NetPacketFieldType CommandType = 'T'; // NetCommandType field |
| 38 | + constexpr const NetPacketFieldType Relay = 'R'; // Relay field |
| 39 | + constexpr const NetPacketFieldType PlayerId = 'P'; // Player ID field |
| 40 | + constexpr const NetPacketFieldType CommandId = 'C'; // Command ID field |
| 41 | + constexpr const NetPacketFieldType Frame = 'F'; // Frame field |
| 42 | + constexpr const NetPacketFieldType Data = 'D'; // Data payload field |
| 43 | +} |
| 44 | + |
33 | 45 | //////////////////////////////////////////////////////////////////////////////// |
34 | 46 | // Common packet field structures |
35 | 47 | //////////////////////////////////////////////////////////////////////////////// |
36 | 48 |
|
37 | 49 | // Command Type field: 'T' + UnsignedByte |
38 | 50 | struct NetPacketCommandTypeField { |
39 | | - char header; // 'T' |
| 51 | + const NetPacketFieldType type; // 'T' |
40 | 52 | UnsignedByte commandType; |
| 53 | + |
| 54 | + NetPacketCommandTypeField() : type(NetPacketFieldTypes::CommandType) {} |
41 | 55 | }; |
42 | 56 |
|
43 | 57 | // Relay field: 'R' + UnsignedByte |
44 | 58 | struct NetPacketRelayField { |
45 | | - char header; // 'R' |
| 59 | + const NetPacketFieldType type; // 'R' |
46 | 60 | UnsignedByte relay; |
| 61 | + |
| 62 | + NetPacketRelayField() : type(NetPacketFieldTypes::Relay) {} |
47 | 63 | }; |
48 | 64 |
|
49 | 65 | // Player ID field: 'P' + UnsignedByte |
50 | 66 | struct NetPacketPlayerIdField { |
51 | | - char header; // 'P' |
| 67 | + const NetPacketFieldType type; // 'P' |
52 | 68 | UnsignedByte playerId; |
| 69 | + |
| 70 | + NetPacketPlayerIdField() : type(NetPacketFieldTypes::PlayerId) {} |
53 | 71 | }; |
54 | 72 |
|
55 | 73 | // Frame field: 'F' + UnsignedInt |
56 | 74 | struct NetPacketFrameField { |
57 | | - char header; // 'F' |
| 75 | + const NetPacketFieldType type; // 'F' |
58 | 76 | UnsignedInt frame; |
| 77 | + |
| 78 | + NetPacketFrameField() : type(NetPacketFieldTypes::Frame) {} |
59 | 79 | }; |
60 | 80 |
|
61 | 81 | // Command ID field: 'C' + UnsignedShort |
62 | 82 | struct NetPacketCommandIdField { |
63 | | - char header; // 'C' |
| 83 | + const NetPacketFieldType type; // 'C' |
64 | 84 | UnsignedShort commandId; |
| 85 | + |
| 86 | + NetPacketCommandIdField() : type(NetPacketFieldTypes::CommandId) {} |
65 | 87 | }; |
66 | 88 |
|
67 | 89 | // Data field header: 'D' (followed by variable-length data) |
68 | 90 | struct NetPacketDataFieldHeader { |
69 | | - char header; // 'D' |
| 91 | + const NetPacketFieldType type; // 'D' |
| 92 | + |
| 93 | + NetPacketDataFieldHeader() : type(NetPacketFieldTypes::Data) {} |
70 | 94 | }; |
71 | 95 |
|
72 | 96 | //////////////////////////////////////////////////////////////////////////////// |
@@ -373,3 +397,85 @@ struct NetPacketFrameResendRequestCommand { |
373 | 397 | // Restore normal struct packing |
374 | 398 | #pragma pack(pop) |
375 | 399 |
|
| 400 | +//////////////////////////////////////////////////////////////////////////////// |
| 401 | +// Static Assert Tests - Verify struct sizes match original manual calculations |
| 402 | +// These tests ensure the refactor maintains exact compatibility |
| 403 | +//////////////////////////////////////////////////////////////////////////////// |
| 404 | + |
| 405 | +// Test function for Frame Resend Request Command |
| 406 | +constexpr UnsignedInt GetFrameResendRequestCommandSize() |
| 407 | +{ |
| 408 | + UnsignedInt msglen = 0; |
| 409 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::CommandType and command type |
| 410 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::PlayerId and player ID |
| 411 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedShort); // NetPacketFieldTypes::CommandId and command ID |
| 412 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::Relay and relay |
| 413 | + |
| 414 | + ++msglen; // NetPacketFieldTypes::Data |
| 415 | + msglen += sizeof(UnsignedInt); // frame to resend |
| 416 | + |
| 417 | + return msglen; |
| 418 | +} |
| 419 | + |
| 420 | +// Test function for ACK Command |
| 421 | +constexpr UnsignedInt GetAckCommandSize() |
| 422 | +{ |
| 423 | + UnsignedInt msglen = 0; |
| 424 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::CommandType and command type |
| 425 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::PlayerId and player ID |
| 426 | + ++msglen; // NetPacketFieldTypes::Data |
| 427 | + msglen += sizeof(UnsignedShort); // command ID being acknowledged |
| 428 | + msglen += sizeof(UnsignedByte); // original player ID |
| 429 | + |
| 430 | + return msglen; |
| 431 | +} |
| 432 | + |
| 433 | +// Test function for Frame Command |
| 434 | +constexpr UnsignedInt GetFrameCommandSize() |
| 435 | +{ |
| 436 | + UnsignedInt msglen = 0; |
| 437 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::CommandType and command type |
| 438 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::Relay and relay |
| 439 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedInt); // NetPacketFieldTypes::Frame and frame |
| 440 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::PlayerId and player ID |
| 441 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedShort); // NetPacketFieldTypes::CommandId and command ID |
| 442 | + ++msglen; // NetPacketFieldTypes::Data |
| 443 | + msglen += sizeof(UnsignedShort); // command count |
| 444 | + |
| 445 | + return msglen; |
| 446 | +} |
| 447 | + |
| 448 | +// Test function for Player Leave Command |
| 449 | +constexpr UnsignedInt GetPlayerLeaveCommandSize() |
| 450 | +{ |
| 451 | + UnsignedInt msglen = 0; |
| 452 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::CommandType and command type |
| 453 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::Relay and relay |
| 454 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedInt); // NetPacketFieldTypes::Frame and frame |
| 455 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::PlayerId and player ID |
| 456 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedShort); // NetPacketFieldTypes::CommandId and command ID |
| 457 | + ++msglen; // NetPacketFieldTypes::Data |
| 458 | + msglen += sizeof(UnsignedByte); // leaving player ID |
| 459 | + |
| 460 | + return msglen; |
| 461 | +} |
| 462 | + |
| 463 | +// Test function for Keep Alive Command |
| 464 | +constexpr UnsignedInt GetKeepAliveCommandSize() |
| 465 | +{ |
| 466 | + UnsignedInt msglen = 0; |
| 467 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::CommandType and command type |
| 468 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::Relay and relay |
| 469 | + msglen += sizeof(UnsignedByte) + sizeof(UnsignedByte); // NetPacketFieldTypes::PlayerId and player ID |
| 470 | + ++msglen; // NetPacketFieldTypes::Data |
| 471 | + |
| 472 | + return msglen; |
| 473 | +} |
| 474 | + |
| 475 | +// Static assertions to verify sizes match |
| 476 | +static_assert(GetFrameResendRequestCommandSize() == sizeof(NetPacketFrameResendRequestCommand), "FrameResendRequestCommand size mismatch"); |
| 477 | +static_assert(GetAckCommandSize() == sizeof(NetPacketAckCommand), "AckCommand size mismatch"); |
| 478 | +static_assert(GetFrameCommandSize() == sizeof(NetPacketFrameCommand), "FrameCommand size mismatch"); |
| 479 | +static_assert(GetPlayerLeaveCommandSize() == sizeof(NetPacketPlayerLeaveCommand), "PlayerLeaveCommand size mismatch"); |
| 480 | +static_assert(GetKeepAliveCommandSize() == sizeof(NetPacketKeepAliveCommand), "KeepAliveCommand size mismatch"); |
| 481 | + |
0 commit comments