# IIA GF2 Software: Final Report

Jamie Magee (jam96) Team 8 Gonville & Caius College

## Contents

| 1            | Description                               | 2          |
|--------------|-------------------------------------------|------------|
| 2            | Development Style                         | 2          |
| 3            | My Contribution                           | 3          |
| 4            | Testing                                   | 3          |
| 5            | Conclusions                               | 3          |
| A            | Code Listings                             | 5          |
|              | A.1 Names Class                           | 5<br>5     |
|              | A.1.2 names.cc                            | 5<br>7     |
|              | A.2.1 scanner.h                           | 7<br>7     |
|              | A.3 Parser Class                          | 10         |
|              | A.4 Test Scripts                          | 10<br>19   |
|              |                                           | 19<br>19   |
| В            | Test Definition Files                     | 20         |
|              | B.1 XOR Gate                              | 20         |
|              | B.1.1 Definition File                     | 20         |
|              | B.1.2 Circuit Diagram                     | 20         |
|              | B.2 4-bit Adder                           | 20         |
|              | B.2.1 Definition File                     | 20         |
|              | B.2.2 Circuit Diagram                     | 22         |
|              | B.3 Serial In Parallel Out Shift Register | 22         |
|              | B.3.1 Definition File                     | 22         |
|              | B.3.2 Circuit Diagram                     | 23         |
|              | B.4 Gated D Latch                         | 23         |
|              |                                           | 23         |
|              |                                           | 23         |
|              |                                           |            |
|              |                                           |            |
|              | B.5.2 Circuit Diagram                     | 24         |
| $\mathbf{C}$ | EBNF                                      | 25         |
| D            | User Guide                                | <b>2</b> 6 |
| $\mathbf{E}$ | File Listing                              | 27         |
| F            | Gantt Chart                               | 28         |

## 1 Description

Our logic simulator is able to simulate any number of circuits which include the following devices:

- Clocks
- Switches
- AND gates (Up to 16 inputs)
- NAND gates (Up to 16 inputs)
- OR gates (Up to 16 inputs)
- NOR gates (Up to 16 inputs)
- XOR gates
- D-Type flip-flops
- Signal generators

It uses files written in the language as specified in our EBNF in Appendix C, with the file extension .gf2 and have MIME type text/plain. Errors in a definition file are reported fully to the user upon reading a definition file, and give unique error codes for each different type of error. There is no limit on the number of devices in a network (except for that implied by the available memory of the computer). There may also be an unlimited number of monitors. For a complete guide on how to operate our logic simulator, please see the user guide in Appendix D.

For the structure of our logic simulator we made use of modularisation through the use of object oriented programming. The logic circuit is internally represented using the network class, in addition to the devices class which stores details about the properties of each device. When a file is opened, it is read by the parser class, by creating an instance of the scanner class

## 2 Development Style

We split the development of our logic simulator into five major phases: specification, design, implementation, testing and maintenance. The timeframe was then decided for each task and each task was assigned to either a team member or the whole team, depending on the nature of the task. A Gantt chart showing our time planning can be seen in Appendix F.

Each member of the team was also assigned a general project role as follows:

**Project manager:** (T Hillel) - Responsible for project planning including delegation of tasks and ensuring that the project runs to the set timescale.

**Programming administrator:** (J Magee) - Responsible for upkeep of the project directory including performing builds and keeping legacy versions of the simulator.

Client representative: (M Jackson) - Responsible for ensuring that the project meets the client's requirements for the logic simulator as defined in Appendix A of the GF2 Project Handout.

We made significant use of git for revision control, as well as GitHub for tracking bugs in the software and features required by the client. It also allowed us to work on the same file independently and merge our changes using the git merge command.

During the maintenance phase of the project we divided up the tasks and tackled them independently. They

were assigned as follows:

Non-zero Hold Time of Bistables: J Magee

Signal Generators: T Hillel

Continuous Simulation: M Jackson

While the tasks were split up, and the person assigned to the task contributed the majority of the code, the power of git allowed other members of the team to contribute easily.

Overall I believe our team worked efficiently, and as a result we were able to achieve all the requirements of the client within the deadlines we were set.

## 3 My Contribution

I took on responsibility for writing the names and scanner classes. In addition, I wrote approximately 25% of the parser class. the names class stores a list of all the words used within a definition file, and methods to manipulate them. It is initialised with only the reserved words, but can be populated as a definition file is read, the scanner class reads through the definition file, character by character, and is able to return complete symbols to the parser. It is able to return the internal representation of a symbol, the type of symbol and optionally the value. The parser class analyses the definition file as it is read in according to the rules laid out in our EBNF. It is then able to create devices, connections and monitors that are laid out in the definition file. I wrote the newConnection, monitorList and newMonitor methods. The newConnection and newMonitor methods deal with reading in a connection and monitor respectively, and creating them using the network and monitor classes respectively. The monitorList class deals with reading in an entire MONITOR block, as defined by our EBNF, which can be seen in Appendix C

Once the majority of the software was written, I designed a definition file for each error and warning our logic simulator can throw and then wrote a shell script which would attempt to run each definition file, and record the output from our logic simulator. In addition to testing for errors, I wrote a shell script which would run through each of our test definition files, in Appendix B, and record the output. Both the shell scripts can be found in Appendix A.4.

During the maintenance phase of the project, I undertook the task to implement a non-zero hold time for d-type flip flops, as well as randomising the order of execution of devices. the non-deterministic characteristics of d-types required editing of the devices class, specifically the **executype** method. Randomising the order of execution of devices was implemented in the network class, and required reading the values from the existing devices list, shuffling the list (while still retaining clocks at the end of the list) and writing it back.

## 4 Testing

We used two main tests of testing - unit and system testing - both of which are industry standard practices. For our unit testing, Martin wrote an errors class which compared the actual output from various units of code, to the expected output. For system testing I wrote a shell script which passed definition files to the logic simulator and recorded the output in a text file. There were two variations on the shell script: One which ran known good definition files and therefore had to input the commands to run the simulation in addition to recording the output; Another which ran known bad definition files and only expected parsing errors which it recorded.

## 5 Conclusions

Overall I believe that we worked well as a team, and were able to achieve the requirements set out by the client.

If we had more time I would improve our logic simulator by:

- $\bullet$  Implementing a WYSIWYG GUI
- $\bullet$  Improving the parser to allow simple error correction
- $\bullet\,$  Expand the device library to include JK flip-flops

## A Code Listings

#### A.1 Names Class

#### A.1.1 names.h

```
#ifndef names_h
  #define names_h
  #include <string>
  #include <vector>
  using namespace std;
  //const int maxnames = 200; /* max number of distinct names */ //const int maxlength = 8; /* max chars in a name string */ const int blankname = -1; /* special name */
   const int lastreservedname = 34;
   typedef int name;
15
   typedef string namestring;
   typedef unsigned int length;
18
19
   class names
20
21
22
     private:
       vector<namestring> namelist; //Stores a list of reserved and declared names
23
24
25
       name lookup(namestring str);
26
       /* Returns the internal representation of the name given in character
27
       /* form. If the name is not already in the name table, it is
28
       /* automatically inserted.
29
30
31
       name cvtname(namestring str);
       /* Returns the internal representation of the name given in character
32
       /* form. If the name is not in the name table then 'blankname' is
       /* returned.
34
35
       void writename(name id);
       /* Prints out the given name on the console
37
38
       int namelength(name id);
39
       /* Returns length ie number of characters in given name
40
41
       namestring getnamestring (name id);
42
       /* Returns the namestring for the given name
43
       names (void):
45
       /st names initialises the name table. This procedure is called at
46
       /* system initialisation before any of the above procedures/functions
47
       /* are used.
48
  };
50
  \#endif /* names_h */
```

Listing 1: names.h

#### A.1.2 names.cc

```
#include "names.h"
#include <iostream>
#include <string>
#include <cstdlib>

using namespace std;

/* Name storage and retrieval routines */
```

```
names::names(void) /* the constructor */
10
11
       //Populate namelist with reserved words
12
      namelist.push_back("DEVICES"); //0
namelist.push_back("CONNECTIONS");
namelist.push_back("MONITORS"); //2
13
14
      namelist.push_back("END"); //3
namelist.push_back("CLOCK"); //4
namelist.push_back("SWITCH"); //5
namelist.push_back("AND"); //6
16
17
18
19
       namelist.push_back("NAND"); //7
namelist.push_back("OR"); //8
namelist.push_back("NOR"); //9
20
21
22
      namelist.push_back("DTYPE"); //10
namelist.push_back("XOR"); //11
namelist.push_back("SIGGEN"); //12
23
24
       namelist.push_back("I1"); //13
namelist.push_back("I2"); //14
26
27
       namelist.push_back("I3"); //15
28
      namelist.push_back("I4"); //16
namelist.push_back("I5"); //17
29
30
       namelist.push_back("I6"); //18
31
      namelist.push_back("I7"); //19
namelist.push_back("I8"); //20
namelist.push_back("I9"); //21
33
34
       namelist.push_back("I10"); //22
35
      namelist.push_back("I11"); //23
namelist.push_back("I12"); //24
36
       namelist.push_back("I13"); //25
       namelist.push_back("I14"); //26
namelist.push_back("I15"); //27
39
40
       namelist.push_back("I16"); //28
41
       namelist.push_back("DATA"); //29
namelist.push_back("CLK"); //30
42
43
       namelist.push_back("SET"); //31
44
      namelist.push_back("CLEAR"); //32
namelist.push_back("Q"); //33
namelist.push_back("QBAR"); //34
45
46
47
48
49
   name names::lookup(namestring str)
50
51
52
       if (cvtname(str) == blankname)
         namelist.push\_back(str); \hspace{0.3cm} // Insert \hspace{0.1cm} new \hspace{0.1cm} string
54
         return namelist.size() - 1; //Return new strings internal name
55
56
       else
57
58
       {
59
         return cvtname(str);
60
61
62
   name names::cvtname(namestring str)
63
64
       if (str == "") return blankname;
65
       for (name id = 0; id < namelist.size(); id++)
66
67
         if (namelist[id] == str) return id; //Linear search of namelist vector
68
69
70
       return blankname;
71
72
73
    void names::writename(name id)
74
       if (id == blankname) cout << "blankname";</pre>
       else if (id > blankname && id < namelist.size()) cout << namelist[id];
76
       else cout << "Incorrect id";
77
78
79
    int names::namelength(name id)
80
81
       if (id > blankname && id < namelist.size()) return namelist[id].length();
82
       else return blankname;
83
```

```
namestring names::getnamestring(name id)

namestring names::getnamestring(name id)

if (id > blankname && id < namelist.size()) return namelist[id];

else return "";
```

Listing 2: names.cc

#### A.2 Scanner Class

#### A.2.1 scanner.h

```
#ifndef scanner_h
  #define scanner_h
  #include <string>
  #include <iostream>
  #include <fstream>
  #include <cstdlib>
  #include "names.h"
  using namespace std;
  typedef int name;
11
  typedef enum {namesym, numsym, devsym, consym, monsym, endsym, classsym, iosym, colon, semicol
      , equals, dot, badsym, eofsym} symbol;
14
  class scanner
    public:
16
17
      symbol s;
      names* nmz; // Pointer to instance of names class
18
19
20
      scanner (names* names_mod,
                                    //Pointer to names class
           const char* defname, //Name of file being read
21
                             //True of file has been opened correctly
           bool& ok);
      ~scanner();
                             //Destructor
23
      void getsymbol(symbol&s, //Symbol type read
24
                               //Return symbol name (if it has one)
                name& id,
25
                int& num,
26
                string& numstring); //Return symbol value (if it's a number)
27
      void writelineerror();
28
29
30
    private:
      ifstream inf; //Input file
31
                    //Current input character
32
      char curch;
                    //Previous input character. Used for finding line end
      char prevch;
      bool eofile;
                     //True for end of file
34
                 //True if the file has been opened correctly
35
      bool ok;
      int linenum; //Number of lines in definition file
36
      int cursymlen; //Length of current symbol. Used for error printing
37
      string line; //Current line contents. Used for error printing
38
39
      void getch(); //Gets next input character
40
      void getnumber(int& number, string& numstring); //Reads number from file
41
      void getname(name& id); //Reads name from file
42
      string getline(); //Reads the line
43
      void skipspaces(); //Skips spaces
44
      void skipcomments(); //Skips comments
45
46
47
  #endif
```

Listing 3: scanner.h

#### A.2.2 scanner.cc

```
#include <iostream>
2 #include "scanner.h"
```

```
using namespace std;
  scanner::scanner(names* names_mod, const char* defname, bool& ok)
    nmz = names\_mod;
    ok = 1;
     inf.open(defname); //Open file
10
     if (!inf)
11
12
       cout << "Error: cannot open file for reading" << endl;</pre>
13
      ok = 0;
14
15
     eofile = (inf.get(curch) == 0); //Get first character
16
17
    linenum = 1;
     s = badsym; //in case getline is called before getsymbol
18
19
20
  scanner:: ~ scanner()
21
22
     inf.close(); //Close file
23
24
25
   void scanner::getsymbol(symbol&s, name&id, int&num, string&numstring)
26
27
  {
    s = badsym;
28
29
     cursymlen = 0;
30
     skipspaces();
     if (eofile) s = eofsym;
31
32
     else
33
     {
       if (isdigit(curch))
34
35
       {
         s = numsvm;
36
37
         getnumber(num, numstring);
38
       else
39
40
         if (isalpha(curch))
41
42
         {
43
           getname(id);
           if (id = 0) s = devsym;
44
           else if (id == 1) s = consym;
45
46
           else if (id == 2) s = monsym;
           else if (id == 3) s = endsym;
47
48
           else if (id > 3 && id < 13) s = classsym;
           else if (id > 12 && id < lastreservedname+1) s = iosym;
49
50
           else s = namesym;
51
         }
         else
           switch (curch)
54
             case '=':
56
               s = equals;
57
                getch();
58
59
               break;
             case ';':
60
61
                s = semicol;
                getch();
62
                break;
63
             case ': ':
64
               s = colon;
65
66
                getch();
67
                break;
             case '.'
68
                s = dot;
69
                getch();
70
                break;
             case '/
72
73
                getch();
                if (curch == '*')
74
75
                  getch();
76
                  skipcomments();
77
                  getsymbol(s, id, num, numstring);
```

```
break:
80
               default:
81
                 s = badsym;
82
                 getch();
83
                 break;
85
             cursymlen = 1;
 86
87
88
89
      }
90
   }
91
   void scanner::writelineerror()
93
94
      string errorptr;
      for (int i = 0; i < ((int) line.length() - cursymlen); <math>i++)
95
96
        \verb|errorptr.push_back(', ')|;
97
98
      errorptr.push_back('^');
cout << "Line " << linenum << ":" << endl;</pre>
99
100
      cout << getline() << endl;</pre>
                                       //Outputs current line
      cout << errorptr << endl; //Outputs a caret at the error
103
104
   void scanner::getch()
106
      prevch = curch;
      eofile = (inf.get(curch) == 0); //get next character
108
      if (prevch = '\n') //If eoline, clear the currently stored line
110
        linenum++;
111
        line.clear();
112
113
      else if (prevch != '\r') //If we're not at the end of a line, add the char to the line
114
        string
115
        line.push_back(prevch);
116
117
      }
118
119
   void scanner::getnumber(int& number, string& numstring)
120
121
   {
      numstring \, = \, "" \, ;
123
      number = 0;
      cursymlen = 0;
124
      while (isdigit (curch) && !eofile)
125
126
        {\tt numstring.push\_back(curch);}
        number *= 10;
128
        number += (int(curch) - int('0'));
        cursymlen++;
130
131
        getch();
   }
134
   void scanner::getname(name& id)
136
      namestring str;
137
      cursymlen = 0;
138
      while (isalnum(curch) && !eofile)
139
140
        str.push_back(curch);
141
142
        cursymlen++;
        getch();
143
144
      id = nmz - > lookup(str);
145
   }
146
147
   void scanner::skipspaces()
148
149
      while (isspace(curch) || curch == '\n')
150
        getch();
152
        if (eofile) break;
```

```
154
156
   void scanner::skipcomments()
157
158
      while (!(prevch == '* ' && curch == '/'))
159
        getch();
161
        if (eofile)
162
          cout << "Reached end of file before comment was terminated" << endl;</pre>
164
167
     getch(); //Get to next useful char
168
169
170
   string scanner::getline()
172
      if (s != semicol)
174
        while (curch != '; ' && ! eofile && curch != '\n')
175
176
          getch();
177
178
        if (curch != '\n' && curch != '\r' && !eofile)
179
180
          line.push_back(curch);
181
182
183
     return line;
184
```

Listing 4: scanner.cc

## A.3 Parser Class

#### A.3.1 parser.cc

```
#include <iostream>
  #include "parser.h"
#include "error.h"
  using namespace std;
   /* The parser for the circuit definition files */
  bool parser::readin(void)
     //EBNF: specfile = devices connections monitors
11
     bool deviceDone = false, connectionDone = false, monitorDone = false;
     devicePresent = connectionPresent = monitorPresent = false;
13
    cursym = badsym;
14
     while (cursym != eofsym)
15
16
       if (cursym != devsym && cursym != consym && cursym != monsym)
17
18
         smz->getsymbol(cursym, curname, curint, numstring);
19
20
21
       if (cursym == devsym)
22
         if (deviceDone)
23
24
         {
           erz->newError(25);//Must only be one devices list
25
26
         deviceDone = true;
27
         deviceList();
28
29
30
       else if (cursym == consym)
31
         if (!deviceDone)
         {
```

```
erz->newError(0); //must have device list first
35
            (connectionDone)
36
37
         {
            erz->newError(28);//Must only be one connections list
38
39
         }
40
41
         connectionDone = true;
         connectionList();
42
43
44
       else if (cursym == monsym)
45
         if (!deviceDone | !connectionDone)
46
47
         {
           erz->newError(2); //Must have monitor list last
48
49
         if (monitorDone)
50
         {
            erz->newError(29);//Must only be one Monitors list
52
         monitorDone = true;
54
55
         monitorList();
56
       else if (cursym != eofsym)
57
58
         while (cursym != devsym && cursym != consym && cursym != monsym && cursym != eofsym)
59
60
           smz->getsymbol(cursym, curname, curint, numstring);
61
           erz->countSymbols();
62
63
         erz->symbolError(deviceDone, connectionDone, monitorDone);
64
       }
65
66
     if (!deviceDone)
67
68
       erz->newError(26);//There must be a DEVICES block, it may not have been initialised
69
       properly
71
     if (!connectionDone)
72
       erz->newError(30);//There must be a CONNECTIONS block, it may not have been initialised
73
       properly
74
75
     if (!monitorDone)
76
     {
       erz->newError(31);//There must be a MONITORS block, it may not have been initialised
77
       properly
     netz->checknetwork(correctOperation);
79
     anyErrors = erz->anyErrors();
80
     return (correctOperation && !anyErrors);
81
82
83
84
   void parser :: deviceList()
85
     //EBNF: devices = 'DEVICES' dev { '; ' dev} '; ' 'END'
86
     bool alreadyGotNextSymbol;
87
     if (!devicePresent)
88
89
       smz->getsymbol(cursym, curname, curint, numstring);
90
       if (cursym == classsym)
91
92
         alreadyGotNextSymbol = newDevice(curname);
93
         devicePresent = true;
94
95
       else if (cursym == endsym)
96
97
         erz->newError(3); //must have at least one device
98
         return;
99
       }
100
101
       else
         erz->newError(4); //need a device type
       if (!alreadyGotNextSymbol)
106
```

```
smz->getsymbol(cursym, curname, curint, numstring);
108
109
     while (cursym = semicol)
       smz->getsymbol(cursym, curname, curint, numstring);
112
       if (cursym == classsym)
114
         alreadyGotNextSymbol = newDevice(curname);
115
       else if (cursym == endsym)
117
118
       {
119
         return;
120
       else if (cursym == consym | cursym == devsym | cursym == monsym)
         erz->newError(32);//Block must be terminated with 'END'
123
         return;
124
125
       else
126
         erz->newError(5);//Expecting device name or END after semicolon (device name must start
       with letter)
130
       if (!alreadyGotNextSymbol)
132
         smz->getsymbol(cursym, curname, curint, numstring);
134
     if (!alreadyGotNextSymbol) erz->newError(24);//must end line in semicolon
     while (cursym != semicol && cursym != endsym && cursym != eofsym)
136
       smz->getsymbol(cursym, curname, curint, numstring);
138
140
     if (cursym == semicol)
141
     {
       deviceList();
142
143
     if (cursym == endsym)
144
145
     {
       return;
146
     }
147
148
149
   bool parser::newDevice(int deviceType)
     //EBNF: dev = clock | switch | gate | dtype | xor | siggen
     bool symbolSkip = false;
     smz->getsymbol(cursym, curname, curint, numstring);
156
     if (cursym == namesym)
157
       devlink nameCheck = netz->finddevice(curname);
158
159
       if (nameCheck=NULL)
160
         name devName = curname;
         if (deviceType == 10)
         {
           dmz->makedevice(dtype, devName, 0, correctOperation); //create DTYPE with name devName
164
            return symbolSkip;
167
         if (deviceType == 11)
168
         {
           dmz->makedevice(xorgate, devName, 2, correctOperation); //create XOR with name devName
            return symbolSkip;
         smz->getsymbol(cursym, curname, curint, numstring);
172
173
         if (cursym == colon)
174
           smz->getsymbol(cursym, curname, curint, numstring);
175
            if (cursym == numsym)
176
177
              switch (deviceType)
178
179
                case 4:
180
                  if (curint > 0)
181
```

```
182
                     dmz->makedevice(aclock, devName, curint, correctOperation); //create clock
183
        with curint and devName
                  }
184
185
                   else
                     erz->newError(6);//clock must have number greater than 0
187
188
                     symbolSkip=true;
189
                   break;
190
                case 5:
                   if (curint = 1 \mid | curint = 0)
192
193
                     dmz->makedevice(aswitch, devName, curint, correctOperation);//create switch
        with curint and devName
195
196
                     erz->newError(7);//switch must have either 0 or 1
198
                     symbolSkip=true;
199
200
                   break;
201
                case 6:
202
203
                case 7:
204
                 case
                      8:
                case 9:
205
                   if (curint > 0 && curint < 17)
206
                   {
207
                     switch (deviceType)
208
209
                       case 6:
210
                         dmz->makedevice(andgate, devName, curint, correctOperation);//create and
211
       gate with curint and devName
212
                         break:
213
                       case 7:
                         dmz->makedevice(nandgate, devName, curint, correctOperation);//create nand
214
        gate with curint and devName
215
                         break;
                       case 8:
216
217
                         dmz->makedevice(orgate, devName, curint, correctOperation);//create or
        gate with curint and devName
                         break:
218
219
                       case 9:
                         dmz->makedevice(norgate, devName, curint, correctOperation);//create nor
220
        gate with curint and devName
221
                         break;
                       default:
                         break;
                     }
224
                   }
226
                   else
                   {
227
                     erz->newError(8);//must have between 1 and 16 inputs to a GATE
228
229
                     symbolSkip=true;
230
231
                  break;
                 case 12:
232
                   if (isBinary(numstring))
233
234
                     sequence waveform;
235
                     for (int i=0; i<numstring.length(); i++)</pre>
236
237
                       waveform.push_back(numstring[i]=='1');
238
                     smz->getsymbol(cursym, curname, curint, numstring);
                     symbolSkip=true;
241
242
                     while (cursym==numsym)
                     {
243
                       if (isBinary(numstring))
245
                          for (int i=0; i<numstring.length(); i++)</pre>
246
                            waveform.push_back(numstring[i]=='1');
248
                         smz->getsymbol(cursym, curname, curint, numstring);
251
```

```
253
                        {
                          erz->newError(36); //Must be a binary input
254
                          symbolSkip=true;
255
256
                          break;
257
                     }
258
                     dmz->makesiggen(devName, waveform); //create SIGGEN with name devName
259
260
                   else
261
262
                      erz->newError(36); //Must be a binary input
263
264
                     symbolSkip=true;
                   break;
266
                 default:
267
                   break;
268
269
270
               return symbolSkip;
            }
271
            else
272
273
            {
              erz->newError(9);//clock needs clock cycle number
274
275
               symbolSkip=true;
276
            }
          }
277
278
          else
          {
279
            erz->newError(10);//need colon after name for CLOCK/SWITCH/GATE type
280
281
            symbolSkip=true;
          }
282
        }
283
284
285
        {
286
          erz->newError(27);//attempting to give two devices the same name, choose an alternative
       name
          symbolSkip=true;
287
288
      }
289
290
      else if (cursym!=badsym)
291
        erz->newError(33);//using reserved word as device name
292
293
        symbolSkip=true;
      }
294
      else
295
296
        erz->newError(11); //name must begin with letter and only containing letter number and _
297
298
        symbolSkip=true;
299
      return symbolSkip;
300
301
302
    void parser::connectionList()
303
304
      //EBNF: connections = 'CONNECTIONS' {con ';'} 'END'
305
      bool connectionError;
306
      if (!connectionPresent)
307
308
309
        smz->getsymbol(cursym, curname, curint, numstring);
        if (cursym == endsym)
310
311
          if (!connectionPresent)
312
313
          {
            erz->newWarning(0);//No Connections
314
315
          return:
316
317
        else if (cursym == namesym)
318
319
          connectionError = newConnection();
          connectionPresent = true;
321
322
323
        else
324
          erz->newError(12);//connection must start with the name of a device
325
326
```

```
if (!connectionError)
327
328
          smz->getsymbol(cursym, curname, curint, numstring);
330
331
      while (cursym = semicol)
332
333
        smz->getsymbol(cursym, curname, curint, numstring);
334
335
        if (cursym == namesym)
336
          connectionError = newConnection();
337
338
        else if (cursym == endsym)
339
340
          return:
341
        else if (cursym == consym | cursym == devsym | cursym == monsym)
343
344
          erz->newError(32);//Block must be terminated with 'END'
345
          return;
346
347
        else
348
349
          \operatorname{erz}\operatorname{->newError}\left(13\right);//\operatorname{connection} must start with the name of a device or end of device
350
        list must be terminated with END (not semicolon)
351
352
        if (!connectionError)
353
          smz->getsymbol(cursym, curname, curint, numstring);
354
355
356
        (!connectionError) erz->newError(24);//must end line in semicolon
      i f
357
      while (cursym != semicol && cursym != endsym && cursym != eofsym)
358
359
360
        smz->getsymbol(cursym, curname, curint, numstring);
361
      if (cursym == semicol)
362
363
        connectionList();
364
365
      if (cursym == endsym)
366
     {
367
368
        return;
369
370
371
   bool parser::newConnection()
372
373
      //EBNF: con = devicename '.' input '=' devicename ['.' output]
374
      bool symbolSkip = false;
375
      devlink devtype = netz->finddevice(curname);
376
      if (devtype != NULL)
377
378
379
        connectionInName = curname;
        smz->getsymbol(cursym, curname, curint, numstring);
380
        if (cursym == dot)
381
382
          smz->getsymbol(cursym, curname, curint, numstring);
383
384
          devtype = netz->finddevice(connectionInName);
          inplink ilist = netz->findinput(devtype, curname);
385
          if (cursym == iosym && ilist != NULL)
386
387
388
            name inputPin = curname;
            smz -\!\!>\! getsymbol\left(\textit{cursym}\;,\;\; curname\;,\;\; curint\;,\;\; numstring\;\right);
389
390
             if (cursym = equals) / SEARCH - you have got to here
391
             {
               smz->getsymbol(cursym, curname, curint, numstring);
392
               devtype = netz->finddevice(curname);
393
               if (devtype != NULL)
394
395
                 connectionOutName = curname;
396
                 switch (devtype ? devtype->kind : baddevice)
397
398
399
                      smz->getsymbol(cursym, curname, curint, numstring);
400
                      if (cursym == dot)
401
```

```
402
                       smz->getsymbol(cursym, curname, curint, numstring);
403
                       outplink olist = netz->findoutput(devtype, curname);
404
                       if (cursym == iosym && olist != NULL)
405
406
                         if (ilist ->connect==NULL)
408
                         {
                            netz->makeconnection(connectionInName, inputPin, connectionOutName,
409
       curname, correctOperation);
                           return symbolSkip;
410
411
                         else if (ilist -> connect == netz -> findoutput (devtype, curname))
412
413
                            namestring repeatedInput = smz->nmz->getnamestring(connectionInName);
414
                            namestring repeatedOutputDevice = smz->nmz->getnamestring(
415
       connectionOutName);
                           namestring repeatedOutputPin = smz->nmz->getnamestring(curname);
                           name string \ repeated Output = repeated Output Device \ + \ ".
417
       repeatedOutputPin;
                           erz->connectionWarning(repeatedInput, repeatedOutput);//generate
418
       warnning for repeated connection
                         else
420
421
                         {
422
                            erz->newError(37);//attempting to input 2 ouputs into same input
                            symbolSkip=true;
423
424
                         }
                       }
425
                       else
426
427
                         erz->newError(34); //Not valid output for dtype
428
                       }
429
430
                     }
431
                     else
432
                       erz->newError(14); //Expect a dot after dtype
433
                       symbolSkip=true;
434
435
                     break:
436
437
                   default:
                   //check the connection is unique
438
                     if (ilist ->connect==NULL)
439
440
                       netz->makeconnection(connectionInName, inputPin, connectionOutName,
441
       blankname, correct Operation):
449
                       return symbolSkip;
443
                     else if (ilist ->connect==netz->findoutput(devtype, blankname))
444
                       namestring repeatedInput = smz->nmz->getnamestring(connectionInName);
446
                       namestring repeatedOutput = smz->nmz->getnamestring(connectionOutName);
447
                       erz->connectionWarning(repeatedInput, repeatedOutput);//generate warnning
448
       for repeated connection
449
                     else
450
451
                       erz->newError(37);//attempting to input 2 ouputs into same input
                       symbolSkip=true;
453
454
                     break;
455
                }
456
457
              }
458
              \mathbf{else}
459
              {
                erz->newError(15); //Device does not exist
                symbolSkip=true;
461
              }
462
            }
463
            else
464
465
              erz->newError(16);//Must specify output to connect to input with equals sign
466
              symbolSkip=true;
467
468
          }
469
470
          else
471
```

```
erz->newError(17);//specify valid input gate after dot
            symbolSkip=true;
473
474
475
        }
        else
476
47
        {
          erz->newError(18); //need to seperate connection input with a '.' (or need to specify
478
        input)
          symbolSkip=true;
479
480
481
      }
      else
482
483
        erz->newError(19); //Device does not exist
484
        symbolSkip=true;
485
486
      return symbolSkip;
487
   }
488
489
    void parser::monitorList()
490
491
      //EBNF: monitors = 'MONITORS' {mon '; '} 'END'
492
      bool monitorError;
493
      if (!monitorPresent)
494
495
        smz->getsymbol(cursym, curname, curint, numstring);
496
497
        if (cursym == endsym)
498
          if (!monitorPresent)
499
500
            erz->newWarning(1);//No Monitors
501
          return;
503
504
505
        else if (cursym == namesym)
506
          monitorError = newMonitor();
507
508
          monitorPresent = true;
509
510
        else
511
          erz->newError(20);//monitor must start with the name of a device
512
513
        if (!monitorError)
514
515
516
          smz->getsymbol(cursym, curname, curint, numstring);
517
518
      while (cursym = semicol)
519
        smz->getsymbol(cursym, curname, curint, numstring);
        if (cursym == namesym)
522
523
          monitorError = newMonitor();
525
        else if (cursym == endsym)
526
527
          return:
528
        else if (cursym == consym | cursym == devsym | cursym == monsym)
530
          erz->newError(32);//Block must be terminated with 'END'
532
533
          return;
          erz->newError(21);//monitor must start with the name of a device or end of device list
        must be terminated with END (not semicolon)
538
        if (!monitorError)
540
        {
          smz->getsymbol(cursym, curname, curint, numstring);
541
542
         \hbox{(!monitorError) erz} {\longrightarrow} \hbox{newError} \hbox{(24);} // \hbox{must end line in semicolon} 
      i f
544
      while (cursym != semicol && cursym != endsym && cursym != eofsym)
```

```
546
       smz->getsymbol(cursym, curname, curint, numstring);
548
     if (cursym == semicol)
549
     {
       monitorList();
55:
     if (cursym == endsym)
554
     {
       return:
     }
557
   }
558
   bool parser::newMonitor()
      //EBNF: mon = devicename['.'output]
561
     bool symbolSkip = false;
562
     devlink devtype = netz->finddevice(curname);
563
564
     if (devtype != NULL)
565
       monitorName = curname;
566
       switch (devtype ? devtype->kind : baddevice)
567
569
          case 7:
570
            smz->getsymbol(cursym, curname, curint, numstring);
            if (cursym == dot)
571
572
573
              smz->getsymbol(cursym, curname, curint, numstring);
              outplink olist = netz->findoutput(devtype, curname);
574
              bool alreadyMonitored = mmz->IsMonitored(olist);
575
              if (!alreadyMonitored)
                if (cursym == iosym && olist != NULL)
578
                {
580
                  mmz->makemonitor(monitorName, curname, correctOperation);
                  return symbolSkip;
581
                }
582
583
                else
                {
584
585
                  erz->newError(34); //Not valid output for dtype
                }
586
              }
587
588
              else
              {
589
                namestring repeatedMonitorDevice = smz->nmz->getnamestring(monitorName);
590
                namestring repeatedMonitorPin = smz->nmz->getnamestring(curname);
                namestring repeatedMonitor = repeatedMonitorDevice + ".
                                                                             " + repeatedMonitorPin;
                erz->monitorWarning(repeatedMonitor); //repeated monitors
                if (cursym == iosym && olist != NULL)
594
                {
                  mmz->makemonitor(monitorName, curname, correctOperation);
596
                  return symbolSkip;
597
598
                }
                else
                {
600
                  \verb|erz->| newError(34); //Not valid output for dtype|
601
602
              }
603
604
            }
605
606
              erz->newError(22); //Expect a dot after dtype
607
              symbolSkip=true;
608
609
610
          default:
            outplink olist = netz->findoutput(devtype, blankname);
611
            bool alreadyMonitored = mmz->IsMonitored(olist);
612
              if (!alreadyMonitored)
613
              {
614
                mmz->makemonitor(monitorName, blankname, correctOperation);
615
              }
616
617
              else
618
              {
                namestring repeated Monitor = smz->nmz->getnamestring (curname);
619
                erz->monitorWarning(repeatedMonitor); //repeated monitors
620
                mmz->makemonitor(monitorName, curname, correctOperation);
621
```

```
622
            return symbolSkip;
623
        }
624
     }
625
626
     else
627
        erz->newError(23);//bad device monitor
628
629
        symbolSkip=true;
630
     return symbolSkip;
631
632
   }
633
   bool parser::isBinary(string numstring)
634
635
      for (int i=0; i<numstring.length(); i++)</pre>
636
637
        if (numstring [i]!= '0' && numstring [i]!= '1')
638
639
          {
640
            return false;
641
642
     return true;
643
644
645
646
   parser::parser(network* network_mod, devices* devices_mod, monitor* monitor_mod, scanner*
       scanner_mod, error* error_mod)
647
                            /* make internal copies of these class pointers */
     netz = network_mod;
648
                            /* so we can call functions from these classes
     dmz = devices_mod;
649
650
     mmz = monitor\_mod;
                            /* eg. to call makeconnection from the network
     smz = scanner_mod;
                            /* class you say:
651
     erz = error_mod; /* netz->makeconnection(i1, i2, o1, o2, ok);
652
     /* any other initialisation you want to do? */
653
654
```

Listing 5: parser.cc

parser.cc was written with joint effort between myself and Tim. I contributed approximately 25% of the code.

## A.4 Test Scripts

#### A.4.1 test.sh

```
#!/bin/bash
echo "" > error.txt
for f in *.gf2

do

echo "processing $f..."
echo "processing $f..." >> test.txt
echo -e "r 50\r\nq\r\n" | ../src/logsim $f >> test.txt
echo -e '\n' >> test.txt
done
```

Listing 6: test.sh

#### A.4.2 error.sh

```
#!/bin/bash
cecho "" > error.txt
for f in *.gf2
do
cecho "processing $f..."
cecho "processing $f..." >> error.txt
cecho ".../.../src/logsim $f >> error.txt
cecho -e '\n' >> error.txt
done
```

Listing 7: error.sh

## B Test Definition Files

All supplied definition files and circuit diagrams were written and designed by myself.

#### B.1 XOR Gate

#### **B.1.1** Definition File

```
DEVICES
  SWITCH S1:0;
  SWITCH S2:1;
  NAND G1:2;
  NAND G2:2;
  NAND G3:2;
  NAND G4:2;
  END
  CONNECTIONS
  G1.I1 = S1;
11
  G1.I2 = S2;
  G2. I1 = S1;
  G2. I2 = G1;
  G3.I1 = G1;
  G3. I2 = S2;
  G4. I1 = G2;
  G4.I2 = G3;
  END
  MONITORS
21
  S1;
22
  S2;
23
24
  G4;
  END
```

Listing 8: xor.gf2

#### B.1.2 Circuit Diagram



Figure 1: Circuit diagram of an XOR gate implemented using NAND gates  $\,$ 

## B.2 4-bit Adder

#### **B.2.1** Definition File

```
DEVICES

/* 4 bit inputs */

SWITCH A0:1;

SWITCH A1:0;

SWITCH A2:0;

SWITCH A3:0;

SWITCH B0:1;

SWITCH B1:0;

SWITCH B2:0;
```

```
10 SWITCH B3:0;
SWITCH C0:1; /* Carry in */
12 AND AND1:2;
13 AND AND2:2;
14 AND AND3:2;
15 AND AND4:2;
16 AND AND5:2;
17 AND AND6:2;
  AND AND7:2;
  AND AND8:2;
20 XOR XOR1;
21
  XOR XOR2;
  XOR XOR3;
  XOR XOR4;
  XOR XOR5;
24
  XOR XOR6:
  XOR XOR7;
  XOR XOR8;
  OR OR1:2;
  OR OR2:2;
  OR OR3:2;
  OR OR4:2;
31
  END
32
  CONNECTIONS
  /* LSB adder */
35
  XOR1.I1 = A0;
  XOR1.I2 = B0;
37
  AND1. I1 = XOR1;
  AND1.I2 = C0;
  AND2.I1 = A0;
  AND2.I2 = B0;
  XOR2.I1 = XOR1;
  XOR2.I2 = C0;
43
  OR1.I1 = AND1;
  OR1.I2 = AND2;
  XOR3. I1 = A1;
  XOR3. I2 = B1;
  AND3.I1 = XOR3;
  AND3. I2 = OR1;
  AND4.I1 = A1;
  AND4. I2 = B1;
  XOR4.I1 = XOR3;
  XOR4.I2 = OR1;
  OR2.I1 = AND3;
  OR2.I2 = AND4;
56
  XOR5.\,I\,1\ =\ A2\,;
  XOR5.I2 = B2;
59
  AND5. I1 = XOR5;
  AND5. I2 = OR2;
  AND6. I1 = A2;
62
  AND6.I2 = B2;
  XOR6.I1 = XOR5;
  XOR6.I2 = OR2;
  OR3. I1 = AND5;
  OR3. I2 = AND6;
67
   /* MSB Adder */
69
  XOR7.I1 = A3;
  XOR7.I2 = B3;
  AND7. I1 = XOR7;
  AND7. I2 = OR3;
  AND8.I1 = A3;
  AND8. I2 = B3;
  XOR8.I1 = XOR7;
  XOR8.I2 = OR3;
  OR4. I1 = AND7;
OR4. I2 = AND8;
79
80
  \overline{\mathrm{END}}
  MONITORS
  /* Outputs */
84 XOR2;
85 XOR4;
```

```
86 XOR6;
87 XOR8;
88 OR4; /* Carry out */
89 END
```

Listing 9: 4bitadder.gf2

#### **B.2.2** Circuit Diagram



Figure 2: Circuit diagram of a 4-bit adder

## B.3 Serial In Parallel Out Shift Register

## **B.3.1** Definition File

```
DEVICES
  CLOCK CLK1:2;
  CLOCK CLK2:1;
SWITCH S:0; /* Set switch */
  SWITCH R:0; /* Reset switch */
  DTYPE D1;
  DTYPE D2;
  DTYPE D3;
  DTYPE D4;
  END
11
  CONNECTIONS
12
  D1.DATA = CLK1;
13
  D2.DATA = D1.Q;
  D3.DATA = D2.Q;
  D4.DATA = D3.Q;
16
  D1.CLK = CLK2;
  D2.CLK = CLK2;
  D3.CLK = CLK2;
19
  D4.CLK = CLK2;
  D1.SET = S;
  D2.SET = S;
22
  D3.SET = S;
  D4.SET = S;
24
  D1.CLEAR = R;
25
  D2.CLEAR = R;
  D3.CLEAR = R;
27
  D4.CLEAR = R;
28
  END
30
  MONITORS
31
32
  CLK2;
33 D1.Q;
34 D2.Q;
  D3.Q;
35
36 D4.Q;
37 END
```

#### B.3.2 Circuit Diagram



Figure 3: Circuit diagram of a serial in parallel out shift register

 ${\bf NB}$  The software used to draw the circuit diagram does not support the same style of D flip-flop used in the definition file, and Fig. 3 was the closest achievable.

#### B.4 Gated D Latch

#### **B.4.1** Definition File

```
DEVICES
  CLOCK CLK1:1;
  CLOCK CLK2:2;
  NAND G1:1;
  AND G2:2;
  AND G3:2;
  NOR G4:2;
  NOR G5:2;
  END
  CONNECTIONS
  G1.I1 = CLK1;
  G2.I1 = G1;
  G2.I2 = CLK2;
  G3. I1 = CLK2;
  G3.I2 = CLK1;
  G4.I1 = G2;
  G4.I2 = G5;
  G5.I1 = G4;
19
  G5.I2 = G3;
  END
  MONITORS
  CLK1; /* D */
24
  CLK2; /* E */
  G4; /* Q */
      /* QBAR */
  G5;
27
  END
```

Listing 11: sipo.gf2

#### B.4.2 Circuit Diagram

**NB** The software used to draw the circuit diagram does not support the NAND gates with one input. Therefore the NAND gate G1 was substituted for a NOT gate as can be seen in Fig. 4.



Figure 4: Circuit diagram of a Gated D Latch

## **B.5** Signal Generator Example

#### **B.5.1** Definition File

```
DEVICES
 SIGGEN S1
    SIGGEN S2: 1 1
              1 0 0 1 0 1 0 1 0 1 0;
 SIGGEN S3:1;
 SIGGEN S4:0;
 AND A1:4;
 END
 CONNECTIONS
10
 A1. I1=S1;
 A1. I2=S2;
 A1. I3=S3;
 A1. I4=S4;
13
 END
 MONITORS
16
 S1;
17
18
 S2;
 S3;
19
 S4;
20
 A1;
21
 END
```

Listing 12: siggen.gf2

#### B.5.2 Circuit Diagram



Figure 5: Circuit diagram using a signal generator

**NB** The software used to draw the circuit diagram does not support the AND gates with four inputs. Therefore three AND gates with two inputs each were substituted for the single AND gate, A1, with four inputs as can be seen in Fig. 5.

## C EBNF

```
specfile = devices connections monitors

devices = 'DEVICES' dev ';' {dev ';'} 'END'
connections = 'CONNECTIONS' {con ';'} 'END'
monitors = 'MONITORS' {mon ';'} 'END'

dev = clock | switch | gate | dtype | xor | siggen
con = devicename'. 'input '=' devicename['.'output]
mon = devicename['.'output]

devicename = letter {'.'|letter | digit}
input = 'I'('1'|'2'|'3'|'4'|'5'|'6'|'7'|'8'|'9'|
'10'|'11'|'12'|'13|'14'|'15'|'16') | 'DATA'| 'CLK'| 'SET'| 'CLEAR'
output = 'Q'['BAR']

clock = 'CLOCK' devicename': 'digit { digit }
switch = 'SWITCH' devicename': '('0'|'1')
gate = ('AND'| 'NAND'| 'OR'| 'NOR') devicename': '('1'|'2'|'3'|'4'|'5'|'6'|'7'|'8'|
'9'|'10'|'11'|'12'|'13'|'14'|'15'|'16')
dtype = 'DTYPE' devicename
xor = 'XOR' devicename
siggen = 'SIGGEN' devicename': '('0'|'1') {'0'|'1'}
```

Listing 13: EBNF

#### D User Guide

Opening files: To open a definition file, click the File menu followed by the Open option. You will be presented with a file selection dialogue. The file selection dialogue will only show definition files (Files with the .gf2 file extension). Upon selecting a file, any errors in the definition file will be written to the message window, otherwise the Logic Simulator is ready to use.



Figure 6: The view upon running a simulation

Running simulations: In order to run a simulation you must first enter a number of cycles you wish the simulation to run for (default is 10) then press the run button. The monitored signals will be displayed in the left display panel. You may choose to continue the simulation by pressing the Continue button, or run the simulation continuously by pressing the Continuous Simulation button.

Adding or removing monitors: You can edit the signals that are monitored and displayed in the main window. To add monitors, click the Add monitors button and select the monitor, or monitors, you wish to add followed by the OK button. To remove monitors, press the Remove monitors button and select the monitor, or monitors, you wish to remove followed by the OK button

**Editing devices:** To edit devices, click the **Edit devices** button. From the Edit devices dialogue you can change the device's name, type and number of inputs (if applicable). You can also change the inputs to, or outures from a device.

Changing switch states: If your circuit contains any switches, you can change the state of the switch by changing the state of the check box beside its name.

Changing simulator settings: You can change certain settings relating to the operation of the logic simulator. Click the Options menu and from here you can change or reset settings as well as adjust the continuous run speed. By clicking Edit options you can change the trace options and debugging options.

## E File Listing



docs contains documentation relevant to our logic simulator such as: EBNF, reserved words, and our Gantt chart.

**examples** contains our example definition files as listed in Appendix B as well as the test shell script listed in Appendix A.4.

errors contains definition files which contain deliberate errors and are used to test our error checking functionality. In addition this folder contains the error.sh shell script listed in Appendix A.4.

report1, report2 and report3 contain our first, second and final report respectively.

src contains the source code for our logic simulator. The functionality of the major classes is outlined beneath.

**names** stores a list of all the words used within a definition file, and methods to manipulate them. It is initialised with only the reserved words, but can be populated as a definition file is read.

scanner reads through the definition file, character by character, and is able to return complete symbols to the parser. It is able to return the internal representation of a symbol, the type of symbol and optionally the value.

parser analyses the definition file as it is read in according to the rules laid out in our EBNF. It is then able to create devices, connections and monitors that are laid out in the definition file.

devices is used to create and execute devices.

network stores information about the devices, including the connections between each device.

**monitor** implements signal monitors, which are used to display the points of interest, as determined from the definition file, in the GUI.

gui provides a graphical user interface for the user, and allows for advanced features such as circuit and monitor editing without having to edit the definition file. We have implemented the GUI with a modular approach, and as such there are several gui related classes e.g. gui-canvas, gui-id, gui-misc, etc.

logsim ties all the other classes together and allows the user to fully simulate logic circuits.

## F Gantt Chart



Figure 7: Gantt chart showing our development cycle