KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
altium_binary_parser.h
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2019-2020 Thomas Pointhuber <thomas.pointhuber@gmx.at>
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
25#ifndef _ALTIUM_BINARY_PARSER_H
26#define _ALTIUM_BINARY_PARSER_H
27
28#include "altium_props_utils.h"
29
30#include <charconv>
31#include <map>
32#include <memory>
33#include <numeric>
34#include <string>
35#include <stdexcept>
36#include <vector>
37
38#include <wx/mstream.h>
39#include <wx/zstream.h>
40#include <math/vector2d.h>
41#include <math/vector3.h>
42
43namespace CFB
44{
45class CompoundFileReader;
46struct COMPOUND_FILE_ENTRY;
47} // namespace CFB
48
54std::string FormatPath( const std::vector<std::string>& aVectorPath );
55
56
58{
59public:
60 const CFB::COMPOUND_FILE_ENTRY* m_symbol;
61 const CFB::COMPOUND_FILE_ENTRY* m_pinsFrac;
62 const CFB::COMPOUND_FILE_ENTRY* m_pinsWideText;
63 const CFB::COMPOUND_FILE_ENTRY* m_pinsTextData;
64 const CFB::COMPOUND_FILE_ENTRY* m_pinsSymbolLineWidth;
65};
66
67
69{
71
72public:
75
81 ALTIUM_COMPOUND_FILE( const wxString& aFilePath );
82
90 ALTIUM_COMPOUND_FILE( const void* aBuffer, size_t aLen );
91
92 ALTIUM_COMPOUND_FILE( const ALTIUM_COMPOUND_FILE& temp_obj ) = delete;
95
103 void InitFromBuffer( const void* aBuffer, size_t aLen );
104
105 const CFB::CompoundFileReader& GetCompoundFileReader() const { return *m_reader; }
106
107 bool DecodeIntLibStream( const CFB::COMPOUND_FILE_ENTRY& cfe, ALTIUM_COMPOUND_FILE* aOutput );
108
109 const CFB::COMPOUND_FILE_ENTRY* FindStream( const std::vector<std::string>& aStreamPath ) const;
110
111 const CFB::COMPOUND_FILE_ENTRY* FindStream( const CFB::COMPOUND_FILE_ENTRY* aStart,
112 const std::vector<std::string>& aStreamPath ) const;
113
114 const CFB::COMPOUND_FILE_ENTRY* FindStreamSingleLevel( const CFB::COMPOUND_FILE_ENTRY* aEntry,
115 const std::string aName,
116 const bool aIsStream ) const;
117
118 std::map<wxString, const CFB::COMPOUND_FILE_ENTRY*> EnumDir( const std::wstring& aDir ) const;
119
120 std::map<wxString, ALTIUM_SYMBOL_DATA> GetLibSymbols( const CFB::COMPOUND_FILE_ENTRY* aStart ) const;
121
122private:
123
124 std::unique_ptr<CFB::CompoundFileReader> m_reader;
125 std::vector<char> m_buffer;
126};
127
128
130{
131public:
133 const CFB::COMPOUND_FILE_ENTRY* aEntry );
134 ALTIUM_BINARY_PARSER( std::unique_ptr<char[]>& aContent, size_t aSize );
136
137 template <typename Type>
138 Type Read()
139 {
140 const size_t remainingBytes = GetRemainingBytes();
141
142 if( remainingBytes >= sizeof( Type ) )
143 {
144 Type val = *(Type*) ( m_pos );
145 m_pos += sizeof( Type );
146 return val;
147 }
148 else
149 {
150 m_pos += remainingBytes; // Ensure remaining bytes are zero
151 m_error = true;
152 return 0;
153 }
154 }
155
156 template <typename Type>
157 Type Peek()
158 {
159 char* const oldPos = m_pos;
160 const bool oldError = m_error;
161 Type result = Read<Type>();
162 m_pos = oldPos;
163 m_error = oldError;
164 return result;
165 }
166
167 wxScopedCharBuffer ReadCharBuffer()
168 {
169 uint8_t len = Read<uint8_t>();
170
171 if( GetRemainingBytes() >= len )
172 {
173 char* buf = static_cast<char*>( malloc( len ) );
174 memcpy( buf, m_pos, len );
175 m_pos += len;
176
177 return wxScopedCharBuffer::CreateOwned( buf, len );
178 }
179 else
180 {
181 m_error = true;
182 return wxScopedCharBuffer();
183 }
184 }
185
186 wxString ReadWxString()
187 {
188 // TODO: Identify where the actual code page is stored. For now, this default code page
189 // has limited impact, because recent Altium files come with a UTF16 string table
190 return wxString( ReadCharBuffer(), wxConvISO8859_1 );
191 }
192
193 std::map<uint32_t, wxString> ReadWideStringTable()
194 {
195 std::map<uint32_t, wxString> table;
196 size_t remaining = GetRemainingBytes();
197
198 while( remaining >= 8 )
199 {
200 uint32_t index = Read<uint32_t>();
201 uint32_t length = Read<uint32_t>();
202 wxString str;
203 remaining -= 8;
204
205 if( length <= 2 )
206 {
207 length = 0; // for empty strings, not even the null bytes are present
208 }
209 else
210 {
211 if( length > remaining )
212 break;
213
214 str = wxString( m_pos, wxMBConvUTF16LE(), length - 2 );
215 }
216
217 table.emplace( index, str );
218 m_pos += length;
219 remaining -= length;
220 }
221
222 return table;
223 }
224
225 std::vector<char> ReadVector( size_t aSize )
226 {
227 if( aSize > GetRemainingBytes() )
228 {
229 m_error = true;
230 return {};
231 }
232 else
233 {
234 std::vector<char> data( m_pos, m_pos + aSize );
235 m_pos += aSize;
236 return data;
237 }
238 }
239
240 int ReadBytes( char* aOut, size_t aSize )
241 {
242 if( aSize > GetRemainingBytes() )
243 {
244 m_error = true;
245 return 0;
246 }
247 else
248 {
249 memcpy( aOut, m_pos, aSize );
250 m_pos += aSize;
251 return aSize;
252 }
253 }
254
256 {
257 return ALTIUM_PROPS_UTILS::ConvertToKicadUnit( Read<int32_t>() );
258 }
259
261 {
262 return ReadKicadUnit();
263 }
264
266 {
267 return -ReadKicadUnit();
268 }
269
271 {
272 int32_t x = ReadKicadUnitX();
273 int32_t y = ReadKicadUnitY();
274 return { x, y };
275 }
276
278 {
279 int32_t x = ReadKicadUnit();
280 int32_t y = ReadKicadUnit();
281 return { x, y };
282 }
283
285 {
286 uint32_t length = Read<uint32_t>();
287 m_subrecord_end = m_pos + length;
288 return length;
289 }
290
291 std::map<wxString, wxString> ReadProperties(
292 std::function<std::map<wxString, wxString>( const std::string& )> handleBinaryData =
293 []( const std::string& )
294 {
295 return std::map<wxString, wxString>();
296 } );
297
298 void Skip( size_t aLength )
299 {
300 if( GetRemainingBytes() >= aLength )
301 {
302 m_pos += aLength;
303 }
304 else
305 {
306 m_error = true;
307 }
308 }
309
311 {
312 if( m_subrecord_end == nullptr || m_subrecord_end < m_pos )
313 {
314 m_error = true;
315 }
316 else
317 {
319 }
320 };
321
322 size_t GetRemainingBytes() const
323 {
324 return m_pos == nullptr ? 0 : m_size - ( m_pos - m_content.get() );
325 }
326
328 {
329 return m_pos == nullptr || m_subrecord_end == nullptr || m_subrecord_end <= m_pos ?
330 0 :
332 };
333
335 {
336 return m_error;
337 }
338
339private:
340 std::unique_ptr<char[]> m_content;
341 size_t m_size;
342
343 char* m_pos; // current read pointer
344 char* m_subrecord_end; // pointer which points to next subrecord start
346};
347
348
350{
351public:
352 ALTIUM_BINARY_READER( const std::string& binaryData ) : m_data( binaryData ), m_position( 0 ) {}
353
354 int32_t ReadInt32()
355 {
356 if( m_position + sizeof( int32_t ) > m_data.size() )
357 throw std::out_of_range( "ALTIUM_BINARY_READER: out of range" );
358
359 int32_t value = *reinterpret_cast<const int32_t*>( &m_data[m_position] );
360 m_position += sizeof( int32_t );
361 return value;
362 }
363
364 int16_t ReadInt16()
365 {
366 if( m_position + sizeof( int16_t ) > m_data.size() )
367 throw std::out_of_range( "ALTIUM_BINARY_READER: out of range" );
368
369 int16_t value = *reinterpret_cast<const int16_t*>( &m_data[m_position] );
370 m_position += sizeof( int16_t );
371 return value;
372 }
373
374 uint8_t ReadByte()
375 {
376 if( m_position + sizeof( uint8_t ) > m_data.size() )
377 throw std::out_of_range( "ALTIUM_BINARY_READER: out of range" );
378
379 uint8_t value = *reinterpret_cast<const uint8_t*>( &m_data[m_position] );
380 m_position += sizeof( uint8_t );
381 return value;
382 }
383
385 {
386 uint8_t length = ReadByte();
387
388 if( m_position + length > m_data.size() )
389 throw std::out_of_range( "ALTIUM_BINARY_READER: out of range" );
390
391 std::string pascalString( &m_data[m_position], &m_data[m_position + length] );
392 m_position += length;
393 return pascalString;
394 }
395
397 {
398 uint32_t length = ReadInt32();
399
400 if( m_position + length > m_data.size() )
401 throw std::out_of_range( "ALTIUM_BINARY_READER: out of range" );
402
403 std::string pascalString( &m_data[m_position], &m_data[m_position + length] );
404 m_position += length;
405 return pascalString;
406 }
407
408private:
409 const std::string& m_data;
411};
412
414{
415public:
416 ALTIUM_COMPRESSED_READER( const std::string& aData ) : ALTIUM_BINARY_READER( aData )
417 {}
418
419 std::pair<int, std::string*> ReadCompressedString()
420 {
421 std::string* result;
422 int id = -1;
423
424 uint8_t byte = ReadByte();
425
426 if( byte != 0xD0 )
427 throw std::runtime_error( "ALTIUM_COMPRESSED_READER: invalid compressed string" );
428
429 std::string str = ReadShortPascalString();
430 std::from_chars( str.data(), str.data() + str.size(), id );
431
432 std::string data = ReadFullPascalString();
433 result = decompressData( data );
434
435 return std::make_pair( id, result );
436 }
437
438private:
439 std::string decompressedData;
440
441 std::string* decompressData( std::string& aData )
442 {
443 // Create a memory input stream with the buffer
444 wxMemoryInputStream memStream( (void*) aData.data(), aData.length() );
445
446 // Create a zlib input stream with the memory input stream
447 wxZlibInputStream zStream( memStream );
448
449 // Read decompressed data from the zlib input stream
450 while( !zStream.Eof() )
451 {
452 char buffer[1024];
453 zStream.Read( buffer, sizeof( buffer ) );
454 size_t bytesRead = zStream.LastRead();
455 decompressedData.append( buffer, bytesRead );
456 }
457
458 return &decompressedData;
459 }
460};
461
462#endif //_ALTIUM_BINARY_PARSER_H
std::string FormatPath(const std::vector< std::string > &aVectorPath)
Helper for debug logging (vector -> string)
size_t GetRemainingBytes() const
~ALTIUM_BINARY_PARSER()=default
void Skip(size_t aLength)
std::vector< char > ReadVector(size_t aSize)
std::unique_ptr< char[]> m_content
wxScopedCharBuffer ReadCharBuffer()
std::map< uint32_t, wxString > ReadWideStringTable()
size_t GetRemainingSubrecordBytes() const
std::map< wxString, wxString > ReadProperties(std::function< std::map< wxString, wxString >(const std::string &)> handleBinaryData=[](const std::string &) { return std::map< wxString, wxString >();})
int ReadBytes(char *aOut, size_t aSize)
std::string ReadShortPascalString()
std::string ReadFullPascalString()
const std::string & m_data
ALTIUM_BINARY_READER(const std::string &binaryData)
ALTIUM_COMPOUND_FILE()
Create an uninitialized file for two-step initialization (e.g. with InitFromBuffer)
const CFB::CompoundFileReader & GetCompoundFileReader() const
const CFB::COMPOUND_FILE_ENTRY * FindStreamSingleLevel(const CFB::COMPOUND_FILE_ENTRY *aEntry, const std::string aName, const bool aIsStream) const
~ALTIUM_COMPOUND_FILE()=default
std::map< wxString, const CFB::COMPOUND_FILE_ENTRY * > EnumDir(const std::wstring &aDir) const
void InitFromBuffer(const void *aBuffer, size_t aLen)
Load a CFB file from memory; may throw an IO_ERROR.
ALTIUM_COMPOUND_FILE & operator=(const ALTIUM_COMPOUND_FILE &temp_obj)=delete
std::vector< char > m_buffer
std::unique_ptr< CFB::CompoundFileReader > m_reader
std::map< wxString, ALTIUM_SYMBOL_DATA > GetLibSymbols(const CFB::COMPOUND_FILE_ENTRY *aStart) const
bool DecodeIntLibStream(const CFB::COMPOUND_FILE_ENTRY &cfe, ALTIUM_COMPOUND_FILE *aOutput)
const CFB::COMPOUND_FILE_ENTRY * FindStream(const std::vector< std::string > &aStreamPath) const
ALTIUM_COMPOUND_FILE(const ALTIUM_COMPOUND_FILE &temp_obj)=delete
std::string * decompressData(std::string &aData)
ALTIUM_COMPRESSED_READER(const std::string &aData)
std::pair< int, std::string * > ReadCompressedString()
static int32_t ConvertToKicadUnit(const double aValue)
const CFB::COMPOUND_FILE_ENTRY * m_pinsSymbolLineWidth
const CFB::COMPOUND_FILE_ENTRY * m_symbol
const CFB::COMPOUND_FILE_ENTRY * m_pinsFrac
const CFB::COMPOUND_FILE_ENTRY * m_pinsWideText
const CFB::COMPOUND_FILE_ENTRY * m_pinsTextData