]> git.ipfire.org Git - thirdparty/strongswan.git/blobdiff - src/libcharon/sa/trap_manager.c
child-sa: Pass default interface ID inherited from IKE_SA
[thirdparty/strongswan.git] / src / libcharon / sa / trap_manager.c
index 85e2207751a5072e5fe41fb3a0dce107b5e33b77..2bc531b387dd2a0d4a6d91dd5422f33627ee27a7 100644 (file)
@@ -1,7 +1,7 @@
 /*
- * Copyright (C) 2011-2015 Tobias Brunner
+ * Copyright (C) 2011-2017 Tobias Brunner
  * Copyright (C) 2009 Martin Willi
- * Hochschule fuer Technik Rapperswil
+ * HSR Hochschule fuer Technik Rapperswil
  *
  * This program is free software; you can redistribute it and/or modify it
  * under the terms of the GNU General Public License as published by the
@@ -115,7 +115,7 @@ typedef struct {
        /** pending IKE_SA connecting upon acquire */
        ike_sa_t *ike_sa;
        /** reqid of pending trap policy */
-       u_int32_t reqid;
+       uint32_t reqid;
        /** destination address (wildcard case) */
        host_t *dst;
 } acquire_t;
@@ -140,25 +140,51 @@ static void destroy_acquire(acquire_t *this)
        free(this);
 }
 
-/**
- * match an acquire entry by reqid
- */
-static bool acquire_by_reqid(acquire_t *this, u_int32_t *reqid)
+CALLBACK(acquire_by_reqid, bool,
+       acquire_t *this, va_list args)
 {
-       return this->reqid == *reqid;
+       uint32_t reqid;
+
+       VA_ARGS_VGET(args, reqid);
+       return this->reqid == reqid;
+}
+
+CALLBACK(acquire_by_dst, bool,
+       acquire_t *this, va_list args)
+{
+       host_t *dst;
+
+       VA_ARGS_VGET(args, dst);
+       return this->dst && this->dst->ip_equals(this->dst, dst);
 }
 
 /**
- * match an acquire entry by destination address
+ * Check if any remote TS are dynamic
  */
-static bool acquire_by_dst(acquire_t *this, host_t *dst)
+static bool dynamic_remote_ts(child_cfg_t *child)
 {
-       return this->dst && this->dst->ip_equals(this->dst, dst);
+       enumerator_t *enumerator;
+       linked_list_t *other_ts;
+       traffic_selector_t *ts;
+       bool found = FALSE;
+
+       other_ts = child->get_traffic_selectors(child, FALSE, NULL, NULL, FALSE);
+       enumerator = other_ts->create_enumerator(other_ts);
+       while (enumerator->enumerate(enumerator, &ts))
+       {
+               if (ts->is_dynamic(ts))
+               {
+                       found = TRUE;
+                       break;
+               }
+       }
+       enumerator->destroy(enumerator);
+       other_ts->destroy_offset(other_ts, offsetof(traffic_selector_t, destroy));
+       return found;
 }
 
-METHOD(trap_manager_t, install, u_int32_t,
-       private_trap_manager_t *this, peer_cfg_t *peer, child_cfg_t *child,
-       u_int32_t reqid)
+METHOD(trap_manager_t, install, bool,
+       private_trap_manager_t *this, peer_cfg_t *peer, child_cfg_t *child)
 {
        entry_t *entry, *found = NULL;
        ike_cfg_t *ike_cfg;
@@ -170,7 +196,7 @@ METHOD(trap_manager_t, install, u_int32_t,
        linked_list_t *proposals;
        proposal_t *proposal;
        protocol_id_t proto = PROTO_ESP;
-       bool wildcard = FALSE;
+       bool result = FALSE, wildcard = FALSE;
 
        /* try to resolve addresses */
        ike_cfg = peer->get_ike_cfg(peer);
@@ -182,27 +208,39 @@ METHOD(trap_manager_t, install, u_int32_t,
                me = host_create_any(other->get_family(other));
                wildcard = TRUE;
        }
-       else if (!other || other->is_anyaddr(other))
+       else if (other && other->is_anyaddr(other))
        {
-               DESTROY_IF(other);
+               other->destroy(other);
                DBG1(DBG_CFG, "installing trap failed, remote address unknown");
-               return 0;
+               return FALSE;
        }
        else
-       {
-               me = ike_cfg->resolve_me(ike_cfg, other->get_family(other));
-               if (!me || me->is_anyaddr(me))
+       {       /* depending on the traffic selectors we don't really need a remote
+                * host yet, but we might fail later if no IP can be resolved */
+               if (!other && dynamic_remote_ts(child))
+               {       /* with dynamic TS we do need a host, otherwise 0.0.0.0/0 is used,
+                        * which is probably not what users expect*/
+                       DBG1(DBG_CFG, "installing trap failed, remote address unknown with "
+                                "dynamic traffic selector");
+                       return FALSE;
+               }
+               me = ike_cfg->resolve_me(ike_cfg, other ? other->get_family(other)
+                                                                                               : AF_UNSPEC);
+               if (!other)
+               {
+                       other = host_create_any(me ? me->get_family(me) : AF_INET);
+               }
+               other->set_port(other, ike_cfg->get_other_port(ike_cfg));
+               if ((!me || me->is_anyaddr(me)) && !other->is_anyaddr(other))
                {
                        DESTROY_IF(me);
                        me = charon->kernel->get_source_addr(charon->kernel, other, NULL);
-                       if (!me)
-                       {
-                               DBG1(DBG_CFG, "installing trap failed, local address unknown");
-                               other->destroy(other);
-                               return 0;
-                       }
-                       me->set_port(me, ike_cfg->get_my_port(ike_cfg));
                }
+               if (!me)
+               {
+                       me = host_create_any(other->get_family(other));
+               }
+               me->set_port(me, ike_cfg->get_my_port(ike_cfg));
        }
 
        this->lock->write_lock(this->lock);
@@ -211,12 +249,14 @@ METHOD(trap_manager_t, install, u_int32_t,
                this->lock->unlock(this->lock);
                other->destroy(other);
                me->destroy(me);
-               return 0;
+               return FALSE;
        }
        enumerator = this->traps->create_enumerator(this->traps);
        while (enumerator->enumerate(enumerator, &entry))
        {
-               if (streq(entry->name, child->get_name(child)))
+               if (streq(entry->name, child->get_name(child)) &&
+                       streq(entry->peer_cfg->get_name(entry->peer_cfg),
+                                 peer->get_name(peer)))
                {
                        found = entry;
                        if (entry->child_sa)
@@ -236,11 +276,10 @@ METHOD(trap_manager_t, install, u_int32_t,
                        this->lock->unlock(this->lock);
                        other->destroy(other);
                        me->destroy(me);
-                       return 0;
+                       return FALSE;
                }
                /* config might have changed so update everything */
                DBG1(DBG_CFG, "updating already routed CHILD_SA '%s'", found->name);
-               reqid = found->child_sa->get_reqid(found->child_sa);
        }
 
        INIT(entry,
@@ -254,14 +293,21 @@ METHOD(trap_manager_t, install, u_int32_t,
        this->lock->unlock(this->lock);
 
        /* create and route CHILD_SA */
-       child_sa = child_sa_create(me, other, child, reqid, FALSE, 0, 0);
+       child_sa_create_t child_data = {
+               /* TODO: no reason to allocate unique interface IDs, there is currently
+                * no event to use them upon trap installation and we'd also have to
+                * pass them in a later initiate() call */
+               .if_id_in_def = peer->get_if_id(peer, TRUE),
+               .if_id_out_def = peer->get_if_id(peer, FALSE),
+       };
+       child_sa = child_sa_create(me, other, child, &child_data);
 
        list = linked_list_create_with_items(me, NULL);
-       my_ts = child->get_traffic_selectors(child, TRUE, NULL, list);
+       my_ts = child->get_traffic_selectors(child, TRUE, NULL, list, FALSE);
        list->destroy_offset(list, offsetof(host_t, destroy));
 
        list = linked_list_create_with_items(other, NULL);
-       other_ts = child->get_traffic_selectors(child, FALSE, NULL, list);
+       other_ts = child->get_traffic_selectors(child, FALSE, NULL, list, FALSE);
        list->destroy_offset(list, offsetof(host_t, destroy));
 
        /* We don't know the finally negotiated protocol (ESP|AH), we install
@@ -274,7 +320,8 @@ METHOD(trap_manager_t, install, u_int32_t,
        proposals->destroy_offset(proposals, offsetof(proposal_t, destroy));
        child_sa->set_protocol(child_sa, proto);
        child_sa->set_mode(child_sa, child->get_mode(child));
-       status = child_sa->add_policies(child_sa, my_ts, other_ts);
+       child_sa->set_policies(child_sa, my_ts, other_ts);
+       status = child_sa->install_policies(child_sa);
        my_ts->destroy_offset(my_ts, offsetof(traffic_selector_t, destroy));
        other_ts->destroy_offset(other_ts, offsetof(traffic_selector_t, destroy));
        if (status != SUCCESS)
@@ -285,14 +332,13 @@ METHOD(trap_manager_t, install, u_int32_t,
                this->lock->unlock(this->lock);
                entry->child_sa = child_sa;
                destroy_entry(entry);
-               reqid = 0;
        }
        else
        {
-               reqid = child_sa->get_reqid(child_sa);
                this->lock->write_lock(this->lock);
                entry->child_sa = child_sa;
                this->lock->unlock(this->lock);
+               result = TRUE;
        }
        if (found)
        {
@@ -303,21 +349,25 @@ METHOD(trap_manager_t, install, u_int32_t,
        this->installing--;
        this->condvar->signal(this->condvar);
        this->lock->unlock(this->lock);
-       return reqid;
+       return result;
 }
 
 METHOD(trap_manager_t, uninstall, bool,
-       private_trap_manager_t *this, u_int32_t reqid)
+       private_trap_manager_t *this, char *peer, char *child)
 {
        enumerator_t *enumerator;
        entry_t *entry, *found = NULL;
 
        this->lock->write_lock(this->lock);
+       while (this->installing)
+       {
+               this->condvar->wait(this->condvar, this->lock);
+       }
        enumerator = this->traps->create_enumerator(this->traps);
        while (enumerator->enumerate(enumerator, &entry))
        {
-               if (entry->child_sa &&
-                       entry->child_sa->get_reqid(entry->child_sa) == reqid)
+               if (streq(entry->name, child) &&
+                  (!peer || streq(peer, entry->peer_cfg->get_name(entry->peer_cfg))))
                {
                        this->traps->remove_at(this->traps, enumerator);
                        found = entry;
@@ -329,32 +379,38 @@ METHOD(trap_manager_t, uninstall, bool,
 
        if (!found)
        {
-               DBG1(DBG_CFG, "trap %d not found to uninstall", reqid);
                return FALSE;
        }
        destroy_entry(found);
        return TRUE;
 }
 
-/**
- * convert enumerated entries to peer_cfg, child_sa
- */
-static bool trap_filter(rwlock_t *lock, entry_t **entry, peer_cfg_t **peer_cfg,
-                                               void *none, child_sa_t **child_sa)
+CALLBACK(trap_filter, bool,
+       rwlock_t *lock, enumerator_t *orig, va_list args)
 {
-       if (!(*entry)->child_sa)
-       {       /* skip entries that are currently being installed */
-               return FALSE;
-       }
-       if (peer_cfg)
-       {
-               *peer_cfg = (*entry)->peer_cfg;
-       }
-       if (child_sa)
+       entry_t *entry;
+       peer_cfg_t **peer_cfg;
+       child_sa_t **child_sa;
+
+       VA_ARGS_VGET(args, peer_cfg, child_sa);
+
+       while (orig->enumerate(orig, &entry))
        {
-               *child_sa = (*entry)->child_sa;
+               if (!entry->child_sa)
+               {       /* skip entries that are currently being installed */
+                       continue;
+               }
+               if (peer_cfg)
+               {
+                       *peer_cfg = entry->peer_cfg;
+               }
+               if (child_sa)
+               {
+                       *child_sa = entry->child_sa;
+               }
+               return TRUE;
        }
-       return TRUE;
+       return FALSE;
 }
 
 METHOD(trap_manager_t, create_enumerator, enumerator_t*,
@@ -362,37 +418,12 @@ METHOD(trap_manager_t, create_enumerator, enumerator_t*,
 {
        this->lock->read_lock(this->lock);
        return enumerator_create_filter(this->traps->create_enumerator(this->traps),
-                                                                       (void*)trap_filter, this->lock,
+                                                                       trap_filter, this->lock,
                                                                        (void*)this->lock->unlock);
 }
 
-METHOD(trap_manager_t, find_reqid, u_int32_t,
-       private_trap_manager_t *this, child_cfg_t *child)
-{
-       enumerator_t *enumerator;
-       entry_t *entry;
-       u_int32_t reqid = 0;
-
-       this->lock->read_lock(this->lock);
-       enumerator = this->traps->create_enumerator(this->traps);
-       while (enumerator->enumerate(enumerator, &entry))
-       {
-               if (streq(entry->name, child->get_name(child)))
-               {
-                       if (entry->child_sa)
-                       {
-                               reqid = entry->child_sa->get_reqid(entry->child_sa);
-                       }
-                       break;
-               }
-       }
-       enumerator->destroy(enumerator);
-       this->lock->unlock(this->lock);
-       return reqid;
-}
-
 METHOD(trap_manager_t, acquire, void,
-       private_trap_manager_t *this, u_int32_t reqid,
+       private_trap_manager_t *this, uint32_t reqid,
        traffic_selector_t *src, traffic_selector_t *dst)
 {
        enumerator_t *enumerator;
@@ -430,11 +461,11 @@ METHOD(trap_manager_t, acquire, void,
        if (wildcard)
        {       /* for wildcard acquires we check that we don't have a pending acquire
                 * with the same peer */
-               u_int8_t mask;
+               uint8_t mask;
 
                dst->to_subnet(dst, &host, &mask);
-               if (this->acquires->find_first(this->acquires, (void*)acquire_by_dst,
-                                                                         (void**)&acquire, host) == SUCCESS)
+               if (this->acquires->find_first(this->acquires, acquire_by_dst,
+                                                                         (void**)&acquire, host))
                {
                        host->destroy(host);
                        ignore = TRUE;
@@ -450,8 +481,8 @@ METHOD(trap_manager_t, acquire, void,
        }
        else
        {
-               if (this->acquires->find_first(this->acquires, (void*)acquire_by_reqid,
-                                                                         (void**)&acquire, &reqid) == SUCCESS)
+               if (this->acquires->find_first(this->acquires, acquire_by_reqid,
+                                                                         (void**)&acquire, reqid))
                {
                        ignore = TRUE;
                }
@@ -483,8 +514,8 @@ METHOD(trap_manager_t, acquire, void,
                if (ike_sa)
                {
                        ike_cfg_t *ike_cfg;
-                       u_int16_t port;
-                       u_int8_t mask;
+                       uint16_t port;
+                       uint8_t mask;
 
                        ike_sa->set_peer_cfg(ike_sa, peer);
                        ike_cfg = ike_sa->get_ike_cfg(ike_sa);
@@ -646,7 +677,6 @@ trap_manager_t *trap_manager_create(void)
                        .install = _install,
                        .uninstall = _uninstall,
                        .create_enumerator = _create_enumerator,
-                       .find_reqid = _find_reqid,
                        .acquire = _acquire,
                        .flush = _flush,
                        .destroy = _destroy,