aboutsummaryrefslogtreecommitdiffstats
path: root/mm
diff options
context:
space:
mode:
authorVasiliy Kulikov <segoon@openwall.com>2011-09-27 13:54:53 -0400
committerPekka Enberg <penberg@kernel.org>2011-09-27 15:59:27 -0400
commitab067e99d22ec78ff646de1283348729d1aa66d4 (patch)
tree813af760ebc66398cb17cfe8652ae8e851eddbfd /mm
parentd20bbfab01802e195a50435940f7e4aa747c217c (diff)
mm: restrict access to slab files under procfs and sysfs
Historically /proc/slabinfo and files under /sys/kernel/slab/* have world read permissions and are accessible to the world. slabinfo contains rather private information related both to the kernel and userspace tasks. Depending on the situation, it might reveal either private information per se or information useful to make another targeted attack. Some examples of what can be learned by reading/watching for /proc/slabinfo entries: 1) dentry (and different *inode*) number might reveal other processes fs activity. The number of dentry "active objects" doesn't strictly show file count opened/touched by a process, however, there is a good correlation between them. The patch "proc: force dcache drop on unauthorized access" relies on the privacy of dentry count. 2) different inode entries might reveal the same information as (1), but these are more fine granted counters. If a filesystem is mounted in a private mount point (or even a private namespace) and fs type differs from other mounted fs types, fs activity in this mount point/namespace is revealed. If there is a single ecryptfs mount point, the whole fs activity of a single user is revealed. Number of files in ecryptfs mount point is a private information per se. 3) fuse_* reveals number of files / fs activity of a user in a user private mount point. It is approx. the same severity as ecryptfs infoleak in (2). 4) sysfs_dir_cache similar to (2) reveals devices' addition/removal, which can be otherwise hidden by "chmod 0700 /sys/". With 0444 slabinfo the precise number of sysfs files is known to the world. 5) buffer_head might reveal some kernel activity. With other information leaks an attacker might identify what specific kernel routines generate buffer_head activity. 6) *kmalloc* infoleaks are very situational. Attacker should watch for the specific kmalloc size entry and filter the noise related to the unrelated kernel activity. If an attacker has relatively silent victim system, he might get rather precise counters. Additional information sources might significantly increase the slabinfo infoleak benefits. E.g. if an attacker knows that the processes activity on the system is very low (only core daemons like syslog and cron), he may run setxid binaries / trigger local daemon activity / trigger network services activity / await sporadic cron jobs activity / etc. and get rather precise counters for fs and network activity of these privileged tasks, which is unknown otherwise. Also hiding slabinfo and /sys/kernel/slab/* is a one step to complicate exploitation of kernel heap overflows (and possibly, other bugs). The related discussion: http://thread.gmane.org/gmane.linux.kernel/1108378 To keep compatibility with old permission model where non-root monitoring daemon could watch for kernel memleaks though slabinfo one should do: groupadd slabinfo usermod -a -G slabinfo $MONITOR_USER And add the following commands to init scripts (to mountall.conf in Ubuntu's upstart case): chmod g+r /proc/slabinfo /sys/kernel/slab/*/* chgrp slabinfo /proc/slabinfo /sys/kernel/slab/*/* Signed-off-by: Vasiliy Kulikov <segoon@openwall.com> Reviewed-by: Kees Cook <kees@ubuntu.com> Reviewed-by: Dave Hansen <dave@linux.vnet.ibm.com> Acked-by: Christoph Lameter <cl@gentwo.org> Acked-by: David Rientjes <rientjes@google.com> CC: Valdis.Kletnieks@vt.edu CC: Linus Torvalds <torvalds@linux-foundation.org> CC: Alan Cox <alan@linux.intel.com> Signed-off-by: Pekka Enberg <penberg@kernel.org>
Diffstat (limited to 'mm')
-rw-r--r--mm/slab.c2
-rw-r--r--mm/slub.c7
2 files changed, 5 insertions, 4 deletions
diff --git a/mm/slab.c b/mm/slab.c
index 5bfc2047afe1..708efe886154 100644
--- a/mm/slab.c
+++ b/mm/slab.c
@@ -4579,7 +4579,7 @@ static const struct file_operations proc_slabstats_operations = {
4579 4579
4580static int __init slab_proc_init(void) 4580static int __init slab_proc_init(void)
4581{ 4581{
4582 proc_create("slabinfo",S_IWUSR|S_IRUGO,NULL,&proc_slabinfo_operations); 4582 proc_create("slabinfo",S_IWUSR|S_IRUSR,NULL,&proc_slabinfo_operations);
4583#ifdef CONFIG_DEBUG_SLAB_LEAK 4583#ifdef CONFIG_DEBUG_SLAB_LEAK
4584 proc_create("slab_allocators", 0, NULL, &proc_slabstats_operations); 4584 proc_create("slab_allocators", 0, NULL, &proc_slabstats_operations);
4585#endif 4585#endif
diff --git a/mm/slub.c b/mm/slub.c
index 3b3f17bc0d17..943f4906131b 100644
--- a/mm/slub.c
+++ b/mm/slub.c
@@ -4386,11 +4386,12 @@ struct slab_attribute {
4386}; 4386};
4387 4387
4388#define SLAB_ATTR_RO(_name) \ 4388#define SLAB_ATTR_RO(_name) \
4389 static struct slab_attribute _name##_attr = __ATTR_RO(_name) 4389 static struct slab_attribute _name##_attr = \
4390 __ATTR(_name, 0400, _name##_show, NULL)
4390 4391
4391#define SLAB_ATTR(_name) \ 4392#define SLAB_ATTR(_name) \
4392 static struct slab_attribute _name##_attr = \ 4393 static struct slab_attribute _name##_attr = \
4393 __ATTR(_name, 0644, _name##_show, _name##_store) 4394 __ATTR(_name, 0600, _name##_show, _name##_store)
4394 4395
4395static ssize_t slab_size_show(struct kmem_cache *s, char *buf) 4396static ssize_t slab_size_show(struct kmem_cache *s, char *buf)
4396{ 4397{
@@ -5231,7 +5232,7 @@ static const struct file_operations proc_slabinfo_operations = {
5231 5232
5232static int __init slab_proc_init(void) 5233static int __init slab_proc_init(void)
5233{ 5234{
5234 proc_create("slabinfo", S_IRUGO, NULL, &proc_slabinfo_operations); 5235 proc_create("slabinfo", S_IRUSR, NULL, &proc_slabinfo_operations);
5235 return 0; 5236 return 0;
5236} 5237}
5237module_init(slab_proc_init); 5238module_init(slab_proc_init);