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>