2015-01-19 17:58:38 +01:00
|
|
|
#include <iostream>
|
|
|
|
#include <string>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <vector>
|
2015-01-19 18:08:00 +01:00
|
|
|
#include <fstream>
|
2015-01-19 17:58:38 +01:00
|
|
|
|
|
|
|
#define DEBUG_EN DEBUG_EN
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
// Source: vsim -gui design.top(RTL)
|
|
|
|
// Target: ghdl -m --ieee=synopsys --warn-no-vital-generic --workdir=simu --work=work testb_file
|
|
|
|
// ./testb_file --stop-time=500ns --vcdgz=testb_file.vcdgz
|
|
|
|
// gunzip --stdout testb_file.vcdgz | gtkwave --vcd
|
|
|
|
|
|
|
|
int run(string args);
|
|
|
|
|
|
|
|
int run(string args) {
|
|
|
|
FILE *proc;
|
|
|
|
char buf[512];
|
|
|
|
vector < string > result;
|
|
|
|
|
|
|
|
proc = popen(args.c_str(), "r");
|
|
|
|
|
|
|
|
if (proc == NULL) {
|
|
|
|
cerr << "Error: Could not invoke GHDL/GtkWave." << endl;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
while(fgets(buf, sizeof(buf), proc)!=NULL){
|
|
|
|
cout << buf;
|
|
|
|
}
|
2015-01-19 18:37:48 +01:00
|
|
|
|
|
|
|
return pclose(proc);
|
|
|
|
}
|
|
|
|
|
|
|
|
string getSimulationTime() { // Very crude but works (for a proof-of-concept anyway^^)
|
|
|
|
FILE *proc;
|
|
|
|
char buf[512];
|
|
|
|
vector < string > result;
|
|
|
|
string defaultValue = "100ns";
|
|
|
|
|
2015-01-19 19:35:33 +01:00
|
|
|
ifstream myfile;
|
|
|
|
char fileTemp[128];
|
|
|
|
myfile.open ("/tmp/ghdl-simtime");
|
|
|
|
myfile.getline(fileTemp, sizeof(fileTemp));
|
|
|
|
myfile.close();
|
|
|
|
if (string(fileTemp) != "")
|
|
|
|
defaultValue = string(fileTemp);
|
|
|
|
|
2015-01-19 18:37:48 +01:00
|
|
|
string temp = "zenity --entry --text \"Enter the duration:\" --title \"Simulation time\" --entry-text=\"" + defaultValue + "\"";
|
|
|
|
proc = popen(temp.c_str(), "r");
|
|
|
|
|
|
|
|
if (proc == NULL) {
|
|
|
|
cerr << "Error: Could not invoke zenity." << endl;
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
while(fgets(buf, sizeof(buf), proc)!=NULL){
|
|
|
|
cout << buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
defaultValue.clear();
|
|
|
|
char *ptr = buf;
|
|
|
|
while (*ptr != '\0' && *ptr != '\n') {
|
|
|
|
defaultValue.append(" ");
|
|
|
|
defaultValue[defaultValue.length()-1] = *ptr;
|
|
|
|
ptr++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pclose(proc)) {
|
|
|
|
defaultValue = "";
|
|
|
|
}
|
2015-01-19 19:35:33 +01:00
|
|
|
else {
|
|
|
|
ofstream myfile;
|
|
|
|
myfile.open ("/tmp/ghdl-simtime");
|
|
|
|
myfile << defaultValue;
|
|
|
|
myfile.flush();
|
|
|
|
myfile.close();
|
|
|
|
}
|
2015-01-19 18:37:48 +01:00
|
|
|
|
|
|
|
return defaultValue;
|
2015-01-19 17:58:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#define ISOPT(cmd) (string(argv[i]) == cmd)
|
|
|
|
#define GETOPT(cmd) (string(argv[i]) == cmd) && (++i < argc)
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
string top = "";
|
|
|
|
string work = "";
|
2015-01-19 18:37:48 +01:00
|
|
|
string simtime = "";
|
2015-01-19 17:58:38 +01:00
|
|
|
int i;
|
|
|
|
char tempdir[256] = ""; // Compile dir
|
|
|
|
|
2015-01-19 18:08:00 +01:00
|
|
|
ifstream myfile;
|
2015-01-19 19:29:35 +01:00
|
|
|
myfile.open ("/tmp/ghdl-workdir");
|
2015-01-19 18:08:00 +01:00
|
|
|
myfile.getline(tempdir, sizeof(tempdir));
|
|
|
|
myfile.close();
|
2015-01-19 17:58:38 +01:00
|
|
|
|
|
|
|
for (i=1; i < argc; ++i) {
|
|
|
|
if (ISOPT("-gui")) {
|
|
|
|
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (argv[i][0] == '-') {
|
|
|
|
cerr << "INFO: Unknown command line opt: " << argv[i] << endl;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (; i < argc; ++i) {
|
|
|
|
top.append(argv[i]);
|
|
|
|
top.append(" ");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// -gui work.toplevel(RTL)
|
|
|
|
string temp = "";
|
|
|
|
for (unsigned int i=0; i < top.length(); ++i) {
|
|
|
|
if (top.at(i) == '.') {
|
|
|
|
work = temp;
|
|
|
|
temp = "";
|
|
|
|
}
|
|
|
|
else if (top.at(i) == '(') {
|
|
|
|
top = temp;
|
|
|
|
temp = "";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
temp.append(" ");
|
|
|
|
temp[temp.length()-1] = top.at(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG_EN
|
|
|
|
cout << "\n\nVSIM CALL PARSED:" << endl;
|
|
|
|
cout << "\twork=" << work << endl;
|
|
|
|
cout << "\ttop=" << top<< endl;
|
|
|
|
cout << "\ttempdir=" << tempdir << endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (work == "" || top == "" || string(tempdir) == "") {
|
|
|
|
cerr << "Error: Incomplete/Unsupported vsim call." << endl;
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
2015-01-19 18:08:00 +01:00
|
|
|
|
2015-01-19 18:37:48 +01:00
|
|
|
string cargs = "cd " + string(tempdir) + "; ghdl -m --ieee=synopsys --warn-no-vital-generic --workdir=" + string(tempdir) + " --work=" + work + " " + top;
|
2015-01-19 18:11:14 +01:00
|
|
|
if (run(cargs)) {
|
2015-01-19 17:58:38 +01:00
|
|
|
cerr << "Error: Compilation failed." << endl;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
cargs = "";
|
|
|
|
// ./testb_file --stop-time=500ns --vcdgz=testb_file.vcdgz
|
2015-01-19 18:37:48 +01:00
|
|
|
if (run("cd " + string(tempdir) + "; ./" + top + " --stop-time=" + getSimulationTime() + " --vcdgz=" + top + ".vcdgz")) {
|
2015-01-19 17:58:38 +01:00
|
|
|
cerr << "Error: Simulation failed." << endl;
|
|
|
|
}
|
2015-01-19 18:11:14 +01:00
|
|
|
else {
|
|
|
|
if (run("gunzip --stdout " + string(tempdir) + "/" + top + ".vcdgz | gtkwave --vcd")) {
|
|
|
|
cerr << "Error: GtkWave failed.";
|
|
|
|
}
|
|
|
|
}
|
2015-01-19 17:58:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|