DPsim
Loading...
Searching...
No Matches
DP_Ph3_PiLine.cpp
1/* Author: Christoph Wirtz <christoph.wirtz@fgh-ma.de>
2 * SPDX-FileCopyrightText: 2025 FGH e.V.
3 * SPDX-License-Identifier: MPL-2.0
4 */
5
6#include <dpsim-models/DP/DP_Ph3_PiLine.h>
7
8using namespace CPS;
9
10DP::Ph3::PiLine::PiLine(String uid, String name, Logger::Level logLevel)
11 : Base::Ph3::PiLine(mAttributes),
12 CompositePowerComp<Complex>(uid, name, true, true, logLevel) {
13 mPhaseType = PhaseType::ABC;
14 setVirtualNodeNumber(1);
15 setTerminalNumber(2);
16
17 SPDLOG_LOGGER_INFO(mSLog, "Create {} {}", this->type(), name);
18 **mIntfVoltage = MatrixComp::Zero(3, 1);
19 **mIntfCurrent = MatrixComp::Zero(3, 1);
20}
21
23SimPowerComp<Complex>::Ptr DP::Ph3::PiLine::clone(String name) {
24 auto copy = PiLine::make(name, mLogLevel);
25 copy->setParameters(**mSeriesRes, **mSeriesInd, **mParallelCap,
27 return copy;
28}
29
31
32 // Static calculation
33 Real omega = 2. * PI * frequency;
34 MatrixComp impedance = MatrixComp::Zero(3, 3);
35 impedance << Complex((**mSeriesRes)(0, 0), omega * (**mSeriesInd)(0, 0)),
36 Complex((**mSeriesRes)(0, 1), omega * (**mSeriesInd)(0, 1)),
37 Complex((**mSeriesRes)(0, 2), omega * (**mSeriesInd)(0, 2)),
38 Complex((**mSeriesRes)(1, 0), omega * (**mSeriesInd)(1, 0)),
39 Complex((**mSeriesRes)(1, 1), omega * (**mSeriesInd)(1, 1)),
40 Complex((**mSeriesRes)(1, 2), omega * (**mSeriesInd)(1, 2)),
41 Complex((**mSeriesRes)(2, 0), omega * (**mSeriesInd)(2, 0)),
42 Complex((**mSeriesRes)(2, 1), omega * (**mSeriesInd)(2, 1)),
43 Complex((**mSeriesRes)(2, 2), omega * (**mSeriesInd)(2, 2));
44 MatrixComp vInitABC = MatrixComp::Zero(3, 1);
45 vInitABC(0, 0) = RMS3PH_TO_PEAK1PH * initialSingleVoltage(1) -
46 RMS3PH_TO_PEAK1PH * initialSingleVoltage(0);
47 vInitABC(1, 0) = vInitABC(0, 0) * SHIFT_TO_PHASE_B;
48 vInitABC(2, 0) = vInitABC(0, 0) * SHIFT_TO_PHASE_C;
49 MatrixComp iInit = impedance.inverse() * vInitABC;
50 **mIntfCurrent = iInit;
51 **mIntfVoltage = vInitABC;
52
53 // Initialization of virtual node
54 // Initial voltage of phase B,C is set after A
55 MatrixComp vInitTerm0 = MatrixComp::Zero(3, 1);
56 vInitTerm0(0, 0) = RMS3PH_TO_PEAK1PH * initialSingleVoltage(0);
57 vInitTerm0(1, 0) = vInitTerm0(0, 0) * SHIFT_TO_PHASE_B;
58 vInitTerm0(2, 0) = vInitTerm0(0, 0) * SHIFT_TO_PHASE_C;
59
60 mVirtualNodes[0]->setInitialVoltage(PEAK1PH_TO_RMS3PH *
61 (vInitTerm0 + **mSeriesRes * iInit));
62
63 // Create series sub components
65 std::make_shared<DP::Ph3::Resistor>(**mName + "_res", mLogLevel);
66 mSubSeriesResistor->setParameters(**mSeriesRes);
67 mSubSeriesResistor->connect({mTerminals[0]->node(), mVirtualNodes[0]});
69 mSubSeriesResistor->initializeFromNodesAndTerminals(frequency);
71 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT,
72 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT, false);
73
75 std::make_shared<DP::Ph3::Inductor>(**mName + "_ind", mLogLevel);
76 mSubSeriesInductor->setParameters(**mSeriesInd);
77 mSubSeriesInductor->connect({mVirtualNodes[0], mTerminals[1]->node()});
79 mSubSeriesInductor->initializeFromNodesAndTerminals(frequency);
81 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT,
82 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT, true);
83
84 // By default there is always a small conductance to ground to
85 // avoid problems with floating nodes.
86 Matrix defaultParallelCond = Matrix::Zero(3, 3);
87 defaultParallelCond << 1e-6, 0, 0, 0, 1e-6, 0, 0, 0, 1e-6;
89 ((**mParallelCond)(0, 0) > 0) ? **mParallelCond : defaultParallelCond;
90
91 // Create parallel sub components
93 std::make_shared<DP::Ph3::Resistor>(**mName + "_con0", mLogLevel);
94 mSubParallelResistor0->setParameters(2. * (**mParallelCond).inverse());
95 mSubParallelResistor0->connect(
96 SimNode::List{SimNode::GND, mTerminals[0]->node()});
98 mSubParallelResistor0->initializeFromNodesAndTerminals(frequency);
100 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT,
101 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT, false);
102
104 std::make_shared<DP::Ph3::Resistor>(**mName + "_con1", mLogLevel);
105 mSubParallelResistor1->setParameters(2. * (**mParallelCond).inverse());
106 mSubParallelResistor1->connect(
107 SimNode::List{SimNode::GND, mTerminals[1]->node()});
109 mSubParallelResistor1->initializeFromNodesAndTerminals(frequency);
111 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT,
112 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT, false);
113
114 if ((**mParallelCap)(0, 0) > 0) {
115 mSubParallelCapacitor0 =
116 std::make_shared<DP::Ph3::Capacitor>(**mName + "_cap0", mLogLevel);
117 mSubParallelCapacitor0->setParameters(**mParallelCap / 2.);
118 mSubParallelCapacitor0->connect(
119 SimNode::List{SimNode::GND, mTerminals[0]->node()});
120 mSubParallelCapacitor0->initialize(mFrequencies);
121 mSubParallelCapacitor0->initializeFromNodesAndTerminals(frequency);
122 addMNASubComponent(mSubParallelCapacitor0,
123 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT,
124 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT, true);
125
127 std::make_shared<DP::Ph3::Capacitor>(**mName + "_cap1", mLogLevel);
128 mSubParallelCapacitor1->setParameters(**mParallelCap / 2.);
129 mSubParallelCapacitor1->connect(
130 SimNode::List{SimNode::GND, mTerminals[1]->node()});
132 mSubParallelCapacitor1->initializeFromNodesAndTerminals(frequency);
134 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT,
135 MNA_SUBCOMP_TASK_ORDER::TASK_BEFORE_PARENT, true);
136 }
137
138 SPDLOG_LOGGER_INFO(
139 mSLog,
140 "\n--- Initialization from powerflow ---"
141 "\nVoltage across: {:s}"
142 "\nCurrent: {:s}"
143 "\nTerminal 0 voltage: {:s}"
144 "\nTerminal 1 voltage: {:s}"
145 "\nVirtual Node 1 voltage: {:s}"
146 "\n--- Initialization from powerflow finished ---",
147 Logger::matrixCompToString(**mIntfVoltage),
148 Logger::matrixCompToString(**mIntfCurrent),
149 Logger::phasorToString(RMS3PH_TO_PEAK1PH * initialSingleVoltage(0)),
150 Logger::phasorToString(RMS3PH_TO_PEAK1PH * initialSingleVoltage(1)),
151 Logger::phasorToString(mVirtualNodes[0]->initialSingleVoltage()));
152}
153
155 AttributeBase::List &prevStepDependencies,
156 AttributeBase::List &attributeDependencies,
157 AttributeBase::List &modifiedAttributes) {
158 // add pre-step dependencies of component itself
159 prevStepDependencies.push_back(mIntfCurrent);
160 prevStepDependencies.push_back(mIntfVoltage);
161 modifiedAttributes.push_back(mRightVector);
162}
163
164void DP::Ph3::PiLine::mnaParentPreStep(Real time, Int timeStepCount) {
165 // pre-step of component itself
167}
168
169void DP::Ph3::PiLine::mnaParentAddPostStepDependencies(
170 AttributeBase::List &prevStepDependencies,
171 AttributeBase::List &attributeDependencies,
172 AttributeBase::List &modifiedAttributes,
173 Attribute<Matrix>::Ptr &leftVector) {
174 // add post-step dependencies of component itself
175 attributeDependencies.push_back(leftVector);
176 modifiedAttributes.push_back(mIntfVoltage);
177 modifiedAttributes.push_back(mIntfCurrent);
178}
179
180void DP::Ph3::PiLine::mnaParentPostStep(Real time, Int timeStepCount,
181 Attribute<Matrix>::Ptr &leftVector) {
182 // post-step of component itself
183 this->mnaUpdateVoltage(**leftVector);
184 this->mnaUpdateCurrent(**leftVector);
185}
186
187void DP::Ph3::PiLine::mnaCompUpdateVoltage(const Matrix &leftVector) {
188 // v1 - v0
189 **mIntfVoltage = MatrixComp::Zero(3, 1);
190 if (terminalNotGrounded(1)) {
191 (**mIntfVoltage)(0, 0) =
192 Math::complexFromVectorElement(leftVector, matrixNodeIndex(1, 0));
193 (**mIntfVoltage)(1, 0) =
194 Math::complexFromVectorElement(leftVector, matrixNodeIndex(1, 1));
195 (**mIntfVoltage)(2, 0) =
196 Math::complexFromVectorElement(leftVector, matrixNodeIndex(1, 2));
197 }
198 if (terminalNotGrounded(0)) {
199 (**mIntfVoltage)(0, 0) =
200 (**mIntfVoltage)(0, 0) -
201 Math::complexFromVectorElement(leftVector, matrixNodeIndex(0, 0));
202 (**mIntfVoltage)(1, 0) =
203 (**mIntfVoltage)(1, 0) -
204 Math::complexFromVectorElement(leftVector, matrixNodeIndex(0, 1));
205 (**mIntfVoltage)(2, 0) =
206 (**mIntfVoltage)(2, 0) -
207 Math::complexFromVectorElement(leftVector, matrixNodeIndex(0, 2));
208 }
209}
210
211void DP::Ph3::PiLine::mnaCompUpdateCurrent(const Matrix &leftVector) {
212 **mIntfCurrent = mSubSeriesInductor->intfCurrent();
213}
214
215// #### Tear Methods ####
216MNAInterface::List DP::Ph3::PiLine::mnaTearGroundComponents() {
217 MNAInterface::List gndComponents;
218
219 gndComponents.push_back(mSubParallelResistor0);
220 gndComponents.push_back(mSubParallelResistor1);
221
222 if ((**mParallelCap)(0, 0) > 0) {
223 gndComponents.push_back(mSubParallelCapacitor0);
224 gndComponents.push_back(mSubParallelCapacitor1);
225 }
226
227 return gndComponents;
228}
229
230void DP::Ph3::PiLine::mnaTearInitialize(Real omega, Real timeStep) {
231 mSubSeriesResistor->mnaTearSetIdx(mTearIdx);
232 mSubSeriesResistor->mnaTearInitialize(omega, timeStep);
233 mSubSeriesInductor->mnaTearSetIdx(mTearIdx);
234 mSubSeriesInductor->mnaTearInitialize(omega, timeStep);
235}
236
237void DP::Ph3::PiLine::mnaTearApplyMatrixStamp(SparseMatrixRow &tearMatrix) {
238 mSubSeriesResistor->mnaTearApplyMatrixStamp(tearMatrix);
239 mSubSeriesInductor->mnaTearApplyMatrixStamp(tearMatrix);
240}
241
242void DP::Ph3::PiLine::mnaTearApplyVoltageStamp(Matrix &voltageVector) {
243 mSubSeriesInductor->mnaTearApplyVoltageStamp(voltageVector);
244}
245
246void DP::Ph3::PiLine::mnaTearPostStep(MatrixComp voltage, MatrixComp current) {
247 mSubSeriesInductor->mnaTearPostStep(voltage - (**mSeriesRes * current),
248 current);
249 (**mIntfCurrent) = mSubSeriesInductor->intfCurrent();
250}
const Attribute< Matrix >::Ptr mParallelCond
Conductance in parallel to the line [S].
const Attribute< Matrix >::Ptr mSeriesRes
Resistance along the line [ohms].
const Attribute< Matrix >::Ptr mSeriesInd
Inductance along the line [H].
const Attribute< Matrix >::Ptr mParallelCap
Capacitance in parallel to the line [F].
void addMNASubComponent(typename SimPowerComp< Complex >::Ptr subc, MNA_SUBCOMP_TASK_ORDER preStepOrder, MNA_SUBCOMP_TASK_ORDER postStepOrder, Bool contributeToRightVector)
CompositePowerComp(String uid, String name, Bool hasPreStep, Bool hasPostStep, Logger::Level logLevel)
void mnaParentPreStep(Real time, Int timeStepCount) override
MNA pre and post step operations.
std::shared_ptr< Resistor > mSubSeriesResistor
Series Resistor submodel.
void mnaCompUpdateCurrent(const Matrix &leftVector) override
Updates internal current variable of the component.
PiLine(String uid, String name, Logger::Level logLevel=Logger::Level::off)
Defines UID, name and logging level.
std::shared_ptr< Capacitor > mSubParallelCapacitor1
Parallel capacitor submodel at Terminal 1.
void mnaCompUpdateVoltage(const Matrix &leftVector) override
Updates internal voltage variable of the component.
SimPowerComp< Complex >::Ptr clone(String copySuffix) override
DEPRECATED: Remove method.
std::shared_ptr< Inductor > mSubSeriesInductor
Series Inductance submodel.
std::shared_ptr< Resistor > mSubParallelResistor0
Parallel Resistor submodel at Terminal 0.
void mnaParentAddPreStepDependencies(AttributeBase::List &prevStepDependencies, AttributeBase::List &attributeDependencies, AttributeBase::List &modifiedAttributes) override
add MNA pre and post step dependencies
void initializeFromNodesAndTerminals(Real frequency) override
Initializes component from power flow data.
std::shared_ptr< Resistor > mSubParallelResistor1
Parallel resistor submodel at Terminal 1.
const Attribute< String >::Ptr mName
Human readable name.
String uid()
Returns unique id.
String type()
Get component type (cross-platform)
AttributeList::Ptr mAttributes
Attribute List.
Attribute< Matrix >::Ptr mRightVector
void mnaApplyRightSideVectorStamp(Matrix &rightVector) final
const Attribute< MatrixVar< Complex > >::Ptr mIntfCurrent
SimTerminal< Complex >::List mTerminals
const Attribute< MatrixVar< Complex > >::Ptr mIntfVoltage
SimNode< Complex >::List mVirtualNodes
Logger::Level mLogLevel
Component logger control for internal variables.
Logger::Log mSLog
Component logger.