| // 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); |