KiCad PCB EDA Suite
Loading...
Searching...
No Matches
tuning_profiles.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) 2020 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 * @author Jon Evans <[email protected]>
7 *
8 * This program is free software: you can redistribute it and/or modify it
9 * under the terms of the GNU General Public License as published by the
10 * Free Software Foundation, either version 3 of the License, or (at your
11 * option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful, but
14 * WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License along
19 * with this program. If not, see <http://www.gnu.org/licenses/>.
20 */
21
22#include <lset.h>
23#include <netclass.h>
25
27
28#include <nlohmann/json.hpp>
29#include <settings/parameters.h>
30
31
33
34TUNING_PROFILES::TUNING_PROFILES( JSON_SETTINGS* aParent, const std::string& aPath ) :
35 NESTED_SETTINGS( "tuning_profiles", tuningParametersSchemaVersion, aParent, aPath, false )
36{
37 auto saveViaOverrideConfigurationLine =
38 []( nlohmann::json& json_array, const DELAY_PROFILE_VIA_OVERRIDE_ENTRY& item )
39 {
40 const nlohmann::json item_json = { { "signal_layer_from", LSET::Name( item.m_SignalLayerFrom ) },
41 { "signal_layer_to", LSET::Name( item.m_SignalLayerTo ) },
42 { "via_layer_from", LSET::Name( item.m_ViaLayerFrom ) },
43 { "via_layer_to", LSET::Name( item.m_ViaLayerTo ) },
44 { "delay", item.m_Delay } };
45
46 json_array.push_back( item_json );
47 };
48
49 auto readViaOverrideConfigurationLine = []( const nlohmann::json& entry )
50 {
51 wxString signalLayerFromName = entry["signal_layer_from"];
52 int signalLayerFromId = LSET::NameToLayer( signalLayerFromName );
53
54 wxString signalLayerToName = entry["signal_layer_to"];
55 int signalLayerToId = LSET::NameToLayer( signalLayerToName );
56
57 wxString viaLayerFromName = entry["via_layer_from"];
58 int viaLayerFromId = LSET::NameToLayer( viaLayerFromName );
59
60 wxString viaLayerToName = entry["via_layer_to"];
61 int viaLayerToId = LSET::NameToLayer( viaLayerToName );
62
63 int delay = entry["delay"];
64
65 DELAY_PROFILE_VIA_OVERRIDE_ENTRY item{ static_cast<PCB_LAYER_ID>( signalLayerFromId ),
66 static_cast<PCB_LAYER_ID>( signalLayerToId ),
67 static_cast<PCB_LAYER_ID>( viaLayerFromId ),
68 static_cast<PCB_LAYER_ID>( viaLayerToId ), delay };
69
70 return item;
71 };
72
73 auto saveUserDefinedProfileConfigurationLine =
74 [saveViaOverrideConfigurationLine]( nlohmann::json& json_array, const TUNING_PROFILE& item )
75 {
76 nlohmann::json layer_entries = nlohmann::json::array();
77
78 for( const DELAY_PROFILE_TRACK_PROPAGATION_ENTRY& trackEntry : item.m_TrackPropagationEntries )
79 {
80 nlohmann::json layer_json;
81
82 layer_json["signal_layer"] = LSET::Name( trackEntry.m_signalLayer );
83 layer_json["top_reference_layer"] = LSET::Name( trackEntry.m_topReferenceLayer );
84 layer_json["bottom_reference_layer"] = LSET::Name( trackEntry.m_bottomReferenceLayer );
85 layer_json["width"] = trackEntry.m_width;
86 layer_json["diff_pair_gap"] = trackEntry.m_diffPairGap;
87 layer_json["delay"] = trackEntry.m_delay;
88
89 layer_entries.push_back( layer_json );
90 }
91
92 nlohmann::json via_overrides = nlohmann::json::array();
93
94 for( const DELAY_PROFILE_VIA_OVERRIDE_ENTRY& viaOverride : item.m_ViaOverrides )
95 {
96 saveViaOverrideConfigurationLine( via_overrides, viaOverride );
97 }
98
99 const nlohmann::json item_json = { { "profile_name", item.m_ProfileName.ToUTF8() },
100 { "type", static_cast<int>( item.m_Type ) },
101 { "target_impedance", item.m_TargetImpedance },
102 { "frequency", item.m_Frequency },
103 { "model_solder_mask", item.m_ModelSolderMask },
104 { "enable_time_domain_tuning", item.m_EnableTimeDomainTuning },
105 { "layer_entries", layer_entries },
106 { "via_prop_delay", item.m_ViaPropagationDelay },
107 { "via_overrides", via_overrides } };
108
109 json_array.push_back( item_json );
110 };
111
112 auto readUserDefinedProfileConfigurationLine = [readViaOverrideConfigurationLine]( const nlohmann::json& entry )
113 {
114 const wxString profileName = entry["profile_name"];
115 const TUNING_PROFILE::PROFILE_TYPE profileType = static_cast<TUNING_PROFILE::PROFILE_TYPE>( entry["type"] );
116 const double targetImpedance = entry["target_impedance"];
117 const double frequency = entry["frequency"];
118 const bool modelSolderMask = entry["model_solder_mask"];
119 const bool enableTimeDomainTuning = entry["enable_time_domain_tuning"];
120 const int viaPropDelay = entry["via_prop_delay"];
121 std::vector<DELAY_PROFILE_TRACK_PROPAGATION_ENTRY> trackEntries;
122 std::map<PCB_LAYER_ID, DELAY_PROFILE_TRACK_PROPAGATION_ENTRY> trackEntriesMap;
123
124 for( const nlohmann::json& layerEntry : entry["layer_entries"] )
125 {
126 if( !layerEntry.is_object() )
127 continue;
128
129 wxString signalLayer = layerEntry["signal_layer"];
130 wxString topRefLayer = layerEntry["top_reference_layer"];
131 wxString bottomRefLayer = layerEntry["bottom_reference_layer"];
132 int signalLayerId = LSET::NameToLayer( signalLayer );
133 int topRefLayerId = LSET::NameToLayer( topRefLayer );
134 int bottomRefLayerId = LSET::NameToLayer( bottomRefLayer );
135
137 trackEntry.m_signalLayer = static_cast<PCB_LAYER_ID>( signalLayerId );
138 trackEntry.m_topReferenceLayer = static_cast<PCB_LAYER_ID>( topRefLayerId );
139 trackEntry.m_bottomReferenceLayer = static_cast<PCB_LAYER_ID>( bottomRefLayerId );
140 trackEntry.m_width = layerEntry["width"];
141 trackEntry.m_diffPairGap = layerEntry["diff_pair_gap"];
142 trackEntry.m_delay = layerEntry["delay"];
143 trackEntry.m_enableTimeDomainTuning = enableTimeDomainTuning;
144
145 trackEntries.push_back( trackEntry );
146 trackEntriesMap[static_cast<PCB_LAYER_ID>( signalLayerId )] = trackEntry;
147 }
148
149 std::vector<DELAY_PROFILE_VIA_OVERRIDE_ENTRY> viaOverrides;
150
151 for( const nlohmann::json& viaEntry : entry["via_overrides"] )
152 {
153 if( !viaEntry.is_object() || !viaEntry.contains( "signal_layer_from" ) )
154 continue;
155
156 viaOverrides.push_back( readViaOverrideConfigurationLine( viaEntry ) );
157 }
158
159 TUNING_PROFILE item{ profileName,
160 profileType,
161 targetImpedance,
162 frequency,
163 enableTimeDomainTuning,
164 modelSolderMask,
165 std::move( trackEntries ),
166 viaPropDelay,
167 std::move( viaOverrides ),
168 std::move( trackEntriesMap ) };
169
170 return item;
171 };
172
173 m_params.emplace_back( new PARAM_LAMBDA<nlohmann::json>(
174 "tuning_profiles_impedance_geometric",
175 [this, saveUserDefinedProfileConfigurationLine]() -> nlohmann::json
176 {
177 nlohmann::json ret = nlohmann::json::array();
178
179 for( const auto& entry : m_tuningProfiles )
180 saveUserDefinedProfileConfigurationLine( ret, entry );
181
182 return ret;
183 },
184 [this, readUserDefinedProfileConfigurationLine]( const nlohmann::json& aJson )
185 {
186 if( !aJson.is_array() )
187 return;
188
190
191 for( const nlohmann::json& entry : aJson )
192 {
193 if( !entry.is_object() || !entry.contains( "profile_name" ) )
194 continue;
195
196 m_tuningProfiles.emplace_back( readUserDefinedProfileConfigurationLine( entry ) );
197 }
198 },
199 {} ) );
200
201 registerMigration( 0, 1, std::bind( &TUNING_PROFILES::migrateSchema0to1, this ) );
202}
203
204
206{
207 // Add frequency and model solder mask fields to tuning profiles
208 if( m_internals->contains( "tuning_profiles_impedance_geometric" )
209 && m_internals->At( "tuning_profiles_impedance_geometric" ).is_array() )
210 {
211 for( auto& profile : m_internals->At( "tuning_profiles_impedance_geometric" ).items() )
212 {
213 profile.value()["frequency"] = 1e9;
214 profile.value()["model_solder_mask"] = false;
215 }
216 }
217
218 return true;
219}
220
221
223{
224 // Release early before destroying members
225 if( m_parent )
226 {
227 m_parent->ReleaseNestedSettings( this );
228 m_parent = nullptr;
229 }
230}
231
232
234{
235 auto itr = std::find_if( m_tuningProfiles.begin(), m_tuningProfiles.end(),
236 [&aProfileName]( const TUNING_PROFILE& aProfile )
237 {
238 return aProfile.m_ProfileName == aProfileName;
239 } );
240
241 if( itr == m_tuningProfiles.end() )
242 return m_nullDelayProfile;
243
244 return *itr;
245}
246
247
249{
250 return m_tuningProfiles == aOther.m_tuningProfiles;
251}
Represents a single line in a time domain profile track propagation setup.
PCB_LAYER_ID m_bottomReferenceLayer
int m_delay
PCB_LAYER_ID m_topReferenceLayer
int m_diffPairGap
bool m_enableTimeDomainTuning
int m_width
PCB_LAYER_ID m_signalLayer
std::vector< PARAM_BASE * > m_params
The list of parameters (owned by this object)
void registerMigration(int aOldSchemaVersion, int aNewSchemaVersion, std::function< bool(void)> aMigrator)
Registers a migration from one schema version to another.
JSON_SETTINGS(const wxString &aFilename, SETTINGS_LOC aLocation, int aSchemaVersion)
std::unique_ptr< JSON_SETTINGS_INTERNALS > m_internals
static int NameToLayer(wxString &aName)
Return the layer number from a layer name.
Definition lset.cpp:117
static wxString Name(PCB_LAYER_ID aLayerId)
Return the fixed name association with aLayerId.
Definition lset.cpp:188
JSON_SETTINGS * m_parent
A pointer to the parent object to load and store from.
NESTED_SETTINGS(const std::string &aName, int aSchemaVersion, JSON_SETTINGS *aParent, const std::string &aPath, bool aLoadFromFile=true)
Like a normal param, but with custom getter and setter functions.
Definition parameters.h:297
TUNING_PROFILE m_nullDelayProfile
TUNING_PROFILE & GetTuningProfile(wxString aProfileName)
std::vector< TUNING_PROFILE > m_tuningProfiles
TUNING_PROFILES(JSON_SETTINGS *aParent, const std::string &aPath)
bool operator==(const TUNING_PROFILES &aOther) const
PCB_LAYER_ID
A quick note on layer IDs:
Definition layer_ids.h:60
Represents a single line in the time domain configuration via overrides configuration grid.
Represents a single line in the tuning profile configuration grid.
constexpr int tuningParametersSchemaVersion