Annotation of embedaddon/strongswan/src/libradius/radius_config.c, revision 1.1.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>