From 2b179a4afafff4d50468eace31bb41144af1ab5c Mon Sep 17 00:00:00 2001 From: Thomas Benz Date: Fri, 22 Sep 2023 15:33:12 +0200 Subject: [PATCH] Initial work towards a config printer --- cheshire.mk | 1 + target/sim/src/cfg_format.c | 121 +++++++++++++++++++++++++ target/sim/src/fixture_cheshire_soc.sv | 8 ++ 3 files changed, 130 insertions(+) create mode 100644 target/sim/src/cfg_format.c diff --git a/cheshire.mk b/cheshire.mk index 5d5d20b2a..a45dc601e 100644 --- a/cheshire.mk +++ b/cheshire.mk @@ -135,6 +135,7 @@ CHS_BOOTROM_ALL += $(CHS_ROOT)/hw/bootrom/cheshire_bootrom.sv $(CHS_ROOT)/hw/boo $(CHS_ROOT)/target/sim/vsim/compile.cheshire_soc.tcl: Bender.yml $(BENDER) script vsim -t sim -t cv64a6_imafdcsclic_sv39 -t test -t cva6 -t rtl --vlog-arg="$(VLOG_ARGS)" > $@ echo 'vlog "$(CURDIR)/$(CHS_ROOT)/target/sim/src/elfloader.cpp" -ccflags "-std=c++11"' >> $@ + echo 'vlog "$(CURDIR)/$(CHS_ROOT)/target/sim/src/cfg_format.c"' >> $@ $(CHS_ROOT)/target/sim/models: mkdir -p $@ diff --git a/target/sim/src/cfg_format.c b/target/sim/src/cfg_format.c new file mode 100644 index 000000000..31e95e7c6 --- /dev/null +++ b/target/sim/src/cfg_format.c @@ -0,0 +1,121 @@ +// Copyright 2022 ETH Zurich and University of Bologna. +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// With a fuction copies from stackoverflow +// (https://stackoverflow.com/questions/8044081/how-to-do-regex-string-replacements-in-pure-c) +// +// Thomas Benz + + +#include +#include +#include +#include + +// replacement patterns +const char *NEWLINE_MATCH = ", ([A-Z])"; +const char *NEWLINE_REPLACE = "\n \1"; + +const char *COLON_MATCH = ":"; +const char *COLON_REPLACE = ": "; + + +int regex_replace(char **str, const char *pattern, const char *replace) { + // https://stackoverflow.com/questions/8044081/how-to-do-regex-string-replacements-in-pure-c + // replaces regex in pattern with replacement observing capture groups + // *str MUST be free-able, i.e. obtained by strdup, malloc, ... + // back references are indicated by char codes 1-31 and none of those chars can be used in the + // replacement string such as a tab. will not search for matches within replaced text, this + // will begin searching for the next match after the end of prev match + // returns: + // -1 if pattern cannot be compiled + // -2 if count of back references and capture groups don't match + // otherwise returns number of matches that were found and replaced + // + regex_t reg; + unsigned int replacements = 0; + // if regex can't commpile pattern, do nothing + if(!regcomp(®, pattern, REG_EXTENDED)) { + size_t nmatch = reg.re_nsub; + regmatch_t m[nmatch + 1]; + const char *rpl, *p; + // count back references in replace + int br = 0; + p = replace; + while(1) { + while(*++p > 31); + if(*p) br++; + else break; + } // if br is not equal to nmatch, leave + if(br != nmatch) { + regfree(®); + return -2; + } + // look for matches and replace + char *new; + char *search_start = *str; + while(!regexec(®, search_start, nmatch + 1, m, REG_NOTBOL)) { + // make enough room + new = (char *)malloc(strlen(*str) + strlen(replace)); + if(!new) exit(EXIT_FAILURE); + *new = '\0'; + strncat(new, *str, search_start - *str); + p = rpl = replace; + int c; + strncat(new, search_start, m[0].rm_so); // test before pattern + for(int k=0; k 31); // skip printable char + c = *p; // back reference (e.g. \1, \2, ...) + strncat(new, rpl, p - rpl); // add head of rpl + // concat match + strncat(new, search_start + m[c].rm_so, m[c].rm_eo - m[c].rm_so); + rpl = p++; // skip back reference, next match + } + strcat(new, p ); // trailing of rpl + unsigned int new_start_offset = strlen(new); + strcat(new, search_start + m[0].rm_eo); // trailing text in *str + free(*str); + *str = (char *)malloc(strlen(new)+1); + strcpy(*str,new); + search_start = *str + new_start_offset; + free(new); + replacements++; + } + regfree(®); + // ajust size + *str = (char *)realloc(*str, strlen(*str) + 1); + return replacements; + } else { + return -1; + } +} + + +// pretty printer +extern void format_cfg(char *cfg_str, int length) { + + // cut curly brackets away + char *cgf_str_cut = cfg_str + 2; + cgf_str_cut[length-3] = "\0"; + + // function works on a copy of the string + char *tmp = (char *)malloc(length + 1 - 4); + strcpy(tmp, cgf_str_cut); + + // infer newlines and fix colons + regex_replace(&tmp, NEWLINE_MATCH, NEWLINE_REPLACE); + regex_replace(&tmp, COLON_MATCH, COLON_REPLACE); + + // format numbers + regex_t find_num_pat; + size_t num_matches = 256; + regmatch_t match_array[256]; + regcomp(&find_num_pat, "[0-9]+", REG_EXTENDED); + regexec(&find_num_pat, tmp, num_matches, match_array, 0); + + // printf("%d", match_array[0].rm_eo); + + // print to questa terminal + printf("\nCheshire Configuration\n----------------------\n\n %s\n", tmp); +} diff --git a/target/sim/src/fixture_cheshire_soc.sv b/target/sim/src/fixture_cheshire_soc.sv index 00f0c4932..4c519d77c 100644 --- a/target/sim/src/fixture_cheshire_soc.sv +++ b/target/sim/src/fixture_cheshire_soc.sv @@ -20,6 +20,14 @@ module fixture_cheshire_soc #( `CHESHIRE_TYPEDEF_ALL(, DutCfg) + import "DPI-C" function void format_cfg(input string cfg_str, input int length); + + initial begin + static string cfg_str = $sformatf("%p", DutCfg); + format_cfg(cfg_str, cfg_str.len()); + end + + /////////// // DUT // ///////////