KiCad PCB EDA Suite
Loading...
Searching...
No Matches
board_stackup.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) 2019 Jean-Pierre Charras, jp.charras at wanadoo.fr
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 3
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
21
22#include "board_stackup.h"
23#include <base_units.h>
24#include <string_utils.h>
25#include <layer_ids.h>
27#include <board.h>
28#include <i18n_utility.h> // For _HKI definition
31#include <richio.h>
32#include <google/protobuf/any.pb.h>
33#include <api/board/board.pb.h>
34#include <api/api_enums.h>
35
36
38{
39 if( m_Material != aOther.m_Material ) return false;
40 if( m_Thickness != aOther.m_Thickness ) return false;
41 if( m_ThicknessLocked != aOther.m_ThicknessLocked ) return false;
42 if( m_EpsilonR != aOther.m_EpsilonR ) return false;
43 if( m_LossTangent != aOther.m_LossTangent ) return false;
44 if( m_Color != aOther.m_Color ) return false;
45
46 return true;
47}
48
49
51{
52 DIELECTRIC_PRMS item_prms;
53 m_DielectricPrmsList.emplace_back( item_prms );
55 m_Type = aType;
57 SetEnabled( true );
58
59 // Initialize parameters to a usual value for allowed types:
60 switch( m_Type )
61 {
65 break;
66
68 m_TypeName = KEY_CORE; // or prepreg
70 SetMaterial( wxT( "FR4" ) ); // or other dielectric name
71 SetLossTangent( 0.02 ); // for FR4
72 SetEpsilonR( 4.5 ); // for FR4
73 SetSpecFreq( 1e9 ); // for FR4
75 break;
76
78 m_TypeName = wxT( "solderpaste" );
79 break;
80
82 m_TypeName = wxT( "soldermask" );
84 SetMaterial( NotSpecifiedPrm() ); // or other solder mask material name
87 break;
88
90 m_TypeName = wxT( "silkscreen" );
92 SetMaterial( NotSpecifiedPrm() ); // or other silkscreen material name
94 break;
95
97 break;
98 }
99}
100
101
112
113
115{
116 if( m_Type != aOther.m_Type ) return false;
117 if( m_LayerName != aOther.m_LayerName ) return false;
118 if( m_TypeName != aOther.m_TypeName ) return false;
119 if( m_LayerId != aOther.m_LayerId ) return false;
120 if( m_DielectricLayerId != aOther.m_DielectricLayerId ) return false;
121 if( m_enabled != aOther.m_enabled ) return false;
122
123 if( !std::equal( std::begin( m_DielectricPrmsList ), std::end( m_DielectricPrmsList ),
124 std::begin( aOther.m_DielectricPrmsList ),
125 []( const DIELECTRIC_PRMS& aA, const DIELECTRIC_PRMS& aB )
126 {
127 return aA == aB;
128 } ) )
129 {
130 return false;
131 }
132
133 return true;
134}
135
136
137void BOARD_STACKUP_ITEM::AddDielectricPrms( int aDielectricPrmsIdx )
138{
139 // add a DIELECTRIC_PRMS item to m_DielectricPrmsList
140 DIELECTRIC_PRMS new_prms;
141
142 m_DielectricPrmsList.emplace( m_DielectricPrmsList.begin() + aDielectricPrmsIdx, new_prms );
143}
144
145
146void BOARD_STACKUP_ITEM::RemoveDielectricPrms( int aDielectricPrmsIdx )
147{
148 // Remove a DIELECTRIC_PRMS item from m_DielectricPrmsList if possible
149
150 if( GetSublayersCount() < 2
151 || aDielectricPrmsIdx < 0
152 || aDielectricPrmsIdx >= GetSublayersCount() )
153 {
154 return;
155 }
156
157 m_DielectricPrmsList.erase( m_DielectricPrmsList.begin() + aDielectricPrmsIdx );
158}
159
160
161
163{
164 // A reasonable thickness for copper layers:
165 return pcbIUScale.mmToIU( 0.035 );
166}
167
168
170{
171 // A reasonable thickness for solder mask:
172 return pcbIUScale.mmToIU( 0.01 );
173}
174
175
176// Getters:
177wxString BOARD_STACKUP_ITEM::GetColor( int aDielectricSubLayer ) const
178{
179 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
180
181 return m_DielectricPrmsList[aDielectricSubLayer].m_Color;
182}
183
184int BOARD_STACKUP_ITEM::GetThickness( int aDielectricSubLayer ) const
185{
186 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
187
188 return m_DielectricPrmsList[aDielectricSubLayer].m_Thickness;
189}
190
191
192double BOARD_STACKUP_ITEM::GetLossTangent( int aDielectricSubLayer ) const
193{
194 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
195
196 return m_DielectricPrmsList[aDielectricSubLayer].m_LossTangent;
197}
198
199
200double BOARD_STACKUP_ITEM::GetSpecFreq( int aDielectricSubLayer ) const
201{
202 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
203
204 return m_DielectricPrmsList[aDielectricSubLayer].m_SpecFreq;
205}
206
207
209{
210 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
211
212 return m_DielectricPrmsList[aDielectricSubLayer].m_DielectricModel;
213}
214
215
216double BOARD_STACKUP_ITEM::GetEpsilonR( int aDielectricSubLayer ) const
217{
218 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
219
220 return m_DielectricPrmsList[aDielectricSubLayer].m_EpsilonR;
221}
222
223
224bool BOARD_STACKUP_ITEM::IsThicknessLocked( int aDielectricSubLayer ) const
225{
226 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
227
228 return m_DielectricPrmsList[aDielectricSubLayer].m_ThicknessLocked;
229}
230
231
232wxString BOARD_STACKUP_ITEM::GetMaterial( int aDielectricSubLayer ) const
233{
234 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
235
236 return m_DielectricPrmsList[aDielectricSubLayer].m_Material;
237}
238
239
240// Setters:
241void BOARD_STACKUP_ITEM::SetColor( const wxString& aColorName , int aDielectricSubLayer )
242{
243 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
244
245 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
246 m_DielectricPrmsList[aDielectricSubLayer].m_Color = aColorName;
247}
248
249
250void BOARD_STACKUP_ITEM::SetThickness( int aThickness, int aDielectricSubLayer )
251{
252 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
253
254 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
255 m_DielectricPrmsList[aDielectricSubLayer].m_Thickness = aThickness;
256}
257
258
259void BOARD_STACKUP_ITEM::SetLossTangent( double aTg, int aDielectricSubLayer )
260{
261 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
262
263 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
264 m_DielectricPrmsList[aDielectricSubLayer].m_LossTangent = aTg;
265}
266
267
268void BOARD_STACKUP_ITEM::SetSpecFreq( const double aSpecFreq, const int aDielectricSubLayer )
269{
270 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
271
272 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
273 m_DielectricPrmsList[aDielectricSubLayer].m_SpecFreq = aSpecFreq;
274}
275
276
277void BOARD_STACKUP_ITEM::SetDielectricModel( DIELECTRIC_MODEL aModel, int aDielectricSubLayer )
278{
279 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
280
281 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
282 m_DielectricPrmsList[aDielectricSubLayer].m_DielectricModel = aModel;
283}
284
285
286void BOARD_STACKUP_ITEM::SetEpsilonR( double aEpsilon, int aDielectricSubLayer )
287{
288 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
289
290 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
291 m_DielectricPrmsList[aDielectricSubLayer].m_EpsilonR = aEpsilon;
292}
293
294
295void BOARD_STACKUP_ITEM::SetThicknessLocked( bool aLocked, int aDielectricSubLayer )
296{
297 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
298
299 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
300 m_DielectricPrmsList[aDielectricSubLayer].m_ThicknessLocked = aLocked;
301}
302
303
304void BOARD_STACKUP_ITEM::SetMaterial( const wxString& aName, int aDielectricSubLayer )
305{
306 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
307
308 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
309 m_DielectricPrmsList[aDielectricSubLayer].m_Material = aName;
310}
311
312
318
319
325
326
331
332
333bool BOARD_STACKUP_ITEM::HasMaterialValue( int aDielectricSubLayer ) const
334{
335 // return true if the material is specified
336 return IsMaterialEditable() && IsPrmSpecified( GetMaterial( aDielectricSubLayer ) );
337}
338
339
346
347
354
355
362
363
364wxString BOARD_STACKUP_ITEM::FormatEpsilonR( int aDielectricSubLayer ) const
365{
366 // return a wxString to print/display Epsilon R
367 // note: we do not want scientific notation
368 wxString txt = UIDouble2Str( GetEpsilonR( aDielectricSubLayer ) );
369 return txt;
370}
371
372
373wxString BOARD_STACKUP_ITEM::FormatLossTangent( int aDielectricSubLayer ) const
374{
375 // return a wxString to print/display Loss Tangent
376 // note: we do not want scientific notation
377 wxString txt = UIDouble2Str( GetLossTangent( aDielectricSubLayer ) );
378 return txt;
379}
380
381
383{
384 // return a wxString to print/display a dielectric name
385 wxString lname;
386 lname.Printf( _( "Dielectric %d" ), GetDielectricLayerId() );
387
388 return lname;
389}
390
391
393{
394 m_HasDielectricConstrains = false; // True if some dielectric layers have constrains
395 // (Loss tg and Epison R)
396 m_HasThicknessConstrains = false; // True if some dielectric or copper layers have constrains
398 m_EdgePlating = false; // True if edge board is plated
399 m_FinishType = wxT( "None" ); // undefined finish type
400}
401
402
404{
409 m_FinishType = aOther.m_FinishType;
410
411 // All items in aOther.m_list have to be duplicated, because aOther.m_list
412 // manage pointers to these items
413 for( BOARD_STACKUP_ITEM* item : aOther.m_list )
414 {
415 BOARD_STACKUP_ITEM* dup_item = new BOARD_STACKUP_ITEM( *item );
416 Add( dup_item );
417 }
418}
419
420
422{
427 m_FinishType = aOther.m_FinishType;
428
429 RemoveAll();
430
431 // All items in aOther.m_list have to be duplicated, because aOther.m_list
432 // manage pointers to these items
433 for( BOARD_STACKUP_ITEM* item : aOther.m_list )
434 {
435 BOARD_STACKUP_ITEM* dup_item = new BOARD_STACKUP_ITEM( *item );
436 Add( dup_item );
437 }
438
439 return *this;
440}
441
442
443bool BOARD_STACKUP::operator==( const BOARD_STACKUP& aOther ) const
444{
445 if( m_HasDielectricConstrains != aOther.m_HasDielectricConstrains ) return false;
446 if( m_HasThicknessConstrains != aOther.m_HasThicknessConstrains ) return false;
447 if( m_EdgeConnectorConstraints != aOther.m_EdgeConnectorConstraints ) return false;
448 if( m_EdgePlating != aOther.m_EdgePlating ) return false;
449 if( m_FinishType != aOther.m_FinishType ) return false;
450
451 if( m_list.size() != aOther.m_list.size() )
452 return false;
453
454 if( !std::equal( std::begin( m_list ), std::end( m_list ), std::begin( aOther.m_list ),
455 []( const BOARD_STACKUP_ITEM* aA, const BOARD_STACKUP_ITEM* aB )
456 {
457 return *aA == *aB;
458 } ) )
459 {
460 return false;
461 }
462
463 return true;
464}
465
466
467void BOARD_STACKUP::Serialize( google::protobuf::Any& aContainer ) const
468{
469 using namespace kiapi::board;
470 BoardStackup stackup;
471
472 for( const BOARD_STACKUP_ITEM* item : m_list )
473 {
474 BoardStackupLayer* layer = stackup.mutable_layers()->Add();
475
476 layer->mutable_thickness()->set_value_nm( item->GetThickness() );
477 layer->set_layer( ToProtoEnum<PCB_LAYER_ID, types::BoardLayer>( item->GetBrdLayerId() ) );
478 layer->set_type(
480
481 switch( item->GetType() )
482 {
484 {
485 layer->set_material_name( "copper" );
486 // (no copper params yet...)
487 break;
488 }
489
491 {
492 BoardStackupDielectricLayer* dielectric = layer->mutable_dielectric()->New();
493
494 for( int i = 0; i < item->GetSublayersCount(); ++i )
495 {
496 BoardStackupDielectricProperties* props = dielectric->mutable_layer()->Add();
497 props->set_epsilon_r( item->GetEpsilonR( i ) );
498 props->set_loss_tangent( item->GetLossTangent( i ) );
499 props->set_material_name( item->GetMaterial( i ).ToUTF8() );
500 props->mutable_thickness()->set_value_nm( item->GetThickness( i ) );
501 }
502
503 break;
504 }
505
506 default:
507 break;
508 }
509 }
510
511 aContainer.PackFrom( stackup );
512}
513
514
515bool BOARD_STACKUP::Deserialize( const google::protobuf::Any& aContainer )
516{
517 // Read-only for now
518 return false;
519}
520
521
523{
524 for( BOARD_STACKUP_ITEM* item : m_list )
525 delete item;
526
527 m_list.clear();
528}
529
530
532{
533 if( aIndex < 0 || aIndex >= GetCount() )
534 return nullptr;
535
536 return GetList()[aIndex];
537}
538
539
541{
542 // return the board thickness from the thickness of BOARD_STACKUP_ITEM list
543 int thickness = 0;
544
545 for( BOARD_STACKUP_ITEM* item : m_list )
546 {
547 if( item->IsThicknessEditable() && item->IsEnabled() )
548 {
549 thickness += item->GetThickness();
550
551 // dielectric layers can have more than one main layer
552 // add thickness of all sublayers
553 for( int idx = 1; idx < item->GetSublayersCount(); idx++ )
554 {
555 thickness += item->GetThickness( idx );
556 }
557 }
558 }
559
560 return thickness;
561}
562
563
565{
566 bool change = false;
567 // Build the suitable stackup:
568 BOARD_STACKUP stackup;
569 stackup.BuildDefaultStackupList( aSettings );
570
571 // First, find removed layers:
572 for( BOARD_STACKUP_ITEM* curr_item: m_list )
573 {
574 bool found = false;
575
576 for( BOARD_STACKUP_ITEM* item: stackup.GetList() )
577 {
578 if( curr_item->GetBrdLayerId() != UNDEFINED_LAYER )
579 {
580 if( item->GetBrdLayerId() == curr_item->GetBrdLayerId() )
581 {
582 found = true;
583 break;
584 }
585 }
586 else // curr_item = dielectric layer
587 {
588 if( item->GetBrdLayerId() != UNDEFINED_LAYER )
589 continue;
590
591 if( item->GetDielectricLayerId() == curr_item->GetDielectricLayerId() )
592 {
593 found = true;
594 break;
595 }
596 }
597 }
598
599 if( !found ) // a layer was removed: a change is found
600 {
601 change = true;
602 break;
603 }
604 }
605
606 // Now initialize all stackup items to the initial values, when exist
607 for( BOARD_STACKUP_ITEM* item : stackup.GetList() )
608 {
609 bool found = false;
610 // Search for initial settings:
611 for( const BOARD_STACKUP_ITEM* initial_item : m_list )
612 {
613 if( item->GetBrdLayerId() != UNDEFINED_LAYER )
614 {
615 if( item->GetBrdLayerId() == initial_item->GetBrdLayerId() )
616 {
617 *item = *initial_item;
618 found = true;
619 break;
620 }
621 }
622 else // dielectric layer: see m_DielectricLayerId for identification
623 {
624 // Compare dielectric layer with dielectric layer
625 if( initial_item->GetBrdLayerId() != UNDEFINED_LAYER )
626 continue;
627
628 if( item->GetDielectricLayerId() == initial_item->GetDielectricLayerId() )
629 {
630 *item = *initial_item;
631 found = true;
632 break;
633 }
634 }
635 }
636
637 if( !found )
638 {
639 change = true;
640 }
641 }
642
643 // Transfer layer settings:
644 *this = stackup;
645
646 // Transfer other stackup settings from aSettings
647 const BOARD_STACKUP& source_stackup = aSettings->GetStackupDescriptor();
650 m_EdgePlating = source_stackup.m_EdgePlating;
651 m_FinishType = source_stackup.m_FinishType;
652
653 return change;
654}
655
656
658 int aActiveCopperLayersCount )
659{
660 // Creates a default stackup, according to the current BOARD_DESIGN_SETTINGS settings.
661 // Note: the m_TypeName string is made translatable using _HKI marker, but is not
662 // translated when building the stackup.
663 // It will be used as this in files, and can be translated only in dialog
664 // if aSettings == NULL, build a full stackup (with 32 copper layers)
665 LSET enabledLayer = aSettings ? aSettings->GetEnabledLayers() : StackupAllowedBrdLayers();
666 int copperLayerCount = aSettings ? aSettings->GetCopperLayerCount() : 32;
667
668 // We need to calculate a suitable dielectric layer thickness.
669 // If no settings, and if aActiveCopperLayersCount is given, use it
670 // (If no settings, and no aActiveCopperLayersCount, the full 32 layers are used)
671 int activeCuLayerCount = copperLayerCount;
672
673 if( aSettings == nullptr && aActiveCopperLayersCount > 0 )
674 activeCuLayerCount = aActiveCopperLayersCount;
675
676 int brd__thickness = aSettings ? aSettings->GetBoardThickness() : pcbIUScale.mmToIU( 1.6 );
677 int diel_thickness = brd__thickness -
678 ( BOARD_STACKUP_ITEM::GetCopperDefaultThickness() * activeCuLayerCount );
679
680 // Take in account the solder mask thickness:
681 int sm_count = ( enabledLayer & LSET( { F_Mask, B_Mask } ) ).count();
682 diel_thickness -= BOARD_STACKUP_ITEM::GetMaskDefaultThickness() * sm_count;
683 diel_thickness /= std::max( 1, activeCuLayerCount - 1 );
684
685 int dielectric_idx = 0;
686
687 // Add silk screen, solder mask and solder paste layers on top
688 if( enabledLayer[F_SilkS] )
689 {
691 item->SetBrdLayerId( F_SilkS );
692 item->SetTypeName( _HKI( "Top Silk Screen" ) );
693 Add( item );
694 }
695
696 if( enabledLayer[F_Paste] )
697 {
699 item->SetBrdLayerId( F_Paste );
700 item->SetTypeName( _HKI( "Top Solder Paste" ) );
701 Add( item );
702 }
703
704 if( enabledLayer[F_Mask] )
705 {
707 item->SetBrdLayerId( F_Mask );
708 item->SetTypeName( _HKI( "Top Solder Mask" ) );
709 Add( item );
710 }
711
712 // Add copper and dielectric layers
713 for( PCB_LAYER_ID layer : enabledLayer.CuStack() )
714 {
716 item->SetBrdLayerId( layer );
717 item->SetTypeName( KEY_COPPER );
718 Add( item );
719
720 if( layer == B_Cu )
721 break;
722
723 // Add the dielectric layer:
725 item->SetThickness( diel_thickness );
726 item->SetDielectricLayerId( dielectric_idx + 1 );
727
728 // Display a dielectric default layer name:
729 if( (dielectric_idx & 1) == 0 )
730 {
731 item->SetTypeName( KEY_CORE );
732 item->SetMaterial( wxT( "FR4" ) );
733 }
734 else
735 {
736 item->SetTypeName( KEY_PREPREG );
737 item->SetMaterial( wxT( "FR4" ) );
738 }
739
740 Add( item );
741 dielectric_idx++;
742 }
743
744 // Add silk screen, solder mask and solder paste layers on bottom
745 if( enabledLayer[B_Mask] )
746 {
748 item->SetBrdLayerId( B_Mask );
749 item->SetTypeName( _HKI( "Bottom Solder Mask" ) );
750 Add( item );
751 }
752
753 if( enabledLayer[B_Paste] )
754 {
756 item->SetBrdLayerId( B_Paste );
757 item->SetTypeName( _HKI( "Bottom Solder Paste" ) );
758 Add( item );
759 }
760
761 if( enabledLayer[B_SilkS] )
762 {
764 item->SetBrdLayerId( B_SilkS );
765 item->SetTypeName( _HKI( "Bottom Silk Screen" ) );
766 Add( item );
767 }
768
769 // Transfer other stackup settings from aSettings
770 if( aSettings )
771 {
772 const BOARD_STACKUP& source_stackup = aSettings->GetStackupDescriptor();
775 m_EdgePlating = source_stackup.m_EdgePlating;
776 m_FinishType = source_stackup.m_FinishType;
777 }
778}
779
780
781void BOARD_STACKUP::FormatBoardStackup( OUTPUTFORMATTER* aFormatter, const BOARD* aBoard ) const
782{
783 // Board stackup is the ordered list from top to bottom of
784 // physical layers and substrate used to build the board.
785 if( m_list.empty() )
786 return;
787
788 aFormatter->Print( "(stackup" );
789
790 // Note:
791 // Unspecified parameters are not stored in file.
792 for( BOARD_STACKUP_ITEM* item: m_list )
793 {
794 wxString layer_name;
795
796 if( item->GetBrdLayerId() == UNDEFINED_LAYER )
797 layer_name.Printf( wxT( "dielectric %d" ), item->GetDielectricLayerId() );
798 else
799 layer_name = LSET::Name( item->GetBrdLayerId() );
800
801 aFormatter->Print( "(layer %s (type %s)",
802 aFormatter->Quotew( layer_name ).c_str(),
803 aFormatter->Quotew( item->GetTypeName() ).c_str() );
804
805 // Output other parameters (in sub layer list there is at least one item)
806 for( int idx = 0; idx < item->GetSublayersCount(); idx++ )
807 {
808 if( idx ) // not for the main (first) layer.
809 aFormatter->Print( " addsublayer" );
810
811 if( item->IsColorEditable() && IsPrmSpecified( item->GetColor( idx ) ) )
812 {
813 aFormatter->Print( "(color %s)",
814 aFormatter->Quotew( item->GetColor( idx ) ).c_str() );
815 }
816
817 if( item->IsThicknessEditable() )
818 {
819 aFormatter->Print( "(thickness %s",
820 EDA_UNIT_UTILS::FormatInternalUnits( pcbIUScale, item->GetThickness( idx ) ).c_str() );
821
822 if( item->GetType() == BS_ITEM_TYPE_DIELECTRIC && item->IsThicknessLocked( idx ) )
823 aFormatter->Print( " locked" );
824
825 aFormatter->Print( ")" );
826 }
827
828 if( item->HasMaterialValue( idx ) )
829 {
830 aFormatter->Print( "(material %s)",
831 aFormatter->Quotew( item->GetMaterial( idx ) ).c_str() );
832 }
833
834 if( item->HasEpsilonRValue() && item->HasMaterialValue( idx ) )
835 aFormatter->Print( "(epsilon_r %s)", FormatDouble2Str( item->GetEpsilonR( idx ) ).c_str() );
836
837 if( item->HasLossTangentValue() && item->HasMaterialValue( idx ) )
838 {
839 aFormatter->Print( "(loss_tangent %s)",
840 FormatDouble2Str( item->GetLossTangent( idx ) ).c_str() );
841 }
842
843 if( item->HasSpecFreqValue() && item->HasMaterialValue( idx ) )
844 {
845 aFormatter->Print( "(spec_frequency %s)", FormatDouble2Str( item->GetSpecFreq( idx ) ).c_str() );
846
847 wxString modelToken = wxT( "constant" );
848
849 if( item->GetDielectricModel( idx ) == DIELECTRIC_MODEL::DJORDJEVIC_SARKAR )
850 modelToken = wxT( "djordjevic_sarkar" );
851
852 aFormatter->Print( "(dielectric_model %s)", modelToken.ToStdString().c_str() );
853 }
854 }
855
856 aFormatter->Print( ")" );
857 }
858
859 // Other infos about board, related to layers and other fabrication specifications
861 aFormatter->Print( "(copper_finish %s)", aFormatter->Quotew( m_FinishType ).c_str() );
862
863 KICAD_FORMAT::FormatBool( aFormatter, "dielectric_constraints", m_HasDielectricConstrains );
864
866 {
867 aFormatter->Print( "(edge_connector %s)",
868 m_EdgeConnectorConstraints > 1 ? "bevelled": "yes" );
869 }
870
871 if( m_EdgePlating )
872 KICAD_FORMAT::FormatBool( aFormatter, "edge_plating", true );
873
874 aFormatter->Print( ")" );
875}
876
877
878int BOARD_STACKUP::GetLayerDistance( PCB_LAYER_ID aFirstLayer, PCB_LAYER_ID aSecondLayer ) const
879{
880 wxASSERT( IsCopperLayer( aFirstLayer ) && IsCopperLayer( aSecondLayer ) );
881
882 if( aFirstLayer == aSecondLayer )
883 return 0;
884
885 // B_Cu is always the last copper layer but doesn't have the last numerical value
886 if( aSecondLayer != B_Cu && ( aSecondLayer < aFirstLayer || aFirstLayer == B_Cu ) )
887 std::swap( aFirstLayer, aSecondLayer );
888
889 int total = 0;
890 bool start = false;
891 bool half = false;
892
893 for( const BOARD_STACKUP_ITEM* item : m_list )
894 {
895 // Will be UNDEFINED_LAYER for dielectrics
896 const PCB_LAYER_ID layer = item->GetBrdLayerId();
897
898 if( layer != UNDEFINED_LAYER && !IsCopperLayer( layer ) )
899 continue; // Silk/mask layer
900
901 // Reached the start copper layer? Start counting the next dielectric after it
902 if( !start && ( layer != UNDEFINED_LAYER && layer == aFirstLayer ) )
903 {
904 start = true;
905
906 // Only count half of each internal copper layer
907 if( aFirstLayer != F_Cu && aFirstLayer != B_Cu )
908 half = true;
909 }
910 else if( !start )
911 continue;
912
913 // Reached the stop copper layer? we're done
914 if( start && ( layer != UNDEFINED_LAYER && layer == aSecondLayer ) )
915 {
916 // Only count half of each internal copper layer
917 if( aSecondLayer != F_Cu && aSecondLayer != B_Cu )
918 half = true;
919 }
920
921 for( int sublayer = 0; sublayer < item->GetSublayersCount(); sublayer++ )
922 {
923 const int subThickness = item->GetThickness( sublayer );
924 total += half ? ( subThickness / 2 ) : subThickness;
925 }
926
927 half = false;
928
929 if( layer != UNDEFINED_LAYER && layer == aSecondLayer )
930 break;
931 }
932
933 return total;
934}
935
936
937bool IsPrmSpecified( const wxString& aPrmValue )
938{
939 // return true if the param value is specified:
940
941 if( !aPrmValue.IsEmpty()
942 && ( aPrmValue.CmpNoCase( NotSpecifiedPrm() ) != 0 )
943 && aPrmValue != wxGetTranslation( NotSpecifiedPrm() ) )
944 return true;
945
946 return false;
947}
types::KiCadObjectType ToProtoEnum(KICAD_T aValue)
constexpr EDA_IU_SCALE pcbIUScale
Definition base_units.h:121
bool IsPrmSpecified(const wxString &aPrmValue)
@ BS_EDGE_CONNECTOR_NONE
BOARD_STACKUP_ITEM_TYPE
@ BS_ITEM_TYPE_UNDEFINED
@ BS_ITEM_TYPE_COPPER
@ BS_ITEM_TYPE_SILKSCREEN
@ BS_ITEM_TYPE_DIELECTRIC
@ BS_ITEM_TYPE_SOLDERPASTE
@ BS_ITEM_TYPE_SOLDERMASK
Container for design settings for a BOARD object.
const LSET & GetEnabledLayers() const
Return a bit-mask of all the layers that are enabled.
int GetBoardThickness() const
The full thickness of the board including copper and masks.
BOARD_STACKUP & GetStackupDescriptor()
Manage one layer needed to make a physical board.
void AddDielectricPrms(int aDielectricPrmsIdx)
Add (insert) a DIELECTRIC_PRMS item to m_DielectricPrmsList all values are set to default.
DIELECTRIC_MODEL GetDielectricModel(int aDielectricSubLayer=0) const
PCB_LAYER_ID m_LayerId
type name of layer (copper, silk screen, core, prepreg ...)
int GetSublayersCount() const
void SetDielectricLayerId(int aLayerId)
bool HasSpecFreqValue() const
double GetEpsilonR(int aDielectricSubLayer=0) const
wxString GetColor(int aDielectricSubLayer=0) const
bool HasEpsilonRValue() const
void SetThickness(int aThickness, int aDielectricSubLayer=0)
bool IsMaterialEditable() const
BOARD_STACKUP_ITEM_TYPE m_Type
void SetDielectricModel(DIELECTRIC_MODEL aModel, int aDielectricSubLayer=0)
bool HasMaterialValue(int aDielectricSubLayer=0) const
void SetThicknessLocked(bool aLocked, int aDielectricSubLayer=0)
wxString FormatDielectricLayerName() const
void SetSpecFreq(double aSpecFreq, int aDielectricSubLayer=0)
void SetMaterial(const wxString &aName, int aDielectricSubLayer=0)
BOARD_STACKUP_ITEM(BOARD_STACKUP_ITEM_TYPE aType)
bool HasLossTangentValue() const
bool IsThicknessEditable() const
void SetLossTangent(double aTg, int aDielectricSubLayer=0)
int GetThickness(int aDielectricSubLayer=0) const
void SetEnabled(bool aEnable)
std::vector< DIELECTRIC_PRMS > m_DielectricPrmsList
the "layer" id for dielectric layers, from 1 (top) to 31 (bottom) (only 31 dielectric layers for 32 c...
static int GetMaskDefaultThickness()
wxString GetMaterial(int aDielectricSubLayer=0) const
wxString m_TypeName
name of layer as shown in layer manager. Useful to create reports
void SetBrdLayerId(PCB_LAYER_ID aBrdLayerId)
void SetTypeName(const wxString &aName)
bool IsThicknessLocked(int aDielectricSubLayer=0) const
wxString FormatEpsilonR(int aDielectricSubLayer=0) const
int m_DielectricLayerId
the layer id (F.Cu to B.Cu, F.Silk, B.silk, F.Mask, B.Mask) and UNDEFINED_LAYER (-1) for dielectric l...
void SetColor(const wxString &aColorName, int aDielectricSubLayer=0)
void SetEpsilonR(double aEpsilon, int aDielectricSubLayer=0)
double GetSpecFreq(int aDielectricSubLayer=0) const
void RemoveDielectricPrms(int aDielectricPrmsIdx)
Remove a DIELECTRIC_PRMS item from m_DielectricPrmsList.
int GetDielectricLayerId() const
bool operator==(const BOARD_STACKUP_ITEM &aOther) const
bool IsColorEditable() const
wxString FormatLossTangent(int aDielectricSubLayer=0) const
double GetLossTangent(int aDielectricSubLayer=0) const
static int GetCopperDefaultThickness()
void RemoveAll()
Delete all items in list and clear the list.
const std::vector< BOARD_STACKUP_ITEM * > & GetList() const
static LSET StackupAllowedBrdLayers()
int GetCount() const
bool SynchronizeWithBoard(BOARD_DESIGN_SETTINGS *aSettings)
Synchronize the BOARD_STACKUP_ITEM* list with the board.
bool Deserialize(const google::protobuf::Any &aContainer) override
Deserializes the given protobuf message into this object.
bool operator==(const BOARD_STACKUP &aOther) const
int BuildBoardThicknessFromStackup() const
bool m_HasDielectricConstrains
True if some layers have impedance controlled tracks or have specific constrains for micro-wave appli...
void Add(BOARD_STACKUP_ITEM *aItem)
Add a new item in stackup layer.
void BuildDefaultStackupList(const BOARD_DESIGN_SETTINGS *aSettings, int aActiveCopperLayersCount=0)
Create a default stackup, according to the current BOARD_DESIGN_SETTINGS settings.
void Serialize(google::protobuf::Any &aContainer) const override
Serializes this object to the given Any message.
BOARD_STACKUP & operator=(const BOARD_STACKUP &aOther)
BOARD_STACKUP_ITEM * GetStackupLayer(int aIndex)
int GetLayerDistance(PCB_LAYER_ID aFirstLayer, PCB_LAYER_ID aSecondLayer) const
Calculate the distance (height) between the two given copper layers.
std::vector< BOARD_STACKUP_ITEM * > m_list
bool m_EdgePlating
True if the edge board is plated.
BS_EDGE_CONNECTOR_CONSTRAINTS m_EdgeConnectorConstraints
If the board has edge connector cards, some constrains can be specified in job file: BS_EDGE_CONNECTO...
void FormatBoardStackup(OUTPUTFORMATTER *aFormatter, const BOARD *aBoard) const
Write the stackup info on board file.
bool m_HasThicknessConstrains
True if some layers (copper and/or dielectric) have specific thickness.
wxString m_FinishType
The name of external copper finish.
Information pertinent to a Pcbnew printed circuit board.
Definition board.h:372
A helper class to manage a dielectric layer set of parameters.
double m_EpsilonR
true for dielectric layers with a fixed thickness (for impedance controlled purposes),...
int m_Thickness
type of material (for dielectric and solder mask)
wxString m_Material
bool operator==(const DIELECTRIC_PRMS &aOther) const
wxString m_Color
For dielectric the dielectric frequency correction model.
bool m_ThicknessLocked
the physical layer thickness in internal units
double m_LossTangent
For dielectric (and solder mask) the dielectric constant.
LSET is a set of PCB_LAYER_IDs.
Definition lset.h:37
LSEQ CuStack() const
Return a sequence of copper layers in starting from the front/top and extending to the back/bottom.
Definition lset.cpp:259
static wxString Name(PCB_LAYER_ID aLayerId)
Return the fixed name association with aLayerId.
Definition lset.cpp:184
An interface used to output 8 bit text in a convenient way.
Definition richio.h:291
std::string Quotew(const wxString &aWrapee) const
Definition richio.cpp:507
int PRINTF_FUNC_N Print(int nestLevel, const char *fmt,...)
Format and write text to the output stream.
Definition richio.cpp:422
#define _(s)
Some functions to handle hotkeys in KiCad.
bool IsCopperLayer(int aLayerId)
Test whether a layer is a copper layer.
Definition layer_ids.h:675
PCB_LAYER_ID
A quick note on layer IDs:
Definition layer_ids.h:56
@ F_Paste
Definition layer_ids.h:100
@ B_Mask
Definition layer_ids.h:94
@ B_Cu
Definition layer_ids.h:61
@ F_Mask
Definition layer_ids.h:93
@ B_Paste
Definition layer_ids.h:101
@ F_SilkS
Definition layer_ids.h:96
@ UNDEFINED_LAYER
Definition layer_ids.h:57
@ B_SilkS
Definition layer_ids.h:97
@ F_Cu
Definition layer_ids.h:60
KICOMMON_API std::string FormatInternalUnits(const EDA_IU_SCALE &aIuScale, int aValue, EDA_DATA_TYPE aDataType=EDA_DATA_TYPE::DISTANCE)
Converts aValue from internal units to a string appropriate for writing to file.
void FormatBool(OUTPUTFORMATTER *aOut, const wxString &aKey, bool aValue)
Writes a boolean to the formatter, in the style (aKey [yes|no])
#define _HKI(x)
Definition page_info.cpp:40
wxString NotSpecifiedPrm()
#define KEY_PREPREG
#define KEY_COPPER
#define KEY_CORE
#define DEFAULT_EPSILON_R_SILKSCREEN
#define DEFAULT_EPSILON_R_SOLDERMASK
std::string UIDouble2Str(double aValue)
Print a float number without using scientific notation and no trailing 0 We want to avoid scientific ...
std::string FormatDouble2Str(double aValue)
Print a float number without using scientific notation and no trailing 0 This function is intended in...
DIELECTRIC_MODEL
Frequency-domain model used for the substrate dielectric properties.