mirror of
				https://github.com/ciphervance/supercell-wx.git
				synced 2025-11-04 12:10:05 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			266 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			266 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#include <scwx/wsr88d/rpg/raster_data_packet.hpp>
 | 
						|
#include <scwx/util/logger.hpp>
 | 
						|
 | 
						|
#include <istream>
 | 
						|
#include <numeric>
 | 
						|
#include <string>
 | 
						|
 | 
						|
namespace scwx
 | 
						|
{
 | 
						|
namespace wsr88d
 | 
						|
{
 | 
						|
namespace rpg
 | 
						|
{
 | 
						|
 | 
						|
static const std::string logPrefix_ = "scwx::wsr88d::rpg::raster_data_packet";
 | 
						|
static const auto        logger_    = util::Logger::Create(logPrefix_);
 | 
						|
 | 
						|
class RasterDataPacketImpl
 | 
						|
{
 | 
						|
public:
 | 
						|
   struct Row
 | 
						|
   {
 | 
						|
      uint16_t             numberOfBytes_;
 | 
						|
      std::vector<uint8_t> data_;
 | 
						|
      std::vector<uint8_t> level_;
 | 
						|
 | 
						|
      Row() : numberOfBytes_ {0}, data_ {}, level_ {} {}
 | 
						|
   };
 | 
						|
 | 
						|
   explicit RasterDataPacketImpl() :
 | 
						|
       packetCode_ {0},
 | 
						|
       opFlag_ {0},
 | 
						|
       iCoordinateStart_ {0},
 | 
						|
       jCoordinateStart_ {0},
 | 
						|
       xScaleInt_ {0},
 | 
						|
       xScaleFractional_ {0},
 | 
						|
       yScaleInt_ {0},
 | 
						|
       yScaleFractional_ {0},
 | 
						|
       numberOfRows_ {0},
 | 
						|
       packagingDescriptor_ {0},
 | 
						|
       row_ {},
 | 
						|
       dataSize_ {0}
 | 
						|
   {
 | 
						|
   }
 | 
						|
   ~RasterDataPacketImpl() = default;
 | 
						|
 | 
						|
   uint16_t                packetCode_;
 | 
						|
   std::array<uint16_t, 2> opFlag_;
 | 
						|
   int16_t                 iCoordinateStart_;
 | 
						|
   int16_t                 jCoordinateStart_;
 | 
						|
   uint16_t                xScaleInt_;
 | 
						|
   uint16_t                xScaleFractional_;
 | 
						|
   uint16_t                yScaleInt_;
 | 
						|
   uint16_t                yScaleFractional_;
 | 
						|
   uint16_t                numberOfRows_;
 | 
						|
   uint16_t                packagingDescriptor_;
 | 
						|
 | 
						|
   // Repeat for each row
 | 
						|
   std::vector<Row> row_;
 | 
						|
 | 
						|
   size_t dataSize_;
 | 
						|
};
 | 
						|
 | 
						|
RasterDataPacket::RasterDataPacket() :
 | 
						|
    p(std::make_unique<RasterDataPacketImpl>())
 | 
						|
{
 | 
						|
}
 | 
						|
RasterDataPacket::~RasterDataPacket() = default;
 | 
						|
 | 
						|
RasterDataPacket::RasterDataPacket(RasterDataPacket&&) noexcept = default;
 | 
						|
RasterDataPacket&
 | 
						|
RasterDataPacket::operator=(RasterDataPacket&&) noexcept = default;
 | 
						|
 | 
						|
uint16_t RasterDataPacket::packet_code() const
 | 
						|
{
 | 
						|
   return p->packetCode_;
 | 
						|
}
 | 
						|
 | 
						|
uint16_t RasterDataPacket::op_flag(size_t i) const
 | 
						|
{
 | 
						|
   return p->opFlag_[i];
 | 
						|
}
 | 
						|
 | 
						|
int16_t RasterDataPacket::i_coordinate_start() const
 | 
						|
{
 | 
						|
   return p->iCoordinateStart_;
 | 
						|
}
 | 
						|
 | 
						|
int16_t RasterDataPacket::j_coordinate_start() const
 | 
						|
{
 | 
						|
   return p->jCoordinateStart_;
 | 
						|
}
 | 
						|
 | 
						|
uint16_t RasterDataPacket::x_scale_int() const
 | 
						|
{
 | 
						|
   return p->xScaleInt_;
 | 
						|
}
 | 
						|
 | 
						|
uint16_t RasterDataPacket::x_scale_fractional() const
 | 
						|
{
 | 
						|
   return p->xScaleFractional_;
 | 
						|
}
 | 
						|
 | 
						|
uint16_t RasterDataPacket::y_scale_int() const
 | 
						|
{
 | 
						|
   return p->yScaleInt_;
 | 
						|
}
 | 
						|
 | 
						|
uint16_t RasterDataPacket::y_scale_fractional() const
 | 
						|
{
 | 
						|
   return p->yScaleFractional_;
 | 
						|
}
 | 
						|
 | 
						|
uint16_t RasterDataPacket::number_of_rows() const
 | 
						|
{
 | 
						|
   return p->numberOfRows_;
 | 
						|
}
 | 
						|
 | 
						|
uint16_t RasterDataPacket::packaging_descriptor() const
 | 
						|
{
 | 
						|
   return p->packagingDescriptor_;
 | 
						|
}
 | 
						|
 | 
						|
const std::vector<uint8_t>& RasterDataPacket::level(uint16_t r) const
 | 
						|
{
 | 
						|
   return p->row_[r].level_;
 | 
						|
}
 | 
						|
 | 
						|
size_t RasterDataPacket::data_size() const
 | 
						|
{
 | 
						|
   return p->dataSize_;
 | 
						|
}
 | 
						|
 | 
						|
bool RasterDataPacket::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->opFlag_.data()), 4);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->iCoordinateStart_), 2);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->jCoordinateStart_), 2);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->xScaleInt_), 2);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->xScaleFractional_), 2);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->yScaleInt_), 2);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->yScaleFractional_), 2);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->numberOfRows_), 2);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->packagingDescriptor_), 2);
 | 
						|
   bytesRead += 22;
 | 
						|
 | 
						|
   p->packetCode_          = ntohs(p->packetCode_);
 | 
						|
   p->iCoordinateStart_    = ntohs(p->iCoordinateStart_);
 | 
						|
   p->jCoordinateStart_    = ntohs(p->jCoordinateStart_);
 | 
						|
   p->xScaleInt_           = ntohs(p->xScaleInt_);
 | 
						|
   p->xScaleFractional_    = ntohs(p->xScaleFractional_);
 | 
						|
   p->yScaleInt_           = ntohs(p->yScaleInt_);
 | 
						|
   p->yScaleFractional_    = ntohs(p->yScaleFractional_);
 | 
						|
   p->numberOfRows_        = ntohs(p->numberOfRows_);
 | 
						|
   p->packagingDescriptor_ = ntohs(p->packagingDescriptor_);
 | 
						|
 | 
						|
   SwapArray(p->opFlag_);
 | 
						|
 | 
						|
   if (is.eof())
 | 
						|
   {
 | 
						|
      logger_->debug("Reached end of file");
 | 
						|
      blockValid = false;
 | 
						|
   }
 | 
						|
   else
 | 
						|
   {
 | 
						|
      if (p->packetCode_ != 0xBA0F && p->packetCode_ != 0xBA07)
 | 
						|
      {
 | 
						|
         logger_->warn("Invalid packet code: {}", p->packetCode_);
 | 
						|
         blockValid = false;
 | 
						|
      }
 | 
						|
      if (p->numberOfRows_ < 1 || p->numberOfRows_ > 464)
 | 
						|
      {
 | 
						|
         logger_->warn("Invalid number of rows: {}", p->numberOfRows_);
 | 
						|
         blockValid = false;
 | 
						|
      }
 | 
						|
   }
 | 
						|
 | 
						|
   if (blockValid)
 | 
						|
   {
 | 
						|
      p->row_.resize(p->numberOfRows_);
 | 
						|
 | 
						|
      for (uint16_t r = 0; r < p->numberOfRows_; r++)
 | 
						|
      {
 | 
						|
         auto& row = p->row_[r];
 | 
						|
 | 
						|
         is.read(reinterpret_cast<char*>(&row.numberOfBytes_), 2);
 | 
						|
         bytesRead += 2;
 | 
						|
 | 
						|
         row.numberOfBytes_ = ntohs(row.numberOfBytes_);
 | 
						|
 | 
						|
         if (row.numberOfBytes_ < 2 || row.numberOfBytes_ > 920 ||
 | 
						|
             row.numberOfBytes_ % 2 != 0)
 | 
						|
         {
 | 
						|
            logger_->warn("Invalid number of bytes in row: {} (Row {})",
 | 
						|
                          row.numberOfBytes_,
 | 
						|
                          r);
 | 
						|
            blockValid = false;
 | 
						|
            break;
 | 
						|
         }
 | 
						|
 | 
						|
         // Read row data
 | 
						|
         size_t dataSize = row.numberOfBytes_;
 | 
						|
         row.data_.resize(dataSize);
 | 
						|
         is.read(reinterpret_cast<char*>(row.data_.data()), dataSize);
 | 
						|
         bytesRead += dataSize;
 | 
						|
 | 
						|
         // If the final byte is 0, truncate it
 | 
						|
         if (row.data_.back() == 0)
 | 
						|
         {
 | 
						|
            row.data_.pop_back();
 | 
						|
         }
 | 
						|
 | 
						|
         // Unpack the levels from the Run Length Encoded data
 | 
						|
         uint16_t binCount =
 | 
						|
            std::accumulate(row.data_.cbegin(),
 | 
						|
                            row.data_.cend(),
 | 
						|
                            static_cast<uint16_t>(0u),
 | 
						|
                            [](const uint16_t& a, const uint8_t& b) -> uint16_t
 | 
						|
                            { return a + (b >> 4); });
 | 
						|
 | 
						|
         row.level_.resize(binCount);
 | 
						|
 | 
						|
         uint16_t b = 0;
 | 
						|
         for (auto it = row.data_.cbegin(); it != row.data_.cend(); it++)
 | 
						|
         {
 | 
						|
            uint8_t run   = *it >> 4;
 | 
						|
            uint8_t level = *it & 0x0f;
 | 
						|
 | 
						|
            for (int i = 0; i < run && b < binCount; i++)
 | 
						|
            {
 | 
						|
               row.level_[b++] = level;
 | 
						|
            }
 | 
						|
         }
 | 
						|
      }
 | 
						|
   }
 | 
						|
 | 
						|
   p->dataSize_ = bytesRead;
 | 
						|
 | 
						|
   if (!ValidateMessage(is, bytesRead))
 | 
						|
   {
 | 
						|
      blockValid = false;
 | 
						|
   }
 | 
						|
 | 
						|
   return blockValid;
 | 
						|
}
 | 
						|
 | 
						|
std::shared_ptr<RasterDataPacket> RasterDataPacket::Create(std::istream& is)
 | 
						|
{
 | 
						|
   std::shared_ptr<RasterDataPacket> packet =
 | 
						|
      std::make_shared<RasterDataPacket>();
 | 
						|
 | 
						|
   if (!packet->Parse(is))
 | 
						|
   {
 | 
						|
      packet.reset();
 | 
						|
   }
 | 
						|
 | 
						|
   return packet;
 | 
						|
}
 | 
						|
 | 
						|
} // namespace rpg
 | 
						|
} // namespace wsr88d
 | 
						|
} // namespace scwx
 |