2015-01-10 17:26:25 +03:00
/***************************************************************************
ngspice . cpp
- - - - - - - - - - - - - - - -
2015-01-25 10:18:06 +03:00
begin : Sat Jan 10 2015
copyright : ( C ) 2015 by Vadim Kuznetsov
2015-01-10 17:26:25 +03:00
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"
2016-01-31 11:28:16 +03:00
# include "xspice_cmbuilder.h"
2015-02-15 18:52:59 +03:00
# include "components/iprobe.h"
# include "components/vprobe.h"
2015-02-19 18:02:47 +03:00
# include "components/equation.h"
2015-03-23 16:12:08 +03:00
# include "components/param_sweep.h"
2019-06-21 21:58:54 -04:00
# include "components/subcircuit.h"
2019-06-21 15:38:33 -04:00
# include "spicecomponents/sp_spiceinit.h"
2016-01-23 11:52:50 +03:00
# include "spicecomponents/xsp_cmlib.h"
2015-04-20 16:37:44 +03:00
# include "main.h"
2022-02-25 23:59:27 +01:00
# include "misc.h"
2017-02-03 18:43:50 +03:00
# ifdef HAVE_CONFIG_H
# include "config.h"
# endif
2015-01-10 17:26:25 +03:00
2015-04-25 13:31:40 +03:00
/*!
\ file ngspice . cpp
\ brief Implementation of the Ngspice class
*/
2015-04-24 13:37:56 +03:00
/*!
* \ brief Ngspice : : Ngspice Class constructor
2022-07-05 07:08:28 -04:00
* \ param sch_ Schematic that need to be simulated with Ngspice .
2015-04-24 13:37:56 +03:00
* \ param parent Parent object
*/
2015-01-10 17:26:25 +03:00
Ngspice : : Ngspice ( Schematic * sch_ , QObject * parent ) :
2015-01-17 12:55:26 +03:00
AbstractSpiceKernel ( sch_ , parent )
2015-01-10 17:26:25 +03:00
{
2015-04-20 16:37:44 +03:00
simulator_cmd = QucsSettings . NgspiceExecutable ;
2015-01-17 12:55:26 +03:00
simulator_parameters = " " ;
2015-01-10 17:26:25 +03:00
}
2015-04-24 13:37:56 +03:00
/*!
* \ 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 .
*/
2015-04-18 14:12:23 +03:00
void Ngspice : : createNetlist ( QTextStream & stream , int ,
2015-01-10 17:26:25 +03:00
QStringList & simulations , QStringList & vars , QStringList & outputs )
{
2017-02-03 18:43:50 +03:00
// include math. functions for inter-simulator compat.
2017-05-01 15:54:01 +03:00
QString mathf_inc ;
bool found = findMathFuncInc ( mathf_inc ) ;
2017-02-03 18:43:50 +03:00
stream < < QString ( " * Qucs %1 %2 \n " ) . arg ( PACKAGE_VERSION ) . arg ( Sch - > DocName ) ;
2017-05-01 15:54:01 +03:00
// Let to simulate schematic without mathfunc.inc file
2018-01-27 22:40:32 +03:00
if ( found & & QucsSettings . DefaultSimulator ! = spicecompat : : simSpiceOpus )
stream < < QString ( " .INCLUDE \" %1 \" \n " ) . arg ( mathf_inc ) ;
2017-02-03 18:43:50 +03:00
2015-01-10 17:26:25 +03:00
QString s ;
2015-03-01 12:15:22 +03:00
if ( ! prepareSpiceNetlist ( stream ) ) return ; // Unable to perform spice simulation
startNetlist ( stream ) ; // output .PARAM and components
2016-02-23 15:30:36 +03:00
if ( DC_OP_only ) {
stream < < " .control \n " // Execute only DC OP analysis
2022-07-05 07:08:28 -04:00
< < " set filetype=ascii \n " // Ignore all other simulations
2016-02-23 15:30:36 +03:00
< < " op \n "
< < " print all > spice4qucs.cir.dc_op \n "
2016-05-09 15:19:07 +03:00
< < " destroy all \n "
2016-02-23 15:30:36 +03:00
< < " quit \n "
2016-05-09 15:19:07 +03:00
< < " .endc \n "
< < " .end \n " ;
2016-02-23 15:30:36 +03:00
outputs . clear ( ) ;
outputs . append ( " spice4qucs.cir.dc_op " ) ;
return ;
}
2015-01-10 17:26:25 +03:00
// 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 " ) ;
2015-04-09 19:45:31 +03:00
if ( sim_typ = = " .CUSTOMSIM " ) simulations . append ( " custom " ) ;
2015-05-20 12:21:58 +03:00
if ( sim_typ = = " .DISTO " ) simulations . append ( " disto " ) ;
2015-05-28 15:20:27 +03:00
if ( sim_typ = = " .NOISE " ) simulations . append ( " noise " ) ;
2015-11-06 10:51:49 +03:00
if ( sim_typ = = " .PZ " ) simulations . append ( " pz " ) ;
2017-09-11 16:40:02 +03:00
if ( sim_typ = = " .SENS " ) simulations . append ( " sens " ) ;
2017-09-16 15:53:42 +03:00
if ( sim_typ = = " .SENS_AC " ) simulations . append ( " sens_ac " ) ;
2022-04-02 19:39:55 +02:00
if ( sim_typ = = " .SP " ) simulations . append ( " sp " ) ;
2022-05-11 21:39:10 +03:00
if ( sim_typ = = " .FFT " ) simulations . append ( " fft " ) ;
2015-02-23 17:51:10 +03:00
if ( ( sim_typ = = " .SW " ) & &
( pc - > Props . at ( 0 ) - > Value . startsWith ( " DC " ) ) ) simulations . append ( " dc " ) ;
2015-04-03 14:04:48 +03:00
// stream<<s;
2015-01-10 17:26:25 +03:00
}
}
// set variable names for named nodes and wires
vars . clear ( ) ;
for ( Node * pn = Sch - > 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 ) ;
}
}
}
2015-02-15 18:52:59 +03:00
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 ) ;
}
}
}
2015-01-10 17:26:25 +03:00
vars . sort ( ) ;
stream < < " .control \n " //execute simulations
2015-11-06 10:51:49 +03:00
< < " echo \" \" > spice4qucs.cir.noise \n "
< < " echo \" \" > spice4qucs.cir.pz \n " ;
2015-03-23 16:12:08 +03:00
2015-04-03 15:46:24 +03:00
QString sim ;
2015-04-03 14:04:48 +03:00
outputs . clear ( ) ;
foreach ( sim , simulations ) {
bool hasParSWP = false ;
2015-05-15 17:02:07 +03:00
bool hasDblSWP = false ;
2016-06-18 10:48:19 +03:00
QString cnt_var ;
2015-04-03 14:04:48 +03:00
2015-06-18 17:08:37 +03:00
// 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 < < eq - > getNgspiceScript ( ) ;
}
}
2016-06-30 09:44:23 +03:00
// 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 ; i < Sch - > DocComps . count ( ) ; i + + ) {
Component * pc = Sch - > DocComps . at ( i ) ;
2015-04-03 14:04:48 +03:00
QString sim_typ = pc - > Model ;
if ( sim_typ = = " .SW " ) {
QString SwpSim = pc - > Props . at ( 0 ) - > Value ;
2015-05-15 17:02:07 +03:00
QString s = pc - > getNgspiceBeforeSim ( sim ) ;
2016-06-18 10:48:19 +03:00
cnt_var = ( ( Param_Sweep * ) pc ) - > getCounterVar ( ) ;
2015-04-03 14:04:48 +03:00
if ( SwpSim . startsWith ( " AC " ) & & ( sim = = " ac " ) ) {
2015-05-15 17:02:07 +03:00
QString s2 = getParentSWPscript ( pc , sim , true , hasDblSWP ) ;
stream < < ( s2 + s ) ;
2015-04-03 14:04:48 +03:00
hasParSWP = true ;
2022-04-02 19:39:55 +02:00
} else if ( SwpSim . startsWith ( " SP " ) & & ( sim = = " sp " ) ) {
QString s2 = getParentSWPscript ( pc , sim , true , hasDblSWP ) ;
stream < < ( s2 + s ) ;
hasParSWP = true ;
2015-05-29 15:56:22 +03:00
} 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 ;
2015-11-08 14:10:56 +03:00
} else if ( SwpSim . startsWith ( " PZ " ) & & ( sim = = " pz " ) ) {
QString s2 = getParentSWPscript ( pc , sim , true , hasDblSWP ) ;
stream < < ( s2 + s ) ;
hasParSWP = true ;
2022-05-11 21:39:10 +03:00
} else if ( SwpSim . startsWith ( " FFT " ) & & ( sim = = " fft " ) ) {
QString s2 = getParentSWPscript ( pc , sim , true , hasDblSWP ) ;
stream < < ( s2 + s ) ;
hasParSWP = true ;
2015-11-08 14:10:56 +03:00
} if ( SwpSim . startsWith ( " TR " ) & & ( sim = = " tran " ) ) {
2015-05-15 17:02:07 +03:00
QString s2 = getParentSWPscript ( pc , sim , true , hasDblSWP ) ;
stream < < ( s2 + s ) ;
2015-04-03 14:04:48 +03:00
hasParSWP = true ;
2015-04-05 15:08:06 +03:00
} 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 " ) ) ) {
2015-05-15 17:02:07 +03:00
QString s2 = getParentSWPscript ( pc , sim , true , hasDblSWP ) ;
stream < < ( s2 + s ) ;
2015-04-05 15:08:06 +03:00
hasParSWP = true ;
}
}
2015-04-03 14:04:48 +03:00
}
2015-03-23 16:12:08 +03:00
}
}
2015-01-10 17:26:25 +03:00
2015-04-09 19:45:31 +03:00
QString custom_vars ;
2022-05-11 21:39:10 +03:00
QStringList ext_vars ;
2016-06-30 09:44:23 +03:00
// 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 ; i < Sch - > DocComps . count ( ) ; i + + ) {
Component * pc = Sch - > DocComps . at ( i ) ;
2015-04-03 14:04:48 +03:00
if ( pc - > isSimulation ) {
QString sim_typ = pc - > Model ;
QString s = pc - > getSpiceNetlist ( ) ;
if ( ( sim_typ = = " .AC " ) & & ( sim = = " ac " ) ) stream < < s ;
2022-04-03 11:54:42 +02:00
if ( ( sim_typ = = " .SP " ) & & ( sim = = " sp " ) ) {
stream < < s ;
2022-05-11 21:39:10 +03:00
ext_vars = pc - > getExtraVariables ( ) ;
}
if ( ( sim_typ = = " .FFT " ) & & ( sim = = " fft " ) ) {
stream < < s ;
//ext_vars = pc->getExtraVariables();
2022-04-03 11:54:42 +02:00
}
2015-05-20 12:21:58 +03:00
if ( ( sim_typ = = " .DISTO " ) & & ( sim = = " disto " ) ) stream < < s ;
2015-05-28 15:20:27 +03:00
if ( ( sim_typ = = " .NOISE " ) & & ( sim = = " noise " ) ) {
outputs . append ( " spice4qucs.cir.noise " ) ;
stream < < s ;
2015-11-06 10:51:49 +03:00
} if ( ( sim_typ = = " .PZ " ) & & ( sim = = " pz " ) ) {
2015-11-06 11:44:58 +03:00
outputs . append ( " spice4qucs.cir.pz " ) ; // Add it twice for poles and zeros
2015-11-06 10:51:49 +03:00
outputs . append ( " spice4qucs.cir.pz " ) ;
stream < < s ;
2017-09-11 16:40:02 +03:00
} else if ( ( sim_typ = = " .SENS " ) & & ( sim = = " sens " ) ) {
2017-09-12 17:56:08 +03:00
outputs . append ( " spice4qucs.ngspice.sens.dc.prn " ) ;
2017-09-11 16:40:02 +03:00
stream < < s ;
2017-09-16 15:53:42 +03:00
} else if ( ( sim_typ = = " .SENS_AC " ) & & ( sim = = " sens_ac " ) ) {
outputs . append ( " spice4qucs.sens.prn " ) ;
stream < < s ;
2015-05-28 15:20:27 +03:00
} if ( ( sim_typ = = " .TR " ) & & ( sim = = " tran " ) ) {
2015-05-18 11:39:58 +03:00
stream < < s ;
Q3PtrList < Component > 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 < < s1 ;
}
}
}
}
2015-04-09 19:45:31 +03:00
if ( ( sim_typ = = " .CUSTOMSIM " ) & & ( sim = = " custom " ) ) {
stream < < s ;
custom_vars = pc - > Props . at ( 1 ) - > Value ;
custom_vars . replace ( " ; " , " " ) ;
2015-05-19 15:19:50 +03:00
QString cust_out = pc - > Props . at ( 2 ) - > Value ;
2022-02-25 23:59:27 +01:00
outputs . append ( cust_out . split ( ' ; ' , qucs : : SkipEmptyParts ) ) ;
2015-04-09 19:45:31 +03:00
}
2015-04-03 14:04:48 +03:00
if ( sim_typ = = " .SW " ) {
QString SwpSim = pc - > Props . at ( 0 ) - > Value ;
if ( SwpSim . startsWith ( " DC " ) & & ( sim = = " dc " ) ) stream < < s ;
}
}
2015-02-19 18:02:47 +03:00
}
2015-04-09 19:45:31 +03:00
QFileInfo inf ( Sch - > DocName ) ;
QString basenam = inf . baseName ( ) ;
if ( sim = = " custom " ) {
2015-05-19 09:37:33 +03:00
QString ss = custom_vars ;
if ( ! ss . remove ( ' ' ) . isEmpty ( ) ) { // if there was no variables
2015-05-19 15:19:50 +03:00
QString filename = basenam + " _custom.txt " ;
outputs . append ( filename ) ;
2015-05-19 09:37:33 +03:00
QString write_str = QString ( " write %1 %2 \n " ) . arg ( filename ) . arg ( custom_vars ) ;
stream < < write_str ;
2016-06-10 11:58:53 +03:00
stream < < " destroy all \n " ;
stream < < " reset \n " ;
2015-05-19 09:37:33 +03:00
}
2015-04-09 19:45:31 +03:00
continue ;
}
2022-05-11 21:39:10 +03:00
QString nod ;
QString nods ;
nods . clear ( ) ;
foreach ( nod , vars ) {
if ( ! nod . endsWith ( " #branch " ) ) {
nods + = QString ( " v(%1) " ) . arg ( nod ) ;
} else {
nods + = QString ( " %1 " ) . arg ( nod ) ;
}
}
if ( sim = = " fft " ) {
stream < < QString ( " linearize %1 \n " ) . arg ( nods ) ;
stream < < QString ( " fft %1 \n " ) . arg ( nods ) ;
}
2015-04-09 19:45:31 +03:00
2015-04-03 14:04:48 +03:00
QStringList vars_eq ;
2015-04-03 15:46:24 +03:00
vars_eq . clear ( ) ;
2015-04-03 14:04:48 +03:00
QStringList Eqns ;
Eqns . clear ( ) ;
for ( Component * pc = Sch - > DocComps . first ( ) ; pc ! = 0 ; pc = Sch - > DocComps . next ( ) ) {
2015-05-22 10:35:36 +03:00
if ( ( pc - > Model = = " Eqn " ) | | ( pc - > Model = = " NutmegEq " ) ) {
2015-04-03 15:46:24 +03:00
QStringList v1 ;
2015-05-22 10:35:36 +03:00
QString s_eq = pc - > getEquations ( sim , v1 ) ;
2015-04-03 14:04:48 +03:00
stream < < s_eq ;
Eqns . append ( s_eq . split ( " \n " ) ) ;
vars_eq . append ( v1 ) ;
}
}
2022-05-11 21:39:10 +03:00
if ( sim = = " sp " ) { // S-parameter and FFT requires specific variables
2022-04-02 19:39:55 +02:00
nods . clear ( ) ;
2022-05-11 21:39:10 +03:00
for ( const auto & var : ext_vars ) {
2022-04-03 11:54:42 +02:00
nods + = " " + var ;
2015-02-15 18:52:59 +03:00
}
2015-01-10 17:26:25 +03:00
}
2015-04-03 15:46:24 +03:00
for ( QStringList : : iterator it = vars_eq . begin ( ) ; it ! = vars_eq . end ( ) ; it + + ) {
nods + = " " + * it ;
2015-02-19 18:02:47 +03:00
}
2015-03-23 16:12:08 +03:00
2016-06-17 11:28:53 +03:00
if ( sim = = " noise " ) {
2016-06-18 10:48:19 +03:00
if ( hasParSWP ) { // Set necessary plot number to output Noise spectrum
// each step of parameter sweep creates new couple of noise plots
stream < < QString ( " let noise_%1 = 2*%1+1 \n " ) . arg ( cnt_var ) ;
stream < < QString ( " setplot noise$&noise_%1 \n " ) . arg ( cnt_var ) ;
} else { // Set Noise1 plot to output noise spectrum
stream < < " setplot noise1 \n " ;
}
2016-06-17 11:28:53 +03:00
nods = " inoise_spectrum onoise_spectrum " ;
}
2017-09-16 15:53:42 +03:00
if ( ( sim ! = " pz " ) & & ( sim ! = " sens " ) & & ( sim ! = " sens_ac " ) ) {
2015-05-28 15:20:27 +03:00
QString filename ;
if ( hasParSWP & & hasDblSWP ) filename = QString ( " %1_%2_swp_swp.txt " ) . arg ( basenam ) . arg ( sim ) ;
else if ( hasParSWP ) filename = QString ( " %1_%2_swp.txt " ) . arg ( basenam ) . arg ( sim ) ;
else filename = QString ( " %1_%2.txt " ) . arg ( basenam ) . arg ( sim ) ;
2015-09-03 15:46:13 +03:00
filename . replace ( ' ' , ' _ ' ) ; // Ngspice cannot understand spaces in filename
2015-05-28 15:20:27 +03:00
QString write_str = QString ( " write %1 %2 \n " ) . arg ( filename ) . arg ( nods ) ;
stream < < write_str ;
outputs . append ( filename ) ;
}
2015-03-23 16:12:08 +03:00
2015-01-10 17:26:25 +03:00
2015-04-03 14:04:48 +03:00
for ( Component * pc = Sch - > DocComps . first ( ) ; pc ! = 0 ; pc = Sch - > DocComps . next ( ) ) {
QString sim_typ = pc - > Model ;
if ( sim_typ = = " .SW " ) {
2015-05-15 17:02:07 +03:00
QString s = pc - > getNgspiceAfterSim ( sim ) ;
2015-04-03 14:04:48 +03:00
QString SwpSim = pc - > Props . at ( 0 ) - > Value ;
2015-05-15 17:02:07 +03:00
bool b ; // value drain
if ( SwpSim . startsWith ( " AC " ) & & ( sim = = " ac " ) ) {
s + = getParentSWPscript ( pc , sim , false , b ) ;
stream < < s ;
2022-04-02 19:39:55 +02:00
} else if ( SwpSim . startsWith ( " SP " ) & & ( sim = = " sp " ) ) {
s + = getParentSWPscript ( pc , sim , false , b ) ;
stream < < s ;
2015-05-29 15:56:22 +03:00
} else if ( SwpSim . startsWith ( " DISTO " ) & & ( sim = = " disto " ) ) {
s + = getParentSWPscript ( pc , sim , false , b ) ;
stream < < s ;
} else if ( SwpSim . startsWith ( " NOISE " ) & & ( sim = = " noise " ) ) {
s + = getParentSWPscript ( pc , sim , false , b ) ;
stream < < s ;
2015-11-08 14:10:56 +03:00
} else if ( SwpSim . startsWith ( " PZ " ) & & ( sim = = " pz " ) ) {
s + = getParentSWPscript ( pc , sim , false , b ) ;
stream < < s ;
2022-05-11 21:39:10 +03:00
} else if ( SwpSim . startsWith ( " FFT " ) & & ( sim = = " fft " ) ) {
s + = getParentSWPscript ( pc , sim , false , b ) ;
stream < < s ;
2015-05-15 17:02:07 +03:00
} else if ( SwpSim . startsWith ( " TR " ) & & ( sim = = " tran " ) ) {
s + = getParentSWPscript ( pc , sim , false , b ) ;
stream < < s ;
} else if ( SwpSim . startsWith ( " SW " ) & & ( sim = = " dc " ) ) {
2015-04-05 15:08:06 +03:00
for ( Component * pc1 = Sch - > DocComps . first ( ) ; pc1 ! = 0 ; pc1 = Sch - > DocComps . next ( ) ) {
if ( ( pc1 - > Name = = SwpSim ) & & ( pc1 - > Props . at ( 0 ) - > Value . startsWith ( " DC " ) ) ) {
2015-05-15 17:02:07 +03:00
s + = getParentSWPscript ( pc , sim , false , b ) ;
2015-04-05 15:08:06 +03:00
stream < < s ;
}
}
}
2015-03-23 16:12:08 +03:00
}
}
2015-04-03 18:39:01 +03:00
stream < < " destroy all \n " ;
2015-05-16 14:15:04 +03:00
stream < < " reset \n \n " ;
2015-03-23 16:12:08 +03:00
}
2015-01-10 17:26:25 +03:00
stream < < " exit \n "
< < " .endc \n " ;
stream < < " .END \n " ;
}
2015-01-13 19:24:05 +03:00
2015-05-15 17:02:07 +03:00
/*!
* \ 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 ;
if ( swp . startsWith ( " DC " ) ) return QString ( " " ) ;
Q3PtrList < Component > Comps ( Sch - > DocComps ) ;
if ( Comps . count ( ) > 0 ) {
for ( unsigned int i = 0 ; i < Comps . count ( ) ; i + + ) {
Component * pc2 = Comps . at ( i ) ;
if ( pc2 - > Model . 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 ( " " ) ;
}
2015-04-24 13:37:56 +03:00
/*!
2022-07-05 07:08:28 -04:00
* \ brief Ngspice : : slotSimulate Create netlist and execute Ngspice simulator . Netlist
2015-04-24 13:37:56 +03:00
* is saved at $ HOME / . qucs / spice4qucs / spice4qucs . cir
*/
2015-01-13 19:24:05 +03:00
void Ngspice : : slotSimulate ( )
2015-04-21 15:38:38 +03:00
{
2015-10-06 11:21:06 +03:00
output . clear ( ) ;
2017-05-01 15:54:01 +03:00
QString mathf_inc ; // drain
if ( ! findMathFuncInc ( mathf_inc ) ) {
output . append ( " [Warning!] " + mathf_inc + " file not found! \n " ) ;
}
2015-07-13 10:40:05 +03:00
QStringList incompat ;
if ( ! checkSchematic ( incompat ) ) {
QString s = incompat . join ( " ; " ) ;
output . append ( " There were SPICE-incompatible components. Simulator cannot proceed. " ) ;
2015-07-26 17:17:38 +03:00
output . append ( " Incompatible components are: " + s + " \n " ) ;
2015-07-13 10:40:05 +03:00
emit finished ( ) ;
emit errors ( QProcess : : FailedToStart ) ;
return ;
}
2017-04-30 11:04:28 +03:00
if ( ! checkGround ( ) ) {
output . append ( " No Ground found. Please add at least one ground! \n " ) ;
emit finished ( ) ;
emit errors ( QProcess : : FailedToStart ) ;
return ;
}
2022-03-13 16:29:31 +01:00
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 ;
}
2016-07-15 13:47:54 +03:00
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 ;
}
2016-04-07 10:24:44 +03:00
QString netfile = " spice4qucs.cir " ;
2022-02-14 00:04:11 +01:00
QString tmp_path = QDir : : toNativeSeparators ( workdir + QDir : : separator ( ) + netfile ) ;
2015-04-21 15:38:38 +03:00
SaveNetlist ( tmp_path ) ;
2016-01-31 11:28:16 +03:00
2016-02-25 14:17:42 +03:00
removeAllSimulatorOutputs ( ) ;
2016-01-31 11:28:16 +03:00
XSPICE_CMbuilder * CMbuilder = new XSPICE_CMbuilder ( Sch ) ;
2016-01-31 11:55:06 +03:00
CMbuilder - > cleanSpiceinit ( ) ;
2019-06-21 21:58:54 -04:00
CMbuilder - > createSpiceinit ( /*initial_spiceinit=*/ collectSpiceinit ( Sch ) ) ;
2016-01-31 11:28:16 +03:00
if ( CMbuilder - > needCompile ( ) ) {
CMbuilder - > cleanCModelTree ( ) ;
2016-02-02 12:41:23 +03:00
CMbuilder - > createCModelTree ( output ) ;
2016-01-31 11:28:16 +03:00
CMbuilder - > compileCMlib ( output ) ;
2016-01-30 15:17:13 +03:00
}
2016-01-31 11:28:16 +03:00
delete CMbuilder ;
2015-04-21 15:38:38 +03:00
//startNgSpice(tmp_path);
SimProcess - > setWorkingDirectory ( workdir ) ;
2016-04-08 15:48:26 +03:00
qDebug ( ) < < workdir ;
2016-04-07 10:24:44 +03:00
QString cmd = QString ( " \" %1 \" %2 %3 " ) . arg ( simulator_cmd , simulator_parameters , netfile ) ;
2022-03-08 00:15:37 +01:00
QStringList cmd_args = misc : : parseCmdArgs ( cmd ) ;
QString ngsp_cmd = cmd_args . at ( 0 ) ;
cmd_args . removeAt ( 0 ) ;
SimProcess - > start ( ngsp_cmd , cmd_args ) ;
2015-04-21 15:38:38 +03:00
emit started ( ) ;
}
2016-07-15 13:47:54 +03:00
/*!
* \ 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 ;
}
2019-06-21 15:38:33 -04:00
/*!
* \ brief Ngspice : : collectSpiceinit Collects user - specified . spiceinit data .
* \ param incompat
* \ return
*/
2019-06-21 21:58:54 -04:00
QString Ngspice : : collectSpiceinit ( Schematic * sch )
2019-06-21 15:38:33 -04:00
{
QStringList collected_spiceinit ;
2019-06-21 21:58:54 -04:00
for ( Component * pc = sch - > DocComps . first ( ) ; pc ! = 0 ; pc = sch - > DocComps . next ( ) ) {
2019-06-21 15:38:33 -04:00
if ( pc - > Model = = " SPICEINIT " ) {
collected_spiceinit + = ( ( SpiceSpiceinit * ) pc ) - > getSpiceinit ( ) ;
2019-06-21 21:58:54 -04:00
} 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 ;
}
2019-06-21 15:38:33 -04:00
}
return collected_spiceinit . join ( " " ) ;
}
2017-05-01 15:54:01 +03:00
/*!
* \ 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 ( ) ;
2017-11-01 09:01:55 +03:00
mathf_inc = QString ( " %1/share/ " QUCS_NAME " /xspice_cmlib/include/ngspice_mathfunc.inc " )
2017-05-01 15:54:01 +03:00
. arg ( qucs_root . absolutePath ( ) ) ;
return QFile : : exists ( mathf_inc ) ;
}
2015-05-04 12:18:45 +03:00
/*!
2022-07-05 07:08:28 -04:00
* \ brief Ngspice : : slotProcessOutput Process Ngspice output and report completion
2015-05-04 12:18:45 +03:00
* percentage .
*/
void Ngspice : : slotProcessOutput ( )
{
QString s = SimProcess - > readAllStandardOutput ( ) ;
2016-04-22 09:59:58 +03:00
QRegExp percentage_pattern ( " ^% \\ d \\ d* \\ . \\ d \\ d.*$ " ) ;
2015-05-04 12:18:45 +03:00
if ( percentage_pattern . exactMatch ( s ) ) {
int percent = round ( s . mid ( 1 , 5 ) . toFloat ( ) ) ;
emit progress ( percent ) ;
2016-04-22 09:59:58 +03:00
} 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 ;
2015-05-04 12:18:45 +03:00
}
}
2015-04-24 13:37:56 +03:00
/*!
* \ brief Ngspice : : SaveNetlist Create netlist and save it to file without execution
* of simulator .
* \ param [ in ] filename Absolute path to netlist
*/
2015-04-21 15:38:38 +03:00
void Ngspice : : SaveNetlist ( QString filename )
2015-01-13 19:24:05 +03:00
{
int num = 0 ;
sims . clear ( ) ;
vars . clear ( ) ;
2015-04-21 15:38:38 +03:00
QFile spice_file ( filename ) ;
2015-01-13 19:24:05 +03:00
if ( spice_file . open ( QFile : : WriteOnly ) ) {
QTextStream stream ( & spice_file ) ;
createNetlist ( stream , num , sims , vars , output_files ) ;
spice_file . close ( ) ;
}
2022-04-02 23:20:14 -04:00
else
{
QString msg = QString ( " Tried to save netlist \n in %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) \n The value for S4Q_workdir (default:/spice4qucs) needs to be writeable! \n For 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 ) ;
}
2015-01-13 19:24:05 +03:00
}
2015-10-07 19:29:17 +03:00
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 ) ;
2020-02-05 20:39:17 +01:00
simulator_parameters = simulator_parameters + " -c " ;
2015-10-07 19:29:17 +03:00
} else { // restore system environment
QProcessEnvironment env = QProcessEnvironment : : systemEnvironment ( ) ;
SimProcess - > setProcessEnvironment ( env ) ;
}
simulator_cmd = cmd ;
}
2020-02-05 20:39:17 +01:00
void Ngspice : : setSimulatorParameters ( QString parameters )
{
simulator_parameters = parameters ;
}