root/kernel/module/procfs.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);