Graphic alphanumeric block

This commit is contained in:
Dan Paulat 2022-01-12 01:15:41 -06:00
parent 09649c0fe7
commit add6c41016
4 changed files with 251 additions and 11 deletions

View file

@ -1,5 +1,6 @@
#include <scwx/wsr88d/level3_file.hpp>
#include <scwx/wsr88d/rpg/ccb_header.hpp>
#include <scwx/wsr88d/rpg/graphic_alphanumeric_block.hpp>
#include <scwx/wsr88d/rpg/level3_message_header.hpp>
#include <scwx/wsr88d/rpg/product_description_block.hpp>
#include <scwx/wsr88d/rpg/product_symbology_block.hpp>
@ -39,14 +40,14 @@ public:
bool LoadFileData(std::istream& is);
bool LoadBlocks(std::istream& is);
std::shared_ptr<rpg::WmoHeader> wmoHeader_;
std::shared_ptr<rpg::CcbHeader> ccbHeader_;
std::shared_ptr<rpg::WmoHeader> innerHeader_;
std::shared_ptr<rpg::Level3MessageHeader> messageHeader_;
std::shared_ptr<rpg::ProductDescriptionBlock> descriptionBlock_;
std::shared_ptr<rpg::ProductSymbologyBlock> symbologyBlock_;
std::shared_ptr<void> graphicBlock_;
std::shared_ptr<void> tabularBlock_;
std::shared_ptr<rpg::WmoHeader> wmoHeader_;
std::shared_ptr<rpg::CcbHeader> ccbHeader_;
std::shared_ptr<rpg::WmoHeader> innerHeader_;
std::shared_ptr<rpg::Level3MessageHeader> messageHeader_;
std::shared_ptr<rpg::ProductDescriptionBlock> descriptionBlock_;
std::shared_ptr<rpg::ProductSymbologyBlock> symbologyBlock_;
std::shared_ptr<rpg::GraphicAlphanumericBlock> graphicBlock_;
std::shared_ptr<void> tabularBlock_;
};
Level3File::Level3File() : p(std::make_unique<Level3FileImpl>()) {}
@ -286,13 +287,19 @@ bool Level3FileImpl::LoadBlocks(std::istream& is)
if (offsetToGraphic >= offsetBase)
{
// TODO
graphicBlock_ = std::make_shared<rpg::GraphicAlphanumericBlock>();
is.seekg(offsetToGraphic - offsetBase, std::ios_base::cur);
graphicValid = graphicBlock_->Parse(is);
is.seekg(offsetBasePos, std::ios_base::beg);
BOOST_LOG_TRIVIAL(debug)
<< logPrefix_
<< "Graphic alphanumeric block found: " << offsetToGraphic;
<< logPrefix_ << "Graphic alphanumeric block valid: " << graphicValid;
if (!graphicValid)
{
graphicBlock_ = nullptr;
}
}
if (offsetToTabular >= offsetBase)

View file

@ -0,0 +1,187 @@
#include <scwx/wsr88d/rpg/graphic_alphanumeric_block.hpp>
#include <scwx/wsr88d/rpg/packet_factory.hpp>
#include <istream>
#include <string>
#include <boost/log/trivial.hpp>
namespace scwx
{
namespace wsr88d
{
namespace rpg
{
static const std::string logPrefix_ =
"[scwx::wsr88d::rpg::graphic_alphanumeric_block] ";
class GraphicAlphanumericBlockImpl
{
public:
explicit GraphicAlphanumericBlockImpl() :
blockDivider_ {0},
blockId_ {0},
lengthOfBlock_ {0},
numberOfPages_ {0},
pageList_ {}
{
}
~GraphicAlphanumericBlockImpl() = default;
int16_t blockDivider_;
int16_t blockId_;
uint32_t lengthOfBlock_;
uint16_t numberOfPages_;
std::vector<std::vector<std::shared_ptr<Packet>>> pageList_;
};
GraphicAlphanumericBlock::GraphicAlphanumericBlock() :
Message(), p(std::make_unique<GraphicAlphanumericBlockImpl>())
{
}
GraphicAlphanumericBlock::~GraphicAlphanumericBlock() = default;
GraphicAlphanumericBlock::GraphicAlphanumericBlock(
GraphicAlphanumericBlock&&) noexcept = default;
GraphicAlphanumericBlock& GraphicAlphanumericBlock::operator=(
GraphicAlphanumericBlock&&) noexcept = default;
int16_t GraphicAlphanumericBlock::block_divider() const
{
return p->blockDivider_;
}
size_t GraphicAlphanumericBlock::data_size() const
{
return p->lengthOfBlock_;
}
bool GraphicAlphanumericBlock::Parse(std::istream& is)
{
bool blockValid = true;
const std::streampos blockStart = is.tellg();
is.read(reinterpret_cast<char*>(&p->blockDivider_), 2);
is.read(reinterpret_cast<char*>(&p->blockId_), 2);
is.read(reinterpret_cast<char*>(&p->lengthOfBlock_), 4);
is.read(reinterpret_cast<char*>(&p->numberOfPages_), 2);
p->blockDivider_ = ntohs(p->blockDivider_);
p->blockId_ = ntohs(p->blockId_);
p->lengthOfBlock_ = ntohl(p->lengthOfBlock_);
p->numberOfPages_ = ntohs(p->numberOfPages_);
if (is.eof())
{
BOOST_LOG_TRIVIAL(debug) << logPrefix_ << "Reached end of file";
blockValid = false;
}
else
{
if (p->blockDivider_ != -1)
{
BOOST_LOG_TRIVIAL(warning)
<< logPrefix_ << "Invalid block divider: " << p->blockDivider_;
blockValid = false;
}
if (p->blockId_ != 2)
{
BOOST_LOG_TRIVIAL(warning)
<< logPrefix_ << "Invalid block ID: " << p->blockId_;
blockValid = false;
}
if (p->lengthOfBlock_ < 10)
{
BOOST_LOG_TRIVIAL(warning)
<< logPrefix_ << "Invalid block length: " << p->lengthOfBlock_;
blockValid = false;
}
if (p->numberOfPages_ < 1 || p->numberOfPages_ > 48)
{
BOOST_LOG_TRIVIAL(warning)
<< logPrefix_ << "Invalid number of pages: " << p->numberOfPages_;
blockValid = false;
}
}
if (blockValid)
{
uint16_t pageNumber;
uint16_t lengthOfPage;
for (uint16_t i = 0; i < p->numberOfPages_; i++)
{
BOOST_LOG_TRIVIAL(trace) << logPrefix_ << "Page " << (i + 1);
std::vector<std::shared_ptr<Packet>> packetList;
uint32_t bytesRead = 0;
is.read(reinterpret_cast<char*>(&pageNumber), 2);
is.read(reinterpret_cast<char*>(&lengthOfPage), 2);
pageNumber = ntohs(pageNumber);
lengthOfPage = ntohs(lengthOfPage);
std::streampos pageStart = is.tellg();
std::streampos pageEnd =
pageStart + static_cast<std::streamoff>(lengthOfPage);
if (pageNumber != i + 1)
{
BOOST_LOG_TRIVIAL(warning)
<< logPrefix_ << "Page out of order: Expected " << (i + 1)
<< ", found " << pageNumber;
}
while (bytesRead < lengthOfPage)
{
std::shared_ptr<Packet> packet = PacketFactory::Create(is);
if (packet != nullptr)
{
packetList.push_back(packet);
bytesRead += static_cast<uint32_t>(packet->data_size());
}
else
{
break;
}
}
if (bytesRead < lengthOfPage)
{
BOOST_LOG_TRIVIAL(trace)
<< logPrefix_
<< "Page bytes read smaller than size: " << bytesRead << " < "
<< lengthOfPage << " bytes";
blockValid = false;
is.seekg(pageEnd, std::ios_base::beg);
}
if (bytesRead > lengthOfPage)
{
BOOST_LOG_TRIVIAL(warning)
<< logPrefix_
<< "Page bytes read larger than size: " << bytesRead << " > "
<< lengthOfPage << " bytes";
blockValid = false;
is.seekg(pageEnd, std::ios_base::beg);
}
p->pageList_.push_back(std::move(packetList));
}
}
const std::streampos blockEnd = is.tellg();
if (!ValidateMessage(is, blockEnd - blockStart))
{
blockValid = false;
}
return blockValid;
}
} // namespace rpg
} // namespace wsr88d
} // namespace scwx