/*************************************************************************** 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 "xspice_cmbuilder.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" #ifdef HAVE_CONFIG_H #include "config.h" #endif /*! \file ngspice.cpp \brief Implementation of the Ngspice class */ /*! * \brief Ngspice::Ngspice Class constructor * \param sch_ Schematic that need to be simulated with Ngspice. * \param parent Parent object */ Ngspice::Ngspice(Schematic *sch_, QObject *parent) : AbstractSpiceKernel(sch_, parent) { simulator_cmd = QucsSettings.NgspiceExecutable; simulator_parameters = ""; } /*! * \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, int , QStringList &simulations, QStringList &vars, QStringList &outputs) { // include math. functions for inter-simulator compat. QString mathf_inc; bool found = findMathFuncInc(mathf_inc); stream<DocName); // Let to simulate schematic without mathfunc.inc file if (found && QucsSettings.DefaultSimulator != spicecompat::simSpiceOpus) stream< spice4qucs.cir.dc_op\n" <<"destroy all\n" <<"quit\n" <<".endc\n" <<".end\n"; outputs.clear(); outputs.append("spice4qucs.cir.dc_op"); return; } // determine which simulations are in use simulations.clear(); for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) { if(pc->isSimulation) { s = pc->getSpiceNetlist(); QString sim_typ = pc->Model; if (sim_typ==".AC") simulations.append("ac"); if (sim_typ==".TR") simulations.append("tran"); if (sim_typ==".CUSTOMSIM") simulations.append("custom"); if (sim_typ==".DISTO") simulations.append("disto"); if (sim_typ==".NOISE") simulations.append("noise"); if (sim_typ==".PZ") simulations.append("pz"); if (sim_typ==".SENS") simulations.append("sens"); if (sim_typ==".SENS_AC") simulations.append("sens_ac"); if (sim_typ==".SP") simulations.append("sp"); if (sim_typ==".FFT") simulations.append("fft"); if ((sim_typ==".SW")&& (pc->Props.at(0)->Value.startsWith("DC"))) simulations.append("dc"); // stream<DocNodes.first(); pn != 0; pn = Sch->DocNodes.next()) { if(pn->Label != 0) { if (!vars.contains(pn->Label->Name)) { vars.append(pn->Label->Name); } } } for(Wire *pw = Sch->DocWires.first(); pw != 0; pw = Sch->DocWires.next()) { if(pw->Label != 0) { if (!vars.contains(pw->Label->Name)) { vars.append(pw->Label->Name); } } } for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) { if (pc->isProbe) { QString var_pr = pc->getProbeVariable(); if (!vars.contains(var_pr)) { vars.append(var_pr); } } } vars.sort(); stream<<".control\n" //execute simulations <<"echo \"\" > spice4qucs.cir.noise\n" <<"echo \"\" > spice4qucs.cir.pz\n"; QString sim; outputs.clear(); foreach(sim, simulations) { bool hasParSWP = false; bool hasDblSWP = false; QString cnt_var; // Duplicate .PARAM in .control section. They may be used in euqations for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) { if (pc->Model=="Eqn") { Equation *eq = (Equation *)pc; stream<getNgspiceScript(); } } // Hack: Such indexation method is needed to avoid entering infinite loop // in some cases of recursive access to Sch->DocComps list for(unsigned int i=0;iDocComps.count();i++) { Component *pc = Sch->DocComps.at(i); QString sim_typ = pc->Model; if (sim_typ==".SW") { QString SwpSim = pc->Props.at(0)->Value; QString s = pc->getNgspiceBeforeSim(sim); cnt_var = ((Param_Sweep *)pc)->getCounterVar(); if (SwpSim.startsWith("AC")&&(sim=="ac")) { QString s2 = getParentSWPscript(pc,sim,true,hasDblSWP); stream<<(s2+s); hasParSWP = true; } else if (SwpSim.startsWith("SP")&&(sim=="sp")) { QString s2 = getParentSWPscript(pc,sim,true,hasDblSWP); stream<<(s2+s); hasParSWP = true; } else if (SwpSim.startsWith("DISTO")&&(sim=="disto")) { QString s2 = getParentSWPscript(pc,sim,true,hasDblSWP); stream<<(s2+s); hasParSWP = true; } else if (SwpSim.startsWith("NOISE")&&(sim=="noise")) { QString s2 = getParentSWPscript(pc,sim,true,hasDblSWP); stream<<(s2+s); hasParSWP = true; } else if (SwpSim.startsWith("PZ")&&(sim=="pz")) { QString s2 = getParentSWPscript(pc,sim,true,hasDblSWP); stream<<(s2+s); hasParSWP = true; } else if (SwpSim.startsWith("FFT")&&(sim=="fft")) { QString s2 = getParentSWPscript(pc,sim,true,hasDblSWP); stream<<(s2+s); hasParSWP = true; } if (SwpSim.startsWith("TR")&&(sim=="tran")) { QString s2 = getParentSWPscript(pc,sim,true,hasDblSWP); stream<<(s2+s); hasParSWP = true; } else if (SwpSim.startsWith("SW")&&(sim=="dc")) { for(Component *pc1 = Sch->DocComps.first(); pc1 != 0; pc1 = Sch->DocComps.next()) { if ((pc1->Name==SwpSim)&&(pc1->Props.at(0)->Value.startsWith("DC"))) { QString s2 = getParentSWPscript(pc,sim,true,hasDblSWP); stream<<(s2+s); hasParSWP = true; } } } } } QString custom_vars; QStringList ext_vars; // Hack: Such indexation method is needed to avoid entering infinite loop // in some cases of recursive access to Sch->DocComps list for(unsigned int i=0;iDocComps.count();i++) { Component *pc = Sch->DocComps.at(i); if(pc->isSimulation) { QString sim_typ = pc->Model; QString s = pc->getSpiceNetlist(); if ((sim_typ==".AC")&&(sim=="ac")) stream<getExtraVariables(); } if ((sim_typ==".FFT")&&(sim=="fft")) { stream<getExtraVariables(); } if ((sim_typ==".DISTO")&&(sim=="disto")) stream< comps(Sch->DocComps); // find Fourier tran for(Component *pc1 = comps.first(); pc1 != 0; pc1 = comps.next()) { if (pc1->Model==".FOURIER") { if (pc1->Props.at(0)->Value==pc->Name) { QString s1 = pc1->getSpiceNetlist(); outputs.append("spice4qucs.four"); stream<Props.at(1)->Value; custom_vars.replace(";"," "); QString cust_out = pc->Props.at(2)->Value; outputs.append(cust_out.split(';',qucs::SkipEmptyParts)); } if (sim_typ==".SW") { QString SwpSim = pc->Props.at(0)->Value; if (SwpSim.startsWith("DC")&&(sim=="dc")) stream<DocName); QString basenam = inf.baseName(); if (sim=="custom") { QString ss = custom_vars; if (!ss.remove(' ').isEmpty()) { // if there was no variables QString filename = basenam + "_custom.txt"; outputs.append(filename); QString write_str = QString("write %1 %2\n").arg(filename).arg(custom_vars); stream<DocComps.first(); pc != 0; pc = Sch->DocComps.next()) { if ((pc->Model == "Eqn")||(pc->Model== "NutmegEq")) { QStringList v1; QString s_eq = pc->getEquations(sim,v1); stream<< s_eq; Eqns.append(s_eq.split("\n")); vars_eq.append(v1); } } if (sim == "sp") { // S-parameter and FFT requires specific variables nods.clear(); for (const auto &var : ext_vars) { nods += " " + var; } } for (QStringList::iterator it = vars_eq.begin();it != vars_eq.end(); it++) { nods += " " + *it; } if (sim=="noise") { if (hasParSWP) { // Set necessary plot number to output Noise spectrum // each step of parameter sweep creates new couple of noise plots stream<DocComps.first(); pc != 0; pc = Sch->DocComps.next()) { QString sim_typ = pc->Model; if (sim_typ==".SW") { QString s = pc->getNgspiceAfterSim(sim); QString SwpSim = pc->Props.at(0)->Value; bool b; // value drain if (SwpSim.startsWith("AC")&&(sim=="ac")) { s += getParentSWPscript(pc,sim,false,b); stream<DocComps.first(); pc1 != 0; pc1 = Sch->DocComps.next()) { if ((pc1->Name==SwpSim)&&(pc1->Props.at(0)->Value.startsWith("DC"))) { s += getParentSWPscript(pc,sim,false,b); stream<Name; if (swp.startsWith("DC")) return QString(""); Q3PtrList Comps(Sch->DocComps); if (Comps.count()>0) { for (unsigned int i=0;iModel.startsWith(".SW")) { if (pc2->Props.at(0)->Value==swp) { if (before) { hasDblSwp = true; return pc2->getNgspiceBeforeSim(sim,1); } else { hasDblSwp = true; return pc2->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() { output.clear(); QString mathf_inc; // drain if (!findMathFuncInc(mathf_inc)) { output.append("[Warning!] " + mathf_inc + " file not found!\n"); } QStringList incompat; if (!checkSchematic(incompat)) { QString s = incompat.join("; "); output.append("There were SPICE-incompatible components. Simulator cannot proceed."); output.append("Incompatible components are: " + s + "\n"); emit finished(); emit errors(QProcess::FailedToStart); return; } if (!checkGround()) { output.append("No Ground found. Please add at least one ground!\n"); emit finished(); emit errors(QProcess::FailedToStart); return; } if (!checkSimulations()) { output.append("No simulation found. Please add at least one simulation!\n"); emit finished(); emit errors(QProcess::FailedToStart); return; } if (!checkDCSimulation()) { output.append("Only DC simulation found in the schematic. It has no effect!" " Add TRAN, AC, or Sweep simulation to proceed.\n"); emit finished(); emit errors(QProcess::FailedToStart); return; } if (!checkNodeNames(incompat)) { QString s = incompat.join("; "); output.append("There were Nutmeg-incompatible node names. Simulator cannot proceed.\n"); output.append("Incompatible node names are: " + s + "\n"); emit finished(); emit errors(QProcess::FailedToStart); return; } QString netfile = "spice4qucs.cir"; QString tmp_path = QDir::toNativeSeparators(workdir+QDir::separator()+netfile); SaveNetlist(tmp_path); removeAllSimulatorOutputs(); XSPICE_CMbuilder *CMbuilder = new XSPICE_CMbuilder(Sch); CMbuilder->cleanSpiceinit(); CMbuilder->createSpiceinit(/*initial_spiceinit=*/collectSpiceinit(Sch)); if (CMbuilder->needCompile()) { CMbuilder->cleanCModelTree(); CMbuilder->createCModelTree(output); CMbuilder->compileCMlib(output); } delete CMbuilder; //startNgSpice(tmp_path); SimProcess->setWorkingDirectory(workdir); qDebug()<start(ngsp_cmd,cmd_args); 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 = Sch->DocNodes.first(); pn != 0; pn = Sch->DocNodes.next()) { if(pn->Label != 0) { if (!spicecompat::check_nodename(pn->Label->Name)) { incompat.append(pn->Label->Name); result = false; } } } for(Wire *pw = Sch->DocWires.first(); pw != 0; pw = Sch->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->DocComps.first(); pc != 0; pc = sch->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 = QString("%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 = SimProcess->readAllStandardOutput(); QRegExp percentage_pattern("^%\\d\\d*\\.\\d\\d.*$"); if (percentage_pattern.exactMatch(s)) { int percent = round(s.mid(1,5).toFloat()); emit progress(percent); } else { s.remove(QRegExp("%\\d\\d*\\.\\d\\d")); // Remove percentage from logs s.remove(QRegExp("\010+")); // Large amount of datar from percentage reports // can freeze QTextEdit for over 100k simulation points output += s; } } /*! * \brief Ngspice::SaveNetlist Create netlist and save it to file without execution * of simulator. * \param[in] filename Absolute path to netlist */ void Ngspice::SaveNetlist(QString filename) { int num=0; sims.clear(); vars.clear(); QFile spice_file(filename); if (spice_file.open(QFile::WriteOnly)) { QTextStream stream(&spice_file); createNetlist(stream,num,sims,vars,output_files); spice_file.close(); } else { QString msg=QString("Tried to save netlist \nin %1\n(could not open for writing!)").arg(filename); QString final_msg=QString("%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); } } void Ngspice::setSimulatorCmd(QString cmd) { if (cmd.contains(QRegExp("spiceopus(....|)$"))) { // spiceopus needs English locale to produce correct decimal point (dot symbol) QProcessEnvironment env = QProcessEnvironment::systemEnvironment(); env.remove("LANG"); env.insert("LANG","en_US"); SimProcess->setProcessEnvironment(env); simulator_parameters = simulator_parameters + "-c"; } else { // restore system environment QProcessEnvironment env = QProcessEnvironment::systemEnvironment(); SimProcess->setProcessEnvironment(env); } simulator_cmd = cmd; } void Ngspice::setSimulatorParameters(QString parameters) { simulator_parameters = parameters; }