|  | @@ -40,19 +40,28 @@ HT_GENERATE(microdesc_map, microdesc_t, node,
 | 
	
		
			
				|  |  |               _microdesc_hash, _microdesc_eq, 0.6,
 | 
	
		
			
				|  |  |               _tor_malloc, _tor_realloc, _tor_free);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +/* returns n bytes written */
 | 
	
		
			
				|  |  |  static int
 | 
	
		
			
				|  |  | -dump_microdescriptor(FILE *f, microdesc_t *md)
 | 
	
		
			
				|  |  | +dump_microdescriptor(FILE *f, microdesc_t *md, int *annotation_len_out)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +  int r = 0;
 | 
	
		
			
				|  |  |    /* XXXX drops unkown annotations. */
 | 
	
		
			
				|  |  |    if (md->last_listed) {
 | 
	
		
			
				|  |  |      char buf[ISO_TIME_LEN+1];
 | 
	
		
			
				|  |  | +    char annotation[ISO_TIME_LEN+32];
 | 
	
		
			
				|  |  |      format_iso_time(buf, md->last_listed);
 | 
	
		
			
				|  |  | -    fprintf(f, "@last-listed %s\n", buf);
 | 
	
		
			
				|  |  | +    tor_snprintf(annotation, sizeof(annotation), "@last-listed %s\n", buf);
 | 
	
		
			
				|  |  | +    fputs(annotation, f);
 | 
	
		
			
				|  |  | +    r += strlen(annotation);
 | 
	
		
			
				|  |  | +    *annotation_len_out = r;
 | 
	
		
			
				|  |  | +  } else {
 | 
	
		
			
				|  |  | +    *annotation_len_out = 0;
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    md->off = (off_t) ftell(f);
 | 
	
		
			
				|  |  |    fwrite(md->body, 1, md->bodylen, f);
 | 
	
		
			
				|  |  | -  return 0;
 | 
	
		
			
				|  |  | +  r += md->bodylen;
 | 
	
		
			
				|  |  | +  return r;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  static microdesc_cache_t *the_microdesc_cache = NULL;
 | 
	
	
		
			
				|  | @@ -72,7 +81,7 @@ get_microdesc_cache(void)
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  /* There are three sources of microdescriptors:
 | 
	
		
			
				|  |  | -   1) Generated us while acting as a directory authority.
 | 
	
		
			
				|  |  | +   1) Generated by us while acting as a directory authority.
 | 
	
		
			
				|  |  |     2) Loaded from the cache on disk.
 | 
	
		
			
				|  |  |     3) Downloaded.
 | 
	
		
			
				|  |  |  */
 | 
	
	
		
			
				|  | @@ -98,7 +107,8 @@ microdescs_add_to_cache(microdesc_cache_t *cache,
 | 
	
		
			
				|  |  |    return added;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -/* Returns list of added microdesc_t. Frees any not added. */
 | 
	
		
			
				|  |  | +/* Returns list of added microdesc_t. Frees any not added. Updates last_listed.
 | 
	
		
			
				|  |  | + */
 | 
	
		
			
				|  |  |  smartlist_t *
 | 
	
		
			
				|  |  |  microdescs_add_list_to_cache(microdesc_cache_t *cache,
 | 
	
		
			
				|  |  |                               smartlist_t *descriptors, saved_location_t where,
 | 
	
	
		
			
				|  | @@ -108,13 +118,17 @@ microdescs_add_list_to_cache(microdesc_cache_t *cache,
 | 
	
		
			
				|  |  |    open_file_t *open_file = NULL;
 | 
	
		
			
				|  |  |    FILE *f = NULL;
 | 
	
		
			
				|  |  |    //  int n_added = 0;
 | 
	
		
			
				|  |  | +  size_t size = 0;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    if (where == SAVED_NOWHERE && !no_save) {
 | 
	
		
			
				|  |  | -    f = start_writing_to_stdio_file(cache->journal_fname, OPEN_FLAGS_APPEND,
 | 
	
		
			
				|  |  | +    f = start_writing_to_stdio_file(cache->journal_fname,
 | 
	
		
			
				|  |  | +                                    OPEN_FLAGS_APPEND|O_BINARY,
 | 
	
		
			
				|  |  |                                      0600, &open_file);
 | 
	
		
			
				|  |  | -    if (!f)
 | 
	
		
			
				|  |  | -      log_warn(LD_DIR, "Couldn't append to journal in %s",
 | 
	
		
			
				|  |  | -               cache->journal_fname);
 | 
	
		
			
				|  |  | +    if (!f) {
 | 
	
		
			
				|  |  | +      log_warn(LD_DIR, "Couldn't append to journal in %s: %s",
 | 
	
		
			
				|  |  | +               cache->journal_fname, strerror(errno));
 | 
	
		
			
				|  |  | +      return NULL;
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    added = smartlist_create();
 | 
	
	
		
			
				|  | @@ -131,8 +145,10 @@ microdescs_add_list_to_cache(microdesc_cache_t *cache,
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      /* Okay, it's a new one. */
 | 
	
		
			
				|  |  |      if (f) {
 | 
	
		
			
				|  |  | -      dump_microdescriptor(f, md);
 | 
	
		
			
				|  |  | +      int annotation_len;
 | 
	
		
			
				|  |  | +      size = dump_microdescriptor(f, md, &annotation_len);
 | 
	
		
			
				|  |  |        md->saved_location = SAVED_IN_JOURNAL;
 | 
	
		
			
				|  |  | +      cache->journal_len += size;
 | 
	
		
			
				|  |  |      } else {
 | 
	
		
			
				|  |  |        md->saved_location = where;
 | 
	
		
			
				|  |  |      }
 | 
	
	
		
			
				|  | @@ -143,7 +159,18 @@ microdescs_add_list_to_cache(microdesc_cache_t *cache,
 | 
	
		
			
				|  |  |      smartlist_add(added, md);
 | 
	
		
			
				|  |  |    } SMARTLIST_FOREACH_END(md);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -  finish_writing_to_file(open_file); /*XXX Check me.*/
 | 
	
		
			
				|  |  | +  if (f)
 | 
	
		
			
				|  |  | +    finish_writing_to_file(open_file); /*XXX Check me.*/
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  {
 | 
	
		
			
				|  |  | +    size_t old_content_len =
 | 
	
		
			
				|  |  | +      cache->cache_content ? cache->cache_content->size : 0;
 | 
	
		
			
				|  |  | +    if (cache->journal_len > 16384 + old_content_len &&
 | 
	
		
			
				|  |  | +        cache->journal_len > old_content_len * 2) {
 | 
	
		
			
				|  |  | +      microdesc_cache_rebuild(cache);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |    return added;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -152,9 +179,11 @@ microdesc_cache_clear(microdesc_cache_t *cache)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  |    microdesc_t **entry, **next;
 | 
	
		
			
				|  |  |    for (entry = HT_START(microdesc_map, &cache->map); entry; entry = next) {
 | 
	
		
			
				|  |  | +    microdesc_t *md = *entry;
 | 
	
		
			
				|  |  |      next = HT_NEXT_RMV(microdesc_map, &cache->map, entry);
 | 
	
		
			
				|  |  | -    microdesc_free(*entry);
 | 
	
		
			
				|  |  | +    microdesc_free(md);
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  | +  HT_CLEAR(microdesc_map, &cache->map);
 | 
	
		
			
				|  |  |    if (cache->cache_content) {
 | 
	
		
			
				|  |  |      tor_munmap_file(cache->cache_content);
 | 
	
		
			
				|  |  |      cache->cache_content = NULL;
 | 
	
	
		
			
				|  | @@ -176,8 +205,10 @@ microdesc_cache_reload(microdesc_cache_t *cache)
 | 
	
		
			
				|  |  |    if (mm) {
 | 
	
		
			
				|  |  |      added = microdescs_add_to_cache(cache, mm->data, mm->data+mm->size,
 | 
	
		
			
				|  |  |                                      SAVED_IN_CACHE, 0);
 | 
	
		
			
				|  |  | -    total += smartlist_len(added);
 | 
	
		
			
				|  |  | -    smartlist_free(added);
 | 
	
		
			
				|  |  | +    if (added) {
 | 
	
		
			
				|  |  | +      total += smartlist_len(added);
 | 
	
		
			
				|  |  | +      smartlist_free(added);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    journal_content = read_file_to_str(cache->journal_fname,
 | 
	
	
		
			
				|  | @@ -186,8 +217,10 @@ microdesc_cache_reload(microdesc_cache_t *cache)
 | 
	
		
			
				|  |  |      added = microdescs_add_to_cache(cache, journal_content,
 | 
	
		
			
				|  |  |                                      journal_content+st.st_size,
 | 
	
		
			
				|  |  |                                      SAVED_IN_JOURNAL, 0);
 | 
	
		
			
				|  |  | -    total += smartlist_len(added);
 | 
	
		
			
				|  |  | -    smartlist_free(added);
 | 
	
		
			
				|  |  | +    if (added) {
 | 
	
		
			
				|  |  | +      total += smartlist_len(added);
 | 
	
		
			
				|  |  | +      smartlist_free(added);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  |      tor_free(journal_content);
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |    log_notice(LD_DIR, "Reloaded microdescriptor cache.  Found %d descriptors.",
 | 
	
	
		
			
				|  | @@ -202,8 +235,16 @@ microdesc_cache_rebuild(microdesc_cache_t *cache)
 | 
	
		
			
				|  |  |    FILE *f;
 | 
	
		
			
				|  |  |    microdesc_t **mdp;
 | 
	
		
			
				|  |  |    smartlist_t *wrote;
 | 
	
		
			
				|  |  | +  int size;
 | 
	
		
			
				|  |  | +  off_t off = 0;
 | 
	
		
			
				|  |  | +  int orig_size, new_size;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -  f = start_writing_to_stdio_file(cache->cache_fname, OPEN_FLAGS_REPLACE,
 | 
	
		
			
				|  |  | +  log_info(LD_DIR, "Rebuilding the microdescriptor cache...");
 | 
	
		
			
				|  |  | +  orig_size = (int)(cache->cache_content ? cache->cache_content->size : 0);
 | 
	
		
			
				|  |  | +  orig_size += (int)cache->journal_len;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  f = start_writing_to_stdio_file(cache->cache_fname,
 | 
	
		
			
				|  |  | +                                  OPEN_FLAGS_REPLACE|O_BINARY,
 | 
	
		
			
				|  |  |                                    0600, &open_file);
 | 
	
		
			
				|  |  |    if (!f)
 | 
	
		
			
				|  |  |      return -1;
 | 
	
	
		
			
				|  | @@ -212,15 +253,17 @@ microdesc_cache_rebuild(microdesc_cache_t *cache)
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    HT_FOREACH(mdp, microdesc_map, &cache->map) {
 | 
	
		
			
				|  |  |      microdesc_t *md = *mdp;
 | 
	
		
			
				|  |  | +    int annotation_len;
 | 
	
		
			
				|  |  |      if (md->no_save)
 | 
	
		
			
				|  |  |        continue;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    dump_microdescriptor(f, md);
 | 
	
		
			
				|  |  | +    size = dump_microdescriptor(f, md, &annotation_len);
 | 
	
		
			
				|  |  | +    md->off = off + annotation_len;
 | 
	
		
			
				|  |  | +    off += size;
 | 
	
		
			
				|  |  |      if (md->saved_location != SAVED_IN_CACHE) {
 | 
	
		
			
				|  |  |        tor_free(md->body);
 | 
	
		
			
				|  |  |        md->saved_location = SAVED_IN_CACHE;
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |      smartlist_add(wrote, md);
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -229,21 +272,29 @@ microdesc_cache_rebuild(microdesc_cache_t *cache)
 | 
	
		
			
				|  |  |    if (cache->cache_content)
 | 
	
		
			
				|  |  |      tor_munmap_file(cache->cache_content);
 | 
	
		
			
				|  |  |    cache->cache_content = tor_mmap_file(cache->cache_fname);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |    if (!cache->cache_content && smartlist_len(wrote)) {
 | 
	
		
			
				|  |  |      log_err(LD_DIR, "Couldn't map file that we just wrote to %s!",
 | 
	
		
			
				|  |  |              cache->cache_fname);
 | 
	
		
			
				|  |  | +    smartlist_free(wrote);
 | 
	
		
			
				|  |  |      return -1;
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |    SMARTLIST_FOREACH_BEGIN(wrote, microdesc_t *, md) {
 | 
	
		
			
				|  |  | -    if (md->no_save)
 | 
	
		
			
				|  |  | -      continue;
 | 
	
		
			
				|  |  |      tor_assert(md->saved_location == SAVED_IN_CACHE);
 | 
	
		
			
				|  |  |      md->body = (char*)cache->cache_content->data + md->off;
 | 
	
		
			
				|  |  |      tor_assert(!memcmp(md->body, "onion-key", 9));
 | 
	
		
			
				|  |  | -  } SMARTLIST_FOREACH_END(wrote);
 | 
	
		
			
				|  |  | +  } SMARTLIST_FOREACH_END(md);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    smartlist_free(wrote);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +  write_str_to_file(cache->journal_fname, "", 1);
 | 
	
		
			
				|  |  | +  cache->journal_len = 0;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  new_size = (int)cache->cache_content->size;
 | 
	
		
			
				|  |  | +  log_info(LD_DIR, "Done rebuilding microdesc cache. "
 | 
	
		
			
				|  |  | +           "Saved %d bytes; %d still used.",
 | 
	
		
			
				|  |  | +           orig_size-new_size, new_size);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |    return 0;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -265,3 +316,13 @@ microdesc_free(microdesc_t *md)
 | 
	
		
			
				|  |  |    tor_free(md);
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +void
 | 
	
		
			
				|  |  | +microdesc_free_all(void)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +  if (the_microdesc_cache) {
 | 
	
		
			
				|  |  | +    microdesc_cache_clear(the_microdesc_cache);
 | 
	
		
			
				|  |  | +    tor_free(the_microdesc_cache->cache_fname);
 | 
	
		
			
				|  |  | +    tor_free(the_microdesc_cache->journal_fname);
 | 
	
		
			
				|  |  | +    tor_free(the_microdesc_cache);
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +}
 |