DPsim
Loading...
Searching...
No Matches
DP_Ph1_Capacitor.cpp
1/* Copyright 2017-2021 Institute for Automation of Complex Power Systems,
2 * EONERC, RWTH Aachen University
3 *
4 * This Source Code Form is subject to the terms of the Mozilla Public
5 * License, v. 2.0. If a copy of the MPL was not distributed with this
6 * file, You can obtain one at https://mozilla.org/MPL/2.0/.
7 *********************************************************************************/
8
9#include <dpsim-models/DP/DP_Ph1_Capacitor.h>
10
11using namespace CPS;
12using namespace CPS::DP::Ph1;
13
14DP::Ph1::Capacitor::Capacitor(String uid, String name, Logger::Level logLevel)
15 : MNASimPowerComp<Complex>(uid, name, true, true, logLevel),
16 Base::Ph1::Capacitor(mAttributes) {
17 mEquivCurrent = {0, 0};
18 **mIntfVoltage = MatrixComp::Zero(1, 1);
19 **mIntfCurrent = MatrixComp::Zero(1, 1);
20 setTerminalNumber(2);
21}
22
23SimPowerComp<Complex>::Ptr DP::Ph1::Capacitor::clone(String name) {
24 auto copy = Capacitor::make(name, mLogLevel);
25 copy->setParameters(**mCapacitance);
26 return copy;
27}
28
29void DP::Ph1::Capacitor::initialize(Matrix frequencies) {
31
32 mEquivCurrent = MatrixComp::Zero(mNumFreqs, 1);
33 mEquivCond = MatrixComp::Zero(mNumFreqs, 1);
34 mPrevVoltCoeff = MatrixComp::Zero(mNumFreqs, 1);
35}
36
38
39 Real omega = 2 * PI * frequency;
40 Complex impedance = {0, -1. / (omega * **mCapacitance)};
41 (**mIntfVoltage)(0, 0) = initialSingleVoltage(1) - initialSingleVoltage(0);
42
43 // Admittance form (I = V*jwC): a zero capacitance gives 0 current, not a NaN from 1/(omega*C).
44 (**mIntfCurrent)(0, 0) =
45 (**mIntfVoltage)(0, 0) * Complex(0, omega * **mCapacitance);
46
47 SPDLOG_LOGGER_INFO(mSLog,
48 "\nCapacitance [F]: {:s}"
49 "\nImpedance [Ohm]: {:s}",
50 Logger::realToString(**mCapacitance),
51 Logger::complexToString(impedance));
52 SPDLOG_LOGGER_INFO(mSLog,
53 "\n--- Initialization from powerflow ---"
54 "\nVoltage across: {:s}"
55 "\nCurrent: {:s}"
56 "\nTerminal 0 voltage: {:s}"
57 "\nTerminal 1 voltage: {:s}"
58 "\n--- Initialization from powerflow finished ---",
59 Logger::phasorToString((**mIntfVoltage)(0, 0)),
60 Logger::phasorToString((**mIntfCurrent)(0, 0)),
61 Logger::phasorToString(initialSingleVoltage(0)),
62 Logger::phasorToString(initialSingleVoltage(1)));
63 mSLog->flush();
64}
65
66void DP::Ph1::Capacitor::mnaCompInitialize(Real omega, Real timeStep,
67 Attribute<Matrix>::Ptr leftVector) {
69
70 Real equivCondReal = 2.0 * **mCapacitance / timeStep;
71 Real prevVoltCoeffReal = 2.0 * **mCapacitance / timeStep;
72
73 for (UInt freq = 0; freq < mNumFreqs; freq++) {
74 Real equivCondImag = 2. * PI * mFrequencies(freq, 0) * **mCapacitance;
75 mEquivCond(freq, 0) = {equivCondReal, equivCondImag};
76 Real prevVoltCoeffImag = -2. * PI * mFrequencies(freq, 0) * **mCapacitance;
77 mPrevVoltCoeff(freq, 0) = {prevVoltCoeffReal, prevVoltCoeffImag};
78
79 mEquivCurrent(freq, 0) =
80 -(**mIntfCurrent)(0, freq) +
81 -mPrevVoltCoeff(freq, 0) * (**mIntfVoltage)(0, freq);
82 (**mIntfCurrent)(0, freq) =
83 mEquivCond(freq, 0) * (**mIntfVoltage)(0, freq) +
84 mEquivCurrent(freq, 0);
85 }
86
87 SPDLOG_LOGGER_INFO(mSLog,
88 "\n--- MNA initialization ---"
89 "\nInitial voltage {:s}"
90 "\nInitial current {:s}"
91 "\nEquiv. current {:s}"
92 "\n--- MNA initialization finished ---",
93 Logger::phasorToString((**mIntfVoltage)(0, 0)),
94 Logger::phasorToString((**mIntfCurrent)(0, 0)),
95 Logger::complexToString(mEquivCurrent(0, 0)));
96 mSLog->flush();
97}
98
99void DP::Ph1::Capacitor::mnaCompInitializeHarm(
100 Real omega, Real timeStep,
101 std::vector<Attribute<Matrix>::Ptr> leftVectors) {
102 updateMatrixNodeIndices();
103
104 Real equivCondReal = 2.0 * **mCapacitance / timeStep;
105 Real prevVoltCoeffReal = 2.0 * **mCapacitance / timeStep;
106
107 for (UInt freq = 0; freq < mNumFreqs; freq++) {
108 Real equivCondImag = 2. * PI * mFrequencies(freq, 0) * **mCapacitance;
109 mEquivCond(freq, 0) = {equivCondReal, equivCondImag};
110 Real prevVoltCoeffImag = -2. * PI * mFrequencies(freq, 0) * **mCapacitance;
111 mPrevVoltCoeff(freq, 0) = {prevVoltCoeffReal, prevVoltCoeffImag};
112
113 mEquivCurrent(freq, 0) =
114 -(**mIntfCurrent)(0, freq) +
115 -mPrevVoltCoeff(freq, 0) * (**mIntfVoltage)(0, freq);
116 (**mIntfCurrent)(0, freq) =
117 mEquivCond(freq, 0) * (**mIntfVoltage)(0, freq) +
118 mEquivCurrent(freq, 0);
119 }
120
121 mMnaTasks.push_back(std::make_shared<MnaPreStepHarm>(*this));
122 mMnaTasks.push_back(std::make_shared<MnaPostStepHarm>(*this, leftVectors));
123 **mRightVector = Matrix::Zero(leftVectors[0]->get().rows(), mNumFreqs);
124}
125
127 SparseMatrixRow &systemMatrix) {
128 for (UInt freq = 0; freq < mNumFreqs; freq++) {
129 MNAStampUtils::stampAdmittance(
130 mEquivCond(freq, 0), systemMatrix, matrixNodeIndex(0),
131 matrixNodeIndex(1), terminalNotGrounded(0), terminalNotGrounded(1),
132 mSLog, mNumFreqs, freq);
133 }
134}
135
136void DP::Ph1::Capacitor::mnaCompApplySystemMatrixStampHarm(
137 SparseMatrixRow &systemMatrix, Int freqIdx) {
138 MNAStampUtils::stampAdmittance(mEquivCond(freqIdx, 0), systemMatrix,
139 matrixNodeIndex(0), matrixNodeIndex(1),
140 terminalNotGrounded(0), terminalNotGrounded(1),
141 mSLog);
142}
143
145 for (UInt freq = 0; freq < mNumFreqs; freq++) {
146 //mCureqr = mCurrr + mGcr * mDeltavr + mGci * mDeltavi;
147 //mCureqi = mCurri + mGcr * mDeltavi - mGci * mDeltavr;
148 mEquivCurrent(freq, 0) =
149 -(**mIntfCurrent)(0, freq) +
150 -mPrevVoltCoeff(freq, 0) * (**mIntfVoltage)(0, freq);
151
152 if (terminalNotGrounded(0))
153 Math::setVectorElement(rightVector, matrixNodeIndex(0),
154 mEquivCurrent(freq, 0), mNumFreqs, freq);
155 if (terminalNotGrounded(1))
156 Math::setVectorElement(rightVector, matrixNodeIndex(1),
157 -mEquivCurrent(freq, 0), mNumFreqs, freq);
158
159 SPDLOG_LOGGER_DEBUG(mSLog, "MNA EquivCurrent {:f}+j{:f}",
160 mEquivCurrent(freq, 0).real(),
161 mEquivCurrent(freq, 0).imag());
162 if (terminalNotGrounded(0))
163 SPDLOG_LOGGER_DEBUG(mSLog, "Add {:f}+j{:f} to source vector at {:d}",
164 mEquivCurrent(freq, 0).real(),
165 mEquivCurrent(freq, 0).imag(), matrixNodeIndex(0));
166 if (terminalNotGrounded(1))
167 SPDLOG_LOGGER_DEBUG(mSLog, "Add {:f}+j{:f} to source vector at {:d}",
168 -mEquivCurrent(freq, 0).real(),
169 -mEquivCurrent(freq, 0).imag(), matrixNodeIndex(1));
170 }
171}
172
173void DP::Ph1::Capacitor::mnaCompApplyRightSideVectorStampHarm(
174 Matrix &rightVector) {
175 for (UInt freq = 0; freq < mNumFreqs; freq++) {
176 //mCureqr = mCurrr + mGcr * mDeltavr + mGci * mDeltavi;
177 //mCureqi = mCurri + mGcr * mDeltavi - mGci * mDeltavr;
178 mEquivCurrent(freq, 0) =
179 -(**mIntfCurrent)(0, freq) +
180 -mPrevVoltCoeff(freq, 0) * (**mIntfVoltage)(0, freq);
181
182 if (terminalNotGrounded(0))
183 Math::setVectorElement(rightVector, matrixNodeIndex(0),
184 mEquivCurrent(freq, 0), 1, 0, freq);
185 if (terminalNotGrounded(1))
186 Math::setVectorElement(rightVector, matrixNodeIndex(1),
187 -mEquivCurrent(freq, 0), 1, 0, freq);
188 }
189}
190
191void DP::Ph1::Capacitor::mnaCompApplyRightSideVectorStampHarm(
192 Matrix &rightVector, Int freq) {
193 //mCureqr = mCurrr + mGcr * mDeltavr + mGci * mDeltavi;
194 //mCureqi = mCurri + mGcr * mDeltavi - mGci * mDeltavr;
195 mEquivCurrent(freq, 0) = -(**mIntfCurrent)(0, freq) +
196 -mPrevVoltCoeff(freq, 0) * (**mIntfVoltage)(0, freq);
197
198 if (terminalNotGrounded(0))
199 Math::setVectorElement(rightVector, matrixNodeIndex(0),
200 mEquivCurrent(freq, 0));
201 if (terminalNotGrounded(1))
202 Math::setVectorElement(rightVector, matrixNodeIndex(1),
203 -mEquivCurrent(freq, 0));
204}
205
207 AttributeBase::List &prevStepDependencies,
208 AttributeBase::List &attributeDependencies,
209 AttributeBase::List &modifiedAttributes) {
210 // actually depends on C, but then we'd have to modify the system matrix anyway
211 prevStepDependencies.push_back(mIntfCurrent);
212 prevStepDependencies.push_back(mIntfVoltage);
213 modifiedAttributes.push_back(mRightVector);
214}
215
216void DP::Ph1::Capacitor::mnaCompPreStep(Real time, Int timeStepCount) {
218}
219
221 AttributeBase::List &prevStepDependencies,
222 AttributeBase::List &attributeDependencies,
223 AttributeBase::List &modifiedAttributes,
224 Attribute<Matrix>::Ptr &leftVector) {
225 attributeDependencies.push_back(leftVector);
226 modifiedAttributes.push_back(mIntfVoltage);
227 modifiedAttributes.push_back(mIntfCurrent);
228}
229
230void DP::Ph1::Capacitor::mnaCompPostStep(Real time, Int timeStepCount,
231 Attribute<Matrix>::Ptr &leftVector) {
232 this->mnaUpdateVoltage(**leftVector);
233 this->mnaUpdateCurrent(**leftVector);
234}
235
236void DP::Ph1::Capacitor::MnaPreStepHarm::execute(Real time, Int timeStepCount) {
237 mCapacitor.mnaCompApplyRightSideVectorStampHarm(**mCapacitor.mRightVector);
238}
239
240void DP::Ph1::Capacitor::MnaPostStepHarm::execute(Real time,
241 Int timeStepCount) {
242 for (UInt freq = 0; freq < mCapacitor.mNumFreqs; freq++)
243 mCapacitor.mnaCompUpdateVoltageHarm(**mLeftVectors[freq], freq);
244 mCapacitor.mnaCompUpdateCurrentHarm();
245}
246
247void DP::Ph1::Capacitor::mnaCompUpdateVoltage(const Matrix &leftVector) {
248 // v1 - v0
249 for (UInt freq = 0; freq < mNumFreqs; freq++) {
250 (**mIntfVoltage)(0, freq) = 0;
251 if (terminalNotGrounded(1))
252 (**mIntfVoltage)(0, freq) = Math::complexFromVectorElement(
253 leftVector, matrixNodeIndex(1), mNumFreqs, freq);
254 if (terminalNotGrounded(0))
255 (**mIntfVoltage)(0, freq) =
256 (**mIntfVoltage)(0, freq) -
257 Math::complexFromVectorElement(leftVector, matrixNodeIndex(0),
258 mNumFreqs, freq);
259
260 SPDLOG_LOGGER_DEBUG(mSLog, "Voltage {:e}<{:e}",
261 std::abs((**mIntfVoltage)(0, freq)),
262 std::arg((**mIntfVoltage)(0, freq)));
263 }
264}
265
266void DP::Ph1::Capacitor::mnaCompUpdateVoltageHarm(const Matrix &leftVector,
267 Int freqIdx) {
268 // v1 - v0
269 (**mIntfVoltage)(0, freqIdx) = 0;
270 if (terminalNotGrounded(1))
271 (**mIntfVoltage)(0, freqIdx) =
272 Math::complexFromVectorElement(leftVector, matrixNodeIndex(1));
273 if (terminalNotGrounded(0))
274 (**mIntfVoltage)(0, freqIdx) =
275 (**mIntfVoltage)(0, freqIdx) -
276 Math::complexFromVectorElement(leftVector, matrixNodeIndex(0));
277
278 SPDLOG_LOGGER_DEBUG(mSLog, "Voltage {:s}",
279 Logger::phasorToString((**mIntfVoltage)(0, freqIdx)));
280}
281
282void DP::Ph1::Capacitor::mnaCompUpdateCurrent(const Matrix &leftVector) {
283 for (UInt freq = 0; freq < mNumFreqs; freq++) {
284 (**mIntfCurrent)(0, freq) =
285 mEquivCond(freq, 0) * (**mIntfVoltage)(0, freq) +
286 mEquivCurrent(freq, 0);
287 SPDLOG_LOGGER_DEBUG(mSLog, "Current {:s}",
288 Logger::phasorToString((**mIntfCurrent)(0, freq)));
289 }
290}
291
292void DP::Ph1::Capacitor::mnaCompUpdateCurrentHarm() {
293 for (UInt freq = 0; freq < mNumFreqs; freq++) {
294 (**mIntfCurrent)(0, freq) =
295 mEquivCond(freq, 0) * (**mIntfVoltage)(0, freq) +
296 mEquivCurrent(freq, 0);
297 SPDLOG_LOGGER_DEBUG(mSLog, "Current {:s}",
298 Logger::phasorToString((**mIntfCurrent)(0, freq)));
299 }
300}
const CPS::Attribute< Real >::Ptr mCapacitance
Capacitance [F].
void mnaCompApplySystemMatrixStamp(SparseMatrixRow &systemMatrix) override
Stamps system matrix.
Capacitor(String uid, String name, Logger::Level logLevel=Logger::Level::off)
Defines UID, name and logging level.
void mnaCompPreStep(Real time, Int timeStepCount) override
MNA pre step operations.
void mnaCompAddPreStepDependencies(AttributeBase::List &prevStepDependencies, AttributeBase::List &attributeDependencies, AttributeBase::List &modifiedAttributes) override
Add MNA pre step dependencies.
void initializeFromNodesAndTerminals(Real frequency) override
Initializes component from power flow data.
MatrixComp mPrevVoltCoeff
Coefficient in front of previous voltage value for harmonics.
void mnaCompAddPostStepDependencies(AttributeBase::List &prevStepDependencies, AttributeBase::List &attributeDependencies, AttributeBase::List &modifiedAttributes, Attribute< Matrix >::Ptr &leftVector) override
Add MNA post step dependencies.
void initialize(Matrix frequencies) override
Initialize components with correct network frequencies.
MatrixComp mEquivCurrent
DC equivalent current source for harmonics [A].
void mnaCompInitialize(Real omega, Real timeStep, Attribute< Matrix >::Ptr leftVector) override
Initializes internal variables of the component.
void mnaCompUpdateVoltage(const Matrix &leftVector) override
Update interface voltage from MNA system result.
MatrixComp mEquivCond
Equivalent conductance for harmonics [S].
void mnaCompUpdateCurrent(const Matrix &leftVector) override
Update interface current from MNA system result.
void mnaCompApplyRightSideVectorStamp(Matrix &rightVector) override
Stamps right side (source) vector.
SimPowerComp< Complex >::Ptr clone(String name) override
Returns a modified copy of the component with the given suffix added to the name and without.
void mnaCompPostStep(Real time, Int timeStepCount, Attribute< Matrix >::Ptr &leftVector) override
MNA post step operations.
String uid()
Returns unique id.
AttributeList::Ptr mAttributes
Attribute List.
void mnaUpdateCurrent(const Matrix &leftVector) final
void mnaUpdateVoltage(const Matrix &leftVector) final
MNASimPowerComp(String uid, String name, Bool hasPreStep, Bool hasPostStep, Logger::Level logLevel)
Attribute< Matrix >::Ptr mRightVector
void mnaApplyRightSideVectorStamp(Matrix &rightVector) final
const Attribute< MatrixVar< Complex > >::Ptr mIntfCurrent
virtual void initialize(Matrix frequencies)
Initialize components with correct network frequencies.
const Attribute< MatrixVar< Complex > >::Ptr mIntfVoltage
Logger::Level mLogLevel
Component logger control for internal variables.
Logger::Log mSLog
Component logger.