#ifndef OPENFPGA_SDC_COMMAND_TEMPLATE_H #define OPENFPGA_SDC_COMMAND_TEMPLATE_H /******************************************************************** * Add commands to the OpenFPGA shell interface, * in purpose of generate SDC files * - write_pnr_sdc : generate SDC to constrain the back-end flow for FPGA fabric * - write_analysis_sdc: TODO: generate SDC based on users' implementations *******************************************************************/ #include "openfpga_sdc_template.h" #include "shell.h" /* begin namespace openfpga */ namespace openfpga { /******************************************************************** * - Add a command to Shell environment: generate PnR SDC * - Add associated options * - Add command dependency *******************************************************************/ template ShellCommandId add_write_pnr_sdc_command_template( openfpga::Shell& shell, const ShellCommandClassId& cmd_class_id, const std::vector& dependent_cmds) { Command shell_cmd("write_pnr_sdc"); /* Add an option '--file' in short '-f'*/ CommandOptionId output_opt = shell_cmd.add_option( "file", true, "Specify the output directory for SDC files"); shell_cmd.set_option_short_name(output_opt, "f"); shell_cmd.set_option_require_value(output_opt, openfpga::OPT_STRING); /* Add an option '--flatten_name' */ shell_cmd.add_option("flatten_names", false, "Use flatten names (no wildcards) in SDC files"); /* Add an option '--hierarchical' */ shell_cmd.add_option( "hierarchical", false, "Output SDC files hierachically (without full path in hierarchy)"); /* Add an option '--output_hierarchy' */ shell_cmd.add_option( "output_hierarchy", false, "Output hierachy of Multiple-Instance-Blocks (MIBs) to plain text file. " "This is applied to constrain timing for grid, SBs and CBs"); /* Add an option '--time_unit' */ CommandOptionId time_unit_opt = shell_cmd.add_option( "time_unit", false, "Specify the time unit in SDC files. Acceptable is [a|f|p|n|u|m|k|M]s"); shell_cmd.set_option_require_value(time_unit_opt, openfpga::OPT_STRING); /* Add an option '--constrain_global_port' */ shell_cmd.add_option("constrain_global_port", false, "Constrain all the global ports of FPGA fabric"); /* Add an option '--constrain_non_clock_global_port' */ shell_cmd.add_option( "constrain_non_clock_global_port", false, "Constrain all the non-clock global ports as clock ports of FPGA fabric"); /* Add an option '--constrain_grid' */ shell_cmd.add_option("constrain_grid", false, "Constrain all the grids of FPGA fabric"); /* Add an option '--constrain_sb' */ shell_cmd.add_option("constrain_sb", false, "Constrain all the switch blocks of FPGA fabric"); /* Add an option '--constrain_cb' */ shell_cmd.add_option("constrain_cb", false, "Constrain all the connection blocks of FPGA fabric"); /* Add an option '--constrain_configurable_memory_outputs' */ shell_cmd.add_option( "constrain_configurable_memory_outputs", false, "Constrain all the outputs of configurable memories of FPGA fabric"); /* Add an option '--constrain_routing_multiplexer_outputs' */ shell_cmd.add_option( "constrain_routing_multiplexer_outputs", false, "Constrain all the outputs of routing multiplexer of FPGA fabric"); /* Add an option '--constrain_switch_block_outputs' */ shell_cmd.add_option( "constrain_switch_block_outputs", false, "Constrain all the outputs of switch blocks of FPGA fabric"); /* Add an option '--constrain_zero_delay_paths' */ shell_cmd.add_option("constrain_zero_delay_paths", false, "Constrain zero-delay paths in FPGA fabric"); /* Add an option '--no_time_stamp' */ shell_cmd.add_option("no_time_stamp", false, "Do not print time stamp in output files"); /* Add an option '--verbose' */ shell_cmd.add_option("verbose", false, "Enable verbose output"); /* Add command 'write_fabric_verilog' to the Shell */ ShellCommandId shell_cmd_id = shell.add_command( shell_cmd, "generate SDC files to constrain the backend flow for FPGA fabric"); shell.set_command_class(shell_cmd_id, cmd_class_id); shell.set_command_const_execute_function(shell_cmd_id, write_pnr_sdc_template); /* Add command dependency to the Shell */ shell.set_command_dependency(shell_cmd_id, dependent_cmds); return shell_cmd_id; } /******************************************************************** * - Add a command to Shell environment: generate PnR SDC for configuration *chain * - Add associated options * - Add command dependency *******************************************************************/ template ShellCommandId add_write_configuration_chain_sdc_command_template( openfpga::Shell& shell, const ShellCommandClassId& cmd_class_id, const std::vector& dependent_cmds) { Command shell_cmd("write_configuration_chain_sdc"); /* Add an option '--file' in short '-f'*/ CommandOptionId output_opt = shell_cmd.add_option( "file", true, "Specify the SDC file to constrain configuration chain"); shell_cmd.set_option_short_name(output_opt, "f"); shell_cmd.set_option_require_value(output_opt, openfpga::OPT_STRING); /* Add an option '--time_unit' */ CommandOptionId time_unit_opt = shell_cmd.add_option( "time_unit", false, "Specify the time unit in SDC files. Acceptable is [a|f|p|n|u|m|k|M]s"); shell_cmd.set_option_require_value(time_unit_opt, openfpga::OPT_STRING); /* Add an option '--min_delay' */ CommandOptionId min_dly_opt = shell_cmd.add_option( "min_delay", false, "Specify the minimum delay to be used."); shell_cmd.set_option_require_value(min_dly_opt, openfpga::OPT_STRING); /* Add an option '--max_delay' */ CommandOptionId max_dly_opt = shell_cmd.add_option( "max_delay", false, "Specify the maximum delay to be used."); shell_cmd.set_option_require_value(max_dly_opt, openfpga::OPT_STRING); /* Add an option '--no_time_stamp' */ shell_cmd.add_option("no_time_stamp", false, "Do not print time stamp in output files"); /* Add command 'write_configuration_chain_sdc' to the Shell */ ShellCommandId shell_cmd_id = shell.add_command( shell_cmd, "generate SDC files to constrain the configuration chain for FPGA fabric"); shell.set_command_class(shell_cmd_id, cmd_class_id); shell.set_command_const_execute_function( shell_cmd_id, write_configuration_chain_sdc_template); /* Add command dependency to the Shell */ shell.set_command_dependency(shell_cmd_id, dependent_cmds); return shell_cmd_id; } /******************************************************************** * - Add a command to Shell environment: generate PnR SDC for configure ports * - Add associated options * - Add command dependency *******************************************************************/ template ShellCommandId add_write_sdc_disable_timing_configure_ports_command_template( openfpga::Shell& shell, const ShellCommandClassId& cmd_class_id, const std::vector& dependent_cmds) { Command shell_cmd("write_sdc_disable_timing_configure_ports"); /* Add an option '--file' in short '-f'*/ CommandOptionId output_opt = shell_cmd.add_option("file", true, "Specify the output directory"); shell_cmd.set_option_short_name(output_opt, "f"); shell_cmd.set_option_require_value(output_opt, openfpga::OPT_STRING); /* Add an option '--flatten_name' */ shell_cmd.add_option("flatten_names", false, "Use flatten names (no wildcards) in SDC files"); /* Add an option '--no_time_stamp' */ shell_cmd.add_option("no_time_stamp", false, "Do not print time stamp in output files"); /* Add an option '--verbose' */ shell_cmd.add_option("verbose", false, "Enable verbose outputs"); /* Add command 'write_configuration_chain_sdc' to the Shell */ ShellCommandId shell_cmd_id = shell.add_command(shell_cmd, "generate SDC files to disable timing for configure " "ports across FPGA fabric"); shell.set_command_class(shell_cmd_id, cmd_class_id); shell.set_command_const_execute_function( shell_cmd_id, write_sdc_disable_timing_configure_ports_template); /* Add command dependency to the Shell */ shell.set_command_dependency(shell_cmd_id, dependent_cmds); return shell_cmd_id; } /******************************************************************** * - Add a command to Shell environment: generate PnR SDC * - Add associated options * - Add command dependency *******************************************************************/ template ShellCommandId add_write_analysis_sdc_command_template( openfpga::Shell& shell, const ShellCommandClassId& cmd_class_id, const std::vector& dependent_cmds) { Command shell_cmd("write_analysis_sdc"); /* Add an option '--file' in short '-f'*/ CommandOptionId output_opt = shell_cmd.add_option( "file", true, "Specify the output directory for SDC files"); shell_cmd.set_option_short_name(output_opt, "f"); shell_cmd.set_option_require_value(output_opt, openfpga::OPT_STRING); /* Add an option '--verbose' */ shell_cmd.add_option("verbose", false, "Enable verbose output"); /* Add an option '--flatten_name' */ shell_cmd.add_option("flatten_names", false, "Use flatten names (no wildcards) in SDC files"); /* Add an option '--time_unit' */ CommandOptionId time_unit_opt = shell_cmd.add_option( "time_unit", false, "Specify the time unit in SDC files. Acceptable is [a|f|p|n|u|m|kM]s"); shell_cmd.set_option_require_value(time_unit_opt, openfpga::OPT_STRING); /* Add an option '--no_time_stamp' */ shell_cmd.add_option("no_time_stamp", false, "Do not print time stamp in output files"); /* Add command 'write_fabric_verilog' to the Shell */ ShellCommandId shell_cmd_id = shell.add_command(shell_cmd, "generate SDC files for timing analysis a PnRed FPGA " "fabric mapped by a benchmark"); shell.set_command_class(shell_cmd_id, cmd_class_id); shell.set_command_const_execute_function(shell_cmd_id, write_analysis_sdc_template); /* Add command dependency to the Shell */ shell.set_command_dependency(shell_cmd_id, dependent_cmds); return shell_cmd_id; } template void add_openfpga_sdc_command_templates(openfpga::Shell& shell) { /* Get the unique id of 'build_fabric' command which is to be used in creating * the dependency graph */ const ShellCommandId& build_fabric_id = shell.command(std::string("build_fabric")); /* Add a new class of commands */ ShellCommandClassId openfpga_sdc_cmd_class = shell.add_command_class("FPGA-SDC"); /******************************** * Command 'write_pnr_sdc' */ /* The 'write_pnr_sdc' command should NOT be executed before 'build_fabric' */ std::vector pnr_sdc_cmd_dependency; pnr_sdc_cmd_dependency.push_back(build_fabric_id); add_write_pnr_sdc_command_template(shell, openfpga_sdc_cmd_class, pnr_sdc_cmd_dependency); /******************************** * Command 'write_configuration_chain_sdc' */ /* The 'write_configuration_chain_sdc' command should NOT be executed before * 'build_fabric' */ std::vector cc_sdc_cmd_dependency; cc_sdc_cmd_dependency.push_back(build_fabric_id); add_write_configuration_chain_sdc_command_template( shell, openfpga_sdc_cmd_class, cc_sdc_cmd_dependency); /******************************** * Command 'write_sdc_disable_timing_configure_ports' */ /* The 'write_sdc_disable_timing_configure_ports' command should NOT be * executed before 'build_fabric' */ std::vector config_port_sdc_cmd_dependency; config_port_sdc_cmd_dependency.push_back(build_fabric_id); add_write_sdc_disable_timing_configure_ports_command_template( shell, openfpga_sdc_cmd_class, config_port_sdc_cmd_dependency); /******************************** * Command 'write_analysis_sdc' */ /* The 'write_analysis_sdc' command should NOT be executed before * 'build_fabric' */ std::vector analysis_sdc_cmd_dependency; analysis_sdc_cmd_dependency.push_back(build_fabric_id); add_write_analysis_sdc_command_template(shell, openfpga_sdc_cmd_class, analysis_sdc_cmd_dependency); } } /* end namespace openfpga */ #endif