Annotation of embedaddon/strongswan/src/libradius/radius_config.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: /*
        !            17:  * Copyright (C) 2015 Thom Troy
        !            18:  *
        !            19:  * Permission is hereby granted, free of charge, to any person obtaining a copy
        !            20:  * of this software and associated documentation files (the "Software"), to deal
        !            21:  * in the Software without restriction, including without limitation the rights
        !            22:  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
        !            23:  * copies of the Software, and to permit persons to whom the Software is
        !            24:  * furnished to do so, subject to the following conditions:
        !            25:  *
        !            26:  * The above copyright notice and this permission notice shall be included in
        !            27:  * all copies or substantial portions of the Software.
        !            28:  *
        !            29:  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
        !            30:  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
        !            31:  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
        !            32:  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
        !            33:  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
        !            34:  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
        !            35:  * THE SOFTWARE.
        !            36:  */
        !            37: 
        !            38: #include "radius_config.h"
        !            39: 
        !            40: #include <threading/mutex.h>
        !            41: #include <threading/condvar.h>
        !            42: #include <collections/linked_list.h>
        !            43: 
        !            44: typedef struct private_radius_config_t private_radius_config_t;
        !            45: 
        !            46: /**
        !            47:  * Private data of an radius_config_t object.
        !            48:  */
        !            49: struct private_radius_config_t {
        !            50: 
        !            51:        /**
        !            52:         * Public radius_config_t interface.
        !            53:         */
        !            54:        radius_config_t public;
        !            55: 
        !            56:        /**
        !            57:         * list of radius sockets, as radius_socket_t
        !            58:         */
        !            59:        linked_list_t *sockets;
        !            60: 
        !            61:        /**
        !            62:         * Total number of sockets, in list + currently in use
        !            63:         */
        !            64:        int socket_count;
        !            65: 
        !            66:        /**
        !            67:         * mutex to lock sockets list
        !            68:         */
        !            69:        mutex_t *mutex;
        !            70: 
        !            71:        /**
        !            72:         * condvar to wait for sockets
        !            73:         */
        !            74:        condvar_t *condvar;
        !            75: 
        !            76:        /**
        !            77:         * Server name
        !            78:         */
        !            79:        char *name;
        !            80: 
        !            81:        /**
        !            82:         * NAS-Identifier
        !            83:         */
        !            84:        chunk_t nas_identifier;
        !            85: 
        !            86:        /**
        !            87:         * Preference boost for this server
        !            88:         */
        !            89:        int preference;
        !            90: 
        !            91:        /**
        !            92:         * Is the server currently reachable
        !            93:         */
        !            94:        bool reachable;
        !            95: 
        !            96:        /**
        !            97:         * Retry counter for unreachable servers
        !            98:         */
        !            99:        int retry;
        !           100: 
        !           101:        /**
        !           102:         * reference count
        !           103:         */
        !           104:        refcount_t ref;
        !           105: };
        !           106: 
        !           107: METHOD(radius_config_t, get_socket, radius_socket_t*,
        !           108:        private_radius_config_t *this)
        !           109: {
        !           110:        radius_socket_t *skt;
        !           111: 
        !           112:        this->mutex->lock(this->mutex);
        !           113:        while (this->sockets->remove_first(this->sockets, (void**)&skt) != SUCCESS)
        !           114:        {
        !           115:                this->condvar->wait(this->condvar, this->mutex);
        !           116:        }
        !           117:        this->mutex->unlock(this->mutex);
        !           118:        return skt;
        !           119: }
        !           120: 
        !           121: METHOD(radius_config_t, put_socket, void,
        !           122:        private_radius_config_t *this, radius_socket_t *skt, bool result)
        !           123: {
        !           124:        this->mutex->lock(this->mutex);
        !           125:        this->sockets->insert_last(this->sockets, skt);
        !           126:        this->mutex->unlock(this->mutex);
        !           127:        this->condvar->signal(this->condvar);
        !           128:        this->reachable = result;
        !           129: }
        !           130: 
        !           131: METHOD(radius_config_t, get_nas_identifier, chunk_t,
        !           132:        private_radius_config_t *this)
        !           133: {
        !           134:        return this->nas_identifier;
        !           135: }
        !           136: 
        !           137: METHOD(radius_config_t, get_preference, int,
        !           138:        private_radius_config_t *this)
        !           139: {
        !           140:        int pref;
        !           141: 
        !           142:        if (this->socket_count == 0)
        !           143:        {       /* don't have sockets, huh? */
        !           144:                return -1;
        !           145:        }
        !           146:        /* calculate preference between 0-100 + boost */
        !           147:        pref = this->preference;
        !           148:        pref += this->sockets->get_count(this->sockets) * 100 / this->socket_count;
        !           149:        if (this->reachable)
        !           150:        {       /* reachable server get a boost: pref = 110-210 + boost */
        !           151:                return pref + 110;
        !           152:        }
        !           153:        /* Not reachable. Increase preference randomly to let it retry from
        !           154:         * time to time, especially if other servers have high load. */
        !           155:        this->retry++;
        !           156:        if (this->retry % 128 == 0)
        !           157:        {       /* every 64th request gets 210, same as unloaded reachable */
        !           158:                return pref + 110;
        !           159:        }
        !           160:        if (this->retry % 32 == 0)
        !           161:        {       /* every 32th request gets 190, wins against average loaded */
        !           162:                return pref + 90;
        !           163:        }
        !           164:        if (this->retry % 8 == 0)
        !           165:        {       /* every 8th request gets 110, same as server under load */
        !           166:                return pref + 10;
        !           167:        }
        !           168:        /* other get ~100, less than fully loaded */
        !           169:        return pref;
        !           170: }
        !           171: 
        !           172: METHOD(radius_config_t, get_name, char*,
        !           173:        private_radius_config_t *this)
        !           174: {
        !           175:        return this->name;
        !           176: }
        !           177: 
        !           178: METHOD(radius_config_t, get_ref, radius_config_t*,
        !           179:        private_radius_config_t *this)
        !           180: {
        !           181:        ref_get(&this->ref);
        !           182:        return &this->public;
        !           183: }
        !           184: 
        !           185: 
        !           186: METHOD(radius_config_t, destroy, void,
        !           187:        private_radius_config_t *this)
        !           188: {
        !           189:        if (ref_put(&this->ref))
        !           190:        {
        !           191:                this->mutex->destroy(this->mutex);
        !           192:                this->condvar->destroy(this->condvar);
        !           193:                this->sockets->destroy_offset(this->sockets,
        !           194:                                                                          offsetof(radius_socket_t, destroy));
        !           195:                free(this);
        !           196:        }
        !           197: }
        !           198: 
        !           199: /**
        !           200:  * See header
        !           201:  */
        !           202: radius_config_t *radius_config_create(char *name, char *address,
        !           203:                                                                          uint16_t auth_port, uint16_t acct_port,
        !           204:                                                                          char *nas_identifier, char *secret,
        !           205:                                                                          int sockets, int preference,
        !           206:                                                                          u_int tries, double timeout, double base)
        !           207: {
        !           208:        private_radius_config_t *this;
        !           209:        radius_socket_t *socket;
        !           210: 
        !           211:        INIT(this,
        !           212:                .public = {
        !           213:                        .get_socket = _get_socket,
        !           214:                        .put_socket = _put_socket,
        !           215:                        .get_nas_identifier = _get_nas_identifier,
        !           216:                        .get_preference = _get_preference,
        !           217:                        .get_name = _get_name,
        !           218:                        .get_ref = _get_ref,
        !           219:                        .destroy = _destroy,
        !           220:                },
        !           221:                .reachable = TRUE,
        !           222:                .nas_identifier = chunk_create(nas_identifier, strlen(nas_identifier)),
        !           223:                .socket_count = sockets,
        !           224:                .sockets = linked_list_create(),
        !           225:                .mutex = mutex_create(MUTEX_TYPE_DEFAULT),
        !           226:                .condvar = condvar_create(CONDVAR_TYPE_DEFAULT),
        !           227:                .name = name,
        !           228:                .preference = preference,
        !           229:                .ref = 1,
        !           230:        );
        !           231: 
        !           232:        while (sockets--)
        !           233:        {
        !           234:                socket = radius_socket_create(address, auth_port, acct_port,
        !           235:                                                                          chunk_create(secret, strlen(secret)),
        !           236:                                                                          tries, timeout, base);
        !           237:                if (!socket)
        !           238:                {
        !           239:                        destroy(this);
        !           240:                        return NULL;
        !           241:                }
        !           242:                this->sockets->insert_last(this->sockets, socket);
        !           243:        }
        !           244:        return &this->public;
        !           245: }

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