KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
kicad_netlist_reader.cpp
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) 1992-2011 Jean-Pierre Charras.
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#include <netlist_lexer.h> // netlist_lexer is common to Eeschema and Pcbnew
26#include <string_utils.h>
27#include <nlohmann/json.hpp>
28
29#include "pcb_netlist.h"
30#include "netlist_reader.h"
32
33using namespace NL_T;
34
35
37{
39
40 parser.Parse();
41
43 {
45
46 // Sort the component pins so they are in the same order as the legacy format. This
47 // is useful for comparing legacy and s-expression netlist dumps.
48 for( unsigned i = 0; i < m_netlist->GetCount(); i++ )
50 }
51}
52
53
54// KICAD_NETLIST_PARSER
56 NETLIST_LEXER( aReader )
57{
58 m_lineReader = aReader;
59 m_netlist = aNetlist;
60 token = T_NONE;
61}
62
63
65{
66 int curr_level = 0;
67
68 while( ( token = NextTok() ) != T_EOF )
69 {
70 if( token == T_LEFT )
71 curr_level--;
72
73 if( token == T_RIGHT )
74 {
75 curr_level++;
76
77 if( curr_level > 0 )
78 return;
79 }
80 }
81}
82
83
85{
86 int plevel = 0; // the count of ')' to read at end of file after parsing all sections
87
88 while( ( token = NextTok() ) != T_EOF )
89 {
90 if( token == T_LEFT )
91 token = NextTok();
92
93 switch( token )
94 {
95 case T_export: // The netlist starts here.
96 // nothing to do here, just increment the count of ')' to read at end of file
97 plevel++;
98 break;
99
100 case T_version: // The netlist starts here.
101 // version id not yet used: read it but does not use it
102 NextTok();
103 NeedRIGHT();
104 break;
105
106 case T_components: // The section comp starts here.
107 while( ( token = NextTok() ) != T_EOF )
108 {
109 if( token == T_RIGHT )
110 break;
111 else if( token == T_LEFT )
112 token = NextTok();
113
114 if( token == T_comp ) // A component section found. Read it
116 }
117
118 break;
119
120 case T_nets: // The section nets starts here.
121 while( ( token = NextTok() ) != T_EOF )
122 {
123 if( token == T_RIGHT )
124 break;
125 else if( token == T_LEFT )
126 token = NextTok();
127
128 if( token == T_net ) // A net section if found. Read it
129 parseNet();
130 }
131
132 break;
133
134 case T_libparts: // The section libparts starts here.
135 while( ( token = NextTok() ) != T_EOF )
136 {
137 if( token == T_RIGHT )
138 break;
139 else if( token == T_LEFT )
140 token = NextTok();
141
142 if( token == T_libpart ) // A libpart section if found. Read it
144 }
145
146 break;
147
148 case T_libraries: // The section libraries starts here.
149 // List of libraries in use.
150 // Not used here, just skip it
151 skipCurrent();
152 break;
153
154 case T_design: // The section design starts here.
155 // Not used (mainly they are comments), just skip it
156 skipCurrent();
157 break;
158
159 case T_RIGHT: // The closing parenthesis of the file.
160 plevel--;
161 break;
162
163 default:
164 skipCurrent();
165 break;
166 }
167 }
168
169 if( plevel != 0 )
170 {
171 wxFAIL_MSG( wxString::Format( wxT( "KICAD_NETLIST_PARSER::Parse(): bad parenthesis "
172 "count (count = %d" ),
173 plevel ) );
174 }
175}
176
177
179{
180 /* Parses a section like
181 * (net (code 20) (name /PC-A0)
182 * (node (ref "BUS1") (pin "62)")
183 * (node (ref "U3") ("pin 3") (pin_function "clock"))
184 * (node (ref "U9") (pin "M6") (pin_function "reset")))
185 */
186
187 wxString code;
188 wxString name;
189 wxString reference;
190 wxString pin_number;
191 wxString pin_function;
192 wxString pin_type;
193
194 // The token net was read, so the next data is (code <number>)
195 while( (token = NextTok() ) != T_EOF )
196 {
197 if( token == T_RIGHT )
198 break;
199 else if( token == T_LEFT )
200 token = NextTok();
201
202 switch( token )
203 {
204 case T_code:
205 NeedSYMBOLorNUMBER();
206 code = From_UTF8( CurText() );
207 NeedRIGHT();
208 break;
209
210 case T_name:
211 NeedSYMBOLorNUMBER();
212 name = From_UTF8( CurText() );
213 NeedRIGHT();
214 break;
215
216 case T_node:
217 // By default: no pin function or type.
218 pin_function.Clear();
219 pin_type.Clear();
220
221 while( (token = NextTok() ) != T_EOF )
222 {
223 if( token == T_RIGHT )
224 break;
225 else if( token == T_LEFT )
226 token = NextTok();
227
228 switch( token )
229 {
230 case T_ref:
231 NeedSYMBOLorNUMBER();
232 reference = From_UTF8( CurText() );
233 NeedRIGHT();
234 break;
235
236 case T_pin:
237 NeedSYMBOLorNUMBER();
238 pin_number = From_UTF8( CurText() );
239 NeedRIGHT();
240 break;
241
242 case T_pinfunction:
243 NeedSYMBOLorNUMBER();
244 pin_function = From_UTF8( CurText() );
245 NeedRIGHT();
246 break;
247
248 case T_pintype:
249 NeedSYMBOLorNUMBER();
250 pin_type = From_UTF8( CurText() );
251 NeedRIGHT();
252 break;
253
254 default:
255 skipCurrent();
256 break;
257 }
258 }
259
260 // Don't assume component will be found; it might be "DNP" or "Exclude from board".
261 if( COMPONENT* component = m_netlist->GetComponentByReference( reference ) )
262 {
263 if( strtol( code.c_str(), nullptr, 10 ) >= 1 )
264 {
265 if( name.IsEmpty() ) // Give a dummy net name like N-000009
266 name = wxT("N-00000") + code;
267
268 component->AddNet( pin_number, name, pin_function, pin_type );
269 }
270 }
271
272 break;
273
274 default:
275 skipCurrent();
276 break;
277 }
278 }
279}
280
281
283{
284 /* Parses a section like
285 * (comp (ref P1)
286 * (value DB25FEMALE)
287 * (footprint DB25FC)
288 * (libsource (lib conn) (part DB25))
289 * (property (name PINCOUNT) (value 25))
290 * (sheetpath (names /) (tstamps /))
291 * (component_classes (class (name "CLASS")))
292 * (tstamp 68183921-93a5-49ac-91b0-49d05a0e1647))
293 *
294 * other fields (unused) are skipped
295 * A component need a reference, value, footprint name and a full time stamp
296 * The full time stamp is the sheetpath time stamp + the component time stamp
297 */
298 LIB_ID fpid;
299 wxString footprint;
300 wxString ref;
301 wxString value;
302 wxString library;
303 wxString name;
304 wxString humanSheetPath;
306
307 std::vector<KIID> uuids;
308 std::map<wxString, wxString> properties;
309 nlohmann::ordered_map<wxString, wxString> fields;
310 std::unordered_set<wxString> componentClasses;
311
312 bool duplicatePinsAreJumpers = false;
313 std::vector<std::set<wxString>> jumperPinGroups;
314
315 // The token comp was read, so the next data is (ref P1)
316 while( (token = NextTok() ) != T_RIGHT )
317 {
318 if( token == T_LEFT )
319 token = NextTok();
320
321 switch( token )
322 {
323 case T_ref:
324 NeedSYMBOLorNUMBER();
325 ref = From_UTF8( CurText() );
326 NeedRIGHT();
327 break;
328
329 case T_value:
330 NeedSYMBOLorNUMBER();
331 value = From_UTF8( CurText() );
332 NeedRIGHT();
333 break;
334
335 case T_footprint:
336 NeedSYMBOLorNUMBER();
337 footprint = FromUTF8();
338 NeedRIGHT();
339 break;
340
341 case T_libsource:
342 // Read libsource
343 while( ( token = NextTok() ) != T_RIGHT )
344 {
345 if( token == T_LEFT )
346 token = NextTok();
347
348 if( token == T_lib )
349 {
350 NeedSYMBOLorNUMBER();
351 library = From_UTF8( CurText() );
352 NeedRIGHT();
353 }
354 else if( token == T_part )
355 {
356 NeedSYMBOLorNUMBER();
357 name = From_UTF8( CurText() );
358 NeedRIGHT();
359 }
360 else if( token == T_description )
361 {
362 NeedSYMBOLorNUMBER();
363 NeedRIGHT();
364 }
365 else
366 {
367 Expecting( "part, lib or description" );
368 }
369 }
370 break;
371
372 case T_property:
373 {
374 wxString propName;
375 wxString propValue;
376
377 while( (token = NextTok() ) != T_RIGHT )
378 {
379 if( token == T_LEFT )
380 token = NextTok();
381
382 if( token == T_name )
383 {
384 NeedSYMBOLorNUMBER();
385 propName = From_UTF8( CurText() );
386 NeedRIGHT();
387 }
388 else if( token == T_value )
389 {
390 NeedSYMBOLorNUMBER();
391 propValue = From_UTF8( CurText() );
392 NeedRIGHT();
393 }
394 else
395 {
396 Expecting( "name or value" );
397 }
398 }
399
400 if( !propName.IsEmpty() )
401 properties[ propName ] = propValue;
402 }
403 break;
404
405 case T_fields:
406 while( ( token = NextTok() ) != T_RIGHT )
407 {
408 if( token == T_LEFT )
409 token = NextTok();
410
411 if( token == T_field )
412 {
413 wxString fieldName;
414 wxString fieldValue;
415
416 while( ( token = NextTok() ) != T_RIGHT )
417 {
418 if( token == T_LEFT )
419 token = NextTok();
420
421 if( token == T_name )
422 {
423 NeedSYMBOLorNUMBER();
424 fieldName = From_UTF8( CurText() );
425 NeedRIGHT();
426 }
427 else if( token == T_STRING )
428 {
429 fieldValue = From_UTF8( CurText() );
430 }
431 }
432
433 if( !fieldName.IsEmpty() )
434 fields[fieldName] = fieldValue;
435 }
436 else
437 {
438 Expecting( "field" );
439 }
440 }
441 break;
442
443 case T_sheetpath:
444 while( ( token = NextTok() ) != T_EOF )
445 {
446 if( token == T_names )
447 {
448 NeedSYMBOLorNUMBER();
449 humanSheetPath = From_UTF8( CurText() );
450 NeedRIGHT();
451 }
452
453 if( token == T_tstamps )
454 {
455 NeedSYMBOLorNUMBER();
456 path = KIID_PATH( From_UTF8( CurText() ) );
457 NeedRIGHT();
458 break;
459 }
460 }
461
462 NeedRIGHT();
463
464 break;
465
466 case T_tstamps:
467 while( ( token = NextTok() ) != T_EOF )
468 {
469 if( token == T_RIGHT )
470 break;
471
472 uuids.emplace_back( From_UTF8( CurText() ) );
473 }
474
475 break;
476
477 case T_component_classes:
478 while( ( token = NextTok() ) != T_RIGHT )
479 {
480 if( token != T_LEFT )
481 Expecting( T_LEFT );
482
483 if( ( token = NextTok() ) != T_class )
484 Expecting( T_class );
485
486 NeedSYMBOLorNUMBER();
487 componentClasses.insert( From_UTF8( CurText() ) );
488 NeedRIGHT();
489 }
490
491 break;
492
493 case T_duplicate_pin_numbers_are_jumpers:
494 {
495 NeedSYMBOLorNUMBER();
496 duplicatePinsAreJumpers = From_UTF8( CurText() ) == wxT( "1" );
497 NeedRIGHT();
498 break;
499 }
500
501 case T_jumper_pin_groups:
502 {
503 std::set<wxString>* currentGroup = nullptr;
504
505 for( token = NextTok(); currentGroup || token != T_RIGHT; token = NextTok() )
506 {
507 if( token == T_LEFT )
508 token = NextTok();
509
510 switch( token )
511 {
512 case T_group:
513 currentGroup = &jumperPinGroups.emplace_back();
514 break;
515
516 case T_pin:
517 {
518 NeedSYMBOLorNUMBER();
519 wxString padName = From_UTF8( CurText() );
520 NeedRIGHT();
521 wxCHECK2( currentGroup, continue );
522 currentGroup->insert( padName );
523 break;
524 }
525
526 case T_RIGHT:
527 currentGroup = nullptr;
528 break;
529
530 default:
531 Expecting( "group or pin" );
532 }
533 }
534
535 break;
536 }
537
538 default:
539 // Skip not used data (i.e all other tokens)
540 skipCurrent();
541 break;
542 }
543 }
544
545 if( !footprint.IsEmpty() && fpid.Parse( footprint, true ) >= 0 )
546 {
547 wxString error;
548 error.Printf( _( "Invalid footprint ID in\nfile: '%s'\nline: %d\nofff: %d" ),
549 CurSource(), CurLineNumber(), CurOffset() );
550
551 THROW_IO_ERROR( error );
552 }
553
554 COMPONENT* component = new COMPONENT( fpid, ref, value, path, uuids );
555 component->SetName( name );
556 component->SetLibrary( library );
557 component->SetProperties( properties );
558 component->SetFields( fields );
559 component->SetHumanReadablePath( humanSheetPath );
560 component->SetComponentClassNames( componentClasses );
561 component->SetDuplicatePadNumbersAreJumpers( duplicatePinsAreJumpers );
562 std::ranges::copy( jumperPinGroups,
563 std::inserter( component->JumperPadGroups(), component->JumperPadGroups().end() ) );
564 m_netlist->AddComponent( component );
565}
566
567
569{
570 /* Parses a section like
571 * (libpart (lib device) (part C)
572 * (aliases
573 * (alias Cxx)
574 * (alias Cyy))
575 * (description "Condensateur non polarise")
576 * (footprints
577 * (fp SM*)
578 * (fp C?)
579 * (fp C1-1))
580 * (fields
581 * (field (name Reference) C)
582 * (field (name Value) C))
583 * (pins
584 * (pin (num 1) (name ~) (type passive))
585 * (pin (num 2) (name ~) (type passive))))
586 *
587 * Currently footprints section/fp are read and data stored
588 * other fields (unused) are skipped
589 */
590 COMPONENT* component = NULL;
591 wxString libName;
592 wxString libPartName;
593 wxArrayString footprintFilters;
594 wxArrayString aliases;
595 int pinCount = 0;
596
597 // The last token read was libpart, so read the next token
598 while( (token = NextTok() ) != T_RIGHT )
599 {
600 if( token == T_LEFT )
601 token = NextTok();
602
603 switch( token )
604 {
605 case T_lib:
606 NeedSYMBOLorNUMBER();
607 libName = From_UTF8( CurText() );
608 NeedRIGHT();
609 break;
610
611 case T_part:
612 NeedSYMBOLorNUMBER();
613 libPartName = From_UTF8( CurText() );
614 NeedRIGHT();
615 break;
616
617 case T_footprints:
618 // Read all fp elements (footprint filter item)
619 while( (token = NextTok() ) != T_RIGHT )
620 {
621 if( token == T_LEFT )
622 token = NextTok();
623
624 if( token != T_fp )
625 Expecting( T_fp );
626
627 NeedSYMBOLorNUMBER();
628 footprintFilters.Add( From_UTF8( CurText() ) );
629 NeedRIGHT();
630 }
631 break;
632
633 case T_aliases:
634 while( (token = NextTok() ) != T_RIGHT )
635 {
636 if( token == T_LEFT )
637 token = NextTok();
638
639 if( token != T_alias )
640 Expecting( T_alias );
641
642 NeedSYMBOLorNUMBER();
643 aliases.Add( From_UTF8( CurText() ) );
644 NeedRIGHT();
645 }
646 break;
647
648 case T_pins:
649 while( (token = NextTok() ) != T_RIGHT )
650 {
651 if( token == T_LEFT )
652 token = NextTok();
653
654 if( token != T_pin )
655 Expecting( T_pin );
656
657 pinCount++;
658
659 skipCurrent();
660 }
661 break;
662
663 default:
664 // Skip not used data (i.e all other tokens)
665 skipCurrent();
666 break;
667 }
668 }
669
670 // Find all of the components that reference this component library part definition.
671 for( unsigned i = 0; i < m_netlist->GetCount(); i++ )
672 {
673 component = m_netlist->GetComponent( i );
674
675 if( component->IsLibSource( libName, libPartName ) )
676 {
677 component->SetFootprintFilters( footprintFilters );
678 component->SetPinCount( pinCount );
679 }
680
681 for( unsigned jj = 0; jj < aliases.GetCount(); jj++ )
682 {
683 if( component->IsLibSource( libName, aliases[jj] ) )
684 {
685 component->SetFootprintFilters( footprintFilters );
686 component->SetPinCount( pinCount );
687 }
688 }
689
690 }
691}
const char * name
Definition: DXF_plotter.cpp:59
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.
Definition: pcb_netlist.h:88
void SetFields(nlohmann::ordered_map< wxString, wxString > &aFields)
Definition: pcb_netlist.h:135
void SetLibrary(const wxString &aLibrary)
Definition: pcb_netlist.h:126
void SetProperties(std::map< wxString, wxString > &aProps)
Definition: pcb_netlist.h:141
void SortPins()
Definition: pcb_netlist.h:121
void SetPinCount(int aPinCount)
Definition: pcb_netlist.h:160
bool IsLibSource(const wxString &aLibrary, const wxString &aName) const
Definition: pcb_netlist.h:170
void SetFootprintFilters(const wxArrayString &aFilters)
Definition: pcb_netlist.h:157
void SetHumanReadablePath(const wxString &aPath)
Definition: pcb_netlist.h:177
void SetDuplicatePadNumbersAreJumpers(bool aEnabled)
Definition: pcb_netlist.h:188
void SetComponentClassNames(const std::unordered_set< wxString > &aClassNames)
Definition: pcb_netlist.h:180
std::vector< std::set< wxString > > & JumperPadGroups()
Definition: pcb_netlist.h:190
void SetName(const wxString &aName)
Definition: pcb_netlist.h:123
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.
Definition: lib_id.h:49
int Parse(const UTF8 &aId, bool aFix=false)
Parse LIB_ID with the information from aId.
Definition: lib_id.cpp:52
An abstract class from which implementation specific LINE_READERs may be derived to read single lines...
Definition: richio.h:93
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.
Definition: pcb_netlist.h:255
unsigned GetCount() const
Definition: pcb_netlist.h:276
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.
Definition: pcb_netlist.h:284
#define _(s)
#define THROW_IO_ERROR(msg)
Definition: ki_exception.h:39
wxString From_UTF8(const char *cstring)