447 lines
17 KiB
C
447 lines
17 KiB
C
/***********************************/
|
|
/* SPICE Modeling for VPR */
|
|
/* Xifan TANG, EPFL/LSI */
|
|
/***********************************/
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <math.h>
|
|
#include <time.h>
|
|
#include <assert.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
|
|
/* Include vpr structs*/
|
|
#include "util.h"
|
|
#include "physical_types.h"
|
|
#include "vpr_types.h"
|
|
#include "globals.h"
|
|
#include "rr_graph.h"
|
|
#include "vpr_utils.h"
|
|
#include "path_delay.h"
|
|
#include "stats.h"
|
|
#include "route_common.h"
|
|
|
|
/* Include spice support headers*/
|
|
#include "read_xml_spice_util.h"
|
|
#include "linkedlist.h"
|
|
#include "fpga_x2p_types.h"
|
|
#include "fpga_x2p_utils.h"
|
|
#include "fpga_x2p_pbtypes_utils.h"
|
|
#include "fpga_x2p_backannotate_utils.h"
|
|
#include "fpga_x2p_globals.h"
|
|
#include "fpga_bitstream.h"
|
|
|
|
|
|
/* Include SPICE generator headers */
|
|
#include "spice_globals.h"
|
|
#include "spice_subckt.h"
|
|
#include "spice_pbtypes.h"
|
|
#include "spice_heads.h"
|
|
#include "spice_lut.h"
|
|
#include "spice_top_netlist.h"
|
|
#include "spice_mux_testbench.h"
|
|
#include "spice_grid_testbench.h"
|
|
#include "spice_routing_testbench.h"
|
|
#include "spice_primitive_testbench.h"
|
|
#include "spice_run_scripts.h"
|
|
|
|
/* For mrFPGA */
|
|
#ifdef MRFPGA_H
|
|
#include "mrfpga_globals.h"
|
|
#endif
|
|
|
|
/* RUN HSPICE Shell Script Name */
|
|
static char* default_spice_dir_path = "spice_netlists/";
|
|
static char* spice_top_tb_dir_name = "top_tb/";
|
|
static char* spice_grid_tb_dir_name = "grid_tb/";
|
|
static char* spice_pb_mux_tb_dir_name = "pb_mux_tb/";
|
|
static char* spice_cb_mux_tb_dir_name = "cb_mux_tb/";
|
|
static char* spice_sb_mux_tb_dir_name = "sb_mux_tb/";
|
|
static char* spice_cb_tb_dir_name = "cb_tb/";
|
|
static char* spice_sb_tb_dir_name = "sb_tb/";
|
|
static char* spice_lut_tb_dir_name = "lut_tb/";
|
|
static char* spice_hardlogic_tb_dir_name = "hardlogic_tb/";
|
|
static char* spice_io_tb_dir_name = "io_tb/";
|
|
|
|
/***** Subroutines Declarations *****/
|
|
static
|
|
void init_list_include_netlists(t_spice* spice);
|
|
|
|
static
|
|
void free_spice_tb_llist();
|
|
|
|
/***** Subroutines *****/
|
|
|
|
static
|
|
void init_list_include_netlists(t_spice* spice) {
|
|
int i, j, cur;
|
|
int to_include = 0;
|
|
int num_to_include = 0;
|
|
|
|
/* Initialize */
|
|
for (i = 0; i < spice->num_include_netlist; i++) {
|
|
FreeSpiceModelNetlist(&(spice->include_netlists[i]));
|
|
}
|
|
my_free(spice->include_netlists);
|
|
spice->include_netlists = NULL;
|
|
spice->num_include_netlist = 0;
|
|
|
|
/* Generate include netlist list */
|
|
vpr_printf(TIO_MESSAGE_INFO, "Listing SPICE Netlist Names to be included...\n");
|
|
for (i = 0; i < spice->num_spice_model; i++) {
|
|
if (NULL != spice->spice_models[i].model_netlist) {
|
|
/* Check if this netlist name has already existed in the list */
|
|
to_include = 1;
|
|
for (j = 0; j < i; j++) {
|
|
if (NULL == spice->spice_models[j].model_netlist) {
|
|
continue;
|
|
}
|
|
if (0 == strcmp(spice->spice_models[j].model_netlist, spice->spice_models[i].model_netlist)) {
|
|
to_include = 0;
|
|
break;
|
|
}
|
|
}
|
|
/* Increamental */
|
|
if (1 == to_include) {
|
|
num_to_include++;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* realloc */
|
|
spice->include_netlists = (t_spice_model_netlist*)my_realloc(spice->include_netlists,
|
|
sizeof(t_spice_model_netlist)*(num_to_include + spice->num_include_netlist));
|
|
|
|
/* Fill the new included netlists */
|
|
cur = spice->num_include_netlist;
|
|
for (i = 0; i < spice->num_spice_model; i++) {
|
|
if (NULL != spice->spice_models[i].model_netlist) {
|
|
/* Check if this netlist name has already existed in the list */
|
|
to_include = 1;
|
|
for (j = 0; j < i; j++) {
|
|
if (NULL == spice->spice_models[j].model_netlist) {
|
|
continue;
|
|
}
|
|
if (0 == strcmp(spice->spice_models[j].model_netlist, spice->spice_models[i].model_netlist)) {
|
|
to_include = 0;
|
|
break;
|
|
}
|
|
}
|
|
/* Increamental */
|
|
if (1 == to_include) {
|
|
spice->include_netlists[cur].path = my_strdup(spice->spice_models[i].model_netlist);
|
|
spice->include_netlists[cur].included = 0;
|
|
vpr_printf(TIO_MESSAGE_INFO, "[%d] %s\n", cur+1, spice->include_netlists[cur].path);
|
|
cur++;
|
|
}
|
|
}
|
|
}
|
|
/* Check */
|
|
assert(cur == (num_to_include + spice->num_include_netlist));
|
|
/* Update */
|
|
spice->num_include_netlist += num_to_include;
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
static
|
|
void free_spice_tb_llist() {
|
|
t_llist* temp = tb_head;
|
|
|
|
while (temp) {
|
|
my_free(((t_spicetb_info*)(temp->dptr))->tb_name);
|
|
my_free(temp->dptr);
|
|
temp->dptr = NULL;
|
|
temp = temp->next;
|
|
}
|
|
free_llist(tb_head);
|
|
|
|
return;
|
|
}
|
|
|
|
/***** Main Function *****/
|
|
void vpr_fpga_spice(t_vpr_setup vpr_setup,
|
|
t_arch Arch,
|
|
char* circuit_name) {
|
|
clock_t t_start;
|
|
clock_t t_end;
|
|
float run_time_sec;
|
|
|
|
int num_clocks = Arch.spice->spice_params.stimulate_params.num_clocks;
|
|
int vpr_crit_path_delay = Arch.spice->spice_params.stimulate_params.vpr_crit_path_delay;
|
|
|
|
char* spice_dir_formatted = NULL;
|
|
char* include_dir_path = NULL;
|
|
char* subckt_dir_path = NULL;
|
|
char* top_netlist_path = NULL;
|
|
char* include_dir_name = vpr_setup.FPGA_SPICE_Opts.SpiceOpts.include_dir;
|
|
char* subckt_dir_name = vpr_setup.FPGA_SPICE_Opts.SpiceOpts.subckt_dir;
|
|
char* chomped_circuit_name = NULL;
|
|
char* chomped_spice_dir = NULL;
|
|
char* top_testbench_dir_path = NULL;
|
|
char* pb_mux_testbench_dir_path = NULL;
|
|
char* cb_mux_testbench_dir_path = NULL;
|
|
char* sb_mux_testbench_dir_path = NULL;
|
|
char* cb_testbench_dir_path = NULL;
|
|
char* sb_testbench_dir_path = NULL;
|
|
char* grid_testbench_dir_path = NULL;
|
|
char* lut_testbench_dir_path = NULL;
|
|
char* hardlogic_testbench_dir_path = NULL;
|
|
char* io_testbench_dir_path = NULL;
|
|
char* top_testbench_file = NULL;
|
|
char* bitstream_file_name = NULL;
|
|
char* bitstream_file_path = NULL;
|
|
|
|
/* Check if the routing architecture we support*/
|
|
if (UNI_DIRECTIONAL != vpr_setup.RoutingArch.directionality) {
|
|
vpr_printf(TIO_MESSAGE_ERROR, "FPGA SPICE netlists only support uni-directional routing architecture!\n");
|
|
exit(1);
|
|
}
|
|
|
|
/* We don't support mrFPGA */
|
|
#ifdef MRFPGA_H
|
|
if (is_mrFPGA) {
|
|
vpr_printf(TIO_MESSAGE_ERROR, "FPGA SPICE netlists do not support mrFPGA!\n");
|
|
exit(1);
|
|
}
|
|
#endif
|
|
|
|
/* assign the global variable of SRAM model */
|
|
assert(NULL != Arch.sram_inf.spice_sram_inf_orgz); /* Check !*/
|
|
sram_spice_model = Arch.sram_inf.spice_sram_inf_orgz->spice_model;
|
|
sram_spice_orgz_type = Arch.sram_inf.spice_sram_inf_orgz->type;
|
|
/* initialize the SRAM organization information struct */
|
|
sram_spice_orgz_info = alloc_one_sram_orgz_info();
|
|
init_sram_orgz_info(sram_spice_orgz_info, sram_spice_orgz_type, sram_spice_model, nx + 2, ny + 2);
|
|
/* Report error: SPICE part only support standalone SRAMs */
|
|
if (SPICE_SRAM_STANDALONE != sram_spice_orgz_info->type) {
|
|
vpr_printf(TIO_MESSAGE_ERROR, "Currently FPGA SPICE netlist only support standalone SRAM organization!\n");
|
|
exit(1);
|
|
}
|
|
/* Check all the SRAM port is using the correct SRAM SPICE MODEL */
|
|
config_spice_models_sram_port_spice_model(Arch.spice->num_spice_model,
|
|
Arch.spice->spice_models,
|
|
Arch.sram_inf.spice_sram_inf_orgz->spice_model);
|
|
|
|
/* Assign global variables of input and output pads */
|
|
iopad_spice_model = find_iopad_spice_model(Arch.spice->num_spice_model, Arch.spice->spice_models);
|
|
assert(NULL != iopad_spice_model);
|
|
|
|
/* Initial Arch SPICE MODELS*/
|
|
/* zero the counter of each spice_model */
|
|
zero_spice_models_cnt(Arch.spice->num_spice_model, Arch.spice->spice_models);
|
|
|
|
/* Move to the top-level function: vpr_fpga_spice_tool_suits */
|
|
/* init_check_arch_spice_models(&Arch, &vpr_setup.RoutingArch); */
|
|
init_list_include_netlists(Arch.spice);
|
|
|
|
/* Initialize the number of configuration bits of all the grids */
|
|
init_grids_num_conf_bits(sram_spice_orgz_info);
|
|
init_grids_num_iopads();
|
|
|
|
/* Add keyword checking */
|
|
/* Move to the top-level function: vpr_fpga_spice_tool_suits */
|
|
/* check_keywords_conflict(Arch); */
|
|
|
|
/*Process the circuit name*/
|
|
split_path_prog_name(circuit_name,'/',&chomped_spice_dir ,&chomped_circuit_name);
|
|
|
|
/* Update the global variable :
|
|
* the number of mutli-thread used in SPICE simulator */
|
|
spice_sim_multi_thread_num = vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_sim_multi_thread_num;
|
|
|
|
/* FPGA-SPICE formally starts*/
|
|
vpr_printf(TIO_MESSAGE_INFO, "\nFPGA-SPICE starts...\n");
|
|
|
|
/* Start Clocking*/
|
|
t_start = clock();
|
|
|
|
/* Format the directory path */
|
|
if (NULL != vpr_setup.FPGA_SPICE_Opts.SpiceOpts.spice_dir) {
|
|
spice_dir_formatted = format_dir_path(vpr_setup.FPGA_SPICE_Opts.SpiceOpts.spice_dir);
|
|
} else {
|
|
spice_dir_formatted = format_dir_path(my_strcat(format_dir_path(chomped_spice_dir),
|
|
default_spice_dir_path));
|
|
}
|
|
|
|
/*Initial directory organization*/
|
|
/* Process include directory */
|
|
(include_dir_path) = my_strcat(spice_dir_formatted,include_dir_name);
|
|
/* Process subckt directory */
|
|
(subckt_dir_path) = my_strcat(spice_dir_formatted,subckt_dir_name);
|
|
|
|
/* Check the spice folders exists if not we create it.*/
|
|
create_dir_path(spice_dir_formatted);
|
|
create_dir_path(include_dir_path);
|
|
create_dir_path(subckt_dir_path);
|
|
|
|
/* Generate Header files */
|
|
spice_print_headers(include_dir_path, vpr_crit_path_delay, num_clocks, *(Arch.spice));
|
|
|
|
/* Generate sub circuits: Inverter, Buffer, Transmission Gate, LUT, DFF, SRAM, MUX*/
|
|
generate_spice_subckts(subckt_dir_path, &Arch ,&vpr_setup.RoutingArch);
|
|
|
|
/* Print MUX testbench if needed */
|
|
if (vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_print_pb_mux_testbench) {
|
|
pb_mux_testbench_dir_path = my_strcat(spice_dir_formatted, spice_pb_mux_tb_dir_name);
|
|
create_dir_path(pb_mux_testbench_dir_path);
|
|
spice_print_mux_testbench(pb_mux_testbench_dir_path, chomped_circuit_name,
|
|
include_dir_path, subckt_dir_path,
|
|
rr_node_indices, num_clocks, Arch,
|
|
SPICE_PB_MUX_TB,
|
|
vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_leakage_only);
|
|
/* Free */
|
|
my_free(pb_mux_testbench_dir_path);
|
|
}
|
|
|
|
if (vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_print_cb_mux_testbench) {
|
|
cb_mux_testbench_dir_path = my_strcat(spice_dir_formatted, spice_cb_mux_tb_dir_name);
|
|
create_dir_path(cb_mux_testbench_dir_path);
|
|
spice_print_mux_testbench(cb_mux_testbench_dir_path, chomped_circuit_name,
|
|
include_dir_path, subckt_dir_path,
|
|
rr_node_indices, num_clocks, Arch, SPICE_CB_MUX_TB,
|
|
vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_leakage_only);
|
|
/* Free */
|
|
my_free(cb_mux_testbench_dir_path);
|
|
}
|
|
|
|
if (vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_print_sb_mux_testbench) {
|
|
sb_mux_testbench_dir_path = my_strcat(spice_dir_formatted, spice_sb_mux_tb_dir_name);
|
|
create_dir_path(sb_mux_testbench_dir_path);
|
|
spice_print_mux_testbench(sb_mux_testbench_dir_path, chomped_circuit_name,
|
|
include_dir_path, subckt_dir_path,
|
|
rr_node_indices, num_clocks, Arch, SPICE_SB_MUX_TB,
|
|
vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_leakage_only);
|
|
/* Free */
|
|
my_free(sb_mux_testbench_dir_path);
|
|
}
|
|
|
|
if (vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_print_cb_testbench) {
|
|
cb_testbench_dir_path = my_strcat(spice_dir_formatted, spice_cb_tb_dir_name);
|
|
create_dir_path(cb_testbench_dir_path);
|
|
spice_print_cb_testbench(cb_testbench_dir_path, chomped_circuit_name,
|
|
include_dir_path, subckt_dir_path,
|
|
rr_node_indices, num_clocks, Arch,
|
|
vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_leakage_only);
|
|
/* Free */
|
|
my_free(cb_testbench_dir_path);
|
|
}
|
|
|
|
if (vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_print_sb_testbench) {
|
|
sb_testbench_dir_path = my_strcat(spice_dir_formatted, spice_sb_tb_dir_name);
|
|
create_dir_path(sb_testbench_dir_path);
|
|
spice_print_sb_testbench(sb_testbench_dir_path, chomped_circuit_name,
|
|
include_dir_path, subckt_dir_path,
|
|
rr_node_indices, num_clocks, Arch,
|
|
vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_leakage_only);
|
|
/* Free */
|
|
my_free(sb_testbench_dir_path);
|
|
}
|
|
|
|
if (vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_print_lut_testbench) {
|
|
lut_testbench_dir_path = my_strcat(spice_dir_formatted, spice_lut_tb_dir_name);
|
|
create_dir_path(lut_testbench_dir_path);
|
|
spice_print_primitive_testbench(lut_testbench_dir_path,
|
|
chomped_circuit_name, include_dir_path, subckt_dir_path,
|
|
rr_node_indices, num_clocks, Arch,
|
|
SPICE_LUT_TB,
|
|
vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_leakage_only);
|
|
/* Free */
|
|
my_free(lut_testbench_dir_path);
|
|
}
|
|
|
|
/* Print hardlogic testbench file if needed */
|
|
if (vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_print_hardlogic_testbench) {
|
|
hardlogic_testbench_dir_path = my_strcat(spice_dir_formatted, spice_hardlogic_tb_dir_name);
|
|
create_dir_path(hardlogic_testbench_dir_path);
|
|
spice_print_primitive_testbench(hardlogic_testbench_dir_path,
|
|
chomped_circuit_name, include_dir_path, subckt_dir_path,
|
|
rr_node_indices, num_clocks, Arch,
|
|
SPICE_HARDLOGIC_TB,
|
|
vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_leakage_only);
|
|
/* Free */
|
|
my_free(hardlogic_testbench_dir_path);
|
|
}
|
|
|
|
/* Print IO testbench file if needed */
|
|
if (vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_print_io_testbench) {
|
|
io_testbench_dir_path = my_strcat(spice_dir_formatted, spice_io_tb_dir_name);
|
|
create_dir_path(io_testbench_dir_path);
|
|
spice_print_primitive_testbench(io_testbench_dir_path,
|
|
chomped_circuit_name, include_dir_path, subckt_dir_path,
|
|
rr_node_indices, num_clocks, Arch,
|
|
SPICE_IO_TB,
|
|
vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_leakage_only);
|
|
/* Free */
|
|
my_free(io_testbench_dir_path);
|
|
}
|
|
|
|
|
|
/* Print Grid testbench if needed */
|
|
if (vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_print_grid_testbench) {
|
|
grid_testbench_dir_path = my_strcat(spice_dir_formatted, spice_grid_tb_dir_name);
|
|
create_dir_path(grid_testbench_dir_path);
|
|
spice_print_grid_testbench(grid_testbench_dir_path, chomped_circuit_name,
|
|
include_dir_path, subckt_dir_path,
|
|
rr_node_indices, num_clocks, Arch,
|
|
vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_leakage_only);
|
|
/* Free */
|
|
my_free(grid_testbench_dir_path);
|
|
}
|
|
|
|
/* Print Netlists of the given FPGA*/
|
|
if (vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_print_top_testbench) {
|
|
top_testbench_file = my_strcat(chomped_circuit_name, spice_top_testbench_postfix);
|
|
/* Process top_netlist_path */
|
|
top_testbench_dir_path = my_strcat(spice_dir_formatted, spice_top_tb_dir_name);
|
|
create_dir_path(top_testbench_dir_path);
|
|
top_netlist_path = my_strcat(top_testbench_dir_path, top_testbench_file);
|
|
spice_print_top_netlist(chomped_circuit_name, top_netlist_path,
|
|
include_dir_path, subckt_dir_path,
|
|
num_rr_nodes, rr_node, rr_node_indices, num_clocks, *(Arch.spice),
|
|
vpr_setup.FPGA_SPICE_Opts.SpiceOpts.fpga_spice_leakage_only);
|
|
/* Free */
|
|
my_free(top_testbench_dir_path);
|
|
my_free(top_testbench_file);
|
|
my_free(top_netlist_path);
|
|
}
|
|
|
|
if (vpr_setup.FPGA_SPICE_Opts.BitstreamGenOpts.gen_bitstream) {
|
|
if (NULL == vpr_setup.FPGA_SPICE_Opts.BitstreamGenOpts.bitstream_output_file) {
|
|
bitstream_file_name = my_strcat(chomped_circuit_name, fpga_spice_bitstream_output_file_postfix);
|
|
bitstream_file_path = my_strcat(spice_dir_formatted, bitstream_file_name);
|
|
} else {
|
|
bitstream_file_path = my_strdup(vpr_setup.FPGA_SPICE_Opts.BitstreamGenOpts.bitstream_output_file);
|
|
}
|
|
/* Dump bitstream file */
|
|
dump_fpga_spice_bitstream(bitstream_file_path, chomped_circuit_name, sram_spice_orgz_info);
|
|
/* Free */
|
|
my_free(bitstream_file_name);
|
|
my_free(bitstream_file_path);
|
|
}
|
|
|
|
/* Generate a shell script for running HSPICE simulations */
|
|
fprint_run_hspice_shell_script(*(Arch.spice), vpr_setup.FPGA_SPICE_Opts.SpiceOpts.simulator_path,
|
|
spice_dir_formatted, subckt_dir_path);
|
|
|
|
/* END Clocking*/
|
|
t_end = clock();
|
|
|
|
run_time_sec = (float)(t_end - t_start) / CLOCKS_PER_SEC;
|
|
vpr_printf(TIO_MESSAGE_INFO, "SPICE netlists dumping took %g seconds\n", run_time_sec);
|
|
|
|
/* Free sram_orgz_info */
|
|
free_sram_orgz_info(sram_spice_orgz_info,
|
|
sram_spice_orgz_info->type);
|
|
/* Free tb_llist */
|
|
free_spice_tb_llist();
|
|
/* Free */
|
|
my_free(spice_dir_formatted);
|
|
my_free(include_dir_path);
|
|
my_free(subckt_dir_path);
|
|
|
|
return;
|
|
}
|