mirror of
https://github.com/ciphervance/supercell-wx.git
synced 2025-10-30 16:00:08 +00:00
258 lines
6.4 KiB
C++
258 lines
6.4 KiB
C++
#include <scwx/wsr88d/rpg/radial_data_packet.hpp>
|
|
#include <scwx/util/logger.hpp>
|
|
|
|
#include <istream>
|
|
#include <string>
|
|
|
|
namespace scwx
|
|
{
|
|
namespace wsr88d
|
|
{
|
|
namespace rpg
|
|
{
|
|
|
|
static const std::string logPrefix_ = "scwx::wsr88d::rpg::radial_data_packet";
|
|
static const auto logger_ = util::Logger::Create(logPrefix_);
|
|
|
|
class RadialDataPacketImpl
|
|
{
|
|
public:
|
|
struct Radial
|
|
{
|
|
uint16_t numberOfRleHalfwords_;
|
|
uint16_t startAngle_;
|
|
uint16_t angleDelta_;
|
|
std::vector<uint8_t> data_;
|
|
std::vector<uint8_t> level_;
|
|
|
|
Radial() :
|
|
numberOfRleHalfwords_ {0},
|
|
startAngle_ {0},
|
|
angleDelta_ {0},
|
|
data_ {},
|
|
level_ {}
|
|
{
|
|
}
|
|
};
|
|
|
|
explicit RadialDataPacketImpl() :
|
|
packetCode_ {0},
|
|
indexOfFirstRangeBin_ {0},
|
|
numberOfRangeBins_ {0},
|
|
iCenterOfSweep_ {0},
|
|
jCenterOfSweep_ {0},
|
|
scaleFactor_ {0},
|
|
radial_ {},
|
|
dataSize_ {0}
|
|
{
|
|
}
|
|
~RadialDataPacketImpl() = default;
|
|
|
|
uint16_t packetCode_;
|
|
uint16_t indexOfFirstRangeBin_;
|
|
uint16_t numberOfRangeBins_;
|
|
int16_t iCenterOfSweep_;
|
|
int16_t jCenterOfSweep_;
|
|
uint16_t scaleFactor_;
|
|
uint16_t numberOfRadials_;
|
|
|
|
// Repeat for each radial
|
|
std::vector<Radial> radial_;
|
|
|
|
size_t dataSize_;
|
|
};
|
|
|
|
RadialDataPacket::RadialDataPacket() :
|
|
p(std::make_unique<RadialDataPacketImpl>())
|
|
{
|
|
}
|
|
RadialDataPacket::~RadialDataPacket() = default;
|
|
|
|
RadialDataPacket::RadialDataPacket(RadialDataPacket&&) noexcept = default;
|
|
RadialDataPacket&
|
|
RadialDataPacket::operator=(RadialDataPacket&&) noexcept = default;
|
|
|
|
uint16_t RadialDataPacket::packet_code() const
|
|
{
|
|
return p->packetCode_;
|
|
}
|
|
|
|
uint16_t RadialDataPacket::index_of_first_range_bin() const
|
|
{
|
|
return p->indexOfFirstRangeBin_;
|
|
}
|
|
|
|
uint16_t RadialDataPacket::number_of_range_bins() const
|
|
{
|
|
return p->numberOfRangeBins_;
|
|
}
|
|
|
|
int16_t RadialDataPacket::i_center_of_sweep() const
|
|
{
|
|
return p->iCenterOfSweep_;
|
|
}
|
|
|
|
int16_t RadialDataPacket::j_center_of_sweep() const
|
|
{
|
|
return p->jCenterOfSweep_;
|
|
}
|
|
|
|
float RadialDataPacket::scale_factor() const
|
|
{
|
|
return p->scaleFactor_ * 0.001f;
|
|
}
|
|
|
|
uint16_t RadialDataPacket::number_of_radials() const
|
|
{
|
|
return p->numberOfRadials_;
|
|
}
|
|
|
|
float RadialDataPacket::start_angle(uint16_t r) const
|
|
{
|
|
return p->radial_[r].startAngle_ * 0.1f;
|
|
}
|
|
|
|
float RadialDataPacket::delta_angle(uint16_t r) const
|
|
{
|
|
return p->radial_[r].angleDelta_ * 0.1f;
|
|
}
|
|
|
|
const std::vector<uint8_t>& RadialDataPacket::level(uint16_t r) const
|
|
{
|
|
return p->radial_[r].level_;
|
|
}
|
|
|
|
size_t RadialDataPacket::data_size() const
|
|
{
|
|
return p->dataSize_;
|
|
}
|
|
|
|
bool RadialDataPacket::Parse(std::istream& is)
|
|
{
|
|
bool blockValid = true;
|
|
size_t bytesRead = 0;
|
|
|
|
is.read(reinterpret_cast<char*>(&p->packetCode_), 2);
|
|
is.read(reinterpret_cast<char*>(&p->indexOfFirstRangeBin_), 2);
|
|
is.read(reinterpret_cast<char*>(&p->numberOfRangeBins_), 2);
|
|
is.read(reinterpret_cast<char*>(&p->iCenterOfSweep_), 2);
|
|
is.read(reinterpret_cast<char*>(&p->jCenterOfSweep_), 2);
|
|
is.read(reinterpret_cast<char*>(&p->scaleFactor_), 2);
|
|
is.read(reinterpret_cast<char*>(&p->numberOfRadials_), 2);
|
|
bytesRead += 14;
|
|
|
|
p->packetCode_ = ntohs(p->packetCode_);
|
|
p->indexOfFirstRangeBin_ = ntohs(p->indexOfFirstRangeBin_);
|
|
p->numberOfRangeBins_ = ntohs(p->numberOfRangeBins_);
|
|
p->iCenterOfSweep_ = ntohs(p->iCenterOfSweep_);
|
|
p->jCenterOfSweep_ = ntohs(p->jCenterOfSweep_);
|
|
p->scaleFactor_ = ntohs(p->scaleFactor_);
|
|
p->numberOfRadials_ = ntohs(p->numberOfRadials_);
|
|
|
|
if (is.eof())
|
|
{
|
|
logger_->debug("Reached end of file");
|
|
blockValid = false;
|
|
}
|
|
else
|
|
{
|
|
if (p->packetCode_ != 0xAF1F)
|
|
{
|
|
logger_->warn("Invalid packet code: {}", p->packetCode_);
|
|
blockValid = false;
|
|
}
|
|
if (p->numberOfRangeBins_ < 1 || p->numberOfRangeBins_ > 460)
|
|
{
|
|
logger_->warn("Invalid number of range bins: {}",
|
|
p->numberOfRangeBins_);
|
|
blockValid = false;
|
|
}
|
|
if (p->numberOfRadials_ < 1 || p->numberOfRadials_ > 400)
|
|
{
|
|
logger_->warn("Invalid number of radials: {}", p->numberOfRadials_);
|
|
blockValid = false;
|
|
}
|
|
}
|
|
|
|
if (blockValid)
|
|
{
|
|
p->radial_.resize(p->numberOfRadials_);
|
|
|
|
for (uint16_t r = 0; r < p->numberOfRadials_; r++)
|
|
{
|
|
auto& radial = p->radial_[r];
|
|
|
|
is.read(reinterpret_cast<char*>(&radial.numberOfRleHalfwords_), 2);
|
|
is.read(reinterpret_cast<char*>(&radial.startAngle_), 2);
|
|
is.read(reinterpret_cast<char*>(&radial.angleDelta_), 2);
|
|
bytesRead += 6;
|
|
|
|
radial.numberOfRleHalfwords_ = ntohs(radial.numberOfRleHalfwords_);
|
|
radial.startAngle_ = ntohs(radial.startAngle_);
|
|
radial.angleDelta_ = ntohs(radial.angleDelta_);
|
|
|
|
if (radial.numberOfRleHalfwords_ < 1 ||
|
|
radial.numberOfRleHalfwords_ > 230)
|
|
{
|
|
logger_->warn("Invalid number of RLE halfwords: {} (Radial {})",
|
|
radial.numberOfRleHalfwords_,
|
|
r);
|
|
blockValid = false;
|
|
break;
|
|
}
|
|
|
|
// Read RLE halfwords
|
|
size_t dataSize = radial.numberOfRleHalfwords_ * 2;
|
|
radial.data_.resize(dataSize);
|
|
is.read(reinterpret_cast<char*>(radial.data_.data()), dataSize);
|
|
bytesRead += dataSize;
|
|
|
|
// If the final byte is 0, truncate it
|
|
if (radial.data_.back() == 0)
|
|
{
|
|
radial.data_.pop_back();
|
|
}
|
|
|
|
// Unpack the levels from the Run Length Encoded data
|
|
radial.level_.resize(p->numberOfRangeBins_);
|
|
|
|
uint16_t b = 0;
|
|
for (auto it = radial.data_.cbegin(); it != radial.data_.cend(); it++)
|
|
{
|
|
uint8_t run = *it >> 4;
|
|
uint8_t level = *it & 0x0f;
|
|
|
|
for (int i = 0; i < run && b < p->numberOfRangeBins_; i++)
|
|
{
|
|
radial.level_[b++] = level;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
p->dataSize_ = bytesRead;
|
|
|
|
if (!ValidateMessage(is, bytesRead))
|
|
{
|
|
blockValid = false;
|
|
}
|
|
|
|
return blockValid;
|
|
}
|
|
|
|
std::shared_ptr<RadialDataPacket> RadialDataPacket::Create(std::istream& is)
|
|
{
|
|
std::shared_ptr<RadialDataPacket> packet =
|
|
std::make_shared<RadialDataPacket>();
|
|
|
|
if (!packet->Parse(is))
|
|
{
|
|
packet.reset();
|
|
}
|
|
|
|
return packet;
|
|
}
|
|
|
|
} // namespace rpg
|
|
} // namespace wsr88d
|
|
} // namespace scwx
|