- some more work for the save/restore feature in the siminterface

* new parameter object type bx_shadow_data_c for binary arrays (e.g. system
    and video memory, CMOS RAM, PCI config space)
  * save_state() completed and restore_hardware() implemented
  * get_sr_root() returns the list with the save/restore subtree
This commit is contained in:
Volker Ruppert 2006-04-15 14:05:18 +00:00
parent a8bfc2d2bf
commit 14107f56fa
2 changed files with 206 additions and 13 deletions

View File

@ -1,5 +1,5 @@
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// $Id: siminterface.cc,v 1.140 2006-04-14 08:07:24 vruppert Exp $ // $Id: siminterface.cc,v 1.141 2006-04-15 14:05:18 vruppert Exp $
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// //
// See siminterface.h for description of the siminterface concept. // See siminterface.h for description of the siminterface concept.
@ -146,6 +146,11 @@ public:
virtual bx_bool restore_config(); virtual bx_bool restore_config();
virtual bx_bool restore_logopts(); virtual bx_bool restore_logopts();
virtual bx_bool restore_hardware(); virtual bx_bool restore_hardware();
virtual bx_list_c *get_sr_root() {
return (bx_list_c*)get_param("save_restore", NULL);
}
private:
void save_sr_param(FILE *fp, bx_param_c *node, const char *sr_path);
#endif #endif
}; };
@ -853,17 +858,16 @@ Bit32s bx_real_sim_c::save_user_options(FILE *fp)
#if BX_SUPPORT_SAVE_RESTORE #if BX_SUPPORT_SAVE_RESTORE
bx_bool bx_real_sim_c::save_state(const char *checkpoint_path) bx_bool bx_real_sim_c::save_state(const char *checkpoint_path)
{ {
char config[BX_PATHNAME_LEN]; char sr_file[BX_PATHNAME_LEN];
char logopts[BX_PATHNAME_LEN];
char prefix[8]; char prefix[8];
int i, dev, ndev = SIM->get_n_log_modules(); int i, dev, ndev = SIM->get_n_log_modules();
int type, ntype = SIM->get_max_log_level(); int type, ntype = SIM->get_max_log_level();
FILE *fp; FILE *fp;
sprintf(config, "%s/config", checkpoint_path); sprintf(sr_file, "%s/config", checkpoint_path);
write_rc(config, 1); write_rc(sr_file, 1);
sprintf(logopts, "%s/logopts", checkpoint_path); sprintf(sr_file, "%s/logopts", checkpoint_path);
fp = fopen(logopts, "w"); fp = fopen(sr_file, "w");
if (fp != NULL) { if (fp != NULL) {
for (dev=0; dev<ndev; dev++) { for (dev=0; dev<ndev; dev++) {
strcpy(prefix, get_prefix(dev)); strcpy(prefix, get_prefix(dev));
@ -882,8 +886,16 @@ bx_bool bx_real_sim_c::save_state(const char *checkpoint_path)
} }
fclose(fp); fclose(fp);
} }
// TODO bx_list_c *sr_list = get_sr_root();
fprintf(stderr, "save_state (not yet complete)\n"); ndev = sr_list->get_size();
for (dev=0; dev<ndev; dev++) {
sprintf(sr_file, "%s/%s", checkpoint_path, sr_list->get(dev)->get_name());
fp = fopen(sr_file, "w");
if (fp != NULL) {
save_sr_param(fp, sr_list->get(dev), checkpoint_path);
fclose(fp);
}
}
return 0; return 0;
} }
@ -900,11 +912,11 @@ bx_bool bx_real_sim_c::restore_config()
bx_bool bx_real_sim_c::restore_logopts() bx_bool bx_real_sim_c::restore_logopts()
{ {
char logopts[BX_PATHNAME_LEN]; char logopts[BX_PATHNAME_LEN];
FILE *fp;
char line[512], string[512], prefix[8]; char line[512], string[512], prefix[8];
char *ret, *ptr; char *ret, *ptr;
int d, i, j, dev = 0, type = 0, action = 0; int d, i, j, dev = 0, type = 0, action = 0;
int ndev = SIM->get_n_log_modules(); int ndev = SIM->get_n_log_modules();
FILE *fp;
sprintf(logopts, "%s/logopts", get_param_string(BXPN_RESTORE_PATH)->getptr()); sprintf(logopts, "%s/logopts", get_param_string(BXPN_RESTORE_PATH)->getptr());
BX_INFO(("restoring '%s'", logopts)); BX_INFO(("restoring '%s'", logopts));
@ -973,10 +985,155 @@ bx_bool bx_real_sim_c::restore_logopts()
bx_bool bx_real_sim_c::restore_hardware() bx_bool bx_real_sim_c::restore_hardware()
{ {
// TODO char devstate[BX_PATHNAME_LEN], devdata[BX_PATHNAME_LEN];
fprintf(stderr, "restore_hardware (not implemented yet)\n"); char line[512], buf[512];
char *ret, *ptr;
int i, j, p, dev, ndev;
unsigned int n;
bx_param_c *param = NULL;
FILE *fp, *fp2;
bx_list_c *sr_list = get_sr_root();
ndev = sr_list->get_size();
for (dev=0; dev<ndev; dev++) {
sprintf(devstate, "%s/%s", get_param_string(BXPN_RESTORE_PATH)->getptr(), sr_list->get(dev)->get_name());
BX_INFO(("restoring '%s'", devstate));
fp = fopen(devstate, "r");
if (fp != NULL) {
do {
ret = fgets(line, sizeof(line)-1, fp);
line[sizeof(line) - 1] = '\0';
int len = strlen(line);
if ((len>0) && (line[len-1] < ' '))
line[len-1] = '\0';
i = 0;
if ((ret != NULL) && strlen(line)) {
ptr = strtok(line, " ");
while (ptr) {
if (i == 0) {
param = get_param(ptr, NULL);
} else if (i == 2) {
switch (param->get_type()) {
case BXT_PARAM_NUM:
if ((ptr[0] == '0') && (ptr[1] == 'x')) {
((bx_param_num_c*)param)->set(strtoul(ptr, NULL, 16));
} else {
((bx_param_num_c*)param)->set(strtoul(ptr, NULL, 10));
}
break;
case BXT_PARAM_BOOL:
((bx_param_bool_c*)param)->set(!strcmp(ptr, "true"));
break;
case BXT_PARAM_ENUM:
((bx_param_enum_c*)param)->set_by_name(ptr);
break;
case BXT_PARAM_STRING:
if (((bx_param_string_c*)param)->get_options()->get() & bx_param_string_c::RAW_BYTES) {
p = 0;
for (j = 0; j < ((bx_param_string_c*)param)->get_maxsize(); j++) {
if (ptr[p] == ((bx_param_string_c*)param)->get_separator()) {
p++;
}
if (sscanf(ptr+p, "%02x", &n) == 1) {
buf[j] = n;
p += 2;
}
}
((bx_param_string_c*)param)->set(buf);
} else {
((bx_param_string_c*)param)->set(ptr);
}
break;
case BXT_PARAM_DATA:
sprintf(devdata, "%s/%s", get_param_string(BXPN_RESTORE_PATH)->getptr(), ptr);
fp2 = fopen(devdata, "rb");
if (fp2 != NULL) {
fread(((bx_shadow_data_c*)param)->getptr(), 1, ((bx_shadow_data_c*)param)->get_size(), fp2);
fclose(fp2);
}
break;
default:
BX_ERROR(("restore_hardware(): unknown parameter type"));
}
}
printf("%d: %s\n", i, ptr);
i++;
ptr = strtok(NULL, " ");
}
}
} while (!feof(fp));
fclose(fp);
}
}
return 0; return 0;
} }
void bx_real_sim_c::save_sr_param(FILE *fp, bx_param_c *node, const char *sr_path)
{
int i;
char tmpstr[BX_PATHNAME_LEN], tmpbyte[4];
FILE *fp2;
if (node == NULL) {
BX_ERROR(("NULL pointer"));
return;
}
if (node->get_type() != BXT_LIST) {
node->get_param_path(tmpstr, BX_PATHNAME_LEN);
fprintf(fp, "%s = ", tmpstr);
}
switch (node->get_type()) {
case BXT_PARAM_NUM:
if (((bx_param_num_c*)node)->get_base() == 10) {
fprintf(fp, "%d\n", ((bx_param_num_c*)node)->get());
} else {
fprintf(fp, "0x%x\n", ((bx_param_num_c*)node)->get());
}
break;
case BXT_PARAM_BOOL:
fprintf(fp, "%s\n", ((bx_param_bool_c*)node)->get()?"true":"false");
break;
case BXT_PARAM_ENUM:
fprintf(fp, "%s\n", ((bx_param_enum_c*)node)->get_selected());
break;
case BXT_PARAM_STRING:
if (((bx_param_string_c*)node)->get_options()->get() & bx_param_string_c::RAW_BYTES) {
tmpstr[0] = 0;
for (i = 0; i < ((bx_param_string_c*)node)->get_maxsize(); i++) {
if (i > 0) {
tmpbyte[0] = ((bx_param_string_c*)node)->get_separator();
tmpbyte[1] = 0;
strcat(tmpstr, tmpbyte);
}
sprintf(tmpbyte, "%02x", (Bit8u)((bx_param_string_c*)node)->getptr()[i]);
strcat(tmpstr, tmpbyte);
}
fprintf(fp, "%s\n", tmpstr);
} else {
fprintf(fp, "%s\n", ((bx_param_string_c*)node)->getptr());
}
break;
case BXT_PARAM_DATA:
fprintf(fp, "%s.%s", node->get_parent()->get_name(), node->get_name());
sprintf(tmpstr, "%s/%s.%s", sr_path, node->get_parent()->get_name(), node->get_name());
fp2 = fopen(tmpstr, "wb");
if (fp != NULL) {
fwrite(((bx_shadow_data_c*)node)->getptr(), 1, ((bx_shadow_data_c*)node)->get_size(), fp2);
fclose(fp2);
}
break;
case BXT_LIST:
{
bx_list_c *list = (bx_list_c*)node;
for (i=0; i < list->get_size(); i++) {
save_sr_param(fp, list->get(i), sr_path);
}
break;
}
default:
BX_ERROR(("save_sr_param(): unknown parameter type"));
}
}
#endif #endif
@ -1542,6 +1699,25 @@ void bx_param_string_c::set_initial_val(char *buf) {
set(initial_val); set(initial_val);
} }
#if BX_SUPPORT_SAVE_RESTORE
bx_shadow_data_c::bx_shadow_data_c(bx_param_c *parent,
char *name,
char *description,
Bit8u *ptr_to_data,
Bit32u data_size)
: bx_param_c(SIM->gen_param_id(), name, description)
{
set_type(BXT_PARAM_DATA);
this->data_ptr = ptr_to_data;
this->data_size = data_size;
if (parent) {
BX_ASSERT(parent->get_type() == BXT_LIST);
this->parent = (bx_list_c *)parent;
this->parent->add(this);
}
}
#endif
bx_list_c::bx_list_c(bx_param_c *parent, int maxsize) bx_list_c::bx_list_c(bx_param_c *parent, int maxsize)
: bx_param_c(SIM->gen_param_id(), "list", "") : bx_param_c(SIM->gen_param_id(), "list", "")
{ {

View File

@ -1,5 +1,5 @@
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// $Id: siminterface.h,v 1.185 2006-04-14 08:07:24 vruppert Exp $ // $Id: siminterface.h,v 1.186 2006-04-15 14:05:18 vruppert Exp $
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// //
// Intro to siminterface by Bryce Denney: // Intro to siminterface by Bryce Denney:
@ -113,6 +113,7 @@ typedef enum {
BXT_PARAM_BOOL, BXT_PARAM_BOOL,
BXT_PARAM_ENUM, BXT_PARAM_ENUM,
BXT_PARAM_STRING, BXT_PARAM_STRING,
BXT_PARAM_DATA,
BXT_LIST BXT_LIST
} bx_objtype; } bx_objtype;
@ -860,6 +861,21 @@ public:
int maxsize=-1); int maxsize=-1);
}; };
#if BX_SUPPORT_SAVE_RESTORE
class BOCHSAPI bx_shadow_data_c : public bx_param_c {
Bit32u data_size;
Bit8u *data_ptr;
public:
bx_shadow_data_c(bx_param_c *parent,
char *name,
char *description,
Bit8u *ptr_to_data,
Bit32u data_size);
Bit8u *getptr() {return data_ptr;}
Bit32u get_size() {return data_size;}
};
#endif
#define BX_DEFAULT_LIST_SIZE 6 #define BX_DEFAULT_LIST_SIZE 6
class BOCHSAPI bx_list_c : public bx_param_c { class BOCHSAPI bx_list_c : public bx_param_c {
@ -1187,6 +1203,7 @@ public:
virtual bx_bool restore_config() {return 0;} virtual bx_bool restore_config() {return 0;}
virtual bx_bool restore_logopts() {return 0;} virtual bx_bool restore_logopts() {return 0;}
virtual bx_bool restore_hardware() {return 0;} virtual bx_bool restore_hardware() {return 0;}
virtual bx_list_c *get_sr_root() {return NULL;}
#endif #endif
}; };