Annotation of embedaddon/strongswan/src/libcharon/plugins/dhcp/dhcp_provider.c, revision 1.1

1.1     ! misho       1: /*
        !             2:  * Copyright (C) 2010 Martin Willi
        !             3:  * Copyright (C) 2010 revosec AG
        !             4:  *
        !             5:  * This program is free software; you can redistribute it and/or modify it
        !             6:  * under the terms of the GNU General Public License as published by the
        !             7:  * Free Software Foundation; either version 2 of the License, or (at your
        !             8:  * option) any later version.  See <http://www.fsf.org/copyleft/gpl.txt>.
        !             9:  *
        !            10:  * This program is distributed in the hope that it will be useful, but
        !            11:  * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
        !            12:  * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
        !            13:  * for more details.
        !            14:  */
        !            15: 
        !            16: #include "dhcp_provider.h"
        !            17: 
        !            18: #include <collections/hashtable.h>
        !            19: #include <threading/mutex.h>
        !            20: 
        !            21: typedef struct private_dhcp_provider_t private_dhcp_provider_t;
        !            22: 
        !            23: /**
        !            24:  * Private data of an dhcp_provider_t object.
        !            25:  */
        !            26: struct private_dhcp_provider_t {
        !            27: 
        !            28:        /**
        !            29:         * Public dhcp_provider_t interface.
        !            30:         */
        !            31:        dhcp_provider_t public;
        !            32: 
        !            33:        /**
        !            34:         * Completed DHCP transactions
        !            35:         */
        !            36:        hashtable_t *transactions;
        !            37: 
        !            38:        /**
        !            39:         * Lock for transactions
        !            40:         */
        !            41:        mutex_t *mutex;
        !            42: 
        !            43:        /**
        !            44:         * DHCP communication socket
        !            45:         */
        !            46:        dhcp_socket_t *socket;
        !            47: };
        !            48: 
        !            49: /**
        !            50:  * Hash ID and host to a key
        !            51:  */
        !            52: static uintptr_t hash_id_host(identification_t *id, host_t *host)
        !            53: {
        !            54:        return chunk_hash_inc(id->get_encoding(id),
        !            55:                                                  chunk_hash(host->get_address(host)));
        !            56: }
        !            57: 
        !            58: /**
        !            59:  * Hash a DHCP transaction to a key, using address and id
        !            60:  */
        !            61: static uintptr_t hash_transaction(dhcp_transaction_t *transaction)
        !            62: {
        !            63:        return hash_id_host(transaction->get_identity(transaction),
        !            64:                                                transaction->get_address(transaction));
        !            65: }
        !            66: 
        !            67: METHOD(attribute_provider_t, acquire_address, host_t*,
        !            68:        private_dhcp_provider_t *this, linked_list_t *pools,
        !            69:        ike_sa_t *ike_sa, host_t *requested)
        !            70: {
        !            71:        dhcp_transaction_t *transaction, *old;
        !            72:        enumerator_t *enumerator;
        !            73:        identification_t *id;
        !            74:        char *pool;
        !            75:        host_t *vip = NULL;
        !            76: 
        !            77:        if (requested->get_family(requested) != AF_INET)
        !            78:        {
        !            79:                return NULL;
        !            80:        }
        !            81:        id = ike_sa->get_other_eap_id(ike_sa);
        !            82:        enumerator = pools->create_enumerator(pools);
        !            83:        while (enumerator->enumerate(enumerator, &pool))
        !            84:        {
        !            85:                if (!streq(pool, "dhcp"))
        !            86:                {
        !            87:                        continue;
        !            88:                }
        !            89:                transaction = this->socket->enroll(this->socket, id);
        !            90:                if (!transaction)
        !            91:                {
        !            92:                        continue;
        !            93:                }
        !            94:                vip = transaction->get_address(transaction);
        !            95:                vip = vip->clone(vip);
        !            96:                this->mutex->lock(this->mutex);
        !            97:                old = this->transactions->put(this->transactions,
        !            98:                                                        (void*)hash_transaction(transaction), transaction);
        !            99:                this->mutex->unlock(this->mutex);
        !           100:                DESTROY_IF(old);
        !           101:                break;
        !           102:        }
        !           103:        enumerator->destroy(enumerator);
        !           104:        return vip;
        !           105: }
        !           106: 
        !           107: METHOD(attribute_provider_t, release_address, bool,
        !           108:        private_dhcp_provider_t *this, linked_list_t *pools,
        !           109:        host_t *address, ike_sa_t *ike_sa)
        !           110: {
        !           111:        dhcp_transaction_t *transaction;
        !           112:        enumerator_t *enumerator;
        !           113:        identification_t *id;
        !           114:        bool found = FALSE;
        !           115:        char *pool;
        !           116: 
        !           117:        if (address->get_family(address) != AF_INET)
        !           118:        {
        !           119:                return FALSE;
        !           120:        }
        !           121:        id = ike_sa->get_other_eap_id(ike_sa);
        !           122:        enumerator = pools->create_enumerator(pools);
        !           123:        while (enumerator->enumerate(enumerator, &pool))
        !           124:        {
        !           125:                if (!streq(pool, "dhcp"))
        !           126:                {
        !           127:                        continue;
        !           128:                }
        !           129:                this->mutex->lock(this->mutex);
        !           130:                transaction = this->transactions->remove(this->transactions,
        !           131:                                                                                (void*)hash_id_host(id, address));
        !           132:                this->mutex->unlock(this->mutex);
        !           133:                if (transaction)
        !           134:                {
        !           135:                        this->socket->release(this->socket, transaction);
        !           136:                        transaction->destroy(transaction);
        !           137:                        found = TRUE;
        !           138:                        break;
        !           139:                }
        !           140:        }
        !           141:        enumerator->destroy(enumerator);
        !           142:        return found;
        !           143: }
        !           144: 
        !           145: METHOD(attribute_provider_t, create_attribute_enumerator, enumerator_t*,
        !           146:        private_dhcp_provider_t *this, linked_list_t *pools, ike_sa_t *ike_sa,
        !           147:        linked_list_t *vips)
        !           148: {
        !           149:        dhcp_transaction_t *transaction = NULL;
        !           150:        enumerator_t *enumerator;
        !           151:        identification_t *id;
        !           152:        host_t *vip;
        !           153: 
        !           154:        if (!pools->find_first(pools, linked_list_match_str, NULL, "dhcp"))
        !           155:        {
        !           156:                return NULL;
        !           157:        }
        !           158: 
        !           159:        id = ike_sa->get_other_eap_id(ike_sa);
        !           160:        this->mutex->lock(this->mutex);
        !           161:        enumerator = vips->create_enumerator(vips);
        !           162:        while (enumerator->enumerate(enumerator, &vip))
        !           163:        {
        !           164:                transaction = this->transactions->get(this->transactions,
        !           165:                                                                                          (void*)hash_id_host(id, vip));
        !           166:                if (transaction)
        !           167:                {
        !           168:                        break;
        !           169:                }
        !           170:        }
        !           171:        enumerator->destroy(enumerator);
        !           172:        if (!transaction)
        !           173:        {
        !           174:                this->mutex->unlock(this->mutex);
        !           175:                return NULL;
        !           176:        }
        !           177:        return enumerator_create_cleaner(
        !           178:                                                transaction->create_attribute_enumerator(transaction),
        !           179:                                                (void*)this->mutex->unlock, this->mutex);
        !           180: }
        !           181: 
        !           182: METHOD(dhcp_provider_t, destroy, void,
        !           183:        private_dhcp_provider_t *this)
        !           184: {
        !           185:        enumerator_t *enumerator;
        !           186:        dhcp_transaction_t *value;
        !           187:        void *key;
        !           188: 
        !           189:        enumerator = this->transactions->create_enumerator(this->transactions);
        !           190:        while (enumerator->enumerate(enumerator, &key, &value))
        !           191:        {
        !           192:                value->destroy(value);
        !           193:        }
        !           194:        enumerator->destroy(enumerator);
        !           195:        this->transactions->destroy(this->transactions);
        !           196:        this->mutex->destroy(this->mutex);
        !           197:        free(this);
        !           198: }
        !           199: 
        !           200: /**
        !           201:  * See header
        !           202:  */
        !           203: dhcp_provider_t *dhcp_provider_create(dhcp_socket_t *socket)
        !           204: {
        !           205:        private_dhcp_provider_t *this;
        !           206: 
        !           207:        INIT(this,
        !           208:                .public = {
        !           209:                        .provider = {
        !           210:                                .acquire_address = _acquire_address,
        !           211:                                .release_address = _release_address,
        !           212:                                .create_attribute_enumerator = _create_attribute_enumerator,
        !           213:                        },
        !           214:                        .destroy = _destroy,
        !           215:                },
        !           216:                .socket = socket,
        !           217:                .mutex = mutex_create(MUTEX_TYPE_DEFAULT),
        !           218:                .transactions = hashtable_create(hashtable_hash_ptr,
        !           219:                                                                                 hashtable_equals_ptr, 8),
        !           220:        );
        !           221: 
        !           222:        return &this->public;
        !           223: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>