mirror of
https://github.com/ra3xdh/qucs_s
synced 2025-03-28 21:13:26 +00:00
2278 lines
71 KiB
C++
2278 lines
71 KiB
C++
/***************************************************************************
|
|
schematic_file.cpp
|
|
--------------------
|
|
begin : Sat Mar 27 2004
|
|
copyright : (C) 2003 by Michael Margraf
|
|
email : michael.margraf@alumni.tu-berlin.de
|
|
***************************************************************************/
|
|
|
|
/***************************************************************************
|
|
* *
|
|
* 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 <QtCore>
|
|
#include <QMessageBox>
|
|
#include <QDir>
|
|
#include <QStringList>
|
|
#include <QPlainTextEdit>
|
|
#include <qt3_compat/q3ptrlist.h>
|
|
#include <QTextStream>
|
|
#include <QList>
|
|
#include <QProcess>
|
|
#include <QDebug>
|
|
|
|
#include "main.h"
|
|
#include "node.h"
|
|
#include "schematic.h"
|
|
#include "diagrams/diagrams.h"
|
|
#include "paintings/paintings.h"
|
|
#include "components/spicefile.h"
|
|
#include "components/vhdlfile.h"
|
|
#include "components/verilogfile.h"
|
|
#include "components/libcomp.h"
|
|
#include "components/sparamfile.h"
|
|
#include "module.h"
|
|
#include "misc.h"
|
|
#include "extsimkernels/abstractspicekernel.h"
|
|
#include "extsimkernels/s2spice.h"
|
|
#include "osdi/osdi_0_3.h"
|
|
|
|
|
|
// Here the subcircuits, SPICE components etc are collected. It must be
|
|
// global to also work within the subcircuits.
|
|
SubMap FileList;
|
|
|
|
// Dummy function for osdi_log callback.
|
|
// Without it, the program will crash if print or display function called
|
|
// in verilog-a model.
|
|
extern void osdi_log_skip(void *handle, char* msg, uint32_t lvl)
|
|
{
|
|
(void)handle;
|
|
(void)msg;
|
|
(void)lvl;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
// Creates a Qucs file format (without document properties) in the returning
|
|
// string. This is used to copy the selected elements into the clipboard.
|
|
QString Schematic::createClipboardFile()
|
|
{
|
|
int z=0; // counts selected elements
|
|
Wire *pw;
|
|
Diagram *pd;
|
|
Painting *pp;
|
|
Component *pc;
|
|
|
|
QString s("<Qucs Schematic " PACKAGE_VERSION ">\n");
|
|
|
|
// Build element document.
|
|
s += "<Components>\n";
|
|
for(pc = Components->first(); pc != 0; pc = Components->next())
|
|
if(pc->isSelected) {
|
|
s += pc->save()+"\n"; z++; }
|
|
s += "</Components>\n";
|
|
|
|
s += "<Wires>\n";
|
|
for(pw = Wires->first(); pw != 0; pw = Wires->next())
|
|
if(pw->isSelected) {
|
|
z++;
|
|
if(pw->Label) if(!pw->Label->isSelected) {
|
|
s += pw->save().section('"', 0, 0)+"\"\" 0 0 0>\n";
|
|
continue;
|
|
}
|
|
s += pw->save()+"\n";
|
|
}
|
|
for(Node *pn = Nodes->first(); pn != 0; pn = Nodes->next())
|
|
if(pn->Label) if(pn->Label->isSelected) {
|
|
s += pn->Label->save()+"\n"; z++; }
|
|
s += "</Wires>\n";
|
|
|
|
s += "<Diagrams>\n";
|
|
for(pd = Diagrams->first(); pd != 0; pd = Diagrams->next())
|
|
if(pd->isSelected) {
|
|
s += pd->save()+"\n"; z++; }
|
|
s += "</Diagrams>\n";
|
|
|
|
s += "<Paintings>\n";
|
|
for(pp = Paintings->first(); pp != 0; pp = Paintings->next())
|
|
if(pp->isSelected)
|
|
if(pp->Name.at(0) != '.') { // subcircuit specific -> do not copy
|
|
s += "<"+pp->save()+">\n"; z++; }
|
|
s += "</Paintings>\n";
|
|
|
|
if(z == 0) return ""; // return empty if no selection
|
|
|
|
return s;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
// Only read fields without loading them.
|
|
bool Schematic::loadIntoNothing(QTextStream *stream)
|
|
{
|
|
QString Line, cstr;
|
|
while(!stream->atEnd()) {
|
|
Line = stream->readLine();
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true;
|
|
}
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\n'Painting' field is not closed!"));
|
|
return false;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
// Paste from clipboard.
|
|
bool Schematic::pasteFromClipboard(QTextStream *stream, Q3PtrList<Element> *pe)
|
|
{
|
|
QString Line;
|
|
|
|
Line = stream->readLine();
|
|
if(Line.left(16) != "<Qucs Schematic ") // wrong file type ?
|
|
return false;
|
|
|
|
QString s = PACKAGE_VERSION;
|
|
Line = Line.mid(16, Line.length()-17);
|
|
if(Line != s) { // wrong version number ?
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Wrong document version: ")+Line);
|
|
return false;
|
|
}
|
|
|
|
// read content in symbol edit mode *************************
|
|
if(symbolMode) {
|
|
while(!stream->atEnd()) {
|
|
Line = stream->readLine();
|
|
if(Line == "<Components>") {
|
|
if(!loadIntoNothing(stream)) return false; }
|
|
else
|
|
if(Line == "<Wires>") {
|
|
if(!loadIntoNothing(stream)) return false; }
|
|
else
|
|
if(Line == "<Diagrams>") {
|
|
if(!loadIntoNothing(stream)) return false; }
|
|
else
|
|
if(Line == "<Paintings>") {
|
|
if(!loadPaintings(stream, (Q3PtrList<Painting>*)pe)) return false; }
|
|
else {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Clipboard Format Error:\nUnknown field!"));
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// read content in schematic edit mode *************************
|
|
while(!stream->atEnd()) {
|
|
Line = stream->readLine();
|
|
if(Line == "<Components>") {
|
|
if(!loadComponents(stream, (Q3PtrList<Component>*)pe)) return false; }
|
|
else
|
|
if(Line == "<Wires>") {
|
|
if(!loadWires(stream, pe)) return false; }
|
|
else
|
|
if(Line == "<Diagrams>") {
|
|
if(!loadDiagrams(stream, (Q3PtrList<Diagram>*)pe)) return false; }
|
|
else
|
|
if(Line == "<Paintings>") {
|
|
if(!loadPaintings(stream, (Q3PtrList<Painting>*)pe)) return false; }
|
|
else {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Clipboard Format Error:\nUnknown field!"));
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
int Schematic::saveSymbolCpp (void)
|
|
{
|
|
QFileInfo info (DocName);
|
|
QString cppfile = info.absolutePath () + QDir::separator() + DataSet;
|
|
QFile file (cppfile);
|
|
|
|
if (!file.open (QIODevice::WriteOnly)) {
|
|
QMessageBox::critical (0, QObject::tr("Error"),
|
|
QObject::tr("Cannot save C++ file \"%1\"!").arg(cppfile));
|
|
return -1;
|
|
}
|
|
|
|
QTextStream stream (&file);
|
|
|
|
// automatically compute boundings of drawing
|
|
int xmin = INT_MAX;
|
|
int ymin = INT_MAX;
|
|
int xmax = INT_MIN;
|
|
int ymax = INT_MIN;
|
|
int x1, y1, x2, y2;
|
|
int maxNum = 0;
|
|
Painting * pp;
|
|
|
|
stream << " // symbol drawing code\n";
|
|
for (pp = SymbolPaints.first (); pp != 0; pp = SymbolPaints.next ()) {
|
|
if (pp->Name == ".ID ") continue;
|
|
if (pp->Name == ".PortSym ") {
|
|
if (((PortSymbol*)pp)->numberStr.toInt() > maxNum)
|
|
maxNum = ((PortSymbol*)pp)->numberStr.toInt();
|
|
x1 = ((PortSymbol*)pp)->cx;
|
|
y1 = ((PortSymbol*)pp)->cy;
|
|
if (x1 < xmin) xmin = x1;
|
|
if (x1 > xmax) xmax = x1;
|
|
if (y1 < ymin) ymin = y1;
|
|
if (y1 > ymax) ymax = y1;
|
|
continue;
|
|
}
|
|
pp->Bounding (x1, y1, x2, y2);
|
|
if (x1 < xmin) xmin = x1;
|
|
if (x2 > xmax) xmax = x2;
|
|
if (y1 < ymin) ymin = y1;
|
|
if (y2 > ymax) ymax = y2;
|
|
stream << " " << pp->saveCpp () << "\n";
|
|
}
|
|
|
|
stream << "\n // terminal definitions\n";
|
|
for (int i = 1; i <= maxNum; i++) {
|
|
for (pp = SymbolPaints.first (); pp != 0; pp = SymbolPaints.next ()) {
|
|
if (pp->Name == ".PortSym ")
|
|
if (((PortSymbol*)pp)->numberStr.toInt() == i)
|
|
stream << " " << pp->saveCpp () << "\n";
|
|
}
|
|
}
|
|
|
|
stream << "\n // symbol boundings\n"
|
|
<< " x1 = " << xmin << "; " << " y1 = " << ymin << ";\n"
|
|
<< " x2 = " << xmax << "; " << " y2 = " << ymax << ";\n";
|
|
|
|
stream << "\n // property text position\n";
|
|
for (pp = SymbolPaints.first (); pp != 0; pp = SymbolPaints.next ())
|
|
if (pp->Name == ".ID ")
|
|
stream << " " << pp->saveCpp () << "\n";
|
|
|
|
file.close ();
|
|
return 0;
|
|
}
|
|
|
|
int Schematic::savePropsJSON()
|
|
{
|
|
QFileInfo info (DocName);
|
|
QString jsonfile = info.absolutePath () + QDir::separator()
|
|
+ info.baseName() + "_props.json";
|
|
QString vafilename = info.absolutePath () + QDir::separator()
|
|
+ info.baseName() + ".va";
|
|
QString osdifile = info.absolutePath() + QDir::separator()
|
|
+ info.baseName() + ".osdi";
|
|
|
|
QFile vafile(vafilename);
|
|
if (!vafile.open (QIODevice::ReadOnly)) {
|
|
QMessageBox::critical (0, QObject::tr("Error"),
|
|
QObject::tr("Cannot open Verilog-A file \"%1\"!").arg(vafilename));
|
|
return -1;
|
|
}
|
|
|
|
// if no osdi file exits, generete json file in the old way
|
|
if (!QFile::exists(osdifile)){
|
|
QString module;
|
|
QStringList prop_name;
|
|
QStringList prop_val;
|
|
QTextStream vastream (&vafile);
|
|
while(!vastream.atEnd()) {
|
|
QString line = vastream.readLine();
|
|
line = line.toLower();
|
|
if (line.contains("module")) {
|
|
auto tokens = line.split(QRegularExpression("[\\s()]"));
|
|
if (tokens.count() > 1) module = tokens.at(1);
|
|
module = module.trimmed();
|
|
continue;
|
|
}
|
|
if (line.contains("parameter")) {
|
|
auto tokens = line.split(QRegularExpression("[\\s=;]"),qucs::SkipEmptyParts);
|
|
if (tokens.count() >= 4) {
|
|
for(int ic = 0; ic <= tokens.count(); ic++) {
|
|
if (tokens.at(ic) == "parameter") {
|
|
prop_name.append(tokens.at(ic+2));
|
|
prop_val.append(tokens.at(ic+3));
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
vafile.close();
|
|
|
|
QFile file (jsonfile);
|
|
|
|
if (!file.open (QIODevice::WriteOnly)) {
|
|
QMessageBox::critical (0, QObject::tr("Error"),
|
|
QObject::tr("Cannot save JSON props file \"%1\"!").arg(jsonfile));
|
|
return -1;
|
|
}
|
|
|
|
QTextStream stream (&file);
|
|
|
|
stream << "{\n";
|
|
|
|
stream << QString(" \"description\" : \"%1 verilog device\",\n").arg(module);
|
|
stream << " \"property\" : [\n";
|
|
auto name = prop_name.begin();
|
|
auto val = prop_val.begin();
|
|
for(; name != prop_name.end(); name++,val++) {
|
|
stream << QString(" { \"name\" : \"%1\", \"value\" : \"%2\", \"display\" : \"false\", \"desc\" : \"-\"},\n")
|
|
.arg(*name,*val);
|
|
}
|
|
stream << " ],\n\n";
|
|
stream << " \"tx\" : 4,\n";
|
|
stream << " \"ty\" : 4,\n";
|
|
stream << QString(" \"Model\" : \"%1\",\n").arg(module);
|
|
stream << " \"NetName\" : \"T\",\n\n\n";
|
|
stream << QString(" \"SymName\" : \"%1\",\n").arg(module);
|
|
stream << QString(" \"BitmapFile\" : \"%1\",\n").arg(module);
|
|
|
|
stream << "}";
|
|
|
|
file.close ();
|
|
}else{
|
|
QString module;
|
|
QStringList prop_name;
|
|
QStringList prop_val;
|
|
QStringList prop_disp;
|
|
QStringList prop_desc;
|
|
|
|
QLibrary osdilib (osdifile);
|
|
if (!osdilib.load()){
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("No valid osdi file. Re-compile verilog-a file first!"));
|
|
return -1;
|
|
}
|
|
|
|
// log function is disabled here.
|
|
// the dummy function osdi_log_backup is assigned to callback pointer
|
|
// to avoid program crash.
|
|
void** osdi_log_ = reinterpret_cast<void**>(osdilib.resolve("osdi_log"));
|
|
void* osdi_log_backup = nullptr;
|
|
if (osdi_log_){
|
|
osdi_log_backup = *osdi_log_;
|
|
}
|
|
*osdi_log_ = (void*)osdi_log_skip;
|
|
|
|
OsdiDescriptor* descriptors = reinterpret_cast<OsdiDescriptor*>(osdilib.resolve("OSDI_DESCRIPTORS"));
|
|
auto descriptor = descriptors[0];
|
|
void* handler = nullptr;
|
|
|
|
// OsdiSimParas and OsdiInitInfo have to be initialized before setup_model and setup_instance
|
|
std::vector<char*> sim_params_names_vec={nullptr};
|
|
std::vector<double> sim_params_vals_vec={};
|
|
std::vector<char*> sim_params_str_vec = {nullptr};
|
|
OsdiSimParas sim_params = {
|
|
.names = sim_params_names_vec.data(),
|
|
.vals = sim_params_vals_vec.data(),
|
|
.names_str = sim_params_str_vec.data(),
|
|
.vals_str = nullptr
|
|
};
|
|
OsdiInitInfo sim_info = {
|
|
.flags = 0,
|
|
.num_errors = 0,
|
|
.errors = nullptr
|
|
};
|
|
|
|
void* model = calloc(1,descriptor.model_size);
|
|
void* instance = calloc(1,descriptor.instance_size);
|
|
|
|
descriptor.setup_model(handler,model,&sim_params,&sim_info);
|
|
descriptor.setup_instance(handler,instance,model,300,descriptor.num_terminals,&sim_params,&sim_info);
|
|
|
|
module = QString(descriptor.name);
|
|
for(uint32_t i=1;i<descriptor.num_params;i++) {
|
|
auto param = descriptor.param_opvar+i;
|
|
void* value;
|
|
if (i<descriptor.num_instance_params){
|
|
value = descriptor.access(instance,model,i,ACCESS_FLAG_INSTANCE);
|
|
prop_disp.append("true");
|
|
}else{
|
|
value = descriptor.access(instance,model,i,ACCESS_FLAG_READ);
|
|
prop_disp.append("false");
|
|
}
|
|
|
|
switch (param->flags & PARA_TY_MASK)
|
|
{
|
|
case PARA_TY_INT:
|
|
prop_val.append(QString::number(*static_cast<uint32_t*>(value)));
|
|
break;
|
|
case PARA_TY_REAL:
|
|
prop_val.append(QString::number(*static_cast<double*>(value)));
|
|
break;
|
|
case PARA_TY_STR:
|
|
prop_val.append(QString(static_cast<char*>(value)));
|
|
break;
|
|
default:
|
|
prop_val.append("");
|
|
break;
|
|
}
|
|
|
|
prop_name.append(param->name[0]);
|
|
prop_desc.append(param->description);
|
|
}
|
|
|
|
free(model);
|
|
free(instance);
|
|
if (osdi_log_backup) {
|
|
*osdi_log_ = osdi_log_backup;
|
|
}
|
|
osdilib.unload();
|
|
|
|
QFile file (jsonfile);
|
|
|
|
if (!file.open (QIODevice::WriteOnly)) {
|
|
QMessageBox::critical (0, QObject::tr("Error"),
|
|
QObject::tr("Cannot save JSON props file \"%1\"!").arg(jsonfile));
|
|
return -1;
|
|
}
|
|
|
|
QTextStream stream (&file);
|
|
|
|
stream << "{\n";
|
|
|
|
stream << QString(" \"description\" : \"%1 verilog device\",\n").arg(module);
|
|
stream << " \"property\" : [\n";
|
|
auto name = prop_name.begin();
|
|
auto val = prop_val.begin();
|
|
auto disp = prop_disp.begin();
|
|
auto desc = prop_desc.begin();
|
|
for(; name != prop_name.end(); name++,val++,disp++,desc++) {
|
|
stream << QString(" { \"name\" : \"%1\", \"value\" : \"%2\", \"display\" : \"%3\", \"desc\" : \"%4\"},\n")
|
|
.arg(*name,*val,*disp,*desc);
|
|
}
|
|
stream << " ],\n\n";
|
|
stream << " \"tx\" : 4,\n";
|
|
stream << " \"ty\" : 4,\n";
|
|
stream << QString(" \"Model\" : \"%1\",\n").arg(module);
|
|
stream << " \"NetName\" : \"T\",\n\n\n";
|
|
stream << QString(" \"SymName\" : \"%1\",\n").arg(module);
|
|
stream << QString(" \"BitmapFile\" : \"%1\",\n").arg(module);
|
|
|
|
stream << "}";
|
|
|
|
file.close ();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// save symbol paintings in JSON format
|
|
int Schematic::saveSymbolJSON()
|
|
{
|
|
QFileInfo info (DocName);
|
|
QString jsonfile = info.absolutePath () + QDir::separator()
|
|
+ info.baseName() + "_sym.json";
|
|
|
|
qDebug() << "saveSymbolJson for " << jsonfile;
|
|
|
|
QFile file (jsonfile);
|
|
|
|
if (!file.open (QIODevice::WriteOnly)) {
|
|
QMessageBox::critical (0, QObject::tr("Error"),
|
|
QObject::tr("Cannot save JSON symbol file \"%1\"!").arg(jsonfile));
|
|
return -1;
|
|
}
|
|
|
|
QTextStream stream (&file);
|
|
|
|
// automatically compute boundings of drawing
|
|
int xmin = INT_MAX;
|
|
int ymin = INT_MAX;
|
|
int xmax = INT_MIN;
|
|
int ymax = INT_MIN;
|
|
int x1, y1, x2, y2;
|
|
int maxNum = 0;
|
|
Painting * pp;
|
|
|
|
stream << "{\n";
|
|
|
|
stream << "\"paintings\" : [\n";
|
|
|
|
// symbol drawing code"
|
|
for (pp = SymbolPaints.first (); pp != 0; pp = SymbolPaints.next ()) {
|
|
if (pp->Name == ".ID ") continue;
|
|
if (pp->Name == ".PortSym ") {
|
|
if (((PortSymbol*)pp)->numberStr.toInt() > maxNum)
|
|
maxNum = ((PortSymbol*)pp)->numberStr.toInt();
|
|
x1 = ((PortSymbol*)pp)->cx;
|
|
y1 = ((PortSymbol*)pp)->cy;
|
|
if (x1 < xmin) xmin = x1;
|
|
if (x1 > xmax) xmax = x1;
|
|
if (y1 < ymin) ymin = y1;
|
|
if (y1 > ymax) ymax = y1;
|
|
continue;
|
|
}
|
|
pp->Bounding (x1, y1, x2, y2);
|
|
if (x1 < xmin) xmin = x1;
|
|
if (x2 > xmax) xmax = x2;
|
|
if (y1 < ymin) ymin = y1;
|
|
if (y2 > ymax) ymax = y2;
|
|
stream << " " << pp->saveJSON() << "\n";
|
|
}
|
|
|
|
// terminal definitions
|
|
//stream << "terminal \n";
|
|
for (int i = 1; i <= maxNum; i++) {
|
|
for (pp = SymbolPaints.first (); pp != 0; pp = SymbolPaints.next ()) {
|
|
if (pp->Name == ".PortSym ")
|
|
if (((PortSymbol*)pp)->numberStr.toInt() == i)
|
|
stream << " " << pp->saveJSON () << "\n";
|
|
}
|
|
}
|
|
|
|
stream << "],\n"; //end of paintings JSON array
|
|
|
|
// symbol boundings
|
|
stream
|
|
<< " \"x1\" : " << xmin << ",\n" << " \"y1\" : " << ymin << ",\n"
|
|
<< " \"x2\" : " << xmax << ",\n" << " \"y2\" : " << ymax << ",\n";
|
|
|
|
// property text position
|
|
for (pp = SymbolPaints.first (); pp != 0; pp = SymbolPaints.next ())
|
|
if (pp->Name == ".ID ")
|
|
stream << " " << pp->saveJSON () << "\n";
|
|
|
|
stream << "}\n";
|
|
|
|
file.close ();
|
|
return 0;
|
|
|
|
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
// Returns the number of subcircuit ports.
|
|
int Schematic::saveDocument()
|
|
{
|
|
QFile file(DocName);
|
|
if(!file.open(QIODevice::WriteOnly)) {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Cannot save document!"));
|
|
return -1;
|
|
}
|
|
|
|
QTextStream stream(&file);
|
|
|
|
stream << "<Qucs Schematic " << PACKAGE_VERSION << ">\n";
|
|
|
|
stream << "<Properties>\n";
|
|
if(symbolMode) {
|
|
stream << " <View=" << tmpViewX1<<","<<tmpViewY1<<","
|
|
<< tmpViewX2<<","<<tmpViewY2<< ",";
|
|
stream <<tmpScale<<","<<tmpPosX<<","<<tmpPosY << ">\n";
|
|
}
|
|
else {
|
|
stream << " <View=" << ViewX1<<","<<ViewY1<<","
|
|
<< ViewX2<<","<<ViewY2<< ",";
|
|
stream << Scale <<","<<contentsX()<<","<<contentsY() << ">\n";
|
|
}
|
|
stream << " <Grid=" << GridX<<","<<GridY<<","
|
|
<< GridOn << ">\n";
|
|
stream << " <DataSet=" << DataSet << ">\n";
|
|
stream << " <DataDisplay=" << DataDisplay << ">\n";
|
|
stream << " <OpenDisplay=" << SimOpenDpl << ">\n";
|
|
stream << " <Script=" << Script << ">\n";
|
|
stream << " <RunScript=" << SimRunScript << ">\n";
|
|
stream << " <showFrame=" << showFrame << ">\n";
|
|
|
|
QString t;
|
|
misc::convert2ASCII(t = Frame_Text0);
|
|
stream << " <FrameText0=" << t << ">\n";
|
|
misc::convert2ASCII(t = Frame_Text1);
|
|
stream << " <FrameText1=" << t << ">\n";
|
|
misc::convert2ASCII(t = Frame_Text2);
|
|
stream << " <FrameText2=" << t << ">\n";
|
|
misc::convert2ASCII(t = Frame_Text3);
|
|
stream << " <FrameText3=" << t << ">\n";
|
|
stream << "</Properties>\n";
|
|
|
|
Painting *pp;
|
|
stream << "<Symbol>\n"; // save all paintings for symbol
|
|
for(pp = SymbolPaints.first(); pp != 0; pp = SymbolPaints.next())
|
|
stream << " <" << pp->save() << ">\n";
|
|
stream << "</Symbol>\n";
|
|
|
|
stream << "<Components>\n"; // save all components
|
|
for(Component *pc = DocComps.first(); pc != 0; pc = DocComps.next())
|
|
stream << " " << pc->save() << "\n";
|
|
stream << "</Components>\n";
|
|
|
|
stream << "<Wires>\n"; // save all wires
|
|
for(Wire *pw = DocWires.first(); pw != 0; pw = DocWires.next())
|
|
stream << " " << pw->save() << "\n";
|
|
|
|
// save all labeled nodes as wires
|
|
for(Node *pn = DocNodes.first(); pn != 0; pn = DocNodes.next())
|
|
if(pn->Label) stream << " " << pn->Label->save() << "\n";
|
|
stream << "</Wires>\n";
|
|
|
|
stream << "<Diagrams>\n"; // save all diagrams
|
|
for(Diagram *pd = DocDiags.first(); pd != 0; pd = DocDiags.next())
|
|
stream << " " << pd->save() << "\n";
|
|
stream << "</Diagrams>\n";
|
|
|
|
stream << "<Paintings>\n"; // save all paintings
|
|
for(pp = DocPaints.first(); pp != 0; pp = DocPaints.next())
|
|
stream << " <" << pp->save() << ">\n";
|
|
stream << "</Paintings>\n";
|
|
|
|
file.close();
|
|
|
|
// additionally save symbol C++ code if in a symbol drawing and the
|
|
// associated file is a Verilog-A file
|
|
if (fileSuffix () == "sym") {
|
|
if (fileSuffix (DataDisplay) == "va") {
|
|
saveSymbolCpp ();
|
|
saveSymbolJSON ();
|
|
if (QucsSettings.DefaultSimulator == spicecompat::simNgspice) {
|
|
savePropsJSON();
|
|
} else if (QucsSettings.DefaultSimulator == spicecompat::simQucsator) {
|
|
// TODO slit this into another method, or merge into saveSymbolJSON
|
|
// handle errors in separate
|
|
qDebug() << " -> Run adms for symbol";
|
|
|
|
QString vaFile;
|
|
|
|
// QDir prefix = QDir(QucsSettings.BinDir);
|
|
|
|
QFileInfo inf(QucsSettings.Qucsator);
|
|
QString QucsatorPath = inf.path()+QDir::separator();
|
|
QDir include = QDir(QucsatorPath+"../include/qucs-core");
|
|
|
|
//pick admsXml from settings
|
|
QString admsXml = QucsSettings.AdmsXmlBinDir.canonicalPath();
|
|
|
|
#ifdef __MINGW32__
|
|
admsXml = QDir::toNativeSeparators(admsXml+"/"+"admsXml.exe");
|
|
#else
|
|
admsXml = QDir::toNativeSeparators(admsXml+"/"+"admsXml");
|
|
#endif
|
|
|
|
QString workDir = QucsSettings.QucsWorkDir.absolutePath();
|
|
|
|
qDebug() << "App path : " << qApp->applicationDirPath();
|
|
qDebug() << "workdir" << workDir;
|
|
qDebug() << "homedir" << QucsSettings.QucsHomeDir.absolutePath();
|
|
|
|
vaFile = QucsSettings.QucsWorkDir.filePath(fileBase()+".va");
|
|
|
|
QStringList Arguments;
|
|
Arguments << QDir::toNativeSeparators(vaFile)
|
|
<< "-I" << QDir::toNativeSeparators(include.absolutePath())
|
|
<< "-e" << QDir::toNativeSeparators(include.absoluteFilePath("qucsMODULEguiJSONsymbol.xml"))
|
|
<< "-A" << "dyload";
|
|
|
|
// QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
|
|
// env.insert("PATH", env.value("PATH") );
|
|
|
|
QFile file(admsXml);
|
|
if ( !file.exists() ){
|
|
QMessageBox::critical(this, tr("Error"),
|
|
tr("Program admsXml not found: %1\n\n"
|
|
"Set the admsXml location on the application settings.").arg(admsXml));
|
|
return -1;
|
|
}
|
|
|
|
qDebug() << "Command: " << admsXml << Arguments.join(" ");
|
|
|
|
// need to cd into project to run admsXml?
|
|
QDir::setCurrent(workDir);
|
|
|
|
QProcess builder;
|
|
builder.setProcessChannelMode(QProcess::MergedChannels);
|
|
|
|
builder.start(admsXml, Arguments);
|
|
|
|
|
|
// how to capture [warning]? need to modify admsXml?
|
|
// TODO put stdout, stderr into a dock window, not messagebox
|
|
if (!builder.waitForFinished()) {
|
|
QString cmdString = QString("%1 %2\n\n").arg(admsXml, Arguments.join(" "));
|
|
cmdString = cmdString + builder.errorString();
|
|
QMessageBox::critical(this, tr("Error"), cmdString);
|
|
}
|
|
else {
|
|
QString cmdString = QString("%1 %2\n\n").arg(admsXml, Arguments.join(" "));
|
|
cmdString = cmdString + builder.readAll();
|
|
QMessageBox::information(this, tr("Status"), cmdString);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Append _sym.json into _props.json, save into _symbol.json
|
|
QFile f1(QucsSettings.QucsWorkDir.filePath(fileBase()+"_props.json"));
|
|
QFile f2(QucsSettings.QucsWorkDir.filePath(fileBase()+"_sym.json"));
|
|
f1.open(QIODevice::ReadOnly | QIODevice::Text);
|
|
f2.open(QIODevice::ReadOnly | QIODevice::Text);
|
|
|
|
QString dat1 = QString(f1.readAll());
|
|
QString dat2 = QString(f2.readAll());
|
|
QString finalJSON = dat1.append(dat2);
|
|
|
|
// remove joining point
|
|
finalJSON = finalJSON.replace("}{", "");
|
|
|
|
QFile f3(QucsSettings.QucsWorkDir.filePath(fileBase()+"_symbol.json"));
|
|
f3.open(QIODevice::WriteOnly | QIODevice::Text);
|
|
QTextStream out(&f3);
|
|
out << finalJSON;
|
|
|
|
f1.close();
|
|
f2.close();
|
|
f3.close();
|
|
|
|
// TODO choose icon, default to something or provided png
|
|
|
|
} // if DataDisplay va
|
|
} // if suffix .sym
|
|
|
|
return 0;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
bool Schematic::loadProperties(QTextStream *stream)
|
|
{
|
|
bool ok = true;
|
|
QString Line, cstr, nstr;
|
|
while(!stream->atEnd()) {
|
|
Line = stream->readLine();
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true; // field end ?
|
|
Line = Line.trimmed();
|
|
if(Line.isEmpty()) continue;
|
|
|
|
if(Line.at(0) != '<') {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\nWrong property field limiter!"));
|
|
return false;
|
|
}
|
|
if(Line.at(Line.length()-1) != '>') {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\nWrong property field limiter!"));
|
|
return false;
|
|
}
|
|
Line = Line.mid(1, Line.length()-2); // cut off start and end character
|
|
|
|
cstr = Line.section('=',0,0); // property type
|
|
nstr = Line.section('=',1,1); // property value
|
|
if(cstr == "View") {
|
|
ViewX1 = nstr.section(',',0,0).toInt(&ok); if(ok) {
|
|
ViewY1 = nstr.section(',',1,1).toInt(&ok); if(ok) {
|
|
ViewX2 = nstr.section(',',2,2).toInt(&ok); if(ok) {
|
|
ViewY2 = nstr.section(',',3,3).toInt(&ok); if(ok) {
|
|
Scale = nstr.section(',',4,4).toDouble(&ok); if(ok) {
|
|
tmpViewX1 = nstr.section(',',5,5).toInt(&ok); if(ok)
|
|
tmpViewY1 = nstr.section(',',6,6).toInt(&ok); }}}}} }
|
|
else if(cstr == "Grid") {
|
|
GridX = nstr.section(',',0,0).toInt(&ok); if(ok) {
|
|
GridY = nstr.section(',',1,1).toInt(&ok); if(ok) {
|
|
if(nstr.section(',',2,2).toInt(&ok) == 0) GridOn = false;
|
|
else GridOn = true; }} }
|
|
else if(cstr == "DataSet") DataSet = nstr;
|
|
else if(cstr == "DataDisplay") DataDisplay = nstr;
|
|
else if(cstr == "OpenDisplay")
|
|
if(nstr.toInt(&ok) == 0) SimOpenDpl = false;
|
|
else SimOpenDpl = true;
|
|
else if(cstr == "Script") Script = nstr;
|
|
else if(cstr == "RunScript")
|
|
if(nstr.toInt(&ok) == 0) SimRunScript = false;
|
|
else SimRunScript = true;
|
|
else if(cstr == "showFrame")
|
|
showFrame = nstr.at(0).toLatin1() - '0';
|
|
else if(cstr == "FrameText0") misc::convert2Unicode(Frame_Text0 = nstr);
|
|
else if(cstr == "FrameText1") misc::convert2Unicode(Frame_Text1 = nstr);
|
|
else if(cstr == "FrameText2") misc::convert2Unicode(Frame_Text2 = nstr);
|
|
else if(cstr == "FrameText3") misc::convert2Unicode(Frame_Text3 = nstr);
|
|
else {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\nUnknown property: ")+cstr);
|
|
return false;
|
|
}
|
|
if(!ok) {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\nNumber expected in property field!"));
|
|
return false;
|
|
}
|
|
}
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\n'Property' field is not closed!"));
|
|
return false;
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
// Inserts a component without performing logic for wire optimization.
|
|
void Schematic::simpleInsertComponent(Component *c)
|
|
{
|
|
Node *pn;
|
|
int x, y;
|
|
// connect every node of component
|
|
for (Port *pp : c->Ports) {
|
|
x = pp->x+c->cx;
|
|
y = pp->y+c->cy;
|
|
|
|
// check if new node lies upon existing node
|
|
for(pn = DocNodes.first(); pn != 0; pn = DocNodes.next())
|
|
if(pn->cx == x) if(pn->cy == y) {
|
|
if (!pn->DType.isEmpty()) {
|
|
pp->Type = pn->DType;
|
|
}
|
|
if (!pp->Type.isEmpty()) {
|
|
pn->DType = pp->Type;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if(pn == nullptr) { // create new node, if no existing one lies at this position
|
|
pn = new Node(x, y);
|
|
DocNodes.append(pn);
|
|
}
|
|
pn->Connections.append(c); // connect schematic node to component node
|
|
if (!pp->Type.isEmpty()) {
|
|
pn->DType = pp->Type;
|
|
}
|
|
|
|
pp->Connection = pn; // connect component node to schematic node
|
|
}
|
|
|
|
DocComps.append(c);
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
bool Schematic::loadComponents(QTextStream *stream, Q3PtrList<Component> *List)
|
|
{
|
|
QString Line, cstr;
|
|
Component *c;
|
|
while(!stream->atEnd()) {
|
|
Line = stream->readLine();
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true;
|
|
Line = Line.trimmed();
|
|
if(Line.isEmpty()) continue;
|
|
|
|
/// \todo enable user to load partial schematic, skip unknown components
|
|
c = getComponentFromName(Line, this);
|
|
if(!c) return false;
|
|
|
|
if(List) { // "paste" ?
|
|
int z;
|
|
for(z=c->Name.length()-1; z>=0; z--) // cut off number of component name
|
|
if(!c->Name.at(z).isDigit()) break;
|
|
c->Name = c->Name.left(z+1);
|
|
List->append(c);
|
|
}
|
|
else simpleInsertComponent(c);
|
|
}
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\n'Component' field is not closed!"));
|
|
return false;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
// Inserts a wire without performing logic for optimizing.
|
|
void Schematic::simpleInsertWire(Wire *pw)
|
|
{
|
|
Node *pn;
|
|
// check if first wire node lies upon existing node
|
|
for(pn = DocNodes.first(); pn != 0; pn = DocNodes.next())
|
|
if(pn->cx == pw->x1) if(pn->cy == pw->y1) break;
|
|
|
|
if(!pn) { // create new node, if no existing one lies at this position
|
|
pn = new Node(pw->x1, pw->y1);
|
|
DocNodes.append(pn);
|
|
}
|
|
|
|
if(pw->x1 == pw->x2) if(pw->y1 == pw->y2) {
|
|
pn->Label = pw->Label; // wire with length zero are just node labels
|
|
if (pn->Label) {
|
|
pn->Label->Type = isNodeLabel;
|
|
pn->Label->pOwner = pn;
|
|
}
|
|
delete pw; // delete wire because this is not a wire
|
|
return;
|
|
}
|
|
pn->Connections.append(pw); // connect schematic node to component node
|
|
pw->Port1 = pn;
|
|
|
|
// check if second wire node lies upon existing node
|
|
for(pn = DocNodes.first(); pn != 0; pn = DocNodes.next())
|
|
if(pn->cx == pw->x2) if(pn->cy == pw->y2) break;
|
|
|
|
if(!pn) { // create new node, if no existing one lies at this position
|
|
pn = new Node(pw->x2, pw->y2);
|
|
DocNodes.append(pn);
|
|
}
|
|
pn->Connections.append(pw); // connect schematic node to component node
|
|
pw->Port2 = pn;
|
|
|
|
DocWires.append(pw);
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
bool Schematic::loadWires(QTextStream *stream, Q3PtrList<Element> *List)
|
|
{
|
|
Wire *w;
|
|
QString Line;
|
|
while(!stream->atEnd()) {
|
|
Line = stream->readLine();
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true;
|
|
Line = Line.trimmed();
|
|
if(Line.isEmpty()) continue;
|
|
|
|
// (Node*)4 = move all ports (later on)
|
|
w = new Wire(0,0,0,0, (Node*)4,(Node*)4);
|
|
if(!w->load(Line)) {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\nWrong 'wire' line format!"));
|
|
delete w;
|
|
return false;
|
|
}
|
|
if(List) {
|
|
if(w->x1 == w->x2) if(w->y1 == w->y2) if(w->Label) {
|
|
w->Label->Type = isMovingLabel;
|
|
List->append(w->Label);
|
|
delete w;
|
|
continue;
|
|
}
|
|
List->append(w);
|
|
if(w->Label) List->append(w->Label);
|
|
}
|
|
else simpleInsertWire(w);
|
|
}
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\n'Wire' field is not closed!"));
|
|
return false;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
bool Schematic::loadDiagrams(QTextStream *stream, Q3PtrList<Diagram> *List)
|
|
{
|
|
Diagram *d;
|
|
QString Line, cstr;
|
|
while(!stream->atEnd()) {
|
|
Line = stream->readLine();
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true;
|
|
Line = Line.trimmed();
|
|
if(Line.isEmpty()) continue;
|
|
|
|
cstr = Line.section(' ',0,0); // diagram type
|
|
if(cstr == "<Rect") d = new RectDiagram();
|
|
else if(cstr == "<Polar") d = new PolarDiagram();
|
|
else if(cstr == "<Tab") d = new TabDiagram();
|
|
else if(cstr == "<Smith") d = new SmithDiagram();
|
|
else if(cstr == "<ySmith") d = new SmithDiagram(0,0,false);
|
|
else if(cstr == "<PS") d = new PSDiagram();
|
|
else if(cstr == "<SP") d = new PSDiagram(0,0,false);
|
|
else if(cstr == "<Rect3D") d = new Rect3DDiagram();
|
|
else if(cstr == "<Curve") d = new CurveDiagram();
|
|
else if(cstr == "<Time") d = new TimingDiagram();
|
|
else if(cstr == "<Truth") d = new TruthDiagram();
|
|
else {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\nUnknown diagram!"));
|
|
return false;
|
|
}
|
|
|
|
if(!d->load(Line, stream)) {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\nWrong 'diagram' line format!"));
|
|
delete d;
|
|
return false;
|
|
}
|
|
List->append(d);
|
|
}
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\n'Diagram' field is not closed!"));
|
|
return false;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
bool Schematic::loadPaintings(QTextStream *stream, Q3PtrList<Painting> *List)
|
|
{
|
|
Painting *p=0;
|
|
QString Line, cstr;
|
|
while(!stream->atEnd()) {
|
|
Line = stream->readLine();
|
|
if (Line.trimmed().isEmpty()) continue;
|
|
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true;
|
|
|
|
Line = Line.trimmed();
|
|
if(Line.isEmpty()) continue;
|
|
if( (Line.at(0) != '<') || (Line.at(Line.length()-1) != '>')) {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\nWrong 'painting' line delimiter!"));
|
|
return false;
|
|
}
|
|
Line = Line.mid(1, Line.length()-2); // cut off start and end character
|
|
|
|
cstr = Line.section(' ',0,0); // painting type
|
|
if(cstr == "Line") p = new GraphicLine();
|
|
else if(cstr == "EArc") p = new EllipseArc();
|
|
else if(cstr == ".PortSym") p = new PortSymbol();
|
|
else if(cstr == ".ID") p = new ID_Text();
|
|
else if(cstr == "Text") p = new GraphicText();
|
|
else if(cstr == "Rectangle") p = new qucs::Rectangle();
|
|
else if(cstr == "Arrow") p = new Arrow();
|
|
else if(cstr == "Ellipse") p = new qucs::Ellipse();
|
|
else {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\nUnknown painting!"));
|
|
return false;
|
|
}
|
|
|
|
if(!p->load(Line)) {
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\nWrong 'painting' line format!"));
|
|
delete p;
|
|
return false;
|
|
}
|
|
List->append(p);
|
|
}
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Format Error:\n'Painting' field is not closed!"));
|
|
return false;
|
|
}
|
|
|
|
/*!
|
|
* \brief Schematic::loadDocument tries to load a schematic document.
|
|
* \return true/false in case of success/failure
|
|
*/
|
|
bool Schematic::loadDocument()
|
|
{
|
|
QFile file(DocName);
|
|
if(!file.open(QIODevice::ReadOnly)) {
|
|
/// \todo implement unified error/warning handling GUI and CLI
|
|
if (QucsMain)
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Cannot load document: ")+DocName);
|
|
else
|
|
qCritical() << "Schematic::loadDocument:"
|
|
<< QObject::tr("Cannot load document: ")+DocName;
|
|
return false;
|
|
}
|
|
|
|
// Keep reference to source file (the schematic file)
|
|
setFileInfo(DocName);
|
|
|
|
QString Line;
|
|
QTextStream stream(&file);
|
|
|
|
// read header **************************
|
|
do {
|
|
if(stream.atEnd()) {
|
|
file.close();
|
|
return true;
|
|
}
|
|
|
|
Line = stream.readLine();
|
|
} while(Line.isEmpty());
|
|
|
|
if(Line.left(16) != "<Qucs Schematic ") { // wrong file type ?
|
|
file.close();
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("Wrong document type: ")+DocName);
|
|
return false;
|
|
}
|
|
|
|
Line = Line.mid(16, Line.length()-17);
|
|
if(!misc::checkVersion(Line)) { // wrong version number ?
|
|
|
|
QMessageBox::StandardButton result;
|
|
result = QMessageBox::warning(0,
|
|
QObject::tr("Warning"),
|
|
QObject::tr("Wrong document version \n") +
|
|
DocName + "\n" +
|
|
QObject::tr("Try to open it anyway?"),
|
|
QMessageBox::Yes|QMessageBox::No);
|
|
|
|
if (result==QMessageBox::No) {
|
|
file.close();
|
|
return false;
|
|
}
|
|
|
|
//QMessageBox::critical(0, QObject::tr("Error"),
|
|
// QObject::tr("Wrong document version: ")+Line);
|
|
}
|
|
|
|
// read content *************************
|
|
while(!stream.atEnd()) {
|
|
Line = stream.readLine();
|
|
Line = Line.trimmed();
|
|
if(Line.isEmpty()) continue;
|
|
|
|
if(Line == "<Symbol>") {
|
|
if(!loadPaintings(&stream, &SymbolPaints)) {
|
|
file.close();
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
if(Line == "<Properties>") {
|
|
if(!loadProperties(&stream)) { file.close(); return false; } }
|
|
else
|
|
if(Line == "<Components>") {
|
|
if(!loadComponents(&stream)) { file.close(); return false; } }
|
|
else
|
|
if(Line == "<Wires>") {
|
|
if(!loadWires(&stream)) { file.close(); return false; } }
|
|
else
|
|
if(Line == "<Diagrams>") {
|
|
if(!loadDiagrams(&stream, &DocDiags)) { file.close(); return false; } }
|
|
else
|
|
if(Line == "<Paintings>") {
|
|
if(!loadPaintings(&stream, &DocPaints)) { file.close(); return false; } }
|
|
else {
|
|
qDebug() << Line;
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
QObject::tr("File Format Error:\nUnknown field!"));
|
|
file.close();
|
|
return false;
|
|
}
|
|
}
|
|
|
|
file.close();
|
|
return true;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
// Creates a Qucs file format (without document properties) in the returning
|
|
// string. This is used to save state for undo operation.
|
|
QString Schematic::createUndoString(char Op)
|
|
{
|
|
Wire *pw;
|
|
Diagram *pd;
|
|
Painting *pp;
|
|
Component *pc;
|
|
|
|
// Build element document.
|
|
QString s = " \n";
|
|
s.replace(0,1,Op);
|
|
for(pc = DocComps.first(); pc != 0; pc = DocComps.next())
|
|
s += pc->save()+"\n";
|
|
s += "</>\n"; // short end flag
|
|
|
|
for(pw = DocWires.first(); pw != 0; pw = DocWires.next())
|
|
s += pw->save()+"\n";
|
|
// save all labeled nodes as wires
|
|
for(Node *pn = DocNodes.first(); pn != 0; pn = DocNodes.next())
|
|
if(pn->Label) s += pn->Label->save()+"\n";
|
|
s += "</>\n";
|
|
|
|
for(pd = DocDiags.first(); pd != 0; pd = DocDiags.next())
|
|
s += pd->save()+"\n";
|
|
s += "</>\n";
|
|
|
|
for(pp = DocPaints.first(); pp != 0; pp = DocPaints.next())
|
|
s += "<"+pp->save()+">\n";
|
|
s += "</>\n";
|
|
|
|
return s;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
// Same as "createUndoString(char Op)" but for symbol edit mode.
|
|
QString Schematic::createSymbolUndoString(char Op)
|
|
{
|
|
Painting *pp;
|
|
|
|
// Build element document.
|
|
QString s = " \n";
|
|
s.replace(0,1,Op);
|
|
s += "</>\n"; // short end flag for components
|
|
s += "</>\n"; // short end flag for wires
|
|
s += "</>\n"; // short end flag for diagrams
|
|
|
|
for(pp = SymbolPaints.first(); pp != 0; pp = SymbolPaints.next())
|
|
s += "<"+pp->save()+">\n";
|
|
s += "</>\n";
|
|
|
|
return s;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
// Is quite similar to "loadDocument()" but with less error checking.
|
|
// Used for "undo" function.
|
|
bool Schematic::rebuild(QString *s)
|
|
{
|
|
DocWires.clear(); // delete whole document
|
|
DocNodes.clear();
|
|
DocComps.clear();
|
|
DocDiags.clear();
|
|
DocPaints.clear();
|
|
|
|
QString Line;
|
|
QTextStream stream(s, QIODevice::ReadOnly);
|
|
Line = stream.readLine(); // skip identity byte
|
|
|
|
// read content *************************
|
|
if(!loadComponents(&stream)) return false;
|
|
if(!loadWires(&stream)) return false;
|
|
if(!loadDiagrams(&stream, &DocDiags)) return false;
|
|
if(!loadPaintings(&stream, &DocPaints)) return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
// -------------------------------------------------------------
|
|
// Same as "rebuild(QString *s)" but for symbol edit mode.
|
|
bool Schematic::rebuildSymbol(QString *s)
|
|
{
|
|
SymbolPaints.clear(); // delete whole document
|
|
|
|
QString Line;
|
|
QTextStream stream(s, QIODevice::ReadOnly);
|
|
Line = stream.readLine(); // skip identity byte
|
|
|
|
// read content *************************
|
|
Line = stream.readLine(); // skip components
|
|
Line = stream.readLine(); // skip wires
|
|
Line = stream.readLine(); // skip diagrams
|
|
if(!loadPaintings(&stream, &SymbolPaints)) return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
// ***************************************************************
|
|
// ***** *****
|
|
// ***** Functions to create netlist *****
|
|
// ***** *****
|
|
// ***************************************************************
|
|
|
|
void Schematic::createNodeSet(QStringList& Collect, int& countInit,
|
|
Conductor *pw, Node *p1)
|
|
{
|
|
if(pw->Label)
|
|
if(!pw->Label->initValue.isEmpty())
|
|
Collect.append("NodeSet:NS" + QString::number(countInit++) + " " +
|
|
p1->Name + " U=\"" + pw->Label->initValue + "\"");
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
void Schematic::throughAllNodes(bool User, QStringList& Collect,
|
|
int& countInit)
|
|
{
|
|
Node *pn;
|
|
int z=0;
|
|
|
|
for(pn = DocNodes.first(); pn != 0; pn = DocNodes.next()) {
|
|
if(pn->Name.isEmpty() == User) {
|
|
continue; // already named ?
|
|
}
|
|
if(!User) {
|
|
if(isAnalog)
|
|
pn->Name = "_net";
|
|
else
|
|
pn->Name = "net_net"; // VHDL names must not begin with '_'
|
|
pn->Name += QString::number(z++); // create numbered node name
|
|
}
|
|
else if(pn->State) {
|
|
continue; // already worked on
|
|
}
|
|
|
|
if(isAnalog) createNodeSet(Collect, countInit, pn, pn);
|
|
|
|
pn->State = 1;
|
|
propagateNode(Collect, countInit, pn);
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------
|
|
// Checks whether this file is a qucs file and whether it is an subcircuit.
|
|
// It returns the number of subcircuit ports.
|
|
int Schematic::testFile(const QString& DocName)
|
|
{
|
|
QFile file(DocName);
|
|
if(!file.open(QIODevice::ReadOnly)) {
|
|
return -1;
|
|
}
|
|
|
|
QString Line;
|
|
// .........................................
|
|
// To strongly speed up the file read operation the whole file is
|
|
// read into the memory in one piece.
|
|
QTextStream ReadWhole(&file);
|
|
QString FileString = ReadWhole.readAll();
|
|
file.close();
|
|
QTextStream stream(&FileString, QIODevice::ReadOnly);
|
|
|
|
|
|
// read header ........................
|
|
do {
|
|
if(stream.atEnd()) {
|
|
file.close();
|
|
return -2;
|
|
}
|
|
Line = stream.readLine();
|
|
Line = Line.trimmed();
|
|
} while(Line.isEmpty());
|
|
|
|
if(Line.left(16) != "<Qucs Schematic ") { // wrong file type ?
|
|
file.close();
|
|
return -3;
|
|
}
|
|
|
|
Line = Line.mid(16, Line.length()-17);
|
|
if(!misc::checkVersion(Line)) { // wrong version number ?
|
|
if (!QucsSettings.IgnoreFutureVersion) {
|
|
file.close();
|
|
return -4;
|
|
}
|
|
//file.close();
|
|
//return -4;
|
|
}
|
|
|
|
// read content ....................
|
|
while(!stream.atEnd()) {
|
|
Line = stream.readLine();
|
|
if(Line == "<Components>") break;
|
|
}
|
|
|
|
int z=0;
|
|
while(!stream.atEnd()) {
|
|
Line = stream.readLine();
|
|
if(Line == "</Components>") {
|
|
file.close();
|
|
return z; // return number of ports
|
|
}
|
|
|
|
Line = Line.trimmed();
|
|
QString s = Line.section(' ',0,0); // component type
|
|
if(s == "<Port") z++;
|
|
}
|
|
return -5; // component field not closed
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
// Collects the signal names for digital simulations.
|
|
void Schematic::collectDigitalSignals(void)
|
|
{
|
|
Node *pn;
|
|
|
|
for(pn = DocNodes.first(); pn != 0; pn = DocNodes.next()) {
|
|
DigMap::Iterator it = Signals.find(pn->Name);
|
|
if(it == Signals.end()) { // avoid redeclaration of signal
|
|
Signals.insert(pn->Name, DigSignal(pn->Name, pn->DType));
|
|
} else if (!pn->DType.isEmpty()) {
|
|
it.value().Type = pn->DType;
|
|
}
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
// Propagates the given node to connected component ports.
|
|
void Schematic::propagateNode(QStringList& Collect,
|
|
int& countInit, Node *pn)
|
|
{
|
|
bool setName=false;
|
|
Q3PtrList<Node> Cons;
|
|
Node *p2;
|
|
Wire *pw;
|
|
Element *pe;
|
|
|
|
Cons.append(pn);
|
|
for(p2 = Cons.first(); p2 != 0; p2 = Cons.next())
|
|
for(pe = p2->Connections.first(); pe != 0; pe = p2->Connections.next())
|
|
if(pe->Type == isWire) {
|
|
pw = (Wire*)pe;
|
|
if(p2 != pw->Port1) {
|
|
if(pw->Port1->Name.isEmpty()) {
|
|
pw->Port1->Name = pn->Name;
|
|
pw->Port1->State = 1;
|
|
Cons.append(pw->Port1);
|
|
setName = true;
|
|
}
|
|
}
|
|
else {
|
|
if(pw->Port2->Name.isEmpty()) {
|
|
pw->Port2->Name = pn->Name;
|
|
pw->Port2->State = 1;
|
|
Cons.append(pw->Port2);
|
|
setName = true;
|
|
}
|
|
}
|
|
if(setName) {
|
|
Cons.findRef(p2); // back to current Connection
|
|
if (isAnalog) createNodeSet(Collect, countInit, pw, pn);
|
|
setName = false;
|
|
}
|
|
}
|
|
Cons.clear();
|
|
}
|
|
|
|
#include <iostream>
|
|
|
|
/*!
|
|
* \brief Schematic::throughAllComps
|
|
* Goes through all schematic components and allows special component
|
|
* handling, e.g. like subcircuit netlisting.
|
|
* \param stream is a pointer to the text stream used to collect the netlist
|
|
* \param countInit is the reference to a counter for nodesets (initial conditions)
|
|
* \param Collect is the reference to a list of collected nodesets
|
|
* \param ErrText is pointer to the QPlainTextEdit used for error messages
|
|
* \param NumPorts counter for the number of ports
|
|
* \return true in case of success (false otherwise)
|
|
*/
|
|
bool Schematic::throughAllComps(QTextStream *stream, int& countInit,
|
|
QStringList& Collect, QPlainTextEdit *ErrText, int NumPorts)
|
|
{
|
|
bool r;
|
|
QString s;
|
|
|
|
// give the ground nodes the name "gnd", and insert subcircuits etc.
|
|
Q3PtrListIterator<Component> it(DocComps);
|
|
Component *pc;
|
|
while((pc = it.current()) != 0) {
|
|
++it;
|
|
if(pc->isActive != COMP_IS_ACTIVE) continue;
|
|
|
|
// check analog/digital typed components
|
|
if(isAnalog) {
|
|
if((pc->Type & isAnalogComponent) == 0) {
|
|
ErrText->appendPlainText(QObject::tr("ERROR: Component \"%1\" has no analog model.").arg(pc->Name));
|
|
return false;
|
|
}
|
|
} else {
|
|
if((pc->Type & isDigitalComponent) == 0) {
|
|
ErrText->appendPlainText(QObject::tr("ERROR: Component \"%1\" has no digital model.").arg(pc->Name));
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// handle ground symbol
|
|
if(pc->Model == "GND") {
|
|
pc->Ports.first()->Connection->Name = "gnd";
|
|
continue;
|
|
}
|
|
|
|
// handle subcircuits
|
|
if(pc->Model == "Sub")
|
|
{
|
|
int i;
|
|
// tell the subcircuit it belongs to this schematic
|
|
pc->setSchematic (this);
|
|
QString f = pc->getSubcircuitFile();
|
|
SubMap::Iterator it = FileList.find(f);
|
|
if(it != FileList.end())
|
|
{
|
|
if (!it.value().PortTypes.isEmpty())
|
|
{
|
|
i = 0;
|
|
// apply in/out signal types of subcircuit
|
|
for (Port *pp : pc->Ports)
|
|
{
|
|
pp->Type = it.value().PortTypes[i];
|
|
pp->Connection->DType = pp->Type;
|
|
i++;
|
|
}
|
|
}
|
|
continue; // insert each subcircuit just one time
|
|
}
|
|
|
|
// The subcircuit has not previously been added
|
|
SubFile sub = SubFile("SCH", f);
|
|
FileList.insert(f, sub);
|
|
|
|
|
|
// load subcircuit schematic
|
|
s = pc->Props.first()->Value;
|
|
Schematic *d = new Schematic(0, pc->getSubcircuitFile());
|
|
if(!d->loadDocument()) // load document if possible
|
|
{
|
|
delete d;
|
|
/// \todo implement error/warning message dispatcher for GUI and CLI modes.
|
|
QString message = QObject::tr("ERROR: Cannot load subcircuit \"%1\".").arg(s);
|
|
if (QucsMain) // GUI is running
|
|
ErrText->appendPlainText(message);
|
|
else // command line
|
|
qCritical() << "Schematic::throughAllComps" << message;
|
|
return false;
|
|
}
|
|
d->DocName = s;
|
|
d->isVerilog = isVerilog;
|
|
d->isAnalog = isAnalog;
|
|
d->creatingLib = creatingLib;
|
|
r = d->createSubNetlist(stream, countInit, Collect, ErrText, NumPorts);
|
|
if (r)
|
|
{
|
|
i = 0;
|
|
// save in/out signal types of subcircuit
|
|
for (Port *pp : pc->Ports)
|
|
{
|
|
//if(i>=d->PortTypes.count())break;
|
|
pp->Type = d->PortTypes[i];
|
|
pp->Connection->DType = pp->Type;
|
|
i++;
|
|
}
|
|
sub.PortTypes = d->PortTypes;
|
|
FileList.insert(f,sub);
|
|
//FileList.replace(f, sub);
|
|
}
|
|
delete d;
|
|
if(!r)
|
|
{
|
|
return false;
|
|
}
|
|
continue;
|
|
} // if(pc->Model == "Sub")
|
|
|
|
if(LibComp* lib = dynamic_cast</*const*/LibComp*>(pc)) {
|
|
if(creatingLib) {
|
|
ErrText->appendPlainText(
|
|
QObject::tr("WARNING: Skipping library component \"%1\".").
|
|
arg(pc->Name));
|
|
continue;
|
|
}
|
|
QString scfile = pc->getSubcircuitFile();
|
|
s = scfile + "/" + pc->Props.at(1)->Value;
|
|
SubMap::Iterator it = FileList.find(s);
|
|
if(it != FileList.end())
|
|
continue; // insert each library subcircuit just one time
|
|
FileList.insert(s, SubFile("LIB", s));
|
|
|
|
|
|
unsigned whatisit = isAnalog?1:(isVerilog?4:2);
|
|
if(isAnalog) {
|
|
if (QucsSettings.DefaultSimulator!=spicecompat::simQucsator) {
|
|
if (QucsSettings.DefaultSimulator==spicecompat::simXyce)
|
|
whatisit = 16;
|
|
else whatisit = 8;
|
|
} else whatisit = 1;
|
|
}
|
|
r = lib->createSubNetlist(stream, Collect, whatisit);
|
|
|
|
if(!r) {
|
|
ErrText->appendPlainText(
|
|
QObject::tr("ERROR: \"%1\": Cannot load library component \"%2\" from \"%3\"").
|
|
arg(pc->Name, pc->Props.at(1)->Value, scfile));
|
|
return false;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
// handle SPICE subcircuit components
|
|
if(pc->Model == "SPICE") {
|
|
s = pc->Props.first()->Value;
|
|
// tell the spice component it belongs to this schematic
|
|
pc->setSchematic (this);
|
|
if(s.isEmpty()) {
|
|
ErrText->appendPlainText(QObject::tr("ERROR: No file name in SPICE component \"%1\".").
|
|
arg(pc->Name));
|
|
return false;
|
|
}
|
|
QString f = pc->getSubcircuitFile();
|
|
SubMap::Iterator it = FileList.find(f);
|
|
if(it != FileList.end())
|
|
continue; // insert each spice component just one time
|
|
FileList.insert(f, SubFile("CIR", f));
|
|
|
|
SpiceFile *sf = (SpiceFile*)pc;
|
|
if (QucsSettings.DefaultSimulator != spicecompat::simQucsator)
|
|
r = sf->createSpiceSubckt(stream);
|
|
else r = sf->createSubNetlist(stream);
|
|
ErrText->appendPlainText(sf->getErrorText());
|
|
if(!r){
|
|
return false;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (pc->Model == "SPfile" &&
|
|
QucsSettings.DefaultSimulator == spicecompat::simNgspice) {
|
|
QString f = pc->getSubcircuitFile();
|
|
QString sub_name = "Sub_" + pc->Model + "_" + pc->Name;
|
|
S2Spice *conv = new S2Spice();
|
|
conv->setFile(f);
|
|
conv->setDeviceName(sub_name);
|
|
bool r = conv->convertTouchstone(stream);
|
|
QString msg = conv->getErrText();
|
|
if (!r) {
|
|
QMessageBox::warning(this,tr("Netlist error"), msg);
|
|
return false;
|
|
} else if (!msg.isEmpty()) {
|
|
QMessageBox::warning(this,tr("S2Spice warning"), msg);
|
|
}
|
|
delete conv;
|
|
}
|
|
|
|
// handle digital file subcircuits
|
|
if(pc->Model == "VHDL" || pc->Model == "Verilog") {
|
|
if(isVerilog && pc->Model == "VHDL")
|
|
continue;
|
|
if(!isVerilog && pc->Model == "Verilog")
|
|
continue;
|
|
s = pc->Props.getFirst()->Value;
|
|
if(s.isEmpty()) {
|
|
ErrText->appendPlainText(QObject::tr("ERROR: No file name in %1 component \"%2\".").
|
|
arg(pc->Model).
|
|
arg(pc->Name));
|
|
return false;
|
|
}
|
|
QString f = pc->getSubcircuitFile();
|
|
SubMap::Iterator it = FileList.find(f);
|
|
if(it != FileList.end())
|
|
continue; // insert each vhdl/verilog component just one time
|
|
s = ((pc->Model == "VHDL") ? "VHD" : "VER");
|
|
FileList.insert(f, SubFile(s, f));
|
|
|
|
if(pc->Model == "VHDL") {
|
|
VHDL_File *vf = (VHDL_File*)pc;
|
|
r = vf->createSubNetlist(stream);
|
|
ErrText->appendPlainText(vf->getErrorText());
|
|
if(!r) {
|
|
return false;
|
|
}
|
|
}
|
|
if(pc->Model == "Verilog") {
|
|
Verilog_File *vf = (Verilog_File*)pc;
|
|
r = vf->createSubNetlist(stream);
|
|
ErrText->appendPlainText(vf->getErrorText());
|
|
if(!r) {
|
|
return false;
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
// Follows the wire lines in order to determine the node names for
|
|
// each component. Output into "stream", NodeSets are collected in
|
|
// "Collect" and counted with "countInit".
|
|
bool Schematic::giveNodeNames(QTextStream *stream, int& countInit,
|
|
QStringList& Collect, QPlainTextEdit *ErrText, int NumPorts)
|
|
{
|
|
// delete the node names
|
|
for(Node *pn = DocNodes.first(); pn != 0; pn = DocNodes.next()) {
|
|
pn->State = 0;
|
|
if(pn->Label) {
|
|
if(isAnalog)
|
|
pn->Name = pn->Label->Name;
|
|
else
|
|
pn->Name = "net" + pn->Label->Name;
|
|
}
|
|
else pn->Name = "";
|
|
}
|
|
|
|
// set the wire names to the connected node
|
|
for(Wire *pw = DocWires.first(); pw != 0; pw = DocWires.next())
|
|
if(pw->Label != 0) {
|
|
if(isAnalog)
|
|
pw->Port1->Name = pw->Label->Name;
|
|
else // avoid to use reserved VHDL words
|
|
pw->Port1->Name = "net" + pw->Label->Name;
|
|
}
|
|
|
|
// go through components
|
|
if(!throughAllComps(stream, countInit, Collect, ErrText, NumPorts)){
|
|
fprintf(stderr, "Error: Could not go throughAllComps\n");
|
|
return false;
|
|
}
|
|
|
|
// work on named nodes first in order to preserve the user given names
|
|
throughAllNodes(true, Collect, countInit);
|
|
|
|
// give names to the remaining (unnamed) nodes
|
|
throughAllNodes(false, Collect, countInit);
|
|
|
|
if(!isAnalog) // collect all node names for VHDL signal declaration
|
|
collectDigitalSignals();
|
|
|
|
return true;
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
bool Schematic::createLibNetlist(QTextStream *stream, QPlainTextEdit *ErrText,
|
|
int NumPorts)
|
|
{
|
|
int countInit = 0;
|
|
QStringList Collect;
|
|
Collect.clear();
|
|
FileList.clear();
|
|
Signals.clear();
|
|
// Apply node names and collect subcircuits and file include
|
|
creatingLib = true;
|
|
if(!giveNodeNames(stream, countInit, Collect, ErrText, NumPorts)) {
|
|
creatingLib = false;
|
|
return false;
|
|
}
|
|
creatingLib = false;
|
|
|
|
// Marking start of actual top-level subcircuit
|
|
QString c;
|
|
if(!isAnalog) {
|
|
if (isVerilog)
|
|
c = "///";
|
|
else
|
|
c = "---";
|
|
}
|
|
else c = "###";
|
|
(*stream) << "\n" << c << " TOP LEVEL MARK " << c << "\n";
|
|
|
|
// Emit subcircuit components
|
|
createSubNetlistPlain(stream, ErrText, NumPorts);
|
|
|
|
Signals.clear(); // was filled in "giveNodeNames()"
|
|
return true;
|
|
}
|
|
|
|
//#define VHDL_SIGNAL_TYPE "bit"
|
|
//#define VHDL_LIBRARIES ""
|
|
#define VHDL_SIGNAL_TYPE "std_logic"
|
|
#define VHDL_LIBRARIES "\nlibrary ieee;\nuse ieee.std_logic_1164.all;\n"
|
|
|
|
// ---------------------------------------------------
|
|
void Schematic::createSubNetlistPlain(QTextStream *stream, QPlainTextEdit *ErrText,
|
|
int NumPorts)
|
|
{
|
|
int i, z;
|
|
QString s;
|
|
QStringList SubcircuitPortNames;
|
|
QStringList SubcircuitPortTypes;
|
|
QStringList InPorts;
|
|
QStringList OutPorts;
|
|
QStringList InOutPorts;
|
|
QStringList::iterator it_name;
|
|
QStringList::iterator it_type;
|
|
Component *pc;
|
|
|
|
// probably creating a library currently
|
|
QTextStream * tstream = stream;
|
|
QFile ofile;
|
|
if(creatingLib) {
|
|
QString f = misc::properAbsFileName(DocName) + ".lst";
|
|
ofile.setFileName(f);
|
|
if(!ofile.open(QIODevice::WriteOnly)) {
|
|
ErrText->appendPlainText(tr("ERROR: Cannot create library file \"%s\".").arg(f));
|
|
return;
|
|
}
|
|
tstream = new QTextStream(&ofile);
|
|
}
|
|
|
|
// collect subcircuit ports and sort their node names into
|
|
// "SubcircuitPortNames"
|
|
PortTypes.clear();
|
|
for(pc = DocComps.first(); pc != 0; pc = DocComps.next()) {
|
|
if(pc->Model.at(0) == '.') { // no simulations in subcircuits
|
|
ErrText->appendPlainText(
|
|
QObject::tr("WARNING: Ignore simulation component in subcircuit \"%1\".").arg(DocName)+"\n");
|
|
continue;
|
|
}
|
|
else if(pc->Model == "Port") {
|
|
i = pc->Props.first()->Value.toInt();
|
|
for(z=SubcircuitPortNames.size(); z<i; z++) { // add empty port names
|
|
SubcircuitPortNames.append(" ");
|
|
SubcircuitPortTypes.append(" ");
|
|
}
|
|
it_name = SubcircuitPortNames.begin();
|
|
it_type = SubcircuitPortTypes.begin();
|
|
for(int n=1;n<i;n++)
|
|
{
|
|
it_name++;
|
|
it_type++;
|
|
}
|
|
(*it_name) = pc->Ports.first()->Connection->Name;
|
|
DigMap::Iterator it = Signals.find(*it_name);
|
|
if(it!=Signals.end())
|
|
(*it_type) = it.value().Type;
|
|
// propagate type to port symbol
|
|
pc->Ports.first()->Connection->DType = *it_type;
|
|
|
|
if(!isAnalog) {
|
|
if (isVerilog) {
|
|
Signals.remove(*it_name); // remove node name
|
|
switch(pc->Props.at(1)->Value.at(0).toLatin1()) {
|
|
case 'a':
|
|
InOutPorts.append(*it_name);
|
|
break;
|
|
case 'o':
|
|
OutPorts.append(*it_name);
|
|
break;
|
|
default:
|
|
InPorts.append(*it_name);
|
|
}
|
|
}
|
|
else {
|
|
// remove node name of output port
|
|
Signals.remove(*it_name);
|
|
switch(pc->Props.at(1)->Value.at(0).toLatin1()) {
|
|
case 'a':
|
|
(*it_name) += " : inout"; // attribute "analog" is "inout"
|
|
break;
|
|
case 'o': // output ports need workaround
|
|
Signals.insert(*it_name, DigSignal(*it_name, *it_type));
|
|
(*it_name) = "net_out" + (*it_name);
|
|
(*it_name) += " : " + pc->Props.at(1)->Value;
|
|
break;
|
|
default:
|
|
(*it_name) += " : " + pc->Props.at(1)->Value;
|
|
}
|
|
(*it_name) += " " + ((*it_type).isEmpty() ?
|
|
VHDL_SIGNAL_TYPE : (*it_type));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// remove empty subcircuit ports (missing port numbers)
|
|
for(it_name = SubcircuitPortNames.begin(),
|
|
it_type = SubcircuitPortTypes.begin();
|
|
it_name != SubcircuitPortNames.end(); ) {
|
|
if(*it_name == " ") {
|
|
it_name = SubcircuitPortNames.erase(it_name);
|
|
it_type = SubcircuitPortTypes.erase(it_type);
|
|
} else {
|
|
PortTypes.append(*it_type);
|
|
it_name++;
|
|
it_type++;
|
|
}
|
|
}
|
|
|
|
QString f = misc::properFileName(DocName);
|
|
QString Type = misc::properName(f);
|
|
|
|
Painting *pi;
|
|
|
|
|
|
|
|
if (QucsSettings.DefaultSimulator == spicecompat::simQucsator) {
|
|
|
|
if(isAnalog) {
|
|
// ..... analog subcircuit ...................................
|
|
(*tstream) << "\n.Def:" << Type << " " << SubcircuitPortNames.join(" ");
|
|
for(pi = SymbolPaints.first(); pi != 0; pi = 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++) {
|
|
s = (*it)->Name; // keep 'Name' unchanged
|
|
(*tstream) << " " << s.replace("=", "=\"") << '"';
|
|
}
|
|
break;
|
|
}
|
|
(*tstream) << '\n';
|
|
|
|
// write all components with node names into netlist file
|
|
for(pc = DocComps.first(); pc != 0; pc = DocComps.next())
|
|
(*tstream) << pc->getNetlist();
|
|
|
|
(*tstream) << ".Def:End\n";
|
|
|
|
}
|
|
else {
|
|
if (isVerilog) {
|
|
// ..... digital subcircuit ...................................
|
|
(*tstream) << "\nmodule Sub_" << Type << " ("
|
|
<< SubcircuitPortNames.join(", ") << ");\n";
|
|
|
|
// subcircuit in/out connections
|
|
if(!InPorts.isEmpty())
|
|
(*tstream) << " input " << InPorts.join(", ") << ";\n";
|
|
if(!OutPorts.isEmpty())
|
|
(*tstream) << " output " << OutPorts.join(", ") << ";\n";
|
|
if(!InOutPorts.isEmpty())
|
|
(*tstream) << " inout " << InOutPorts.join(", ") << ";\n";
|
|
|
|
// subcircuit connections
|
|
if(!Signals.isEmpty()) {
|
|
QList<DigSignal> values = Signals.values();
|
|
QList<DigSignal>::const_iterator it;
|
|
for (it = values.constBegin(); it != values.constEnd(); ++it) {
|
|
(*tstream) << " wire " << (*it).Name << ";\n";
|
|
}
|
|
}
|
|
(*tstream) << "\n";
|
|
|
|
// subcircuit parameters
|
|
for(pi = SymbolPaints.first(); pi != 0; pi = SymbolPaints.next())
|
|
if(pi->Name == ".ID ") {
|
|
QList<SubParameter *>::const_iterator it;
|
|
ID_Text *pid = (ID_Text*)pi;
|
|
for(it = pid->Parameter.constBegin(); it != pid->Parameter.constEnd(); it++) {
|
|
s = (*it)->Name.section('=', 0,0);
|
|
QString v = misc::Verilog_Param((*it)->Name.section('=', 1,1));
|
|
(*tstream) << " parameter " << s << " = " << v << ";\n";
|
|
}
|
|
(*tstream) << "\n";
|
|
break;
|
|
}
|
|
|
|
// write all equations into netlist file
|
|
for(pc = DocComps.first(); pc != 0; pc = DocComps.next()) {
|
|
if(pc->Model == "Eqn") {
|
|
(*tstream) << pc->get_Verilog_Code(NumPorts);
|
|
}
|
|
}
|
|
|
|
if(Signals.find("gnd") != Signals.end())
|
|
(*tstream) << " assign gnd = 0;\n"; // should appear only once
|
|
|
|
// write all components into netlist file
|
|
for(pc = DocComps.first(); pc != 0; pc = DocComps.next()) {
|
|
if(pc->Model != "Eqn") {
|
|
s = pc->get_Verilog_Code(NumPorts);
|
|
if(s.length()>0 && s.at(0) == '\xA7') { //section symbol
|
|
ErrText->insertPlainText(s.mid(1));
|
|
}
|
|
else (*tstream) << s;
|
|
}
|
|
}
|
|
|
|
(*tstream) << "endmodule\n";
|
|
} else {
|
|
// ..... digital subcircuit ...................................
|
|
(*tstream) << VHDL_LIBRARIES;
|
|
(*tstream) << "entity Sub_" << Type << " is\n"
|
|
<< " port ("
|
|
<< SubcircuitPortNames.join(";\n ") << ");\n";
|
|
|
|
for(pi = SymbolPaints.first(); pi != 0; pi = SymbolPaints.next())
|
|
if(pi->Name == ".ID ") {
|
|
ID_Text *pid = (ID_Text*)pi;
|
|
QList<SubParameter *>::const_iterator it;
|
|
|
|
(*tstream) << " generic (";
|
|
|
|
for(it = pid->Parameter.constBegin(); it != pid->Parameter.constEnd(); it++) {
|
|
s = (*it)->Name;
|
|
QString t = (*it)->Type.isEmpty() ? "real" : (*it)->Type;
|
|
(*tstream) << s.replace("=", " : "+t+" := ") << ";\n ";
|
|
}
|
|
|
|
(*tstream) << ");\n";
|
|
break;
|
|
}
|
|
|
|
(*tstream) << "end entity;\n"
|
|
<< "use work.all;\n"
|
|
<< "architecture Arch_Sub_" << Type << " of Sub_" << Type
|
|
<< " is\n";
|
|
|
|
if(!Signals.isEmpty()) {
|
|
QList<DigSignal> values = Signals.values();
|
|
QList<DigSignal>::const_iterator it;
|
|
for (it = values.constBegin(); it != values.constEnd(); ++it) {
|
|
(*tstream) << " signal " << (*it).Name << " : "
|
|
<< ((*it).Type.isEmpty() ?
|
|
VHDL_SIGNAL_TYPE : (*it).Type) << ";\n";
|
|
}
|
|
}
|
|
|
|
// write all equations into netlist file
|
|
for(pc = DocComps.first(); pc != 0; pc = DocComps.next()) {
|
|
if(pc->Model == "Eqn") {
|
|
ErrText->insertPlainText(
|
|
QObject::tr("WARNING: Equations in \"%1\" are 'time' typed.").
|
|
arg(pc->Name));
|
|
(*tstream) << pc->get_VHDL_Code(NumPorts);
|
|
}
|
|
}
|
|
|
|
(*tstream) << "begin\n";
|
|
|
|
if(Signals.find("gnd") != Signals.end())
|
|
(*tstream) << " gnd <= '0';\n"; // should appear only once
|
|
|
|
// write all components into netlist file
|
|
for(pc = DocComps.first(); pc != 0; pc = DocComps.next()) {
|
|
if(pc->Model != "Eqn") {
|
|
s = pc->get_VHDL_Code(NumPorts);
|
|
if(s.length()>0 && s.at(0) == '\xA7') { //section symbol
|
|
ErrText->insertPlainText(s.mid(1));
|
|
}
|
|
else (*tstream) << s;
|
|
}
|
|
}
|
|
|
|
(*tstream) << "end architecture;\n";
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
// close file
|
|
if(creatingLib) {
|
|
ofile.close();
|
|
delete tstream;
|
|
}
|
|
}
|
|
// ---------------------------------------------------
|
|
// Write the netlist as subcircuit to the text stream 'stream'.
|
|
bool Schematic::createSubNetlist(QTextStream *stream, int& countInit,
|
|
QStringList& Collect, QPlainTextEdit *ErrText, int NumPorts)
|
|
{
|
|
// int Collect_count = Collect.count(); // position for this subcircuit
|
|
|
|
// TODO: NodeSets have to be put into the subcircuit block.
|
|
if(!giveNodeNames(stream, countInit, Collect, ErrText, NumPorts)){
|
|
fprintf(stderr, "Error giving NodeNames in createSubNetlist\n");
|
|
return false;
|
|
}
|
|
|
|
/* Example for TODO
|
|
for(it = Collect.at(Collect_count); it != Collect.end(); )
|
|
if((*it).left(4) == "use ") { // output all subcircuit uses
|
|
(*stream) << (*it);
|
|
it = Collect.remove(it);
|
|
}
|
|
else it++;*/
|
|
|
|
// Emit subcircuit components
|
|
createSubNetlistPlain(stream, ErrText, NumPorts);
|
|
if (QucsSettings.DefaultSimulator != spicecompat::simQucsator) {
|
|
AbstractSpiceKernel *kern = new AbstractSpiceKernel(this);
|
|
QStringList err_lst;
|
|
if (!kern->checkSchematic(err_lst)) {
|
|
QString s = QString("Subcircuit %1 contains SPICE-incompatible components.\n"
|
|
"Check these components: %2 \n")
|
|
.arg(this->DocName).arg(err_lst.join("; "));
|
|
ErrText->insertPlainText(s);
|
|
return false;
|
|
}
|
|
kern->createSubNetlsit(*stream);
|
|
|
|
delete kern;
|
|
}
|
|
|
|
|
|
Signals.clear(); // was filled in "giveNodeNames()"
|
|
return true;
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
// Detect simulation domain (analog/digital) by looking at component types.
|
|
bool Schematic::isDigitalCircuit()
|
|
{
|
|
for(Component *pc = DocComps.first(); pc != 0; pc = DocComps.next()) {
|
|
if(pc->isActive == COMP_IS_OPEN) continue;
|
|
if(pc->Model.at(0) == '.' && pc->Model == ".Digi") {
|
|
return true; // Verilog simulation detected
|
|
}
|
|
}
|
|
return false; // Verilog simulation not found
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
// Determines the node names and writes subcircuits into netlist file.
|
|
int Schematic::prepareNetlist(QTextStream& stream, QStringList& Collect,
|
|
QPlainTextEdit *ErrText)
|
|
{
|
|
if(showBias > 0) showBias = -1; // do not show DC bias anymore
|
|
|
|
isVerilog = false;
|
|
isAnalog = true;
|
|
bool isTruthTable = false;
|
|
int allTypes = 0, NumPorts = 0;
|
|
|
|
// Detect simulation domain (analog/digital) by looking at component types.
|
|
for(Component *pc = DocComps.first(); pc != 0; pc = DocComps.next()) {
|
|
if(pc->isActive == COMP_IS_OPEN) continue;
|
|
if(pc->Model.at(0) == '.') {
|
|
if(pc->Model == ".Digi") {
|
|
if(allTypes & isDigitalComponent) {
|
|
ErrText->appendPlainText(
|
|
QObject::tr("ERROR: Only one digital simulation allowed."));
|
|
return -10;
|
|
}
|
|
if(pc->Props.getFirst()->Value != "TimeList")
|
|
isTruthTable = true;
|
|
if(pc->Props.getLast()->Value != "VHDL")
|
|
isVerilog = true;
|
|
allTypes |= isDigitalComponent;
|
|
isAnalog = false;
|
|
}
|
|
else allTypes |= isAnalogComponent;
|
|
if((allTypes & isComponent) == isComponent) {
|
|
ErrText->appendPlainText(
|
|
QObject::tr("ERROR: Analog and digital simulations cannot be mixed."));
|
|
return -10;
|
|
}
|
|
}
|
|
else if(pc->Model == "DigiSource") NumPorts++;
|
|
}
|
|
|
|
if((allTypes & isAnalogComponent) == 0) {
|
|
if(allTypes == 0) {
|
|
// If no simulation exists, assume analog simulation. There may
|
|
// be a simulation within a SPICE file. Otherwise Qucsator will
|
|
// output an error.
|
|
isAnalog = true;
|
|
allTypes |= isAnalogComponent;
|
|
NumPorts = -1;
|
|
}
|
|
else {
|
|
if(NumPorts < 1 && isTruthTable) {
|
|
ErrText->appendPlainText(
|
|
QObject::tr("ERROR: Digital simulation needs at least one digital source."));
|
|
return -10;
|
|
}
|
|
if(!isTruthTable) NumPorts = 0;
|
|
}
|
|
}
|
|
else {
|
|
NumPorts = -1;
|
|
isAnalog = true;
|
|
}
|
|
|
|
// first line is documentation
|
|
if(allTypes & isAnalogComponent) {
|
|
if (QucsSettings.DefaultSimulator != spicecompat::simQucsator)
|
|
stream << "*";
|
|
else stream << '#';
|
|
} else if (isVerilog)
|
|
stream << "//";
|
|
else
|
|
stream << "--";
|
|
stream << " Qucs " << PACKAGE_VERSION << " " << DocName << "\n";
|
|
|
|
// set timescale property for verilog schematics
|
|
if (isVerilog) {
|
|
stream << "\n`timescale 1ps/100fs\n";
|
|
}
|
|
|
|
int countInit = 0; // counts the nodesets to give them unique names
|
|
|
|
if(!giveNodeNames(&stream, countInit, Collect, ErrText, NumPorts)){
|
|
fprintf(stderr, "Error giving NodeNames\n");
|
|
return -10;
|
|
}
|
|
|
|
if(allTypes & isAnalogComponent){
|
|
return NumPorts;
|
|
}
|
|
|
|
if (!isVerilog) {
|
|
stream << VHDL_LIBRARIES;
|
|
stream << "entity TestBench is\n"
|
|
<< "end entity;\n"
|
|
<< "use work.all;\n";
|
|
}
|
|
return NumPorts;
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
// Write the beginning of digital netlist to the text stream 'stream'.
|
|
void Schematic::beginNetlistDigital(QTextStream& stream)
|
|
{
|
|
if (isVerilog) {
|
|
stream << "module TestBench ();\n";
|
|
QList<DigSignal> values = Signals.values();
|
|
QList<DigSignal>::const_iterator it;
|
|
for (it = values.constBegin(); it != values.constEnd(); ++it) {
|
|
stream << " wire " << (*it).Name << ";\n";
|
|
}
|
|
stream << "\n";
|
|
} else {
|
|
stream << "architecture Arch_TestBench of TestBench is\n";
|
|
QList<DigSignal> values = Signals.values();
|
|
QList<DigSignal>::const_iterator it;
|
|
for (it = values.constBegin(); it != values.constEnd(); ++it) {
|
|
stream << " signal " << (*it).Name << " : "
|
|
<< ((*it).Type.isEmpty() ?
|
|
VHDL_SIGNAL_TYPE : (*it).Type) << ";\n";
|
|
}
|
|
stream << "begin\n";
|
|
}
|
|
|
|
if(Signals.find("gnd") != Signals.end()) {
|
|
if (isVerilog) {
|
|
stream << " assign gnd = 0;\n";
|
|
} else {
|
|
stream << " gnd <= '0';\n"; // should appear only once
|
|
}
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
// Write the end of digital netlist to the text stream 'stream'.
|
|
void Schematic::endNetlistDigital(QTextStream& stream)
|
|
{
|
|
if (isVerilog) {
|
|
} else {
|
|
stream << "end architecture;\n";
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------
|
|
// write all components with node names into the netlist file
|
|
QString Schematic::createNetlist(QTextStream& stream, int NumPorts)
|
|
{
|
|
if(!isAnalog) {
|
|
beginNetlistDigital(stream);
|
|
}
|
|
|
|
Signals.clear(); // was filled in "giveNodeNames()"
|
|
FileList.clear();
|
|
|
|
QString s, Time;
|
|
for(Component *pc = DocComps.first(); pc != 0; pc = DocComps.next()) {
|
|
if(isAnalog) {
|
|
s = pc->getNetlist();
|
|
}
|
|
else {
|
|
if(pc->Model == ".Digi" && pc->isActive) { // simulation component ?
|
|
if(NumPorts > 0) { // truth table simulation ?
|
|
if (isVerilog)
|
|
Time = QString::number((1 << NumPorts));
|
|
else
|
|
Time = QString::number((1 << NumPorts) - 1) + " ns";
|
|
} else {
|
|
Time = pc->Props.at(1)->Value;
|
|
if (isVerilog) {
|
|
if(!misc::Verilog_Time(Time, pc->Name)) return Time;
|
|
} else {
|
|
if(!misc::VHDL_Time(Time, pc->Name)) return Time; // wrong time format
|
|
}
|
|
}
|
|
}
|
|
if (isVerilog) {
|
|
s = pc->get_Verilog_Code(NumPorts);
|
|
} else {
|
|
s = pc->get_VHDL_Code(NumPorts);
|
|
}
|
|
if (s.length()>0 && s.at(0) == '\xA7'){
|
|
return s; // return error
|
|
}
|
|
}
|
|
stream << s;
|
|
}
|
|
|
|
if(!isAnalog) {
|
|
endNetlistDigital(stream);
|
|
}
|
|
|
|
return Time;
|
|
}
|
|
|
|
|
|
void Schematic::clearSignalsAndFileList()
|
|
{
|
|
Signals.clear(); // was filled in "giveNodeNames()"
|
|
FileList.clear();
|
|
}
|
|
|
|
void Schematic::clearSignals()
|
|
{
|
|
Signals.clear();
|
|
}
|
|
// vim:ts=8:sw=2:noet
|