KiCad PCB EDA Suite
Loading...
Searching...
No Matches
pcb_io_kicad_sexpr_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) 2012 CERN
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, see <https://www.gnu.org/licenses/>.
19 */
20
25
26#ifndef _PCBNEW_PARSER_H_
27#define _PCBNEW_PARSER_H_
28
29#include <eda_units.h>
30#include <core/wx_stl_compat.h>
31#include <hashtables.h>
32#include <lib_id.h>
33#include <layer_ids.h> // PCB_LAYER_ID
34#include <lset.h>
35#include <pcb_lexer.h>
36#include <kiid.h>
37#include <math/box2.h>
38#include <string_any_map.h>
39#include <padstack.h>
41
42#include <chrono>
43#include <unordered_map>
44
45
46class PCB_ARC;
47class BOARD;
48class BOARD_ITEM;
49class ZONE_SETTINGS;
52class PAD;
55class PCB_SHAPE;
57class EDA_TEXT;
58class PCB_TEXT;
59class PCB_TEXTBOX;
60class PCB_TRACK;
61class PCB_TABLE;
62class PCB_TABLECELL;
63class FOOTPRINT;
64class PCB_GROUP;
65class PCB_POINT;
66class PCB_TARGET;
67class PCB_VIA;
68class ZONE;
70class PCB_BARCODE;
71class FP_3DMODEL;
73struct LAYER;
76
77
82class PCB_IO_KICAD_SEXPR_PARSER : public PCB_LEXER
83{
84public:
85
86 typedef std::unordered_map< std::string, PCB_LAYER_ID > LAYER_ID_MAP;
87 typedef std::unordered_map< std::string, LSET > LSET_MAP;
88 typedef std::unordered_map< wxString, KIID > KIID_MAP;
89
91 std::function<bool( wxString, int, wxString, wxString )> aQueryUserCallback,
92 PROGRESS_REPORTER* aProgressReporter = nullptr, unsigned aLineCount = 0,
93 bool aPreserveDestinationStackup = false ) :
94 PCB_LEXER( aReader ),
95 m_board( aAppendToMe ),
96 m_appendToExisting( aAppendToMe != nullptr ),
97 m_preserveDestinationStackup( aPreserveDestinationStackup ),
98 m_progressReporter( aProgressReporter ),
99 m_lastProgressTime( std::chrono::steady_clock::now() ),
100 m_lineCount( aLineCount ),
101 m_queryUserCallback( std::move( aQueryUserCallback ) )
102 {
103 init();
104 }
105
106 // ~PCB_IO_KICAD_SEXPR_PARSER() {}
107
108 BOARD_ITEM* Parse();
109
116 FOOTPRINT* parseFOOTPRINT( wxArrayString* aInitialComments = nullptr );
117
122 {
123 return m_tooRecent;
124 }
125
130 wxString GetRequiredVersion();
131
136 bool IsValidBoardHeader();
137
142 const std::vector<wxString>& GetParseWarnings() const { return m_parseWarnings; }
143
145 void SetLayerMappingHandler( LAYER_MAPPING_HANDLER aHandler ) { m_layerMappingHandler = std::move( aHandler ); }
146
147private:
148
149 // Group membership info refers to other Uuids in the file.
150 // We don't want to rely on group declarations being last in the file, so
151 // we store info about the group declarations here during parsing and then resolve
152 // them into BOARD_ITEM* after we've parsed the rest of the file.
154 {
155 virtual ~GROUP_INFO() = default; // Make polymorphic
156
158 wxString name;
159 bool locked;
162 std::vector<KIID> memberUuids;
163 };
164
171
174 inline int getNetCode( int aNetCode )
175 {
176 if( ( aNetCode >= 0 ) && ( aNetCode < (int) m_netCodes.size() ) )
177 return m_netCodes[aNetCode];
178
179 return aNetCode;
180 }
181
190 void pushValueIntoMap( int aIndex, int aValue );
191
198 void init();
199
200 void checkpoint();
201
209 void createOldLayerMapping( std::unordered_map< std::string, std::string >& aMap );
210
215 void skipCurrent();
216
217 void parseHeader();
218 void parseGeneralSection();
219 void parsePAGE_INFO();
220 void parseTITLE_BLOCK();
221
222 void parseLayers();
223
225 void remapAppendedLayers( const std::vector<LAYER>& aSourceLayers, const LSET& aDestInitialEnabled,
226 int aDestInitialCopperCount );
227 void parseLayer( LAYER* aLayer );
228
229 void parseBoardStackup();
230
231 void parseSetup();
232 void parseDefaults( BOARD_DESIGN_SETTINGS& aSettings );
233 void parseDefaultTextDims( BOARD_DESIGN_SETTINGS& aSettings, int aLayer );
234 void parseNETINFO_ITEM();
235 void parseNETCLASS();
237
239
240 void parseTextBoxContent( PCB_TEXTBOX* aTextBox );
241
242 void bakeTextBoxLib( PCB_TEXTBOX* aTextBox );
243
245 PCB_TEXT* parsePCB_TEXT( BOARD_ITEM* aParent, PCB_TEXT* aBaseText = nullptr );
246 void parsePCB_TEXT_effects( PCB_TEXT* aText, PCB_TEXT* aBaseText = nullptr );
253
254 // Parse a footprint, but do not replace PARSE_ERROR with FUTURE_FORMAT_ERROR automatically.
255 FOOTPRINT* parseFOOTPRINT_unchecked( wxArrayString* aInitialComments = nullptr );
256 void parseFootprintStackup( FOOTPRINT& aFootprint );
257
258 PAD* parsePAD( FOOTPRINT* aParent = nullptr );
259
260 // Parse only the (option ...) inside a pad description
261 bool parsePAD_option( PAD* aPad );
263
264 void parsePadstack( PAD* aPad );
265
266 PCB_ARC* parseARC();
269 void parseViastack( PCB_VIA* aVia );
273 BOARD* parseBOARD();
274 void parseGROUP_members( GROUP_INFO& aGroupInfo );
275 void parseGROUP( BOARD_ITEM* aParent );
276 void parseGENERATOR( BOARD_ITEM* aParent );
277
278 // Parse a board, but do not replace PARSE_ERROR with FUTURE_FORMAT_ERROR automatically.
280
290 LSET lookUpLayerSet( const LSET_MAP& aMap );
291
300
309
319
331
332 void parseXY( int* aX, int* aY );
333
334 void parseMargins( int& aLeft, int& aTop, int& aRight, int& aBottom );
335
336 void parseZoneDefaults( ZONE_SETTINGS& aZoneSettings );
337
338 void parseZoneLayerProperty( std::map<PCB_LAYER_ID, ZONE_LAYER_PROPERTIES>& aProperties );
339
340 std::pair<wxString, wxString> parseBoardProperty();
341
342 void parseVariants();
343 void parseFootprintVariant( FOOTPRINT* aFootprint );
344
353
360 void parseEDA_TEXT( EDA_TEXT* aText );
361
369
370 FP_3DMODEL* parse3DModel( bool aFileNameAlreadyParsed = false );
371
379
380 int parseBoardUnits();
381
382 int parseBoardUnits( const char* aExpected, EDA_DATA_TYPE aDataType );
383
384 inline int parseBoardUnits( const PCB_KEYS_T::T aToken, const EDA_DATA_TYPE aDataType = EDA_DATA_TYPE::DISTANCE )
385 {
386 return parseBoardUnits( GetTokenText( aToken ), aDataType );
387 }
388
389 inline int parseInt()
390 {
391 return (int)strtol( CurText(), nullptr, 10 );
392 }
393
394 inline int parseInt( const char* aExpected )
395 {
396 NeedNUMBER( aExpected );
397 return parseInt();
398 }
399
400 inline long parseHex()
401 {
402 NextTok();
403 return strtol( CurText(), nullptr, 16 );
404 }
405
406 bool parseBool();
407
408 std::optional<bool> parseOptBool();
409
420 bool parseMaybeAbsentBool( bool aDefaultValue );
421
422 std::pair<std::optional<bool>, std::optional<bool>> parseFrontBackOptBool( bool aAllowLegacyFormat = false );
423
424 void parseNet( BOARD_CONNECTED_ITEM* aItem );
425
426 /*
427 * @return if m_appendToExisting, returns new KIID(), otherwise returns CurStr() as KIID.
428 */
430
435 void resolveGroups( BOARD_ITEM* aParent );
436
438 using TIMEOUT = std::chrono::milliseconds;
439
441 using CLOCK = std::chrono::steady_clock;
442
444 using TIME_PT = std::chrono::time_point<CLOCK>;
445
449 std::set<wxString> m_undefinedLayers;
450 std::vector<int> m_netCodes;
457
460
463
466 unsigned m_lineCount;
467
468 std::vector<GROUP_INFO> m_groupInfos;
469 std::vector<GENERATOR_INFO> m_generatorInfos;
470
471 std::function<bool( wxString aTitle, int aIcon, wxString aMsg, wxString aAction )> m_queryUserCallback;
472
473 std::vector<wxString> m_parseWarnings;
474};
475
476
477#endif // _PCBNEW_PARSER_H_
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
Container for design settings for a BOARD object.
Abstract interface for BOARD_ITEMs capable of storing other items inside.
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition board_item.h:81
Information pertinent to a Pcbnew printed circuit board.
Definition board.h:372
A mix-in class (via multiple inheritance) that handles texts such as labels, parts,...
Definition eda_text.h:89
Definition kiid.h:44
A logical library item identifier and consists of various portions much like a URI.
Definition lib_id.h:45
An abstract class from which implementation specific LINE_READERs may be derived to read single lines...
Definition richio.h:62
LSET is a set of PCB_LAYER_IDs.
Definition lset.h:37
Definition pad.h:61
Abstract dimension API.
A set of BOARD_ITEMs (i.e., without duplicates).
Definition pcb_group.h:49
wxString m_generatorVersion
Set to the generator version this board requires.
PCB_TABLECELL * parsePCB_TABLECELL(BOARD_ITEM *aParent)
std::unordered_map< std::string, PCB_LAYER_ID > LAYER_ID_MAP
std::vector< int > m_netCodes
net codes mapping for boards being loaded
void parseOutlinePoints(SHAPE_LINE_CHAIN &aPoly)
Parses possible outline points and stores them into aPoly.
std::set< wxString > m_undefinedLayers
set of layers not defined in layers section
LAYER_MAPPING_HANDLER m_layerMappingHandler
optional remap of appended layers onto dest
void parseZoneLayerProperty(std::map< PCB_LAYER_ID, ZONE_LAYER_PROPERTIES > &aProperties)
PROGRESS_REPORTER * m_progressReporter
optional; may be nullptr
void parseFootprintStackup(FOOTPRINT &aFootprint)
void createOldLayerMapping(std::unordered_map< std::string, std::string > &aMap)
Create a mapping from the (short-lived) bug where layer names were translated.
void parseZoneDefaults(ZONE_SETTINGS &aZoneSettings)
std::unordered_map< std::string, LSET > LSET_MAP
void parseEDA_TEXT(EDA_TEXT *aText)
Parse the common settings for any object derived from EDA_TEXT.
int parseInt(const char *aExpected)
bool m_tooRecent
true if version parses as later than supported
PCB_LAYER_ID lookUpLayer(const LAYER_ID_MAP &aMap)
Parse the current token for the layer definition of a BOARD_ITEM object.
void remapAppendedLayers(const std::vector< LAYER > &aSourceLayers, const LSET &aDestInitialEnabled, int aDestInitialCopperCount)
Remap the appended layers onto the destination using m_layerMappingHandler, on mismatch.
PCB_REFERENCE_IMAGE * parsePCB_REFERENCE_IMAGE(BOARD_ITEM *aParent)
LAYER_ID_MAP m_layerIndices
map layer name to it's index
const std::vector< wxString > & GetParseWarnings() const
Return any non-fatal parse warnings that occurred during parsing.
void parsePostMachining(PADSTACK::POST_MACHINING_PROPS &aProps)
FP_3DMODEL * parse3DModel(bool aFileNameAlreadyParsed=false)
void parseTextBoxContent(PCB_TEXTBOX *aTextBox)
FOOTPRINT * parseFOOTPRINT(wxArrayString *aInitialComments=nullptr)
void pushValueIntoMap(int aIndex, int aValue)
Add aValue value in netcode mapping (m_netCodes) at aIndex.
bool m_preserveDestinationStackup
append keeps destination stackup
std::chrono::steady_clock CLOCK
The type of the time stamps.
void init()
Clear and re-establish m_layerMap with the default layer names.
std::pair< std::optional< bool >, std::optional< bool > > parseFrontBackOptBool(bool aAllowLegacyFormat=false)
void skipCurrent()
Skip the current token level, i.e search for the RIGHT parenthesis which closes the current descripti...
void parseMargins(int &aLeft, int &aTop, int &aRight, int &aBottom)
PCB_LAYER_ID parseBoardItemLayer()
Parse the layer definition of a BOARD_ITEM object.
LSET parseLayersForCuItemWithSoldermask()
Parse the layers definition of a BOARD_ITEM object that has a single copper layer and optional solder...
void parseGENERATOR(BOARD_ITEM *aParent)
PCB_IO_KICAD_SEXPR_PARSER(LINE_READER *aReader, BOARD *aAppendToMe, std::function< bool(wxString, int, wxString, wxString)> aQueryUserCallback, PROGRESS_REPORTER *aProgressReporter=nullptr, unsigned aLineCount=0, bool aPreserveDestinationStackup=false)
std::chrono::milliseconds TIMEOUT
The clock used for the timestamp (guaranteed to be monotonic).
LSET parseBoardItemLayersAsMask()
Parse the layers definition of a BOARD_ITEM object.
void resolveGroups(BOARD_ITEM *aParent)
Called after parsing a footprint definition or board to build the group membership lists.
void parseDefaultTextDims(BOARD_DESIGN_SETTINGS &aSettings, int aLayer)
std::vector< GROUP_INFO > m_groupInfos
ZONE * parseZONE(BOARD_ITEM_CONTAINER *aParent)
PCB_TABLE * parsePCB_TABLE(BOARD_ITEM *aParent)
std::vector< GENERATOR_INFO > m_generatorInfos
PCB_TEXTBOX * parsePCB_TEXTBOX(BOARD_ITEM *aParent)
std::chrono::time_point< CLOCK > TIME_PT
PCB_TEXT * parsePCB_TEXT(BOARD_ITEM *aParent, PCB_TEXT *aBaseText=nullptr)
unsigned m_lineCount
for progress reporting
VECTOR2I parseXY()
Parse a coordinate pair (xy X Y) in board units (mm).
void parseTEARDROP_PARAMETERS(TEARDROP_PARAMETERS *tdParams)
int m_requiredVersion
set to the KiCad format version this board requires
PAD * parsePAD(FOOTPRINT *aParent=nullptr)
std::function< bool(wxString aTitle, int aIcon, wxString aMsg, wxString aAction)> m_queryUserCallback
void parseNet(BOARD_CONNECTED_ITEM *aItem)
FOOTPRINT * parseFOOTPRINT_unchecked(wxArrayString *aInitialComments=nullptr)
void SetLayerMappingHandler(LAYER_MAPPING_HANDLER aHandler)
Handler to remap an appended board's layers onto the destination board, used on mismatch.
void parseRenderCache(EDA_TEXT *text)
Parse the render cache for any object derived from EDA_TEXT.
int parseBoardUnits(const PCB_KEYS_T::T aToken, const EDA_DATA_TYPE aDataType=EDA_DATA_TYPE::DISTANCE)
TIME_PT m_lastProgressTime
for progress reporting
void parseGROUP_members(GROUP_INFO &aGroupInfo)
bool IsTooRecent()
Return whether a version number, if any was parsed, was too recent.
bool IsValidBoardHeader()
Partially parse the input and check if it matches expected header.
void parseFootprintVariant(FOOTPRINT *aFootprint)
std::pair< wxString, wxString > parseBoardProperty()
LSET_MAP m_layerMasks
map layer names to their masks
bool parseMaybeAbsentBool(bool aDefaultValue)
Parses a boolean flag inside a list that existed before boolean normalization.
int parseBoardUnits()
Parse the current token as an ASCII numeric string with possible leading whitespace into a double pre...
void bakeTextBoxLib(PCB_TEXTBOX *aTextBox)
Lift disk-parsed values into PCB_TEXTBOX lib storage for new format files.
std::unordered_map< wxString, KIID > KIID_MAP
PCB_DIMENSION_BASE * parseDIMENSION(BOARD_ITEM *aParent)
LSET lookUpLayerSet(const LSET_MAP &aMap)
std::vector< wxString > m_parseWarnings
Non-fatal warnings collected during parsing.
bool m_appendToExisting
reading into an existing board; reset UUIDs
void parsePCB_TEXT_effects(PCB_TEXT *aText, PCB_TEXT *aBaseText=nullptr)
PCB_SHAPE * parsePCB_SHAPE(BOARD_ITEM *aParent)
void parseDefaults(BOARD_DESIGN_SETTINGS &aSettings)
wxString GetRequiredVersion()
Return a string representing the version of KiCad required to open this file.
PCB_BARCODE * parsePCB_BARCODE(BOARD_ITEM *aParent)
A PCB_POINT is a 0-dimensional point that is used to mark a position on a PCB, or more usually a foot...
Definition pcb_point.h:39
Object to handle a bitmap image that can be inserted in a PCB.
A progress reporter interface for use in multi-threaded environments.
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
A name/value tuple with unique names and wxAny values.
TEARDROP_PARAMETARS is a helper class to handle parameters needed to build teardrops for a board thes...
ZONE_SETTINGS handles zones parameters.
Handle a list of polygons defining a copper zone.
Definition zone.h:70
#define LAYER(n, l)
EDA_DATA_TYPE
The type of unit.
Definition eda_units.h:34
PCB_LAYER_ID
A quick note on layer IDs:
Definition layer_ids.h:56
STL namespace.
std::function< std::map< wxString, PCB_LAYER_ID >(const std::vector< INPUT_LAYER_DESC > &)> LAYER_MAPPING_HANDLER
Pointer to a function that takes a map of source and KiCad layers and returns a re-mapped version.
Convert net code using the mapping table if available, otherwise returns unchanged net code if < 0 or...
VECTOR2< int32_t > VECTOR2I
Definition vector2d.h:683