summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/or/config.c53
1 files changed, 41 insertions, 12 deletions
diff --git a/src/or/config.c b/src/or/config.c
index a62ce5f1fa..ecfaf401cc 100644
--- a/src/or/config.c
+++ b/src/or/config.c
@@ -348,6 +348,7 @@ static int check_nickname_list(const char *lst, const char *name);
static void config_register_addressmaps(or_options_t *options);
static int parse_dir_server_line(const char *line, int validate_only);
+static int config_cmp_single_addr_policy(addr_policy_t *a, addr_policy_t *b);
static int parse_redirect_line(smartlist_t *result,
config_line_t *line);
static int parse_log_severity_range(const char *range, int *min_out,
@@ -3001,9 +3002,10 @@ config_expand_exit_policy_aliases(smartlist_t *entries, int assume_action)
static void
config_exit_policy_remove_redundancies(addr_policy_t **dest)
{
- addr_policy_t *ap, *tmp;
+ addr_policy_t *ap, *tmp, *victim;
int have_seen_accept=0;
+ /* Step one: find a *:* entry and cut off everything after it. */
for (ap=*dest; ap; ap=ap->next) {
if (ap->policy_type == ADDR_POLICY_ACCEPT)
have_seen_accept=1;
@@ -3028,6 +3030,22 @@ config_exit_policy_remove_redundancies(addr_policy_t **dest)
}
}
}
+
+ /* Step two: for every entry, see if there's an exact duplicate
+ * later on, and remove it. */
+ for (ap=*dest; ap; ap=ap->next) {
+ tmp=ap;
+ while (tmp) {
+ if (tmp->next && !config_cmp_single_addr_policy(ap, tmp->next)) {
+ victim = tmp->next;
+ tmp->next = victim->next;
+ victim->next = NULL;
+ addr_policy_free(victim);
+ } else {
+ tmp=tmp->next;
+ }
+ }
+ }
}
#define DEFAULT_EXIT_POLICY \
@@ -3104,22 +3122,33 @@ config_parse_addr_policy(config_line_t *cfg,
return r;
}
-/** Compare two provided address policies, and return -1, 0, or 1 if the first
- * is less than, equal to, or greater than the second. */
+/** Compare two provided address policy items, and return -1, 0, or 1
+ * if the first is less than, equal to, or greater than the second. */
+static int
+config_cmp_single_addr_policy(addr_policy_t *a, addr_policy_t *b)
+{
+ int r;
+ if ((r=((int)a->policy_type - (int)b->policy_type)))
+ return r;
+ if ((r=((int)a->addr - (int)b->addr)))
+ return r;
+ if ((r=((int)a->msk - (int)b->msk)))
+ return r;
+ if ((r=((int)a->prt_min - (int)b->prt_min)))
+ return r;
+ if ((r=((int)a->prt_max - (int)b->prt_max)))
+ return r;
+ return 0;
+}
+
+/** Like config_cmp_single_addr_policy() above, but looks at the
+ * whole set of policies in each case. */
int
config_cmp_addr_policies(addr_policy_t *a, addr_policy_t *b)
{
int r;
while (a && b) {
- if ((r=((int)a->policy_type - (int)b->policy_type)))
- return r;
- if ((r=((int)a->addr - (int)b->addr)))
- return r;
- if ((r=((int)a->msk - (int)b->msk)))
- return r;
- if ((r=((int)a->prt_min - (int)b->prt_min)))
- return r;
- if ((r=((int)a->prt_max - (int)b->prt_max)))
+ if ((r=config_cmp_single_addr_policy(a,b)))
return r;
a = a->next;
b = b->next;