30#include <wx/sstream.h>
32#include <wx/translation.h>
56 uint32_t w1 = aStream.
ReadU32();
57 uint32_t w2 = aStream.
ReadU32();
78 const uint32_t a = aStream.
ReadU32();
79 const uint32_t b = aStream.
ReadU32();
82 const uint64_t
combined = (
static_cast<uint64_t
>( a ) << 32 ) + b;
91template <
typename ARRAY>
94 for(
size_t i = 0; i < aArray.size(); i++ )
105 if constexpr( std::is_same_v<T, uint32_t> )
109 else if constexpr( std::is_same_v<T, uint8_t> )
113 else if constexpr( std::is_same_v<T, uint16_t> )
117 else if constexpr( std::is_same_v<T, int16_t> )
121 else if constexpr( std::is_same_v<T, int32_t> )
125 else if constexpr( std::is_same_v<T, FILE_HEADER::LINKED_LIST> )
127 field =
ReadLL( aStream, aFmtVer );
129 else if constexpr(
requires { std::tuple_size<T>::value; } )
131 for(
size_t i = 0; i < std::tuple_size_v<T>; ++i )
147template <
typename COND_T>
150 if( COND_T::exists( aFmtVer ) )
159 uint32_t masked = aMagic & 0xFFFFFF00;
182 uint32_t majorVer = ( aMagic >> 16 ) & 0xFFFF;
184 if( majorVer <= 0x0012 )
189 THROW_IO_ERROR( wxString::Format(
"Unknown Allegro file version %#010x (rev %d)", aMagic, majorVer - 3 ) );
195 auto header = std::make_unique<ALLEGRO::FILE_HEADER>();
196 const size_t headerStartPos =
m_stream.Position();
198 uint32_t fileMagic =
m_stream.ReadU32();
201 header->m_Magic = fileMagic;
268 wxASSERT(
m_stream.Position() - headerStartPos == 0xF8 );
270 wxASSERT(
m_stream.Position() - headerStartPos == 0x124 );
293 THROW_IO_ERROR( wxString::Format(
"Unknown board units %d", units ) );
318 for(
size_t i = 0; i <
header->m_LayerMap.size(); ++i )
325 header->GetStringsCount() );
335 static constexpr size_t STRING_TABLE_OFFSET = 0x1200;
337 stream.
Seek( STRING_TABLE_OFFSET );
339 for( uint32_t i = 0; i < count; ++i )
341 uint32_t
id = stream.
ReadU32();
351 uint8_t classCode = aStream.
ReadU8();
352 uint8_t subclassCode = aStream.
ReadU8();
357 .m_Class = classCode,
358 .m_Subclass = subclassCode,
365 auto block = std::make_unique<BLOCK<BLK_0x01_ARC>>( aStream.
Position() );
367 auto& data = block->GetData();
371 data.m_UnknownByte = aStream.
ReadU8();
372 data.m_SubType = aStream.
ReadU8();
373 data.m_Key = aStream.
ReadU32();
374 block->SetKey( data.m_Key );
375 data.m_Next = aStream.
ReadU32();
376 block->SetNext( data.m_Next );
377 data.m_Parent = aStream.
ReadU32();
378 data.m_Unknown1 = aStream.
ReadU32();
380 ReadCond( aStream, aVer, data.m_Unknown6 );
382 data.m_Width = aStream.
ReadU32();
384 data.m_StartX = aStream.
ReadS32();
385 data.m_StartY = aStream.
ReadS32();
386 data.m_EndX = aStream.
ReadS32();
387 data.m_EndY = aStream.
ReadS32();
393 for(
size_t i = 0; i < data.m_BoundingBoxCoords.size(); ++i )
395 data.m_BoundingBoxCoords[i] = aStream.
ReadS32();
404 auto block = std::make_unique<BLOCK<BLK_0x03_FIELD>>( aStream.
Position() );
406 auto& data = block->GetData();
410 data.m_Hdr1 = aStream.
ReadU16();
411 data.m_Key = aStream.
ReadU32();
412 block->SetKey( data.m_Key );
413 data.m_Next = aStream.
ReadU32();
414 block->SetNext( data.m_Next );
416 ReadCond( aStream, aVer, data.m_Unknown1 );
418 data.m_SubType = aStream.
ReadU8();
419 data.m_Hdr2 = aStream.
ReadU8();
420 data.m_Size = aStream.
ReadU16();
422 ReadCond( aStream, aVer, data.m_Unknown2 );
427 switch( data.m_SubType )
437 data.m_Substruct = aStream.
ReadU32();
442 data.m_Substruct = std::array<uint32_t, 2>{
468 "Block 0x03 subtype 0x6C entry count %u exceeds limit at offset %#010zx",
478 data.m_Substruct = std::move( sub );
489 const size_t numEntries = ( sub.
m_X1 + 4 * sub.
m_X0 );
491 for(
size_t i = 0; i < numEntries; ++i )
495 data.m_Substruct = std::move( sub );
501 for(
size_t i = 0; i < sub.
m_Entries.size(); ++i )
505 data.m_Substruct = std::move( sub );
510 if( data.m_Size == 4 )
512 data.m_Substruct = aStream.
ReadU32();
514 else if( data.m_Size == 8 )
516 std::array<uint32_t, 2> sub;
519 data.m_Substruct = std::move( sub );
524 wxString::Format(
"Unknown substruct type %#02x with size %d", data.m_SubType, data.m_Size ) );
536 auto block = std::make_unique<BLOCK<BLK_0x04_NET_ASSIGNMENT>>( aStream.
Position() );
538 auto& data = block->GetData();
540 data.m_Type = aStream.
ReadU8();
542 data.m_Key = aStream.
ReadU32();
543 block->SetKey( data.m_Key );
544 data.m_Next = aStream.
ReadU32();
545 block->SetNext( data.m_Next );
546 data.m_Net = aStream.
ReadU32();
547 data.m_ConnItem = aStream.
ReadU32();
549 ReadCond( aStream, aVer, data.m_Unknown );
557 auto block = std::make_unique<BLOCK<BLK_0x05_TRACK>>( aStream.
Position() );
559 auto& data = block->GetData();
564 data.m_Key = aStream.
ReadU32();
565 block->SetKey( data.m_Key );
566 data.m_Next = aStream.
ReadU32();
567 block->SetNext( data.m_Next );
568 data.m_NetAssignment = aStream.
ReadU32();
569 data.m_UnknownPtr1 = aStream.
ReadU32();
570 data.m_Unknown2 = aStream.
ReadU32();
571 data.m_Unknown3 = aStream.
ReadU32();
572 data.m_UnknownPtr2a = aStream.
ReadU32();
573 data.m_UnknownPtr2b = aStream.
ReadU32();
574 data.m_Unknown4 = aStream.
ReadU32();
575 data.m_UnknownPtr3a = aStream.
ReadU32();
576 data.m_UnknownPtr3b = aStream.
ReadU32();
578 ReadCond( aStream, aVer, data.m_Unknown5a );
579 ReadCond( aStream, aVer, data.m_Unknown5b );
581 data.m_FirstSegPtr = aStream.
ReadU32();
582 data.m_UnknownPtr5 = aStream.
ReadU32();
583 data.m_Unknown6 = aStream.
ReadU32();
591 auto block = std::make_unique<BLOCK<BLK_0x06_COMPONENT>>( stream.
Position() );
593 auto& data = block->GetData();
598 block->SetKey( data.m_Key );
599 data.m_Next = stream.
ReadU32();
600 block->SetNext( data.m_Next );
601 data.m_CompDeviceType = stream.
ReadU32();
602 data.m_SymbolName = stream.
ReadU32();
603 data.m_FirstInstPtr = stream.
ReadU32();
604 data.m_PtrFunctionSlot = stream.
ReadU32();
605 data.m_PtrPinNumber = stream.
ReadU32();
606 data.m_Fields = stream.
ReadU32();
608 ReadCond( stream, aVer, data.m_Unknown1 );
616 auto block = std::make_unique<BLOCK<BLK_0x07_COMPONENT_INST>>( stream.
Position() );
618 auto& data = block->GetData();
623 block->SetKey( data.m_Key );
624 data.m_Next = stream.
ReadU32();
625 block->SetNext( data.m_Next );
627 ReadCond( stream, aVer, data.m_UnknownPtr1 );
628 ReadCond( stream, aVer, data.m_Unknown2 );
629 ReadCond( stream, aVer, data.m_Unknown3 );
631 data.m_FpInstPtr = stream.
ReadU32();
633 ReadCond( stream, aVer, data.m_Unknown4 );
635 data.m_RefDesStrPtr = stream.
ReadU32();
636 data.m_FunctionInstPtr = stream.
ReadU32();
637 data.m_X03Ptr = stream.
ReadU32();
638 data.m_Unknown5 = stream.
ReadU32();
639 data.m_FirstPadPtr = stream.
ReadU32();
647 auto block = std::make_unique<BLOCK<BLK_0x08_PIN_NUMBER>>( aStream.
Position() );
649 auto& data = block->GetData();
651 data.m_Type = aStream.
ReadU8();
653 data.m_Key = aStream.
ReadU32();
654 block->SetKey( data.m_Key );
656 ReadCond( aStream, aVer, data.m_Previous );
657 ReadCond( aStream, aVer, data.m_StrPtr16x );
659 data.m_Next = aStream.
ReadU32();
660 block->SetNext( data.m_Next );
662 ReadCond( aStream, aVer, data.m_StrPtr );
664 data.m_PinNamePtr = aStream.
ReadU32();
666 ReadCond( aStream, aVer, data.m_Unknown1 );
668 data.m_Ptr4 = aStream.
ReadU32();
676 auto block = std::make_unique<BLOCK<BLK_0x09_FILL_LINK>>( aStream.
Position() );
678 auto& data = block->GetData();
682 data.m_Key = aStream.
ReadU32();
683 block->SetKey( data.m_Key );
685 for(
size_t i = 0; i < data.m_UnknownArray.size(); ++i )
687 data.m_UnknownArray[i] = aStream.
ReadU32();
690 ReadCond( aStream, aVer, data.m_Unknown1 );
692 data.m_UnknownPtr1 = aStream.
ReadU32();
693 data.m_UnknownPtr2 = aStream.
ReadU32();
694 data.m_Unknown2 = aStream.
ReadU32();
695 data.m_UnknownPtr3 = aStream.
ReadU32();
696 data.m_UnknownPtr4 = aStream.
ReadU32();
698 ReadCond( aStream, aVer, data.m_Unknown3 );
706 auto block = std::make_unique<BLOCK<BLK_0x0A_DRC>>( aStream.
Position() );
708 auto& data = block->GetData();
710 data.m_T = aStream.
ReadU8();
712 data.m_Key = aStream.
ReadU32();
713 block->SetKey( data.m_Key );
714 data.m_Next = aStream.
ReadU32();
715 block->SetNext( data.m_Next );
716 data.m_Unknown1 = aStream.
ReadU32();
718 ReadCond( aStream, aVer, data.m_Unknown2 );
720 for(
size_t i = 0; i < data.m_Coords.size(); ++i )
722 data.m_Coords[i] = aStream.
ReadS32();
728 ReadCond( aStream, aVer, data.m_Unknown6 );
736 auto block = std::make_unique<BLOCK<BLK_0x0C_PIN_DEF>>( aStream.
Position() );
738 auto& data = block->GetData();
740 data.m_T = aStream.
ReadU8();
743 data.m_Key = aStream.
ReadU32();
744 block->SetKey( data.m_Key );
745 data.m_Next = aStream.
ReadU32();
746 block->SetNext( data.m_Next );
748 data.m_Unknown1 = aStream.
ReadU32();
749 data.m_Unknown2 = aStream.
ReadU32();
752 ReadCond( aStream, aVer, data.m_Shape );
753 ReadCond( aStream, aVer, data.m_DrillChar );
754 ReadCond( aStream, aVer, data.m_UnknownPadding );
757 ReadCond( aStream, aVer, data.m_Shape16x );
758 ReadCond( aStream, aVer, data.m_DrillChars );
759 ReadCond( aStream, aVer, data.m_Unknown_16x );
761 data.m_Unknown4 = aStream.
ReadU32();
762 ReadCond( aStream, aVer, data.m_Unknown5 );
764 for(
size_t i = 0; i < data.m_Coords.size(); ++i )
766 data.m_Coords[i] = aStream.
ReadS32();
769 for(
size_t i = 0; i < data.m_Size.size(); ++i )
771 data.m_Size[i] = aStream.
ReadS32();
774 data.m_GroupPtr = aStream.
ReadU32();
775 data.m_Unknown6 = aStream.
ReadU32();
776 data.m_Unknown7 = aStream.
ReadU32();
778 ReadCond( aStream, aVer, data.m_Unknown8 );
786 auto block = std::make_unique<BLOCK<BLK_0x0D_PAD>>( aStream.
Position() );
788 auto& data = block->GetData();
792 data.m_Key = aStream.
ReadU32();
793 block->SetKey( data.m_Key );
794 data.m_NameStrId = aStream.
ReadU32();
795 data.m_Next = aStream.
ReadU32();
796 block->SetNext( data.m_Next );
798 ReadCond( aStream, aVer, data.m_Unknown1 );
800 data.m_CoordsX = aStream.
ReadS32();
801 data.m_CoordsY = aStream.
ReadS32();
803 data.m_PadStack = aStream.
ReadU32();
804 data.m_Unknown2 = aStream.
ReadU32();
806 ReadCond( aStream, aVer, data.m_Unknown3 );
808 data.m_Flags = aStream.
ReadU32();
809 data.m_Rotation = aStream.
ReadU32();
817 auto block = std::make_unique<BLOCK<BLK_0x0E_RECT>>( aStream.
Position() );
819 auto& data = block->GetData();
821 data.m_T = aStream.
ReadU8();
823 data.m_Key = aStream.
ReadU32();
824 block->SetKey( data.m_Key );
825 data.m_Next = aStream.
ReadU32();
826 block->SetNext( data.m_Next );
827 data.m_FpPtr = aStream.
ReadU32();
829 data.m_Unknown1 = aStream.
ReadU32();
830 data.m_Unknown2 = aStream.
ReadU32();
831 data.m_Unknown3 = aStream.
ReadU32();
833 ReadCond( aStream, aVer, data.m_Unknown4 );
834 ReadCond( aStream, aVer, data.m_Unknown5 );
836 for(
size_t i = 0; i < data.m_Coords.size(); ++i )
838 data.m_Coords[i] = aStream.
ReadS32();
841 for(
size_t i = 0; i < data.m_UnknownArr.size(); ++i )
843 data.m_UnknownArr[i] = aStream.
ReadU32();
846 data.m_Rotation = aStream.
ReadU32();
854 auto block = std::make_unique<BLOCK<BLK_0x0F_FUNCTION_SLOT>>( stream.
Position() );
856 auto& data = block->GetData();
861 block->SetKey( data.m_Key );
862 data.m_SlotName = stream.
ReadU32();
864 ReadCond( stream, aVer, data.m_Unknown1 );
866 stream.
ReadBytes( data.m_CompDeviceType.data(), data.m_CompDeviceType.size() );
868 ReadCond( stream, aVer, data.m_Next );
869 block->SetNext( data.m_Next.value_or( 0 ) );
871 data.m_Ptr0x06 = stream.
ReadU32();
872 data.m_Ptr0x11 = stream.
ReadU32();
873 data.m_Unknown2 = stream.
ReadU32();
881 auto block = std::make_unique<BLOCK<BLK_0x10_FUNCTION_INST>>( stream.
Position() );
883 auto& data = block->GetData();
888 block->SetKey( data.m_Key );
890 ReadCond( stream, aVer, data.m_Unknown1 );
891 data.m_ComponentInstPtr = stream.
ReadU32();
892 ReadCond( stream, aVer, data.m_Unknown2 );
893 data.m_PtrX12 = stream.
ReadU32();
894 data.m_Unknown3 = stream.
ReadU32();
895 data.m_FunctionName = stream.
ReadU32();
896 data.m_Slots = stream.
ReadU32();
897 data.m_Fields = stream.
ReadU32();
905 auto block = std::make_unique<BLOCK<BLK_0x11_PIN_NAME>>( aStream.
Position() );
907 auto& data = block->GetData();
909 data.m_Type = aStream.
ReadU8();
911 data.m_Key = aStream.
ReadU32();
912 block->SetKey( data.m_Key );
913 data.m_PinNameStrPtr = aStream.
ReadU32();
914 data.m_Next = aStream.
ReadU32();
915 block->SetNext( data.m_Next );
916 data.m_PinNumberPtr = aStream.
ReadU32();
917 data.m_Unknown1 = aStream.
ReadU32();
919 ReadCond( aStream, aVer, data.m_Unknown2 );
927 auto block = std::make_unique<BLOCK<BLK_0x12_XREF>>( aStream.
Position() );
929 auto& data = block->GetData();
931 data.m_Type = aStream.
ReadU8();
933 data.m_Key = aStream.
ReadU32();
934 block->SetKey( data.m_Key );
935 data.m_Ptr1 = aStream.
ReadU32();
936 data.m_Ptr2 = aStream.
ReadU32();
937 data.m_Ptr3 = aStream.
ReadU32();
938 data.m_Unknown1 = aStream.
ReadU32();
940 ReadCond( aStream, aVer, data.m_Unknown2 );
941 ReadCond( aStream, aVer, data.m_Unknown3 );
949 auto block = std::make_unique<BLOCK<BLK_0x14_GRAPHIC>>( aStream.
Position() );
951 auto& data = block->GetData();
953 data.m_Type = aStream.
ReadU8();
955 data.m_Key = aStream.
ReadU32();
956 block->SetKey( data.m_Key );
957 data.m_Next = aStream.
ReadU32();
958 block->SetNext( data.m_Next );
959 data.m_Parent = aStream.
ReadU32();
960 data.m_Flags = aStream.
ReadU32();
962 ReadCond( aStream, aVer, data.m_Unknown2 );
964 data.m_SegmentPtr = aStream.
ReadU32();
965 data.m_Ptr0x03 = aStream.
ReadU32();
966 data.m_Ptr0x26 = aStream.
ReadU32();
974 auto block = std::make_unique<BLOCK<BLK_0x15_16_17_SEGMENT>>( aType, aStream.
Position() );
976 auto& data = block->GetData();
980 data.m_Key = aStream.
ReadU32();
981 block->SetKey( data.m_Key );
982 data.m_Next = aStream.
ReadU32();
983 block->SetNext( data.m_Next );
984 data.m_Parent = aStream.
ReadU32();
985 data.m_Flags = aStream.
ReadU32();
987 ReadCond( aStream, aVer, data.m_Unknown2 );
989 data.m_Width = aStream.
ReadU32();
991 data.m_StartX = aStream.
ReadS32();
992 data.m_StartY = aStream.
ReadS32();
993 data.m_EndX = aStream.
ReadS32();
994 data.m_EndY = aStream.
ReadS32();
1002 auto block = std::make_unique<BLOCK<BLK_0x1B_NET>>( stream.
Position() );
1004 auto& data = block->GetData();
1008 data.m_Key = stream.
ReadU32();
1009 block->SetKey( data.m_Key );
1010 data.m_Next = stream.
ReadU32();
1011 block->SetNext( data.m_Next );
1012 data.m_NetName = stream.
ReadU32();
1013 data.m_Unknown1 = stream.
ReadU32();
1015 ReadCond( stream, aVer, data.m_Unknown2 );
1017 data.m_Type = stream.
ReadU32();
1018 data.m_Assignment = stream.
ReadU32();
1019 data.m_Ratline = stream.
ReadU32();
1020 data.m_FieldsPtr = stream.
ReadU32();
1021 data.m_MatchGroupPtr = stream.
ReadU32();
1022 data.m_ModelPtr = stream.
ReadU32();
1023 data.m_UnknownPtr4 = stream.
ReadU32();
1024 data.m_UnknownPtr5 = stream.
ReadU32();
1025 data.m_UnknownPtr6 = stream.
ReadU32();
1034 switch( (aVal & 0xF0) )
1051 THROW_IO_ERROR( wxString::Format(
"Unknown padstack type 0x%x", aVal ) );
1059 auto block = std::make_unique<BLOCK<BLK_0x1C_PADSTACK>>( aStream.
Position() );
1061 auto& data = block->GetData();
1063 data.m_UnknownByte1 = aStream.
ReadU8();
1064 data.m_N = aStream.
ReadU8();
1065 data.m_UnknownByte2 = aStream.
ReadU8();
1067 data.m_Key = aStream.
ReadU32();
1068 block->SetKey( data.m_Key );
1069 data.m_Next = aStream.
ReadU32();
1070 block->SetNext( data.m_Next );
1071 data.m_PadStr = aStream.
ReadU32();
1114 uint8_t padTypeAndA = aStream.
ReadU8();
1116 hdr.
m_A = ( padTypeAndA & 0x0F );
1156 static const uint16_t MAX_LAYER_COUNT = 256;
1157 if( data.GetLayerCount() > MAX_LAYER_COUNT )
1158 throw std::runtime_error(
"Layer count exceeds maximum of " + std::to_string( MAX_LAYER_COUNT ) );
1162 data.m_NumFixedCompEntries = 10;
1164 data.m_NumFixedCompEntries = 11;
1166 data.m_NumFixedCompEntries = 21;
1171 const size_t nComps = data.m_NumFixedCompEntries + ( data.GetLayerCount() * data.m_NumCompsPerLayer );
1173 data.m_Components.reserve( nComps );
1174 for(
size_t i = 0; i < nComps; ++i )
1204 const size_t nElems = data.m_N * ( ( aVer <
FMT_VER::V_172 ) ? 8 : 10 );
1206 data.m_UnknownArrN.reserve( nElems );
1207 for(
size_t i = 0; i < nElems; ++i )
1209 data.m_UnknownArrN.push_back( aStream.
ReadU32() );
1219 auto block = std::make_unique<BLOCK<BLK_0x1D_CONSTRAINT_SET>>( aStream.
Position() );
1221 auto& data = block->GetData();
1225 data.m_Key = aStream.
ReadU32();
1226 block->SetKey( data.m_Key );
1227 data.m_Next = aStream.
ReadU32();
1228 block->SetNext( data.m_Next );
1229 data.m_NameStrKey = aStream.
ReadU32();
1230 data.m_FieldPtr = aStream.
ReadU32();
1232 data.m_SizeA = aStream.
ReadU16();
1233 data.m_SizeB = aStream.
ReadU16();
1235 data.m_DataB.resize( data.m_SizeB );
1237 for(
auto& item : data.m_DataB )
1238 aStream.
ReadBytes( item.data(), item.size() );
1240 data.m_DataA.resize( data.m_SizeA );
1242 for(
auto& item : data.m_DataA )
1243 aStream.
ReadBytes( item.data(), item.size() );
1245 ReadCond( aStream, aVer, data.m_Unknown4 );
1253 auto block = std::make_unique<BLOCK<BLK_0x1E_SI_MODEL>>( aStream.
Position() );
1255 auto& data = block->GetData();
1257 data.m_Type = aStream.
ReadU8();
1258 data.m_T2 = aStream.
ReadU16();
1259 data.m_Key = aStream.
ReadU32();
1260 block->SetKey( data.m_Key );
1261 data.m_Next = aStream.
ReadU32();
1262 block->SetNext( data.m_Next );
1264 ReadCond( aStream, aVer, data.m_Unknown2 );
1265 ReadCond( aStream, aVer, data.m_Unknown3 );
1267 data.m_StrPtr = aStream.
ReadU32();
1268 data.m_Size = aStream.
ReadU32();
1272 ReadCond( aStream, aVer, data.m_Unknown4 );
1280 auto block = std::make_unique<BLOCK<BLK_0x1F_PADSTACK_DIM>>( aStream.
Position() );
1282 auto& data = block->GetData();
1286 data.m_Key = aStream.
ReadU32();
1287 block->SetKey( data.m_Key );
1289 data.m_Next = aStream.
ReadU32();
1290 block->SetNext( data.m_Next );
1291 data.m_Unknown2 = aStream.
ReadU32();
1292 data.m_Unknown3 = aStream.
ReadU32();
1293 data.m_Unknown4 = aStream.
ReadU32();
1294 data.m_Unknown5 = aStream.
ReadU16();
1296 data.m_Size = aStream.
ReadU16();
1298 size_t substructSize = 0;
1300 substructSize = data.m_Size * 384 + 8;
1302 substructSize = data.m_Size * 280 + 8;
1304 substructSize = data.m_Size * 280 + 4;
1306 substructSize = data.m_Size * 240 + 4;
1308 data.m_Substruct.resize( substructSize );
1309 aStream.
ReadBytes( data.m_Substruct.data(), substructSize );
1317 auto block = std::make_unique<BLOCK<BLK_0x20_UNKNOWN>>( aStream.
Position() );
1319 auto& data = block->GetData();
1321 data.m_Type = aStream.
ReadU8();
1323 data.m_Key = aStream.
ReadU32();
1324 block->SetKey( data.m_Key );
1325 data.m_Next = aStream.
ReadU32();
1326 block->SetNext( data.m_Next );
1329 ReadCond( aStream, aVer, data.m_UnknownArray2 );
1337 auto block = std::make_unique<BLOCK<BLK_0x21_BLOB>>( aStream.
Position() );
1339 auto& data = block->GetData();
1341 data.m_Type = aStream.
ReadU8();
1344 data.m_Size = aStream.
ReadU32();
1346 if( data.m_Size < 12 )
1348 THROW_IO_ERROR( wxString::Format(
"Block 0x21 size %u too small (minimum 12) at offset %#010zx",
1349 data.m_Size, aStream.
Position() ) );
1352 data.m_Key = aStream.
ReadU32();
1353 block->SetKey( data.m_Key );
1355 const size_t nBytes = data.m_Size - 12;
1356 data.m_Data.resize( nBytes );
1357 aStream.
ReadBytes( data.m_Data.data(), nBytes );
1365 auto block = std::make_unique<BLOCK<BLK_0x22_UNKNOWN>>( aStream.
Position() );
1367 auto& data = block->GetData();
1369 data.m_Type = aStream.
ReadU8();
1370 data.m_T2 = aStream.
ReadU16();
1371 data.m_Key = aStream.
ReadU32();
1372 block->SetKey( data.m_Key );
1374 ReadCond( aStream, aVer, data.m_Unknown1 );
1384 auto block = std::make_unique<BLOCK<BLK_0x23_RATLINE>>( aStream.
Position() );
1386 auto& data = block->GetData();
1388 data.m_Type = aStream.
ReadU8();
1390 data.m_Key = aStream.
ReadU32();
1391 block->SetKey( data.m_Key );
1392 data.m_Next = aStream.
ReadU32();
1393 block->SetNext( data.m_Next );
1397 data.m_Ptr1 = aStream.
ReadU32();
1398 data.m_Ptr2 = aStream.
ReadU32();
1399 data.m_Ptr3 = aStream.
ReadU32();
1401 for(
size_t i = 0; i < data.m_Coords.size(); ++i )
1403 data.m_Coords[i] = aStream.
ReadS32();
1408 ReadCond( aStream, aVer, data.m_Unknown2 );
1409 ReadCond( aStream, aVer, data.m_Unknown3 );
1417 auto block = std::make_unique<BLOCK<BLK_0x24_RECT>>( aStream.
Position() );
1419 auto& data = block->GetData();
1421 data.m_Type = aStream.
ReadU8();
1423 data.m_Key = aStream.
ReadU32();
1424 block->SetKey( data.m_Key );
1425 data.m_Next = aStream.
ReadU32();
1426 block->SetNext( data.m_Next );
1427 data.m_Parent = aStream.
ReadU32();
1428 data.m_Unknown1 = aStream.
ReadU32();
1430 ReadCond( aStream, aVer, data.m_Unknown2 );
1432 for(
size_t i = 0; i < data.m_Coords.size(); ++i )
1434 data.m_Coords[i] = aStream.
ReadS32();
1437 data.m_Ptr2 = aStream.
ReadU32();
1439 data.m_Unknown3 = aStream.
ReadU32();
1440 data.m_Unknown4 = aStream.
ReadU32();
1441 data.m_Rotation = aStream.
ReadU32();
1449 auto block = std::make_unique<BLOCK<BLK_0x26_MATCH_GROUP>>( aStream.
Position() );
1451 auto& data = block->GetData();
1453 data.m_Type = aStream.
ReadU8();
1455 data.m_Key = aStream.
ReadU32();
1456 block->SetKey( data.m_Key );
1457 data.m_MemberPtr = aStream.
ReadU32();
1459 ReadCond( aStream, aVer, data.m_Unknown1 );
1461 data.m_GroupPtr = aStream.
ReadU32();
1462 data.m_ConstPtr = aStream.
ReadU32();
1464 ReadCond( aStream, aVer, data.m_Unknown2 );
1472 auto block = std::make_unique<BLOCK<BLK_0x27_CSTRMGR_XREF>>( aStream.
Position() );
1474 auto& data = block->GetData();
1476 const size_t totalBytes = aEndOff - 1 - aStream.
Position();
1479 constexpr size_t kPadding = 3;
1481 if( totalBytes <= kPadding )
1483 aStream.
Skip( totalBytes );
1487 aStream.
Skip( kPadding );
1489 const size_t payloadBytes = totalBytes - kPadding;
1490 const size_t numValues = payloadBytes / 4;
1491 const size_t remainder = payloadBytes % 4;
1493 data.m_Refs.resize( numValues );
1495 for(
size_t i = 0; i < numValues; i++ )
1496 data.m_Refs[i] = aStream.
ReadU32();
1499 aStream.
Skip( remainder );
1507 auto block = std::make_unique<BLOCK<BLK_0x28_SHAPE>>( aStream.
Position() );
1509 auto& data = block->GetData();
1511 data.m_Type = aStream.
ReadU8();
1513 data.m_Key = aStream.
ReadU32();
1514 block->SetKey( data.m_Key );
1515 data.m_Next = aStream.
ReadU32();
1516 block->SetNext( data.m_Next );
1517 data.m_Ptr1 = aStream.
ReadU32();
1518 data.m_Unknown1 = aStream.
ReadU32();
1520 ReadCond( aStream, aVer, data.m_Unknown2 );
1521 ReadCond( aStream, aVer, data.m_Unknown3 );
1523 data.m_Ptr2 = aStream.
ReadU32();
1524 data.m_Ptr3 = aStream.
ReadU32();
1525 data.m_FirstKeepoutPtr = aStream.
ReadU32();
1526 data.m_FirstSegmentPtr = aStream.
ReadU32();
1527 data.m_Unknown4 = aStream.
ReadU32();
1528 data.m_Unknown5 = aStream.
ReadU32();
1530 ReadCond( aStream, aVer, data.m_TablePtr );
1532 data.m_Ptr6 = aStream.
ReadU32();
1534 ReadCond( aStream, aVer, data.m_TablePtr_16x );
1536 for(
size_t i = 0; i < data.m_Coords.size(); ++i )
1538 data.m_Coords[i] = aStream.
ReadS32();
1547 auto block = std::make_unique<BLOCK<BLK_0x29_PIN>>( aStream.
Position() );
1549 auto& data = block->GetData();
1551 data.m_Type = aStream.
ReadU8();
1553 data.m_Key = aStream.
ReadU32();
1554 block->SetKey( data.m_Key );
1556 data.m_Ptr1 = aStream.
ReadU32();
1557 data.m_Ptr2 = aStream.
ReadU32();
1559 data.m_Null = aStream.
ReadU32();
1561 data.m_Ptr3 = aStream.
ReadU32();
1563 data.m_Coord1 = aStream.
ReadS32();
1564 data.m_Coord2 = aStream.
ReadS32();
1566 data.m_PtrPadstack = aStream.
ReadU32();
1568 data.m_Unknown1 = aStream.
ReadU32();
1570 data.m_PtrX30 = aStream.
ReadU32();
1572 data.m_Unknown2 = aStream.
ReadU32();
1573 data.m_Unknown3 = aStream.
ReadU32();
1574 data.m_Unknown4 = aStream.
ReadU32();
1582 auto block = std::make_unique<BLOCK<BLK_0x2A_LAYER_LIST>>( aStream.
Position() );
1584 auto& data = block->GetData();
1588 data.m_NumEntries = aStream.
ReadU16();
1590 ReadCond( aStream, aVer, data.m_Unknown );
1592 if( data.m_NonRefEntries.exists( aVer ) )
1594 data.m_NonRefEntries = std::vector<BLK_0x2A_LAYER_LIST::NONREF_ENTRY>();
1595 data.m_NonRefEntries->reserve( data.m_NumEntries );
1596 for(
size_t i = 0; i < data.m_NumEntries; ++i )
1605 data.m_RefEntries = std::vector<BLK_0x2A_LAYER_LIST::REF_ENTRY>();
1606 data.m_RefEntries->reserve( data.m_NumEntries );
1607 for(
size_t i = 0; i < data.m_NumEntries; ++i )
1617 data.m_Key = aStream.
ReadU32();
1618 block->SetKey( data.m_Key );
1626 auto block = std::make_unique<BLOCK<BLK_0x2B_FOOTPRINT_DEF>>( stream.
Position() );
1628 auto& data = block->GetData();
1632 data.m_Key = stream.
ReadU32();
1633 block->SetKey( data.m_Key );
1634 data.m_FpStrRef = stream.
ReadU32();
1635 data.m_Unknown1 = stream.
ReadU32();
1637 data.m_Next = stream.
ReadU32();
1638 block->SetNext( data.m_Next );
1639 data.m_FirstInstPtr = stream.
ReadU32();
1640 data.m_UnknownPtr3 = stream.
ReadU32();
1641 data.m_UnknownPtr4 = stream.
ReadU32();
1642 data.m_UnknownPtr5 = stream.
ReadU32();
1643 data.m_SymLibPathPtr = stream.
ReadU32();
1644 data.m_UnknownPtr6 = stream.
ReadU32();
1645 data.m_UnknownPtr7 = stream.
ReadU32();
1646 data.m_UnknownPtr8 = stream.
ReadU32();
1648 ReadCond( stream, aVer, data.m_Unknown2 );
1649 ReadCond( stream, aVer, data.m_Unknown3 );
1657 auto block = std::make_unique<BLOCK<BLK_0x2C_TABLE>>( aStream.
Position() );
1659 auto& data = block->GetData();
1661 data.m_Type = aStream.
ReadU8();
1662 data.m_SubType = aStream.
ReadU16();
1663 data.m_Key = aStream.
ReadU32();
1664 block->SetKey( data.m_Key );
1665 data.m_Next = aStream.
ReadU32();
1666 block->SetNext( data.m_Next );
1668 ReadCond( aStream, aVer, data.m_Unknown1 );
1669 ReadCond( aStream, aVer, data.m_Unknown2 );
1670 ReadCond( aStream, aVer, data.m_Unknown3 );
1672 data.m_StringPtr = aStream.
ReadU32();
1674 ReadCond( aStream, aVer, data.m_Unknown4 );
1676 data.m_Ptr1 = aStream.
ReadU32();
1677 data.m_Ptr2 = aStream.
ReadU32();
1678 data.m_Ptr3 = aStream.
ReadU32();
1680 data.m_Flags = aStream.
ReadU32();
1688 auto block = std::make_unique<BLOCK<BLK_0x2D_FOOTPRINT_INST>>( stream.
Position() );
1690 auto& data = block->GetData();
1692 data.m_UnknownByte1 = stream.
ReadU8();
1693 data.m_Layer = stream.
ReadU8();
1694 data.m_UnknownByte2 = stream.
ReadU8();
1696 data.m_Key = stream.
ReadU32();
1697 block->SetKey( data.m_Key );
1698 data.m_Next = stream.
ReadU32();
1699 block->SetNext( data.m_Next );
1701 ReadCond( stream, aVer, data.m_Unknown1 );
1703 ReadCond( stream, aVer, data.m_InstRef16x );
1705 data.m_Unknown2 = stream.
ReadU16();
1706 data.m_Unknown3 = stream.
ReadU16();
1708 ReadCond( stream, aVer, data.m_Unknown4 );
1710 data.m_Flags = stream.
ReadU32();
1712 data.m_Rotation = stream.
ReadU32();
1713 data.m_CoordX = stream.
ReadS32();
1714 data.m_CoordY = stream.
ReadS32();
1716 ReadCond( stream, aVer, data.m_InstRef );
1718 data.m_GraphicPtr = stream.
ReadU32();
1719 data.m_FirstPadPtr = stream.
ReadU32();
1720 data.m_TextPtr = stream.
ReadU32();
1722 data.m_AssemblyPtr = stream.
ReadU32();
1723 data.m_AreasPtr = stream.
ReadU32();
1724 data.m_UnknownPtr1 = stream.
ReadU32();
1725 data.m_UnknownPtr2 = stream.
ReadU32();
1733 auto block = std::make_unique<BLOCK<BLK_0x2E_CONNECTION>>( aStream.
Position() );
1735 auto& data = block->GetData();
1737 data.m_Type = aStream.
ReadU8();
1738 data.m_T2 = aStream.
ReadU16();
1739 data.m_Key = aStream.
ReadU32();
1740 block->SetKey( data.m_Key );
1741 data.m_Next = aStream.
ReadU32();
1742 block->SetNext( data.m_Next );
1743 data.m_NetAssignment = aStream.
ReadU32();
1744 data.m_Unknown1 = aStream.
ReadU32();
1745 data.m_CoordX = aStream.
ReadU32();
1746 data.m_CoordY = aStream.
ReadU32();
1747 data.m_Connection = aStream.
ReadU32();
1748 data.m_Unknown2 = aStream.
ReadU32();
1750 ReadCond( aStream, aVer, data.m_Unknown3 );
1758 auto block = std::make_unique<BLOCK<BLK_0x2F_UNKNOWN>>( aStream.
Position() );
1760 auto& data = block->GetData();
1762 data.m_Type = aStream.
ReadU8();
1763 data.m_T2 = aStream.
ReadU16();
1764 data.m_Key = aStream.
ReadU32();
1765 block->SetKey( data.m_Key );
1780 uint8_t alignment = aStream.
ReadU8();
1792 uint8_t reversal = aStream.
ReadU8();
1808 auto block = std::make_unique<BLOCK<BLK_0x30_STR_WRAPPER>>( aStream.
Position() );
1810 auto& data = block->GetData();
1812 data.m_Type = aStream.
ReadU8();
1814 data.m_Key = aStream.
ReadU32();
1815 block->SetKey( data.m_Key );
1816 data.m_Next = aStream.
ReadU32();
1817 block->SetNext( data.m_Next );
1819 ReadCond( aStream, aVer, data.m_Unknown1 );
1820 ReadCond( aStream, aVer, data.m_Unknown2 );
1822 if( data.m_Font.exists( aVer ) )
1827 ReadCond( aStream, aVer, data.m_Ptr1 );
1828 ReadCond( aStream, aVer, data.m_Unknown3 );
1830 data.m_StrGraphicPtr = aStream.
ReadU32();
1832 ReadCond( aStream, aVer, data.m_PtrGroup_17x );
1833 ReadCond( aStream, aVer, data.m_Unknown4 );
1835 if( data.m_Font16x.exists( aVer ) )
1840 ReadCond( aStream, aVer, data.m_Ptr2 );
1842 data.m_CoordsX = aStream.
ReadU32();
1843 data.m_CoordsY = aStream.
ReadU32();
1845 data.m_Unknown5 = aStream.
ReadU32();
1846 data.m_Rotation = aStream.
ReadU32();
1848 ReadCond( aStream, aVer, data.m_PtrGroup_16x );
1856 auto block = std::make_unique<BLOCK<BLK_0x31_SGRAPHIC>>( aStream.
Position() );
1858 auto& data = block->GetData();
1860 data.m_T = aStream.
ReadU8();
1863 const uint16_t layer = aStream.
ReadU16();
1878 data.m_Key = aStream.
ReadU32();
1879 block->SetKey( data.m_Key );
1880 data.m_StrGraphicWrapperPtr = aStream.
ReadU32();
1882 data.m_CoordsX = aStream.
ReadU32();
1883 data.m_CoordsY = aStream.
ReadU32();
1885 data.m_Unknown = aStream.
ReadU16();
1886 data.m_Len = aStream.
ReadU16();
1888 ReadCond( aStream, aVer, data.m_Un2 );
1898 auto block = std::make_unique<BLOCK<BLK_0x32_PLACED_PAD>>( aStream.
Position() );
1900 auto& data = block->GetData();
1902 data.m_Type = aStream.
ReadU8();
1904 data.m_Key = aStream.
ReadU32();
1905 block->SetKey( data.m_Key );
1906 data.m_Next = aStream.
ReadU32();
1907 block->SetNext( data.m_Next );
1908 data.m_NetPtr = aStream.
ReadU32();
1909 data.m_Flags = aStream.
ReadU32();
1911 ReadCond( aStream, aVer, data.m_Prev );
1913 data.m_NextInFp = aStream.
ReadU32();
1914 data.m_ParentFp = aStream.
ReadU32();
1915 data.m_Track = aStream.
ReadU32();
1916 data.m_PadPtr = aStream.
ReadU32();
1917 data.m_Ptr6 = aStream.
ReadU32();
1918 data.m_Ratline = aStream.
ReadU32();
1919 data.m_PtrPinNumber = aStream.
ReadU32();
1920 data.m_NextInCompInst = aStream.
ReadU32();
1922 ReadCond( aStream, aVer, data.m_Unknown2 );
1924 data.m_NameText = aStream.
ReadU32();
1925 data.m_Ptr11 = aStream.
ReadU32();
1927 for(
size_t i = 0; i < data.m_Coords.size(); ++i )
1929 data.m_Coords[i] = aStream.
ReadS32();
1938 auto block = std::make_unique<BLOCK<BLK_0x33_VIA>>( aStream.
Position() );
1940 auto& data = block->GetData();
1945 data.m_Key = aStream.
ReadU32();
1946 block->SetKey( data.m_Key );
1947 data.m_Next = aStream.
ReadU32();
1948 block->SetNext( data.m_Next );
1949 data.m_NetPtr = aStream.
ReadU32();
1950 data.m_Unknown2 = aStream.
ReadU32();
1952 ReadCond( aStream, aVer, data.m_Unknown3 );
1954 data.m_UnknownPtr1 = aStream.
ReadU32();
1956 ReadCond( aStream, aVer, data.m_UnknownPtr2 );
1958 data.m_CoordsX = aStream.
ReadS32();
1959 data.m_CoordsY = aStream.
ReadS32();
1961 data.m_Connection = aStream.
ReadU32();
1962 data.m_Padstack = aStream.
ReadU32();
1963 data.m_UnknownPtr5 = aStream.
ReadU32();
1964 data.m_UnknownPtr6 = aStream.
ReadU32();
1966 data.m_Unknown4 = aStream.
ReadU32();
1967 data.m_Unknown5 = aStream.
ReadU32();
1969 for(
size_t i = 0; i < data.m_BoundingBoxCoords.size(); ++i )
1971 data.m_BoundingBoxCoords[i] = aStream.
ReadS32();
1980 auto block = std::make_unique<BLOCK<BLK_0x34_KEEPOUT>>( aStream.
Position() );
1982 auto& data = block->GetData();
1984 data.m_T = aStream.
ReadU8();
1986 data.m_Key = aStream.
ReadU32();
1987 block->SetKey( data.m_Key );
1988 data.m_Next = aStream.
ReadU32();
1989 block->SetNext( data.m_Next );
1990 data.m_Ptr1 = aStream.
ReadU32();
1992 ReadCond( aStream, aVer, data.m_Unknown1 );
1994 data.m_Flags = aStream.
ReadU32();
1995 data.m_FirstSegmentPtr = aStream.
ReadU32();
1996 data.m_Ptr3 = aStream.
ReadU32();
1997 data.m_Unknown2 = aStream.
ReadU32();
2005 auto block = std::make_unique<BLOCK<BLK_0x35_FILE_REF>>( aStream.
Position() );
2007 auto& data = block->GetData();
2009 data.m_T2 = aStream.
ReadU8();
2010 data.m_T3 = aStream.
ReadU16();
2011 aStream.
ReadBytes( data.m_Content.data(), data.m_Content.size() );
2019 auto block = std::make_unique<BLOCK<BLK_0x36_DEF_TABLE>>( aStream.
Position() );
2021 auto& data = block->GetData();
2025 data.m_Code = aStream.
ReadU16();
2026 data.m_Key = aStream.
ReadU32();
2027 block->SetKey( data.m_Key );
2028 data.m_Next = aStream.
ReadU32();
2029 block->SetNext( data.m_Next );
2031 ReadCond( aStream, aVer, data.m_Unknown1 );
2033 data.m_NumItems = aStream.
ReadU32();
2034 data.m_Count = aStream.
ReadU32();
2035 data.m_LastIdx = aStream.
ReadU32();
2036 data.m_Unknown2 = aStream.
ReadU32();
2038 ReadCond( aStream, aVer, data.m_Unknown3 );
2040 if( data.m_NumItems > 1000000 )
2043 "Block 0x36 item count %u exceeds limit at offset %#010zx",
2044 data.m_NumItems, aStream.
Position() ) );
2047 if( data.m_Count > data.m_NumItems )
2050 "Block 0x36 filled count %u exceeds capacity %u at offset %#010zx",
2051 data.m_Count, data.m_NumItems, aStream.
Position() ) );
2057 data.m_Items.reserve( data.m_Count );
2058 for( uint32_t i = 0; i < data.m_NumItems; ++i )
2060 const bool keep = i < data.m_Count;
2062 switch( data.m_Code )
2074 data.m_Items.emplace_back( std::move( item ) );
2088 data.m_Items.emplace_back( std::move( item ) );
2099 data.m_Items.emplace_back( std::move( item ) );
2114 data.m_Items.emplace_back( std::move( item ) );
2136 data.m_Items.emplace_back( std::move( item ) );
2144 data.m_Items.emplace_back( std::move( item ) );
2152 data.m_Items.emplace_back( std::move( item ) );
2160 data.m_Items.emplace_back( std::move( item ) );
2170 data.m_Items.emplace_back( std::move( item ) );
2179 data.m_Items.emplace_back( std::move( item ) );
2186 aStream.
Skip( 1052 );
2188 data.m_Items.emplace_back( std::move( item ) );
2191 default:
THROW_IO_ERROR( wxString::Format(
"Unknown substruct type %#02x in block 0x36", data.m_Code ) );
2201 auto block = std::make_unique<BLOCK<BLK_0x37_PTR_ARRAY>>( aStream.
Position() );
2203 auto& data = block->GetData();
2205 data.m_T = aStream.
ReadU8();
2206 data.m_T2 = aStream.
ReadU16();
2207 data.m_Key = aStream.
ReadU32();
2208 block->SetKey( data.m_Key );
2209 data.m_GroupPtr = aStream.
ReadU32();
2210 data.m_Next = aStream.
ReadU32();
2211 block->SetNext( data.m_Next );
2212 data.m_Capacity = aStream.
ReadU32();
2213 data.m_Count = aStream.
ReadU32();
2214 data.m_Unknown2 = aStream.
ReadU32();
2216 ReadCond( aStream, aVer, data.m_Unknown3 );
2226 auto block = std::make_unique<BLOCK<BLK_0x38_FILM>>( aStream.
Position() );
2228 auto& data = block->GetData();
2232 data.m_Key = aStream.
ReadU32();
2233 block->SetKey( data.m_Key );
2234 data.m_Next = aStream.
ReadU32();
2235 block->SetNext( data.m_Next );
2236 data.m_LayerList = aStream.
ReadU32();
2238 if( data.m_FilmName.exists( aVer ) )
2243 ReadCond( aStream, aVer, data.m_LayerNameStr );
2244 ReadCond( aStream, aVer, data.m_Unknown2 );
2246 for(
size_t i = 0; i < data.m_UnknownArray1.size(); ++i )
2248 data.m_UnknownArray1[i] = aStream.
ReadU32();
2251 ReadCond( aStream, aVer, data.m_Unknown3 );
2259 auto block = std::make_unique<BLOCK<BLK_0x39_FILM_LAYER_LIST>>( aStream.
Position() );
2261 auto& data = block->GetData();
2265 data.m_Key = aStream.
ReadU32();
2266 block->SetKey( data.m_Key );
2267 data.m_Parent = aStream.
ReadU32();
2268 data.m_Head = aStream.
ReadU32();
2270 for(
size_t i = 0; i < data.m_X.size(); ++i )
2272 data.m_X[i] = aStream.
ReadU16();
2281 auto block = std::make_unique<BLOCK<BLK_0x3A_FILM_LIST_NODE>>( aStream.
Position() );
2283 auto& data = block->GetData();
2288 data.m_Key = aStream.
ReadU32();
2289 block->SetKey( data.m_Key );
2290 data.m_Next = aStream.
ReadU32();
2291 block->SetNext( data.m_Next );
2292 data.m_Unknown = aStream.
ReadU32();
2294 ReadCond( aStream, aVer, data.m_Unknown1 );
2302 auto block = std::make_unique<BLOCK<BLK_0x3B_PROPERTY>>( aStream.
Position() );
2304 auto& data = block->GetData();
2306 data.m_T = aStream.
ReadU8();
2307 data.m_SubType = aStream.
ReadU16();
2308 data.m_Len = aStream.
ReadU32();
2313 data.m_Unknown1 = aStream.
ReadU32();
2314 data.m_Unknown2 = aStream.
ReadU32();
2316 ReadCond( aStream, aVer, data.m_Unknown3 );
2326 auto block = std::make_unique<BLOCK<BLK_0x3C_KEY_LIST>>( aStream.
Position() );
2328 auto& data = block->GetData();
2330 data.m_T = aStream.
ReadU8();
2331 data.m_T2 = aStream.
ReadU16();
2332 data.m_Key = aStream.
ReadU32();
2333 block->SetKey( data.m_Key );
2335 ReadCond( aStream, aVer, data.m_Unknown );
2337 data.m_NumEntries = aStream.
ReadU32();
2339 if( data.m_NumEntries > 1000000 )
2342 "Block 0x3C entry count %u exceeds limit at offset %#010zx",
2343 data.m_NumEntries, aStream.
Position() ) );
2346 data.m_Entries.reserve( data.m_NumEntries );
2347 for( uint32_t i = 0; i < data.m_NumEntries; ++i )
2349 data.m_Entries.push_back( aStream.
ReadU32() );
2360 uint8_t type = 0x00;
2363 aEndOfObjectsMarker =
true;
2367 std::unique_ptr<BLOCK_BASE> block;
2523 wxString::Format(
"Current offset %#010zx is at or past the expected end of block 0x27 at %#010zx",
2637 aEndOfObjectsMarker =
true;
2661 THROTTLE refreshThrottle( std::chrono::milliseconds( 100 ) );
2665 const size_t offset =
m_stream.Position();
2668 wxASSERT_MSG( offset % 4 == 0,
2669 wxString::Format(
"Allegro object at %#010zx, offset not aligned to 4 bytes", offset ) );
2674 uint8_t blockTypeByte = 0;
2678 bool endOfObjectsMarker =
false;
2679 std::unique_ptr<BLOCK_BASE> block = blockParser.
ParseBlock( endOfObjectsMarker );
2681 if( endOfObjectsMarker )
2687 size_t scanPos =
m_stream.Position();
2688 uint8_t nextByte = 0;
2690 while(
m_stream.GetU8( nextByte ) && nextByte == 0x00 )
2693 if( nextByte > 0x00 && nextByte <= 0x3C )
2695 size_t blockStart = scanPos;
2697 if( blockStart % 4 != 0 )
2698 blockStart -= ( blockStart % 4 );
2703 uint8_t alignedByte = 0;
2708 if( alignedByte == 0x00 || alignedByte > 0x3C )
2712 wxString::Format(
"V18 zero gap from %#010zx to %#010zx, "
2713 "continuing at block type %#04x",
2714 offset, blockStart, nextByte ) );
2722 wxString::Format(
"End of objects marker (0x00) at index %zu, offset %#010zx",
2734 "Do not have parser for block index %zu type %#02x available at offset %#010zx",
2740 wxString::Format(
"Ending at unknown block, index %zu type %#04x at offset %#010zx",
2743 wxFAIL_MSG(
"Failed to create block" );
2750 wxString::Format(
"Added block %zu, type %#04x from %#010zx to %#010zx",
2768template <
typename T>
2771 if constexpr( std::is_same_v<T, FILE_HEADER::LINKED_LIST> )
2775 else if constexpr( VERSIONED_COND_FIELD<T> &&
2776 std::is_same_v<typename T::value_type, FILE_HEADER::LINKED_LIST> )
2778 if( aLL.has_value() )
2786 std::unique_ptr<BRD_DB> board = std::make_unique<BRD_DB>();
2801 if( !board->m_Header )
2810 board->m_Header->m_Magic );
2812 dumpLL(
"V18_1", board->m_Header->m_LL_V18_1 );
2813 dumpLL(
"V18_2", board->m_Header->m_LL_V18_2 );
2814 dumpLL(
"V18_3", board->m_Header->m_LL_V18_3 );
2815 dumpLL(
"V18_4", board->m_Header->m_LL_V18_4 );
2816 dumpLL(
"V18_5", board->m_Header->m_LL_V18_5 );
2818 dumpLL(
"0x04", board->m_Header->m_LL_0x04 );
2819 dumpLL(
"0x06", board->m_Header->m_LL_0x06 );
2820 dumpLL(
"0x0C", board->m_Header->m_LL_0x0C );
2821 dumpLL(
"Shapes", board->m_Header->m_LL_Shapes );
2822 dumpLL(
"0x14", board->m_Header->m_LL_0x14 );
2823 dumpLL(
"0x1B_Nets", board->m_Header->m_LL_0x1B_Nets );
2824 dumpLL(
"0x1C", board->m_Header->m_LL_0x1C );
2825 dumpLL(
"0x24_0x28", board->m_Header->m_LL_0x24_0x28 );
2826 dumpLL(
"Unknown1", board->m_Header->m_LL_Unknown1 );
2827 dumpLL(
"0x2B", board->m_Header->m_LL_0x2B );
2828 dumpLL(
"0x03_0x30", board->m_Header->m_LL_0x03_0x30 );
2829 dumpLL(
"0x0A", board->m_Header->m_LL_0x0A );
2830 dumpLL(
"0x1D_0x1E_0x1F", board->m_Header->m_LL_0x1D_0x1E_0x1F );
2831 dumpLL(
"Unknown2", board->m_Header->m_LL_Unknown2 );
2832 dumpLL(
"0x38", board->m_Header->m_LL_0x38 );
2833 dumpLL(
"0x2C", board->m_Header->m_LL_0x2C );
2834 dumpLL(
"0x0C_2", board->m_Header->m_LL_0x0C_2 );
2835 dumpLL(
"Unknown3", board->m_Header->m_LL_Unknown3 );
2836 dumpLL(
"Unknown5", board->m_Header->m_LL_Unknown5_preV18 );
2837 dumpLL(
"Unknown5", board->m_Header->m_LL_Unknown5_V18 );
2838 dumpLL(
"0x36", board->m_Header->m_LL_0x36 );
2839 dumpLL(
"Unknown6", board->m_Header->m_LL_Unknown6 );
2840 dumpLL(
"0x0A_2", board->m_Header->m_LL_0x0A_2 );
2842 dumpLL(
"V18_6", board->m_Header->m_LL_V18_6 );
2848 s += wxString::Format(
"Error parsing Allegro file: %s\n", e.
What() );
2849 s += wxString::Format(
"Stream position: %#010lx\n",
m_stream.Position() );
2851 if( board->m_Header )
2852 s += wxString::Format(
"File magic: %#010x\n", board->m_Header->m_Magic );
2854 s += wxString::Format(
"File magic: Unknown\n" );
2863 wxString verStr( board->m_Header->m_AllegroVersion.data(), 60 );
2867 _(
"This file was created with %s, which uses a binary format that "
2868 "predates Allegro 16.0 and is not supported by this importer.\n\n"
2869 "To import this design, open it in Cadence Allegro PCB Editor "
2870 "version 16.0 or later and re-save, then import the resulting file." ),
2874 const uint32_t stringsCount = board->m_Header->GetStringsCount();
2875 board->ReserveCapacity( board->m_Header->m_ObjectCount, stringsCount );
2881 wxLogTrace(
traceAllegroPerf, wxT(
" ReadStringMap (%u strings): %.3f ms" ),
2882 stringsCount, parseTimer.
msecs(
true ) );
2886 wxLogTrace(
traceAllegroPerf, wxT(
" readObjects (%zu objects): %.3f ms" ),
2887 board->GetObjectCount(), parseTimer.
msecs(
true ) );
2892 s += wxString::Format(
"Error parsing Allegro file: %s\n", e.
What() );
2893 s += wxString::Format(
"Stream position: %#010lx\n",
m_stream.Position() );
2895 if( board->m_Header )
2896 s += wxString::Format(
"File magic: %#010x\n", board->m_Header->m_Magic );
2898 s += wxString::Format(
"File magic: Unknown\n" );
2904 board->ResolveAndValidate();
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x35(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x36(FILE_STREAM &aStream, FMT_VER aVer)
void dumpLL(const char *name, const T &aLL)
static FILE_HEADER::LINKED_LIST ReadLL(FILE_STREAM &aStream, FMT_VER aVer)
static PAD_TYPE decodePadType(uint8_t aVal)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x04_NET_ASSIGNMENT(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x2C_TABLE(FILE_STREAM &aStream, FMT_VER aVer)
static void ReadStringMap(FILE_STREAM &stream, BRD_DB &aDb, uint32_t count)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x1F(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x3A_FILM_LIST_NODE(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x3C(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x14(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x08(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x10(FILE_STREAM &stream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x1B_NET(FILE_STREAM &stream, FMT_VER aVer)
static const wxChar *const traceAllegroParserBlocks
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x0F(FILE_STREAM &stream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x05_TRACK(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x1D(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x1C_PADSTACK(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x34_KEEPOUT(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x01_ARC(FILE_STREAM &aStream, FMT_VER aVer)
static BLK_0x30_STR_WRAPPER::TEXT_PROPERTIES ParseTextProps(FILE_STREAM &aStream)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x0D_PAD(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x1E(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x20(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x0A_DRC(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x06(FILE_STREAM &stream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x0E(FILE_STREAM &aStream, FMT_VER aVer)
static void ReadArrayU32(FILE_STREAM &aStream, ARRAY &aArray)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x39_FILM_LAYER_LIST(FILE_STREAM &aStream, FMT_VER aVer)
static void ReadCond(FILE_STREAM &aStream, FMT_VER aFmtVer, COND_T &aField)
Read a single conditional field from the stream, if it exists at the current version.
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x09(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x38_FILM(FILE_STREAM &aStream, FMT_VER aVer)
static double ReadAllegroFloat(FILE_STREAM &aStream)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x24_RECT(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x2E(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x03(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x12(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x15_16_17_SEGMENT(FILE_STREAM &aStream, FMT_VER aVer, uint8_t aType)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x31_SGRAPHIC(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x22(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x27(FILE_STREAM &aStream, FMT_VER aVer, size_t aEndOff)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x2A(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x2F(FILE_STREAM &aStream, FMT_VER aVer)
static LAYER_INFO ParseLayerInfo(FILE_STREAM &aStream)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x07(FILE_STREAM &stream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x33_VIA(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x32_PLACED_PAD(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x37(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x2B(FILE_STREAM &stream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x2D(FILE_STREAM &stream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x21(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x23_RATLINE(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x28_SHAPE(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x11(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x26(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x3B(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x0C(FILE_STREAM &aStream, FMT_VER aVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x29_PIN(FILE_STREAM &aStream, FMT_VER aVer)
static T ReadField(FILE_STREAM &aStream, FMT_VER aFmtVer)
static std::unique_ptr< BLOCK_BASE > ParseBlock_0x30_STR_WRAPPER(FILE_STREAM &aStream, FMT_VER aVer)
The block parser is responsible for parsing individual blocks of data from the file stream.
std::unique_ptr< BLOCK_BASE > ParseBlock(bool &aEndOfObjectsMarker)
Parse one block from the stream, returning a BLOCK_BASE representing the raw data of the block.
const size_t m_x27_end
To parse an 0x27 block, we need to know where the end of the block is in the stream.
An Allegro board database representing the contents of a .brd (and presumably .dra) file.
std::unique_ptr< FILE_HEADER > m_Header
size_t GetObjectCount() const
void InsertBlock(std::unique_ptr< BLOCK_BASE > aBlock)
void AddString(uint32_t aKey, wxString &&aStr)
Stream that reads primitive types from a memory buffer containing Allegro .brd (or ....
void ReadBytes(void *aDest, size_t aSize)
Read a number of bytes from the stream into the destination buffer.
std::string ReadString(bool aRoundToNextU32)
std::string ReadStringFixed(size_t aLen, bool aRoundToNextU32)
PROGRESS_REPORTER * m_progressReporter
void readObjects(BRD_DB &aDb)
std::unique_ptr< BRD_DB > Parse()
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
virtual const wxString What() const
A composite of Problem() and Where()
A small class to help profiling.
double msecs(bool aSinceLast=false)
Rate-limiter that fires at most once per interval.
static const wxChar *const traceAllegroParser
Flag to enable debug output of Allegro parsing.
#define THROW_IO_ERROR(msg)
macro which captures the "call site" values of FILE_, __FUNCTION & LINE
PAD_TYPE
The type of the padstack.
FMT_VER
The format of an Allegro file.
static const wxChar *const traceAllegroPerf
std::vector< uint32_t > m_Entries
std::vector< uint32_t > m_Entries
std::array< uint32_t, 20 > m_Entries
TEXT_ALIGNMENT m_Alignment
COND_GE< FMT_VER::V_172, std::array< uint32_t, 8 > > m_Ys
uint32_t m_CharacterSpace
COND_GE< FMT_VER::V_174, uint32_t > m_Unknown2
std::array< uint32_t, 14 > m_Xs
COND_GE< FMT_VER::V_172, std::array< uint32_t, 2 > > m_Zs
COND_GE< FMT_VER::V_164, std::array< uint32_t, 3 > > m_Ys
COND_GE< FMT_VER::V_172, std::string > m_Str
COND_GE< FMT_VER::V_174, uint32_t > m_Unknown1
COND_LT< FMT_VER::V_172, std::string > m_Str16x
std::array< uint8_t, 28 > m_Unknown
COND_GE< FMT_VER::V_174, uint32_t > m_Unknown2
COND_LT< FMT_VER::V_172, std::array< uint32_t, 50 > > m_Unknown2
std::array< uint8_t, 1016 > m_Unknown
std::array< uint8_t, 232 > m_Unknown
std::array< uint8_t, 200 > m_Unknown
std::array< uint32_t, 3 > m_Ptrs
std::array< uint8_t, 108 > m_Unknown
COND_GE< FMT_VER::V_180, uint32_t > m_Unknown2
Substruct in a padstack object.
A type that is always false.
std::vector< std::string > header
wxString result
Test unit parsing edge cases and error handling.