2014-11-09 14:59:34 +04:00
|
|
|
/***************************************************************************
|
2015-01-10 17:26:25 +03:00
|
|
|
abstractspicekernel.cpp
|
2014-11-09 14:59:34 +04:00
|
|
|
----------------
|
2015-01-10 17:26:25 +03:00
|
|
|
begin : Sat Jan 10 2015
|
|
|
|
copyright : (C) 2015 by Vadim Kuznetsov
|
2014-11-09 14:59:34 +04: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. *
|
|
|
|
* *
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include <config.h>
|
|
|
|
#endif
|
|
|
|
|
2015-01-14 19:30:27 +03:00
|
|
|
#include "externsimdialog.h"
|
2015-04-21 14:41:00 +03:00
|
|
|
#include "simsettingsdialog.h"
|
|
|
|
#include "main.h"
|
2014-11-07 10:11:12 +04:00
|
|
|
|
2015-01-13 11:16:20 +03:00
|
|
|
ExternSimDialog::ExternSimDialog(Schematic *sch,QWidget *parent) :
|
2014-11-07 10:11:12 +04:00
|
|
|
QDialog(parent)
|
|
|
|
{
|
2014-11-07 10:38:35 +04:00
|
|
|
Sch = sch;
|
2015-06-13 14:48:27 +03:00
|
|
|
wasSimulated = false;
|
2014-11-09 14:48:51 +04:00
|
|
|
|
2015-07-13 09:09:26 +03:00
|
|
|
workdir = QucsSettings.S4Qworkdir;
|
2014-11-12 11:31:15 +04:00
|
|
|
QFileInfo inf(workdir);
|
|
|
|
if (!inf.exists()) {
|
|
|
|
QDir dir;
|
|
|
|
dir.mkpath(workdir);
|
|
|
|
}
|
2014-11-10 14:46:40 +04:00
|
|
|
|
2015-01-10 17:26:25 +03:00
|
|
|
ngspice = new Ngspice(sch,this);
|
2015-01-17 11:03:29 +03:00
|
|
|
xyce = new Xyce(sch,this);
|
2015-01-10 17:26:25 +03:00
|
|
|
|
2014-11-09 14:48:51 +04:00
|
|
|
|
|
|
|
buttonSimulate = new QPushButton(tr("Simulate"),this);
|
2015-01-10 17:26:25 +03:00
|
|
|
connect(buttonSimulate,SIGNAL(clicked()),this,SLOT(slotStart()));
|
2015-01-17 11:03:29 +03:00
|
|
|
|
2014-11-09 14:48:51 +04:00
|
|
|
buttonStopSim = new QPushButton(tr("Stop"),this);
|
2015-01-10 17:26:25 +03:00
|
|
|
connect(buttonStopSim,SIGNAL(clicked()),ngspice,SLOT(killThemAll()));
|
2015-07-12 13:29:19 +03:00
|
|
|
connect(buttonStopSim,SIGNAL(clicked()),xyce,SLOT(killThemAll()));
|
2014-11-09 14:48:51 +04:00
|
|
|
buttonStopSim->setEnabled(false);
|
|
|
|
|
2015-04-30 11:19:26 +03:00
|
|
|
buttonSaveNetlist = new QPushButton(tr("Save netlist"),this);
|
2015-04-21 15:38:38 +03:00
|
|
|
connect(buttonSaveNetlist,SIGNAL(clicked()),this,SLOT(slotSaveNetlist()));
|
|
|
|
|
2015-07-12 13:29:19 +03:00
|
|
|
buttonExit = new QPushButton(tr("Exit"),this);
|
|
|
|
connect(buttonExit,SIGNAL(clicked()),this,SLOT(reject()));
|
|
|
|
connect(buttonExit,SIGNAL(clicked()),ngspice,SLOT(killThemAll()));
|
|
|
|
connect(buttonExit,SIGNAL(clicked()),xyce,SLOT(killThemAll()));
|
|
|
|
|
2022-02-20 16:25:39 +01:00
|
|
|
QGroupBox *grp_1 = new QGroupBox(tr("Simulation console"),this);
|
2014-11-09 14:48:51 +04:00
|
|
|
QVBoxLayout *vbl1 = new QVBoxLayout;
|
2015-01-17 11:03:29 +03:00
|
|
|
|
2015-07-16 17:56:35 +03:00
|
|
|
editSimConsole = new QPlainTextEdit(this);
|
|
|
|
QFont font;
|
|
|
|
font.setFamily("monospace");
|
|
|
|
font.setPointSize(10);
|
|
|
|
editSimConsole->setFont(font);
|
2014-11-09 14:48:51 +04:00
|
|
|
editSimConsole->setReadOnly(true);
|
|
|
|
vbl1->addWidget(editSimConsole);
|
2022-02-20 16:25:39 +01:00
|
|
|
grp_1->setLayout(vbl1);
|
2014-11-09 14:48:51 +04:00
|
|
|
|
2015-05-04 12:18:45 +03:00
|
|
|
simProgress = new QProgressBar(this);
|
|
|
|
connect(ngspice,SIGNAL(progress(int)),simProgress,SLOT(setValue(int)));
|
|
|
|
connect(xyce,SIGNAL(progress(int)),simProgress,SLOT(setValue(int)));
|
|
|
|
|
2014-11-09 14:48:51 +04:00
|
|
|
QVBoxLayout *vl_top = new QVBoxLayout;
|
2022-02-20 16:25:39 +01:00
|
|
|
vl_top->addWidget(grp_1);
|
2015-05-04 12:18:45 +03:00
|
|
|
vl_top->addWidget(simProgress);
|
2014-11-09 14:48:51 +04:00
|
|
|
QHBoxLayout *hl1 = new QHBoxLayout;
|
|
|
|
hl1->addWidget(buttonSimulate);
|
|
|
|
hl1->addWidget(buttonStopSim);
|
2015-04-21 15:38:38 +03:00
|
|
|
hl1->addWidget(buttonSaveNetlist);
|
2015-07-12 13:29:19 +03:00
|
|
|
hl1->addWidget(buttonExit);
|
2014-11-09 14:48:51 +04:00
|
|
|
vl_top->addLayout(hl1);
|
|
|
|
this->setLayout(vl_top);
|
2015-04-22 11:25:23 +03:00
|
|
|
this->setWindowTitle(tr("Simulate with external simulator"));
|
2015-01-17 11:03:29 +03:00
|
|
|
|
|
|
|
slotSetSimulator();
|
2016-01-07 17:51:20 +03:00
|
|
|
buttonSimulate->click(); // Start simulation
|
|
|
|
|
2014-11-07 10:38:35 +04:00
|
|
|
}
|
|
|
|
|
2015-01-13 11:16:20 +03:00
|
|
|
ExternSimDialog::~ExternSimDialog()
|
2014-11-07 10:38:35 +04:00
|
|
|
{
|
2015-01-10 17:26:25 +03:00
|
|
|
ngspice->killThemAll();
|
2014-11-07 10:38:35 +04:00
|
|
|
}
|
|
|
|
|
2015-01-17 11:03:29 +03:00
|
|
|
void ExternSimDialog::slotSetSimulator()
|
|
|
|
{
|
2016-01-07 17:41:51 +03:00
|
|
|
switch (QucsSettings.DefaultSimulator) {
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simNgspice: {
|
2015-04-26 11:07:00 +03:00
|
|
|
xyce->setParallel(false);
|
2016-01-07 19:43:01 +03:00
|
|
|
connect(ngspice,SIGNAL(started()),this,SLOT(slotNgspiceStarted()));
|
|
|
|
connect(ngspice,SIGNAL(finished()),this,SLOT(slotProcessOutput()));
|
|
|
|
connect(ngspice,SIGNAL(errors(QProcess::ProcessError)),this,SLOT(slotNgspiceStartError(QProcess::ProcessError)));
|
|
|
|
connect(buttonSimulate,SIGNAL(clicked()),ngspice,SLOT(slotSimulate()));
|
2015-10-06 11:21:06 +03:00
|
|
|
ngspice->setSimulatorCmd(QucsSettings.NgspiceExecutable);
|
2020-02-05 20:39:17 +01:00
|
|
|
ngspice->setSimulatorParameters(QucsSettings.SimParameters);
|
2015-01-17 11:03:29 +03:00
|
|
|
}
|
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simXyceSer: {
|
2015-04-26 11:07:00 +03:00
|
|
|
xyce->setParallel(false);
|
2015-01-17 11:03:29 +03:00
|
|
|
connect(xyce,SIGNAL(started()),this,SLOT(slotNgspiceStarted()));
|
2015-10-13 11:34:02 +03:00
|
|
|
connect(xyce,SIGNAL(finished()),this,SLOT(slotProcessOutput()));
|
2015-06-13 13:01:11 +03:00
|
|
|
connect(xyce,SIGNAL(errors(QProcess::ProcessError)),this,SLOT(slotNgspiceStartError(QProcess::ProcessError)));
|
2015-01-17 11:03:29 +03:00
|
|
|
connect(buttonSimulate,SIGNAL(clicked()),xyce,SLOT(slotSimulate()));
|
2020-02-05 20:39:17 +01:00
|
|
|
xyce->setSimulatorParameters(QucsSettings.SimParameters);
|
2015-01-17 11:03:29 +03:00
|
|
|
}
|
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simXycePar: {
|
2015-04-26 14:30:38 +03:00
|
|
|
#ifdef Q_OS_UNIX
|
2015-04-26 11:07:00 +03:00
|
|
|
xyce->setParallel(true);
|
2015-04-26 14:30:38 +03:00
|
|
|
#else
|
|
|
|
xyce->setParallel(false);
|
|
|
|
#endif
|
2015-03-11 10:46:37 +03:00
|
|
|
connect(xyce,SIGNAL(started()),this,SLOT(slotNgspiceStarted()));
|
2015-10-13 11:34:02 +03:00
|
|
|
connect(xyce,SIGNAL(finished()),this,SLOT(slotProcessOutput()));
|
2015-06-13 13:01:11 +03:00
|
|
|
connect(xyce,SIGNAL(errors(QProcess::ProcessError)),this,SLOT(slotNgspiceStartError(QProcess::ProcessError)));
|
2015-03-11 10:46:37 +03:00
|
|
|
connect(buttonSimulate,SIGNAL(clicked()),xyce,SLOT(slotSimulate()));
|
2020-02-05 20:39:17 +01:00
|
|
|
xyce->setSimulatorParameters(QucsSettings.SimParameters);
|
2015-01-17 11:03:29 +03:00
|
|
|
}
|
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simSpiceOpus: {
|
2015-10-06 11:21:06 +03:00
|
|
|
xyce->setParallel(false);
|
|
|
|
connect(ngspice,SIGNAL(started()),this,SLOT(slotNgspiceStarted()),Qt::UniqueConnection);
|
2015-10-13 11:34:02 +03:00
|
|
|
connect(ngspice,SIGNAL(finished()),this,SLOT(slotProcessOutput()),Qt::UniqueConnection);
|
2015-10-06 11:21:06 +03:00
|
|
|
connect(ngspice,SIGNAL(errors(QProcess::ProcessError)),this,SLOT(slotNgspiceStartError(QProcess::ProcessError)),Qt::UniqueConnection);
|
|
|
|
connect(buttonSimulate,SIGNAL(clicked()),ngspice,SLOT(slotSimulate()),Qt::UniqueConnection);
|
|
|
|
ngspice->setSimulatorCmd(QucsSettings.SpiceOpusExecutable);
|
2020-02-05 20:39:17 +01:00
|
|
|
ngspice->setSimulatorParameters(QucsSettings.SimParameters);
|
2015-10-06 11:21:06 +03:00
|
|
|
}
|
|
|
|
break;
|
2015-01-17 11:03:29 +03:00
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
}
|
2014-11-09 14:48:51 +04:00
|
|
|
|
|
|
|
|
2015-10-13 11:34:02 +03:00
|
|
|
void ExternSimDialog::slotProcessOutput()
|
2014-11-09 14:48:51 +04:00
|
|
|
{
|
2015-04-21 15:38:38 +03:00
|
|
|
buttonSaveNetlist->setEnabled(true);
|
2014-11-09 14:48:51 +04:00
|
|
|
buttonStopSim->setEnabled(false);
|
2015-10-13 11:34:02 +03:00
|
|
|
QString out;
|
|
|
|
|
|
|
|
// Set temporary safe output name
|
|
|
|
|
|
|
|
QString ext;
|
2016-01-07 17:41:51 +03:00
|
|
|
switch (QucsSettings.DefaultSimulator) {
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simNgspice:
|
2015-10-13 11:34:02 +03:00
|
|
|
ext = ".dat.ngspice";
|
|
|
|
out = ngspice->getOutput();
|
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simXycePar:
|
|
|
|
case spicecompat::simXyceSer:
|
2015-10-13 11:34:02 +03:00
|
|
|
ext = ".dat.xyce";
|
|
|
|
out = xyce->getOutput();
|
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simSpiceOpus:
|
2015-10-13 11:34:02 +03:00
|
|
|
out = ngspice->getOutput();
|
|
|
|
ext = ".dat.spopus";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
out = "dummy";
|
|
|
|
ext = ".dat";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-07-15 15:37:01 +03:00
|
|
|
if (out.contains("warning",Qt::CaseInsensitive)||
|
2015-07-27 14:26:26 +03:00
|
|
|
out.contains("error",Qt::CaseInsensitive)) {
|
2015-07-15 15:37:01 +03:00
|
|
|
emit warnings();
|
|
|
|
} else emit success();
|
2014-11-12 15:51:40 +04:00
|
|
|
//editSimConsole->clear();
|
2015-07-16 17:56:35 +03:00
|
|
|
editSimConsole->insertPlainText(out);
|
2015-09-25 09:53:40 +03:00
|
|
|
editSimConsole->moveCursor(QTextCursor::End);
|
2015-07-12 14:06:27 +03:00
|
|
|
saveLog();
|
2022-02-19 12:37:38 +01:00
|
|
|
editSimConsole->insertPlainText("Simulation finished\n");
|
2015-10-13 11:34:02 +03:00
|
|
|
|
2014-11-12 15:51:40 +04:00
|
|
|
QFileInfo inf(Sch->DocName);
|
2015-07-15 17:13:46 +03:00
|
|
|
//QString qucs_dataset = inf.canonicalPath()+QDir::separator()+inf.baseName()+"_ngspice.dat";
|
2015-10-13 11:34:02 +03:00
|
|
|
QString qucs_dataset = inf.canonicalPath()+QDir::separator()+inf.baseName()+ext;
|
2016-01-07 17:41:51 +03:00
|
|
|
switch (QucsSettings.DefaultSimulator) {
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simNgspice:
|
|
|
|
case spicecompat::simSpiceOpus:
|
2015-10-13 11:34:02 +03:00
|
|
|
ngspice->convertToQucsData(qucs_dataset);
|
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simXycePar:
|
|
|
|
case spicecompat::simXyceSer:
|
2015-10-13 11:34:02 +03:00
|
|
|
xyce->convertToQucsData(qucs_dataset);
|
|
|
|
break;
|
|
|
|
default:break;
|
|
|
|
}
|
2015-06-13 14:48:27 +03:00
|
|
|
emit simulated();
|
|
|
|
wasSimulated = true;
|
2016-02-25 13:47:29 +03:00
|
|
|
if (Sch->showBias>0) this->close();
|
2014-11-12 15:51:40 +04:00
|
|
|
}
|
|
|
|
|
2015-01-17 18:04:33 +03:00
|
|
|
|
2015-01-13 11:16:20 +03:00
|
|
|
void ExternSimDialog::slotNgspiceStarted()
|
2014-11-12 15:51:40 +04:00
|
|
|
{
|
|
|
|
editSimConsole->clear();
|
2015-07-12 10:43:04 +03:00
|
|
|
QString sim;
|
2016-01-07 17:41:51 +03:00
|
|
|
switch (QucsSettings.DefaultSimulator) {
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simNgspice: sim = "Ngspice";
|
2015-07-12 10:43:04 +03:00
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simXyceSer: sim = "Xyce (serial) ";
|
2015-07-12 10:43:04 +03:00
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simXycePar: sim = "Xyce (parallel) ";
|
2015-07-12 10:43:04 +03:00
|
|
|
break;
|
|
|
|
default: sim = "Simulator "; // Some other simulators could be added ...
|
|
|
|
break;
|
|
|
|
}
|
2015-07-16 17:56:35 +03:00
|
|
|
editSimConsole->insertPlainText(sim + tr(" started...\n"));
|
2014-11-12 15:51:40 +04:00
|
|
|
}
|
|
|
|
|
2015-06-13 13:01:11 +03:00
|
|
|
void ExternSimDialog::slotNgspiceStartError(QProcess::ProcessError err)
|
2014-11-12 15:51:40 +04:00
|
|
|
{
|
2015-06-13 13:01:11 +03:00
|
|
|
QString msg;
|
|
|
|
switch (err) {
|
|
|
|
case QProcess::FailedToStart : msg = tr("Failed to start simulator!");
|
|
|
|
break;
|
|
|
|
case QProcess::Crashed : msg = tr("Simulator crashed!");
|
|
|
|
break;
|
2022-07-05 07:08:28 -04:00
|
|
|
default : msg = tr("Simulator error!");
|
2015-06-13 13:01:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QMessageBox::critical(this,tr("Simulate with SPICE"),msg,QMessageBox::Ok);
|
2015-07-13 10:40:05 +03:00
|
|
|
|
|
|
|
QString sim;
|
2016-01-07 17:41:51 +03:00
|
|
|
switch (QucsSettings.DefaultSimulator) {
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simNgspice: sim = "Ngspice";
|
2015-07-13 10:40:05 +03:00
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simXyceSer: sim = "Xyce (serial) ";
|
2015-07-13 10:40:05 +03:00
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simXycePar: sim = "Xyce (parallel) ";
|
2015-07-13 10:40:05 +03:00
|
|
|
break;
|
|
|
|
default: sim = "Simulator "; // Some other simulators could be added ...
|
|
|
|
break;
|
|
|
|
}
|
2015-07-16 17:56:35 +03:00
|
|
|
editSimConsole->insertPlainText(sim + tr(" error..."));
|
2014-11-09 17:22:04 +04:00
|
|
|
}
|
|
|
|
|
2015-01-13 11:16:20 +03:00
|
|
|
void ExternSimDialog::slotStart()
|
2014-11-09 17:22:04 +04:00
|
|
|
{
|
2015-01-10 17:26:25 +03:00
|
|
|
buttonStopSim->setEnabled(true);
|
2015-04-21 15:38:38 +03:00
|
|
|
buttonSaveNetlist->setEnabled(false);
|
2014-11-07 10:11:12 +04:00
|
|
|
}
|
2014-11-10 13:36:21 +04:00
|
|
|
|
2015-01-13 11:16:20 +03:00
|
|
|
void ExternSimDialog::slotStop()
|
2014-11-10 13:36:21 +04:00
|
|
|
{
|
2015-01-10 17:26:25 +03:00
|
|
|
buttonStopSim->setEnabled(false);
|
2015-04-21 15:38:38 +03:00
|
|
|
buttonSaveNetlist->setEnabled(true);
|
2015-01-10 17:26:25 +03:00
|
|
|
ngspice->killThemAll();
|
|
|
|
}
|
2014-11-10 13:36:21 +04:00
|
|
|
|
2015-04-21 15:38:38 +03:00
|
|
|
void ExternSimDialog::slotSaveNetlist()
|
|
|
|
{
|
2015-07-14 08:58:58 +03:00
|
|
|
QFileInfo inf(Sch->DocName);
|
|
|
|
QString filename = QFileDialog::getSaveFileName(this,tr("Save netlist"),inf.path()+QDir::separator()+"netlist.cir",
|
2015-04-21 15:38:38 +03:00
|
|
|
"All files (*)");
|
2015-06-13 13:16:43 +03:00
|
|
|
if (filename.isEmpty()) return;
|
2015-04-21 15:38:38 +03:00
|
|
|
|
2016-01-07 17:41:51 +03:00
|
|
|
switch (QucsSettings.DefaultSimulator) {
|
2016-05-10 13:38:00 +03:00
|
|
|
case spicecompat::simNgspice:
|
|
|
|
case spicecompat::simSpiceOpus: {
|
2015-04-21 15:38:38 +03:00
|
|
|
ngspice->SaveNetlist(filename);
|
|
|
|
}
|
|
|
|
break;
|
2016-01-07 17:11:09 +03:00
|
|
|
case spicecompat::simXyceSer:
|
|
|
|
case spicecompat::simXycePar: {
|
2015-04-21 15:38:38 +03:00
|
|
|
xyce->SaveNetlist(filename);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!QFile::exists(filename)) {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Save netlist"),
|
|
|
|
QObject::tr("Disk write error!"), QMessageBox::Ok);
|
|
|
|
}
|
|
|
|
}
|
2014-11-10 16:15:44 +04:00
|
|
|
|
2015-07-12 14:06:27 +03:00
|
|
|
void ExternSimDialog::saveLog()
|
|
|
|
{
|
|
|
|
QString filename = QucsSettings.QucsHomeDir.filePath("log.txt");
|
|
|
|
QFile log(filename);
|
|
|
|
if (log.open(QIODevice::WriteOnly)) {
|
|
|
|
QTextStream ts_log(&log);
|
|
|
|
ts_log<<editSimConsole->toPlainText();
|
|
|
|
log.flush();
|
|
|
|
log.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-11 16:00:22 +04:00
|
|
|
|