SystemC Questa Tutorial
SystemC Questa Tutorial
SystemC Questa Tutorial
Revision History June, 2008: Initial version using Questa 6.3c (Modelsim 6.3d) R. Jenkal /P. Franzon)
1. Introduction
The aim of this tutorial is to provide an understanding of the representation of a design using SystemC and its simulation using Questa/Modelsim. To this end, we provide an implementation of a system consisting of a producer (traffic generator) which talks to a consumer, here an FIR filter, over a FIFO channel. These blocks are re-configurable and controllable using compile time parameters. Moreover, the interface between the producer and the consumer consists of both bit-accurate signals as well as the use of the sc_fifo class based interface.
2. Learning Objectives
Understanding of the coding structure for the SystemC design description Understanding the requirements for the simulation of SystemC code in Questa /Modelsim.
PLEASE DO NOT CUT AND PASTE COMMANDS FROM THE TUTORIAL. THIS LEADS TO ERRORS. YOU ARE ADVISED TO TYPE THEM OUT INSTEAD.
Please download files associated with this tutorial into a directory of your choice. All the work from this point assumes that you are within this directory.
1. Header files: consumer.h, producer.h, Top.h 2. Source Files: consumer.cc, producer.cc, Test_Top.cc 3. Definition file: defs.h 4. Modelsim initialization file: modelsim.ini
model that represents the environment, say, data from the memory interface which sends in data one at a time using the FIFO. But at the same time there is a fixed interface as well which has the following signals: a. ready from the consumer to producer: 1 if consumer is ready to work with the producer b. valid from consumer to producer: 1 if result from consumer is valid c. result from consumer to producer: contains the result of the FIR computation for that cycle. d. program_coeff from producer to consumer: 1 implies the producer is sending coefficients to the consumer to program the coefficients e. coeff which contains the coefficients that will be programmed into the FIR. The producer, on a start_new pulse and the ready signal from the consumer, begins by creating a programmable number of inputs and set of coefficients that go to the consumer. This is done by asserting the program_coeff signal and sending the coefficients one at a time over the coeff bus. The result bus is read when the valid signal is asserted to record the result being sent after computation from the consumer. 2. Consumer: This is going to be timing accurate but not bit accurate representation of the FIR filter with instance configurable computation size, bit sizes and programmable coefficients. The consumer has a protocol where we assume that the computation resource is shared and hence we can only perform a computation when the resource is available. This protocol is modeled using a (rand()%m > n) functionality to create a n-in-m probability of availability. If the resource is available, the consumer reads the FIFO and proceeds to produce results. 3. Abstracted Communication Channel: FIFO: This allows for setting up an abstracted yet standardized channel between a producer and consumer. This comes with the ability to do things like checking for free slots to write to, data presence, determining the number of data values present etc.
The following protocol is followed for setting up the test and computation: a. Reset the design b. On receiving a start_new pulse: 1. We setup an arbitrary number of values to be worked on in 1 test. 2. We setup a new set of coefficients that need to used in this run. c. As long as enable is 1 1. Wait for ready to go high, then program the coefficients in a clocked manner 2. At the producer: At each clock edge i. send data to FIFO until full ii. read result value if valid == 1 3. At the consumer: At each clock edge: iii. Determine availability using (rand()%m > n) clause and compute FIR output result using the incoming FIFO values. iv. Assert the valid signal to 1 when consumer is available and after the first N input data points have been provided. 4. Finish when all the results have been computed by the consumer by asserting the finished signal.
5. SC_MODULE_EXPORT is used to export the top level instance name of the testbench. In this case it is done as SC_MODULE_EXPORT(Test_Top);
(Note that the name mti_lib corresponds to the variable work within the modelsim.ini file and is the library to which all the source code would be compiled to create a design single entity). Note that in some cases, if the compilation seems to crash for a reason you think is incorrect, it would be advisable to delete the mti_lib directory (Use: \rm rf mti_lib OR vdel all) and re-create it as shown above. 2. One time setup for a set of simulations within a directory: We assume step 1 has already been followed. Let us assume a directory has been setup up correctly and you come into this directory for a future simulation. You would still need to run the following commands each time you start a set of simulations for a given design within a directory. a. set paths to the Modelsim tool: b. prompt%> add questasim63 OR prompt%> add modelsim c. set environment variable MODELSIM to modelsim.ini
prompt%> setenv MODELSIM modelsim.ini
At this point we have all the relevant variables and directories needed for a successful simulation setup.
We assume that Step 5.1 has already been followed and both the mti_lib library has been created and the modelsim.ini file is already present in the simulation and compilation directory. For the present example, the compilation of the code is done as (making it generic to the case where the files are in a different directory as the compilation):
prompt%> sccom <path to source files>/consumer.cc prompt%> sccom <path to source files>/producer.cc prompt%> sccom <path to source files>/test_Top.cc
Here, sccom is the compiler for SystemC descriptions (the equivalent of vlog for verilog). After the compilation of the files, you will receive the following message from the compiler,
Exported modules: Test_Top
This implies that the top level instance i.e. the testbench module Test_Top in this case, has been successfully exported as a design unit. We now need to link the multiple design units and module declarations using :
prompt%> sccom -link
At any stage of the above compilation or simulation, errors in coding would be displayed as messages with the respective error code. 6.2. Simulation of design: We are now poised to simulate the design example using the vsim simulation engine. The invocation of the engine is performed as:
prompt%> vsim novopt Test_Top
This invokes the GUI based simulation engine. The novopt command line option is used to direct the simulator that no optimizations should be performed in the invocation of the design unit. This is done to ensure that we have visibility to as much of the design as possible. It you are interested in just having the simulator run as fast as possible while not needing to see all the design intricacies, this option can be removed. It must also be noted that, if you are interested in having the simulator run without invoking the GUI, you can do so using the c option (i.e. vsim c novopt <top level module>) WITHOUT the &. [ASIDE] All of the above sccom commands (and ONLY the sccom commands) can be run either on the command prompt or after the invocation of the simulation engine. Thus the prompt would either be the command prompt on the UNIX/Linux terminal or VSIM#> which is the command prompt of the simulator. Another useful option is the use of .do files to prevent having to type all the commands. This will be talked about in Appendix A. The result of the above invocation is the GUI shown in Figure 2. with the Test_Top design unit loaded.
Design Hierarchy Verilog allows you to build modules that contain other modules. You might recall that we instantiated the modules producer and consumer within the module Top (look within Top.h), to enable us to test this architecture. The module Top is itself instantiated with the name (sc_module_name) TopInst within Test_Top.cc (view code for conformation) as
Top_inst = new Top("TopInst", 10, 5, 50);
You might recall that we had declared the following signals inside Test_Top
sc_signal sc_signal sc_signal sc_signal <bool <bool <bool <bool > > > > reset; start_new; enable; finished;
Thus within the instance, Test_Top we should be able to find the signals declared within that module and a reference to the TopInst instance of the module Top.
To this end, let us look at the contents of the GUI window. Note the presence of the workspace window and the objects window. The workspace contains all the design units within the mti_lib working library. At the top of the workspace window you will see the Test_Top design unit. This is the logical top level of the simulation. Given that the Top module is instantiated within the test_fixture as TopInst, you will notice the presence of the instance TopInst under Test_Top. The hierarchy of the design is captured by virtue of a series of buttons with + on them. To go down the hierarchy you would need to click on the + to expand it and view the contents of the design unit. The objects window lists the objects i.e. inputs, outputs, internal nets, parameters etc, corresponding to the design unit that is highlighted in the workspace window. In the above case, the instance TopInst has been highlighted by clicking on it. It is interesting to note that when Test_Top is highlighted we see reset, start_new, enable and finished, while on highlighting TopInst we see clock, start_new, enable, reset, ready, valid and a slew of other signals that correspond to the connectivity between the producer and consumer. Moreover, on expanding the TopInst tab, you will also notice the instances of the producer and consumer as Prod and Cons in keeping with their instance within Top.h as:
prod_inst = new producer("Prod", coeff_size, input_size); cons_inst = new consumer("Cons", coeff_size);
We now need to use the GUI to enable us to view the waveforms at the level of the instance TopInst to determine correctness of the design operation. One of two things can be done now to view waveforms: a. We can either display all the nets in the objects window by right clicking on the Signals in Region in the list objects window and doing: add to wave that pops up. b. You can highlight all signals that are of relevance to you by keeping the ctrl signal depressed and clicking all the signals of interest once. Once this is done, you can do add to wave Selected Signals in the list that pops up. The preview of case a. is shown below:
The result of the above command is the creation of the waveform window. This is shown below. The window would generally result in a window that would be docked with the rest of the GUI. It is easiest to view the wave in a free moving window for the waves. To do this, undock the waveform window by hitting the button at the top of the waveform window.
We are now poised to run the simulation for this design. To this end, we would need to do the following at the GUI command prompt (say we are running only for 200ns).
VSIM #> run 200ns
The result of the above command can be seen in the waveform window where we can note simulation progress. A preview of the waveforms is shown below:
To view the waveform in its entirety you would need to fit the waveform in the window by either hitting the f key on the keyboard or by hitting the key to zoom to fit the waveform in the window. At this point we can ensure that the design works correctly by visual inspection. [ASIDE]: To run this simulation to its completion we would need to do VSIM#> run all where it will take the simulation onwards till a sc_stop() is found. When a sc_stop() is hit, the tool prompts a "Are you sure you want to finish?" to which you hit No. It is always advised to run the simulation to a controlled amount of time. Something that might prove useful immediately is the tools support for changes in radix. To be able to change the radix of a signal of choice (here the coeff bus), right click in the signal name on the waveform window and then perform: radix -> decimal (or anything else you might need). Let us assume that you want to change a few parameters in your simulation environment, say the number of coefficients or the number of inputs. This would be done in Test_Top.cc by changing the line Top_inst = new Top("TopInst", 10, 5,
50); where the second, third and fourth entries in the call correspond to the size of the
FIFO, the number of coefficients and the number of inputs that are going to be sent into the consumer. The idea here is to note that we do not need to leave the GUI based environment to do the necessary compilation after the modification of the code. To recompile and simulate, do the following a. Modify Top_inst = new Top("TopInst", 10, 5, 50); to Top_inst = new Top("TopInst", 5, 7, 50); This is going to cause greater stalling in the design because the size of the FIFO is smaller. . b. In the vsim GUI type the following : VSIM#> sccom <path to source file>/Test_Top.cc. This compiles the edited file within the GUI environment. c. Link the modified design using VSIM#> sccom -link d. Restart the simulation that is presently running by doing a
VSIM#> restart f VSIM#> run 400ns
At this point you should be able to note the modified initial loading of the coeff bus (we have 7 values instead of the previous 5). The resultant simulation would look like something shown below:
The simulation results might differ a bit by virtue of the difference in the behavior of the rand() function called within the consumer. Also note the presence of a set of tools (circled in the previous figure), which can be used to create cursors (we have added a cursor in the above figure), lock the cursors, remove them, change the names of the signals from the full path to the short signal name etc.
10
APPENDIX A: ADDITIONAL TOOL INFORMATION 1. Working on the command prompt: It is useful to note that, if the GUI is to be avoided (debug using only text commands/generation of files/checking for compilation correctness.), the simulation can be invoked in the non-GUI mode by doing a vsim c. This causes a Modelsim prompt to appear where you compile (vlog), invoke the simulation (vsim) and run the entire simulation (run). This is sometimes a good alternative when X-windows proves troublesome or you do not have a connection strong enough to support a GUI based operation. 2. Compiling selectively: If you make a modification in just a few .cc files, then you can compile just those files with a sccom (instead of the entire set of files you are working with). Be sure to do a sccom link after the modified file and just do a VSIM#> restart f to restart the simulation. There is no need to quit the GUI. 3. Using .do files: An extremely useful feature in Modelsim is the presence of do files. These are the equivalent of shell scripts for Modelsim. All the necessary compilation commands, simulation commands, waveforms etc can be stored within a .do file. An example .do file is provided on the EDA page. This allows you to avoid typing in all the commands manually. Instead, you can call this file within the vsim environment ( GUI / no GUI) by doing a VSIM #> do <filename>.do. Of particular importance in working with do files is working with waveforms. a. Saving Waveforms: Once you have set up all the relevant waves on the waveform window, it is feasible in Modelsim to store the format of these waves in a .do file and restore the same set of waves the next time you come back to the simulator. This is done by saving all the waveforms by doing a File Save and saving the format as a .do file. The next time you invoke the simulator, MAKE SURE THAT THE DESIGN HAS BEEN LOADED using a vsim <modulename> for eg, vsim novopt test_fixture, after which you can open the waveform window and do a File Load <filename>.do to get the same set of waveforms as before.
11