mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	module_layout manages different types of memory (text, data, rodata, etc.)
in one allocation, which is problematic for some reasons:
1. It is hard to enable CONFIG_STRICT_MODULE_RWX.
2. It is hard to use huge pages in modules (and not break strict rwx).
3. Many archs uses module_layout for arch-specific data, but it is not
   obvious how these data are used (are they RO, RX, or RW?)
Improve the scenario by replacing 2 (or 3) module_layout per module with
up to 7 module_memory per module:
        MOD_TEXT,
        MOD_DATA,
        MOD_RODATA,
        MOD_RO_AFTER_INIT,
        MOD_INIT_TEXT,
        MOD_INIT_DATA,
        MOD_INIT_RODATA,
and allocating them separately. This adds slightly more entries to
mod_tree (from up to 3 entries per module, to up to 7 entries per
module). However, this at most adds a small constant overhead to
__module_address(), which is expected to be fast.
Various archs use module_layout for different data. These data are put
into different module_memory based on their location in module_layout.
IOW, data that used to go with text is allocated with MOD_MEM_TYPE_TEXT;
data that used to go with data is allocated with MOD_MEM_TYPE_DATA, etc.
module_memory simplifies quite some of the module code. For example,
ARCH_WANTS_MODULES_DATA_IN_VMALLOC is a lot cleaner, as it just uses a
different allocator for the data. kernel/module/strict_rwx.c is also
much cleaner with module_memory.
Signed-off-by: Song Liu <song@kernel.org>
Cc: Luis Chamberlain <mcgrof@kernel.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Guenter Roeck <linux@roeck-us.net>
Cc: Christophe Leroy <christophe.leroy@csgroup.eu>
Reviewed-by: Thomas Gleixner <tglx@linutronix.de>
Reviewed-by: Christophe Leroy <christophe.leroy@csgroup.eu>
Reviewed-by: Luis Chamberlain <mcgrof@kernel.org>
Signed-off-by: Luis Chamberlain <mcgrof@kernel.org>
		
	
			
		
			
				
	
	
		
			152 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			152 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
						|
/*
 | 
						|
 * Module proc support
 | 
						|
 *
 | 
						|
 * Copyright (C) 2008 Alexey Dobriyan
 | 
						|
 */
 | 
						|
 | 
						|
#include <linux/module.h>
 | 
						|
#include <linux/kallsyms.h>
 | 
						|
#include <linux/mutex.h>
 | 
						|
#include <linux/seq_file.h>
 | 
						|
#include <linux/proc_fs.h>
 | 
						|
#include "internal.h"
 | 
						|
 | 
						|
#ifdef CONFIG_MODULE_UNLOAD
 | 
						|
static inline void print_unload_info(struct seq_file *m, struct module *mod)
 | 
						|
{
 | 
						|
	struct module_use *use;
 | 
						|
	int printed_something = 0;
 | 
						|
 | 
						|
	seq_printf(m, " %i ", module_refcount(mod));
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Always include a trailing , so userspace can differentiate
 | 
						|
	 * between this and the old multi-field proc format.
 | 
						|
	 */
 | 
						|
	list_for_each_entry(use, &mod->source_list, source_list) {
 | 
						|
		printed_something = 1;
 | 
						|
		seq_printf(m, "%s,", use->source->name);
 | 
						|
	}
 | 
						|
 | 
						|
	if (mod->init && !mod->exit) {
 | 
						|
		printed_something = 1;
 | 
						|
		seq_puts(m, "[permanent],");
 | 
						|
	}
 | 
						|
 | 
						|
	if (!printed_something)
 | 
						|
		seq_puts(m, "-");
 | 
						|
}
 | 
						|
#else /* !CONFIG_MODULE_UNLOAD */
 | 
						|
static inline void print_unload_info(struct seq_file *m, struct module *mod)
 | 
						|
{
 | 
						|
	/* We don't know the usage count, or what modules are using. */
 | 
						|
	seq_puts(m, " - -");
 | 
						|
}
 | 
						|
#endif /* CONFIG_MODULE_UNLOAD */
 | 
						|
 | 
						|
/* Called by the /proc file system to return a list of modules. */
 | 
						|
static void *m_start(struct seq_file *m, loff_t *pos)
 | 
						|
{
 | 
						|
	mutex_lock(&module_mutex);
 | 
						|
	return seq_list_start(&modules, *pos);
 | 
						|
}
 | 
						|
 | 
						|
static void *m_next(struct seq_file *m, void *p, loff_t *pos)
 | 
						|
{
 | 
						|
	return seq_list_next(p, &modules, pos);
 | 
						|
}
 | 
						|
 | 
						|
static void m_stop(struct seq_file *m, void *p)
 | 
						|
{
 | 
						|
	mutex_unlock(&module_mutex);
 | 
						|
}
 | 
						|
 | 
						|
static unsigned int module_total_size(struct module *mod)
 | 
						|
{
 | 
						|
	int size = 0;
 | 
						|
 | 
						|
	for_each_mod_mem_type(type)
 | 
						|
		size += mod->mem[type].size;
 | 
						|
	return size;
 | 
						|
}
 | 
						|
 | 
						|
static int m_show(struct seq_file *m, void *p)
 | 
						|
{
 | 
						|
	struct module *mod = list_entry(p, struct module, list);
 | 
						|
	char buf[MODULE_FLAGS_BUF_SIZE];
 | 
						|
	void *value;
 | 
						|
	unsigned int size;
 | 
						|
 | 
						|
	/* We always ignore unformed modules. */
 | 
						|
	if (mod->state == MODULE_STATE_UNFORMED)
 | 
						|
		return 0;
 | 
						|
 | 
						|
	size = module_total_size(mod);
 | 
						|
	seq_printf(m, "%s %u", mod->name, size);
 | 
						|
	print_unload_info(m, mod);
 | 
						|
 | 
						|
	/* Informative for users. */
 | 
						|
	seq_printf(m, " %s",
 | 
						|
		   mod->state == MODULE_STATE_GOING ? "Unloading" :
 | 
						|
		   mod->state == MODULE_STATE_COMING ? "Loading" :
 | 
						|
		   "Live");
 | 
						|
	/* Used by oprofile and other similar tools. */
 | 
						|
	value = m->private ? NULL : mod->mem[MOD_TEXT].base;
 | 
						|
	seq_printf(m, " 0x%px", value);
 | 
						|
 | 
						|
	/* Taints info */
 | 
						|
	if (mod->taints)
 | 
						|
		seq_printf(m, " %s", module_flags(mod, buf, true));
 | 
						|
 | 
						|
	seq_puts(m, "\n");
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Format: modulename size refcount deps address
 | 
						|
 *
 | 
						|
 * Where refcount is a number or -, and deps is a comma-separated list
 | 
						|
 * of depends or -.
 | 
						|
 */
 | 
						|
static const struct seq_operations modules_op = {
 | 
						|
	.start	= m_start,
 | 
						|
	.next	= m_next,
 | 
						|
	.stop	= m_stop,
 | 
						|
	.show	= m_show
 | 
						|
};
 | 
						|
 | 
						|
/*
 | 
						|
 * This also sets the "private" pointer to non-NULL if the
 | 
						|
 * kernel pointers should be hidden (so you can just test
 | 
						|
 * "m->private" to see if you should keep the values private).
 | 
						|
 *
 | 
						|
 * We use the same logic as for /proc/kallsyms.
 | 
						|
 */
 | 
						|
static int modules_open(struct inode *inode, struct file *file)
 | 
						|
{
 | 
						|
	int err = seq_open(file, &modules_op);
 | 
						|
 | 
						|
	if (!err) {
 | 
						|
		struct seq_file *m = file->private_data;
 | 
						|
 | 
						|
		m->private = kallsyms_show_value(file->f_cred) ? NULL : (void *)8ul;
 | 
						|
	}
 | 
						|
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static const struct proc_ops modules_proc_ops = {
 | 
						|
	.proc_flags	= PROC_ENTRY_PERMANENT,
 | 
						|
	.proc_open	= modules_open,
 | 
						|
	.proc_read	= seq_read,
 | 
						|
	.proc_lseek	= seq_lseek,
 | 
						|
	.proc_release	= seq_release,
 | 
						|
};
 | 
						|
 | 
						|
static int __init proc_modules_init(void)
 | 
						|
{
 | 
						|
	proc_create("modules", 0, NULL, &modules_proc_ops);
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
module_init(proc_modules_init);
 |