1a5494dcdSEric W. Biederman /* 2a5494dcdSEric W. Biederman * Copyright (C) 2007 3a5494dcdSEric W. Biederman * 4a5494dcdSEric W. Biederman * Author: Eric Biederman <ebiederm@xmision.com> 5a5494dcdSEric W. Biederman * 6a5494dcdSEric W. Biederman * This program is free software; you can redistribute it and/or 7a5494dcdSEric W. Biederman * modify it under the terms of the GNU General Public License as 8a5494dcdSEric W. Biederman * published by the Free Software Foundation, version 2 of the 9a5494dcdSEric W. Biederman * License. 10a5494dcdSEric W. Biederman */ 11a5494dcdSEric W. Biederman 12a5494dcdSEric W. Biederman #include <linux/module.h> 13a5494dcdSEric W. Biederman #include <linux/ipc.h> 14a5494dcdSEric W. Biederman #include <linux/nsproxy.h> 15a5494dcdSEric W. Biederman #include <linux/sysctl.h> 16a5494dcdSEric W. Biederman #include <linux/uaccess.h> 17ae5e1b22SPavel Emelyanov #include <linux/ipc_namespace.h> 186546bc42SNadia Derbey #include <linux/msg.h> 196546bc42SNadia Derbey #include "util.h" 20a5494dcdSEric W. Biederman 21a5494dcdSEric W. Biederman static void *get_ipc(ctl_table *table) 22a5494dcdSEric W. Biederman { 23a5494dcdSEric W. Biederman char *which = table->data; 24a5494dcdSEric W. Biederman struct ipc_namespace *ipc_ns = current->nsproxy->ipc_ns; 25a5494dcdSEric W. Biederman which = (which - (char *)&init_ipc_ns) + (char *)ipc_ns; 26a5494dcdSEric W. Biederman return which; 27a5494dcdSEric W. Biederman } 28a5494dcdSEric W. Biederman 29a5494dcdSEric W. Biederman #ifdef CONFIG_PROC_FS 30a5494dcdSEric W. Biederman static int proc_ipc_dointvec(ctl_table *table, int write, struct file *filp, 31a5494dcdSEric W. Biederman void __user *buffer, size_t *lenp, loff_t *ppos) 32a5494dcdSEric W. Biederman { 33a5494dcdSEric W. Biederman struct ctl_table ipc_table; 34a5494dcdSEric W. Biederman memcpy(&ipc_table, table, sizeof(ipc_table)); 35a5494dcdSEric W. Biederman ipc_table.data = get_ipc(table); 36a5494dcdSEric W. Biederman 37a5494dcdSEric W. Biederman return proc_dointvec(&ipc_table, write, filp, buffer, lenp, ppos); 38a5494dcdSEric W. Biederman } 39a5494dcdSEric W. Biederman 4091cfb2b4SNadia Derbey static int proc_ipc_callback_dointvec(ctl_table *table, int write, 4191cfb2b4SNadia Derbey struct file *filp, void __user *buffer, size_t *lenp, loff_t *ppos) 4291cfb2b4SNadia Derbey { 436546bc42SNadia Derbey struct ctl_table ipc_table; 4491cfb2b4SNadia Derbey size_t lenp_bef = *lenp; 4591cfb2b4SNadia Derbey int rc; 4691cfb2b4SNadia Derbey 476546bc42SNadia Derbey memcpy(&ipc_table, table, sizeof(ipc_table)); 486546bc42SNadia Derbey ipc_table.data = get_ipc(table); 496546bc42SNadia Derbey 506546bc42SNadia Derbey rc = proc_dointvec(&ipc_table, write, filp, buffer, lenp, ppos); 5191cfb2b4SNadia Derbey 5291cfb2b4SNadia Derbey if (write && !rc && lenp_bef == *lenp) 539eefe520SNadia Derbey /* 549eefe520SNadia Derbey * Tunable has successfully been changed by hand. Disable its 559eefe520SNadia Derbey * automatic adjustment. This simply requires unregistering 569eefe520SNadia Derbey * the notifiers that trigger recalculation. 579eefe520SNadia Derbey */ 589eefe520SNadia Derbey unregister_ipcns_notifier(current->nsproxy->ipc_ns); 5991cfb2b4SNadia Derbey 6091cfb2b4SNadia Derbey return rc; 6191cfb2b4SNadia Derbey } 6291cfb2b4SNadia Derbey 63a5494dcdSEric W. Biederman static int proc_ipc_doulongvec_minmax(ctl_table *table, int write, 64a5494dcdSEric W. Biederman struct file *filp, void __user *buffer, size_t *lenp, loff_t *ppos) 65a5494dcdSEric W. Biederman { 66a5494dcdSEric W. Biederman struct ctl_table ipc_table; 67a5494dcdSEric W. Biederman memcpy(&ipc_table, table, sizeof(ipc_table)); 68a5494dcdSEric W. Biederman ipc_table.data = get_ipc(table); 69a5494dcdSEric W. Biederman 70a5494dcdSEric W. Biederman return proc_doulongvec_minmax(&ipc_table, write, filp, buffer, 71a5494dcdSEric W. Biederman lenp, ppos); 72a5494dcdSEric W. Biederman } 73a5494dcdSEric W. Biederman 74*4c2c3b4aSakpm@linux-foundation.org /* 75*4c2c3b4aSakpm@linux-foundation.org * Routine that is called when the file "auto_msgmni" has successfully been 76*4c2c3b4aSakpm@linux-foundation.org * written. 77*4c2c3b4aSakpm@linux-foundation.org * Two values are allowed: 78*4c2c3b4aSakpm@linux-foundation.org * 0: unregister msgmni's callback routine from the ipc namespace notifier 79*4c2c3b4aSakpm@linux-foundation.org * chain. This means that msgmni won't be recomputed anymore upon memory 80*4c2c3b4aSakpm@linux-foundation.org * add/remove or ipc namespace creation/removal. 81*4c2c3b4aSakpm@linux-foundation.org * 1: register back the callback routine. 82*4c2c3b4aSakpm@linux-foundation.org */ 83*4c2c3b4aSakpm@linux-foundation.org static void ipc_auto_callback(int val) 84*4c2c3b4aSakpm@linux-foundation.org { 85*4c2c3b4aSakpm@linux-foundation.org if (!val) 86*4c2c3b4aSakpm@linux-foundation.org unregister_ipcns_notifier(current->nsproxy->ipc_ns); 87*4c2c3b4aSakpm@linux-foundation.org else { 88*4c2c3b4aSakpm@linux-foundation.org /* 89*4c2c3b4aSakpm@linux-foundation.org * Re-enable automatic recomputing only if not already 90*4c2c3b4aSakpm@linux-foundation.org * enabled. 91*4c2c3b4aSakpm@linux-foundation.org */ 92*4c2c3b4aSakpm@linux-foundation.org recompute_msgmni(current->nsproxy->ipc_ns); 93*4c2c3b4aSakpm@linux-foundation.org cond_register_ipcns_notifier(current->nsproxy->ipc_ns); 94*4c2c3b4aSakpm@linux-foundation.org } 95*4c2c3b4aSakpm@linux-foundation.org } 96*4c2c3b4aSakpm@linux-foundation.org 979eefe520SNadia Derbey static int proc_ipcauto_dointvec_minmax(ctl_table *table, int write, 989eefe520SNadia Derbey struct file *filp, void __user *buffer, size_t *lenp, loff_t *ppos) 999eefe520SNadia Derbey { 1009eefe520SNadia Derbey struct ctl_table ipc_table; 1019eefe520SNadia Derbey size_t lenp_bef = *lenp; 1029eefe520SNadia Derbey int oldval; 1039eefe520SNadia Derbey int rc; 1049eefe520SNadia Derbey 1059eefe520SNadia Derbey memcpy(&ipc_table, table, sizeof(ipc_table)); 1069eefe520SNadia Derbey ipc_table.data = get_ipc(table); 1079eefe520SNadia Derbey oldval = *((int *)(ipc_table.data)); 1089eefe520SNadia Derbey 1099eefe520SNadia Derbey rc = proc_dointvec_minmax(&ipc_table, write, filp, buffer, lenp, ppos); 1109eefe520SNadia Derbey 1119eefe520SNadia Derbey if (write && !rc && lenp_bef == *lenp) { 1129eefe520SNadia Derbey int newval = *((int *)(ipc_table.data)); 1139eefe520SNadia Derbey /* 1149eefe520SNadia Derbey * The file "auto_msgmni" has correctly been set. 1159eefe520SNadia Derbey * React by (un)registering the corresponding tunable, if the 1169eefe520SNadia Derbey * value has changed. 1179eefe520SNadia Derbey */ 1189eefe520SNadia Derbey if (newval != oldval) 1199eefe520SNadia Derbey ipc_auto_callback(newval); 1209eefe520SNadia Derbey } 1219eefe520SNadia Derbey 1229eefe520SNadia Derbey return rc; 1239eefe520SNadia Derbey } 1249eefe520SNadia Derbey 125a5494dcdSEric W. Biederman #else 126a5494dcdSEric W. Biederman #define proc_ipc_doulongvec_minmax NULL 127a5494dcdSEric W. Biederman #define proc_ipc_dointvec NULL 12891cfb2b4SNadia Derbey #define proc_ipc_callback_dointvec NULL 1299eefe520SNadia Derbey #define proc_ipcauto_dointvec_minmax NULL 130a5494dcdSEric W. Biederman #endif 131a5494dcdSEric W. Biederman 132a5494dcdSEric W. Biederman #ifdef CONFIG_SYSCTL_SYSCALL 133a5494dcdSEric W. Biederman /* The generic sysctl ipc data routine. */ 134f221e726SAlexey Dobriyan static int sysctl_ipc_data(ctl_table *table, 135a5494dcdSEric W. Biederman void __user *oldval, size_t __user *oldlenp, 136a5494dcdSEric W. Biederman void __user *newval, size_t newlen) 137a5494dcdSEric W. Biederman { 138a5494dcdSEric W. Biederman size_t len; 139a5494dcdSEric W. Biederman void *data; 140a5494dcdSEric W. Biederman 141a5494dcdSEric W. Biederman /* Get out of I don't have a variable */ 142a5494dcdSEric W. Biederman if (!table->data || !table->maxlen) 143a5494dcdSEric W. Biederman return -ENOTDIR; 144a5494dcdSEric W. Biederman 145a5494dcdSEric W. Biederman data = get_ipc(table); 146a5494dcdSEric W. Biederman if (!data) 147a5494dcdSEric W. Biederman return -ENOTDIR; 148a5494dcdSEric W. Biederman 149a5494dcdSEric W. Biederman if (oldval && oldlenp) { 150a5494dcdSEric W. Biederman if (get_user(len, oldlenp)) 151a5494dcdSEric W. Biederman return -EFAULT; 152a5494dcdSEric W. Biederman if (len) { 153a5494dcdSEric W. Biederman if (len > table->maxlen) 154a5494dcdSEric W. Biederman len = table->maxlen; 155a5494dcdSEric W. Biederman if (copy_to_user(oldval, data, len)) 156a5494dcdSEric W. Biederman return -EFAULT; 157a5494dcdSEric W. Biederman if (put_user(len, oldlenp)) 158a5494dcdSEric W. Biederman return -EFAULT; 159a5494dcdSEric W. Biederman } 160a5494dcdSEric W. Biederman } 161a5494dcdSEric W. Biederman 162a5494dcdSEric W. Biederman if (newval && newlen) { 163a5494dcdSEric W. Biederman if (newlen > table->maxlen) 164a5494dcdSEric W. Biederman newlen = table->maxlen; 165a5494dcdSEric W. Biederman 166a5494dcdSEric W. Biederman if (copy_from_user(data, newval, newlen)) 167a5494dcdSEric W. Biederman return -EFAULT; 168a5494dcdSEric W. Biederman } 169a5494dcdSEric W. Biederman return 1; 170a5494dcdSEric W. Biederman } 17191cfb2b4SNadia Derbey 172f221e726SAlexey Dobriyan static int sysctl_ipc_registered_data(ctl_table *table, 173f221e726SAlexey Dobriyan void __user *oldval, size_t __user *oldlenp, 17491cfb2b4SNadia Derbey void __user *newval, size_t newlen) 17591cfb2b4SNadia Derbey { 17691cfb2b4SNadia Derbey int rc; 17791cfb2b4SNadia Derbey 178f221e726SAlexey Dobriyan rc = sysctl_ipc_data(table, oldval, oldlenp, newval, newlen); 17991cfb2b4SNadia Derbey 1809eefe520SNadia Derbey if (newval && newlen && rc > 0) 18191cfb2b4SNadia Derbey /* 1826546bc42SNadia Derbey * Tunable has successfully been changed from userland 18391cfb2b4SNadia Derbey */ 1849eefe520SNadia Derbey unregister_ipcns_notifier(current->nsproxy->ipc_ns); 18591cfb2b4SNadia Derbey 18691cfb2b4SNadia Derbey return rc; 18791cfb2b4SNadia Derbey } 188a5494dcdSEric W. Biederman #else 189a5494dcdSEric W. Biederman #define sysctl_ipc_data NULL 19091cfb2b4SNadia Derbey #define sysctl_ipc_registered_data NULL 191a5494dcdSEric W. Biederman #endif 192a5494dcdSEric W. Biederman 1939eefe520SNadia Derbey static int zero; 1949eefe520SNadia Derbey static int one = 1; 1959eefe520SNadia Derbey 196a5494dcdSEric W. Biederman static struct ctl_table ipc_kern_table[] = { 197a5494dcdSEric W. Biederman { 198a5494dcdSEric W. Biederman .ctl_name = KERN_SHMMAX, 199a5494dcdSEric W. Biederman .procname = "shmmax", 200a5494dcdSEric W. Biederman .data = &init_ipc_ns.shm_ctlmax, 201a5494dcdSEric W. Biederman .maxlen = sizeof (init_ipc_ns.shm_ctlmax), 202a5494dcdSEric W. Biederman .mode = 0644, 203a5494dcdSEric W. Biederman .proc_handler = proc_ipc_doulongvec_minmax, 204a5494dcdSEric W. Biederman .strategy = sysctl_ipc_data, 205a5494dcdSEric W. Biederman }, 206a5494dcdSEric W. Biederman { 207a5494dcdSEric W. Biederman .ctl_name = KERN_SHMALL, 208a5494dcdSEric W. Biederman .procname = "shmall", 209a5494dcdSEric W. Biederman .data = &init_ipc_ns.shm_ctlall, 210a5494dcdSEric W. Biederman .maxlen = sizeof (init_ipc_ns.shm_ctlall), 211a5494dcdSEric W. Biederman .mode = 0644, 212a5494dcdSEric W. Biederman .proc_handler = proc_ipc_doulongvec_minmax, 213a5494dcdSEric W. Biederman .strategy = sysctl_ipc_data, 214a5494dcdSEric W. Biederman }, 215a5494dcdSEric W. Biederman { 216a5494dcdSEric W. Biederman .ctl_name = KERN_SHMMNI, 217a5494dcdSEric W. Biederman .procname = "shmmni", 218a5494dcdSEric W. Biederman .data = &init_ipc_ns.shm_ctlmni, 219a5494dcdSEric W. Biederman .maxlen = sizeof (init_ipc_ns.shm_ctlmni), 220a5494dcdSEric W. Biederman .mode = 0644, 221a5494dcdSEric W. Biederman .proc_handler = proc_ipc_dointvec, 222a5494dcdSEric W. Biederman .strategy = sysctl_ipc_data, 223a5494dcdSEric W. Biederman }, 224a5494dcdSEric W. Biederman { 225a5494dcdSEric W. Biederman .ctl_name = KERN_MSGMAX, 226a5494dcdSEric W. Biederman .procname = "msgmax", 227a5494dcdSEric W. Biederman .data = &init_ipc_ns.msg_ctlmax, 228a5494dcdSEric W. Biederman .maxlen = sizeof (init_ipc_ns.msg_ctlmax), 229a5494dcdSEric W. Biederman .mode = 0644, 230a5494dcdSEric W. Biederman .proc_handler = proc_ipc_dointvec, 231a5494dcdSEric W. Biederman .strategy = sysctl_ipc_data, 232a5494dcdSEric W. Biederman }, 233a5494dcdSEric W. Biederman { 234a5494dcdSEric W. Biederman .ctl_name = KERN_MSGMNI, 235a5494dcdSEric W. Biederman .procname = "msgmni", 236a5494dcdSEric W. Biederman .data = &init_ipc_ns.msg_ctlmni, 237a5494dcdSEric W. Biederman .maxlen = sizeof (init_ipc_ns.msg_ctlmni), 238a5494dcdSEric W. Biederman .mode = 0644, 23991cfb2b4SNadia Derbey .proc_handler = proc_ipc_callback_dointvec, 24091cfb2b4SNadia Derbey .strategy = sysctl_ipc_registered_data, 241a5494dcdSEric W. Biederman }, 242a5494dcdSEric W. Biederman { 243a5494dcdSEric W. Biederman .ctl_name = KERN_MSGMNB, 244a5494dcdSEric W. Biederman .procname = "msgmnb", 245a5494dcdSEric W. Biederman .data = &init_ipc_ns.msg_ctlmnb, 246a5494dcdSEric W. Biederman .maxlen = sizeof (init_ipc_ns.msg_ctlmnb), 247a5494dcdSEric W. Biederman .mode = 0644, 248a5494dcdSEric W. Biederman .proc_handler = proc_ipc_dointvec, 249a5494dcdSEric W. Biederman .strategy = sysctl_ipc_data, 250a5494dcdSEric W. Biederman }, 251a5494dcdSEric W. Biederman { 252a5494dcdSEric W. Biederman .ctl_name = KERN_SEM, 253a5494dcdSEric W. Biederman .procname = "sem", 254a5494dcdSEric W. Biederman .data = &init_ipc_ns.sem_ctls, 255a5494dcdSEric W. Biederman .maxlen = 4*sizeof (int), 256a5494dcdSEric W. Biederman .mode = 0644, 257a5494dcdSEric W. Biederman .proc_handler = proc_ipc_dointvec, 258a5494dcdSEric W. Biederman .strategy = sysctl_ipc_data, 259a5494dcdSEric W. Biederman }, 2609eefe520SNadia Derbey { 2619eefe520SNadia Derbey .ctl_name = CTL_UNNUMBERED, 2629eefe520SNadia Derbey .procname = "auto_msgmni", 2639eefe520SNadia Derbey .data = &init_ipc_ns.auto_msgmni, 2649eefe520SNadia Derbey .maxlen = sizeof(int), 2659eefe520SNadia Derbey .mode = 0644, 2669eefe520SNadia Derbey .proc_handler = proc_ipcauto_dointvec_minmax, 2679eefe520SNadia Derbey .extra1 = &zero, 2689eefe520SNadia Derbey .extra2 = &one, 2699eefe520SNadia Derbey }, 270a5494dcdSEric W. Biederman {} 271a5494dcdSEric W. Biederman }; 272a5494dcdSEric W. Biederman 273a5494dcdSEric W. Biederman static struct ctl_table ipc_root_table[] = { 274a5494dcdSEric W. Biederman { 275a5494dcdSEric W. Biederman .ctl_name = CTL_KERN, 276a5494dcdSEric W. Biederman .procname = "kernel", 277a5494dcdSEric W. Biederman .mode = 0555, 278a5494dcdSEric W. Biederman .child = ipc_kern_table, 279a5494dcdSEric W. Biederman }, 280a5494dcdSEric W. Biederman {} 281a5494dcdSEric W. Biederman }; 282a5494dcdSEric W. Biederman 283a5494dcdSEric W. Biederman static int __init ipc_sysctl_init(void) 284a5494dcdSEric W. Biederman { 2850b4d4147SEric W. Biederman register_sysctl_table(ipc_root_table); 286a5494dcdSEric W. Biederman return 0; 287a5494dcdSEric W. Biederman } 288a5494dcdSEric W. Biederman 289a5494dcdSEric W. Biederman __initcall(ipc_sysctl_init); 290