25#include <netlist_lexer.h>
27#include <nlohmann/json.hpp>
56 NETLIST_LEXER( aReader )
68 while( (
token = NextTok() ) != T_EOF )
73 if(
token == T_RIGHT )
88 while( (
token = NextTok() ) != T_EOF )
107 while( (
token = NextTok() ) != T_EOF )
109 if(
token == T_RIGHT )
111 else if(
token == T_LEFT )
114 if(
token == T_comp )
121 while( (
token = NextTok() ) != T_EOF )
123 if(
token == T_RIGHT )
125 else if(
token == T_LEFT )
135 while( (
token = NextTok() ) != T_EOF )
137 if(
token == T_RIGHT )
139 else if(
token == T_LEFT )
142 if(
token == T_libpart )
171 wxFAIL_MSG( wxString::Format( wxT(
"KICAD_NETLIST_PARSER::Parse(): bad parenthesis "
172 "count (count = %d" ),
191 wxString pin_function;
195 while( (
token = NextTok() ) != T_EOF )
197 if(
token == T_RIGHT )
199 else if(
token == T_LEFT )
205 NeedSYMBOLorNUMBER();
211 NeedSYMBOLorNUMBER();
218 pin_function.Clear();
221 while( (
token = NextTok() ) != T_EOF )
223 if(
token == T_RIGHT )
225 else if(
token == T_LEFT )
231 NeedSYMBOLorNUMBER();
237 NeedSYMBOLorNUMBER();
243 NeedSYMBOLorNUMBER();
249 NeedSYMBOLorNUMBER();
263 if( strtol( code.c_str(),
nullptr, 10 ) >= 1 )
266 name = wxT(
"N-00000") + code;
268 component->AddNet( pin_number,
name, pin_function, pin_type );
304 wxString humanSheetPath;
307 std::vector<KIID> uuids;
308 std::map<wxString, wxString> properties;
309 nlohmann::ordered_map<wxString, wxString> fields;
310 std::unordered_set<wxString> componentClasses;
312 bool duplicatePinsAreJumpers =
false;
313 std::vector<std::set<wxString>> jumperPinGroups;
316 while( (
token = NextTok() ) != T_RIGHT )
318 if(
token == T_LEFT )
324 NeedSYMBOLorNUMBER();
330 NeedSYMBOLorNUMBER();
336 NeedSYMBOLorNUMBER();
337 footprint = FromUTF8();
343 while( (
token = NextTok() ) != T_RIGHT )
345 if(
token == T_LEFT )
350 NeedSYMBOLorNUMBER();
354 else if(
token == T_part )
356 NeedSYMBOLorNUMBER();
360 else if(
token == T_description )
362 NeedSYMBOLorNUMBER();
367 Expecting(
"part, lib or description" );
377 while( (
token = NextTok() ) != T_RIGHT )
379 if(
token == T_LEFT )
382 if(
token == T_name )
384 NeedSYMBOLorNUMBER();
388 else if(
token == T_value )
390 NeedSYMBOLorNUMBER();
396 Expecting(
"name or value" );
400 if( !propName.IsEmpty() )
401 properties[ propName ] = propValue;
406 while( (
token = NextTok() ) != T_RIGHT )
408 if(
token == T_LEFT )
411 if(
token == T_field )
416 while( (
token = NextTok() ) != T_RIGHT )
418 if(
token == T_LEFT )
421 if(
token == T_name )
423 NeedSYMBOLorNUMBER();
427 else if(
token == T_STRING )
433 if( !fieldName.IsEmpty() )
434 fields[fieldName] = fieldValue;
438 Expecting(
"field" );
444 while( (
token = NextTok() ) != T_EOF )
446 if(
token == T_names )
448 NeedSYMBOLorNUMBER();
453 if(
token == T_tstamps )
455 NeedSYMBOLorNUMBER();
467 while( (
token = NextTok() ) != T_EOF )
469 if(
token == T_RIGHT )
472 uuids.emplace_back(
From_UTF8( CurText() ) );
477 case T_component_classes:
478 while( (
token = NextTok() ) != T_RIGHT )
480 if(
token != T_LEFT )
483 if( (
token = NextTok() ) != T_class )
484 Expecting( T_class );
486 NeedSYMBOLorNUMBER();
487 componentClasses.insert(
From_UTF8( CurText() ) );
493 case T_duplicate_pin_numbers_are_jumpers:
495 NeedSYMBOLorNUMBER();
496 duplicatePinsAreJumpers =
From_UTF8( CurText() ) == wxT(
"1" );
501 case T_jumper_pin_groups:
503 std::set<wxString>* currentGroup =
nullptr;
505 for(
token = NextTok(); currentGroup ||
token != T_RIGHT;
token = NextTok() )
507 if(
token == T_LEFT )
513 currentGroup = &jumperPinGroups.emplace_back();
518 NeedSYMBOLorNUMBER();
519 wxString padName =
From_UTF8( CurText() );
521 wxCHECK2( currentGroup,
continue );
522 currentGroup->insert( padName );
527 currentGroup =
nullptr;
531 Expecting(
"group or pin" );
545 if( !footprint.IsEmpty() && fpid.
Parse( footprint,
true ) >= 0 )
548 error.Printf(
_(
"Invalid footprint ID in\nfile: '%s'\nline: %d\nofff: %d" ),
549 CurSource(), CurLineNumber(), CurOffset() );
562 std::ranges::copy( jumperPinGroups,
592 wxString libPartName;
593 wxArrayString footprintFilters;
594 wxArrayString aliases;
598 while( (
token = NextTok() ) != T_RIGHT )
600 if(
token == T_LEFT )
606 NeedSYMBOLorNUMBER();
612 NeedSYMBOLorNUMBER();
619 while( (
token = NextTok() ) != T_RIGHT )
621 if(
token == T_LEFT )
627 NeedSYMBOLorNUMBER();
628 footprintFilters.Add(
From_UTF8( CurText() ) );
634 while( (
token = NextTok() ) != T_RIGHT )
636 if(
token == T_LEFT )
639 if(
token != T_alias )
640 Expecting( T_alias );
642 NeedSYMBOLorNUMBER();
649 while( (
token = NextTok() ) != T_RIGHT )
651 if(
token == T_LEFT )
675 if( component->
IsLibSource( libName, libPartName ) )
681 for(
unsigned jj = 0; jj < aliases.GetCount(); jj++ )
683 if( component->
IsLibSource( libName, aliases[jj] ) )
bool Load(NETLIST *aNetlist)
Read the *.cmp file format contains the component footprint assignments created by CvPcb into aNetlis...
Store all of the related footprint information found in a netlist.
void SetFields(nlohmann::ordered_map< wxString, wxString > &aFields)
void SetLibrary(const wxString &aLibrary)
void SetProperties(std::map< wxString, wxString > &aProps)
void SetPinCount(int aPinCount)
bool IsLibSource(const wxString &aLibrary, const wxString &aName) const
void SetFootprintFilters(const wxArrayString &aFilters)
void SetHumanReadablePath(const wxString &aPath)
void SetDuplicatePadNumbersAreJumpers(bool aEnabled)
void SetComponentClassNames(const std::unordered_set< wxString > &aClassNames)
std::vector< std::set< wxString > > & JumperPadGroups()
void SetName(const wxString &aName)
The parser for reading the KiCad s-expression netlist format.
KICAD_NETLIST_PARSER(LINE_READER *aReader, NETLIST *aNetlist)
void parseComponent()
Parse a component description: (comp (ref P1) (value DB25FEMELLE) (footprint DB25FC) (libsource (lib ...
void Parse()
Function Parse parse the full netlist.
void parseNet()
Parse a net section (net (code 20) (name /PC-A0) (node (ref BUS1) (pin 62)) (node (ref U3) (pin 3)) (...
NETLIST * m_netlist
The netlist to parse into. Not owned.
void parseLibPartList()
Read the section "libparts" in the netlist: (libparts (libpart (lib device) (part C) (description "Co...
LINE_READER * m_lineReader
The line reader used to parse the netlist. Not owned.
void skipCurrent()
Skip the current token level, i.e search for the RIGHT parenthesis which closes the current descripti...
virtual void LoadNetlist() override
Load the contents of the netlist file into aNetlist.
A logical library item identifier and consists of various portions much like a URI.
int Parse(const UTF8 &aId, bool aFix=false)
Parse LIB_ID with the information from aId.
An abstract class from which implementation specific LINE_READERs may be derived to read single lines...
NETLIST * m_netlist
The net list to read the file(s) into.
LINE_READER * m_lineReader
The line reader of the netlist.
CMP_READER * m_footprintReader
The reader used to load the footprint links. If NULL, footprint links are not read.
Store information read from a netlist along with the flags used to update the NETLIST in the BOARD.
unsigned GetCount() const
void AddComponent(COMPONENT *aComponent)
Add aComponent to the NETLIST.
COMPONENT * GetComponentByReference(const wxString &aReference)
Return a COMPONENT by aReference.
COMPONENT * GetComponent(unsigned aIndex)
Return the COMPONENT at aIndex.
#define THROW_IO_ERROR(msg)
wxString From_UTF8(const char *cstring)