2022-10-06 19:08:50 -05:00
# include "config_protocol.h"
2023-04-22 02:12:38 -05:00
# include "openfpga_tokenizer.h"
2020-01-18 22:19:20 -06:00
# include "vtr_assert.h"
2021-09-23 16:25:25 -05:00
# include "vtr_log.h"
2020-01-18 22:19:20 -06:00
/************************************************************************
* Member functions for class ConfigProtocol
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/************************************************************************
* Constructors
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2023-04-22 02:12:38 -05:00
ConfigProtocol : : ConfigProtocol ( ) { INDICE_STRING_DELIM_ = ' , ' ; }
2020-01-18 22:19:20 -06:00
/************************************************************************
* Public Accessors
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2022-10-06 19:08:50 -05:00
e_config_protocol_type ConfigProtocol : : type ( ) const { return type_ ; }
2020-01-18 22:19:20 -06:00
std : : string ConfigProtocol : : memory_model_name ( ) const {
return memory_model_name_ ;
}
2022-10-06 19:08:50 -05:00
CircuitModelId ConfigProtocol : : memory_model ( ) const { return memory_model_ ; }
2020-01-18 22:19:20 -06:00
2022-10-06 19:08:50 -05:00
int ConfigProtocol : : num_regions ( ) const { return num_regions_ ; }
2020-09-28 14:51:43 -05:00
2023-04-23 01:26:08 -05:00
size_t ConfigProtocol : : num_prog_clocks ( ) const {
VTR_ASSERT ( type_ = = CONFIG_MEM_SCAN_CHAIN ) ;
return prog_clk_port_ . get_width ( ) ;
}
openfpga : : BasicPort ConfigProtocol : : prog_clock_port_info ( ) const {
VTR_ASSERT ( type_ = = CONFIG_MEM_SCAN_CHAIN ) ;
2023-04-23 00:57:37 -05:00
return prog_clk_port_ ;
}
std : : vector < openfpga : : BasicPort > ConfigProtocol : : prog_clock_pins ( ) const {
2023-04-23 01:26:08 -05:00
VTR_ASSERT ( type_ = = CONFIG_MEM_SCAN_CHAIN ) ;
2023-04-22 02:09:39 -05:00
std : : vector < openfpga : : BasicPort > keys ;
2023-04-23 00:57:37 -05:00
for ( auto pin : prog_clk_port_ . pins ( ) ) {
keys . push_back ( openfpga : : BasicPort ( prog_clk_port_ . get_name ( ) , pin , pin ) ) ;
2023-04-21 04:01:51 -05:00
}
return keys ;
}
2023-04-23 01:26:08 -05:00
std : : string ConfigProtocol : : prog_clock_pin_ccff_head_indices_str (
2023-04-22 02:12:38 -05:00
const openfpga : : BasicPort & port ) const {
2023-04-23 01:26:08 -05:00
VTR_ASSERT ( type_ = = CONFIG_MEM_SCAN_CHAIN ) ;
2023-04-21 04:01:51 -05:00
std : : string ret ( " " ) ;
2023-04-23 01:26:08 -05:00
std : : vector < size_t > raw = prog_clock_pin_ccff_head_indices ( port ) ;
2023-04-21 23:44:34 -05:00
if ( ! raw . empty ( ) ) {
for ( size_t idx : raw ) {
2023-04-21 04:01:51 -05:00
/* TODO: We need a join function */
2023-04-22 02:09:39 -05:00
ret + = std : : to_string ( idx ) + std : : to_string ( INDICE_STRING_DELIM_ ) ;
2023-04-21 04:01:51 -05:00
}
/* Remove the last comma */
2023-04-22 02:09:39 -05:00
ret . pop_back ( ) ;
2023-04-21 04:01:51 -05:00
}
return ret ;
}
2023-04-23 01:26:08 -05:00
std : : vector < size_t > ConfigProtocol : : prog_clock_pin_ccff_head_indices (
2023-04-22 02:12:38 -05:00
const openfpga : : BasicPort & port ) const {
2023-04-23 01:26:08 -05:00
VTR_ASSERT ( type_ = = CONFIG_MEM_SCAN_CHAIN ) ;
2023-04-21 23:44:34 -05:00
std : : vector < size_t > ret ;
2023-04-23 00:57:37 -05:00
if ( port . get_width ( ) ! = 1 ) {
VTR_LOG_ERROR ( " The programming clock pin must have a width of 1 while the width specified is %ld! \n " , port . get_width ( ) ) ;
2023-04-21 23:44:34 -05:00
}
2023-04-23 01:26:08 -05:00
VTR_ASSERT ( port . get_width ( ) = = 1 ) ;
2023-04-23 00:57:37 -05:00
if ( ! prog_clk_port_ . contained ( port ) ) {
VTR_LOG_ERROR ( " The programming clock pin '%s[%ld]' is not out of the range [%ld, %ld]! \n " , port . get_name ( ) . c_str ( ) , port . get_lsb ( ) , prog_clk_port_ . get_lsb ( ) , prog_clk_port_ . get_msb ( ) ) ;
}
VTR_ASSERT ( prog_clk_port_ . contained ( port ) ) ;
return prog_clk_ccff_head_indices_ [ port . get_lsb ( ) ] ;
2023-04-21 23:44:34 -05:00
}
2021-09-23 16:25:25 -05:00
e_blwl_protocol_type ConfigProtocol : : bl_protocol_type ( ) const {
return bl_protocol_type_ ;
}
std : : string ConfigProtocol : : bl_memory_model_name ( ) const {
return bl_memory_model_name_ ;
}
CircuitModelId ConfigProtocol : : bl_memory_model ( ) const {
return bl_memory_model_ ;
}
2022-10-06 19:08:50 -05:00
size_t ConfigProtocol : : bl_num_banks ( ) const { return bl_num_banks_ ; }
2021-09-28 16:20:35 -05:00
2021-09-23 16:25:25 -05:00
e_blwl_protocol_type ConfigProtocol : : wl_protocol_type ( ) const {
return wl_protocol_type_ ;
}
std : : string ConfigProtocol : : wl_memory_model_name ( ) const {
return wl_memory_model_name_ ;
}
CircuitModelId ConfigProtocol : : wl_memory_model ( ) const {
return wl_memory_model_ ;
}
2022-10-06 19:08:50 -05:00
size_t ConfigProtocol : : wl_num_banks ( ) const { return wl_num_banks_ ; }
2021-09-28 16:20:35 -05:00
2020-01-18 22:19:20 -06:00
/************************************************************************
* Public Mutators
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void ConfigProtocol : : set_type ( const e_config_protocol_type & type ) {
type_ = type ;
}
2022-10-06 19:08:50 -05:00
void ConfigProtocol : : set_memory_model_name (
const std : : string & memory_model_name ) {
2020-01-18 22:19:20 -06:00
memory_model_name_ = memory_model_name ;
}
void ConfigProtocol : : set_memory_model ( const CircuitModelId & memory_model ) {
memory_model_ = memory_model ;
}
2020-09-28 14:51:43 -05:00
void ConfigProtocol : : set_num_regions ( const int & num_regions ) {
num_regions_ = num_regions ;
}
2021-09-23 16:25:25 -05:00
2023-04-23 01:26:08 -05:00
void ConfigProtocol : : set_prog_clock_port ( const openfpga : : BasicPort & port ) {
2023-04-23 00:57:37 -05:00
prog_clk_port_ = port ;
prog_clk_ccff_head_indices_ . resize ( prog_clk_port_ . get_width ( ) ) ;
}
2023-04-23 01:26:08 -05:00
void ConfigProtocol : : set_prog_clock_pin_ccff_head_indices_pair (
2023-04-22 02:12:38 -05:00
const openfpga : : BasicPort & port , const std : : string & indices_str ) {
2023-04-21 04:01:51 -05:00
openfpga : : StringToken tokenizer ( indices_str ) ;
2023-04-22 02:09:39 -05:00
std : : vector < size_t > token_int ;
2023-04-21 04:01:51 -05:00
token_int . reserve ( tokenizer . split ( INDICE_STRING_DELIM_ ) . size ( ) ) ;
for ( std : : string token : tokenizer . split ( INDICE_STRING_DELIM_ ) ) {
2023-04-22 02:09:39 -05:00
token_int . push_back ( std : : stoi ( token ) ) ;
2023-04-21 04:01:51 -05:00
}
2023-04-23 00:57:37 -05:00
if ( port . get_width ( ) ! = 1 ) {
VTR_LOG_ERROR ( " The programming clock pin must have a width of 1 while the width specified is %ld! \n " , port . get_width ( ) ) ;
}
2023-04-23 01:26:08 -05:00
VTR_ASSERT ( port . get_width ( ) = = 1 ) ;
2023-04-23 00:57:37 -05:00
if ( ! prog_clk_port_ . contained ( port ) ) {
VTR_LOG_ERROR ( " The programming clock pin '%s[%ld]' is not out of the range [%ld, %ld]! \n " , port . get_name ( ) . c_str ( ) , port . get_lsb ( ) , prog_clk_port_ . get_lsb ( ) , prog_clk_port_ . get_msb ( ) ) ;
}
VTR_ASSERT ( prog_clk_port_ . contained ( port ) ) ;
if ( ! prog_clk_ccff_head_indices_ [ port . get_lsb ( ) ] . empty ( ) ) {
2023-04-22 02:12:38 -05:00
VTR_LOG_WARN (
" Overwrite the pair between programming clock port '%s[%d:%d]' and ccff "
" head indices (previous: '%s', current: '%s')! \n " ,
port . get_name ( ) . c_str ( ) , port . get_lsb ( ) , port . get_msb ( ) ,
2023-04-23 01:26:08 -05:00
prog_clock_pin_ccff_head_indices_str ( port ) . c_str ( ) , indices_str . c_str ( ) ) ;
2023-04-21 04:01:51 -05:00
}
2023-04-23 00:57:37 -05:00
prog_clk_ccff_head_indices_ [ port . get_lsb ( ) ] = token_int ;
2023-04-21 04:01:51 -05:00
}
2021-09-23 16:25:25 -05:00
void ConfigProtocol : : set_bl_protocol_type ( const e_blwl_protocol_type & type ) {
if ( CONFIG_MEM_QL_MEMORY_BANK ! = type_ ) {
2022-10-06 19:08:50 -05:00
VTR_LOG_ERROR (
" BL protocol type is only applicable for configuration protocol '%d' " ,
CONFIG_PROTOCOL_TYPE_STRING [ type_ ] ) ;
2021-09-23 16:25:25 -05:00
return ;
}
bl_protocol_type_ = type ;
}
2022-10-06 19:08:50 -05:00
void ConfigProtocol : : set_bl_memory_model_name (
const std : : string & memory_model_name ) {
2021-09-23 16:25:25 -05:00
if ( BLWL_PROTOCOL_SHIFT_REGISTER ! = bl_protocol_type_ ) {
2022-10-06 19:08:50 -05:00
VTR_LOG_ERROR (
" BL protocol memory model is only applicable when '%d' is defined " ,
BLWL_PROTOCOL_TYPE_STRING [ bl_protocol_type_ ] ) ;
2021-09-23 16:25:25 -05:00
return ;
}
bl_memory_model_name_ = memory_model_name ;
}
void ConfigProtocol : : set_bl_memory_model ( const CircuitModelId & memory_model ) {
if ( BLWL_PROTOCOL_SHIFT_REGISTER ! = bl_protocol_type_ ) {
2022-10-06 19:08:50 -05:00
VTR_LOG_ERROR (
" BL protocol memory model is only applicable when '%d' is defined " ,
BLWL_PROTOCOL_TYPE_STRING [ bl_protocol_type_ ] ) ;
2021-09-23 16:25:25 -05:00
return ;
}
bl_memory_model_ = memory_model ;
}
2021-09-28 16:20:35 -05:00
void ConfigProtocol : : set_bl_num_banks ( const size_t & num_banks ) {
if ( BLWL_PROTOCOL_SHIFT_REGISTER ! = bl_protocol_type_ ) {
2022-10-06 19:08:50 -05:00
VTR_LOG_ERROR (
" BL protocol memory model is only applicable when '%d' is defined " ,
BLWL_PROTOCOL_TYPE_STRING [ bl_protocol_type_ ] ) ;
2021-09-28 16:20:35 -05:00
return ;
}
bl_num_banks_ = num_banks ;
}
2021-09-23 16:25:25 -05:00
void ConfigProtocol : : set_wl_protocol_type ( const e_blwl_protocol_type & type ) {
if ( CONFIG_MEM_QL_MEMORY_BANK ! = type_ ) {
2022-10-06 19:08:50 -05:00
VTR_LOG_ERROR (
" WL protocol type is only applicable for configuration protocol '%d' " ,
CONFIG_PROTOCOL_TYPE_STRING [ type_ ] ) ;
2021-09-23 16:25:25 -05:00
return ;
}
wl_protocol_type_ = type ;
}
2022-10-06 19:08:50 -05:00
void ConfigProtocol : : set_wl_memory_model_name (
const std : : string & memory_model_name ) {
2021-09-23 16:25:25 -05:00
if ( BLWL_PROTOCOL_SHIFT_REGISTER ! = wl_protocol_type_ ) {
2022-10-06 19:08:50 -05:00
VTR_LOG_ERROR (
" WL protocol memory model is only applicable when '%d' is defined " ,
BLWL_PROTOCOL_TYPE_STRING [ wl_protocol_type_ ] ) ;
2021-09-23 16:25:25 -05:00
return ;
}
wl_memory_model_name_ = memory_model_name ;
}
void ConfigProtocol : : set_wl_memory_model ( const CircuitModelId & memory_model ) {
if ( BLWL_PROTOCOL_SHIFT_REGISTER ! = wl_protocol_type_ ) {
2022-10-06 19:08:50 -05:00
VTR_LOG_ERROR (
" WL protocol memory model is only applicable when '%d' is defined " ,
BLWL_PROTOCOL_TYPE_STRING [ wl_protocol_type_ ] ) ;
2021-09-23 16:25:25 -05:00
return ;
}
wl_memory_model_ = memory_model ;
}
2021-09-28 16:20:35 -05:00
void ConfigProtocol : : set_wl_num_banks ( const size_t & num_banks ) {
if ( BLWL_PROTOCOL_SHIFT_REGISTER ! = wl_protocol_type_ ) {
2022-10-06 19:08:50 -05:00
VTR_LOG_ERROR (
" WL protocol memory model is only applicable when '%d' is defined " ,
BLWL_PROTOCOL_TYPE_STRING [ wl_protocol_type_ ] ) ;
2021-09-28 16:20:35 -05:00
return ;
}
wl_num_banks_ = num_banks ;
}
2023-04-21 23:44:34 -05:00
/************************************************************************
* Private Validators
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2023-04-22 02:09:39 -05:00
int ConfigProtocol : : validate_ccff_prog_clocks ( ) const {
2023-04-21 23:44:34 -05:00
int num_err = 0 ;
/* Initialize scoreboard */
std : : vector < int > ccff_head_scoreboard ( num_regions ( ) , 0 ) ;
2023-04-23 01:26:08 -05:00
for ( openfpga : : BasicPort port : prog_clock_pins ( ) ) {
2023-04-22 02:12:38 -05:00
/* Must be valid first */
2023-04-21 23:44:34 -05:00
if ( port . is_valid ( ) ) {
2023-04-22 02:12:38 -05:00
VTR_LOG_ERROR ( " Programming clock '%s[%d:%d]' is not a valid port! \n " ,
port . get_name ( ) . c_str ( ) , port . get_lsb ( ) , port . get_msb ( ) ) ;
2023-04-21 23:44:34 -05:00
num_err + + ;
}
2023-04-22 02:12:38 -05:00
/* Each port should have a width of 1 */
2023-04-21 23:44:34 -05:00
if ( port . get_width ( ) ! = 1 ) {
2023-04-22 02:12:38 -05:00
VTR_LOG_ERROR (
" Expect each programming clock has a size of 1 in the definition. "
" '%s[%d:%d]' violates the rule! \n " ,
port . get_name ( ) . c_str ( ) , port . get_lsb ( ) , port . get_msb ( ) ) ;
2023-04-21 23:44:34 -05:00
num_err + + ;
}
/* Fill scoreboard */
2023-04-23 01:26:08 -05:00
for ( size_t ccff_head_idx : prog_clock_pin_ccff_head_indices ( port ) ) {
2023-04-22 02:09:39 -05:00
if ( ccff_head_idx > = ccff_head_scoreboard . size ( ) ) {
2023-04-22 02:12:38 -05:00
VTR_LOG_ERROR (
" Programming clock '%s[%d:%d]' controlls an invalid ccff head '%ld' "
" (Expect [0, '%ld'])! \n " ,
port . get_name ( ) . c_str ( ) , port . get_lsb ( ) , port . get_msb ( ) ,
ccff_head_idx , ccff_head_scoreboard . size ( ) - 1 ) ;
2023-04-21 23:44:34 -05:00
num_err + + ;
}
ccff_head_scoreboard [ ccff_head_idx ] + + ;
}
}
2023-04-23 01:26:08 -05:00
if ( prog_clock_pins ( ) . size ( ) ! = ( size_t ) num_regions ( ) ) {
2023-04-22 02:12:38 -05:00
VTR_LOG_ERROR (
" Number of programming clocks '%ld' does not match the number of "
" configuration regions '%ld'! \n " ,
2023-04-23 01:26:08 -05:00
prog_clock_pins ( ) . size ( ) , num_regions ( ) ) ;
2023-04-21 23:44:34 -05:00
num_err + + ;
}
for ( size_t iregion = 0 ; iregion < ccff_head_scoreboard . size ( ) ; iregion + + ) {
if ( ccff_head_scoreboard [ iregion ] = = 0 ) {
2023-04-22 02:12:38 -05:00
VTR_LOG_ERROR (
" Configuration chain '%ld' is not driven by any programming clock! \n " ,
iregion ) ;
2023-04-21 23:44:34 -05:00
num_err + + ;
}
if ( ccff_head_scoreboard [ iregion ] > 1 ) {
2023-04-22 02:12:38 -05:00
VTR_LOG_ERROR (
" Configuration chain '%ld' is driven by %ld programming clock! \n " ,
iregion , ccff_head_scoreboard [ iregion ] ) ;
2023-04-21 23:44:34 -05:00
num_err + + ;
}
}
2023-04-22 02:12:38 -05:00
return num_err ;
2023-04-21 23:44:34 -05:00
}
/************************************************************************
* Public Validators
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2023-04-22 02:09:39 -05:00
int ConfigProtocol : : validate ( ) const {
2023-04-21 23:44:34 -05:00
int num_err = 0 ;
if ( type ( ) = = CONFIG_MEM_SCAN_CHAIN ) {
num_err + = validate_ccff_prog_clocks ( ) ;
}
2023-04-22 02:12:38 -05:00
return num_err ;
2023-04-21 23:44:34 -05:00
}