

## Simulating Multiple Gates

### Background

The first installment of the project dealt with the capability to simulate the actions of four very simple gate types, an INV gate, an AND gate, an OR gate and an XOR (exclusive or) gate. The second installment extended that capability by describing the simple logic gates using Verilog, but the Verilog was restricted to descriptions of circuits which only contained a single gate. For the third installment, we will allow our Verilog circuits to contain multiple gates.

### Verilog Infrastructure

There are very few changes in the Verilog infrastructure provided to you between the second installment of the project, and the third installment, but there have been a couple of minor changes, so the name of the support file has changed to verilog2.c – just to ensure you use the latest version. Also, experience with installment 2 shows that the documentation for the functions provided to you can be improved. Therefore, at the end of these instructions, I've added a Verilog infrastructure reference section which describes in more detail what each function does.

The specific changes from the previous version are:

1. The order of instance name and instance module name have been corrected to match true Verilog syntax. The correct order is instance module name first, then instance name.
2. The name of an inverting gate has been corrected to match true Verilog syntax. Verilog recognizes “not” as the name of an inverting gate (and does not recognize “inv”). (Note that the Verilog gate names never appear in verilog2.c, but the associated Verilog files provided as examples have been fixed.)

As in installment 2, I have also included a C program called “printCircuit.c” which is an example of how to use the Verilog infrastructure. Look at this code to get some idea of how to use the infrastructure.

You may also look through the code that implements the Verilog infrastructure, but there are some concepts used in that code that we haven't discussed yet. We will talk more about the code when we discuss these concepts.

### Project Description

Create a C program called simCircuit.c. This program is very similar to the simGate.c program you created for installment 2. You may start with the code you used for installment 2, and modify it to accommodate the new installment 3 requirements.

The simCircuit.c program needs to:

1. Check to make sure there are at least 3 command line inputs specified. If not, print a message that indicates what the command line should be, and exit with a non-zero return code.
2. Invoke the Verilog openModule statement passing argv[1] in as the argument, and keeping track of the module number returned. (If there is a problem reading the Verilog file, openModule will exit the program for you, so there is no need to check to see if it worked.)
3. Loop through the remaining command line inputs. Invoke the “atoi()” library function to get the integer value represented by each command line argument, and use that value to set each module input pin in the Verilog circuit to that value, using the order the module input pins appear in the Verilog module header. If there are not enough command line values, print the names of the input module pins which were not set. If there were too many command line values specified, print the values not used in a message.
4. Loop through the circuit as long as some activity occurs, where activity is either setting a pin in the circuit to a known value, or setting a net in the circuit to a known value. Inside this loop you will net to:
  - a. Propagate all source pin values to nets.
  - b. Propagate all net values to net sinks.
  - c. Evaluate all instances and set the instance output pin based on the input values.
5. Print the result to the user using the printModuleIO function.
6. Invoke the freeModule function for your module.
7. If your program worked correctly, return an exit code of zero.

The following is an example of a correctly coded program:

```
./simCircuit AND_2 1 0
AND_2 Inputs: A0=1, A1=0; Outputs: Z=0;
./simCircuit MUX2 1 0 1
MUX2 Inputs: I0=1, I1=0, SEL=1; Outputs: Z=0;
```

#### Hints and Suggestions

- Don't forget to include “verilog.h” in your program, and to compile using the command:  
    gcc -g -Wall -o simCircuit simCircuit.c verilog2.c
- To manage the loop in instruction 4 above, create an integer variable called “active” that is initialized to 1, and then control the loop using “while(active)”. Then the first line of the body of the loop sets active to 0. Then, in either a, b, or c of the body of the loop, if the setPinValue or setNetValue functions return a 1, set active to 1.
- Use the printModuleState function to see what is set and what is not set to debug your code.
- Use the gdb debugger to help find specific problems in your code.

### Project Submission

Upload your simCircuit.c file in the Project Installment 3 submission area on MyCourses in the Content area under “Project Submissions”.

### Project Grading

After the due date, your submission will be graded as follows. The project is worth 100 points. Your code will be compared to all other student's code using an automated code plagiarism checker that can detect copied code even if you try to “fix” it after copying. Then, your code will be compiled on a BU Linux machine. If there are compiler errors, the professor will attempt to fix your code. If your code can be fixed, the rest of the grading will be performed on that fixed code. Once compiled, your code will be tested with various Verilog circuits that contain one or more gates with various inputs. The results will be compared to correct results, based on the requirements above. Points will be deducted for the following reasons:

| Problem                                     | Deduction                                    |
|---------------------------------------------|----------------------------------------------|
| Bad submission (e.g. name not simCircuit.c) | -5 points                                    |
| Compiler or logic error that can be fixed   | -15 points per error                         |
| Unfixable error, or >4 errors               | -60 points                                   |
| Compiler warning messages                   | -5 points per type of warning message        |
| Illegible or Poorly formatted code          | Up to -10 points                             |
| Incorrect results                           | -10 points per invocation (up to -40 points) |
| Late submission                             | -10 points per every 24 hours late           |
| Code matches another student's code         | -100 points                                  |

## Verilog Infrastructure Reference

The Verilog Infrastructure consists of a series of functions provided to you. These functions will read a Verilog file, and return information to you about the contents of that file. The functions also provide the capability to specify and retrieve Boolean values on pins and nets in the Verilog circuit defined within the Verilog file.

The infrastructure keeps track of entities in the Verilog circuit, such as modules, pins, nets, and instances using arbitrary numbers which can be thought of as module numbers, pin numbers, net numbers, and instance numbers. I will use variable names “mn”, “pn”, “nn”, and “in” to keep track of the module number, pin number, net number, and instance number in this reference. When there are lists of entities, such as the list of nets in a module, there will be a function that returns the size of that list (e.g. `moduleNumNets(mn)`). You can then specify an index into that list – any number between 0 and the size of the list – 1 as the “index” into the list. For each list, there is a function which takes an argument that is the index into the list, and returns the number associated with the entity in the list. I will use variable names “pi”, “ni”, “ii”, and “si” to reference the index of a pin in a pin list, the index of a net in a net list, the index of an instance in an instance list, and the index of a sink in a sink list respectively in the following documentation. For instance, to loop through all the nets in the module with module number mn, you can use the following code:

```
for(int ni=0;ni<moduleNumNets(mn);ni++) {
    int nn=moduleNet(mn,ni);
    // Handle net with net number nn
}
```

### Top Level Functions

`int openModule(moduleName)`

- Argument: `char * moduleName` – the name of a Verilog module.
- Function: Reads file `moduleName.v` (where `moduleName` is the argument) and collects all the information about the circuit described in that file in internal tables. If there are any errors, such as file not found or incorrect Verilog data, a message will be printed, and the program will abort. If everything is successful, the module number will be returned for later use.
- Return Value: The module number (mn) to be used in other calls.

`char * moduleName(mn)`

- Argument: `int mn` – the module number of a Verilog circuit.
- Function: retrieves the module name of the function
- Return Value: A string (character array) that contains the module name

`void printModuleState(mn)`

- Argument: int mn – the module number of a Verilog circuit.
- Function: Prints out all known information about the Verilog module mn. To be used to debug – to see all the nets, pins, and instances and all the known information about those entities.

`void printModuleIO(mn)`

- Argument: int mn – the module number of a Verilog circuit.
- Function: Prints the module name, and the module pins and their values. This is the expected form of output when simulating a gate or a circuit in this project.

`void freeModule(mn)`

- Argument: int mn – the module number of a Verilog circuit.
- Function: Releases all the information associated with module mn (and recovers space used by that module). After freeing a module, the module number should not be used as arguments to subsequent calls to other infrastructure functions.

#### Module to Pin Functions

Each module contains both module pins and internal pins. Both are kept in a single list, but the module pins occur first in that list.

`int moduleNumModulePins(mn)`

- Argument: int mn – the module number of a Verilog circuit.
- Function: Retrieve the number of module pins contained in module mn, so that you can calculate the pin indexes (pi) from 0 to this number to be used in the modulePin function.
- Return Value: The number of module pins in module mn.

`int moduleNumPins(mn)`

- Argument: int mn – the module number of a Verilog circuit.
- Function: Retrieve the number of pins (both module pins and internal pins) contained in module mn, so that you can calculate the pin indexes (pi) from 0 to this number to be used in the modulePin function.
- Return Value: The number of pins in module mn.

`int modulePin(mn,pi)`

- Arguments: int mn – the module number of a Verilog circuit.  
`int pi` – The pin index of a pin in module mn
- Function: Asserts that `pi < moduleNumPin(mn)`, and retrieves the pin number associated with pin index pi in module mn. Note that for very simple Verilog circuits, the pin numbers may match the pin indexes, but in general this will not be true.
- Return Value: The pin number (pn) of the pi<sup>th</sup> pin in module mn.

### Pin Functions

The following functions assume you know the pin number (pn).

`char * pinName(pn)`

- Argument: int pn – the pin number of a Verilog pin.
- Function: asserts that pn is a valid pin number, then retrieves the name of the pin. For module pins, this is the name of the pin itself. For instance pins, this is the instance name, followed by a dot, followed by the name of the pin in the instance. For gate instances, pin names are arbitrarily A0, A1, ... for input pins, and Z for the output pin.
- Return Value: A string (character array) that contains the pin name of pin pn.

`char pinDirection(pn)`

- Argument: int pn – the pin number of a Verilog pin.
- Function: asserts that pn is a valid pin number, then retrieves the direction of the pin.
- Return Value: 'I' if pn is an input pin, 'O' if pn is an output pin.

`int pinInstance(pn)`

- Argument: int pn – the pin number of a Verilog pin.
- Function: asserts that pn is a valid pin number, then retrieves the instance number of the instance associated with the pin. If the pin is a module pin, it does not have an instance, so a -1 is returned.
- Return value: The instance number (in) of the instance associated with pin pn.

`int pinNet(pn)`

- Argument: int pn – the pin number of a Verilog pin.
- Function: asserts that pn is a valid pin number, then retrieves the net associated with pin pn. Since all pins in Verilog are connected to a single net, this will return a single result.
- Return Value: The net number (nn) of the net connect to pin pn.

`int pinValue(pn)`

- Argument: int pn – the pin number of a Verilog pin.
- Function: asserts that pn is a valid pin number, then retrieves the Boolean value of the pin. The value is initialized to -1 (unknown) but can be set by the setPinValue function.
- Return Value: The Boolean value associated with pin pn.

`int setPinValue(pn,val)`

- Argument: int pn – the pin number of a Verilog pin.  
int val – The Boolean value (should be 0 or 1) to be put on pin pn.

- Function: asserts that pn is a valid pin number. If the pin value matches the val argument, returns zero. Asserts that the pin value is -1 (unknown) because it is an error to set the same pin to a different value. Sets pin pn's value to val, and returns a 1.
- Return Value: A zero if the pin value did not change, or a 1 if the pin value was set to the specified val.

### Module to Net Functions

int moduleNumNets(mn)

- Argument: int mn – the module number of a Verilog circuit.
- Function: Retrieve the number of nets contained in module mn, so that you can calculate the net indexes (ni) from 0 to this number to be used in the moduleNet function.
- Return Value: The number of nets in module mn.

int moduleNet(mn,ni)

- Arguments: int mn – the module number of a Verilog circuit.  
int ni – The net index of a net in module mn
- Function: Asserts that ni<moduleNumNets(mn), and retrieves the net number associated with net index ni in module mn. Note that for very simple Verilog circuits, the net numbers may match the net indexes, but in general this will not be true.
- Return Value: The net number (nn) of the ni<sup>th</sup> net in module mn.

### Net Functions

The following functions assume you know the net number (nn).

char \* netName(nn)

- Argument: int nn – the net number of a Verilog net.
- Function: asserts that nn is a valid net number, then retrieves the name of the net.
- Return Value: A string (character array) that contains the net name

int netSource(nn)

- Argument: int nn – the net number of a Verilog net.
- Function: asserts that nn is a valid net number, then retrieves the pin number of the source pin of net nn.
- Return Value: The pin number (pn) of the source pin of net nn.

int netNumSinks(nn)

- Argument: int nn – the net number of a Verilog net.
- Function: asserts that nn is a valid net number, then retrieves the number of sink pins so that you can calculate the sink indexes (si) from 0 to this number to be used in the netSink function.

- Return Value: The number of sink pins for net nn.

int netSink(nn,si)

- Arguments: int nn – the net number of a Verilog net.  
int si – The sink index of a sink for net nn
- Function: Asserts that  $si < \text{netNumSinks}(nn)$ , and retrieves the pin number associated with sink index si in net nn.
- Return Value: The pin number (pn) of the  $si^{\text{th}}$  sink of net nn.

int netValue(nn)

- Argument: int nn – the net number of a Verilog net.
- Function: asserts that nn is a valid net number, then retrieves the Boolean value of the net. The value is initialized to -1 (unknown) but can be set by the setNetValue function.
- Return Value: The Boolean value associated with net nn.

int setNetValue(nn)

- Argument: int nn – the net number of a Verilog net.
- Function: asserts that nn is a valid net number. Then retrieves the source pin number for the source pin of this net, and that source pin's value. If the net value matches the source pin's value, returns zero. Asserts that the net value is -1 (unknown) because it is an error to set the same net to a different value. Sets net nn's value to source pin's value and returns a 1.
- Return Value: A zero if the net value did not change, or a 1 if the net value was set to its source pin's value.

### Module to Instance Functions

int moduleNumInstances(mn)

- Argument: int mn – the module number of a Verilog circuit.
- Function: Retrieve the number of instances contained in module mn, so that you can calculate the instance indexes (ii) from 0 to this number to be used in the moduleInstance function.
- Return Value: The number of instances in module mn.

int moduleInstance(mn,ii)

- Arguments: int mn – the module number of a Verilog circuit.  
int ii – The instance index of an instance in module mn
- Function: Asserts that  $ii < \text{moduleNumInstances}(mn)$ , and retrieves the instance number associated with instance index ii in module mn. Note that for very simple Verilog circuits, the instance numbers may match the instance indexes, but in general this will not be true.
- Return Value: The instance number (in) of the  $ii^{\text{th}}$  instance in module mn.

### Instance Functions

The following functions assume you know the instance number (in).

`char * instanceName(in)`

- Argument: int in – the instance number of a Verilog instance.
- Function: asserts that in is a valid instance number, then retrieves the instance name of the instance. Note that the instance name is a unique name within this Verilog circuit of the instance, and is different from the instance module name (see `instanceModule` `instanceModuleName`).
- Return Value: A string (character array) that contains the instance name

`char * instanceModule(in) instanceModuleName(in)`

- Argument: int in – the instance number of a Verilog instance.
- Function: asserts that in is a valid net number, then retrieves the name of the gate that is being instanced in this instance, which should be one of “and”, “or”, “xor”, or “not”. Note that the instance module name is different from the instance name (see `instanceName`).
- Return Value: A string (character array) that contains the name of the gate instanced by instance in

`int instanceNumPins(in)`

- Argument: int in – the instance number of a Verilog instance.
- Function: asserts that in is a valid instance number, then retrieves the number of pins connected to this instance so that you can calculate the pin indexes (pi) from 0 to this number to be used in the `instancePin` or `instanceNet` functions.
- Return Value: The number of pins connected to instance in.

`int instancePin(in,pi)`

- Arguments: int in – the instance number of a Verilog instance.  
int pi – The index of a pin on instance in
- Function: Asserts that `pi < instanceNumPins(in)`, and retrieves the pin number associated with pin index pi in instance in.
- Return Value: The pin number (pn) of the  $pi^{th}$  pin of instance in.

`int instanceNet(in,pi)`

- Arguments: int in – the instance number of a Verilog instance.  
int pi – The index of a pin/connection on instance in
- Function: Asserts that `pi < instanceNumPins(in)`, and retrieves the net number associated with connection index pi in instance in.
- Return Value: The net number (nn) of the  $pi^{th}$  connection of instance in.