[Author Prev][Author Next][Thread Prev][Thread Next][Author Index][Thread Index]

[or-cvs] Possibly broken implementation of persistant state; handles...



Update of /home/or/cvsroot/tor/src/or
In directory moria:/tmp/cvs-serv22251/src/or

Modified Files:
	circuitbuild.c config.c hibernate.c main.c or.h 
Log Message:
Possibly broken implementation of persistant state; handles helper nodes; does not handle accounting info yet.

Index: circuitbuild.c
===================================================================
RCS file: /home/or/cvsroot/tor/src/or/circuitbuild.c,v
retrieving revision 1.128
retrieving revision 1.129
diff -u -d -r1.128 -r1.129
--- circuitbuild.c	23 Jul 2005 04:35:58 -0000	1.128
+++ circuitbuild.c	28 Jul 2005 19:01:48 -0000	1.129
@@ -31,6 +31,7 @@
 
 /** A list of our chosen helper nodes. */
 static smartlist_t *helper_nodes = NULL;
+static int helper_nodes_dirty = 0;
 
 /********* END VARIABLES ************/
 
@@ -46,6 +47,7 @@
 static routerinfo_t *choose_random_helper(void);
 static void clear_helper_nodes(void);
 static void remove_dead_helpers(void);
+static void helper_nodes_changed(void);
 
 /** Iterate over values of circ_id, starting from conn-\>next_circ_id,
  * and with the high bit specified by circ_id_type (see
@@ -1635,6 +1637,7 @@
     strlcpy(helper->nickname, entry->nickname, sizeof(helper->nickname));
     memcpy(helper->identity, entry->identity_digest, DIGEST_LEN);
     smartlist_add(helper_nodes, helper);
+    helper_nodes_changed();
   }
 }
 
@@ -1644,6 +1647,7 @@
 {
   SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h, tor_free(h));
   smartlist_clear(helper_nodes);
+  helper_nodes_changed();
 }
 
 /** How long (in seconds) do we allow a helper node to be nonfunctional before
@@ -1681,6 +1685,7 @@
              helper->nickname, dbuf, why, tbuf);
       tor_free(helper);
       smartlist_del(helper_nodes, i);
+      helper_nodes_changed();
     } else
       ++i;
   }
@@ -1747,9 +1752,11 @@
       }
     });
 
-  if (changed)
+  if (changed) {
     log_fn(LOG_WARN, "    (%d/%d helpers are usable)",
            num_live_helpers(), smartlist_len(helper_nodes));
+    helper_nodes_changed();
+  }
 
   remove_dead_helpers();
   pick_helper_nodes();
@@ -1775,6 +1782,7 @@
                    "Connection to formerly down helper node '%s' succeeded. "
                    "%d/%d helpers usable.", helper->nickname,
                    num_live_helpers(), smartlist_len(helper_nodes));
+            helper_nodes_changed();
           }
           helper->down_since = 0;
         } else if (!helper->down_since) {
@@ -1782,6 +1790,7 @@
           log_fn(LOG_WARN,
                  "Connection to helper node '%s' failed. %d/%d helpers usable.",
                  helper->nickname, num_live_helpers(), smartlist_len(helper_nodes));
+          helper_nodes_changed();
         }
       }
     });
@@ -1818,3 +1827,115 @@
   return r;
 }
 
+/** DOCDOC */
+int
+helper_nodes_parse_state(or_state_t *state, int set, const char **err)
+{
+  helper_node_t *node = NULL;
+  smartlist_t *helpers = smartlist_create();
+  config_line_t *line;
+
+  *err = NULL;
+  for (line = state->HelperNodes; line; line = line->next) {
+    if (!strcasecmp(line->key, "HelperNode")) {
+      smartlist_t *args = smartlist_create();
+      node = tor_malloc_zero(sizeof(helper_node_t));
+      smartlist_add(helpers, node);
+      smartlist_split_string(args, line->value, " ",
+                             SPLIT_SKIP_SPACE|SPLIT_IGNORE_BLANK, 0);
+      if (smartlist_len(args)<2) {
+        *err = "Too few arguments to HelperNode";
+        break;
+      }
+      if (!is_legal_nickname(smartlist_get(args,0))) {
+        *err = "Bad nickname for HelperNode";
+        break;
+      }
+      strlcpy(node->nickname, smartlist_get(args,0), MAX_NICKNAME_LEN+1);
+      if (base16_decode(node->identity, DIGEST_LEN, smartlist_get(args,1),
+                        strlen(smartlist_get(args,11)))<0) {
+        *err = "Bad hex digest for HelperNode";
+        break;
+      }
+    } else {
+      time_t when;
+      if (!node) {
+        *err = "HelperNodeDownSince/UnlistedSince without HelperNode";
+        break;
+      }
+      if (parse_iso_time(line->value, &when)<0) {
+        *err = "Bad time in HelperNodeDownSince/UnlistedSince";
+        break;
+      }
+      if (!strcasecmp(line->key, "HelperNodeDownSince"))
+        node->down_since = when;
+      else
+        node->unlisted_since = when;
+    }
+  }
+
+  if (*err || !set) {
+    SMARTLIST_FOREACH(helpers, helper_node_t *, h, tor_free(h));
+    smartlist_free(helpers);
+  }
+  if (!*err && set) {
+    if (helper_nodes) {
+      SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h, tor_free(h));
+      smartlist_free(helper_nodes);
+    }
+    helper_nodes = helpers;
+    helper_nodes_dirty = 0;
+  }
+  return *err ? -1 : 0;
+}
+
+/** DOCDOC */
+static void
+helper_nodes_changed(void)
+{
+  helper_nodes_dirty = 1;
+
+  or_state_save();
+}
+
+/** DOCDOC */
+int
+helper_nodes_update_state(or_state_t *state)
+{
+  config_line_t **next, *line;
+  if (! helper_nodes_dirty)
+    return 0;
+
+  config_free_lines(state->HelperNodes);
+  next = &state->HelperNodes;
+  *next = NULL;
+  SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h,
+    {
+      char dbuf[HEX_DIGEST_LEN+1];
+      *next = line = tor_malloc_zero(sizeof(config_line_t));
+      line->key = tor_strdup("HelperNode");
+      line->value = tor_malloc(HEX_DIGEST_LEN+MAX_NICKNAME_LEN+2);
+      base16_encode(dbuf, sizeof(dbuf), h->identity, DIGEST_LEN);
+      tor_snprintf(line->value,HEX_DIGEST_LEN+MAX_NICKNAME_LEN+2,
+                   "%s %s", h->nickname, dbuf);
+      next = &(line->next);
+      if (h->down_since) {
+        *next = line = tor_malloc_zero(sizeof(config_line_t));
+        line->key = tor_strdup("HelperNodeDownSince");
+        line->value = tor_malloc(ISO_TIME_LEN+1);
+        format_iso_time(line->value, h->down_since);
+        next = &(line->next);
+      }
+      if (h->unlisted_since) {
+        *next = line = tor_malloc_zero(sizeof(config_line_t));
+        line->key = tor_strdup("HelperNodeUnlistedSince");
+        line->value = tor_malloc(ISO_TIME_LEN+1);
+        format_iso_time(line->value, h->unlisted_since);
+        next = &(line->next);
+      }
+    });
+  state->dirty = 1;
+  helper_nodes_dirty = 0;
+
+  return 1;
+}

Index: config.c
===================================================================
RCS file: /home/or/cvsroot/tor/src/or/config.c,v
retrieving revision 1.375
retrieving revision 1.376
diff -u -d -r1.375 -r1.376
--- config.c	25 Jul 2005 10:29:21 -0000	1.375
+++ config.c	28 Jul 2005 19:01:48 -0000	1.376
@@ -24,6 +24,7 @@
   CONFIG_TYPE_MEMUNIT,      /**< A number of bytes, with optional units*/
   CONFIG_TYPE_DOUBLE,       /**< A floating-point value */
   CONFIG_TYPE_BOOL,         /**< A boolean value, expressed as 0 or 1. */
+  CONFIG_TYPE_ISOTIME,      /**< An ISO-formated time relative to GMT. */
   CONFIG_TYPE_CSV,          /**< A list of strings, separated by commas and optional
                               * whitespace. */
   CONFIG_TYPE_LINELIST,     /**< Uninterpreted config lines */
@@ -47,7 +48,7 @@
 #define PLURAL(tok) { #tok, #tok "s", 0 }
 
 /* A list of command-line abbreviations. */
-static config_abbrev_t _config_abbrevs[] = {
+static config_abbrev_t _option_abbrevs[] = {
   PLURAL(ExitNode),
   PLURAL(EntryNode),
   PLURAL(ExcludeNode),
@@ -75,6 +76,7 @@
   config_type_t type; /**< How to interpret the type and turn it into a value. */
   off_t var_offset; /**< Offset of the corresponding member of or_options_t. */
   const char *initvalue; /**< String (or null) describing initial value. */
+  const char *description;
 } config_var_t;
 
 /** Return the offset of <b>member</b> within the type <b>tp</b>, in bytes */
@@ -83,7 +85,7 @@
  * CONFIG_TYPE_<b>conftype</b>, and corresponds to
  * or_options_t.<b>member</b>"
  */
-#define VAR(name,conftype,member,initvalue) \
+#define VAR(name,conftype,member,initvalue)                            \
   { name, CONFIG_TYPE_ ## conftype, STRUCT_OFFSET(or_options_t, member), initvalue }
 /** An entry for config_vars: "The option <b>name</b> is obsolete." */
 #define OBSOLETE(name) { name, CONFIG_TYPE_OBSOLETE, 0, NULL }
@@ -92,7 +94,7 @@
  * abbreviations, order is significant, since the first matching option will
  * be chosen first.
  */
-static config_var_t _config_vars[] = {
+static config_var_t _option_vars[] = {
   VAR("Address",             STRING,   Address,              NULL),
   VAR("AccountingStart",     STRING,   AccountingStart,      NULL),
   VAR("AllowUnverifiedNodes",CSV,      AllowUnverifiedNodes, "middle,rendezvous"),
@@ -186,8 +188,50 @@
   { NULL, CONFIG_TYPE_OBSOLETE, 0, NULL }
 };
 #undef VAR
+
+#define VAR(name,conftype,member,initvalue) \
+  { name, CONFIG_TYPE_ ## conftype, STRUCT_OFFSET(or_state_t, member), initvalue }
+static config_var_t _state_vars[] = {
+  VAR("LastWritten",             ISOTIME,     LastWritten,          NULL),
+
+  VAR("AccountingIntervalStart", ISOTIME,  AccountingIntervalStart, NULL),
+  VAR("AccountingBytesWrittenInInterval", MEMUNIT,
+      AccountingBytesWrittenInInterval, NULL),
+  VAR("AccountingBytesReadInterval", MEMUNIT, AccountingBytesReadInInterval,NULL),
+  VAR("AccountingSecondsActive", INTERVAL,    AccountingSecondsActive, NULL),
+  VAR("AccountingExpectedUsage", MEMUNIT,     AccountingExpectedUsage, NULL),
+
+  VAR("HelperNodes",             LINELIST_V,  HelperNodes,          NULL),
+  VAR("HelperNode",              LINELIST_S,  HelperNodes,          NULL),
+  VAR("HelperNodeDownSince",     LINELIST_S,  HelperNodes,          NULL),
+  VAR("HelperNodeUnlistedSince", LINELIST_S,  HelperNodes,          NULL),
+  { NULL, CONFIG_TYPE_OBSOLETE, 0, NULL }
+};
+
+#undef VAR
 #undef OBSOLETE
 
+/** DOCDOC*/
+typedef struct config_var_description_t {
+  const char *name;
+  const char *description;
+} config_var_description_t;
+
+static config_var_description_t options_description[] = {
+  { "Address", "The advertised (external) address we should use" },
+  // { "AccountingStart", ""},
+  { NULL, NULL },
+};
+
+static config_var_description_t state_description[] = {
+  { "HelperNode", "One of the nodes we have chosen as a fixed entry" },
+  { "HelperNodeDownSince",
+    "The last helper node has been down since this time." },
+  { "HelperNodeUnlistedSince",
+    "The last helper node has been unlisted since this time." },
+  { NULL, NULL },
+};
+
 typedef int (*validate_fn_t)(void*);
 
 typedef struct {
@@ -197,6 +241,7 @@
   config_abbrev_t *abbrevs;
   config_var_t *vars;
   validate_fn_t validate_fn;
+  config_var_description_t *descriptions;
 } config_format_t;
 
 #define CHECK(fmt, cfg) do {                                            \
@@ -211,7 +256,7 @@
                                const char *key, const char *val);
 static void option_reset(config_format_t *fmt, or_options_t *options,
                          config_var_t *var);
-static void options_free(config_format_t *fmt, or_options_t *options);
+static void config_free(config_format_t *fmt, void *options);
 static int option_is_same(config_format_t *fmt,
                           or_options_t *o1, or_options_t *o2,const char *name);
 static or_options_t *options_dup(config_format_t *fmt, or_options_t *old);
@@ -236,7 +281,8 @@
 static int write_configuration_file(const char *fname, or_options_t *options);
 static config_line_t *get_assigned_option(config_format_t *fmt,
                                      or_options_t *options, const char *key);
-static void config_init(config_format_t *fmt, or_options_t *options);
+static void config_init(config_format_t *fmt, void *options);
+static int or_state_validate(or_state_t *options);
 
 static uint64_t config_parse_memunit(const char *s, int *ok);
 static int config_parse_interval(const char *s, int *ok);
@@ -252,9 +298,22 @@
   sizeof(or_options_t),
   OR_OPTIONS_MAGIC,
   STRUCT_OFFSET(or_options_t, _magic),
-  _config_abbrevs,
-  _config_vars,
-  (validate_fn_t)options_validate
+  _option_abbrevs,
+  _option_vars,
+  (validate_fn_t)options_validate,
+  options_description
+};
+
+#define OR_STATE_MAGIC 0x57A73f57
+
+static config_format_t state_format = {
+  sizeof(or_state_t),
+  OR_STATE_MAGIC,
+  STRUCT_OFFSET(or_state_t, _magic),
+  NULL,
+  _state_vars,
+  (validate_fn_t)or_state_validate,
+  state_description
 };
 
 /*
@@ -262,9 +321,11 @@
  */
 
 /** Command-line and config-file options. */
-static or_options_t *global_options=NULL;
+static or_options_t *global_options = NULL;
 /** Name of most recently read torrc file. */
 static char *config_fname = NULL;
+/** Persistant serialized state. */
+static or_state_t *global_state = NULL;
 
 static void *
 config_alloc(config_format_t *fmt)
@@ -290,14 +351,14 @@
 set_options(or_options_t *new_val)
 {
   if (global_options)
-    options_free(&config_format, global_options);
+    config_free(&config_format, global_options);
   global_options = new_val;
 }
 
 void
 config_free_all(void)
 {
-  options_free(&config_format, global_options);
+  config_free(&config_format, global_options);
   tor_free(config_fname);
 }
 
@@ -388,6 +449,11 @@
     libevent_initialized = 1;
   }
 
+  /* Load state */
+  if (! global_state)
+    if (or_state_load())
+      return -1;
+
   options->_ConnLimit =
     set_max_file_descriptors((unsigned)options->ConnLimit, MAXCONNECTIONS);
   if (options->_ConnLimit < 0)
@@ -695,6 +761,13 @@
     *(double *)lvalue = atof(c->value);
     break;
 
+  case CONFIG_TYPE_ISOTIME:
+    if (parse_iso_time(c->value, (time_t *)lvalue)) {
+      log(LOG_WARN, "Invalid time '%s' for keyword '%s'", c->value, c->key);
+      return -2;
+    }
+    break;
+
   case CONFIG_TYPE_CSV:
     if (*(smartlist_t**)lvalue) {
       SMARTLIST_FOREACH(*(smartlist_t**)lvalue, char *, cp, tor_free(cp));
@@ -814,6 +887,15 @@
         return NULL;
       }
       break;
+    case CONFIG_TYPE_ISOTIME:
+      if (*(time_t*)value) {
+        result->value = tor_malloc(ISO_TIME_LEN+1);
+        format_iso_time(result->value, *(time_t*)value);
+      } else {
+        tor_free(result->key);
+        tor_free(result);
+      }
+      break;
     case CONFIG_TYPE_INTERVAL:
     case CONFIG_TYPE_UINT:
       /* This means every or_options_t uint or bool element
@@ -866,7 +948,7 @@
  */
 static int
 config_assign(config_format_t *fmt,
-              or_options_t *options, config_line_t *list, int reset)
+              void *options, config_line_t *list, int reset)
 {
   config_line_t *p;
 
@@ -911,17 +993,17 @@
   or_options_t *trial_options = options_dup(&config_format, get_options());
 
   if ((r=config_assign(&config_format, trial_options, list, reset)) < 0) {
-    options_free(&config_format, trial_options);
+    config_free(&config_format, trial_options);
     return r;
   }
 
   if (options_validate(trial_options) < 0) {
-    options_free(&config_format, trial_options);
+    config_free(&config_format, trial_options);
     return -2;
   }
 
   if (options_transition_allowed(get_options(), trial_options) < 0) {
-    options_free(&config_format, trial_options);
+    config_free(&config_format, trial_options);
     return -3;
   }
 
@@ -947,6 +1029,8 @@
     case CONFIG_TYPE_DOUBLE:
       *(double*)lvalue = 0.0;
       break;
+    case CONFIG_TYPE_ISOTIME:
+      *(time_t*)lvalue = 0;
     case CONFIG_TYPE_INTERVAL:
     case CONFIG_TYPE_UINT:
     case CONFIG_TYPE_BOOL:
@@ -1132,7 +1216,7 @@
 
 /** Release storage held by <b>options</b> */
 static void
-options_free(config_format_t *fmt,or_options_t *options)
+config_free(config_format_t *fmt, void *options)
 {
   int i;
   void *lvalue;
@@ -1147,6 +1231,7 @@
       case CONFIG_TYPE_UINT:
       case CONFIG_TYPE_BOOL:
       case CONFIG_TYPE_DOUBLE:
+      case CONFIG_TYPE_ISOTIME:
       case CONFIG_TYPE_OBSOLETE:
         break; /* nothing to free for these config types */
       case CONFIG_TYPE_STRING:
@@ -1238,8 +1323,9 @@
   config_init(&config_format, options);
 }
 
+/* DOCDOC */
 static void
-config_init(config_format_t *fmt, or_options_t *options)
+config_init(config_format_t *fmt, void *options)
 {
   int i;
   config_var_t *var;
@@ -1253,14 +1339,15 @@
   }
 }
 
+/* DOCDOC */
 static char *
-config_dump(config_format_t *fmt, or_options_t *options, int minimal)
+config_dump(config_format_t *fmt, void *options, int minimal)
 {
   smartlist_t *elements;
   or_options_t *defaults;
   config_line_t *line;
   char *result;
-  int i;
+  int i, j;
 
   defaults = config_alloc(fmt);
   config_init(fmt, defaults);
@@ -1276,6 +1363,18 @@
       continue;
     if (minimal && option_is_same(fmt, options, defaults, fmt->vars[i].name))
       continue;
+
+    for (j=0; fmt->descriptions[j].name; ++j) {
+      if (!strcasecmp(fmt->vars[i].name, fmt->descriptions[j].name)) {
+        const char *desc = fmt->descriptions[j].description;
+        size_t len = strlen(desc)+8;
+        char *tmp = tor_malloc(len);
+        tor_snprintf(tmp, len, "# %s\n",desc);
+        smartlist_add(elements, tmp);
+        break;
+      }
+    }
+
     line = get_assigned_option(fmt, options, fmt->vars[i].name);
     for (; line; line = line->next) {
       size_t len = strlen(line->key) + strlen(line->value) + 3;
@@ -1735,32 +1834,32 @@
   }
 
   if (old->RunAsDaemon != new_val->RunAsDaemon) {
-    log_fn(LOG_WARN,"During reload, changing RunAsDaemon is not allowed. Failing.");
+    log_fn(LOG_WARN,"While Tor is running, changing RunAsDaemon is not allowed. Failing.");
     return -1;
   }
 
   if (old->ORPort != new_val->ORPort) {
-    log_fn(LOG_WARN,"During reload, changing ORPort is not allowed. Failing.");
+    log_fn(LOG_WARN,"While Tor is running, changing ORPort is not allowed. Failing.");
     return -1;
   }
 
   if (strcmp(old->DataDirectory,new_val->DataDirectory)!=0) {
-    log_fn(LOG_WARN,"During reload, changing DataDirectory (%s->%s) is not allowed. Failing.", old->DataDirectory, new_val->DataDirectory);
+    log_fn(LOG_WARN,"While Tor is running, changing DataDirectory (%s->%s) is not allowed. Failing.", old->DataDirectory, new_val->DataDirectory);
     return -1;
   }
 
   if (!opt_streq(old->User, new_val->User)) {
-    log_fn(LOG_WARN,"During reload, changing User is not allowed. Failing.");
+    log_fn(LOG_WARN,"While Tor is running, changing User is not allowed. Failing.");
     return -1;
   }
 
   if (!opt_streq(old->Group, new_val->Group)) {
-    log_fn(LOG_WARN,"During reload, changing Group is not allowed. Failing.");
+    log_fn(LOG_WARN,"While Tor is running, changing Group is not allowed. Failing.");
     return -1;
   }
 
   if (old->HardwareAccel != new_val->HardwareAccel) {
-    log_fn(LOG_WARN,"During reload, changing HardwareAccel is not allowed. Failing.");
+    log_fn(LOG_WARN,"While Tor is running, changing HardwareAccel is not allowed. Failing.");
     return -1;
   }
 
@@ -1984,7 +2083,7 @@
   return 0;
  err:
   tor_free(fname);
-  options_free(&config_format, newoptions);
+  config_free(&config_format, newoptions);
   return -1;
 }
 
@@ -2795,6 +2894,162 @@
 }
 #endif
 
+/* Versioning issues and state: we want to be able to understand old state
+ * files, and not choke on new ones.
+ *
+ * We could preserve all unrecognized variables across invocations, but we could
+ * screw up order, if their order is significant with respect to existing
+ * options.
+ *
+ * We could just dump unrecognized variables if you downgrade.
+ *
+ * This needs thought. XXXX NM
+ */
+
+/** DOCDOC */
+or_state_t *
+get_or_state(void)
+{
+  return global_state;
+}
+
+/** DOCDOC */
+static char *
+get_or_state_fname(void)
+{
+  char *fname = NULL;
+  or_options_t *options = get_options();
+  size_t len = strlen(options->DataDirectory) + 16;
+  fname = tor_malloc(len);
+  tor_snprintf(fname, len, "%s/state", options->DataDirectory);
+  return fname;
+}
+
+/** DOCDOC */
+static int
+or_state_validate(or_state_t *state)
+{
+  const char *err;
+  if (helper_nodes_parse_state(state, 0, &err)<0) {
+    log_fn(LOG_WARN, "Unable to parse helper nodes: %s", err);
+    return -1;
+  }
+  return 0;
+}
+
+/** DOCDOC */
+static void
+or_state_set(or_state_t *new_state)
+{
+  const char *err;
+  tor_assert(new_state);
+  if (global_state)
+    config_free(&state_format, global_state);
+  global_state = new_state;
+  if (helper_nodes_parse_state(global_state, 1, &err)<0)
+    log_fn(LOG_WARN,"Unparseable helper nodes state: %s",err);
+
+}
+
+/* DOCDOC */
+int
+or_state_load(void)
+{
+  or_state_t *new_state = NULL;
+  char *contents = NULL, *fname;
+  int r = -1;
+
+  fname = get_or_state_fname();
+  switch (file_status(fname)) {
+    case FN_FILE:
+      if (!(contents = read_file_to_str(fname, 0))) {
+        log_fn(LOG_WARN, "Unable to read state file %s", fname);
+        goto done;
+      }
+      break;
+    case FN_NOENT:
+      break;
+    default:
+      log_fn(LOG_WARN,"State file %s is not a file? Failing.", fname);
+      goto done;
+  }
+  new_state = tor_malloc_zero(sizeof(or_state_t));
+  new_state->_magic = OR_STATE_MAGIC;
+  config_init(&state_format, new_state);
+  if (contents) {
+    config_line_t *lines=NULL;
+    int assign_retval;
+    if (config_get_lines(contents, &lines)<0)
+      goto done;
+    assign_retval = config_assign(&state_format, new_state, lines, 0);
+    config_free_lines(lines);
+    if (assign_retval<0)
+      goto done;
+  }
+
+  if (or_state_validate(new_state) < 0)
+    goto done;
+
+  if (contents)
+    log_fn(LOG_INFO, "Loaded state from %s", fname);
+  else
+    log_fn(LOG_INFO, "Initialized state");
+  or_state_set(new_state);
+  new_state = NULL;
+  if (!contents) {
+    global_state->dirty = 1;
+    or_state_save();
+  }
+
+  r = 0;
+ done:
+  tor_free(fname);
+  tor_free(contents);
+  if (new_state)
+    config_free(&state_format, new_state);
+
+  return r;
+}
+
+/** DOCDOC */
+int
+or_state_save(void)
+{
+  char *state, *contents;
+  char tbuf[ISO_TIME_LEN+1];
+  size_t len;
+  char *fname;
+
+  helper_nodes_update_state(global_state);
+
+  if (!global_state->dirty)
+    return 0;
+
+  global_state->LastWritten = time(NULL);
+  state = config_dump(&state_format, global_state, 0);
+  len = strlen(state)+128;
+  contents = tor_malloc(len);
+  format_local_iso_time(tbuf, time(NULL));
+  tor_snprintf(contents, len,
+               "# Tor state file last generated on %s\n"
+               "# You *do not* need to edit this file.\n\n%s",
+               tbuf, state);
+  tor_free(state);
+  fname = get_or_state_fname();
+  if (write_str_to_file(fname, contents, 0)<0) {
+    log_fn(LOG_WARN, "Unable to write state to file %s", fname);
+    tor_free(fname);
+    tor_free(contents);
+    return -1;
+  }
+  log_fn(LOG_INFO, "Saved state to %s", fname);
+  tor_free(fname);
+  tor_free(contents);
+
+  global_state->dirty = 0;
+  return 0;
+}
+
 /** Dump the version of every file to the log. */
 static void
 print_cvs_version(void)

Index: hibernate.c
===================================================================
RCS file: /home/or/cvsroot/tor/src/or/hibernate.c,v
retrieving revision 1.57
retrieving revision 1.58
diff -u -d -r1.57 -r1.58
--- hibernate.c	23 Jul 2005 01:52:24 -0000	1.57
+++ hibernate.c	28 Jul 2005 19:01:48 -0000	1.58
@@ -539,6 +539,9 @@
   tor_snprintf(fname, sizeof(fname), "%s/bw_accounting",
                get_options()->DataDirectory);
 
+
+ 
+
   return write_str_to_file(fname, buf, 0);
 }
 

Index: main.c
===================================================================
RCS file: /home/or/cvsroot/tor/src/or/main.c,v
retrieving revision 1.532
retrieving revision 1.533
diff -u -d -r1.532 -r1.533
--- main.c	25 Jul 2005 10:36:01 -0000	1.532
+++ main.c	28 Jul 2005 19:01:48 -0000	1.533
@@ -1320,6 +1320,7 @@
     log_fn(LOG_ERR, "Unable to seed random number generator. Exiting.");
     return -1;
   }
+
   return 0;
 }
 

Index: or.h
===================================================================
RCS file: /home/or/cvsroot/tor/src/or/or.h,v
retrieving revision 1.632
retrieving revision 1.633
diff -u -d -r1.632 -r1.633
--- or.h	25 Jul 2005 10:29:21 -0000	1.632
+++ or.h	28 Jul 2005 19:01:48 -0000	1.633
@@ -1176,6 +1176,22 @@
   int RephistTrackTime; /**< How many seconds do we keep rephist info? */
 } or_options_t;
 
+/** Persistent state for an onion router, as saved to disk. */
+typedef struct {
+  uint32_t _magic;
+  int dirty;
+
+  /* XXXX These options aren't actually attached to anything yet. */
+  time_t LastWritten;
+  time_t AccountingIntervalStart;
+  uint64_t AccountingBytesReadInInterval;
+  uint64_t AccountingBytesWrittenInInterval;
+  int AccountingSecondsActive;
+  uint64_t AccountingExpectedUsage;
+
+  config_line_t *HelperNodes;
+} or_state_t;
+
 #define MAX_SOCKS_REPLY_LEN 1024
 #define MAX_SOCKS_ADDR_LEN 256
 #define SOCKS_COMMAND_CONNECT 0x01
@@ -1261,6 +1277,8 @@
 
 void helper_node_set_status(const char *digest, int succeeded);
 void helper_nodes_set_status_from_directory(void);
+int helper_nodes_update_state(or_state_t *state);
+int helper_nodes_parse_state(or_state_t *state, int set, const char **err);
 
 /********************************* circuitlist.c ***********************/
 
@@ -1350,6 +1368,10 @@
 char *options_dump(or_options_t *options, int minimal);
 int options_save_current(void);
 
+or_state_t *get_or_state(void);
+int or_state_load(void);
+int or_state_save(void);
+
 /********************************* connection.c ***************************/
 
 const char *conn_type_to_string(int type);