diff options
-rw-r--r-- | src/or/config.c | 56 | ||||
-rw-r--r-- | src/or/or.h | 13 |
2 files changed, 69 insertions, 0 deletions
diff --git a/src/or/config.c b/src/or/config.c index 7d36ddc059..91337515d5 100644 --- a/src/or/config.c +++ b/src/or/config.c @@ -277,6 +277,7 @@ config_assign(or_options_t *options, struct config_line_t *list) config_compare(list, "PidFile", CONFIG_TYPE_STRING, &options->PidFile) || config_compare(list, "PathlenCoinWeight",CONFIG_TYPE_DOUBLE, &options->PathlenCoinWeight) || + config_compare(list, "RedirectExit", CONFIG_TYPE_LINELIST, &options->RedirectExit) || config_compare(list, "RouterFile", CONFIG_TYPE_OBSOLETE, NULL) || config_compare(list, "RunAsDaemon", CONFIG_TYPE_BOOL, &options->RunAsDaemon) || config_compare(list, "RunTesting", CONFIG_TYPE_BOOL, &options->RunTesting) || @@ -478,6 +479,12 @@ free_options(or_options_t *options) config_free_lines(options->DirServers); config_free_lines(options->RecommendedVersions); config_free_lines(options->NodeFamilies); + config_free_lines(options->RedirectExit); + if (options->RedirectExitList) { + SMARTLIST_FOREACH(options->RedirectExit,exit_redirect_t *, p, tor_free(p)); + smartlist_free(options->RedirectExit); + options->RedirectExit = NULL; + } if (options->FirewallPorts) { SMARTLIST_FOREACH(options->FirewallPorts, char *, cp, tor_free(cp)); smartlist_free(options->FirewallPorts); @@ -868,6 +875,13 @@ getconfig(int argc, char **argv, or_options_t *options) return -1; } + if (!options->RedirectExitList) + options->RedirectExitList = smartlist_create(); + for (cl = options->RedirectExit; cl; cl = cl->next) { + if (parse_redirect_line(options, cl)<0) + return -1; + } + clear_trusted_dir_servers(); if (!options->DirServers) { add_default_trusted_dirservers(); @@ -1051,6 +1065,48 @@ void exit_policy_free(struct exit_policy_t *p) { } } +static int parse_redirect_line(or_options_t *options, + struct config_line_t *line) +{ + smartlist_t *elements = NULL; + exit_redirect_t *r; + + tor_assert(options); + tor_assert(options->RedirectExitList); + tor_assert(line); + + r = tor_malloc_zero(sizeof(exit_redirect_t)); + smartlist_split_string(elements, line->value, " ", + SPLIT_SKIP_SPACE|SPLIT_IGNORE_BLANK, 0); + if (smartlist_len(elements) != 2) { + log_fn(LOG_WARN, "Wrong number of elements in RedirectExit line"); + goto err; + } + if (parse_addr_and_port_range(smartlist_get(elements,0),&r->addr,&r->mask, + &r->port_min,&r->port_max)) { + log_fn(LOG_WARN, "Error parsing source address in RedirectExit line"); + goto err; + } + if (parse_addr_port(smartlist_get(elements,1),NULL,&r->addr_dest, + &r->port_dest)) { + log_fn(LOG_WARN, "Error parseing dest address in RedirectExit line"); + goto err; + } + + goto done; + err: + tor_free(r); + done: + SMARTLIST_FOREACH(elements, char *, cp, tor_free(cp)); + smartlist_free(elements); + if (r) { + smartlist_add(options->RedirectExitList, r); + return 0; + } else { + return -1; + } +} + static int parse_dir_server_line(const char *line) { smartlist_t *items = NULL; diff --git a/src/or/or.h b/src/or/or.h index da321a44c0..2193a853ca 100644 --- a/src/or/or.h +++ b/src/or/or.h @@ -818,6 +818,16 @@ typedef struct circuit_t circuit_t; #define ALLOW_UNVERIFIED_RENDEZVOUS 8 #define ALLOW_UNVERIFIED_INTRODUCTION 16 +typedef struct exit_redirect_t { + uint32_t addr; + uint32_t mask; + uint16_t port_min; + uint16_t port_max; + + uint32_t addr_dest; + uint32_t port_dest; +} exit_redirect_t; + /** Configuration options for a Tor process */ typedef struct { struct config_line_t *LogOptions; /**< List of configuration lines @@ -905,6 +915,9 @@ typedef struct { char *MyFamily; /**< Declared family for this OR. */ struct config_line_t *NodeFamilies; /**< List of config lines for * node families */ + struct config_line_t *RedirectExit; /**< List of config lines for simple + * addr/port redirection */ + smartlist_t *RedirectExitList; /** List of exit_redirect_t */ } or_options_t; /* XXX are these good enough defaults? */ |