Linux Audio

Check our new training course

Loading...
v6.9.4
  1// SPDX-License-Identifier: GPL-2.0-or-later
  2/* SCTP kernel implementation
  3 * (C) Copyright IBM Corp. 2002, 2004
  4 * Copyright (c) 2002 Intel Corp.
  5 *
  6 * This file is part of the SCTP kernel implementation
  7 *
  8 * Sysctl related interfaces for SCTP.
  9 *
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 10 * Please send any bug reports or fixes you make to the
 11 * email address(es):
 12 *    lksctp developers <linux-sctp@vger.kernel.org>
 13 *
 14 * Written or modified by:
 15 *    Mingqin Liu           <liuming@us.ibm.com>
 16 *    Jon Grimm             <jgrimm@us.ibm.com>
 17 *    Ardelle Fan           <ardelle.fan@intel.com>
 18 *    Ryan Layer            <rmlayer@us.ibm.com>
 19 *    Sridhar Samudrala     <sri@us.ibm.com>
 20 */
 21
 22#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
 23
 24#include <net/sctp/structs.h>
 25#include <net/sctp/sctp.h>
 26#include <linux/sysctl.h>
 27
 
 
 28static int timer_max = 86400000; /* ms in one day */
 
 29static int sack_timer_min = 1;
 30static int sack_timer_max = 500;
 31static int addr_scope_max = SCTP_SCOPE_POLICY_MAX;
 32static int rwnd_scale_max = 16;
 33static int rto_alpha_min = 0;
 34static int rto_beta_min = 0;
 35static int rto_alpha_max = 1000;
 36static int rto_beta_max = 1000;
 37static int pf_expose_max = SCTP_PF_EXPOSE_MAX;
 38static int ps_retrans_max = SCTP_PS_RETRANS_MAX;
 39static int udp_port_max = 65535;
 40
 41static unsigned long max_autoclose_min = 0;
 42static unsigned long max_autoclose_max =
 43	(MAX_SCHEDULE_TIMEOUT / HZ > UINT_MAX)
 44	? UINT_MAX : MAX_SCHEDULE_TIMEOUT / HZ;
 45
 46static int proc_sctp_do_hmac_alg(struct ctl_table *ctl, int write,
 47				 void *buffer, size_t *lenp, loff_t *ppos);
 
 48static int proc_sctp_do_rto_min(struct ctl_table *ctl, int write,
 49				void *buffer, size_t *lenp, loff_t *ppos);
 50static int proc_sctp_do_rto_max(struct ctl_table *ctl, int write, void *buffer,
 51				size_t *lenp, loff_t *ppos);
 52static int proc_sctp_do_udp_port(struct ctl_table *ctl, int write, void *buffer,
 53				 size_t *lenp, loff_t *ppos);
 54static int proc_sctp_do_alpha_beta(struct ctl_table *ctl, int write,
 55				   void *buffer, size_t *lenp, loff_t *ppos);
 
 56static int proc_sctp_do_auth(struct ctl_table *ctl, int write,
 57			     void *buffer, size_t *lenp, loff_t *ppos);
 58static int proc_sctp_do_probe_interval(struct ctl_table *ctl, int write,
 59				       void *buffer, size_t *lenp, loff_t *ppos);
 60
 61static struct ctl_table sctp_table[] = {
 62	{
 63		.procname	= "sctp_mem",
 64		.data		= &sysctl_sctp_mem,
 65		.maxlen		= sizeof(sysctl_sctp_mem),
 66		.mode		= 0644,
 67		.proc_handler	= proc_doulongvec_minmax
 68	},
 69	{
 70		.procname	= "sctp_rmem",
 71		.data		= &sysctl_sctp_rmem,
 72		.maxlen		= sizeof(sysctl_sctp_rmem),
 73		.mode		= 0644,
 74		.proc_handler	= proc_dointvec,
 75	},
 76	{
 77		.procname	= "sctp_wmem",
 78		.data		= &sysctl_sctp_wmem,
 79		.maxlen		= sizeof(sysctl_sctp_wmem),
 80		.mode		= 0644,
 81		.proc_handler	= proc_dointvec,
 82	},
 83
 84	{ /* sentinel */ }
 85};
 86
 87/* The following index defines are used in sctp_sysctl_net_register().
 88 * If you add new items to the sctp_net_table, please ensure that
 89 * the index values of these defines hold the same meaning indicated by
 90 * their macro names when they appear in sctp_net_table.
 91 */
 92#define SCTP_RTO_MIN_IDX       0
 93#define SCTP_RTO_MAX_IDX       1
 94#define SCTP_PF_RETRANS_IDX    2
 95#define SCTP_PS_RETRANS_IDX    3
 96
 97static struct ctl_table sctp_net_table[] = {
 98	[SCTP_RTO_MIN_IDX] = {
 
 
 
 
 
 
 
 
 
 99		.procname	= "rto_min",
100		.data		= &init_net.sctp.rto_min,
101		.maxlen		= sizeof(unsigned int),
102		.mode		= 0644,
103		.proc_handler	= proc_sctp_do_rto_min,
104		.extra1         = SYSCTL_ONE,
105		.extra2         = &init_net.sctp.rto_max
106	},
107	[SCTP_RTO_MAX_IDX] =  {
108		.procname	= "rto_max",
109		.data		= &init_net.sctp.rto_max,
110		.maxlen		= sizeof(unsigned int),
111		.mode		= 0644,
112		.proc_handler	= proc_sctp_do_rto_max,
113		.extra1         = &init_net.sctp.rto_min,
114		.extra2         = &timer_max
115	},
116	[SCTP_PF_RETRANS_IDX] = {
117		.procname	= "pf_retrans",
118		.data		= &init_net.sctp.pf_retrans,
119		.maxlen		= sizeof(int),
120		.mode		= 0644,
121		.proc_handler	= proc_dointvec_minmax,
122		.extra1		= SYSCTL_ZERO,
123		.extra2		= &init_net.sctp.ps_retrans,
124	},
125	[SCTP_PS_RETRANS_IDX] = {
126		.procname	= "ps_retrans",
127		.data		= &init_net.sctp.ps_retrans,
128		.maxlen		= sizeof(int),
129		.mode		= 0644,
130		.proc_handler	= proc_dointvec_minmax,
131		.extra1		= &init_net.sctp.pf_retrans,
132		.extra2		= &ps_retrans_max,
133	},
134	{
135		.procname	= "rto_initial",
136		.data		= &init_net.sctp.rto_initial,
137		.maxlen		= sizeof(unsigned int),
138		.mode		= 0644,
139		.proc_handler	= proc_dointvec_minmax,
140		.extra1         = SYSCTL_ONE,
141		.extra2         = &timer_max
142	},
143	{
144		.procname	= "rto_alpha_exp_divisor",
145		.data		= &init_net.sctp.rto_alpha,
146		.maxlen		= sizeof(int),
147		.mode		= 0644,
148		.proc_handler	= proc_sctp_do_alpha_beta,
149		.extra1		= &rto_alpha_min,
150		.extra2		= &rto_alpha_max,
151	},
152	{
153		.procname	= "rto_beta_exp_divisor",
154		.data		= &init_net.sctp.rto_beta,
155		.maxlen		= sizeof(int),
156		.mode		= 0644,
157		.proc_handler	= proc_sctp_do_alpha_beta,
158		.extra1		= &rto_beta_min,
159		.extra2		= &rto_beta_max,
160	},
161	{
162		.procname	= "max_burst",
163		.data		= &init_net.sctp.max_burst,
164		.maxlen		= sizeof(int),
165		.mode		= 0644,
166		.proc_handler	= proc_dointvec_minmax,
167		.extra1		= SYSCTL_ZERO,
168		.extra2		= SYSCTL_INT_MAX,
169	},
170	{
171		.procname	= "cookie_preserve_enable",
172		.data		= &init_net.sctp.cookie_preserve_enable,
173		.maxlen		= sizeof(int),
174		.mode		= 0644,
175		.proc_handler	= proc_dointvec,
176	},
177	{
178		.procname	= "cookie_hmac_alg",
179		.data		= &init_net.sctp.sctp_hmac_alg,
180		.maxlen		= 8,
181		.mode		= 0644,
182		.proc_handler	= proc_sctp_do_hmac_alg,
183	},
184	{
185		.procname	= "valid_cookie_life",
186		.data		= &init_net.sctp.valid_cookie_life,
187		.maxlen		= sizeof(unsigned int),
188		.mode		= 0644,
189		.proc_handler	= proc_dointvec_minmax,
190		.extra1         = SYSCTL_ONE,
191		.extra2         = &timer_max
192	},
193	{
194		.procname	= "sack_timeout",
195		.data		= &init_net.sctp.sack_timeout,
196		.maxlen		= sizeof(int),
197		.mode		= 0644,
198		.proc_handler	= proc_dointvec_minmax,
199		.extra1         = &sack_timer_min,
200		.extra2         = &sack_timer_max,
201	},
202	{
203		.procname	= "hb_interval",
204		.data		= &init_net.sctp.hb_interval,
205		.maxlen		= sizeof(unsigned int),
206		.mode		= 0644,
207		.proc_handler	= proc_dointvec_minmax,
208		.extra1         = SYSCTL_ONE,
209		.extra2         = &timer_max
210	},
211	{
212		.procname	= "association_max_retrans",
213		.data		= &init_net.sctp.max_retrans_association,
214		.maxlen		= sizeof(int),
215		.mode		= 0644,
216		.proc_handler	= proc_dointvec_minmax,
217		.extra1		= SYSCTL_ONE,
218		.extra2		= SYSCTL_INT_MAX,
219	},
220	{
221		.procname	= "path_max_retrans",
222		.data		= &init_net.sctp.max_retrans_path,
223		.maxlen		= sizeof(int),
224		.mode		= 0644,
225		.proc_handler	= proc_dointvec_minmax,
226		.extra1		= SYSCTL_ONE,
227		.extra2		= SYSCTL_INT_MAX,
228	},
229	{
230		.procname	= "max_init_retransmits",
231		.data		= &init_net.sctp.max_retrans_init,
232		.maxlen		= sizeof(int),
233		.mode		= 0644,
234		.proc_handler	= proc_dointvec_minmax,
235		.extra1		= SYSCTL_ONE,
236		.extra2		= SYSCTL_INT_MAX,
 
 
 
 
 
 
 
 
 
237	},
238	{
239		.procname	= "sndbuf_policy",
240		.data		= &init_net.sctp.sndbuf_policy,
241		.maxlen		= sizeof(int),
242		.mode		= 0644,
243		.proc_handler	= proc_dointvec,
244	},
245	{
246		.procname	= "rcvbuf_policy",
247		.data		= &init_net.sctp.rcvbuf_policy,
248		.maxlen		= sizeof(int),
249		.mode		= 0644,
250		.proc_handler	= proc_dointvec,
251	},
252	{
253		.procname	= "default_auto_asconf",
254		.data		= &init_net.sctp.default_auto_asconf,
255		.maxlen		= sizeof(int),
256		.mode		= 0644,
257		.proc_handler	= proc_dointvec,
258	},
259	{
260		.procname	= "addip_enable",
261		.data		= &init_net.sctp.addip_enable,
262		.maxlen		= sizeof(int),
263		.mode		= 0644,
264		.proc_handler	= proc_dointvec,
265	},
266	{
267		.procname	= "addip_noauth_enable",
268		.data		= &init_net.sctp.addip_noauth,
269		.maxlen		= sizeof(int),
270		.mode		= 0644,
271		.proc_handler	= proc_dointvec,
272	},
273	{
274		.procname	= "prsctp_enable",
275		.data		= &init_net.sctp.prsctp_enable,
276		.maxlen		= sizeof(int),
277		.mode		= 0644,
278		.proc_handler	= proc_dointvec,
279	},
280	{
281		.procname	= "reconf_enable",
282		.data		= &init_net.sctp.reconf_enable,
283		.maxlen		= sizeof(int),
284		.mode		= 0644,
285		.proc_handler	= proc_dointvec,
286	},
287	{
288		.procname	= "auth_enable",
289		.data		= &init_net.sctp.auth_enable,
290		.maxlen		= sizeof(int),
291		.mode		= 0644,
292		.proc_handler	= proc_sctp_do_auth,
293	},
294	{
295		.procname	= "intl_enable",
296		.data		= &init_net.sctp.intl_enable,
297		.maxlen		= sizeof(int),
298		.mode		= 0644,
299		.proc_handler	= proc_dointvec,
300	},
301	{
302		.procname	= "ecn_enable",
303		.data		= &init_net.sctp.ecn_enable,
304		.maxlen		= sizeof(int),
305		.mode		= 0644,
306		.proc_handler	= proc_dointvec,
307	},
308	{
309		.procname	= "plpmtud_probe_interval",
310		.data		= &init_net.sctp.probe_interval,
311		.maxlen		= sizeof(int),
312		.mode		= 0644,
313		.proc_handler	= proc_sctp_do_probe_interval,
314	},
315	{
316		.procname	= "udp_port",
317		.data		= &init_net.sctp.udp_port,
318		.maxlen		= sizeof(int),
319		.mode		= 0644,
320		.proc_handler	= proc_sctp_do_udp_port,
321		.extra1		= SYSCTL_ZERO,
322		.extra2		= &udp_port_max,
323	},
324	{
325		.procname	= "encap_port",
326		.data		= &init_net.sctp.encap_port,
327		.maxlen		= sizeof(int),
328		.mode		= 0644,
329		.proc_handler	= proc_dointvec_minmax,
330		.extra1		= SYSCTL_ZERO,
331		.extra2		= &udp_port_max,
332	},
333	{
334		.procname	= "addr_scope_policy",
335		.data		= &init_net.sctp.scope_policy,
336		.maxlen		= sizeof(int),
337		.mode		= 0644,
338		.proc_handler	= proc_dointvec_minmax,
339		.extra1		= SYSCTL_ZERO,
340		.extra2		= &addr_scope_max,
341	},
342	{
343		.procname	= "rwnd_update_shift",
344		.data		= &init_net.sctp.rwnd_upd_shift,
345		.maxlen		= sizeof(int),
346		.mode		= 0644,
347		.proc_handler	= &proc_dointvec_minmax,
348		.extra1		= SYSCTL_ONE,
349		.extra2		= &rwnd_scale_max,
350	},
351	{
352		.procname	= "max_autoclose",
353		.data		= &init_net.sctp.max_autoclose,
354		.maxlen		= sizeof(unsigned long),
355		.mode		= 0644,
356		.proc_handler	= &proc_doulongvec_minmax,
357		.extra1		= &max_autoclose_min,
358		.extra2		= &max_autoclose_max,
359	},
360#ifdef CONFIG_NET_L3_MASTER_DEV
361	{
362		.procname	= "l3mdev_accept",
363		.data		= &init_net.sctp.l3mdev_accept,
364		.maxlen		= sizeof(int),
365		.mode		= 0644,
366		.proc_handler	= proc_dointvec_minmax,
367		.extra1		= SYSCTL_ZERO,
368		.extra2		= SYSCTL_ONE,
369	},
370#endif
371	{
372		.procname	= "pf_enable",
373		.data		= &init_net.sctp.pf_enable,
374		.maxlen		= sizeof(int),
375		.mode		= 0644,
376		.proc_handler	= proc_dointvec,
377	},
378	{
379		.procname	= "pf_expose",
380		.data		= &init_net.sctp.pf_expose,
381		.maxlen		= sizeof(int),
382		.mode		= 0644,
383		.proc_handler	= proc_dointvec_minmax,
384		.extra1		= SYSCTL_ZERO,
385		.extra2		= &pf_expose_max,
386	},
387
388	{ /* sentinel */ }
389};
390
391static int proc_sctp_do_hmac_alg(struct ctl_table *ctl, int write,
392				 void *buffer, size_t *lenp, loff_t *ppos)
 
393{
394	struct net *net = current->nsproxy->net_ns;
395	struct ctl_table tbl;
396	bool changed = false;
397	char *none = "none";
398	char tmp[8] = {0};
399	int ret;
400
401	memset(&tbl, 0, sizeof(struct ctl_table));
402
403	if (write) {
404		tbl.data = tmp;
405		tbl.maxlen = sizeof(tmp);
406	} else {
407		tbl.data = net->sctp.sctp_hmac_alg ? : none;
408		tbl.maxlen = strlen(tbl.data);
409	}
410
411	ret = proc_dostring(&tbl, write, buffer, lenp, ppos);
412	if (write && ret == 0) {
413#ifdef CONFIG_CRYPTO_MD5
414		if (!strncmp(tmp, "md5", 3)) {
415			net->sctp.sctp_hmac_alg = "md5";
416			changed = true;
417		}
418#endif
419#ifdef CONFIG_CRYPTO_SHA1
420		if (!strncmp(tmp, "sha1", 4)) {
421			net->sctp.sctp_hmac_alg = "sha1";
422			changed = true;
423		}
424#endif
425		if (!strncmp(tmp, "none", 4)) {
426			net->sctp.sctp_hmac_alg = NULL;
427			changed = true;
428		}
429		if (!changed)
430			ret = -EINVAL;
431	}
432
433	return ret;
434}
435
436static int proc_sctp_do_rto_min(struct ctl_table *ctl, int write,
437				void *buffer, size_t *lenp, loff_t *ppos)
 
438{
439	struct net *net = current->nsproxy->net_ns;
440	unsigned int min = *(unsigned int *) ctl->extra1;
441	unsigned int max = *(unsigned int *) ctl->extra2;
442	struct ctl_table tbl;
443	int ret, new_value;
444
445	memset(&tbl, 0, sizeof(struct ctl_table));
446	tbl.maxlen = sizeof(unsigned int);
447
448	if (write)
449		tbl.data = &new_value;
450	else
451		tbl.data = &net->sctp.rto_min;
452
453	ret = proc_dointvec(&tbl, write, buffer, lenp, ppos);
454	if (write && ret == 0) {
455		if (new_value > max || new_value < min)
456			return -EINVAL;
457
458		net->sctp.rto_min = new_value;
459	}
460
461	return ret;
462}
463
464static int proc_sctp_do_rto_max(struct ctl_table *ctl, int write,
465				void *buffer, size_t *lenp, loff_t *ppos)
 
466{
467	struct net *net = current->nsproxy->net_ns;
468	unsigned int min = *(unsigned int *) ctl->extra1;
469	unsigned int max = *(unsigned int *) ctl->extra2;
470	struct ctl_table tbl;
471	int ret, new_value;
472
473	memset(&tbl, 0, sizeof(struct ctl_table));
474	tbl.maxlen = sizeof(unsigned int);
475
476	if (write)
477		tbl.data = &new_value;
478	else
479		tbl.data = &net->sctp.rto_max;
480
481	ret = proc_dointvec(&tbl, write, buffer, lenp, ppos);
482	if (write && ret == 0) {
483		if (new_value > max || new_value < min)
484			return -EINVAL;
485
486		net->sctp.rto_max = new_value;
487	}
488
489	return ret;
490}
491
492static int proc_sctp_do_alpha_beta(struct ctl_table *ctl, int write,
493				   void *buffer, size_t *lenp, loff_t *ppos)
 
494{
495	if (write)
496		pr_warn_once("Changing rto_alpha or rto_beta may lead to "
497			     "suboptimal rtt/srtt estimations!\n");
498
499	return proc_dointvec_minmax(ctl, write, buffer, lenp, ppos);
500}
501
502static int proc_sctp_do_auth(struct ctl_table *ctl, int write,
503			     void *buffer, size_t *lenp, loff_t *ppos)
 
504{
505	struct net *net = current->nsproxy->net_ns;
506	struct ctl_table tbl;
507	int new_value, ret;
508
509	memset(&tbl, 0, sizeof(struct ctl_table));
510	tbl.maxlen = sizeof(unsigned int);
511
512	if (write)
513		tbl.data = &new_value;
514	else
515		tbl.data = &net->sctp.auth_enable;
516
517	ret = proc_dointvec(&tbl, write, buffer, lenp, ppos);
518	if (write && ret == 0) {
519		struct sock *sk = net->sctp.ctl_sock;
520
521		net->sctp.auth_enable = new_value;
522		/* Update the value in the control socket */
523		lock_sock(sk);
524		sctp_sk(sk)->ep->auth_enable = new_value;
525		release_sock(sk);
526	}
527
528	return ret;
529}
530
531static int proc_sctp_do_udp_port(struct ctl_table *ctl, int write,
532				 void *buffer, size_t *lenp, loff_t *ppos)
533{
534	struct net *net = current->nsproxy->net_ns;
535	unsigned int min = *(unsigned int *)ctl->extra1;
536	unsigned int max = *(unsigned int *)ctl->extra2;
537	struct ctl_table tbl;
538	int ret, new_value;
539
540	memset(&tbl, 0, sizeof(struct ctl_table));
541	tbl.maxlen = sizeof(unsigned int);
542
543	if (write)
544		tbl.data = &new_value;
545	else
546		tbl.data = &net->sctp.udp_port;
547
548	ret = proc_dointvec(&tbl, write, buffer, lenp, ppos);
549	if (write && ret == 0) {
550		struct sock *sk = net->sctp.ctl_sock;
551
552		if (new_value > max || new_value < min)
553			return -EINVAL;
554
555		net->sctp.udp_port = new_value;
556		sctp_udp_sock_stop(net);
557		if (new_value) {
558			ret = sctp_udp_sock_start(net);
559			if (ret)
560				net->sctp.udp_port = 0;
561		}
562
563		/* Update the value in the control socket */
564		lock_sock(sk);
565		sctp_sk(sk)->udp_port = htons(net->sctp.udp_port);
566		release_sock(sk);
567	}
568
569	return ret;
570}
571
572static int proc_sctp_do_probe_interval(struct ctl_table *ctl, int write,
573				       void *buffer, size_t *lenp, loff_t *ppos)
574{
575	struct net *net = current->nsproxy->net_ns;
576	struct ctl_table tbl;
577	int ret, new_value;
578
579	memset(&tbl, 0, sizeof(struct ctl_table));
580	tbl.maxlen = sizeof(unsigned int);
581
582	if (write)
583		tbl.data = &new_value;
584	else
585		tbl.data = &net->sctp.probe_interval;
586
587	ret = proc_dointvec(&tbl, write, buffer, lenp, ppos);
588	if (write && ret == 0) {
589		if (new_value && new_value < SCTP_PROBE_TIMER_MIN)
590			return -EINVAL;
591
592		net->sctp.probe_interval = new_value;
593	}
594
595	return ret;
596}
597
598int sctp_sysctl_net_register(struct net *net)
599{
600	struct ctl_table *table;
601	int i;
602
603	table = kmemdup(sctp_net_table, sizeof(sctp_net_table), GFP_KERNEL);
604	if (!table)
605		return -ENOMEM;
606
607	for (i = 0; table[i].data; i++)
608		table[i].data += (char *)(&net->sctp) - (char *)&init_net.sctp;
609
610	table[SCTP_RTO_MIN_IDX].extra2 = &net->sctp.rto_max;
611	table[SCTP_RTO_MAX_IDX].extra1 = &net->sctp.rto_min;
612	table[SCTP_PF_RETRANS_IDX].extra2 = &net->sctp.ps_retrans;
613	table[SCTP_PS_RETRANS_IDX].extra1 = &net->sctp.pf_retrans;
614
615	net->sctp.sysctl_header = register_net_sysctl_sz(net, "net/sctp",
616							 table,
617							 ARRAY_SIZE(sctp_net_table));
618	if (net->sctp.sysctl_header == NULL) {
619		kfree(table);
620		return -ENOMEM;
621	}
622	return 0;
623}
624
625void sctp_sysctl_net_unregister(struct net *net)
626{
627	struct ctl_table *table;
628
629	table = net->sctp.sysctl_header->ctl_table_arg;
630	unregister_net_sysctl_table(net->sctp.sysctl_header);
631	kfree(table);
632}
633
634static struct ctl_table_header *sctp_sysctl_header;
635
636/* Sysctl registration.  */
637void sctp_sysctl_register(void)
638{
639	sctp_sysctl_header = register_net_sysctl(&init_net, "net/sctp", sctp_table);
640}
641
642/* Sysctl deregistration.  */
643void sctp_sysctl_unregister(void)
644{
645	unregister_net_sysctl_table(sctp_sysctl_header);
646}
v4.6
 
  1/* SCTP kernel implementation
  2 * (C) Copyright IBM Corp. 2002, 2004
  3 * Copyright (c) 2002 Intel Corp.
  4 *
  5 * This file is part of the SCTP kernel implementation
  6 *
  7 * Sysctl related interfaces for SCTP.
  8 *
  9 * This SCTP implementation is free software;
 10 * you can redistribute it and/or modify it under the terms of
 11 * the GNU General Public License as published by
 12 * the Free Software Foundation; either version 2, or (at your option)
 13 * any later version.
 14 *
 15 * This SCTP implementation is distributed in the hope that it
 16 * will be useful, but WITHOUT ANY WARRANTY; without even the implied
 17 *                 ************************
 18 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 19 * See the GNU General Public License for more details.
 20 *
 21 * You should have received a copy of the GNU General Public License
 22 * along with GNU CC; see the file COPYING.  If not, see
 23 * <http://www.gnu.org/licenses/>.
 24 *
 25 * Please send any bug reports or fixes you make to the
 26 * email address(es):
 27 *    lksctp developers <linux-sctp@vger.kernel.org>
 28 *
 29 * Written or modified by:
 30 *    Mingqin Liu           <liuming@us.ibm.com>
 31 *    Jon Grimm             <jgrimm@us.ibm.com>
 32 *    Ardelle Fan           <ardelle.fan@intel.com>
 33 *    Ryan Layer            <rmlayer@us.ibm.com>
 34 *    Sridhar Samudrala     <sri@us.ibm.com>
 35 */
 36
 37#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
 38
 39#include <net/sctp/structs.h>
 40#include <net/sctp/sctp.h>
 41#include <linux/sysctl.h>
 42
 43static int zero = 0;
 44static int one = 1;
 45static int timer_max = 86400000; /* ms in one day */
 46static int int_max = INT_MAX;
 47static int sack_timer_min = 1;
 48static int sack_timer_max = 500;
 49static int addr_scope_max = 3; /* check sctp_scope_policy_t in include/net/sctp/constants.h for max entries */
 50static int rwnd_scale_max = 16;
 51static int rto_alpha_min = 0;
 52static int rto_beta_min = 0;
 53static int rto_alpha_max = 1000;
 54static int rto_beta_max = 1000;
 
 
 
 55
 56static unsigned long max_autoclose_min = 0;
 57static unsigned long max_autoclose_max =
 58	(MAX_SCHEDULE_TIMEOUT / HZ > UINT_MAX)
 59	? UINT_MAX : MAX_SCHEDULE_TIMEOUT / HZ;
 60
 61static int proc_sctp_do_hmac_alg(struct ctl_table *ctl, int write,
 62				void __user *buffer, size_t *lenp,
 63				loff_t *ppos);
 64static int proc_sctp_do_rto_min(struct ctl_table *ctl, int write,
 65				void __user *buffer, size_t *lenp,
 66				loff_t *ppos);
 67static int proc_sctp_do_rto_max(struct ctl_table *ctl, int write,
 68				void __user *buffer, size_t *lenp,
 69				loff_t *ppos);
 70static int proc_sctp_do_alpha_beta(struct ctl_table *ctl, int write,
 71				   void __user *buffer, size_t *lenp,
 72				   loff_t *ppos);
 73static int proc_sctp_do_auth(struct ctl_table *ctl, int write,
 74			     void __user *buffer, size_t *lenp,
 75			     loff_t *ppos);
 
 76
 77static struct ctl_table sctp_table[] = {
 78	{
 79		.procname	= "sctp_mem",
 80		.data		= &sysctl_sctp_mem,
 81		.maxlen		= sizeof(sysctl_sctp_mem),
 82		.mode		= 0644,
 83		.proc_handler	= proc_doulongvec_minmax
 84	},
 85	{
 86		.procname	= "sctp_rmem",
 87		.data		= &sysctl_sctp_rmem,
 88		.maxlen		= sizeof(sysctl_sctp_rmem),
 89		.mode		= 0644,
 90		.proc_handler	= proc_dointvec,
 91	},
 92	{
 93		.procname	= "sctp_wmem",
 94		.data		= &sysctl_sctp_wmem,
 95		.maxlen		= sizeof(sysctl_sctp_wmem),
 96		.mode		= 0644,
 97		.proc_handler	= proc_dointvec,
 98	},
 99
100	{ /* sentinel */ }
101};
102
 
 
 
 
 
 
 
 
 
 
103static struct ctl_table sctp_net_table[] = {
104	{
105		.procname	= "rto_initial",
106		.data		= &init_net.sctp.rto_initial,
107		.maxlen		= sizeof(unsigned int),
108		.mode		= 0644,
109		.proc_handler	= proc_dointvec_minmax,
110		.extra1         = &one,
111		.extra2         = &timer_max
112	},
113	{
114		.procname	= "rto_min",
115		.data		= &init_net.sctp.rto_min,
116		.maxlen		= sizeof(unsigned int),
117		.mode		= 0644,
118		.proc_handler	= proc_sctp_do_rto_min,
119		.extra1         = &one,
120		.extra2         = &init_net.sctp.rto_max
121	},
122	{
123		.procname	= "rto_max",
124		.data		= &init_net.sctp.rto_max,
125		.maxlen		= sizeof(unsigned int),
126		.mode		= 0644,
127		.proc_handler	= proc_sctp_do_rto_max,
128		.extra1         = &init_net.sctp.rto_min,
129		.extra2         = &timer_max
130	},
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
131	{
132		.procname	= "rto_alpha_exp_divisor",
133		.data		= &init_net.sctp.rto_alpha,
134		.maxlen		= sizeof(int),
135		.mode		= 0644,
136		.proc_handler	= proc_sctp_do_alpha_beta,
137		.extra1		= &rto_alpha_min,
138		.extra2		= &rto_alpha_max,
139	},
140	{
141		.procname	= "rto_beta_exp_divisor",
142		.data		= &init_net.sctp.rto_beta,
143		.maxlen		= sizeof(int),
144		.mode		= 0644,
145		.proc_handler	= proc_sctp_do_alpha_beta,
146		.extra1		= &rto_beta_min,
147		.extra2		= &rto_beta_max,
148	},
149	{
150		.procname	= "max_burst",
151		.data		= &init_net.sctp.max_burst,
152		.maxlen		= sizeof(int),
153		.mode		= 0644,
154		.proc_handler	= proc_dointvec_minmax,
155		.extra1		= &zero,
156		.extra2		= &int_max
157	},
158	{
159		.procname	= "cookie_preserve_enable",
160		.data		= &init_net.sctp.cookie_preserve_enable,
161		.maxlen		= sizeof(int),
162		.mode		= 0644,
163		.proc_handler	= proc_dointvec,
164	},
165	{
166		.procname	= "cookie_hmac_alg",
167		.data		= &init_net.sctp.sctp_hmac_alg,
168		.maxlen		= 8,
169		.mode		= 0644,
170		.proc_handler	= proc_sctp_do_hmac_alg,
171	},
172	{
173		.procname	= "valid_cookie_life",
174		.data		= &init_net.sctp.valid_cookie_life,
175		.maxlen		= sizeof(unsigned int),
176		.mode		= 0644,
177		.proc_handler	= proc_dointvec_minmax,
178		.extra1         = &one,
179		.extra2         = &timer_max
180	},
181	{
182		.procname	= "sack_timeout",
183		.data		= &init_net.sctp.sack_timeout,
184		.maxlen		= sizeof(int),
185		.mode		= 0644,
186		.proc_handler	= proc_dointvec_minmax,
187		.extra1         = &sack_timer_min,
188		.extra2         = &sack_timer_max,
189	},
190	{
191		.procname	= "hb_interval",
192		.data		= &init_net.sctp.hb_interval,
193		.maxlen		= sizeof(unsigned int),
194		.mode		= 0644,
195		.proc_handler	= proc_dointvec_minmax,
196		.extra1         = &one,
197		.extra2         = &timer_max
198	},
199	{
200		.procname	= "association_max_retrans",
201		.data		= &init_net.sctp.max_retrans_association,
202		.maxlen		= sizeof(int),
203		.mode		= 0644,
204		.proc_handler	= proc_dointvec_minmax,
205		.extra1		= &one,
206		.extra2		= &int_max
207	},
208	{
209		.procname	= "path_max_retrans",
210		.data		= &init_net.sctp.max_retrans_path,
211		.maxlen		= sizeof(int),
212		.mode		= 0644,
213		.proc_handler	= proc_dointvec_minmax,
214		.extra1		= &one,
215		.extra2		= &int_max
216	},
217	{
218		.procname	= "max_init_retransmits",
219		.data		= &init_net.sctp.max_retrans_init,
220		.maxlen		= sizeof(int),
221		.mode		= 0644,
222		.proc_handler	= proc_dointvec_minmax,
223		.extra1		= &one,
224		.extra2		= &int_max
225	},
226	{
227		.procname	= "pf_retrans",
228		.data		= &init_net.sctp.pf_retrans,
229		.maxlen		= sizeof(int),
230		.mode		= 0644,
231		.proc_handler	= proc_dointvec_minmax,
232		.extra1		= &zero,
233		.extra2		= &int_max
234	},
235	{
236		.procname	= "sndbuf_policy",
237		.data		= &init_net.sctp.sndbuf_policy,
238		.maxlen		= sizeof(int),
239		.mode		= 0644,
240		.proc_handler	= proc_dointvec,
241	},
242	{
243		.procname	= "rcvbuf_policy",
244		.data		= &init_net.sctp.rcvbuf_policy,
245		.maxlen		= sizeof(int),
246		.mode		= 0644,
247		.proc_handler	= proc_dointvec,
248	},
249	{
250		.procname	= "default_auto_asconf",
251		.data		= &init_net.sctp.default_auto_asconf,
252		.maxlen		= sizeof(int),
253		.mode		= 0644,
254		.proc_handler	= proc_dointvec,
255	},
256	{
257		.procname	= "addip_enable",
258		.data		= &init_net.sctp.addip_enable,
259		.maxlen		= sizeof(int),
260		.mode		= 0644,
261		.proc_handler	= proc_dointvec,
262	},
263	{
264		.procname	= "addip_noauth_enable",
265		.data		= &init_net.sctp.addip_noauth,
266		.maxlen		= sizeof(int),
267		.mode		= 0644,
268		.proc_handler	= proc_dointvec,
269	},
270	{
271		.procname	= "prsctp_enable",
272		.data		= &init_net.sctp.prsctp_enable,
273		.maxlen		= sizeof(int),
274		.mode		= 0644,
275		.proc_handler	= proc_dointvec,
276	},
277	{
 
 
 
 
 
 
 
278		.procname	= "auth_enable",
279		.data		= &init_net.sctp.auth_enable,
280		.maxlen		= sizeof(int),
281		.mode		= 0644,
282		.proc_handler	= proc_sctp_do_auth,
283	},
284	{
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
285		.procname	= "addr_scope_policy",
286		.data		= &init_net.sctp.scope_policy,
287		.maxlen		= sizeof(int),
288		.mode		= 0644,
289		.proc_handler	= proc_dointvec_minmax,
290		.extra1		= &zero,
291		.extra2		= &addr_scope_max,
292	},
293	{
294		.procname	= "rwnd_update_shift",
295		.data		= &init_net.sctp.rwnd_upd_shift,
296		.maxlen		= sizeof(int),
297		.mode		= 0644,
298		.proc_handler	= &proc_dointvec_minmax,
299		.extra1		= &one,
300		.extra2		= &rwnd_scale_max,
301	},
302	{
303		.procname	= "max_autoclose",
304		.data		= &init_net.sctp.max_autoclose,
305		.maxlen		= sizeof(unsigned long),
306		.mode		= 0644,
307		.proc_handler	= &proc_doulongvec_minmax,
308		.extra1		= &max_autoclose_min,
309		.extra2		= &max_autoclose_max,
310	},
 
 
 
 
 
 
 
 
 
 
 
311	{
312		.procname	= "pf_enable",
313		.data		= &init_net.sctp.pf_enable,
314		.maxlen		= sizeof(int),
315		.mode		= 0644,
316		.proc_handler	= proc_dointvec,
317	},
 
 
 
 
 
 
 
 
 
318
319	{ /* sentinel */ }
320};
321
322static int proc_sctp_do_hmac_alg(struct ctl_table *ctl, int write,
323				void __user *buffer, size_t *lenp,
324				loff_t *ppos)
325{
326	struct net *net = current->nsproxy->net_ns;
327	struct ctl_table tbl;
328	bool changed = false;
329	char *none = "none";
330	char tmp[8] = {0};
331	int ret;
332
333	memset(&tbl, 0, sizeof(struct ctl_table));
334
335	if (write) {
336		tbl.data = tmp;
337		tbl.maxlen = sizeof(tmp);
338	} else {
339		tbl.data = net->sctp.sctp_hmac_alg ? : none;
340		tbl.maxlen = strlen(tbl.data);
341	}
342
343	ret = proc_dostring(&tbl, write, buffer, lenp, ppos);
344	if (write && ret == 0) {
345#ifdef CONFIG_CRYPTO_MD5
346		if (!strncmp(tmp, "md5", 3)) {
347			net->sctp.sctp_hmac_alg = "md5";
348			changed = true;
349		}
350#endif
351#ifdef CONFIG_CRYPTO_SHA1
352		if (!strncmp(tmp, "sha1", 4)) {
353			net->sctp.sctp_hmac_alg = "sha1";
354			changed = true;
355		}
356#endif
357		if (!strncmp(tmp, "none", 4)) {
358			net->sctp.sctp_hmac_alg = NULL;
359			changed = true;
360		}
361		if (!changed)
362			ret = -EINVAL;
363	}
364
365	return ret;
366}
367
368static int proc_sctp_do_rto_min(struct ctl_table *ctl, int write,
369				void __user *buffer, size_t *lenp,
370				loff_t *ppos)
371{
372	struct net *net = current->nsproxy->net_ns;
373	unsigned int min = *(unsigned int *) ctl->extra1;
374	unsigned int max = *(unsigned int *) ctl->extra2;
375	struct ctl_table tbl;
376	int ret, new_value;
377
378	memset(&tbl, 0, sizeof(struct ctl_table));
379	tbl.maxlen = sizeof(unsigned int);
380
381	if (write)
382		tbl.data = &new_value;
383	else
384		tbl.data = &net->sctp.rto_min;
385
386	ret = proc_dointvec(&tbl, write, buffer, lenp, ppos);
387	if (write && ret == 0) {
388		if (new_value > max || new_value < min)
389			return -EINVAL;
390
391		net->sctp.rto_min = new_value;
392	}
393
394	return ret;
395}
396
397static int proc_sctp_do_rto_max(struct ctl_table *ctl, int write,
398				void __user *buffer, size_t *lenp,
399				loff_t *ppos)
400{
401	struct net *net = current->nsproxy->net_ns;
402	unsigned int min = *(unsigned int *) ctl->extra1;
403	unsigned int max = *(unsigned int *) ctl->extra2;
404	struct ctl_table tbl;
405	int ret, new_value;
406
407	memset(&tbl, 0, sizeof(struct ctl_table));
408	tbl.maxlen = sizeof(unsigned int);
409
410	if (write)
411		tbl.data = &new_value;
412	else
413		tbl.data = &net->sctp.rto_max;
414
415	ret = proc_dointvec(&tbl, write, buffer, lenp, ppos);
416	if (write && ret == 0) {
417		if (new_value > max || new_value < min)
418			return -EINVAL;
419
420		net->sctp.rto_max = new_value;
421	}
422
423	return ret;
424}
425
426static int proc_sctp_do_alpha_beta(struct ctl_table *ctl, int write,
427				   void __user *buffer, size_t *lenp,
428				   loff_t *ppos)
429{
430	if (write)
431		pr_warn_once("Changing rto_alpha or rto_beta may lead to "
432			     "suboptimal rtt/srtt estimations!\n");
433
434	return proc_dointvec_minmax(ctl, write, buffer, lenp, ppos);
435}
436
437static int proc_sctp_do_auth(struct ctl_table *ctl, int write,
438			     void __user *buffer, size_t *lenp,
439			     loff_t *ppos)
440{
441	struct net *net = current->nsproxy->net_ns;
442	struct ctl_table tbl;
443	int new_value, ret;
444
445	memset(&tbl, 0, sizeof(struct ctl_table));
446	tbl.maxlen = sizeof(unsigned int);
447
448	if (write)
449		tbl.data = &new_value;
450	else
451		tbl.data = &net->sctp.auth_enable;
452
453	ret = proc_dointvec(&tbl, write, buffer, lenp, ppos);
454	if (write && ret == 0) {
455		struct sock *sk = net->sctp.ctl_sock;
456
457		net->sctp.auth_enable = new_value;
458		/* Update the value in the control socket */
459		lock_sock(sk);
460		sctp_sk(sk)->ep->auth_enable = new_value;
461		release_sock(sk);
462	}
463
464	return ret;
465}
466
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
467int sctp_sysctl_net_register(struct net *net)
468{
469	struct ctl_table *table;
470	int i;
471
472	table = kmemdup(sctp_net_table, sizeof(sctp_net_table), GFP_KERNEL);
473	if (!table)
474		return -ENOMEM;
475
476	for (i = 0; table[i].data; i++)
477		table[i].data += (char *)(&net->sctp) - (char *)&init_net.sctp;
478
479	net->sctp.sysctl_header = register_net_sysctl(net, "net/sctp", table);
 
 
 
 
 
 
 
480	if (net->sctp.sysctl_header == NULL) {
481		kfree(table);
482		return -ENOMEM;
483	}
484	return 0;
485}
486
487void sctp_sysctl_net_unregister(struct net *net)
488{
489	struct ctl_table *table;
490
491	table = net->sctp.sysctl_header->ctl_table_arg;
492	unregister_net_sysctl_table(net->sctp.sysctl_header);
493	kfree(table);
494}
495
496static struct ctl_table_header *sctp_sysctl_header;
497
498/* Sysctl registration.  */
499void sctp_sysctl_register(void)
500{
501	sctp_sysctl_header = register_net_sysctl(&init_net, "net/sctp", sctp_table);
502}
503
504/* Sysctl deregistration.  */
505void sctp_sysctl_unregister(void)
506{
507	unregister_net_sysctl_table(sctp_sysctl_header);
508}