2015-01-10 17:26:25 +03:00
|
|
|
/***************************************************************************
|
|
|
|
abstractspicekernel.h
|
|
|
|
----------------
|
|
|
|
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. *
|
|
|
|
* *
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include <config.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "abstractspicekernel.h"
|
2015-03-01 16:49:15 +03:00
|
|
|
#include "misc.h"
|
2015-07-13 10:18:26 +03:00
|
|
|
#include "main.h"
|
2015-03-01 16:49:15 +03:00
|
|
|
#include "../paintings/id_text.h"
|
2016-02-24 11:17:49 +03:00
|
|
|
#include "dialogs/sweepdialog.h"
|
2022-02-23 22:54:50 +01:00
|
|
|
|
|
|
|
|
2015-06-05 13:58:01 +03:00
|
|
|
#include <QPlainTextEdit>
|
2022-02-23 22:54:50 +01:00
|
|
|
#include <algorithm>
|
2015-01-10 17:26:25 +03:00
|
|
|
|
2015-04-25 13:31:40 +03:00
|
|
|
/*!
|
2015-05-18 14:39:00 +03:00
|
|
|
\file abstractspicekernel.cpp
|
|
|
|
\brief Implementation of the AbstractSpiceKernel class
|
2015-04-25 13:31:40 +03:00
|
|
|
*/
|
|
|
|
|
2015-01-10 17:26:25 +03:00
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::AbstractSpiceKernel class constructor
|
2022-07-05 07:08:28 -04:00
|
|
|
* \param sch_ Schematic that should be simulated with Spice-compatible
|
2015-04-24 13:37:56 +03:00
|
|
|
* simulator
|
|
|
|
* \param parent Parent object
|
|
|
|
*/
|
2015-01-17 12:55:26 +03:00
|
|
|
AbstractSpiceKernel::AbstractSpiceKernel(Schematic *sch_, QObject *parent) :
|
2015-01-10 17:26:25 +03:00
|
|
|
QObject(parent)
|
|
|
|
{
|
|
|
|
Sch = sch_;
|
|
|
|
|
2016-02-23 15:30:36 +03:00
|
|
|
if (Sch->showBias == 0) DC_OP_only = true;
|
|
|
|
else DC_OP_only = false;
|
|
|
|
|
2015-07-13 10:18:26 +03:00
|
|
|
workdir = QucsSettings.S4Qworkdir;
|
2015-01-10 17:26:25 +03:00
|
|
|
QFileInfo inf(workdir);
|
|
|
|
if (!inf.exists()) {
|
|
|
|
QDir dir;
|
|
|
|
dir.mkpath(workdir);
|
|
|
|
}
|
|
|
|
|
|
|
|
SimProcess = new QProcess(this);
|
|
|
|
SimProcess->setProcessChannelMode(QProcess::MergedChannels);
|
|
|
|
connect(SimProcess,SIGNAL(finished(int)),this,SLOT(slotFinished()));
|
2015-05-04 12:18:45 +03:00
|
|
|
connect(SimProcess,SIGNAL(readyRead()),this,SLOT(slotProcessOutput()));
|
2015-06-13 13:01:11 +03:00
|
|
|
connect(SimProcess,SIGNAL(error(QProcess::ProcessError)),this,SLOT(slotErrors(QProcess::ProcessError)));
|
2015-01-10 17:26:25 +03:00
|
|
|
connect(this,SIGNAL(destroyed()),this,SLOT(killThemAll()));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
AbstractSpiceKernel::~AbstractSpiceKernel()
|
|
|
|
{
|
|
|
|
killThemAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractSpiceKernel::killThemAll()
|
|
|
|
{
|
|
|
|
if (SimProcess->state()!=QProcess::NotRunning) {
|
|
|
|
SimProcess->kill();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::prepareSpiceNetlist Fill components nodes
|
|
|
|
* with approate node numbers
|
|
|
|
* \param stream QTextStream that associated with spice netlist file
|
|
|
|
* \param xyce Default is false. Should be set in true if netlist is
|
|
|
|
* prepared for Xyce simulator. For Ngspice should be false.
|
|
|
|
* \return Returns true if success, false if netlist preparation fails
|
|
|
|
*/
|
2016-10-09 10:08:26 +03:00
|
|
|
bool AbstractSpiceKernel::prepareSpiceNetlist(QTextStream &stream, bool isSubckt)
|
2015-01-31 10:22:13 +03:00
|
|
|
{
|
|
|
|
QStringList collect;
|
2015-06-05 13:58:01 +03:00
|
|
|
QPlainTextEdit *err = new QPlainTextEdit;
|
2016-01-08 15:53:29 +03:00
|
|
|
if (Sch->prepareNetlist(stream,collect,err)==-10) { // Broken netlist
|
2015-07-14 09:57:17 +03:00
|
|
|
output.append(err->toPlainText());
|
|
|
|
delete err;
|
|
|
|
return false;
|
|
|
|
}
|
2015-01-31 10:22:13 +03:00
|
|
|
delete err;
|
2016-10-09 10:08:26 +03:00
|
|
|
if (isSubckt) Sch->clearSignals();
|
|
|
|
else Sch->clearSignalsAndFileList(); // for proper build of subckts
|
2015-01-31 10:22:13 +03:00
|
|
|
return true; // TODO: Add feature to determine ability of spice simulation
|
|
|
|
}
|
|
|
|
|
2015-07-13 10:18:26 +03:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::checkSchematic Check SPICE-compatibility of
|
|
|
|
* all components.
|
|
|
|
* \param incompat[out] QStringList filled by incompatible components names
|
|
|
|
* \return true --- All components are SPICE-compatible; false --- otherwise
|
|
|
|
*/
|
|
|
|
bool AbstractSpiceKernel::checkSchematic(QStringList &incompat)
|
|
|
|
{
|
|
|
|
incompat.clear();
|
|
|
|
for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) {
|
|
|
|
if ((!pc->isEquation)&&!(pc->isProbe)) {
|
2015-07-26 17:17:38 +03:00
|
|
|
if (pc->SpiceModel.isEmpty() && pc->isActive) incompat.append(pc->Name);
|
2015-07-13 10:18:26 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return incompat.isEmpty();
|
|
|
|
}
|
|
|
|
|
2017-04-30 11:04:28 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::checkGround Check if schematic contain at least one ground.
|
|
|
|
* \return True if ground found, false otherwise
|
|
|
|
*/
|
|
|
|
bool AbstractSpiceKernel::checkGround()
|
|
|
|
{
|
|
|
|
bool r = false;
|
|
|
|
for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) {
|
|
|
|
if (pc->Model=="GND") {
|
|
|
|
r = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2022-03-13 16:29:31 +01:00
|
|
|
bool AbstractSpiceKernel::checkSimulations()
|
|
|
|
{
|
|
|
|
if (DC_OP_only) return true;
|
|
|
|
bool r = false;
|
|
|
|
for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) {
|
|
|
|
if (pc->isSimulation) {
|
|
|
|
r = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AbstractSpiceKernel::checkDCSimulation()
|
|
|
|
{
|
|
|
|
if (DC_OP_only) return true;
|
|
|
|
bool r = false;
|
|
|
|
for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) {
|
|
|
|
if (pc->isSimulation && pc->Model != ".DC") {
|
|
|
|
r = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::startNetlist Outputs .PARAM , .GLOABAL_PARAM,
|
|
|
|
* and .OPTIONS sections to netlist. These sections are placed on schematic
|
|
|
|
* directly or converted form Equation components. Then outputs common
|
|
|
|
* components to netlist.
|
|
|
|
* \param stream QTextStream that associated with spice netlist file
|
|
|
|
* \param xyce Default is false. Should be set in true if netlist is
|
|
|
|
* prepared for Xyce simulator. For Ngspice should be false.
|
|
|
|
*/
|
2015-03-01 12:15:22 +03:00
|
|
|
void AbstractSpiceKernel::startNetlist(QTextStream &stream, bool xyce)
|
|
|
|
{
|
|
|
|
QString s;
|
2015-12-07 17:33:47 +03:00
|
|
|
|
2017-03-03 14:19:31 +03:00
|
|
|
// User-defined functions
|
|
|
|
for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) {
|
2017-03-03 16:09:51 +03:00
|
|
|
if ((pc->SpiceModel==".FUNC")||
|
|
|
|
(pc->SpiceModel=="INCLSCR")) {
|
2017-03-03 14:19:31 +03:00
|
|
|
s = pc->getExpression();
|
|
|
|
stream<<s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-27 18:13:28 +03:00
|
|
|
QStringList incls;
|
2017-03-03 14:19:31 +03:00
|
|
|
// Include Directives
|
2015-12-07 17:33:47 +03:00
|
|
|
for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) {
|
2016-03-08 17:45:17 +03:00
|
|
|
if ((pc->SpiceModel==".INCLUDE")||
|
|
|
|
(pc->Model=="SpLib")) {
|
2015-12-07 17:33:47 +03:00
|
|
|
s = pc->getSpiceModel();
|
2016-08-27 18:13:28 +03:00
|
|
|
if (!incls.contains(s)) {
|
|
|
|
// prevent multiple libraries inclusion
|
|
|
|
incls.append(s);
|
|
|
|
stream<<s;
|
|
|
|
}
|
2015-12-07 17:33:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-01 18:11:28 +03:00
|
|
|
// Parameters, Initial conditions, Options
|
2015-03-01 12:15:22 +03:00
|
|
|
for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) {
|
|
|
|
if (pc->isEquation) {
|
|
|
|
s = pc->getExpression(xyce);
|
|
|
|
stream<<s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-01 18:11:28 +03:00
|
|
|
// Components
|
2015-03-01 12:15:22 +03:00
|
|
|
for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) {
|
|
|
|
if(Sch->isAnalog &&
|
|
|
|
!(pc->isSimulation) &&
|
|
|
|
!(pc->isEquation)) {
|
|
|
|
s = pc->getSpiceNetlist(xyce);
|
|
|
|
stream<<s;
|
|
|
|
}
|
|
|
|
}
|
2015-10-01 18:11:28 +03:00
|
|
|
|
|
|
|
// Modelcards
|
|
|
|
for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) {
|
2015-12-07 17:33:47 +03:00
|
|
|
if (pc->SpiceModel==".MODEL") {
|
|
|
|
s = pc->getSpiceModel();
|
|
|
|
stream<<s;
|
|
|
|
}
|
2015-10-01 18:11:28 +03:00
|
|
|
}
|
2015-03-01 12:15:22 +03:00
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::createNetlist Writes netlist in stream TextStream.
|
|
|
|
* This is overloaded method. Should be reimplemted for Ngspice and Xyce.
|
|
|
|
*/
|
2015-04-18 14:12:23 +03:00
|
|
|
void AbstractSpiceKernel::createNetlist(QTextStream&, int ,QStringList&,
|
|
|
|
QStringList&, QStringList &)
|
2015-01-10 17:26:25 +03:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::createSubNetlsit Output Netlist with
|
|
|
|
* Subcircuit header .SUBCKT
|
|
|
|
* \param stream QTextStream that associated with spice netlist file
|
|
|
|
* \param xyce Default is false. Should be set in true if netlist is
|
|
|
|
* prepared for Xyce simulator. For Ngspice should be false.
|
|
|
|
*/
|
2016-04-05 09:47:16 +03:00
|
|
|
void AbstractSpiceKernel::createSubNetlsit(QTextStream &stream, bool lib)
|
2015-03-01 12:15:22 +03:00
|
|
|
{
|
2015-03-01 16:49:15 +03:00
|
|
|
QString header;
|
2015-03-05 19:12:30 +03:00
|
|
|
QString f = misc::properFileName(Sch->DocName);
|
|
|
|
header = QString(".SUBCKT %1 ").arg(misc::properName(f));
|
2015-03-01 16:49:15 +03:00
|
|
|
|
|
|
|
QList< QPair<int,QString> > ports;
|
2016-10-09 10:08:26 +03:00
|
|
|
if(!prepareSpiceNetlist(stream,true)) {
|
2015-07-14 09:57:17 +03:00
|
|
|
emit finished();
|
|
|
|
emit errors(QProcess::FailedToStart);
|
|
|
|
return;
|
|
|
|
} // Unable to perform spice simulation
|
2015-03-01 12:15:22 +03:00
|
|
|
for(Component *pc = Sch->DocComps.first(); pc != 0; pc = Sch->DocComps.next()) {
|
|
|
|
if (pc->Model=="Port") {
|
2015-03-01 16:49:15 +03:00
|
|
|
ports.append(qMakePair(pc->Props.first()->Value.toInt(),
|
|
|
|
pc->Ports.first()->Connection->Name));
|
2015-03-01 12:15:22 +03:00
|
|
|
}
|
|
|
|
}
|
2022-02-23 22:54:50 +01:00
|
|
|
std::sort(ports.begin(), ports.end());
|
2015-03-01 16:49:15 +03:00
|
|
|
QPair<int,QString> pp;
|
2016-04-05 09:47:16 +03:00
|
|
|
if (lib) header += " gnd "; // Ground node forwarding for Library
|
2015-03-01 16:49:15 +03:00
|
|
|
foreach(pp,ports) {
|
|
|
|
header += pp.second + " ";
|
|
|
|
}
|
|
|
|
|
|
|
|
Painting *pi;
|
|
|
|
for(pi = Sch->SymbolPaints.first(); pi != 0; pi = Sch->SymbolPaints.next())
|
|
|
|
if(pi->Name == ".ID ") {
|
|
|
|
ID_Text *pid = (ID_Text*)pi;
|
|
|
|
QList<SubParameter *>::const_iterator it;
|
|
|
|
for(it = pid->Parameter.constBegin(); it != pid->Parameter.constEnd(); it++) {
|
|
|
|
header += (*it)->Name + " "; // keep 'Name' unchanged
|
|
|
|
//(*tstream) << " " << s.replace("=", "=\"") << '"';
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-03-01 12:15:22 +03:00
|
|
|
header += "\n";
|
2016-04-05 09:47:16 +03:00
|
|
|
if (lib) stream<<"\n";
|
2015-03-01 12:15:22 +03:00
|
|
|
stream<<header;
|
2016-04-05 09:26:11 +03:00
|
|
|
bool xyce = false;
|
|
|
|
if ((QucsSettings.DefaultSimulator == spicecompat::simXyceSer)||
|
|
|
|
(QucsSettings.DefaultSimulator == spicecompat::simXycePar)) xyce = true;
|
2015-03-01 12:15:22 +03:00
|
|
|
startNetlist(stream,xyce);
|
|
|
|
stream<<".ENDS\n";
|
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::slotSimulate Executes simulator
|
|
|
|
*/
|
2015-01-10 17:26:25 +03:00
|
|
|
void AbstractSpiceKernel::slotSimulate()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::parseNgSpiceSimOutput This method parses text raw spice
|
|
|
|
* output. Extracts a simulation points array and variables names and types (Real
|
|
|
|
* or Complex) from output.
|
|
|
|
* \param ngspice_file Spice output file name
|
|
|
|
* \param sim_points 2D array in which simulation points should be extracted
|
2022-07-05 07:08:28 -04:00
|
|
|
* \param var_list This list is filled by simulation variables. There is a list of dependent
|
|
|
|
* and independent variables. An independent variable is the first in list.
|
2015-04-24 13:37:56 +03:00
|
|
|
* \param isComplex Type of variables. True if complex. False if real.
|
|
|
|
*/
|
2015-01-10 17:26:25 +03:00
|
|
|
void AbstractSpiceKernel::parseNgSpiceSimOutput(QString ngspice_file,QList< QList<double> > &sim_points,QStringList &var_list, bool &isComplex)
|
|
|
|
{
|
|
|
|
isComplex = false;
|
2016-04-26 15:56:50 +03:00
|
|
|
bool isBinary = false;
|
|
|
|
int NumPoints = 0;
|
|
|
|
int bin_offset = 0;
|
|
|
|
QByteArray content;
|
2016-04-23 19:04:09 +03:00
|
|
|
|
2015-01-10 17:26:25 +03:00
|
|
|
QFile ofile(ngspice_file);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
2016-04-26 15:56:50 +03:00
|
|
|
//QTextStream ts(&ofile);
|
|
|
|
content = ofile.readAll();
|
2016-04-23 19:04:09 +03:00
|
|
|
ofile.close();
|
|
|
|
}
|
2015-01-10 17:26:25 +03:00
|
|
|
|
2016-04-23 19:04:09 +03:00
|
|
|
QTextStream ngsp_data(&content);
|
|
|
|
sim_points.clear();
|
|
|
|
bool start_values_sec = false;
|
|
|
|
int NumVars=0; // Number of dep. and indep.variables
|
2016-04-26 15:56:50 +03:00
|
|
|
while (!ngsp_data.atEnd()) { // Parse header;
|
2016-04-23 19:04:09 +03:00
|
|
|
QRegExp sep("[ \t,]");
|
|
|
|
QString lin = ngsp_data.readLine();
|
|
|
|
if (lin.isEmpty()) continue;
|
|
|
|
if (lin.contains("Flags")&&lin.contains("complex")) { // output consists of
|
|
|
|
isComplex = true; // complex numbers
|
|
|
|
continue; // maybe ac_analysis
|
|
|
|
}
|
|
|
|
if (lin.contains("No. Variables")) { // get number of variables
|
|
|
|
NumVars=lin.section(sep,2,2,QString::SectionSkipEmpty).toInt();
|
|
|
|
continue;
|
|
|
|
}
|
2016-04-26 15:56:50 +03:00
|
|
|
if (lin.contains("No. Points:")) { // get number of variables
|
|
|
|
NumPoints=lin.section(sep,2,2,QString::SectionSkipEmpty).toInt();
|
|
|
|
continue;
|
|
|
|
}
|
2016-04-23 19:04:09 +03:00
|
|
|
if (lin=="Variables:") {
|
|
|
|
var_list.clear();
|
|
|
|
QString indep_var = ngsp_data.readLine().section(sep,1,1,QString::SectionSkipEmpty);
|
|
|
|
var_list.append(indep_var);
|
|
|
|
|
|
|
|
for (int i=1;i<NumVars;i++) {
|
|
|
|
lin = ngsp_data.readLine();
|
|
|
|
QString dep_var = lin.section(sep,1,1,QString::SectionSkipEmpty);
|
|
|
|
var_list.append(dep_var);
|
2015-01-10 17:26:25 +03:00
|
|
|
}
|
2016-04-23 19:04:09 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (lin=="Values:") {
|
|
|
|
start_values_sec = true;
|
|
|
|
continue;
|
|
|
|
}
|
2016-04-26 15:56:50 +03:00
|
|
|
if (lin=="Binary:") {
|
|
|
|
isBinary = true;
|
|
|
|
bin_offset = ngsp_data.pos();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isBinary) {
|
|
|
|
QDataStream dbl(content);
|
|
|
|
dbl.setByteOrder(QDataStream::LittleEndian);
|
|
|
|
dbl.device()->seek(bin_offset);
|
2016-04-29 10:48:47 +03:00
|
|
|
extractBinSamples(dbl, sim_points, NumPoints, NumVars, isComplex);
|
2016-04-26 15:56:50 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-04-23 19:04:09 +03:00
|
|
|
if (start_values_sec) {
|
2016-04-29 10:48:47 +03:00
|
|
|
extractASCIISamples(lin,ngsp_data,sim_points,NumVars,isComplex);
|
2015-01-10 17:26:25 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::parseHBOutput Parse Xyce Harmonic balance (HB) simulation output.
|
|
|
|
* \param ngspice_file Spice output file name
|
|
|
|
* \param sim_points 2D array in which simulation points should be extracted
|
2022-07-05 07:08:28 -04:00
|
|
|
* \param var_list This list is filled by simulation variables. There is a list of dependent
|
|
|
|
* variables. Independent hbfrequency variable is always the first in this list.
|
2017-10-13 13:30:08 +03:00
|
|
|
* \param hasParSweep[out] Set to true if dataset contains parameter sweep output.
|
2015-04-24 13:37:56 +03:00
|
|
|
*/
|
2017-10-13 13:30:08 +03:00
|
|
|
void AbstractSpiceKernel::parseHBOutput(QString ngspice_file, QList<QList<double> > &sim_points,
|
|
|
|
QStringList &var_list, bool &hasParSweep)
|
2015-01-19 18:19:44 +03:00
|
|
|
{
|
2015-01-20 15:03:24 +03:00
|
|
|
var_list.clear();
|
|
|
|
sim_points.clear();
|
2017-10-13 13:30:08 +03:00
|
|
|
hasParSweep = false;
|
2015-01-20 19:21:44 +03:00
|
|
|
QFile ofile(ngspice_file);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream hb_data(&ofile);
|
|
|
|
var_list.append("hbfrequency");
|
|
|
|
while (!hb_data.atEnd()) {
|
|
|
|
QString lin = hb_data.readLine();
|
|
|
|
if (lin.isEmpty()) continue;
|
2017-10-13 13:30:08 +03:00
|
|
|
if (lin.contains("Parameter Sweep")) {
|
|
|
|
hasParSweep = true;
|
|
|
|
continue;
|
|
|
|
}
|
2015-01-20 19:21:44 +03:00
|
|
|
if (lin.startsWith("Index")) { // CSV heading
|
2022-02-24 23:45:28 +01:00
|
|
|
QStringList vars1 = lin.split(" ",qucs::SkipEmptyParts);
|
2015-01-20 19:21:44 +03:00
|
|
|
vars1.removeFirst();
|
|
|
|
vars1.removeFirst();
|
2015-12-13 10:20:02 +03:00
|
|
|
QStringList norm_vars;
|
|
|
|
foreach(QString v, vars1) { // Normalize variables
|
|
|
|
QString nv = v;
|
|
|
|
nv.remove(0,3).chop(1); // extract variable between "Re|Im(" and ")"
|
|
|
|
if (!norm_vars.contains(nv))
|
|
|
|
norm_vars.append(nv);
|
|
|
|
}
|
|
|
|
var_list.append(norm_vars);
|
2015-01-20 19:21:44 +03:00
|
|
|
}
|
2015-12-13 10:20:02 +03:00
|
|
|
if ((lin.contains(QRegExp("\\d*\\.\\d+[+-]*[eE]*[\\d]*")))) { // CSV dataline
|
2022-02-24 23:45:28 +01:00
|
|
|
QStringList vals = lin.split(" ",qucs::SkipEmptyParts);
|
2015-01-20 19:21:44 +03:00
|
|
|
QList <double> sim_point;
|
|
|
|
sim_point.clear();
|
2015-12-13 10:20:02 +03:00
|
|
|
for (int i=1;i<vals.count();i++) {
|
2015-01-20 19:21:44 +03:00
|
|
|
sim_point.append(vals.at(i).toDouble());
|
|
|
|
}
|
|
|
|
sim_points.append(sim_point);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ofile.close();
|
|
|
|
}
|
2015-01-19 18:19:44 +03:00
|
|
|
}
|
|
|
|
|
2015-05-18 11:39:58 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::parseFourierOutput Parse output of fourier simulation.
|
|
|
|
* \param ngspice_file[in] Spice output file name
|
|
|
|
* \param sim_points[out] 2D array in which simulation points should be extracted
|
2022-07-05 07:08:28 -04:00
|
|
|
* \param var_list[out] This list is filled by simulation variables. There is a list of dependent
|
|
|
|
* and independent variables. An independent variable is the first in list.
|
2015-05-18 11:39:58 +03:00
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::parseFourierOutput(QString ngspice_file, QList<QList<double> > &sim_points,
|
2016-08-21 14:09:45 +03:00
|
|
|
QStringList &var_list)
|
2015-05-18 11:39:58 +03:00
|
|
|
{
|
|
|
|
QFile ofile(ngspice_file);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream ngsp_data(&ofile);
|
|
|
|
sim_points.clear();
|
|
|
|
var_list.clear();
|
|
|
|
var_list.append("fourierfreq");
|
|
|
|
int Nharm; // number of harmonics
|
2015-05-18 14:24:16 +03:00
|
|
|
bool firstgroup = false;
|
2015-05-18 11:39:58 +03:00
|
|
|
while (!ngsp_data.atEnd()) {
|
|
|
|
QRegExp sep("[ \t,]");
|
|
|
|
QString lin = ngsp_data.readLine();
|
|
|
|
if (lin.isEmpty()) continue;
|
|
|
|
if (lin.contains("Fourier analysis for")) {
|
2022-02-24 23:45:28 +01:00
|
|
|
QStringList tokens = lin.split(sep,qucs::SkipEmptyParts);
|
2016-07-10 15:44:08 +03:00
|
|
|
QString var;
|
|
|
|
foreach(var,tokens) {
|
|
|
|
if (var.contains('(')&&var.contains(')')) break;
|
|
|
|
}
|
|
|
|
|
2015-05-18 11:39:58 +03:00
|
|
|
if (var.endsWith(':')) var.chop(1);
|
|
|
|
var_list.append("magnitude("+var+")");
|
|
|
|
var_list.append("phase("+var+")");
|
2015-05-19 15:29:21 +03:00
|
|
|
var_list.append("norm(mag("+var+"))");
|
|
|
|
var_list.append("norm(phase("+var+"))");
|
2015-05-18 11:39:58 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (lin.contains("No. Harmonics:")) {
|
|
|
|
QString ss = lin.section(sep,2,2,QString::SectionSkipEmpty);
|
|
|
|
if (ss.endsWith(',')) ss.chop(1);
|
|
|
|
Nharm = ss.toInt();
|
|
|
|
while (!ngsp_data.readLine().contains(QRegExp("Harmonic\\s+Frequency")));
|
2016-08-21 14:09:45 +03:00
|
|
|
if (!(QucsSettings.DefaultSimulator == spicecompat::simXyceSer||
|
|
|
|
QucsSettings.DefaultSimulator == spicecompat::simXycePar)) lin = ngsp_data.readLine(); // dummy line
|
2015-05-18 11:39:58 +03:00
|
|
|
for (int i=0;i<Nharm;i++) {
|
|
|
|
lin = ngsp_data.readLine();
|
2015-05-18 14:24:16 +03:00
|
|
|
if (!firstgroup) {
|
|
|
|
QList<double> sim_point;
|
|
|
|
sim_point.append(lin.section(sep,1,1,QString::SectionSkipEmpty).toDouble()); // freq
|
|
|
|
sim_point.append(lin.section(sep,2,2,QString::SectionSkipEmpty).toDouble()); // magnitude
|
|
|
|
sim_point.append(lin.section(sep,3,3,QString::SectionSkipEmpty).toDouble()); // phase
|
2015-05-19 15:29:21 +03:00
|
|
|
sim_point.append(lin.section(sep,4,4,QString::SectionSkipEmpty).toDouble()); // normalized magnitude
|
|
|
|
sim_point.append(lin.section(sep,5,5,QString::SectionSkipEmpty).toDouble()); // normalized phase
|
2015-05-18 14:24:16 +03:00
|
|
|
sim_points.append(sim_point);
|
|
|
|
} else {
|
|
|
|
sim_points[i].append(lin.section(sep,2,2,QString::SectionSkipEmpty).toDouble()); // magnitude
|
|
|
|
sim_points[i].append(lin.section(sep,3,3,QString::SectionSkipEmpty).toDouble()); // phase
|
2015-05-19 15:29:21 +03:00
|
|
|
sim_points[i].append(lin.section(sep,4,4,QString::SectionSkipEmpty).toDouble()); // normalized magnitude
|
|
|
|
sim_points[i].append(lin.section(sep,5,5,QString::SectionSkipEmpty).toDouble()); // normalized phase
|
2015-05-18 14:24:16 +03:00
|
|
|
}
|
2015-05-18 11:39:58 +03:00
|
|
|
}
|
2015-05-18 14:24:16 +03:00
|
|
|
firstgroup = true;
|
2015-05-18 11:39:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ofile.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-26 17:52:37 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::parseNoiseOutput Parse output of .NOISE simulation.
|
2015-05-29 15:56:22 +03:00
|
|
|
* \param[in] ngspice_file Spice output file name
|
|
|
|
* \param[out] sim_points 2D array in which simulation points should be extracted. All simulation
|
2015-05-26 17:52:37 +03:00
|
|
|
* points from all sweep variable steps are extracted in a single array
|
2022-07-05 07:08:28 -04:00
|
|
|
* \param[out] var_list This list is filled by simulation variables. There is a list of dependent
|
|
|
|
* and independent variables. An independent variable is the first in list.
|
2015-05-29 15:56:22 +03:00
|
|
|
* \param[out] ParSwp Set to true if there was parameter sweep
|
2015-05-26 17:52:37 +03:00
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::parseNoiseOutput(QString ngspice_file, QList<QList<double> > &sim_points,
|
2015-05-29 15:56:22 +03:00
|
|
|
QStringList &var_list, bool &ParSwp)
|
2015-05-26 17:52:37 +03:00
|
|
|
{
|
2015-06-17 15:48:41 +03:00
|
|
|
var_list.clear();
|
2015-05-28 15:20:27 +03:00
|
|
|
var_list.append(""); // dummy indep var
|
|
|
|
var_list.append("inoise_total");
|
|
|
|
var_list.append("onoise_total");
|
2015-05-26 17:52:37 +03:00
|
|
|
|
2015-05-29 15:56:22 +03:00
|
|
|
ParSwp = false;
|
2015-05-28 15:20:27 +03:00
|
|
|
QFile ofile(ngspice_file);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream ngsp_data(&ofile);
|
|
|
|
sim_points.clear();
|
2015-05-29 15:56:22 +03:00
|
|
|
int cnt = 0;
|
2015-05-28 15:20:27 +03:00
|
|
|
while (!ngsp_data.atEnd()) {
|
|
|
|
QString line = ngsp_data.readLine();
|
|
|
|
if (line.contains('=')) {
|
|
|
|
QList <double> sim_point;
|
2015-06-05 09:02:34 +03:00
|
|
|
sim_point.append(0.0);
|
2015-05-28 15:20:27 +03:00
|
|
|
sim_point.append(line.section('=',1,1).toDouble());
|
|
|
|
line = ngsp_data.readLine();
|
|
|
|
sim_point.append(line.section('=',1,1).toDouble());
|
|
|
|
sim_points.append(sim_point);
|
2015-05-29 15:56:22 +03:00
|
|
|
cnt++;
|
2015-05-28 15:20:27 +03:00
|
|
|
}
|
|
|
|
}
|
2015-05-29 15:56:22 +03:00
|
|
|
if (cnt>1) ParSwp = true;
|
2015-05-28 15:20:27 +03:00
|
|
|
ofile.close();
|
|
|
|
}
|
2015-05-26 17:52:37 +03:00
|
|
|
}
|
|
|
|
|
2015-11-06 10:51:49 +03:00
|
|
|
void AbstractSpiceKernel::parsePZOutput(QString ngspice_file, QList<QList<double> > &sim_points,
|
|
|
|
QStringList &var_list, bool &ParSwp)
|
|
|
|
{
|
2015-11-06 11:44:58 +03:00
|
|
|
static bool zeros = false; // first run --- poles; second run --- zeros
|
|
|
|
// because poles and zeros vectors have unequal dimension
|
|
|
|
QString var;
|
|
|
|
if (zeros) var = "zero";
|
|
|
|
else var = "pole";
|
|
|
|
|
2015-11-06 10:51:49 +03:00
|
|
|
var_list.clear();
|
|
|
|
sim_points.clear();
|
|
|
|
ParSwp = false;
|
|
|
|
QFile ofile(ngspice_file);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream ngsp_data(&ofile);
|
|
|
|
QStringList lines = ngsp_data.readAll().split("\n");
|
2015-11-08 14:10:56 +03:00
|
|
|
|
|
|
|
if (lines.count("PZ analysis")>1) ParSwp = true;
|
|
|
|
|
2015-11-06 10:51:49 +03:00
|
|
|
foreach (QString lin, lines) { // Extract poles
|
2015-11-06 11:44:58 +03:00
|
|
|
if (lin.contains(var + "(")) {
|
2015-11-08 14:10:56 +03:00
|
|
|
if (!var_list.contains(var)) {
|
|
|
|
var_list.append(var+"_number");
|
|
|
|
var_list.append(var);
|
|
|
|
}
|
2015-11-06 10:51:49 +03:00
|
|
|
QList <double> sim_point;
|
2015-11-08 14:10:56 +03:00
|
|
|
sim_point.append(lin.section('(',1,1).section(')',0,0).toDouble());
|
2015-11-06 10:51:49 +03:00
|
|
|
QString right = lin.section("=",1,1);
|
|
|
|
sim_point.append(right.section(",",0,0).toDouble());
|
|
|
|
sim_point.append(right.section(",",1,1).toDouble());
|
|
|
|
sim_points.append(sim_point);
|
|
|
|
}
|
|
|
|
}
|
2015-11-06 11:44:58 +03:00
|
|
|
zeros = !zeros;
|
2015-11-06 10:51:49 +03:00
|
|
|
ofile.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-12 17:56:08 +03:00
|
|
|
/*!
|
2022-07-05 07:08:28 -04:00
|
|
|
* \brief AbstractSpiceKernel::parseSENSOutput Parse output after DC sensitivity analysis.
|
2017-09-12 17:56:08 +03:00
|
|
|
* \param[in] ngspice_file Spice output file name
|
|
|
|
* \param[out] sim_points 2D array in which simulation points should be extracted. All simulation
|
|
|
|
* points from all sweep variable steps are extracted in a single array
|
2022-07-05 07:08:28 -04:00
|
|
|
* \param[out] var_list This list is filled by simulation variables. There is a list of dependent
|
|
|
|
* and independent variables. An independent variable is the first in list.
|
2017-09-12 17:56:08 +03:00
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::parseSENSOutput(QString ngspice_file, QList<QList<double> > &sim_points,
|
|
|
|
QStringList &var_list)
|
|
|
|
{
|
|
|
|
QFile ofile(ngspice_file);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream ngsp_data(&ofile);
|
|
|
|
QStringList lines = ngsp_data.readAll().split("\n");
|
2017-09-15 15:47:32 +03:00
|
|
|
// Extract variables
|
|
|
|
int cnt = 0;
|
|
|
|
for (auto lin=lines.begin(); lin != lines.end(); lin++) {
|
|
|
|
if (lin->contains("Sens analysis")) cnt++;
|
|
|
|
if ( lin->contains('=')) {
|
|
|
|
QString var = (*lin).section("=",0,0).trimmed();
|
2017-09-12 17:56:08 +03:00
|
|
|
var_list.append(var);
|
2017-09-15 15:47:32 +03:00
|
|
|
}
|
|
|
|
if (cnt>=2) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Extract values
|
|
|
|
QList <double> sim_point;
|
|
|
|
cnt = 0;
|
|
|
|
for (auto lin=lines.begin(); lin != lines.end(); lin++) {
|
|
|
|
if (lin->contains('=')) {
|
|
|
|
double val = (*lin).section("=",1,1).trimmed().toDouble();
|
2017-09-12 17:56:08 +03:00
|
|
|
sim_point.append(val);
|
2017-09-15 15:47:32 +03:00
|
|
|
cnt++;
|
|
|
|
}
|
|
|
|
if (cnt >= var_list.count()) {
|
|
|
|
sim_points.append(sim_point);
|
|
|
|
sim_point.clear();
|
|
|
|
cnt = 0;
|
2017-09-12 17:56:08 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ofile.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-23 15:30:36 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::parseDC_OPoutput Parse DC OP simulation result and setup
|
|
|
|
* schematic node names to show DC bias
|
|
|
|
* \param ngspice_file[in] DC OP results test file
|
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::parseDC_OPoutput(QString ngspice_file)
|
|
|
|
{
|
2016-02-24 11:17:49 +03:00
|
|
|
QHash<QString,double> NodeVals;
|
|
|
|
QFile ofile(ngspice_file);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream ngsp_data(&ofile);
|
|
|
|
QStringList lines = ngsp_data.readAll().split("\n");
|
|
|
|
foreach (QString lin,lines) {
|
|
|
|
if (lin.contains('=')) {
|
|
|
|
QString nod = lin.section('=',0,0).remove(' ');
|
|
|
|
double val = lin.section('=',1,1).toDouble();
|
|
|
|
NodeVals.insert(nod,val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ofile.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update Node labels on schematic
|
|
|
|
SweepDialog *swpdlg = new SweepDialog(Sch,&NodeVals);
|
|
|
|
delete swpdlg;
|
|
|
|
|
2016-02-23 15:30:36 +03:00
|
|
|
Sch->showBias = 1;
|
|
|
|
}
|
2015-11-06 10:51:49 +03:00
|
|
|
|
2016-06-15 14:41:15 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::parseDC_OPoutputXY Parse DC OP simulation result for XYCE
|
|
|
|
* and setup schematic node names to show DC bias
|
|
|
|
* \param ngspice_file[in] DC OP results test file
|
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::parseDC_OPoutputXY(QString xyce_file)
|
|
|
|
{
|
|
|
|
QHash<QString,double> NodeVals;
|
|
|
|
QFile ofile(xyce_file);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream ngsp_data(&ofile);
|
|
|
|
QStringList lines = ngsp_data.readAll().split("\n");
|
|
|
|
if (lines.count()>=2) {
|
2022-02-24 23:45:28 +01:00
|
|
|
QStringList nods = lines.at(0).split(QRegExp("\\s"),qucs::SkipEmptyParts);
|
|
|
|
QStringList vals = lines.at(1).split(QRegExp("\\s"),qucs::SkipEmptyParts);
|
2016-06-15 14:41:15 +03:00
|
|
|
QStringList::iterator n,v;
|
|
|
|
for(n = nods.begin(),v = vals.begin();n!=nods.end()||v!=vals.end();n++,v++) {
|
|
|
|
if ((*n).startsWith("I(")) {
|
|
|
|
(*n).remove(0,2).chop(1);
|
|
|
|
(*n) += "#branch"; // Ngspice compatible
|
|
|
|
} else (*n).remove(0,2).chop(1); // Remove (I|V), starting ( and closing )
|
|
|
|
NodeVals.insert((*n).toLower(),(*v).toDouble());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ofile.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update Node labels on schematic
|
|
|
|
SweepDialog *swpdlg = new SweepDialog(Sch,&NodeVals);
|
|
|
|
delete swpdlg;
|
|
|
|
|
|
|
|
Sch->showBias = 1;
|
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::parseSTEPOutput This method parses text raw spice
|
|
|
|
* output from Parameter sweep analysis. Can parse data that uses appedwrite.
|
|
|
|
* Extracts a simulation points array and variables names and types (Real
|
|
|
|
* or Complex) from output.
|
|
|
|
* \param ngspice_file Spice output file name
|
|
|
|
* \param sim_points 2D array in which simulation points should be extracted. All simulation
|
|
|
|
* points from all sweep variable steps are extracted in a single array
|
2022-07-05 07:08:28 -04:00
|
|
|
* \param var_list This list is filled by simulation variables. There is a list of dependent
|
|
|
|
* and independent variables. An independent variable is the first in list.
|
2015-04-24 13:37:56 +03:00
|
|
|
* \param isComplex Type of variables. True if complex. False if real.
|
|
|
|
*/
|
2015-01-19 18:19:44 +03:00
|
|
|
void AbstractSpiceKernel::parseSTEPOutput(QString ngspice_file,
|
|
|
|
QList< QList<double> > &sim_points,
|
|
|
|
QStringList &var_list, bool &isComplex)
|
|
|
|
{
|
2015-03-20 17:40:23 +03:00
|
|
|
isComplex = false;
|
2016-04-27 11:54:24 +03:00
|
|
|
bool isBinary = false;
|
|
|
|
int bin_offset = 0;
|
|
|
|
QByteArray content;
|
2015-03-20 17:40:23 +03:00
|
|
|
|
|
|
|
QFile ofile(ngspice_file);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
2016-04-27 11:54:24 +03:00
|
|
|
content = ofile.readAll();
|
|
|
|
ofile.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextStream ngsp_data(&content);
|
|
|
|
sim_points.clear();
|
|
|
|
bool start_values_sec = false;
|
|
|
|
bool header_parsed = false;
|
|
|
|
int NumVars=0; // Number of dep. and indep.variables
|
|
|
|
int NumPoints=0; // Number of simulation points
|
|
|
|
while (!ngsp_data.atEnd()) {
|
|
|
|
QRegExp sep("[ \t,]");
|
|
|
|
QString lin = ngsp_data.readLine();
|
|
|
|
if (lin.isEmpty()) continue;
|
|
|
|
if (lin.contains("Plotname:")&& // skip operating point
|
|
|
|
(lin.contains("DC operating point"))) {
|
|
|
|
for(bool t = false; !t; t = (ngsp_data.readLine().startsWith("Plotname:")));
|
|
|
|
}
|
|
|
|
if (!header_parsed) {
|
|
|
|
if (lin.contains("Flags")&&lin.contains("complex")) { // output consists of
|
|
|
|
isComplex = true; // complex numbers
|
|
|
|
continue; // maybe ac_analysis
|
2015-03-20 17:40:23 +03:00
|
|
|
}
|
2016-04-27 11:54:24 +03:00
|
|
|
if (lin.contains("No. Variables")) { // get number of variables
|
|
|
|
NumVars=lin.section(sep,2,2,QString::SectionSkipEmpty).toInt();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (lin.contains("No. Points:")) { // get number of variables
|
|
|
|
NumPoints=lin.section(sep,2,2,QString::SectionSkipEmpty).toInt();
|
|
|
|
continue;
|
2015-03-20 17:40:23 +03:00
|
|
|
}
|
2016-04-27 11:54:24 +03:00
|
|
|
if (lin=="Variables:") {
|
|
|
|
var_list.clear();
|
|
|
|
QString indep_var = ngsp_data.readLine().section(sep,1,1,QString::SectionSkipEmpty);
|
|
|
|
var_list.append(indep_var);
|
2015-03-20 17:40:23 +03:00
|
|
|
|
2016-04-27 11:54:24 +03:00
|
|
|
for (int i=1;i<NumVars;i++) {
|
|
|
|
lin = ngsp_data.readLine();
|
|
|
|
QString dep_var = lin.section(sep,1,1,QString::SectionSkipEmpty);
|
|
|
|
var_list.append(dep_var);
|
|
|
|
}
|
|
|
|
header_parsed = true;
|
2015-03-20 17:40:23 +03:00
|
|
|
continue;
|
|
|
|
}
|
2016-04-27 11:54:24 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (lin=="Values:") {
|
|
|
|
start_values_sec = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (lin=="Binary:") {
|
|
|
|
isBinary = true;
|
|
|
|
bin_offset = ngsp_data.pos();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isBinary) {
|
|
|
|
QDataStream dbl(content);
|
|
|
|
dbl.setByteOrder(QDataStream::LittleEndian);
|
|
|
|
dbl.device()->seek(bin_offset);
|
2016-04-29 10:48:47 +03:00
|
|
|
extractBinSamples(dbl,sim_points,NumPoints,NumVars,isComplex);
|
2016-04-27 11:54:24 +03:00
|
|
|
int pos = dbl.device()->pos();
|
|
|
|
ngsp_data.seek(pos);
|
|
|
|
isBinary = false;
|
|
|
|
continue;
|
|
|
|
}
|
2015-01-19 18:19:44 +03:00
|
|
|
|
2016-04-27 11:54:24 +03:00
|
|
|
|
|
|
|
if (start_values_sec) {
|
2016-04-29 10:48:47 +03:00
|
|
|
if (!extractASCIISamples(lin,ngsp_data,sim_points,NumVars,isComplex)) continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void AbstractSpiceKernel::extractBinSamples(QDataStream &dbl, QList<QList<double> > &sim_points,
|
|
|
|
int NumPoints, int NumVars, bool isComplex)
|
|
|
|
{
|
|
|
|
int cnt = NumPoints;
|
|
|
|
while (cnt>0) {
|
|
|
|
QList<double> sim_point;
|
|
|
|
double re,im;
|
|
|
|
dbl>>re; // Indep. variable
|
|
|
|
sim_point.append(re);
|
|
|
|
if (isComplex) dbl>>im; // drop Im part of indep.var
|
|
|
|
for (int i=1;i<NumVars;i++) { // first variable is independent
|
|
|
|
if (isComplex) {
|
|
|
|
dbl>>re; // Re
|
|
|
|
dbl>>im; // Im
|
|
|
|
sim_point.append(re);
|
|
|
|
sim_point.append(im);
|
|
|
|
} else {
|
|
|
|
dbl>>re;
|
|
|
|
sim_point.append(re); // Re
|
2016-04-27 11:54:24 +03:00
|
|
|
}
|
2015-03-20 17:40:23 +03:00
|
|
|
}
|
2016-04-29 10:48:47 +03:00
|
|
|
sim_points.append(sim_point);
|
|
|
|
cnt--;
|
|
|
|
}
|
|
|
|
}
|
2016-04-27 11:54:24 +03:00
|
|
|
|
2016-04-29 10:48:47 +03:00
|
|
|
bool AbstractSpiceKernel::extractASCIISamples(QString &lin, QTextStream &ngsp_data,
|
|
|
|
QList<QList<double> > &sim_points, int NumVars, bool isComplex)
|
|
|
|
{
|
|
|
|
QRegExp sep("[ \t,]");
|
|
|
|
QList<double> sim_point;
|
|
|
|
bool ok = false;
|
|
|
|
QRegExp dataline_patter("^ *[0-9]+[ \t]+.*");
|
|
|
|
if (!dataline_patter.exactMatch(lin)) return false;
|
|
|
|
double indep_val = lin.section(sep,1,1,QString::SectionSkipEmpty).toDouble(&ok);
|
|
|
|
//double indep_val = lin.split(sep,QString::SkipEmptyParts).at(1).toDouble(&ok); // only real indep vars
|
|
|
|
if (!ok) return false;
|
|
|
|
sim_point.append(indep_val);
|
|
|
|
for (int i=0;i<NumVars;i++) {
|
|
|
|
if (isComplex) {
|
2022-02-24 23:45:28 +01:00
|
|
|
QStringList lst = ngsp_data.readLine().split(sep,qucs::SkipEmptyParts);
|
2016-04-29 10:48:47 +03:00
|
|
|
if (lst.count()==2) {
|
|
|
|
double re_dep_val = lst.at(0).toDouble(); // for complex sim results
|
|
|
|
double im_dep_val = lst.at(1).toDouble(); // imaginary part follows
|
|
|
|
sim_point.append(re_dep_val); // real part
|
|
|
|
sim_point.append(im_dep_val);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
double dep_val = ngsp_data.readLine().remove(sep).toDouble();
|
|
|
|
sim_point.append(dep_val);
|
|
|
|
}
|
2015-03-20 17:40:23 +03:00
|
|
|
}
|
2016-04-29 10:48:47 +03:00
|
|
|
sim_points.append(sim_point);
|
|
|
|
return true;
|
2015-01-19 18:19:44 +03:00
|
|
|
}
|
|
|
|
|
2016-06-16 19:39:12 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::parseXYCESTDOutput
|
|
|
|
* \param std_file[in] XYCE STD output file name
|
|
|
|
* \param sim_points[out] 2D array in which simulation points should be extracted
|
2022-07-05 07:08:28 -04:00
|
|
|
* \param var_list[out] This list is filled by simulation variables. There is a list of dependent
|
|
|
|
* and independent variables. An independent variable is the first in list.
|
2016-06-16 19:39:12 +03:00
|
|
|
* \param isComplex[out] Type of variables. True if complex. False if real.
|
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::parseXYCESTDOutput(QString std_file, QList<QList<double> > &sim_points,
|
2022-03-20 12:55:42 +01:00
|
|
|
QStringList &var_list, bool &isComplex, bool &hasParSweep)
|
2016-06-16 19:39:12 +03:00
|
|
|
{
|
|
|
|
isComplex = false;
|
|
|
|
QString content;
|
|
|
|
|
|
|
|
QFile ofile(std_file);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream ts(&ofile);
|
|
|
|
content = ts.readAll();
|
|
|
|
ofile.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextStream ngsp_data(&content);
|
|
|
|
sim_points.clear();
|
|
|
|
var_list.clear();
|
2022-03-13 12:48:03 +01:00
|
|
|
QStringList complex_var_list;
|
|
|
|
QList<int> complex_var_idx;
|
2016-06-16 19:39:12 +03:00
|
|
|
while (!ngsp_data.atEnd()) { // Parse header;
|
|
|
|
QString lin = ngsp_data.readLine();
|
|
|
|
if (lin.isEmpty()) continue;
|
2022-03-20 12:55:42 +01:00
|
|
|
if (lin.contains("Parameter Sweep")) {
|
|
|
|
hasParSweep = true;
|
|
|
|
continue;
|
|
|
|
}
|
2016-06-16 19:39:12 +03:00
|
|
|
if (lin.startsWith("End of ")) continue;
|
|
|
|
if (lin.startsWith("Index ",Qt::CaseInsensitive)) {
|
2022-02-24 23:45:28 +01:00
|
|
|
var_list = lin.split(" ",qucs::SkipEmptyParts);
|
2016-06-16 19:39:12 +03:00
|
|
|
var_list.removeFirst(); // Drop Index
|
2022-03-13 12:48:03 +01:00
|
|
|
for(int i = 0; i < var_list.count()-1; i++) {
|
|
|
|
QString var_re = var_list.at(i);
|
|
|
|
QString var_im = var_list.at(i+1);
|
|
|
|
if (var_re.startsWith("Re(") &&
|
|
|
|
var_im.startsWith("Im(")) {
|
|
|
|
QString var = var_re;
|
|
|
|
var.remove(0,3);
|
|
|
|
var.chop(1);
|
|
|
|
complex_var_list.append(var);
|
|
|
|
complex_var_idx.append(i+1);
|
|
|
|
isComplex = true;
|
|
|
|
}
|
|
|
|
}
|
2016-06-16 19:39:12 +03:00
|
|
|
continue;
|
|
|
|
} else {
|
2022-02-24 23:45:28 +01:00
|
|
|
QStringList val_lst = lin.split(" ",qucs::SkipEmptyParts);
|
2016-06-16 19:39:12 +03:00
|
|
|
QList<double> sim_point;
|
2022-03-13 12:48:03 +01:00
|
|
|
for (int i = 1; i <= var_list.count(); i++ ) {
|
|
|
|
if (isComplex && i != 1) {
|
|
|
|
sim_point.append(val_lst.at(i).toDouble()); // Re and Im
|
|
|
|
sim_point.append(0.0); // real vars
|
|
|
|
} else {
|
|
|
|
sim_point.append(val_lst.at(i).toDouble());
|
|
|
|
}
|
2016-06-16 19:39:12 +03:00
|
|
|
}
|
2022-03-13 12:48:03 +01:00
|
|
|
if (isComplex) { // reassemble complex variables
|
|
|
|
for (int j = 0; j < complex_var_list.count(); j++) {
|
|
|
|
int idx = complex_var_idx[j];
|
|
|
|
sim_point.append(val_lst.at(idx).toDouble());
|
|
|
|
sim_point.append(val_lst.at(idx+1).toDouble());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//sim_point.removeFirst(); // Index
|
2016-06-16 19:39:12 +03:00
|
|
|
sim_points.append(sim_point);
|
|
|
|
}
|
|
|
|
}
|
2022-03-13 12:48:03 +01:00
|
|
|
if (isComplex) {
|
|
|
|
var_list.append(complex_var_list);
|
|
|
|
}
|
2016-06-16 19:39:12 +03:00
|
|
|
}
|
|
|
|
|
2016-06-21 17:44:22 +03:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::parseXYCENoiseLog
|
|
|
|
* \param logfile
|
|
|
|
* \param sim_points
|
|
|
|
* \param var_list
|
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::parseXYCENoiseLog(QString logfile, QList<QList<double> > &sim_points,
|
|
|
|
QStringList &var_list)
|
|
|
|
{
|
|
|
|
var_list.clear();
|
|
|
|
var_list.append(""); // dummy indep var
|
|
|
|
var_list.append("ONOISE_TOTAL");
|
|
|
|
var_list.append("INOISE_TOTAL");
|
|
|
|
QString content;
|
|
|
|
QList <double> sim_point;
|
|
|
|
sim_point.append(0.0);
|
|
|
|
|
|
|
|
QFile ofile(logfile);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream ts(&ofile);
|
|
|
|
content = ts.readAll();
|
|
|
|
ofile.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextStream data(&content);
|
|
|
|
sim_points.clear();
|
|
|
|
while (!data.atEnd()) { // Parse header;
|
|
|
|
QString lin = data.readLine();
|
|
|
|
if (lin.startsWith("Total Output Noise")) {
|
|
|
|
double val = lin.section('=',1,1).toDouble();
|
|
|
|
sim_point.append(val);
|
|
|
|
}
|
|
|
|
if (lin.startsWith("Total Input Noise")) {
|
|
|
|
double val = lin.section('=',1,1).toDouble();
|
|
|
|
sim_point.append(val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sim_points.append(sim_point);
|
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::parseResFile Extract sweep variable name and
|
|
|
|
* values from Ngspice or Xyce *.res output
|
|
|
|
* \param resfile A name of a *.res file
|
|
|
|
* \param var QString in which war is stored
|
|
|
|
* \param values String list in which values are extracted
|
|
|
|
*/
|
2015-03-19 19:46:34 +03:00
|
|
|
void AbstractSpiceKernel::parseResFile(QString resfile, QString &var, QStringList &values)
|
|
|
|
{
|
|
|
|
var.clear();
|
|
|
|
values.clear();
|
|
|
|
|
|
|
|
QFile ofile(resfile);
|
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream swp_data(&ofile);
|
|
|
|
while (!swp_data.atEnd()) {
|
|
|
|
QRegExp point_pattern("^\\s*[0-9]+ .*");
|
|
|
|
QRegExp var_pattern("^STEP\\s+.*");
|
|
|
|
QRegExp sep("\\s");
|
|
|
|
QString lin = swp_data.readLine();
|
|
|
|
if (var_pattern.exactMatch(lin)) {
|
2022-02-24 23:45:28 +01:00
|
|
|
var = lin.split(sep,qucs::SkipEmptyParts).last();
|
2015-03-19 19:46:34 +03:00
|
|
|
}
|
|
|
|
if (point_pattern.exactMatch(lin)) {
|
2022-02-24 23:45:28 +01:00
|
|
|
values.append(lin.split(sep,qucs::SkipEmptyParts).last());
|
2015-03-19 19:46:34 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ofile.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-19 09:37:33 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::checkRawOutupt Determine Ngspice Raw output contains
|
2016-06-22 17:44:23 +03:00
|
|
|
* parameter sweep or is XYCE STD output.
|
2015-05-19 09:37:33 +03:00
|
|
|
* \param ngspice_file[in] Raw output file name
|
|
|
|
* \param values[out] Numbers of parameter sweep steps
|
|
|
|
* \return true if parameter sweep presents, false otherwise
|
|
|
|
*/
|
2016-06-22 17:44:23 +03:00
|
|
|
int AbstractSpiceKernel::checkRawOutupt(QString ngspice_file, QStringList &values)
|
2015-05-19 09:37:33 +03:00
|
|
|
{
|
|
|
|
values.clear();
|
|
|
|
|
|
|
|
QFile ofile(ngspice_file);
|
|
|
|
int plots_cnt = 0;
|
2017-10-19 17:16:01 +03:00
|
|
|
int zeroindex_cnt = 0;
|
|
|
|
bool isXyce = false;
|
2015-05-19 09:37:33 +03:00
|
|
|
if (ofile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream ngsp_data(&ofile);
|
|
|
|
while (!ngsp_data.atEnd()) {
|
|
|
|
QString lin = ngsp_data.readLine();
|
|
|
|
if (lin.startsWith("Plotname: ")) {
|
|
|
|
plots_cnt++;
|
|
|
|
values.append(QString::number(plots_cnt));
|
|
|
|
}
|
2017-10-19 17:16:01 +03:00
|
|
|
if (lin.startsWith("End of Xyce(TM)")) isXyce = true;
|
|
|
|
QRegExp rx("^0\\s+[0-9].*"); // Zero index pattern
|
|
|
|
if (rx.exactMatch(lin)) {
|
|
|
|
zeroindex_cnt++;
|
|
|
|
values.append(QString::number(zeroindex_cnt));
|
|
|
|
}
|
2015-05-19 09:37:33 +03:00
|
|
|
}
|
|
|
|
ofile.close();
|
|
|
|
}
|
2017-10-19 17:16:01 +03:00
|
|
|
int filetype = Unknown;
|
|
|
|
if (plots_cnt>1) filetype = spiceRawSwp;
|
|
|
|
else if ((plots_cnt == 0)&&(isXyce)){
|
|
|
|
if (zeroindex_cnt>1) filetype = xyceSTDswp;
|
|
|
|
else filetype = xyceSTD;
|
|
|
|
} else filetype = spiceRaw;
|
|
|
|
return filetype;
|
2015-05-19 09:37:33 +03:00
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::convertToQucsData Put data extracted from spice raw
|
|
|
|
* text output files (given in outputs_files property) into single XML
|
|
|
|
* Qucs Dataset.
|
|
|
|
* \param qucs_dataset A file name of Qucs Dataset to create
|
|
|
|
* \param xyce True if Xyce simulator was used.
|
|
|
|
*/
|
2016-08-21 14:09:45 +03:00
|
|
|
void AbstractSpiceKernel::convertToQucsData(const QString &qucs_dataset)
|
2015-01-10 17:26:25 +03:00
|
|
|
{
|
2016-02-25 13:47:29 +03:00
|
|
|
if (DC_OP_only) { // Don't touch existing datasets when only DC was simulated
|
|
|
|
// It's need to show DC bias on schematic only
|
|
|
|
foreach(QString output,output_files) {
|
|
|
|
QString full_outfile = workdir+QDir::separator()+output;
|
|
|
|
if (output.endsWith(".dc_op")) parseDC_OPoutput(full_outfile);
|
2016-06-15 14:41:15 +03:00
|
|
|
else if (output.endsWith(".dc_op_xyce")) parseDC_OPoutputXY(full_outfile);
|
2016-02-25 13:47:29 +03:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Merge all outputs in a single Qucs dataset otherwise
|
2016-04-23 19:04:09 +03:00
|
|
|
QString ds_str;
|
|
|
|
QTextStream ds_stream(&ds_str);
|
|
|
|
|
|
|
|
ds_stream<<"<Qucs Dataset " PACKAGE_VERSION ">\n";
|
|
|
|
|
|
|
|
QString sim,indep;
|
|
|
|
QStringList indep_vars;
|
|
|
|
|
|
|
|
QString ngspice_output_filename;
|
|
|
|
foreach(ngspice_output_filename,output_files) { // For every simulation convert results to Qucs dataset
|
2016-07-10 16:38:54 +03:00
|
|
|
QList< QList<double> > sim_points;
|
|
|
|
QStringList var_list;
|
|
|
|
QString swp_var,swp_var2;
|
|
|
|
QStringList swp_var_val,swp_var2_val;
|
|
|
|
bool isComplex = false;
|
|
|
|
bool hasParSweep = false;
|
|
|
|
bool hasDblParSweep = false;
|
|
|
|
|
2022-02-18 22:05:43 +01:00
|
|
|
QRegExp four_rx(".*\\.four[0-9]+$");
|
2016-04-23 19:04:09 +03:00
|
|
|
QString full_outfile = workdir+QDir::separator()+ngspice_output_filename;
|
|
|
|
if (ngspice_output_filename.endsWith("HB.FD.prn")) {
|
2022-03-20 12:55:42 +01:00
|
|
|
//parseHBOutput(full_outfile,sim_points,var_list,hasParSweep);
|
|
|
|
//isComplex = true;
|
|
|
|
parseXYCESTDOutput(full_outfile,sim_points,var_list,isComplex,hasParSweep);
|
2017-10-13 13:30:08 +03:00
|
|
|
if (hasParSweep) {
|
2022-02-14 00:04:11 +01:00
|
|
|
QString res_file = QDir::toNativeSeparators(workdir + QDir::separator()
|
2017-10-13 13:30:08 +03:00
|
|
|
+ "spice4qucs.hb.cir.res");
|
|
|
|
parseResFile(res_file,swp_var,swp_var_val);
|
|
|
|
}
|
2022-02-18 22:05:43 +01:00
|
|
|
} else if (ngspice_output_filename.endsWith(".four") ||
|
|
|
|
four_rx.exactMatch(ngspice_output_filename)) {
|
2016-04-23 19:04:09 +03:00
|
|
|
isComplex=false;
|
2016-08-21 14:09:45 +03:00
|
|
|
parseFourierOutput(full_outfile,sim_points,var_list);
|
2017-09-12 17:56:08 +03:00
|
|
|
} else if (ngspice_output_filename.endsWith(".ngspice.sens.dc.prn")) {
|
|
|
|
isComplex = false;
|
|
|
|
parseSENSOutput(full_outfile,sim_points,var_list);
|
2016-06-16 19:39:12 +03:00
|
|
|
} else if (ngspice_output_filename.endsWith(".txt_std")) {
|
2022-03-20 12:55:42 +01:00
|
|
|
parseXYCESTDOutput(full_outfile,sim_points,var_list,isComplex,hasParSweep);
|
2016-06-21 17:44:22 +03:00
|
|
|
} else if (ngspice_output_filename.endsWith(".noise_log")) {
|
|
|
|
isComplex = false;
|
|
|
|
parseXYCENoiseLog(full_outfile,sim_points,var_list);
|
2016-04-23 19:04:09 +03:00
|
|
|
} else if (ngspice_output_filename.endsWith(".noise")) {
|
|
|
|
isComplex = false;
|
|
|
|
parseNoiseOutput(full_outfile,sim_points,var_list,hasParSweep);
|
|
|
|
if (hasParSweep) {
|
2022-02-14 00:04:11 +01:00
|
|
|
QString res_file = QDir::toNativeSeparators(workdir + QDir::separator()
|
2016-04-23 19:04:09 +03:00
|
|
|
+ "spice4qucs.noise.cir.res");
|
|
|
|
parseResFile(res_file,swp_var,swp_var_val);
|
|
|
|
}
|
|
|
|
} else if (ngspice_output_filename.endsWith(".pz")) {
|
|
|
|
isComplex = true;
|
|
|
|
parsePZOutput(full_outfile,sim_points,var_list,hasParSweep);
|
|
|
|
if (hasParSweep) {
|
2022-02-14 00:04:11 +01:00
|
|
|
QString res_file = QDir::toNativeSeparators(workdir + QDir::separator()
|
2016-04-23 19:04:09 +03:00
|
|
|
+ "spice4qucs.pz.cir.res");
|
2015-03-19 19:46:34 +03:00
|
|
|
parseResFile(res_file,swp_var,swp_var_val);
|
2016-04-23 19:04:09 +03:00
|
|
|
}
|
2017-10-19 18:06:32 +03:00
|
|
|
} else if (ngspice_output_filename.endsWith(".SENS.prn")) {
|
|
|
|
QStringList vals;
|
|
|
|
int type = checkRawOutupt(full_outfile,vals);
|
2022-03-20 12:55:42 +01:00
|
|
|
parseXYCESTDOutput(full_outfile,sim_points,var_list,isComplex,hasParSweep);
|
2017-10-19 18:06:32 +03:00
|
|
|
if (type == xyceSTDswp) {
|
|
|
|
hasParSweep = true;
|
2022-02-14 00:04:11 +01:00
|
|
|
QString res_file = QDir::toNativeSeparators(workdir + QDir::separator()
|
2017-10-19 18:06:32 +03:00
|
|
|
+ "spice4qucs.sens.cir.res");
|
|
|
|
parseResFile(res_file,swp_var,swp_var_val);
|
|
|
|
}
|
2016-04-23 19:04:09 +03:00
|
|
|
} else if (ngspice_output_filename.endsWith("_swp.txt")) {
|
|
|
|
hasParSweep = true;
|
|
|
|
QString simstr = full_outfile;
|
|
|
|
simstr.remove("_swp.txt");
|
|
|
|
if (ngspice_output_filename.endsWith("_swp_swp.txt")) { // 2-var parameter sweep
|
|
|
|
hasDblParSweep = true;
|
|
|
|
simstr.chop(4);
|
|
|
|
simstr = simstr.split('_').last();
|
2022-02-14 00:04:11 +01:00
|
|
|
QString res2_file = QDir::toNativeSeparators(workdir + QDir::separator()
|
2016-04-23 19:04:09 +03:00
|
|
|
+ "spice4qucs." + simstr + ".cir.res1");
|
|
|
|
parseResFile(res2_file,swp_var2,swp_var2_val);
|
|
|
|
} else {
|
|
|
|
simstr = simstr.split('_').last();
|
|
|
|
}
|
2015-03-19 19:46:34 +03:00
|
|
|
|
2022-02-14 00:04:11 +01:00
|
|
|
QString res_file = QDir::toNativeSeparators(workdir + QDir::separator()
|
2016-04-23 19:04:09 +03:00
|
|
|
+ "spice4qucs." + simstr + ".cir.res");
|
|
|
|
parseResFile(res_file,swp_var,swp_var_val);
|
2015-04-01 19:30:34 +03:00
|
|
|
|
2017-10-13 13:30:08 +03:00
|
|
|
parseSTEPOutput(full_outfile,sim_points,var_list,isComplex);
|
2016-04-23 19:04:09 +03:00
|
|
|
} else {
|
2016-06-22 17:44:23 +03:00
|
|
|
int OutType = checkRawOutupt(full_outfile,swp_var_val);
|
2022-03-20 12:55:42 +01:00
|
|
|
bool hasSwp = false;
|
2016-06-22 17:44:23 +03:00
|
|
|
switch (OutType) {
|
|
|
|
case spiceRawSwp:
|
|
|
|
hasParSweep = true;
|
2016-04-23 19:04:09 +03:00
|
|
|
swp_var = "Number";
|
|
|
|
parseSTEPOutput(full_outfile,sim_points,var_list,isComplex);
|
2016-06-22 17:44:23 +03:00
|
|
|
break;
|
|
|
|
case spiceRaw:
|
2016-04-23 19:04:09 +03:00
|
|
|
parseNgSpiceSimOutput(full_outfile,sim_points,var_list,isComplex);
|
2016-06-22 17:44:23 +03:00
|
|
|
break;
|
|
|
|
case xyceSTD:
|
2022-03-20 12:55:42 +01:00
|
|
|
parseXYCESTDOutput(full_outfile,sim_points,var_list,isComplex,hasSwp);
|
2016-06-22 17:44:23 +03:00
|
|
|
break;
|
2017-10-19 17:16:01 +03:00
|
|
|
case xyceSTDswp:
|
|
|
|
hasParSweep = true;
|
|
|
|
swp_var = "Number";
|
2022-03-20 12:55:42 +01:00
|
|
|
parseXYCESTDOutput(full_outfile,sim_points,var_list,isComplex,hasSwp);
|
2016-06-22 17:44:23 +03:00
|
|
|
default: break;
|
2015-01-20 15:03:24 +03:00
|
|
|
}
|
2016-04-23 19:04:09 +03:00
|
|
|
}
|
2022-07-05 07:08:28 -04:00
|
|
|
if (var_list.isEmpty()) continue; // nothing to convert
|
2016-04-23 19:04:09 +03:00
|
|
|
normalizeVarsNames(var_list);
|
2015-03-20 17:40:23 +03:00
|
|
|
|
2016-04-23 19:04:09 +03:00
|
|
|
QString indep = var_list.first();
|
|
|
|
QList<double> sim_point;
|
2015-03-20 17:40:23 +03:00
|
|
|
|
|
|
|
|
2016-04-23 19:04:09 +03:00
|
|
|
if (hasParSweep) {
|
|
|
|
int indep_cnt;
|
2016-07-10 16:38:54 +03:00
|
|
|
if (swp_var_val.isEmpty()) continue;
|
|
|
|
if (hasDblParSweep&&swp_var2_val.isEmpty()) continue;
|
2016-04-23 19:04:09 +03:00
|
|
|
if (hasDblParSweep) indep_cnt = sim_points.count()/(swp_var_val.count()*swp_var2_val.count());
|
|
|
|
else indep_cnt = sim_points.count()/swp_var_val.count();
|
|
|
|
if (!indep.isEmpty()) {
|
|
|
|
ds_stream<<QString("<indep %1 %2>\n").arg(indep).arg(indep_cnt); // output indep var: TODO: parameter sweep
|
|
|
|
for (int i=0;i<indep_cnt;i++) {
|
2022-02-24 23:16:43 +01:00
|
|
|
ds_stream<<QString::number(sim_points.at(i).at(0),'e',12)<<"\n";
|
2015-03-20 17:40:23 +03:00
|
|
|
}
|
2016-04-23 19:04:09 +03:00
|
|
|
ds_stream<<"</indep>\n";
|
|
|
|
}
|
2015-03-20 17:40:23 +03:00
|
|
|
|
2016-04-23 19:04:09 +03:00
|
|
|
ds_stream<<QString("<indep %1 %2>\n").arg(swp_var).arg(swp_var_val.count());
|
|
|
|
foreach (QString val,swp_var_val) {
|
2022-02-24 23:16:43 +01:00
|
|
|
ds_stream<<val<<"\n";
|
2016-04-23 19:04:09 +03:00
|
|
|
}
|
|
|
|
ds_stream<<"</indep>\n";
|
|
|
|
if (indep.isEmpty()) indep = swp_var;
|
|
|
|
else indep += " " + swp_var;
|
|
|
|
if (hasDblParSweep) {
|
|
|
|
ds_stream<<QString("<indep %1 %2>\n").arg(swp_var2).arg(swp_var2_val.count());
|
|
|
|
foreach (QString val,swp_var2_val) {
|
2022-02-24 23:16:43 +01:00
|
|
|
ds_stream<<val<<"\n";
|
2015-03-20 12:38:04 +03:00
|
|
|
}
|
|
|
|
ds_stream<<"</indep>\n";
|
2016-04-23 19:04:09 +03:00
|
|
|
indep += " " + swp_var2;
|
|
|
|
}
|
|
|
|
} else if (!indep.isEmpty()) {
|
|
|
|
ds_stream<<QString("<indep %1 %2>\n").arg(indep).arg(sim_points.count()); // output indep var: TODO: parameter sweep
|
|
|
|
foreach (sim_point,sim_points) {
|
2022-02-24 23:16:43 +01:00
|
|
|
ds_stream<<QString::number(sim_point.at(0),'e',12)<<"\n";
|
2015-03-20 12:38:04 +03:00
|
|
|
}
|
2016-04-23 19:04:09 +03:00
|
|
|
ds_stream<<"</indep>\n";
|
|
|
|
}
|
2015-03-20 17:40:23 +03:00
|
|
|
|
2016-04-23 19:04:09 +03:00
|
|
|
for(int i=1;i<var_list.count();i++) { // output dep var
|
|
|
|
if (indep.isEmpty()) ds_stream<<QString("<indep %1 %2>\n").arg(var_list.at(i)).arg(sim_points.count());
|
|
|
|
else ds_stream<<QString("<dep %1 %2>\n").arg(var_list.at(i)).arg(indep);
|
|
|
|
foreach (sim_point,sim_points) {
|
|
|
|
if (isComplex) {
|
|
|
|
double re=sim_point.at(2*(i-1)+1);
|
|
|
|
double im = sim_point.at(2*i);
|
|
|
|
QString s;
|
|
|
|
s += QString::number(re,'e',12);
|
|
|
|
if (im<0) s += "-j";
|
|
|
|
else s += "+j";
|
|
|
|
s += QString::number(fabs(im),'e',12) + "\n";
|
|
|
|
ds_stream<<s;
|
|
|
|
} else {
|
2022-02-24 23:16:43 +01:00
|
|
|
ds_stream<<QString::number(sim_point.at(i),'e',12)<<"\n";
|
2015-01-10 17:26:25 +03:00
|
|
|
}
|
|
|
|
}
|
2016-04-23 19:04:09 +03:00
|
|
|
if (indep.isEmpty()) ds_stream<<"</indep>\n";
|
|
|
|
else ds_stream<<"</dep>\n";
|
2015-01-10 17:26:25 +03:00
|
|
|
}
|
2016-04-23 19:04:09 +03:00
|
|
|
}
|
2015-01-10 17:26:25 +03:00
|
|
|
|
2016-04-23 19:04:09 +03:00
|
|
|
QFile dataset(qucs_dataset);
|
|
|
|
if (dataset.open(QFile::WriteOnly)) {
|
|
|
|
QTextStream ts(&dataset);
|
|
|
|
ts<<ds_str;
|
2015-01-10 17:26:25 +03:00
|
|
|
dataset.close();
|
|
|
|
}
|
2015-08-03 09:50:28 +03:00
|
|
|
#ifdef NDEBUG
|
|
|
|
removeAllSimulatorOutputs();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::removeAllSimulatorOutputs Clean temporary simulator
|
|
|
|
* datasets.
|
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::removeAllSimulatorOutputs()
|
|
|
|
{
|
|
|
|
foreach(QString output_filename,output_files) {
|
|
|
|
QString full_outfile = workdir+QDir::separator()+output_filename;
|
|
|
|
QFile::remove(full_outfile);
|
|
|
|
}
|
2015-01-10 17:26:25 +03:00
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::normalizeVarsNames Convert spice-style variable names to
|
|
|
|
* Qucs style and add simualation type prefix (i.e. AC, TRAN, DC). Conversion
|
|
|
|
* for harmonic balance variable and current probes variables are supported.
|
|
|
|
* \param var_list This list contains variable names that need normalization.
|
|
|
|
*/
|
2015-01-17 18:34:44 +03:00
|
|
|
void AbstractSpiceKernel::normalizeVarsNames(QStringList &var_list)
|
|
|
|
{
|
|
|
|
QString prefix="";
|
2015-03-22 17:11:27 +03:00
|
|
|
QString iprefix="";
|
2015-01-17 18:34:44 +03:00
|
|
|
QString indep = var_list.first();
|
2015-01-20 19:21:44 +03:00
|
|
|
bool HB = false;
|
2015-01-17 18:34:44 +03:00
|
|
|
indep = indep.toLower();
|
|
|
|
if (indep=="time") {
|
|
|
|
prefix = "tran.";
|
2015-03-22 17:11:27 +03:00
|
|
|
iprefix = "i(tran.";
|
2015-01-17 18:34:44 +03:00
|
|
|
} else if (indep=="frequency") {
|
|
|
|
prefix = "ac.";
|
2015-03-22 17:11:27 +03:00
|
|
|
iprefix = "i(ac.";
|
2015-01-20 19:21:44 +03:00
|
|
|
} else if (indep=="hbfrequency") {
|
|
|
|
HB = true;
|
2015-01-17 18:34:44 +03:00
|
|
|
}
|
|
|
|
|
2017-04-22 15:11:21 +03:00
|
|
|
for(auto it = var_list.begin();it!=var_list.end();it++) { // For subcircuit nodes output i.e. v(X1:n1)
|
|
|
|
(*it).replace(":","_"); // colon symbol is reserved in Qucs as dataset specifier
|
|
|
|
}
|
|
|
|
|
2015-03-22 17:11:27 +03:00
|
|
|
QStringList::iterator it=var_list.begin();
|
|
|
|
|
|
|
|
for (it++;it!=var_list.end();it++) {
|
|
|
|
if ((!(it->startsWith(prefix)||it->startsWith(iprefix)))||(HB)) {
|
|
|
|
if (HB) {
|
2016-07-05 13:39:23 +03:00
|
|
|
QString suffix;
|
|
|
|
if ((*it).startsWith('I')) suffix = ".Ib";
|
|
|
|
else suffix = ".Vb";
|
|
|
|
int idx = it->indexOf('(');
|
2015-03-22 17:11:27 +03:00
|
|
|
int cnt = it->count();
|
2016-07-05 13:39:23 +03:00
|
|
|
*it = it->right(cnt-idx-1);
|
2015-03-22 17:11:27 +03:00
|
|
|
it->remove(')');
|
2016-07-05 13:39:23 +03:00
|
|
|
*it += suffix;
|
|
|
|
QRegExp iprobe_pattern("^[Vv][Pp][Rr][0-9]+.*");
|
|
|
|
if (iprobe_pattern.exactMatch(*it)) (*it).remove(0,1);
|
2015-03-22 17:11:27 +03:00
|
|
|
} else {
|
|
|
|
*it = prefix + *it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
QStringList lst = it->split('(');
|
|
|
|
if (lst.count()>1) {
|
|
|
|
QRegExp ivprobe_pattern("^[Vv][Pp][Rr][0-9]+.*");
|
|
|
|
QRegExp ivprobe_pattern_ngspice("^(ac\\.|tran\\.)[Vv][Pp][Rr][0-9]+.*");
|
|
|
|
if (ivprobe_pattern.exactMatch(lst.at(1))) {
|
|
|
|
lst[1].remove(0,1);
|
|
|
|
*it = lst.join("(");
|
|
|
|
} else if (ivprobe_pattern_ngspice.exactMatch(lst.at(1))) {
|
|
|
|
lst[1].replace(".v",".",Qt::CaseInsensitive);
|
|
|
|
*it = lst.join("(");
|
2015-01-17 18:34:44 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-03-22 17:11:27 +03:00
|
|
|
|
2015-01-17 18:34:44 +03:00
|
|
|
}
|
2015-01-10 17:26:25 +03:00
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::slotErrors Simulator errors handler
|
|
|
|
* \param err
|
|
|
|
*/
|
2015-01-10 17:26:25 +03:00
|
|
|
void AbstractSpiceKernel::slotErrors(QProcess::ProcessError err)
|
|
|
|
{
|
|
|
|
emit errors(err);
|
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::slotFinished Simulation process normal finish handler
|
|
|
|
*/
|
2015-01-10 17:26:25 +03:00
|
|
|
void AbstractSpiceKernel::slotFinished()
|
|
|
|
{
|
2015-05-04 12:18:45 +03:00
|
|
|
//output.clear();
|
|
|
|
output += SimProcess->readAllStandardOutput();
|
2015-01-10 17:26:25 +03:00
|
|
|
emit finished();
|
2015-05-04 14:07:38 +03:00
|
|
|
emit progress(100);
|
2015-05-04 12:18:45 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::slotProcessOutput Process SimProcess output and report progress
|
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::slotProcessOutput()
|
|
|
|
{
|
|
|
|
|
2015-01-10 17:26:25 +03:00
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::getOutput Get sdtout and stderr output of simulation
|
|
|
|
* process.
|
|
|
|
* \return Simulation process output
|
|
|
|
*/
|
2015-01-10 17:26:25 +03:00
|
|
|
QString AbstractSpiceKernel::getOutput()
|
|
|
|
{
|
|
|
|
return output;
|
|
|
|
}
|
2015-04-21 14:41:00 +03:00
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
2020-02-05 20:39:17 +01:00
|
|
|
* \brief AbstractSpiceKernel::setSimulatorCmd Set simulator executable location
|
2015-04-24 13:37:56 +03:00
|
|
|
* \param cmd Simulator executable absolute path. For example /usr/bin/ngspice
|
|
|
|
*/
|
2015-04-21 14:41:00 +03:00
|
|
|
void AbstractSpiceKernel::setSimulatorCmd(QString cmd)
|
|
|
|
{
|
|
|
|
simulator_cmd = cmd;
|
|
|
|
}
|
2015-04-21 15:38:38 +03:00
|
|
|
|
2020-02-05 20:39:17 +01:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::setSimulatorParameters Set simulator parameters
|
|
|
|
* \param cmd Simulator executable absolute path. For example -plugin lib/Xyce_ADMS_Plugin.so
|
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::setSimulatorParameters(QString parameters)
|
|
|
|
{
|
|
|
|
simulator_parameters = parameters;
|
|
|
|
}
|
|
|
|
|
2015-07-13 10:18:26 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::setWorkdir Set simulator working directory path
|
|
|
|
* to store netlist and temp data.
|
|
|
|
* \param path[in] New working directory path
|
|
|
|
*/
|
|
|
|
void AbstractSpiceKernel::setWorkdir(QString path)
|
|
|
|
{
|
|
|
|
workdir = path;
|
|
|
|
QFileInfo inf(workdir);
|
|
|
|
if (!inf.exists()) {
|
|
|
|
QDir dir;
|
|
|
|
dir.mkpath(workdir);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-24 13:37:56 +03:00
|
|
|
/*!
|
|
|
|
* \brief AbstractSpiceKernel::SaveNetlist Save netlist to file. Reimplemented
|
|
|
|
* in Ngspice and Xyce classes.
|
|
|
|
*/
|
2015-04-21 15:38:38 +03:00
|
|
|
void AbstractSpiceKernel::SaveNetlist(QString)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
2015-07-21 17:32:52 +03:00
|
|
|
|
|
|
|
bool AbstractSpiceKernel::waitEndOfSimulation()
|
|
|
|
{
|
|
|
|
return SimProcess->waitForFinished(10000);
|
|
|
|
}
|
2016-02-25 13:47:29 +03:00
|
|
|
|