1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /*
23  * Copyright (c) 2004, 2010, Oracle and/or its affiliates. All rights reserved.
24  */
25 
26 #include <inet/ip.h>
27 #include <inet/ip6.h>
28 #include <inet/sctp/sctp_stack.h>
29 #include <inet/sctp/sctp_impl.h>
30 #include <sys/sunddi.h>
31 
32 /* Max size IP datagram is 64k - 1 */
33 #define	SCTP_MSS_MAX_IPV4 (IP_MAXPACKET - (sizeof (ipha_t) + \
34 					sizeof (sctp_hdr_t)))
35 #define	SCTP_MSS_MAX_IPV6 (IP_MAXPACKET - (sizeof (ip6_t) + \
36 					sizeof (sctp_hdr_t)))
37 /* Max of the above */
38 #define	SCTP_MSS_MAX	SCTP_MSS_MAX_IPV4
39 
40 /*
41  * returns the current list of listener limit configuration.
42  */
43 /* ARGSUSED */
44 static int
45 sctp_listener_conf_get(void *cbarg, mod_prop_info_t *pinfo, const char *ifname,
46     void *val, uint_t psize, uint_t flags)
47 {
48 	sctp_stack_t	*sctps = (sctp_stack_t *)cbarg;
49 	sctp_listener_t	*sl;
50 	char		*pval = val;
51 	size_t		nbytes = 0, tbytes = 0;
52 	uint_t		size;
53 	int		err = 0;
54 
55 	bzero(pval, psize);
56 	size = psize;
57 
58 	if (flags & (MOD_PROP_DEFAULT|MOD_PROP_PERM|MOD_PROP_POSSIBLE))
59 		return (0);
60 
61 	mutex_enter(&sctps->sctps_listener_conf_lock);
62 	for (sl = list_head(&sctps->sctps_listener_conf); sl != NULL;
63 	    sl = list_next(&sctps->sctps_listener_conf, sl)) {
64 		if (psize == size)
65 			nbytes = snprintf(pval, size, "%d:%d",  sl->sl_port,
66 			    sl->sl_ratio);
67 		else
68 			nbytes = snprintf(pval, size, ",%d:%d",  sl->sl_port,
69 			    sl->sl_ratio);
70 		size -= nbytes;
71 		pval += nbytes;
72 		tbytes += nbytes;
73 		if (tbytes >= psize) {
74 			/* Buffer overflow, stop copying information */
75 			err = ENOBUFS;
76 			break;
77 		}
78 	}
79 
80 	mutex_exit(&sctps->sctps_listener_conf_lock);
81 	return (err);
82 }
83 
84 /*
85  * add a new listener limit configuration.
86  */
87 /* ARGSUSED */
88 static int
89 sctp_listener_conf_add(void *cbarg, cred_t *cr, mod_prop_info_t *pinfo,
90     const char *ifname, const void* pval, uint_t flags)
91 {
92 	sctp_listener_t	*new_sl;
93 	sctp_listener_t	*sl;
94 	long		lport;
95 	long		ratio;
96 	char		*colon;
97 	sctp_stack_t	*sctps = (sctp_stack_t *)cbarg;
98 
99 	if (flags & MOD_PROP_DEFAULT)
100 		return (ENOTSUP);
101 
102 	if (ddi_strtol(pval, &colon, 10, &lport) != 0 || lport <= 0 ||
103 	    lport > USHRT_MAX || *colon != ':') {
104 		return (EINVAL);
105 	}
106 	if (ddi_strtol(colon + 1, NULL, 10, &ratio) != 0 || ratio <= 0)
107 		return (EINVAL);
108 
109 	mutex_enter(&sctps->sctps_listener_conf_lock);
110 	for (sl = list_head(&sctps->sctps_listener_conf); sl != NULL;
111 	    sl = list_next(&sctps->sctps_listener_conf, sl)) {
112 		/* There is an existing entry, so update its ratio value. */
113 		if (sl->sl_port == lport) {
114 			sl->sl_ratio = ratio;
115 			mutex_exit(&sctps->sctps_listener_conf_lock);
116 			return (0);
117 		}
118 	}
119 
120 	if ((new_sl = kmem_alloc(sizeof (sctp_listener_t), KM_NOSLEEP)) ==
121 	    NULL) {
122 		mutex_exit(&sctps->sctps_listener_conf_lock);
123 		return (ENOMEM);
124 	}
125 
126 	new_sl->sl_port = lport;
127 	new_sl->sl_ratio = ratio;
128 	list_insert_tail(&sctps->sctps_listener_conf, new_sl);
129 	mutex_exit(&sctps->sctps_listener_conf_lock);
130 	return (0);
131 }
132 
133 /*
134  * remove a listener limit configuration.
135  */
136 /* ARGSUSED */
137 static int
138 sctp_listener_conf_del(void *cbarg, cred_t *cr, mod_prop_info_t *pinfo,
139     const char *ifname, const void* pval, uint_t flags)
140 {
141 	sctp_listener_t	*sl;
142 	long		lport;
143 	sctp_stack_t	*sctps = (sctp_stack_t *)cbarg;
144 
145 	if (flags & MOD_PROP_DEFAULT)
146 		return (ENOTSUP);
147 
148 	if (ddi_strtol(pval, NULL, 10, &lport) != 0 || lport <= 0 ||
149 	    lport > USHRT_MAX) {
150 		return (EINVAL);
151 	}
152 	mutex_enter(&sctps->sctps_listener_conf_lock);
153 	for (sl = list_head(&sctps->sctps_listener_conf); sl != NULL;
154 	    sl = list_next(&sctps->sctps_listener_conf, sl)) {
155 		if (sl->sl_port == lport) {
156 			list_remove(&sctps->sctps_listener_conf, sl);
157 			mutex_exit(&sctps->sctps_listener_conf_lock);
158 			kmem_free(sl, sizeof (sctp_listener_t));
159 			return (0);
160 		}
161 	}
162 	mutex_exit(&sctps->sctps_listener_conf_lock);
163 	return (ESRCH);
164 }
165 
166 /*
167  * All of these are alterable, within the min/max values given, at run time.
168  *
169  * Note: All those tunables which do not start with "sctp_" are Committed and
170  * therefore are public. See PSARC 2009/306.
171  */
172 mod_prop_info_t sctp_propinfo_tbl[] = {
173 	{ "sctp_max_init_retr", MOD_PROTO_SCTP,
174 	    mod_set_uint32, mod_get_uint32,
175 	    {0, 128, 8}, {8} },
176 
177 	{ "sctp_pa_max_retr", MOD_PROTO_SCTP,
178 	    mod_set_uint32, mod_get_uint32,
179 	    {1, 128, 10}, {10} },
180 
181 	{ "sctp_pp_max_retr", MOD_PROTO_SCTP,
182 	    mod_set_uint32, mod_get_uint32,
183 	    {1, 128, 5}, {5} },
184 
185 	{ "sctp_cwnd_max", MOD_PROTO_SCTP,
186 	    mod_set_uint32, mod_get_uint32,
187 	    {128, (1<<30), 1024*1024}, {1024*1024} },
188 
189 	{ "smallest_nonpriv_port", MOD_PROTO_SCTP,
190 	    mod_set_uint32, mod_get_uint32,
191 	    {1024, (32*1024), 1024}, {1024} },
192 
193 	{ "sctp_ipv4_ttl", MOD_PROTO_SCTP,
194 	    mod_set_uint32, mod_get_uint32,
195 	    {1, 255, 64}, {64} },
196 
197 	{ "sctp_heartbeat_interval", MOD_PROTO_SCTP,
198 	    mod_set_uint32, mod_get_uint32,
199 	    {0, 1*DAYS, 30*SECONDS}, {30*SECONDS} },
200 
201 	{ "sctp_initial_mtu", MOD_PROTO_SCTP,
202 	    mod_set_uint32, mod_get_uint32,
203 	    {68, 65535, 1500}, {1500} },
204 
205 	{ "sctp_mtu_probe_interval", MOD_PROTO_SCTP,
206 	    mod_set_uint32, mod_get_uint32,
207 	    {0, 1*DAYS, 10*MINUTES}, {10*MINUTES} },
208 
209 	{ "sctp_new_secret_interval", MOD_PROTO_SCTP,
210 	    mod_set_uint32, mod_get_uint32,
211 	    {0, 1*DAYS, 2*MINUTES}, {2*MINUTES} },
212 
213 	/* tunable - 10 */
214 	{ "sctp_deferred_ack_interval", MOD_PROTO_SCTP,
215 	    mod_set_uint32, mod_get_uint32,
216 	    {10*MS, 1*MINUTES, 100*MS}, {100*MS} },
217 
218 	{ "sctp_snd_lowat_fraction", MOD_PROTO_SCTP,
219 	    mod_set_uint32, mod_get_uint32,
220 	    {0, 16, 0}, {0} },
221 
222 	{ "sctp_ignore_path_mtu", MOD_PROTO_SCTP,
223 	    mod_set_boolean, mod_get_boolean,
224 	    {B_FALSE}, {B_FALSE} },
225 
226 	{ "sctp_initial_ssthresh", MOD_PROTO_SCTP,
227 	    mod_set_uint32, mod_get_uint32,
228 	    {1024, UINT32_MAX, SCTP_RECV_HIWATER}, { SCTP_RECV_HIWATER} },
229 
230 	{ "smallest_anon_port", MOD_PROTO_SCTP,
231 	    mod_set_uint32, mod_get_uint32,
232 	    {1024, ULP_MAX_PORT, 32*1024}, {32*1024} },
233 
234 	{ "largest_anon_port", MOD_PROTO_SCTP,
235 	    mod_set_uint32, mod_get_uint32,
236 	    {1024, ULP_MAX_PORT, ULP_MAX_PORT}, {ULP_MAX_PORT} },
237 
238 	{ "send_maxbuf", MOD_PROTO_SCTP,
239 	    mod_set_uint32, mod_get_uint32,
240 	    {SCTP_XMIT_LOWATER,  (1<<30),  SCTP_XMIT_HIWATER},
241 	    {SCTP_XMIT_HIWATER} },
242 
243 	{ "sctp_xmit_lowat", MOD_PROTO_SCTP,
244 	    mod_set_uint32, mod_get_uint32,
245 	    {SCTP_XMIT_LOWATER,  (1<<30),  SCTP_XMIT_LOWATER},
246 	    {SCTP_XMIT_LOWATER} },
247 
248 	{ "recv_maxbuf", MOD_PROTO_SCTP,
249 	    mod_set_uint32, mod_get_uint32,
250 	    {SCTP_RECV_LOWATER,  (1<<30),  SCTP_RECV_HIWATER},
251 	    {SCTP_RECV_HIWATER} },
252 
253 	{ "sctp_max_buf", MOD_PROTO_SCTP,
254 	    mod_set_uint32, mod_get_uint32,
255 	    {8192, (1<<30), 1024*1024}, {1024*1024} },
256 
257 	/* tunable - 20 */
258 	{ "sctp_rtt_updates", MOD_PROTO_SCTP,
259 	    mod_set_uint32, mod_get_uint32,
260 	    {0, 65536, 20}, {20} },
261 
262 	{ "sctp_ipv6_hoplimit", MOD_PROTO_SCTP,
263 	    mod_set_uint32, mod_get_uint32,
264 	    {0, IPV6_MAX_HOPS, IPV6_DEFAULT_HOPS}, {IPV6_DEFAULT_HOPS} },
265 
266 	{ "sctp_rto_min", MOD_PROTO_SCTP,
267 	    mod_set_uint32, mod_get_uint32,
268 	    {500*MS, 60*SECONDS, 1*SECONDS}, {1*SECONDS} },
269 
270 	{ "sctp_rto_max", MOD_PROTO_SCTP,
271 	    mod_set_uint32, mod_get_uint32,
272 	    {1*SECONDS, 60000*SECONDS, 60*SECONDS}, {60*SECONDS} },
273 
274 	{ "sctp_rto_initial", MOD_PROTO_SCTP,
275 	    mod_set_uint32, mod_get_uint32,
276 	    {1*SECONDS, 60000*SECONDS, 3*SECONDS}, {3*SECONDS} },
277 
278 	{ "sctp_cookie_life", MOD_PROTO_SCTP,
279 	    mod_set_uint32, mod_get_uint32,
280 	    {10*MS, 60000*SECONDS, 60*SECONDS}, {60*SECONDS} },
281 
282 	{ "sctp_max_in_streams", MOD_PROTO_SCTP,
283 	    mod_set_uint32, mod_get_uint32,
284 	    {1, UINT16_MAX, 32}, {32} },
285 
286 	{ "sctp_initial_out_streams", MOD_PROTO_SCTP,
287 	    mod_set_uint32, mod_get_uint32,
288 	    {1, UINT16_MAX, 32}, {32} },
289 
290 	{ "sctp_shutack_wait_bound", MOD_PROTO_SCTP,
291 	    mod_set_uint32, mod_get_uint32,
292 	    {0, 300*SECONDS, 60*SECONDS}, {60*SECONDS} },
293 
294 	{ "sctp_maxburst", MOD_PROTO_SCTP,
295 	    mod_set_uint32, mod_get_uint32,
296 	    {2, 8, 4}, {4} },
297 
298 	/* tunable - 30 */
299 	{ "sctp_addip_enabled", MOD_PROTO_SCTP,
300 	    mod_set_boolean, mod_get_boolean,
301 	    {B_FALSE}, {B_FALSE} },
302 
303 	{ "sctp_recv_hiwat_minmss", MOD_PROTO_SCTP,
304 	    mod_set_uint32, mod_get_uint32,
305 	    {1, 65536, 4}, {4} },
306 
307 	{ "sctp_slow_start_initial", MOD_PROTO_SCTP,
308 	    mod_set_uint32, mod_get_uint32,
309 	    {1, 16, 4}, {4} },
310 
311 	{ "sctp_slow_start_after_idle", MOD_PROTO_SCTP,
312 	    mod_set_uint32, mod_get_uint32,
313 	    {1, 16384, 4}, {4} },
314 
315 	{ "sctp_prsctp_enabled", MOD_PROTO_SCTP,
316 	    mod_set_boolean, mod_get_boolean,
317 	    {B_TRUE}, {B_TRUE} },
318 
319 	{ "sctp_fast_rxt_thresh", MOD_PROTO_SCTP,
320 	    mod_set_uint32, mod_get_uint32,
321 	    {1, 10000, 3}, {3} },
322 
323 	{ "sctp_deferred_acks_max", MOD_PROTO_SCTP,
324 	    mod_set_uint32, mod_get_uint32,
325 	    { 1, 16, 2}, {2} },
326 
327 	/*
328 	 * sctp_wroff_xtra is the extra space in front of SCTP/IP header
329 	 * for link layer header.  It has to be a multiple of 8.
330 	 */
331 	{ "sctp_wroff_xtra", MOD_PROTO_SCTP,
332 	    mod_set_aligned, mod_get_uint32,
333 	    {0, 256, 32}, {32} },
334 
335 	{ "extra_priv_ports", MOD_PROTO_SCTP,
336 	    mod_set_extra_privports, mod_get_extra_privports,
337 	    {1, ULP_MAX_PORT, 0}, {0} },
338 
339 	{ "sctp_listener_limit_conf", MOD_PROTO_SCTP,
340 	    NULL, sctp_listener_conf_get, {0}, {0} },
341 
342 	{ "sctp_listener_limit_conf_add", MOD_PROTO_SCTP,
343 	    sctp_listener_conf_add, NULL, {0}, {0} },
344 
345 	{ "sctp_listener_limit_conf_del", MOD_PROTO_SCTP,
346 	    sctp_listener_conf_del, NULL, {0}, {0} },
347 
348 	{ "?", MOD_PROTO_SCTP, NULL, mod_get_allprop, {0}, {0} },
349 
350 	{ NULL, 0, NULL, NULL, {0}, {0} }
351 };
352 
353 int sctp_propinfo_count = A_CNT(sctp_propinfo_tbl);
354