diff options
-rw-r--r-- | ipc/ipc_sysctl.c | 23 | ||||
-rw-r--r-- | ipc/util.h | 9 |
2 files changed, 31 insertions, 1 deletions
diff --git a/ipc/ipc_sysctl.c b/ipc/ipc_sysctl.c index f87cb29436ef..49f9bf4ffc7f 100644 --- a/ipc/ipc_sysctl.c +++ b/ipc/ipc_sysctl.c | |||
@@ -88,12 +88,33 @@ static int proc_ipc_auto_msgmni(struct ctl_table *table, int write, | |||
88 | return proc_dointvec_minmax(&ipc_table, write, buffer, lenp, ppos); | 88 | return proc_dointvec_minmax(&ipc_table, write, buffer, lenp, ppos); |
89 | } | 89 | } |
90 | 90 | ||
91 | static int proc_ipc_sem_dointvec(struct ctl_table *table, int write, | ||
92 | void __user *buffer, size_t *lenp, loff_t *ppos) | ||
93 | { | ||
94 | int ret, semmni; | ||
95 | struct ipc_namespace *ns = current->nsproxy->ipc_ns; | ||
96 | |||
97 | semmni = ns->sem_ctls[3]; | ||
98 | ret = proc_ipc_dointvec(table, write, buffer, lenp, ppos); | ||
99 | |||
100 | if (!ret) | ||
101 | ret = sem_check_semmni(current->nsproxy->ipc_ns); | ||
102 | |||
103 | /* | ||
104 | * Reset the semmni value if an error happens. | ||
105 | */ | ||
106 | if (ret) | ||
107 | ns->sem_ctls[3] = semmni; | ||
108 | return ret; | ||
109 | } | ||
110 | |||
91 | #else | 111 | #else |
92 | #define proc_ipc_doulongvec_minmax NULL | 112 | #define proc_ipc_doulongvec_minmax NULL |
93 | #define proc_ipc_dointvec NULL | 113 | #define proc_ipc_dointvec NULL |
94 | #define proc_ipc_dointvec_minmax NULL | 114 | #define proc_ipc_dointvec_minmax NULL |
95 | #define proc_ipc_dointvec_minmax_orphans NULL | 115 | #define proc_ipc_dointvec_minmax_orphans NULL |
96 | #define proc_ipc_auto_msgmni NULL | 116 | #define proc_ipc_auto_msgmni NULL |
117 | #define proc_ipc_sem_dointvec NULL | ||
97 | #endif | 118 | #endif |
98 | 119 | ||
99 | static int zero; | 120 | static int zero; |
@@ -175,7 +196,7 @@ static struct ctl_table ipc_kern_table[] = { | |||
175 | .data = &init_ipc_ns.sem_ctls, | 196 | .data = &init_ipc_ns.sem_ctls, |
176 | .maxlen = 4*sizeof(int), | 197 | .maxlen = 4*sizeof(int), |
177 | .mode = 0644, | 198 | .mode = 0644, |
178 | .proc_handler = proc_ipc_dointvec, | 199 | .proc_handler = proc_ipc_sem_dointvec, |
179 | }, | 200 | }, |
180 | #ifdef CONFIG_CHECKPOINT_RESTORE | 201 | #ifdef CONFIG_CHECKPOINT_RESTORE |
181 | { | 202 | { |
diff --git a/ipc/util.h b/ipc/util.h index 1ee81bce25e9..d768fdbed515 100644 --- a/ipc/util.h +++ b/ipc/util.h | |||
@@ -217,6 +217,15 @@ int ipcget(struct ipc_namespace *ns, struct ipc_ids *ids, | |||
217 | void free_ipcs(struct ipc_namespace *ns, struct ipc_ids *ids, | 217 | void free_ipcs(struct ipc_namespace *ns, struct ipc_ids *ids, |
218 | void (*free)(struct ipc_namespace *, struct kern_ipc_perm *)); | 218 | void (*free)(struct ipc_namespace *, struct kern_ipc_perm *)); |
219 | 219 | ||
220 | static inline int sem_check_semmni(struct ipc_namespace *ns) { | ||
221 | /* | ||
222 | * Check semmni range [0, IPCMNI] | ||
223 | * semmni is the last element of sem_ctls[4] array | ||
224 | */ | ||
225 | return ((ns->sem_ctls[3] < 0) || (ns->sem_ctls[3] > IPCMNI)) | ||
226 | ? -ERANGE : 0; | ||
227 | } | ||
228 | |||
220 | #ifdef CONFIG_COMPAT | 229 | #ifdef CONFIG_COMPAT |
221 | #include <linux/compat.h> | 230 | #include <linux/compat.h> |
222 | struct compat_ipc_perm { | 231 | struct compat_ipc_perm { |