mirror of
https://github.com/ra3xdh/qucs_s
synced 2025-03-28 21:13:26 +00:00

-Introduced cli parameter --cdl for netlisting CDL -Implemented netlisting to console for ngspice and xyce -Increased readability and introduced smart-pointer semantics for some qucs-s main.cpp functions -Increased readability and introduced c++ cast's for casting to Schematic* for affected QucsApp::slotSimulateWithSpice and QucsApp::slotSaveNetlist Signed-off-by: ThomasZecha <zecha@ihp-microelectronics.com>
689 lines
26 KiB
C++
689 lines
26 KiB
C++
/***************************************************************************
|
|
ngspice.cpp
|
|
----------------
|
|
begin : Sat Jan 10 2015
|
|
copyright : (C) 2015 by Vadim Kuznetsov
|
|
email : ra3xdh@gmail.com
|
|
***************************************************************************/
|
|
|
|
/***************************************************************************
|
|
* *
|
|
* This program is free software; you can redistribute it and/or modify *
|
|
* it under the terms of the GNU General Public License as published by *
|
|
* the Free Software Foundation; either version 2 of the License, or *
|
|
* (at your option) any later version. *
|
|
* *
|
|
***************************************************************************/
|
|
|
|
|
|
#include "ngspice.h"
|
|
#include "components/iprobe.h"
|
|
#include "components/vprobe.h"
|
|
#include "components/equation.h"
|
|
#include "components/param_sweep.h"
|
|
#include "components/subcircuit.h"
|
|
#include "spicecomponents/sp_spiceinit.h"
|
|
#include "spicecomponents/xsp_cmlib.h"
|
|
#include "main.h"
|
|
#include "misc.h"
|
|
#include "qucs.h"
|
|
#include "settings.h"
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <QScopedPointer>
|
|
|
|
#include <iostream>
|
|
|
|
/*!
|
|
\file ngspice.cpp
|
|
\brief Implementation of the Ngspice class
|
|
*/
|
|
|
|
/*!
|
|
* \brief Ngspice::Ngspice Class constructor
|
|
* \param schematic Schematic that need to be simulated with Ngspice.
|
|
* \param parent Parent object
|
|
*/
|
|
Ngspice::Ngspice(Schematic* schematic, QObject *parent) :
|
|
AbstractSpiceKernel(schematic, parent),
|
|
a_spinit_name()
|
|
{
|
|
if (QFileInfo(QucsSettings.NgspiceExecutable).isRelative()) { // this check is related to MacOS
|
|
a_simulator_cmd = QFileInfo(QucsSettings.BinDir + QucsSettings.NgspiceExecutable).absoluteFilePath();
|
|
} else {
|
|
a_simulator_cmd = QFileInfo(QucsSettings.NgspiceExecutable).absoluteFilePath();
|
|
}
|
|
if (!QFileInfo::exists(a_simulator_cmd)) {
|
|
a_simulator_cmd = QucsSettings.NgspiceExecutable; //rely on $PATH
|
|
}
|
|
a_simulator_parameters = "";
|
|
a_spinit_name = QDir::toNativeSeparators(QucsSettings.S4Qworkdir+"/.spiceinit");
|
|
}
|
|
|
|
/*!
|
|
* \brief Ngspice::createNetlist Output Ngspice-style netlist to text stream.
|
|
* Netlist contains sections necessary for Ngspice.
|
|
* \param[out] stream QTextStream that associated with spice netlist file
|
|
* \param[out] simulations The list of simulations used by schematic.
|
|
* \param[out] vars The list of output variables and node names.
|
|
* \param[out] outputs The list of spice output raw text files.
|
|
*/
|
|
void Ngspice::createNetlist(
|
|
QTextStream& stream,
|
|
QStringList& simulations,
|
|
QStringList& vars,
|
|
QStringList& outputs)
|
|
{
|
|
Q_UNUSED(simulations);
|
|
|
|
stream << "* Qucs " << PACKAGE_VERSION << " " << a_schematic->getDocName() << "\n";
|
|
|
|
// include math. functions for inter-simulator compat.
|
|
QString mathf_inc;
|
|
bool found = findMathFuncInc(mathf_inc);
|
|
// Let to simulate schematic without mathfunc.inc file
|
|
if (found && QucsSettings.DefaultSimulator != spicecompat::simSpiceOpus)
|
|
stream<<QStringLiteral(".INCLUDE \"%1\"\n").arg(mathf_inc);
|
|
|
|
stream<<collectSpiceLibs(a_schematic); // collect libraries on the top of netlist
|
|
if(!prepareSpiceNetlist(stream)) return; // Unable to perform spice simulation
|
|
startNetlist(stream); // output .PARAM and components
|
|
|
|
if (a_DC_OP_only) {
|
|
stream<<".control\n" // Execute only DC OP analysis
|
|
<<"set filetype=ascii\n" // Ignore all other simulations
|
|
<<"op\n"
|
|
<<"print all > spice4qucs.cir.dc_op\n"
|
|
<<"destroy all\n"
|
|
<<"quit\n"
|
|
<<".endc\n"
|
|
<<".end\n";
|
|
outputs.clear();
|
|
outputs.append("spice4qucs.cir.dc_op");
|
|
return;
|
|
}
|
|
|
|
// set variable names for named nodes and wires
|
|
vars.clear();
|
|
for(Node *pn = a_schematic->a_DocNodes.first(); pn != 0; pn = a_schematic->a_DocNodes.next()) {
|
|
if(pn->Label != 0) {
|
|
if (!vars.contains(pn->Label->Name)) {
|
|
vars.append(pn->Label->Name);
|
|
}
|
|
}
|
|
}
|
|
for(Wire *pw = a_schematic->a_DocWires.first(); pw != 0; pw = a_schematic->a_DocWires.next()) {
|
|
if(pw->Label != 0) {
|
|
if (!vars.contains(pw->Label->Name)) {
|
|
vars.append(pw->Label->Name);
|
|
}
|
|
}
|
|
}
|
|
|
|
for(Component *pc = a_schematic->a_DocComps.first(); pc != 0; pc = a_schematic->a_DocComps.next()) {
|
|
if (pc->isProbe) {
|
|
QString var_pr = pc->getProbeVariable();
|
|
if (!vars.contains(var_pr)) {
|
|
vars.append(var_pr);
|
|
}
|
|
}
|
|
}
|
|
vars.sort();
|
|
|
|
stream << "\n.control\n\n"; //execute simulations
|
|
|
|
if (QucsMain != nullptr) { // if not run from CLI
|
|
if (!QucsMain->ProjName.isEmpty()) {
|
|
// always load osdi from the project directory
|
|
QStringList osdi_ext;
|
|
osdi_ext<<"*.osdi";
|
|
QStringList osdi_files = QucsSettings.QucsWorkDir.entryList(osdi_ext,QDir::Files);
|
|
for(const auto &file : osdi_files) {
|
|
QString abs_file = QucsSettings.QucsWorkDir.absolutePath() +
|
|
QDir::separator() + file;
|
|
stream<<QStringLiteral("pre_osdi '%1'\n").arg(abs_file);
|
|
}
|
|
}
|
|
}
|
|
|
|
// determine which simulations are in use
|
|
unsigned int dcSims = 0;
|
|
unsigned int freqSims = 0;
|
|
unsigned int timeSims = 0;
|
|
unsigned int fourSims = 0;
|
|
unsigned int pzSims = 0;
|
|
|
|
outputs.clear();
|
|
for ( unsigned int i = 0 ; i < a_schematic->a_DocComps.count() ; i++ ) {
|
|
Component *pc = a_schematic->a_DocComps.at(i);
|
|
if ( !pc->isSimulation ) continue;
|
|
if ( pc->isActive != COMP_IS_ACTIVE ) continue;
|
|
|
|
QString sim_typ = pc->Model;
|
|
QString sim_name = pc->Name.toLower();
|
|
QString spiceNetlist;
|
|
|
|
bool hasParSWP = false;
|
|
bool hasDblSWP = false;
|
|
QString cnt_var;
|
|
|
|
// Duplicate .PARAM in .control section. They may be used in euqations
|
|
for ( unsigned int i = 0 ; i < a_schematic->a_DocComps.count() ; i++ ) {
|
|
Component *pc1 = a_schematic->a_DocComps.at(i);
|
|
if ( pc1->isActive != COMP_IS_ACTIVE ) continue;
|
|
if ( pc1->Model == "Eqn" ) {
|
|
spiceNetlist.append((reinterpret_cast<Equation *>(pc1))->getNgspiceScript());
|
|
}
|
|
}
|
|
|
|
QString nods;
|
|
for (const QString& nod : vars) {
|
|
if ( nod.endsWith("#branch") )
|
|
nods.append(QStringLiteral("i(%1) ").arg(nod.section('#', 0, 0)));
|
|
else
|
|
nods.append(QStringLiteral("v(%1) ").arg(nod));
|
|
}
|
|
|
|
for ( unsigned int i = 0 ; i < a_schematic->a_DocComps.count() ; i++ ) {
|
|
Component *pc1 = a_schematic->a_DocComps.at(i);
|
|
if ( !pc1->isSimulation ) continue;
|
|
if ( pc1->isActive != COMP_IS_ACTIVE ) continue;
|
|
QString sim_typ = pc1->Model;
|
|
if ( sim_typ == ".SW" ) {
|
|
QString SwpSim = pc1->Props.at(0)->Value.toLower();
|
|
if ( SwpSim == sim_name ) {
|
|
cnt_var = (reinterpret_cast<Param_Sweep *>(pc1))->getCounterVar();
|
|
if ( !sim_name.startsWith("dc") ) {
|
|
spiceNetlist.append(getParentSWPscript(pc1, sim_name, true, hasDblSWP));
|
|
spiceNetlist.append(pc1->getNgspiceBeforeSim(sim_name));
|
|
hasParSWP = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( sim_typ == ".AC" ) {
|
|
freqSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
} else if ( sim_typ == ".TR" ) {
|
|
timeSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
for ( unsigned int i = 0 ; i < a_schematic->a_DocComps.count() ; i++ ) {
|
|
Component *pc1 = a_schematic->a_DocComps.at(i);
|
|
if ( !pc1->isSimulation ) continue;
|
|
if ( pc1->isActive != COMP_IS_ACTIVE ) continue;
|
|
if ( pc1->Model == ".FOURIER" ) {
|
|
if ( pc1->Props.at(0)->Value.toLower() == sim_name ) {
|
|
fourSims++;
|
|
// Add it twice for THD
|
|
outputs.append("spice4qucs." + pc1->Name.toLower() + ".four");
|
|
outputs.append("spice4qucs." + pc1->Name.toLower() + ".four");
|
|
spiceNetlist.append(pc1->getSpiceNetlist());
|
|
}
|
|
}
|
|
}
|
|
} else if ( sim_typ == ".CUSTOMSIM" ) {
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
nods = pc->Props.at(1)->Value;
|
|
nods.replace(';', ' ');
|
|
outputs.append(pc->Props.at(2)->Value.split(';', Qt::SkipEmptyParts));
|
|
|
|
QRegularExpression ac_rx("^\\s*ac\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression sp_rx("^\\s*sp\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression noise_rx("^\\s*noise\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression disto_rx("^\\s*disto\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression fft_rx("^\\s*fft\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression four_rx("^\\s*(four|fourier)\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression dc_rx("^\\s*dc\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression op_rx("^\\s*op\\s*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression tran_rx("^\\s*tran\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression sens_ac_rx("^\\s*sens\\s.*ac\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression sens_dc_rx("^\\s*sens\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
QRegularExpression pz_rx("^\\s*pz\\s.*", QRegularExpression::CaseInsensitiveOption);
|
|
|
|
QStringList lines = pc->getSpiceNetlist().split('\n');
|
|
for ( const QString& line : lines ) {
|
|
if ( ac_rx.match(line).hasMatch() ) freqSims++ ;
|
|
else if ( sp_rx.match(line).hasMatch() ) freqSims++ ;
|
|
else if ( noise_rx.match(line).hasMatch() ) freqSims++ ;
|
|
else if ( disto_rx.match(line).hasMatch() ) freqSims++ ;
|
|
else if ( fft_rx.match(line).hasMatch() ) freqSims++ ;
|
|
else if ( four_rx.match(line).hasMatch() ) fourSims++ ;
|
|
else if ( dc_rx.match(line).hasMatch() ) dcSims++ ;
|
|
else if ( op_rx.match(line).hasMatch() ) dcSims++ ;
|
|
else if ( tran_rx.match(line).hasMatch() ) timeSims++ ;
|
|
else if ( sens_ac_rx.match(line).hasMatch() ) freqSims++ ;
|
|
else if ( sens_dc_rx.match(line).hasMatch() ) dcSims++ ;
|
|
else if ( pz_rx.match(line).hasMatch() ) pzSims++ ;
|
|
}
|
|
} else if ( sim_typ == ".DISTO" ) {
|
|
freqSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
nods.clear();
|
|
} else if ( sim_typ == ".NOISE" ) {
|
|
freqSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
outputs.append("spice4qucs." + sim_name + ".cir.noise");
|
|
if ( hasParSWP ) { // Set necessary plot number to output Noise spectrum
|
|
// each step of parameter sweep creates new couple of noise plots
|
|
spiceNetlist.append(QStringLiteral("let noise_%1 = 2*%1+1\n").arg(cnt_var));
|
|
spiceNetlist.append(QStringLiteral("setplot noise$&noise_%1\n").arg(cnt_var));
|
|
} else { // Set Noise1 plot to output noise spectrum
|
|
spiceNetlist.append("setplot noise1\n");
|
|
}
|
|
nods = "inoise_spectrum onoise_spectrum";
|
|
} else if ( sim_typ == ".PZ" ) {
|
|
pzSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
QString out = "spice4qucs." + sim_name + ".cir.pz";
|
|
// Add it twice for poles and zeros
|
|
outputs.append(out);
|
|
outputs.append(out);
|
|
} else if ( sim_typ == ".SENS" ) {
|
|
dcSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
outputs.append("spice4qucs." + sim_name + ".ngspice.sens.dc.prn");
|
|
} else if ( sim_typ == ".SENS_AC" ) {
|
|
freqSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
outputs.append("spice4qucs." + sim_name + ".sens.prn");
|
|
} else if ( sim_typ == ".SP" ) {
|
|
freqSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
nods.clear();
|
|
nods.append(' ' + pc->getExtraVariables().join(' '));
|
|
} else if ( sim_typ == ".FFT" ) {
|
|
freqSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
spiceNetlist.append(QStringLiteral("linearize %1\n").arg(nods));
|
|
spiceNetlist.append(QStringLiteral("fft %1\n").arg(nods));
|
|
} else if ( sim_typ == ".DC" ) {
|
|
dcSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
} else if ( sim_typ == ".SW" ) {
|
|
QString SwpSim = pc->Props.at(0)->Value.toLower();
|
|
if ( SwpSim.startsWith("dc") ) {
|
|
dcSims++;
|
|
spiceNetlist.append(pc->getSpiceNetlist());
|
|
} else
|
|
continue;
|
|
} else
|
|
continue;
|
|
|
|
if ( (sim_typ != ".PZ") && (sim_typ != ".SENS") && (sim_typ != ".SENS_AC") ) {
|
|
QStringList dep_vars;
|
|
for ( unsigned int i = 0 ; i < a_schematic->a_DocComps.count() ; i++ ) {
|
|
Component *pc1 = a_schematic->a_DocComps.at(i);
|
|
if ( pc1->isActive != COMP_IS_ACTIVE ) continue;
|
|
if ( pc1->Model == "Eqn" || pc1->Model == "NutmegEq" )
|
|
spiceNetlist.append(pc1->getEquations(sim_name, dep_vars));
|
|
}
|
|
nods.append(' ' + dep_vars.join(' '));
|
|
}
|
|
|
|
if ( sim_typ == ".DC" ) {
|
|
QString out = "spice4qucs." + sim_name + ".ngspice.dc.print";
|
|
spiceNetlist.append(QStringLiteral("print %1 > %2\n").arg(nods).arg(out));
|
|
outputs.append(out);
|
|
} else if ( (sim_typ != ".PZ") && (sim_typ != ".SENS") && (sim_typ != ".SENS_AC") ) {
|
|
nods = nods.simplified();
|
|
if ( !nods.isEmpty() ) {
|
|
QString basenam = "spice4qucs";
|
|
QString filename;
|
|
if ( hasParSWP && hasDblSWP )
|
|
filename = QStringLiteral("%1.%2._swp_swp.plot").arg(basenam).arg(sim_name);
|
|
else if ( hasParSWP )
|
|
filename = QStringLiteral("%1.%2._swp.plot").arg(basenam).arg(sim_name);
|
|
else
|
|
filename = QStringLiteral("%1.%2.plot").arg(basenam).arg(sim_name);
|
|
filename.replace(' ', '_'); // Ngspice cannot understand spaces in filename
|
|
spiceNetlist.append(QStringLiteral("write %1 %2\n").arg(filename).arg(nods));
|
|
outputs.append(filename);
|
|
}
|
|
}
|
|
|
|
for ( unsigned int i = 0 ; i < a_schematic->a_DocComps.count() ; i++ ) {
|
|
Component *pc1 = a_schematic->a_DocComps.at(i);
|
|
if ( !pc1->isSimulation ) continue;
|
|
if ( pc1->isActive != COMP_IS_ACTIVE ) continue;
|
|
QString sim_typ = pc1->Model;
|
|
if ( sim_typ == ".SW" ) {
|
|
QString SwpSim = pc1->Props.at(0)->Value.toLower();
|
|
if ( SwpSim == sim_name ) {
|
|
if ( !sim_name.startsWith("dc") ) {
|
|
spiceNetlist.append(pc1->getNgspiceAfterSim(sim_name));
|
|
spiceNetlist.append(getParentSWPscript(pc1, sim_name, false, hasDblSWP));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
spiceNetlist.append("destroy all\n");
|
|
spiceNetlist.append("reset\n\n");
|
|
stream << spiceNetlist;
|
|
}
|
|
|
|
stream << "exit\n"
|
|
<< ".endc\n";
|
|
stream << ".END\n";
|
|
|
|
a_needsPrefix = ( (dcSims | freqSims | timeSims | fourSims | pzSims) > 1 );
|
|
|
|
qDebug() << '\n'
|
|
<< "Simulations:\n"
|
|
<< "DC: " << dcSims << '\n'
|
|
<< "Frequency: " << freqSims << '\n'
|
|
<< "Time: " << timeSims << '\n'
|
|
<< "Fourier: " << fourSims << '\n'
|
|
<< "Pole-Zero: " << pzSims << '\n'
|
|
<< '\n';
|
|
}
|
|
|
|
/*!
|
|
* \brief Ngspice::getParentSWPscript
|
|
* \param pc_swp
|
|
* \param sim
|
|
* \param before
|
|
* \return
|
|
*/
|
|
QString Ngspice::getParentSWPscript(Component *pc_swp, QString sim, bool before, bool &hasDblSwp)
|
|
{
|
|
hasDblSwp = false;
|
|
QString swp = pc_swp->Name.toLower();
|
|
for ( unsigned int i = 0 ; i < a_schematic->a_DocComps.count() ; i++ ) {
|
|
Component *pc = a_schematic->a_DocComps.at(i);
|
|
if ( !pc->isSimulation ) continue;
|
|
if ( pc->isActive != COMP_IS_ACTIVE ) continue;
|
|
if ( pc->Model == ".SW" ) {
|
|
if ( pc->Props.at(0)->Value.toLower() == swp ) {
|
|
if (before) {
|
|
hasDblSwp = true;
|
|
return pc->getNgspiceBeforeSim(sim, 1);
|
|
} else {
|
|
hasDblSwp = true;
|
|
return pc->getNgspiceAfterSim(sim, 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return QString();
|
|
}
|
|
|
|
/*!
|
|
* \brief Ngspice::slotSimulate Create netlist and execute Ngspice simulator. Netlist
|
|
* is saved at $HOME/.qucs/spice4qucs/spice4qucs.cir
|
|
*/
|
|
void Ngspice::slotSimulate()
|
|
{
|
|
a_output.clear();
|
|
|
|
QString mathf_inc; // drain
|
|
if (!findMathFuncInc(mathf_inc)) {
|
|
a_output.append("[Warning!] " + mathf_inc + " file not found!\n");
|
|
}
|
|
|
|
bool checker_error = false;
|
|
QStringList incompat;
|
|
if (!checkSchematic(incompat)) {
|
|
QString s = incompat.join("; ");
|
|
a_output.append("There were SPICE-incompatible components. Simulator cannot proceed.");
|
|
a_output.append("Incompatible components are: " + s + "\n");
|
|
checker_error = true;
|
|
}
|
|
|
|
if (!checkGround()) {
|
|
a_output.append("No Ground found. Please add at least one ground!\n"
|
|
"Press Insert->Ground in the main menu and connect ground to one "
|
|
"of the schematic nodes.\n");
|
|
checker_error = true;
|
|
}
|
|
|
|
if (!checkSimulations()) {
|
|
a_output.append("No simulation found. Please add at least one simulation!\n"
|
|
"Navigate to the \"simulations\" group in the components panel (left)"
|
|
" and drag simulation to the schematic sheet. Then define its parameters.\n");
|
|
checker_error = true;
|
|
}
|
|
|
|
if (!checkDCSimulation()) {
|
|
a_output.append("Only DC simulation found in the schematic. It has no effect!"
|
|
" Add TRAN, AC, or Sweep simulation to proceed.\n");
|
|
checker_error = true;
|
|
}
|
|
|
|
if (!checkNodeNames(incompat)) {
|
|
QString s = incompat.join("; ");
|
|
a_output.append("There were Nutmeg-incompatible node names. Simulator cannot proceed.\n");
|
|
a_output.append("Incompatible node names are: " + s + "\n");
|
|
checker_error = true;
|
|
}
|
|
|
|
if (checker_error) {
|
|
if (a_console != nullptr)
|
|
a_console->insertPlainText(a_output);
|
|
//emit finished();
|
|
emit errors(QProcess::FailedToStart);
|
|
return;
|
|
}
|
|
|
|
QString netfile = "spice4qucs.cir";
|
|
QString tmp_path = QDir::toNativeSeparators(a_workdir+QDir::separator()+netfile);
|
|
SaveNetlist(tmp_path, false);
|
|
|
|
removeAllSimulatorOutputs();
|
|
|
|
/*XSPICE_CMbuilder *CMbuilder = new XSPICE_CMbuilder(a_schematic);
|
|
CMbuilder->cleanSpiceinit();
|
|
CMbuilder->createSpiceinit(collectSpiceinit(a_schematic));
|
|
if (CMbuilder->needCompile()) {
|
|
CMbuilder->cleanCModelTree();
|
|
CMbuilder->createCModelTree(a_output);
|
|
CMbuilder->compileCMlib(a_output);
|
|
}
|
|
delete CMbuilder;*/
|
|
cleanSpiceinit();
|
|
createSpiceinit(/*initial_spiceinit=*/collectSpiceinit(a_schematic));
|
|
|
|
//startNgSpice(tmp_path);
|
|
a_simProcess->setWorkingDirectory(a_workdir);
|
|
qDebug()<<a_workdir;
|
|
QString cmd = QStringLiteral("\"%1\" %2 %3").arg(a_simulator_cmd,a_simulator_parameters,netfile);
|
|
QStringList cmd_args = misc::parseCmdArgs(cmd);
|
|
QString ngsp_cmd = cmd_args.at(0);
|
|
cmd_args.removeAt(0);
|
|
a_simProcess->start(ngsp_cmd,cmd_args);
|
|
if (QucsMain != nullptr)
|
|
emit started();
|
|
}
|
|
|
|
/*!
|
|
* \brief Ngspice::checkNodeNames Check schematic node names on reserved Nutmeg keywords.
|
|
* \param incompat
|
|
* \return
|
|
*/
|
|
bool Ngspice::checkNodeNames(QStringList &incompat)
|
|
{
|
|
bool result = true;
|
|
for(Node *pn = a_schematic->a_DocNodes.first(); pn != 0; pn = a_schematic->a_DocNodes.next()) {
|
|
if(pn->Label != 0) {
|
|
if (!spicecompat::check_nodename(pn->Label->Name)) {
|
|
incompat.append(pn->Label->Name);
|
|
result = false;
|
|
}
|
|
}
|
|
}
|
|
for(Wire *pw = a_schematic->a_DocWires.first(); pw != 0; pw = a_schematic->a_DocWires.next()) {
|
|
if(pw->Label != 0) {
|
|
if (!spicecompat::check_nodename(pw->Label->Name)) {
|
|
incompat.append(pw->Label->Name);
|
|
result = false;
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*!
|
|
* \brief Ngspice::collectSpiceinit Collects user-specified .spiceinit data.
|
|
* \param incompat
|
|
* \return
|
|
*/
|
|
QString Ngspice::collectSpiceinit(Schematic* sch)
|
|
{
|
|
QStringList collected_spiceinit;
|
|
for(Component *pc = sch->a_DocComps.first(); pc != 0; pc = sch->a_DocComps.next()) {
|
|
if (pc->Model == "SPICEINIT") {
|
|
collected_spiceinit += ((SpiceSpiceinit*)pc)->getSpiceinit();
|
|
} else if (pc->Model == "Sub") {
|
|
Schematic *sub = new Schematic(0, ((Subcircuit *)pc)->getSubcircuitFile());
|
|
if(!sub->loadDocument()) // load document if possible
|
|
{
|
|
delete sub;
|
|
continue;
|
|
}
|
|
collected_spiceinit += collectSpiceinit(sub);
|
|
delete sub;
|
|
}
|
|
}
|
|
return collected_spiceinit.join("");
|
|
}
|
|
|
|
/*!
|
|
* \brief Ngspice::findMathFuncInc Find the ngspice_mathfunc.inc file. This file
|
|
* contains math.functions definitions for Ngspice. It's need to let to simulate
|
|
* circuit if it is not found.
|
|
* \param mathf_inc[out] The filename of include file
|
|
* \return True if found. False otherwise
|
|
*/
|
|
bool Ngspice::findMathFuncInc(QString &mathf_inc)
|
|
{
|
|
QDir qucs_root(QucsSettings.BinDir);
|
|
qucs_root.cdUp();
|
|
mathf_inc = QStringLiteral("%1/share/" QUCS_NAME "/xspice_cmlib/include/ngspice_mathfunc.inc")
|
|
.arg(qucs_root.absolutePath());
|
|
return QFile::exists(mathf_inc);
|
|
}
|
|
|
|
/*!
|
|
* \brief Ngspice::slotProcessOutput Process Ngspice output and report completion
|
|
* percentage.
|
|
*/
|
|
void Ngspice::slotProcessOutput()
|
|
{
|
|
QString s = a_simProcess->readAllStandardOutput();
|
|
QRegularExpression percentage_pattern("^%\\d\\d*\\.\\d\\d.*$");
|
|
if (percentage_pattern.match(s).hasMatch()) {
|
|
int percent = round(s.mid(1,5).toFloat());
|
|
emit progress(percent);
|
|
}
|
|
a_output += s;
|
|
if (a_console != nullptr) {
|
|
a_console->insertPlainText(s);
|
|
a_console->moveCursor(QTextCursor::End);
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* \brief Ngspice::SaveNetlist Create netlist and save it to file without execution
|
|
* of simulator.
|
|
* \param[in] filename Absolute path to netlist
|
|
* \param[in] netlist2Console Whether netlist to console instead to file
|
|
*/
|
|
void Ngspice::SaveNetlist(QString filename, bool netlist2Console)
|
|
{
|
|
a_sims.clear();
|
|
a_vars.clear();
|
|
|
|
QScopedPointer<QString> netlistString;
|
|
QScopedPointer<QTextStream> netlistStream;
|
|
QScopedPointer<QFile> netlistFile;
|
|
|
|
if (netlist2Console)
|
|
{
|
|
netlistString.reset(new QString);
|
|
netlistStream.reset(new QTextStream(netlistString.get()));
|
|
}
|
|
else
|
|
{
|
|
netlistFile.reset(new QFile(filename));
|
|
if (netlistFile->open(QFile::WriteOnly))
|
|
{
|
|
netlistStream.reset(new QTextStream(netlistFile.get()));
|
|
}
|
|
else
|
|
{
|
|
QString msg = QStringLiteral("Tried to save netlist \nin %1\n(could not open for writing!)").arg(filename);
|
|
QString final_msg = QStringLiteral("%1\n This could be an error in the QSettings settings file\n(usually in ~/.config/qucs/qucs_s.conf)\nThe value for S4Q_workdir (default:/spice4qucs) needs to be writeable!\nFor a Simulation Simulation will raise error! (most likely S4Q_workdir does not exists)").arg(msg);
|
|
QMessageBox::critical(nullptr,tr("Problem with SaveNetlist"), final_msg, QMessageBox::Ok);
|
|
return;
|
|
}
|
|
}
|
|
|
|
createNetlist(*netlistStream, a_sims, a_vars, a_output_files);
|
|
|
|
if (netlist2Console)
|
|
{
|
|
std::cout << netlistString->toUtf8().constData() << std::endl;
|
|
}
|
|
}
|
|
|
|
void Ngspice::setSimulatorCmd(QString cmd)
|
|
{
|
|
if (cmd.contains(QRegularExpression("spiceopus(....|)$"))) {
|
|
// spiceopus needs English locale to produce correct decimal point (dot symbol)
|
|
QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
|
|
env.remove("LANG");
|
|
env.insert("LANG","en_US");
|
|
a_simProcess->setProcessEnvironment(env);
|
|
a_simulator_parameters = a_simulator_parameters + "-c";
|
|
} else { // restore system environment
|
|
QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
|
|
a_simProcess->setProcessEnvironment(env);
|
|
}
|
|
|
|
a_simulator_cmd = cmd;
|
|
}
|
|
|
|
void Ngspice::setSimulatorParameters(QString parameters)
|
|
{
|
|
a_simulator_parameters = parameters;
|
|
}
|
|
|
|
void Ngspice::cleanSpiceinit()
|
|
{
|
|
QFileInfo inf(a_spinit_name);
|
|
if (inf.exists()) QFile::remove(a_spinit_name);
|
|
}
|
|
|
|
void Ngspice::createSpiceinit(const QString &initial_spiceinit)
|
|
{
|
|
auto compat_mode = _settings::Get().item<int>("NgspiceCompatMode");
|
|
QString compat_str;
|
|
switch(compat_mode) {
|
|
case spicecompat::NgspLTspice:
|
|
compat_str = "set ngbehavior=ltpsa\n";
|
|
break;
|
|
case spicecompat::NgspHSPICE:
|
|
compat_str = "set ngbehavior=hsa\n";
|
|
break;
|
|
case spicecompat::NgspS3:
|
|
compat_str = "set ngbehavior=s3\n";
|
|
break;
|
|
default: break;
|
|
}
|
|
if (initial_spiceinit.isEmpty() &&
|
|
compat_str.isEmpty()) {
|
|
return;
|
|
}
|
|
QFile spinit(a_spinit_name);
|
|
if (spinit.open(QIODevice::WriteOnly)) {
|
|
QTextStream stream(&spinit);
|
|
stream << compat_str << '\n';
|
|
stream << initial_spiceinit << '\n';
|
|
spinit.close();
|
|
}
|
|
}
|