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 (C) 2009-2021 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 along
18 * with this program. If not, see <http://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
30#include <richio.h>
31#include <google/protobuf/any.pb.h>
32#include <api/board/board.pb.h>
33#include <api/api_enums.h>
34
35
37{
38 if( m_Material != aOther.m_Material ) return false;
39 if( m_Thickness != aOther.m_Thickness ) return false;
40 if( m_ThicknessLocked != aOther.m_ThicknessLocked ) return false;
41 if( m_EpsilonR != aOther.m_EpsilonR ) return false;
42 if( m_LossTangent != aOther.m_LossTangent ) return false;
43 if( m_Color != aOther.m_Color ) return false;
44
45 return true;
46}
47
48
50{
51 DIELECTRIC_PRMS item_prms;
52 m_DielectricPrmsList.emplace_back( item_prms );
54 m_Type = aType;
56 SetEnabled( true );
57
58 // Initialize parameters to a usual value for allowed types:
59 switch( m_Type )
60 {
64 break;
65
67 m_TypeName = KEY_CORE; // or prepreg
69 SetMaterial( wxT( "FR4" ) ); // or other dielectric name
70 SetLossTangent( 0.02 ); // for FR4
71 SetEpsilonR( 4.5 ); // for FR4
72 break;
73
75 m_TypeName = wxT( "solderpaste" );
76 break;
77
79 m_TypeName = wxT( "soldermask" );
81 SetMaterial( NotSpecifiedPrm() ); // or other solder mask material name
84 break;
85
87 m_TypeName = wxT( "silkscreen" );
89 SetMaterial( NotSpecifiedPrm() ); // or other silkscreen material name
91 break;
92
94 break;
95 }
96}
97
98
100{
101 m_LayerId = aOther.m_LayerId;
103 m_Type = aOther.m_Type;
104 m_enabled = aOther.m_enabled;
106 m_TypeName = aOther.m_TypeName;
107 m_LayerName = aOther.m_LayerName;
108}
109
110
112{
113 if( m_Type != aOther.m_Type ) return false;
114 if( m_LayerName != aOther.m_LayerName ) return false;
115 if( m_TypeName != aOther.m_TypeName ) return false;
116 if( m_LayerId != aOther.m_LayerId ) return false;
117 if( m_DielectricLayerId != aOther.m_DielectricLayerId ) return false;
118 if( m_enabled != aOther.m_enabled ) return false;
119
120 if( !std::equal( std::begin( m_DielectricPrmsList ), std::end( m_DielectricPrmsList ),
121 std::begin( aOther.m_DielectricPrmsList ),
122 []( const DIELECTRIC_PRMS& aA, const DIELECTRIC_PRMS& aB )
123 {
124 return aA == aB;
125 } ) )
126 {
127 return false;
128 }
129
130 return true;
131}
132
133
134void BOARD_STACKUP_ITEM::AddDielectricPrms( int aDielectricPrmsIdx )
135{
136 // add a DIELECTRIC_PRMS item to m_DielectricPrmsList
137 DIELECTRIC_PRMS new_prms;
138
139 m_DielectricPrmsList.emplace( m_DielectricPrmsList.begin() + aDielectricPrmsIdx, new_prms );
140}
141
142
143void BOARD_STACKUP_ITEM::RemoveDielectricPrms( int aDielectricPrmsIdx )
144{
145 // Remove a DIELECTRIC_PRMS item from m_DielectricPrmsList if possible
146
147 if( GetSublayersCount() < 2
148 || aDielectricPrmsIdx < 0
149 || aDielectricPrmsIdx >= GetSublayersCount() )
150 {
151 return;
152 }
153
154 m_DielectricPrmsList.erase( m_DielectricPrmsList.begin() + aDielectricPrmsIdx );
155}
156
157
158
160{
161 // A reasonable thickness for copper layers:
162 return pcbIUScale.mmToIU( 0.035 );
163}
164
165
167{
168 // A reasonable thickness for solder mask:
169 return pcbIUScale.mmToIU( 0.01 );
170}
171
172
173// Getters:
174wxString BOARD_STACKUP_ITEM::GetColor( int aDielectricSubLayer ) const
175{
176 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
177
178 return m_DielectricPrmsList[aDielectricSubLayer].m_Color;
179}
180
181int BOARD_STACKUP_ITEM::GetThickness( int aDielectricSubLayer ) const
182{
183 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
184
185 return m_DielectricPrmsList[aDielectricSubLayer].m_Thickness;
186}
187
188
189double BOARD_STACKUP_ITEM::GetLossTangent( int aDielectricSubLayer ) const
190{
191 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
192
193 return m_DielectricPrmsList[aDielectricSubLayer].m_LossTangent;
194}
195
196
197double BOARD_STACKUP_ITEM::GetEpsilonR( int aDielectricSubLayer ) const
198{
199 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
200
201 return m_DielectricPrmsList[aDielectricSubLayer].m_EpsilonR;
202}
203
204
205bool BOARD_STACKUP_ITEM::IsThicknessLocked( int aDielectricSubLayer ) const
206{
207 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
208
209 return m_DielectricPrmsList[aDielectricSubLayer].m_ThicknessLocked;
210}
211
212
213wxString BOARD_STACKUP_ITEM::GetMaterial( int aDielectricSubLayer ) const
214{
215 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
216
217 return m_DielectricPrmsList[aDielectricSubLayer].m_Material;
218}
219
220
221// Setters:
222void BOARD_STACKUP_ITEM::SetColor( const wxString& aColorName , int aDielectricSubLayer )
223{
224 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
225
226 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
227 m_DielectricPrmsList[aDielectricSubLayer].m_Color = aColorName;
228}
229
230
231void BOARD_STACKUP_ITEM::SetThickness( int aThickness, int aDielectricSubLayer )
232{
233 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
234
235 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
236 m_DielectricPrmsList[aDielectricSubLayer].m_Thickness = aThickness;
237}
238
239
240void BOARD_STACKUP_ITEM::SetLossTangent( double aTg, int aDielectricSubLayer )
241{
242 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
243
244 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
245 m_DielectricPrmsList[aDielectricSubLayer].m_LossTangent = aTg;
246}
247
248
249void BOARD_STACKUP_ITEM::SetEpsilonR( double aEpsilon, int aDielectricSubLayer )
250{
251 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
252
253 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
254 m_DielectricPrmsList[aDielectricSubLayer].m_EpsilonR = aEpsilon;
255}
256
257
258void BOARD_STACKUP_ITEM::SetThicknessLocked( bool aLocked, int aDielectricSubLayer )
259{
260 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
261
262 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
263 m_DielectricPrmsList[aDielectricSubLayer].m_ThicknessLocked = aLocked;
264}
265
266
267void BOARD_STACKUP_ITEM::SetMaterial( const wxString& aName, int aDielectricSubLayer )
268{
269 wxASSERT( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() );
270
271 if( aDielectricSubLayer >= 0 && aDielectricSubLayer < GetSublayersCount() )
272 m_DielectricPrmsList[aDielectricSubLayer].m_Material = aName;
273}
274
275
277{
280};
281
282
284{
287};
288
289
290bool BOARD_STACKUP_ITEM::HasMaterialValue( int aDielectricSubLayer ) const
291{
292 // return true if the material is specified
293 return IsMaterialEditable() && IsPrmSpecified( GetMaterial( aDielectricSubLayer ) );
294}
295
296
298{
302}
303
304
306{
310}
311
312
314{
318}
319
320
321wxString BOARD_STACKUP_ITEM::FormatEpsilonR( int aDielectricSubLayer ) const
322{
323 // return a wxString to print/display Epsilon R
324 // note: we do not want scientific notation
325 wxString txt = UIDouble2Str( GetEpsilonR( aDielectricSubLayer ) );
326 return txt;
327}
328
329
330wxString BOARD_STACKUP_ITEM::FormatLossTangent( int aDielectricSubLayer ) const
331{
332 // return a wxString to print/display Loss Tangent
333 // note: we do not want scientific notation
334 wxString txt = UIDouble2Str( GetLossTangent( aDielectricSubLayer ) );
335 return txt;
336}
337
338
340{
341 // return a wxString to print/display a dielectric name
342 wxString lname;
343 lname.Printf( _( "Dielectric %d" ), GetDielectricLayerId() );
344
345 return lname;
346}
347
348
350{
351 m_HasDielectricConstrains = false; // True if some dielectric layers have constrains
352 // (Loss tg and Epison R)
353 m_HasThicknessConstrains = false; // True if some dielectric or copper layers have constrains
355 m_CastellatedPads = false; // True if some castellated pads exist
356 m_EdgePlating = false; // True if edge board is plated
357 m_FinishType = wxT( "None" ); // undefined finish type
358}
359
360
362{
368 m_FinishType = aOther.m_FinishType;
369
370 // All items in aOther.m_list have to be duplicated, because aOther.m_list
371 // manage pointers to these items
372 for( BOARD_STACKUP_ITEM* item : aOther.m_list )
373 {
374 BOARD_STACKUP_ITEM* dup_item = new BOARD_STACKUP_ITEM( *item );
375 Add( dup_item );
376 }
377}
378
379
381{
387 m_FinishType = aOther.m_FinishType;
388
389 RemoveAll();
390
391 // All items in aOther.m_list have to be duplicated, because aOther.m_list
392 // manage pointers to these items
393 for( BOARD_STACKUP_ITEM* item : aOther.m_list )
394 {
395 BOARD_STACKUP_ITEM* dup_item = new BOARD_STACKUP_ITEM( *item );
396 Add( dup_item );
397 }
398
399 return *this;
400}
401
402
403bool BOARD_STACKUP::operator==( const BOARD_STACKUP& aOther ) const
404{
405 if( m_HasDielectricConstrains != aOther.m_HasDielectricConstrains ) return false;
406 if( m_HasThicknessConstrains != aOther.m_HasThicknessConstrains ) return false;
407 if( m_EdgeConnectorConstraints != aOther.m_EdgeConnectorConstraints ) return false;
408 if( m_CastellatedPads != aOther.m_CastellatedPads ) return false;
409 if( m_EdgePlating != aOther.m_EdgePlating ) return false;
410 if( m_FinishType != aOther.m_FinishType ) return false;
411
412 if( !std::equal( std::begin( m_list ), std::end( m_list ), std::begin( aOther.m_list ),
413 []( const BOARD_STACKUP_ITEM* aA, const BOARD_STACKUP_ITEM* aB )
414 {
415 return *aA == *aB;
416 } ) )
417 {
418 return false;
419 }
420
421 return true;
422}
423
424
425void BOARD_STACKUP::Serialize( google::protobuf::Any& aContainer ) const
426{
427 kiapi::board::BoardStackup stackup;
428
429 for( const BOARD_STACKUP_ITEM* item : m_list )
430 {
431 kiapi::board::BoardStackupLayer* layer = stackup.mutable_layers()->Add();
432
433 // TODO dielectric sub-layers
434 layer->mutable_thickness()->set_value_nm( item->GetThickness() );
435 layer->set_layer( ToProtoEnum<PCB_LAYER_ID, kiapi::board::types::BoardLayer>(
436 item->GetBrdLayerId() ) );
437
438 switch( item->GetType() )
439 {
440 case BOARD_STACKUP_ITEM_TYPE::BS_ITEM_TYPE_COPPER:
441 {
442 layer->mutable_copper()->New();
443 // (no copper params yet...)
444 break;
445 }
446
447 default:
448 // TODO
449 break;
450 }
451 }
452
453 aContainer.PackFrom( stackup );
454}
455
456
457bool BOARD_STACKUP::Deserialize( const google::protobuf::Any& aContainer )
458{
459 return true;
460}
461
462
464{
465 for( BOARD_STACKUP_ITEM* item : m_list )
466 delete item;
467
468 m_list.clear();
469}
470
471
473{
474 if( aIndex < 0 || aIndex >= GetCount() )
475 return nullptr;
476
477 return GetList()[aIndex];
478}
479
480
482{
483 // return the board thickness from the thickness of BOARD_STACKUP_ITEM list
484 int thickness = 0;
485
486 for( BOARD_STACKUP_ITEM* item : m_list )
487 {
488 if( item->IsThicknessEditable() && item->IsEnabled() )
489 {
490 thickness += item->GetThickness();
491
492 // dielectric layers can have more than one main layer
493 // add thickness of all sublayers
494 for( int idx = 1; idx < item->GetSublayersCount(); idx++ )
495 {
496 thickness += item->GetThickness( idx );
497 }
498 }
499 }
500
501 return thickness;
502}
503
504
506{
507 bool change = false;
508 // Build the suitable stackup:
509 BOARD_STACKUP stackup;
510 stackup.BuildDefaultStackupList( aSettings );
511
512 // First, find removed layers:
513 for( BOARD_STACKUP_ITEM* curr_item: m_list )
514 {
515 bool found = false;
516
517 for( BOARD_STACKUP_ITEM* item: stackup.GetList() )
518 {
519 if( curr_item->GetBrdLayerId() != UNDEFINED_LAYER )
520 {
521 if( item->GetBrdLayerId() == curr_item->GetBrdLayerId() )
522 {
523 found = true;
524 break;
525 }
526 }
527 else // curr_item = dielectric layer
528 {
529 if( item->GetBrdLayerId() != UNDEFINED_LAYER )
530 continue;
531
532 if( item->GetDielectricLayerId() == curr_item->GetDielectricLayerId() )
533 {
534 found = true;
535 break;
536 }
537 }
538 }
539
540 if( !found ) // a layer was removed: a change is found
541 {
542 change = true;
543 break;
544 }
545 }
546
547 // Now initialize all stackup items to the initial values, when exist
548 for( BOARD_STACKUP_ITEM* item : stackup.GetList() )
549 {
550 bool found = false;
551 // Search for initial settings:
552 for( const BOARD_STACKUP_ITEM* initial_item : m_list )
553 {
554 if( item->GetBrdLayerId() != UNDEFINED_LAYER )
555 {
556 if( item->GetBrdLayerId() == initial_item->GetBrdLayerId() )
557 {
558 *item = *initial_item;
559 found = true;
560 break;
561 }
562 }
563 else // dielectric layer: see m_DielectricLayerId for identification
564 {
565 // Compare dielectric layer with dielectric layer
566 if( initial_item->GetBrdLayerId() != UNDEFINED_LAYER )
567 continue;
568
569 if( item->GetDielectricLayerId() == initial_item->GetDielectricLayerId() )
570 {
571 *item = *initial_item;
572 found = true;
573 break;
574 }
575 }
576 }
577
578 if( !found )
579 {
580 change = true;
581 }
582 }
583
584 // Transfer layer settings:
585 *this = stackup;
586
587 // Transfer other stackup settings from aSettings
588 const BOARD_STACKUP& source_stackup = aSettings->GetStackupDescriptor();
591 m_CastellatedPads = source_stackup.m_CastellatedPads;
592 m_EdgePlating = source_stackup.m_EdgePlating;
593 m_FinishType = source_stackup.m_FinishType;
594
595 return change;
596}
597
598
600 int aActiveCopperLayersCount )
601{
602 // Creates a default stackup, according to the current BOARD_DESIGN_SETTINGS settings.
603 // Note: the m_TypeName string is made translatable using _HKI marker, but is not
604 // translated when building the stackup.
605 // It will be used as this in files, and can be translated only in dialog
606 // if aSettings == NULL, build a full stackup (with 32 copper layers)
607 LSET enabledLayer = aSettings ? aSettings->GetEnabledLayers() : StackupAllowedBrdLayers();
608 int copperLayerCount = aSettings ? aSettings->GetCopperLayerCount() : 32;
609
610 // We need to calculate a suitable dielectric layer thickness.
611 // If no settings, and if aActiveCopperLayersCount is given, use it
612 // (If no settings, and no aActiveCopperLayersCount, the full 32 layers are used)
613 int activeCuLayerCount = copperLayerCount;
614
615 if( aSettings == nullptr && aActiveCopperLayersCount > 0 )
616 activeCuLayerCount = aActiveCopperLayersCount;
617
618 int brd__thickness = aSettings ? aSettings->GetBoardThickness() : pcbIUScale.mmToIU( 1.6 );
619 int diel_thickness = brd__thickness -
620 ( BOARD_STACKUP_ITEM::GetCopperDefaultThickness() * activeCuLayerCount );
621
622 // Take in account the solder mask thickness:
623 int sm_count = ( enabledLayer & LSET( { F_Mask, B_Mask } ) ).count();
624 diel_thickness -= BOARD_STACKUP_ITEM::GetMaskDefaultThickness() * sm_count;
625 diel_thickness /= std::max( 1, activeCuLayerCount - 1 );
626
627 int dielectric_idx = 0;
628
629 // Add silk screen, solder mask and solder paste layers on top
630 if( enabledLayer[F_SilkS] )
631 {
633 item->SetBrdLayerId( F_SilkS );
634 item->SetTypeName( _HKI( "Top Silk Screen" ) );
635 Add( item );
636 }
637
638 if( enabledLayer[F_Paste] )
639 {
641 item->SetBrdLayerId( F_Paste );
642 item->SetTypeName( _HKI( "Top Solder Paste" ) );
643 Add( item );
644 }
645
646 if( enabledLayer[F_Mask] )
647 {
649 item->SetBrdLayerId( F_Mask );
650 item->SetTypeName( _HKI( "Top Solder Mask" ) );
651 Add( item );
652 }
653
654 // Add copper and dielectric layers
655 for( PCB_LAYER_ID layer : enabledLayer.CuStack() )
656 {
658 item->SetBrdLayerId( layer );
659 item->SetTypeName( KEY_COPPER );
660 Add( item );
661
662 if( layer == B_Cu )
663 break;
664
665 // Add the dielectric layer:
667 item->SetThickness( diel_thickness );
668 item->SetDielectricLayerId( dielectric_idx + 1 );
669
670 // Display a dielectric default layer name:
671 if( (dielectric_idx & 1) == 0 )
672 {
673 item->SetTypeName( KEY_CORE );
674 item->SetMaterial( wxT( "FR4" ) );
675 }
676 else
677 {
678 item->SetTypeName( KEY_PREPREG );
679 item->SetMaterial( wxT( "FR4" ) );
680 }
681
682 Add( item );
683 dielectric_idx++;
684 }
685
686 // Add silk screen, solder mask and solder paste layers on bottom
687 if( enabledLayer[B_Mask] )
688 {
690 item->SetBrdLayerId( B_Mask );
691 item->SetTypeName( _HKI( "Bottom Solder Mask" ) );
692 Add( item );
693 }
694
695 if( enabledLayer[B_Paste] )
696 {
698 item->SetBrdLayerId( B_Paste );
699 item->SetTypeName( _HKI( "Bottom Solder Paste" ) );
700 Add( item );
701 }
702
703 if( enabledLayer[B_SilkS] )
704 {
706 item->SetBrdLayerId( B_SilkS );
707 item->SetTypeName( _HKI( "Bottom Silk Screen" ) );
708 Add( item );
709 }
710
711 // Transfer other stackup settings from aSettings
712 if( aSettings )
713 {
714 const BOARD_STACKUP& source_stackup = aSettings->GetStackupDescriptor();
717 m_CastellatedPads = source_stackup.m_CastellatedPads;
718 m_EdgePlating = source_stackup.m_EdgePlating;
719 m_FinishType = source_stackup.m_FinishType;
720 }
721}
722
723
725 const BOARD* aBoard, int aNestLevel ) const
726{
727 // Board stackup is the ordered list from top to bottom of
728 // physical layers and substrate used to build the board.
729 if( m_list.empty() )
730 return;
731
732 aFormatter->Print( aNestLevel, "(stackup\n" );
733 int nest_level = aNestLevel+1;
734
735 // Note:
736 // Unspecified parameters are not stored in file.
737 for( BOARD_STACKUP_ITEM* item: m_list )
738 {
739 wxString layer_name;
740
741 if( item->GetBrdLayerId() == UNDEFINED_LAYER )
742 layer_name.Printf( wxT( "dielectric %d" ), item->GetDielectricLayerId() );
743 else
744 layer_name = LSET::Name( item->GetBrdLayerId() );
745
746 aFormatter->Print( nest_level, "(layer %s (type %s)",
747 aFormatter->Quotew( layer_name ).c_str(),
748 aFormatter->Quotew( item->GetTypeName() ).c_str() );
749
750 // Output other parameters ( in sub layer list there is at least one item)
751 for( int idx = 0; idx < item->GetSublayersCount(); idx++ )
752 {
753 if( idx ) // not for the main (first) layer.
754 {
755 aFormatter->Print( 0, "\n" );
756 aFormatter->Print( nest_level+1, "addsublayer" );
757 }
758
759 if( item->IsColorEditable() && IsPrmSpecified( item->GetColor( idx ) ) )
760 {
761 aFormatter->Print( 0, " (color %s)",
762 aFormatter->Quotew( item->GetColor( idx ) ).c_str() );
763 }
764
765 if( item->IsThicknessEditable() )
766 {
767 if( item->GetType() == BS_ITEM_TYPE_DIELECTRIC && item->IsThicknessLocked( idx ) )
768 aFormatter->Print( 0, " (thickness %s locked)",
769 EDA_UNIT_UTILS::FormatInternalUnits( pcbIUScale, item->GetThickness( idx ) ).c_str() );
770 else
771 aFormatter->Print( 0, " (thickness %s)",
772 EDA_UNIT_UTILS::FormatInternalUnits( pcbIUScale, item->GetThickness( idx ) ).c_str() );
773 }
774
775 if( item->HasMaterialValue( idx ) )
776 aFormatter->Print( 0, " (material %s)",
777 aFormatter->Quotew( item->GetMaterial( idx ) ).c_str() );
778
779 if( item->HasEpsilonRValue() && item->HasMaterialValue( idx ) )
780 aFormatter->Print( 0, " (epsilon_r %g)", item->GetEpsilonR( idx ) );
781
782 if( item->HasLossTangentValue() && item->HasMaterialValue( idx ) )
783 aFormatter->Print( 0, " (loss_tangent %s)",
784 FormatDouble2Str( item->GetLossTangent( idx ) ).c_str() );
785 }
786
787 aFormatter->Print( 0, ")\n" );
788 }
789
790 // Other infos about board, related to layers and other fabrication specifications
792 {
793 aFormatter->Print( nest_level, "(copper_finish %s)\n",
794 aFormatter->Quotew( m_FinishType ).c_str() );
795 }
796
797 aFormatter->Print( nest_level, "(dielectric_constraints %s)\n",
798 m_HasDielectricConstrains ? "yes" : "no" );
799
801 {
802 aFormatter->Print( nest_level, "(edge_connector %s)\n",
803 m_EdgeConnectorConstraints > 1 ? "bevelled": "yes" );
804 }
805
807 aFormatter->Print( nest_level, "(castellated_pads yes)\n" );
808
809 if( m_EdgePlating )
810 aFormatter->Print( nest_level, "(edge_plating yes)\n" );
811
812 aFormatter->Print( aNestLevel, ")\n" );
813}
814
815
816int BOARD_STACKUP::GetLayerDistance( PCB_LAYER_ID aFirstLayer, PCB_LAYER_ID aSecondLayer ) const
817{
818 wxASSERT( IsCopperLayer( aFirstLayer ) && IsCopperLayer( aSecondLayer ) );
819
820 if( aFirstLayer == aSecondLayer )
821 return 0;
822
823 // B_Cu is always the last copper layer but doesn't have the last numerical value
824 if( aSecondLayer != B_Cu && ( aSecondLayer < aFirstLayer || aFirstLayer == B_Cu ) )
825 std::swap( aFirstLayer, aSecondLayer );
826
827 int total = 0;
828 bool start = false;
829 bool half = false;
830
831 for( BOARD_STACKUP_ITEM* item : m_list )
832 {
833 // Will be UNDEFINED_LAYER for dielectrics
834 PCB_LAYER_ID layer = item->GetBrdLayerId();
835
836 if( layer != UNDEFINED_LAYER && !IsCopperLayer( layer ) )
837 continue; // Silk/mask layer
838
839 // Reached the start copper layer? Start counting the next dielectric after it
840 if( !start && ( layer != UNDEFINED_LAYER && layer == aFirstLayer ) )
841 {
842 start = true;
843 half = true;
844 }
845 else if( !start )
846 continue;
847
848 // Reached the stop copper layer? we're done
849 if( start && ( layer != UNDEFINED_LAYER && layer == aSecondLayer ) )
850 half = true;
851
852 for( int sublayer = 0; sublayer < item->GetSublayersCount(); sublayer++ )
853 {
854 int subThickness = item->GetThickness( sublayer );
855 total += half ? ( subThickness / 2 ) : subThickness;
856 }
857
858 half = false;
859
860 if( layer != UNDEFINED_LAYER && layer == aSecondLayer )
861 break;
862 }
863
864 return total;
865}
866
867
868bool IsPrmSpecified( const wxString& aPrmValue )
869{
870 // return true if the param value is specified:
871
872 if( !aPrmValue.IsEmpty()
873 && ( aPrmValue.CmpNoCase( NotSpecifiedPrm() ) != 0 )
874 && aPrmValue != wxGetTranslation( NotSpecifiedPrm() ) )
875 return true;
876
877 return false;
878}
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
bool IsPrmSpecified(const wxString &aPrmValue)
@ BS_EDGE_CONNECTOR_NONE
Definition: board_stackup.h:57
BOARD_STACKUP_ITEM_TYPE
Definition: board_stackup.h:43
@ BS_ITEM_TYPE_UNDEFINED
Definition: board_stackup.h:44
@ BS_ITEM_TYPE_COPPER
Definition: board_stackup.h:45
@ BS_ITEM_TYPE_SILKSCREEN
Definition: board_stackup.h:51
@ BS_ITEM_TYPE_DIELECTRIC
Definition: board_stackup.h:46
@ BS_ITEM_TYPE_SOLDERPASTE
Definition: board_stackup.h:48
@ BS_ITEM_TYPE_SOLDERMASK
Definition: board_stackup.h:49
Container for design settings for a BOARD object.
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.
Definition: board_stackup.h:96
void AddDielectricPrms(int aDielectricPrmsIdx)
Add (insert) a DIELECTRIC_PRMS item to m_DielectricPrmsList all values are set to default.
PCB_LAYER_ID m_LayerId
type name of layer (copper, silk screen, core, prepreg ...)
int GetSublayersCount() const
void SetDielectricLayerId(int aLayerId)
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
bool HasMaterialValue(int aDielectricSubLayer=0) const
void SetThicknessLocked(bool aLocked, int aDielectricSubLayer=0)
wxString FormatDielectricLayerName() const
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)
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()
Manage layers needed to make a physical board.
void RemoveAll()
Delete all items in list and clear the list.
void FormatBoardStackup(OUTPUTFORMATTER *aFormatter, const BOARD *aBoard, int aNestLevel) const
Write the stackup info on board file.
bool m_CastellatedPads
True if castellated pads exist.
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...
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:290
A helper class to manage a dielectric layer set of parameters.
Definition: board_stackup.h:67
double m_EpsilonR
true for dielectric layers with a fixed thickness (for impedance controlled purposes),...
Definition: board_stackup.h:84
int m_Thickness
type of material (for dielectric and solder mask)
Definition: board_stackup.h:81
wxString m_Material
Definition: board_stackup.h:80
bool operator==(const DIELECTRIC_PRMS &aOther) const
wxString m_Color
For dielectric (and solder mask) the dielectric loss.
Definition: board_stackup.h:86
bool m_ThicknessLocked
the physical layer thickness in internal units
Definition: board_stackup.h:82
double m_LossTangent
For dielectric (and solder mask) the dielectric constant.
Definition: board_stackup.h:85
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:36
LSEQ CuStack() const
Return a sequence of copper layers in starting from the front/top and extending to the back/bottom.
Definition: lset.cpp:240
static wxString Name(PCB_LAYER_ID aLayerId)
Return the fixed name association with aLayerId.
Definition: lset.cpp:183
An interface used to output 8 bit text in a convenient way.
Definition: richio.h:322
std::string Quotew(const wxString &aWrapee) const
Definition: richio.cpp:526
int PRINTF_FUNC Print(int nestLevel, const char *fmt,...)
Format and write text to the output stream.
Definition: richio.cpp:458
#define _HKI(x)
#define _(s)
Some functions to handle hotkeys in KiCad.
bool IsCopperLayer(int aLayerId)
Tests whether a layer is a copper layer.
Definition: layer_ids.h:531
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ F_Paste
Definition: layer_ids.h:104
@ B_Mask
Definition: layer_ids.h:98
@ B_Cu
Definition: layer_ids.h:65
@ F_Mask
Definition: layer_ids.h:97
@ B_Paste
Definition: layer_ids.h:105
@ F_SilkS
Definition: layer_ids.h:100
@ UNDEFINED_LAYER
Definition: layer_ids.h:61
@ B_SilkS
Definition: layer_ids.h:101
KICOMMON_API std::string FormatInternalUnits(const EDA_IU_SCALE &aIuScale, int aValue)
Converts aValue from internal units to a string appropriate for writing to file.
Definition: eda_units.cpp:169
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...
constexpr int mmToIU(double mm) const
Definition: base_units.h:88