2009-03-12 Stefan Jahn <stefan@lkcc.org>

* schematic_file.cpp: Some code cleanup.  Also fixed a problem
        with subcircuits containing VHDL files: passing signal types up
        also when more than one instance of the subcircuit is on the
        schematic.
This commit is contained in:
ela 2009-03-12 19:36:13 +00:00
parent dc143d46b4
commit 6bddc2855a
6 changed files with 263 additions and 179 deletions

1
NEWS
View File

@ -26,6 +26,7 @@ files.
Version 0.0.15 Version 0.0.15
-------------- --------------
* allowing VHDL files on schematic with arbitrary in/out signals
* many new digital primitives defined for analogue, VHDL and Verilog * many new digital primitives defined for analogue, VHDL and Verilog
simulations simulations
* additional install step for MacOSX and appropriate icons * additional install step for MacOSX and appropriate icons

2
TODO
View File

@ -26,7 +26,7 @@ unfixed) have a look at the file BUGS.
- different color/thickness of wires in order to visualize - different color/thickness of wires in order to visualize
different types of signals different types of signals
- busses for digital designs (instead of wires only) - busses for digital designs (instead of wires only)
- keeping input/output type of VHDL subcircuits based on VHDL files - keeping input/output type of VHDL subcircuits based on VHDL files
- real library creation using VHDL (precompiled objects as well as - real library creation using VHDL (precompiled objects as well as
libraries) libraries)
- eye-diagram implementation - eye-diagram implementation

View File

@ -1,3 +1,10 @@
2009-03-12 Stefan Jahn <stefan@lkcc.org>
* schematic_file.cpp: Some code cleanup. Also fixed a problem
with subcircuits containing VHDL files: passing signal types up
also when more than one instance of the subcircuit is on the
schematic.
2009-03-10 Stefan Jahn <stefan@lkcc.org> 2009-03-10 Stefan Jahn <stefan@lkcc.org>
* schematic_file.cpp (createSubNetlistPlain): Adjusting type * schematic_file.cpp (createSubNetlistPlain): Adjusting type

View File

@ -40,7 +40,7 @@
#include "main.h" #include "main.h"
#include "schematic.h" #include "schematic.h"
extern QStringList StringList; extern SubMap FileList;
LibraryDialog::LibraryDialog(QucsApp *App_, QListViewItem *SchematicList) LibraryDialog::LibraryDialog(QucsApp *App_, QListViewItem *SchematicList)
: QDialog(App_, 0, TRUE, Qt::WDestructiveClose) : QDialog(App_, 0, TRUE, Qt::WDestructiveClose)
@ -301,19 +301,21 @@ void LibraryDialog::slotNext()
intoStream(Stream, tmp, "Model"); intoStream(Stream, tmp, "Model");
int error = 0; int error = 0;
QStringList IFiles; QStringList IFiles;
while(!StringList.isEmpty()) { SubMap::Iterator it = FileList.begin();
QString f = StringList.last(); while(it != FileList.end()) {
StringList.remove(f); QString f = it.data().File;
QString ifn, ofn; QString ifn, ofn;
if(f.find("SCH") == 0) { if(it.data().Type == "SCH") {
ifn = f.mid(4) + ".lst"; ifn = f + ".lst";
ofn = ifn; ofn = ifn;
} else if(f.find("CIR") == 0) { } else if(it.data().Type == "CIR") {
ifn = f.mid(4) + ".lst"; ifn = f + ".lst";
ofn = ifn; ofn = ifn;
} }
error += intoFile(ifn, ofn, IFiles); error += intoFile(ifn, ofn, IFiles);
it++;
} }
FileList.clear();
if(!IFiles.isEmpty()) { if(!IFiles.isEmpty()) {
Stream << " <ModelIncludes \"" << IFiles.join("\" \"") << "\">\n"; Stream << " <ModelIncludes \"" << IFiles.join("\" \"") << "\">\n";
} }
@ -330,19 +332,21 @@ void LibraryDialog::slotNext()
intoStream(Stream, tmp, "VerilogModel"); intoStream(Stream, tmp, "VerilogModel");
int error = 0; int error = 0;
QStringList IFiles; QStringList IFiles;
while(!StringList.isEmpty()) { SubMap::Iterator it = FileList.begin();
QString f = StringList.last(); while(it != FileList.end()) {
StringList.remove(f); QString f = it.data().File;
QString ifn, ofn; QString ifn, ofn;
if(f.find("SCH") == 0) { if(it.data().Type == "SCH") {
ifn = f.mid(4) + ".lst"; ifn = f + ".lst";
ofn = f.mid(4) + ".v"; ofn = f + ".v";
} else if(f.find("VER") == 0) { } else if(it.data().Type == "VER") {
ifn = f.mid(4); ifn = f;
ofn = ifn; ofn = ifn;
} }
error += intoFile(ifn, ofn, IFiles); error += intoFile(ifn, ofn, IFiles);
it++;
} }
FileList.clear();
if(!IFiles.isEmpty()) { if(!IFiles.isEmpty()) {
Stream << " <VerilogModelIncludes \"" Stream << " <VerilogModelIncludes \""
<< IFiles.join("\" \"") << "\">\n"; << IFiles.join("\" \"") << "\">\n";
@ -360,19 +364,20 @@ void LibraryDialog::slotNext()
intoStream(Stream, tmp, "VHDLModel"); intoStream(Stream, tmp, "VHDLModel");
int error = 0; int error = 0;
QStringList IFiles; QStringList IFiles;
while(!StringList.isEmpty()) { SubMap::Iterator it = FileList.begin();
QString f = StringList.last(); while(it != FileList.end()) {
StringList.remove(f); QString f = it.data().File;
QString ifn, ofn; QString ifn, ofn;
if(f.find("SCH") == 0) { if(it.data().Type == "SCH") {
ifn = f.mid(4) + ".lst"; ifn = f + ".lst";
ofn = f.mid(4) + ".vhdl"; ofn = f + ".vhdl";
} else if(f.find("VHD") == 0) { } else if(it.data().Type == "VHD") {
ifn = f.mid(4); ifn = f;
ofn = ifn; ofn = ifn;
} }
error += intoFile(ifn, ofn, IFiles); error += intoFile(ifn, ofn, IFiles);
} }
FileList.clear();
if(!IFiles.isEmpty()) { if(!IFiles.isEmpty()) {
Stream << " <VHDLModelIncludes \"" Stream << " <VHDLModelIncludes \""
<< IFiles.join("\" \"") << "\">\n"; << IFiles.join("\" \"") << "\">\n";

View File

@ -34,6 +34,26 @@
class QTextEdit; class QTextEdit;
// digital signal data
struct DigSignal {
DigSignal() { Name=""; Type=""; }
DigSignal(const QString& _Name, const QString& _Type = "")
: Name(_Name), Type(_Type) {}
QString Name; // name
QString Type; // type of signal
};
typedef QMap<QString, DigSignal> DigMap;
// subcircuit, vhdl, etc. file structure
struct SubFile {
SubFile() { Type=""; File=""; PortTypes.clear(); }
SubFile(const QString& _Type, const QString& _File)
: Type(_Type), File(_File) { PortTypes.clear(); }
QString Type; // type of file
QString File; // file name identifier
QStringList PortTypes; // data types of in/out signals
};
typedef QMap<QString, SubFile> SubMap;
class Schematic : public QScrollView, public QucsDoc { class Schematic : public QScrollView, public QucsDoc {
Q_OBJECT Q_OBJECT
@ -235,16 +255,19 @@ private:
bool rebuildSymbol(QString *); bool rebuildSymbol(QString *);
static void createNodeSet(QStringList&, int&, Conductor*, Node*); static void createNodeSet(QStringList&, int&, Conductor*, Node*);
void throughAllNodes(bool, QStringList&, int&, bool); void throughAllNodes(bool, QStringList&, int&);
void propagateNode(QStringList&, int&, bool, Node*); void propagateNode(QStringList&, int&, Node*);
void collectDigitalSignals(void); void collectDigitalSignals(void);
bool giveNodeNames(QTextStream*, int&, QStringList&, QTextEdit*, int); bool giveNodeNames(QTextStream*, int&, QStringList&, QTextEdit*, int);
void beginNetlistDigital(QTextStream&);
void endNetlistDigital(QTextStream&);
bool throughAllComps(QTextStream *, int&, QStringList&, QTextEdit *, int);
QStringList Signals; // collecting node names for VHDL signal declarations DigMap Signals; // collecting node names for VHDL signal declarations
QStringList SignalTypes;
QStringList PortTypes; QStringList PortTypes;
public: public:
bool isAnalog;
bool isVerilog; bool isVerilog;
bool creatingLib; bool creatingLib;
}; };

View File

@ -42,7 +42,7 @@ extern QDir QucsWorkDir;
// Here the subcircuits, SPICE components etc are collected. It must be // Here the subcircuits, SPICE components etc are collected. It must be
// global to also work within the subcircuits. // global to also work within the subcircuits.
QStringList StringList; SubMap FileList;
// ------------------------------------------------------------- // -------------------------------------------------------------
@ -741,7 +741,7 @@ bool Schematic::rebuildSymbol(QString *s)
// *************************************************************** // ***************************************************************
void Schematic::createNodeSet(QStringList& Collect, int& countInit, void Schematic::createNodeSet(QStringList& Collect, int& countInit,
Conductor *pw, Node *p1) Conductor *pw, Node *p1)
{ {
if(pw->Label) if(pw->Label)
if(!pw->Label->initValue.isEmpty()) if(!pw->Label->initValue.isEmpty())
@ -751,7 +751,7 @@ void Schematic::createNodeSet(QStringList& Collect, int& countInit,
// --------------------------------------------------- // ---------------------------------------------------
void Schematic::throughAllNodes(bool User, QStringList& Collect, void Schematic::throughAllNodes(bool User, QStringList& Collect,
int& countInit, bool Analog) int& countInit)
{ {
Node *pn; Node *pn;
int z=0; int z=0;
@ -761,19 +761,20 @@ void Schematic::throughAllNodes(bool User, QStringList& Collect,
continue; // already named ? continue; // already named ?
} }
if(!User) { if(!User) {
if(Analog) pn->Name = "_net"; if(isAnalog)
else pn->Name = "net_net"; // VHDL names must not begin with '_' pn->Name = "_net";
else
pn->Name = "net_net"; // VHDL names must not begin with '_'
pn->Name += QString::number(z++); // create numbered node name pn->Name += QString::number(z++); // create numbered node name
} }
else else if(pn->State) {
if(pn->State) { continue; // already worked on
continue; // already worked on }
}
if(Analog) createNodeSet(Collect, countInit, pn, pn); if(isAnalog) createNodeSet(Collect, countInit, pn, pn);
pn->State = 1; pn->State = 1;
propagateNode(Collect, countInit, Analog, pn); propagateNode(Collect, countInit, pn);
} }
} }
@ -781,15 +782,14 @@ void Schematic::throughAllNodes(bool User, QStringList& Collect,
// Collects the signal names for digital simulations. // Collects the signal names for digital simulations.
void Schematic::collectDigitalSignals(void) void Schematic::collectDigitalSignals(void)
{ {
int i;
Node *pn; Node *pn;
for(pn = DocNodes.first(); pn != 0; pn = DocNodes.next()) { for(pn = DocNodes.first(); pn != 0; pn = DocNodes.next()) {
if((i=Signals.findIndex(pn->Name)) < 0) { // avoid redeclaration of signal DigMap::Iterator it = Signals.find(pn->Name);
Signals.append(pn->Name); if(it == Signals.end()) { // avoid redeclaration of signal
SignalTypes.append(pn->Type); Signals.insert(pn->Name, DigSignal(pn->Name, pn->Type));
} else if (!pn->Type.isEmpty()) { } else if (!pn->Type.isEmpty()) {
SignalTypes[i] = pn->Type; it.data().Type = pn->Type;
} }
} }
} }
@ -797,7 +797,7 @@ void Schematic::collectDigitalSignals(void)
// --------------------------------------------------- // ---------------------------------------------------
// Propagates the given node to connected component ports. // Propagates the given node to connected component ports.
void Schematic::propagateNode(QStringList& Collect, void Schematic::propagateNode(QStringList& Collect,
int& countInit, bool Analog, Node *pn) int& countInit, Node *pn)
{ {
bool setName=false; bool setName=false;
QPtrList<Node> Cons; QPtrList<Node> Cons;
@ -828,43 +828,23 @@ void Schematic::propagateNode(QStringList& Collect,
} }
if(setName) { if(setName) {
Cons.findRef(p2); // back to current Connection Cons.findRef(p2); // back to current Connection
if (Analog) createNodeSet(Collect, countInit, pw, pn); if (isAnalog) createNodeSet(Collect, countInit, pw, pn);
setName = false; setName = false;
} }
} }
Cons.clear(); Cons.clear();
} }
// --------------------------------------------------- // ---------------------------------------------------
// Follows the wire lines in order to determine the node names for // Goes through all schematic components and allows special component
// each component. Output into "stream", NodeSets are collected in // handling, e.g. like subcircuit netlisting.
// "Collect" and counted with "countInit". bool Schematic::throughAllComps(QTextStream *stream, int& countInit,
bool Schematic::giveNodeNames(QTextStream *stream, int& countInit,
QStringList& Collect, QTextEdit *ErrText, int NumPorts) QStringList& Collect, QTextEdit *ErrText, int NumPorts)
{ {
// delete the node names
for(Node *pn = DocNodes.first(); pn != 0; pn = DocNodes.next()) {
pn->State = 0;
if(pn->Label) {
if(NumPorts < 0)
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(NumPorts < 0)
pw->Port1->Name = pw->Label->Name;
else // avoid to use reserved VHDL words
pw->Port1->Name = "net" + pw->Label->Name;
}
bool r; bool r;
QString s; QString s;
// give the ground nodes the name "gnd", and insert subcircuits etc. // give the ground nodes the name "gnd", and insert subcircuits etc.
QPtrListIterator<Component> it(DocComps); QPtrListIterator<Component> it(DocComps);
Component *pc; Component *pc;
@ -872,30 +852,43 @@ bool Schematic::giveNodeNames(QTextStream *stream, int& countInit,
++it; ++it;
if(pc->isActive != COMP_IS_ACTIVE) continue; if(pc->isActive != COMP_IS_ACTIVE) continue;
if(NumPorts < 0) { // check analog/digital typed components
if(isAnalog) {
if((pc->Type & isAnalogComponent) == 0) { if((pc->Type & isAnalogComponent) == 0) {
ErrText->insert(QObject::tr("ERROR: Component \"%1\" has no analog model.").arg(pc->Name)); ErrText->insert(QObject::tr("ERROR: Component \"%1\" has no analog model.").arg(pc->Name));
return false; return false;
} }
} } else {
else {
if((pc->Type & isDigitalComponent) == 0) { if((pc->Type & isDigitalComponent) == 0) {
ErrText->insert(QObject::tr("ERROR: Component \"%1\" has no digital model.").arg(pc->Name)); ErrText->insert(QObject::tr("ERROR: Component \"%1\" has no digital model.").arg(pc->Name));
return false; return false;
} }
} }
// handle ground symbol
if(pc->Model == "GND") { if(pc->Model == "GND") {
pc->Ports.getFirst()->Connection->Name = "gnd"; pc->Ports.getFirst()->Connection->Name = "gnd";
continue; continue;
} }
// handle subcircuits
if(pc->Model == "Sub") { if(pc->Model == "Sub") {
QString f = "SCH:"+pc->getSubcircuitFile(); int i;
if(StringList.findIndex(f) >= 0) QString f = pc->getSubcircuitFile();
SubMap::Iterator it = FileList.find(f);
if(it != FileList.end()) {
i = 0;
// apply in/out signal types of subcircuit
for(Port *pp = pc->Ports.first(); pp; pp = pc->Ports.next(), i++) {
pp->Type = it.data().PortTypes[i];
pp->Connection->Type = pp->Type;
}
continue; // insert each subcircuit just one time continue; // insert each subcircuit just one time
StringList.append(f); }
SubFile sub = SubFile("SCH", f);
FileList.insert(f, sub);
// load subcircuit schematic
s = pc->Props.first()->Value; s = pc->Props.first()->Value;
Schematic *d = new Schematic(0, QucsWorkDir.filePath(s)); Schematic *d = new Schematic(0, QucsWorkDir.filePath(s));
if(!d->loadDocument()) { // load document if possible if(!d->loadDocument()) { // load document if possible
@ -905,18 +898,23 @@ bool Schematic::giveNodeNames(QTextStream *stream, int& countInit,
} }
d->DocName = s; d->DocName = s;
d->isVerilog = isVerilog; d->isVerilog = isVerilog;
d->isAnalog = isAnalog;
d->creatingLib = creatingLib; d->creatingLib = creatingLib;
r = d->createSubNetlist(stream, countInit, Collect, ErrText, NumPorts); r = d->createSubNetlist(stream, countInit, Collect, ErrText, NumPorts);
int in=0; i = 0;
for(Port *pp = pc->Ports.first(); pp != 0; pp = pc->Ports.next(), in++) { // save in/out signal types of subcircuit
pp->Type = d->PortTypes[in]; for(Port *pp = pc->Ports.first(); pp; pp = pc->Ports.next(), i++) {
pp->Type = d->PortTypes[i];
pp->Connection->Type = pp->Type; pp->Connection->Type = pp->Type;
} }
sub.PortTypes = d->PortTypes;
FileList.replace(f, sub);
delete d; delete d;
if(!r) return false; if(!r) return false;
continue; continue;
} }
// handle library symbols
if(pc->Model == "Lib") { if(pc->Model == "Lib") {
if(creatingLib) { if(creatingLib) {
ErrText->insert( ErrText->insert(
@ -924,19 +922,19 @@ bool Schematic::giveNodeNames(QTextStream *stream, int& countInit,
arg(pc->Name)); arg(pc->Name));
continue; continue;
} }
s = "LIB:" + pc->getSubcircuitFile(); s = pc->getSubcircuitFile() + "/" + pc->Props.at(1)->Value;
s += "/" + pc->Props.next()->Value; SubMap::Iterator it = FileList.find(s);
if(StringList.findIndex(s) >= 0) if(it != FileList.end())
continue; // insert each subcircuit just one time continue; // insert each library subcircuit just one time
StringList.append(s); FileList.insert(s, SubFile("LIB", s));
if(NumPorts < 0) if(isAnalog)
r = ((LibComp*)pc)->createSubNetlist(stream, StringList, 1); r = ((LibComp*)pc)->createSubNetlist(stream, Collect, 1);
else { else {
if(isVerilog) if(isVerilog)
r = ((LibComp*)pc)->createSubNetlist(stream, StringList, 4); r = ((LibComp*)pc)->createSubNetlist(stream, Collect, 4);
else else
r = ((LibComp*)pc)->createSubNetlist(stream, StringList, 2); r = ((LibComp*)pc)->createSubNetlist(stream, Collect, 2);
} }
if(!r) { if(!r) {
ErrText->insert( ErrText->insert(
@ -947,6 +945,7 @@ bool Schematic::giveNodeNames(QTextStream *stream, int& countInit,
continue; continue;
} }
// handle SPICE subcircuit components
if(pc->Model == "SPICE") { if(pc->Model == "SPICE") {
s = pc->Props.first()->Value; s = pc->Props.first()->Value;
if(s.isEmpty()) { if(s.isEmpty()) {
@ -954,17 +953,11 @@ bool Schematic::giveNodeNames(QTextStream *stream, int& countInit,
arg(pc->Name)); arg(pc->Name));
return false; return false;
} }
QString f = "CIR:"+pc->getSubcircuitFile(); QString f = pc->getSubcircuitFile();
if(StringList.findIndex(f) >= 0) SubMap::Iterator it = FileList.find(f);
if(it != FileList.end())
continue; // insert each spice component just one time continue; // insert each spice component just one time
StringList.append(f); FileList.insert(f, SubFile("CIR", f));
#if 0
s += '"'+pc->Props.next()->Value;
if(pc->Props.next()->Value == "yes") s = "SPICE \""+s;
else s = "SPICEo\""+s;
Collect.append(s);
#endif
SpiceFile *sf = (SpiceFile*)pc; SpiceFile *sf = (SpiceFile*)pc;
r = sf->createSubNetlist(stream); r = sf->createSubNetlist(stream);
@ -973,6 +966,7 @@ bool Schematic::giveNodeNames(QTextStream *stream, int& countInit,
continue; continue;
} }
// handle digital file subcircuits
if(pc->Model == "VHDL" || pc->Model == "Verilog") { if(pc->Model == "VHDL" || pc->Model == "Verilog") {
if(isVerilog && pc->Model == "VHDL") if(isVerilog && pc->Model == "VHDL")
continue; continue;
@ -986,10 +980,11 @@ bool Schematic::giveNodeNames(QTextStream *stream, int& countInit,
return false; return false;
} }
QString f = pc->getSubcircuitFile(); QString f = pc->getSubcircuitFile();
f = ((pc->Model == "VHDL") ? "VHD:" : "VER:") + f; SubMap::Iterator it = FileList.find(f);
if(StringList.findIndex(f) >= 0) if(it != FileList.end())
continue; // insert each vhdl/verilog component just one time continue; // insert each vhdl/verilog component just one time
StringList.append(f); QString s = ((pc->Model == "VHDL") ? "VHD" : "VER");
FileList.insert(f, SubFile(s, f));
if(pc->Model == "VHDL") { if(pc->Model == "VHDL") {
VHDL_File *vf = (VHDL_File*)pc; VHDL_File *vf = (VHDL_File*)pc;
@ -1006,14 +1001,48 @@ bool Schematic::giveNodeNames(QTextStream *stream, int& countInit,
continue; 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, QTextEdit *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))
return false;
// work on named nodes first in order to preserve the user given names // work on named nodes first in order to preserve the user given names
throughAllNodes(true, Collect, countInit, NumPorts<0); throughAllNodes(true, Collect, countInit);
// give names to the remaining (unnamed) nodes // give names to the remaining (unnamed) nodes
throughAllNodes(false, Collect, countInit, NumPorts<0); throughAllNodes(false, Collect, countInit);
if(NumPorts>=0) // collect all node names for VHDL signal declaration if(!isAnalog) // collect all node names for VHDL signal declaration
collectDigitalSignals(); collectDigitalSignals();
return true; return true;
@ -1026,9 +1055,8 @@ bool Schematic::createLibNetlist(QTextStream *stream, QTextEdit *ErrText,
int countInit = 0; int countInit = 0;
QStringList Collect; QStringList Collect;
Collect.clear(); Collect.clear();
StringList.clear(); FileList.clear();
Signals.clear(); Signals.clear();
SignalTypes.clear();
// Apply node names and collect subcircuits and file include // Apply node names and collect subcircuits and file include
creatingLib = true; creatingLib = true;
@ -1040,7 +1068,7 @@ bool Schematic::createLibNetlist(QTextStream *stream, QTextEdit *ErrText,
// Marking start of actual top-level subcircuit // Marking start of actual top-level subcircuit
QString c; QString c;
if(NumPorts >= 0) { if(!isAnalog) {
if (isVerilog) if (isVerilog)
c = "///"; c = "///";
else else
@ -1053,7 +1081,6 @@ bool Schematic::createLibNetlist(QTextStream *stream, QTextEdit *ErrText,
createSubNetlistPlain(stream, ErrText, NumPorts); createSubNetlistPlain(stream, ErrText, NumPorts);
Signals.clear(); // was filled in "giveNodeNames()" Signals.clear(); // was filled in "giveNodeNames()"
SignalTypes.clear();
return true; return true;
} }
@ -1066,7 +1093,7 @@ bool Schematic::createLibNetlist(QTextStream *stream, QTextEdit *ErrText,
void Schematic::createSubNetlistPlain(QTextStream *stream, QTextEdit *ErrText, void Schematic::createSubNetlistPlain(QTextStream *stream, QTextEdit *ErrText,
int NumPorts) int NumPorts)
{ {
int i, z, r; int i, z;
QString s; QString s;
QStringList SubcircuitPortNames; QStringList SubcircuitPortNames;
QStringList SubcircuitPortTypes; QStringList SubcircuitPortTypes;
@ -1108,14 +1135,14 @@ void Schematic::createSubNetlistPlain(QTextStream *stream, QTextEdit *ErrText,
it_name = SubcircuitPortNames.at(i-1); it_name = SubcircuitPortNames.at(i-1);
it_type = SubcircuitPortTypes.at(i-1); it_type = SubcircuitPortTypes.at(i-1);
(*it_name) = pc->Ports.getFirst()->Connection->Name; (*it_name) = pc->Ports.getFirst()->Connection->Name;
r = Signals.findIndex(*it_name); DigMap::Iterator it = Signals.find(*it_name);
(*it_type) = SignalTypes[r]; (*it_type) = it.data().Type;
// propagate type to port symbol // propagate type to port symbol
pc->Ports.getFirst()->Connection->Type = SignalTypes[r]; pc->Ports.getFirst()->Connection->Type = *it_type;
if(NumPorts >= 0) { if(!isAnalog) {
if (isVerilog) { if (isVerilog) {
Signals.remove(Signals.find(*it_name)); // remove node name Signals.erase(*it_name); // remove node name
switch(pc->Props.at(1)->Value.at(0).latin1()) { switch(pc->Props.at(1)->Value.at(0).latin1()) {
case 'a': case 'a':
InOutPorts.append(*it_name); InOutPorts.append(*it_name);
@ -1129,16 +1156,13 @@ void Schematic::createSubNetlistPlain(QTextStream *stream, QTextEdit *ErrText,
} }
else { else {
// remove node name of output port // remove node name of output port
r = Signals.findIndex(*it_name); Signals.erase(*it_name);
Signals.remove(Signals.at(r));
SignalTypes.remove(SignalTypes.at(r));
switch(pc->Props.at(1)->Value.at(0).latin1()) { switch(pc->Props.at(1)->Value.at(0).latin1()) {
case 'a': case 'a':
(*it_name) += " : inout"; // attribute "analog" is "inout" (*it_name) += " : inout"; // attribute "analog" is "inout"
break; break;
case 'o': case 'o': // output ports need workaround
Signals.append(*it_name); // output ports need workaround Signals.insert(*it_name, DigSignal(*it_name, *it_type));
SignalTypes.append(*it_type);
(*it_name) = "net_out" + (*it_name); (*it_name) = "net_out" + (*it_name);
// no "break;" here !!! // no "break;" here !!!
default: default:
@ -1169,7 +1193,7 @@ void Schematic::createSubNetlistPlain(QTextStream *stream, QTextEdit *ErrText,
QString Type = properName(f); QString Type = properName(f);
Painting *pi; Painting *pi;
if(NumPorts < 0) { if(isAnalog) {
// ..... analog subcircuit ................................... // ..... analog subcircuit ...................................
(*tstream) << "\n.Def:" << Type << " " << SubcircuitPortNames.join(" "); (*tstream) << "\n.Def:" << Type << " " << SubcircuitPortNames.join(" ");
for(pi = SymbolPaints.first(); pi != 0; pi = SymbolPaints.next()) for(pi = SymbolPaints.first(); pi != 0; pi = SymbolPaints.next())
@ -1202,12 +1226,16 @@ void Schematic::createSubNetlistPlain(QTextStream *stream, QTextEdit *ErrText,
(*tstream) << " output " << OutPorts.join(", ") << ";\n"; (*tstream) << " output " << OutPorts.join(", ") << ";\n";
if(!InOutPorts.isEmpty()) if(!InOutPorts.isEmpty())
(*tstream) << " inout " << InOutPorts.join(", ") << ";\n"; (*tstream) << " inout " << InOutPorts.join(", ") << ";\n";
if(!Signals.isEmpty()) if(!Signals.isEmpty()) {
(*tstream) << " wire " << Signals.join(",\n ") QValueList<DigSignal> values = Signals.values();
<< ";\n"; QValueList<DigSignal>::iterator it;
for (it = values.begin(); it != values.end(); ++it) {
(*tstream) << " wire " << (*it).Name << ";\n";
}
}
(*tstream) << "\n"; (*tstream) << "\n";
if(Signals.findIndex("gnd") >= 0) if(Signals.find("gnd") != Signals.end())
(*tstream) << " assign gnd = 0;\n"; // should appear only once (*tstream) << " assign gnd = 0;\n"; // should appear only once
// write all components into netlist file // write all components into netlist file
@ -1226,17 +1254,18 @@ void Schematic::createSubNetlistPlain(QTextStream *stream, QTextEdit *ErrText,
<< "architecture Arch_Sub_" << Type << " of Sub_" << Type << "architecture Arch_Sub_" << Type << " of Sub_" << Type
<< " is\n"; << " is\n";
if(!Signals.isEmpty()) { if(!Signals.isEmpty()) {
for(unsigned int j=0; j<Signals.count(); j++) { QValueList<DigSignal> values = Signals.values();
(*tstream) << " signal " << Signals[j] << " : " QValueList<DigSignal>::iterator it;
<< (SignalTypes[j].isEmpty() ? for (it = values.begin(); it != values.end(); ++it) {
VHDL_SIGNAL_TYPE : SignalTypes[j]) (*tstream) << " signal " << (*it).Name << " : "
<< ";\n"; << ((*it).Type.isEmpty() ?
VHDL_SIGNAL_TYPE : (*it).Type) << ";\n";
} }
} }
(*tstream) << "begin\n"; (*tstream) << "begin\n";
if(Signals.findIndex("gnd") >= 0) if(Signals.find("gnd") != Signals.end())
(*tstream) << " gnd <= '0';\n"; // should appear only once (*tstream) << " gnd <= '0';\n"; // should appear only once
// write all components into netlist file // write all components into netlist file
@ -1277,7 +1306,6 @@ bool Schematic::createSubNetlist(QTextStream *stream, int& countInit,
createSubNetlistPlain(stream, ErrText, NumPorts); createSubNetlistPlain(stream, ErrText, NumPorts);
Signals.clear(); // was filled in "giveNodeNames()" Signals.clear(); // was filled in "giveNodeNames()"
SignalTypes.clear();
return true; return true;
} }
@ -1286,11 +1314,13 @@ bool Schematic::createSubNetlist(QTextStream *stream, int& countInit,
int Schematic::prepareNetlist(QTextStream& stream, QStringList& Collect, int Schematic::prepareNetlist(QTextStream& stream, QStringList& Collect,
QTextEdit *ErrText) QTextEdit *ErrText)
{ {
if(showBias > 0) showBias = -1; // do not show DC bias anymore if(showBias > 0) showBias = -1; // do not show DC bias anymore
isVerilog = false; isVerilog = false;
isAnalog = true;
bool isTruthTable = false; bool isTruthTable = false;
int allTypes = 0, NumPorts = 0; int allTypes = 0, NumPorts = 0;
// Detect simulation domain (analog/digital) by looking at component types. // Detect simulation domain (analog/digital) by looking at component types.
for(Component *pc = DocComps.first(); pc != 0; pc = DocComps.next()) { for(Component *pc = DocComps.first(); pc != 0; pc = DocComps.next()) {
if(pc->isActive == COMP_IS_OPEN) continue; if(pc->isActive == COMP_IS_OPEN) continue;
@ -1306,6 +1336,7 @@ int Schematic::prepareNetlist(QTextStream& stream, QStringList& Collect,
if(pc->Props.getLast()->Value != "VHDL") if(pc->Props.getLast()->Value != "VHDL")
isVerilog = true; isVerilog = true;
allTypes |= isDigitalComponent; allTypes |= isDigitalComponent;
isAnalog = false;
} }
else allTypes |= isAnalogComponent; else allTypes |= isAnalogComponent;
@ -1315,32 +1346,31 @@ int Schematic::prepareNetlist(QTextStream& stream, QStringList& Collect,
return -10; return -10;
} }
} }
else if(pc->Model == "DigiSource") NumPorts++; else if(pc->Model == "DigiSource") NumPorts++;
} }
if((allTypes & isAnalogComponent) == 0) { if((allTypes & isAnalogComponent) == 0) {
if(allTypes == 0) { if(allTypes == 0) {
/* ErrText->insert(
QObject::tr("ERROR: No simulation specified on this page."));
return -10;*/
// If no simulation exists, assume analog simulation. There may // If no simulation exists, assume analog simulation. There may
// be a simulation within a SPICE file. Otherwise Qucsator will // be a simulation within a SPICE file. Otherwise Qucsator will
// output an error. // output an error.
isAnalog = true;
allTypes |= isAnalogComponent; allTypes |= isAnalogComponent;
NumPorts = -1; NumPorts = -1;
} }
else { else {
if(NumPorts < 1) { if(NumPorts < 1 && isTruthTable) {
ErrText->insert( ErrText->insert(
QObject::tr("ERROR: Digital simulation needs at least one digital source.")); QObject::tr("ERROR: Digital simulation needs at least one digital source."));
return -10; return -10;
} }
if(!isTruthTable) NumPorts = 0; if(!isTruthTable) NumPorts = 0;
} }
} }
else NumPorts = -1; else {
NumPorts = -1;
isAnalog = true;
}
// first line is documentation // first line is documentation
if(allTypes & isAnalogComponent) if(allTypes & isAnalogComponent)
@ -1372,41 +1402,63 @@ int Schematic::prepareNetlist(QTextStream& stream, QStringList& Collect,
return NumPorts; return NumPorts;
} }
// ................................................. // ---------------------------------------------------
// Write the beginning of digital netlist to the text stream 'stream'.
void Schematic::beginNetlistDigital(QTextStream& stream)
{
if (isVerilog) {
stream << "module TestBench ();\n";
QValueList<DigSignal> values = Signals.values();
QValueList<DigSignal>::iterator it;
for (it = values.begin(); it != values.end(); ++it) {
stream << " wire " << (*it).Name << ";\n";
}
stream << "\n";
} else {
stream << "architecture Arch_TestBench of TestBench is\n";
QValueList<DigSignal> values = Signals.values();
QValueList<DigSignal>::iterator it;
for (it = values.begin(); it != values.end(); ++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 // write all components with node names into the netlist file
QString Schematic::createNetlist(QTextStream& stream, int NumPorts) QString Schematic::createNetlist(QTextStream& stream, int NumPorts)
{ {
if(NumPorts >= 0) { if(!isAnalog) {
if (isVerilog) { beginNetlistDigital(stream);
stream << "module TestBench ();\n"
<< " wire " << Signals.join(",\n ")
<< ";\n\n";
} else {
stream << "architecture Arch_TestBench of TestBench is\n";
for(unsigned int i=0; i<Signals.count(); i++) {
stream << " signal " << Signals[i] << " : "
<< (SignalTypes[i].isEmpty() ?
VHDL_SIGNAL_TYPE : SignalTypes[i])
<< ";\n";
}
stream << "begin\n";
}
if(Signals.findIndex("gnd") >= 0) {
if (isVerilog) {
stream << " assign gnd = 0;\n";
} else {
stream << " gnd <= '0';\n"; // should appear only once
}
}
} }
Signals.clear(); // was filled in "giveNodeNames()" Signals.clear(); // was filled in "giveNodeNames()"
SignalTypes.clear(); FileList.clear();
StringList.clear();
QString s, Time; QString s, Time;
for(Component *pc = DocComps.first(); pc != 0; pc = DocComps.next()) { for(Component *pc = DocComps.first(); pc != 0; pc = DocComps.next()) {
if(NumPorts < 0) { if(isAnalog) {
s = pc->getNetlist(); s = pc->getNetlist();
} }
else { else {
@ -1430,17 +1482,13 @@ QString Schematic::createNetlist(QTextStream& stream, int NumPorts)
} else { } else {
s = pc->get_VHDL_Code(NumPorts); s = pc->get_VHDL_Code(NumPorts);
} }
if(s.at(0) == '§') return s; // return error if(s.at(0) == '§') return s; // return error
} }
stream << s; stream << s;
} }
if(NumPorts >= 0) { if(!isAnalog) {
if (isVerilog) { endNetlistDigital(stream);
} else {
stream << "end architecture;\n";
}
} }
return Time; return Time;