#include #include #include #include namespace scwx { namespace wsr88d { namespace rpg { static const std::string logPrefix_ = "[scwx::wsr88d::rpg::unlinked_vector_packet] "; class UnlinkedVectorPacketImpl { public: explicit UnlinkedVectorPacketImpl() : packetCode_ {0}, lengthOfBlock_ {0}, valueOfVector_ {0}, beginI_ {}, beginJ_ {}, endI_ {}, endJ_ {} { } ~UnlinkedVectorPacketImpl() = default; uint16_t packetCode_; uint16_t lengthOfBlock_; uint16_t valueOfVector_; std::vector beginI_; std::vector beginJ_; std::vector endI_; std::vector endJ_; }; UnlinkedVectorPacket::UnlinkedVectorPacket() : p(std::make_unique()) { } UnlinkedVectorPacket::~UnlinkedVectorPacket() = default; UnlinkedVectorPacket::UnlinkedVectorPacket(UnlinkedVectorPacket&&) noexcept = default; UnlinkedVectorPacket& UnlinkedVectorPacket::operator=(UnlinkedVectorPacket&&) noexcept = default; uint16_t UnlinkedVectorPacket::packet_code() const { return p->packetCode_; } uint16_t UnlinkedVectorPacket::length_of_block() const { return p->lengthOfBlock_; } std::optional UnlinkedVectorPacket::value_of_vector() const { std::optional value; if (p->packetCode_ == 10) { value = p->valueOfVector_; } return value; } size_t UnlinkedVectorPacket::data_size() const { return p->lengthOfBlock_ + 4u; } bool UnlinkedVectorPacket::Parse(std::istream& is) { bool blockValid = true; is.read(reinterpret_cast(&p->packetCode_), 2); is.read(reinterpret_cast(&p->lengthOfBlock_), 2); p->packetCode_ = ntohs(p->packetCode_); p->lengthOfBlock_ = ntohs(p->lengthOfBlock_); int vectorSize = static_cast(p->lengthOfBlock_); if (is.eof()) { BOOST_LOG_TRIVIAL(debug) << logPrefix_ << "Reached end of file"; blockValid = false; } else if (p->packetCode_ == 10) { is.read(reinterpret_cast(&p->valueOfVector_), 2); p->valueOfVector_ = ntohs(p->valueOfVector_); vectorSize -= 2; } // The number of vectors is equal to the size divided by the number of bytes // in a vector int vectorCount = vectorSize / 8; p->beginI_.resize(vectorCount); p->beginJ_.resize(vectorCount); p->endI_.resize(vectorCount); p->endJ_.resize(vectorCount); for (int v = 0; v < vectorCount && !is.eof(); v++) { is.read(reinterpret_cast(&p->beginI_[v]), 2); is.read(reinterpret_cast(&p->beginJ_[v]), 2); is.read(reinterpret_cast(&p->endI_[v]), 2); is.read(reinterpret_cast(&p->endJ_[v]), 2); p->beginI_[v] = ntohs(p->beginI_[v]); p->beginJ_[v] = ntohs(p->beginJ_[v]); p->endI_[v] = ntohs(p->endI_[v]); p->endJ_[v] = ntohs(p->endJ_[v]); } if (is.eof()) { BOOST_LOG_TRIVIAL(debug) << logPrefix_ << "Reached end of file"; blockValid = false; } else { if (p->packetCode_ != 7 && p->packetCode_ != 10) { BOOST_LOG_TRIVIAL(warning) << logPrefix_ << "Invalid packet code: " << p->packetCode_; blockValid = false; } } return blockValid; } std::shared_ptr UnlinkedVectorPacket::Create(std::istream& is) { std::shared_ptr packet = std::make_shared(); if (!packet->Parse(is)) { packet.reset(); } return packet; } } // namespace rpg } // namespace wsr88d } // namespace scwx