mirror of
				https://github.com/ciphervance/supercell-wx.git
				synced 2025-11-03 19:10:05 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			219 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			219 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#include <scwx/wsr88d/rpg/cell_trend_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::cell_trend_data_packet";
 | 
						|
static const auto logger_ = util::Logger::Create(logPrefix_);
 | 
						|
 | 
						|
struct CellTrendData
 | 
						|
{
 | 
						|
   uint16_t              trendCode_;
 | 
						|
   uint8_t               numberOfVolumes_;
 | 
						|
   uint8_t               latestVolumePointer_;
 | 
						|
   std::vector<uint16_t> trendData_;
 | 
						|
 | 
						|
   CellTrendData() :
 | 
						|
       trendCode_ {0},
 | 
						|
       numberOfVolumes_ {0},
 | 
						|
       latestVolumePointer_ {0},
 | 
						|
       trendData_ {}
 | 
						|
   {
 | 
						|
   }
 | 
						|
};
 | 
						|
 | 
						|
class CellTrendDataPacketImpl
 | 
						|
{
 | 
						|
public:
 | 
						|
   explicit CellTrendDataPacketImpl() :
 | 
						|
       packetCode_ {0},
 | 
						|
       lengthOfBlock_ {0},
 | 
						|
       cellId_ {0},
 | 
						|
       iPosition_ {0},
 | 
						|
       jPosition_ {0},
 | 
						|
       trendData_ {}
 | 
						|
   {
 | 
						|
   }
 | 
						|
   ~CellTrendDataPacketImpl() = default;
 | 
						|
 | 
						|
   uint16_t    packetCode_;
 | 
						|
   uint16_t    lengthOfBlock_;
 | 
						|
   std::string cellId_;
 | 
						|
   int16_t     iPosition_;
 | 
						|
   int16_t     jPosition_;
 | 
						|
 | 
						|
   std::vector<CellTrendData> trendData_;
 | 
						|
};
 | 
						|
 | 
						|
CellTrendDataPacket::CellTrendDataPacket() :
 | 
						|
    p(std::make_unique<CellTrendDataPacketImpl>())
 | 
						|
{
 | 
						|
}
 | 
						|
CellTrendDataPacket::~CellTrendDataPacket() = default;
 | 
						|
 | 
						|
CellTrendDataPacket::CellTrendDataPacket(CellTrendDataPacket&&) noexcept =
 | 
						|
   default;
 | 
						|
CellTrendDataPacket&
 | 
						|
CellTrendDataPacket::operator=(CellTrendDataPacket&&) noexcept = default;
 | 
						|
 | 
						|
uint16_t CellTrendDataPacket::packet_code() const
 | 
						|
{
 | 
						|
   return p->packetCode_;
 | 
						|
}
 | 
						|
 | 
						|
uint16_t CellTrendDataPacket::length_of_block() const
 | 
						|
{
 | 
						|
   return p->lengthOfBlock_;
 | 
						|
}
 | 
						|
 | 
						|
std::string CellTrendDataPacket::cell_id() const
 | 
						|
{
 | 
						|
   return p->cellId_;
 | 
						|
}
 | 
						|
 | 
						|
int16_t CellTrendDataPacket::i_position() const
 | 
						|
{
 | 
						|
   return p->iPosition_;
 | 
						|
}
 | 
						|
 | 
						|
int16_t CellTrendDataPacket::j_position() const
 | 
						|
{
 | 
						|
   return p->iPosition_;
 | 
						|
}
 | 
						|
 | 
						|
uint16_t CellTrendDataPacket::number_of_trends() const
 | 
						|
{
 | 
						|
   return static_cast<uint16_t>(p->trendData_.size());
 | 
						|
}
 | 
						|
 | 
						|
uint16_t CellTrendDataPacket::trend_code(uint16_t t) const
 | 
						|
{
 | 
						|
   return p->trendData_[t].trendCode_;
 | 
						|
}
 | 
						|
 | 
						|
uint8_t CellTrendDataPacket::number_of_volumes(uint16_t t) const
 | 
						|
{
 | 
						|
   return p->trendData_[t].numberOfVolumes_;
 | 
						|
}
 | 
						|
 | 
						|
uint8_t CellTrendDataPacket::latest_volume_pointer(uint16_t t) const
 | 
						|
{
 | 
						|
   return p->trendData_[t].latestVolumePointer_;
 | 
						|
}
 | 
						|
 | 
						|
uint16_t CellTrendDataPacket::trend_data(uint16_t t, uint8_t v) const
 | 
						|
{
 | 
						|
   return p->trendData_[t].trendData_[v];
 | 
						|
}
 | 
						|
 | 
						|
size_t CellTrendDataPacket::data_size() const
 | 
						|
{
 | 
						|
   return p->lengthOfBlock_ + 4u;
 | 
						|
}
 | 
						|
 | 
						|
bool CellTrendDataPacket::Parse(std::istream& is)
 | 
						|
{
 | 
						|
   bool blockValid = true;
 | 
						|
 | 
						|
   std::streampos isBegin = is.tellg();
 | 
						|
 | 
						|
   is.read(reinterpret_cast<char*>(&p->packetCode_), 2);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->lengthOfBlock_), 2);
 | 
						|
 | 
						|
   p->packetCode_    = ntohs(p->packetCode_);
 | 
						|
   p->lengthOfBlock_ = ntohs(p->lengthOfBlock_);
 | 
						|
 | 
						|
   if (is.eof())
 | 
						|
   {
 | 
						|
      logger_->debug("Reached end of file");
 | 
						|
      blockValid = false;
 | 
						|
   }
 | 
						|
   else
 | 
						|
   {
 | 
						|
      if (p->packetCode_ != 21)
 | 
						|
      {
 | 
						|
         logger_->warn("Invalid packet code: {}", p->packetCode_);
 | 
						|
         blockValid = false;
 | 
						|
      }
 | 
						|
      else if (p->lengthOfBlock_ < 12 || p->lengthOfBlock_ > 198)
 | 
						|
      {
 | 
						|
         logger_->warn("Invalid length of block: {}", p->packetCode_);
 | 
						|
         blockValid = false;
 | 
						|
      }
 | 
						|
   }
 | 
						|
 | 
						|
   p->cellId_.resize(2);
 | 
						|
 | 
						|
   is.read(p->cellId_.data(), 2);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->iPosition_), 2);
 | 
						|
   is.read(reinterpret_cast<char*>(&p->jPosition_), 2);
 | 
						|
 | 
						|
   p->iPosition_ = ntohs(p->iPosition_);
 | 
						|
   p->jPosition_ = ntohs(p->jPosition_);
 | 
						|
 | 
						|
   if (blockValid)
 | 
						|
   {
 | 
						|
      size_t trendSize = p->lengthOfBlock_ - 6;
 | 
						|
      size_t bytesRead = 0;
 | 
						|
 | 
						|
      while (bytesRead < trendSize)
 | 
						|
      {
 | 
						|
         CellTrendData trendData;
 | 
						|
 | 
						|
         is.read(reinterpret_cast<char*>(&trendData.trendCode_), 2);
 | 
						|
         is.read(reinterpret_cast<char*>(&trendData.numberOfVolumes_), 1);
 | 
						|
         is.read(reinterpret_cast<char*>(&trendData.latestVolumePointer_), 1);
 | 
						|
 | 
						|
         trendData.trendCode_ = ntohs(trendData.trendCode_);
 | 
						|
 | 
						|
         size_t trendDataSize =
 | 
						|
            static_cast<size_t>(trendData.numberOfVolumes_) * 2u;
 | 
						|
         trendData.trendData_.resize(trendData.numberOfVolumes_);
 | 
						|
 | 
						|
         is.read(reinterpret_cast<char*>(trendData.trendData_.data()),
 | 
						|
                 trendDataSize);
 | 
						|
 | 
						|
         SwapVector(trendData.trendData_);
 | 
						|
 | 
						|
         p->trendData_.push_back(std::move(trendData));
 | 
						|
 | 
						|
         bytesRead += 4 + trendDataSize;
 | 
						|
      }
 | 
						|
   }
 | 
						|
 | 
						|
   std::streampos isEnd = is.tellg();
 | 
						|
 | 
						|
   if (!ValidateMessage(is, isEnd - isBegin))
 | 
						|
   {
 | 
						|
      blockValid = false;
 | 
						|
   }
 | 
						|
 | 
						|
   return blockValid;
 | 
						|
}
 | 
						|
 | 
						|
std::shared_ptr<CellTrendDataPacket>
 | 
						|
CellTrendDataPacket::Create(std::istream& is)
 | 
						|
{
 | 
						|
   std::shared_ptr<CellTrendDataPacket> packet =
 | 
						|
      std::make_shared<CellTrendDataPacket>();
 | 
						|
 | 
						|
   if (!packet->Parse(is))
 | 
						|
   {
 | 
						|
      packet.reset();
 | 
						|
   }
 | 
						|
 | 
						|
   return packet;
 | 
						|
}
 | 
						|
 | 
						|
} // namespace rpg
 | 
						|
} // namespace wsr88d
 | 
						|
} // namespace scwx
 |