kmod_module_probe_insert_module() is supposed to return 0 for builtin
modules, but only when libkmod can locate the modules.builtin index. If
the index is missing (e.g. a kernel built with the driver as builtin
but installed without running modules_install), libkmod falls through
to the real init_module() syscall and returns an error such as -ENOENT,
producing a spurious "insert failure" even though the driver is already
part of the running kernel.

Add a helper util_kmod_skip_probe_insert() that returns true when the
module state is KMOD_MODULE_BUILTIN or KMOD_MODULE_LIVE. As an
additional heuristic, treat KMOD_MODULE_COMING as builtin when
/sys/module/<name>/ exists but the initstate file does not - this is
the exact pattern libkmod's sysfs fallback emits for builtin drivers
when the modules.builtin index is unavailable. The pattern mirrors the
KMOD_MODULE_LIVE / KMOD_MODULE_BUILTIN check already used by ndctl's
own test/core.c (see test/core.c:218-236).

The helper also returns the observed libkmod state via an out parameter
so daxctl_insert_kmod_for_mode() can distinguish LIVE (retain the kmod
reference in dev->module) from BUILTIN (drop it, since builtin drivers
cannot be unloaded) without re-reading /sys/module/<name>/initstate.
__util_bind() passes NULL since it does not need the state.

Reported-by: Jonathan Cameron <[email protected]>
Suggested-by: Alison Schofield <[email protected]>
Signed-off-by: Chen Pei <[email protected]>
---
 daxctl/lib/libdaxctl.c | 22 +++++++++++++++++++--
 util/sysfs.c           | 44 +++++++++++++++++++++++++++++++++++++++++-
 util/sysfs.h           | 16 +++++++++++++++
 3 files changed, 79 insertions(+), 3 deletions(-)

diff --git a/daxctl/lib/libdaxctl.c b/daxctl/lib/libdaxctl.c
index ffc81eb..1596dc0 100644
--- a/daxctl/lib/libdaxctl.c
+++ b/daxctl/lib/libdaxctl.c
@@ -910,7 +910,7 @@ static int daxctl_insert_kmod_for_mode(struct daxctl_dev 
*dev,
        const char *devname = daxctl_dev_get_devname(dev);
        struct daxctl_ctx *ctx = daxctl_dev_get_ctx(dev);
        struct kmod_module *kmod;
-       int rc;
+       int state, rc;
 
        rc = kmod_module_new_from_name(ctx->kmod_ctx, mod_name, &kmod);
        if (rc < 0) {
@@ -919,7 +919,25 @@ static int daxctl_insert_kmod_for_mode(struct daxctl_dev 
*dev,
                return rc;
        }
 
-       /* if the driver is builtin, this Just Works */
+       /*
+        * If the driver is builtin or already live, skip probe-insert.
+        * For live modules retain the local reference in dev->module so
+        * the module can be unreffed alongside the device; for builtin
+        * drivers drop it because builtin modules cannot be unloaded.
+        */
+       if (util_kmod_skip_probe_insert(kmod, ctx, &state)) {
+               if (state == KMOD_MODULE_LIVE) {
+                       dbg(ctx, "%s: module %s already loaded\n", devname,
+                               kmod_module_get_name(kmod));
+                       dev->module = kmod;
+               } else {
+                       dbg(ctx, "%s: module %s is builtin\n", devname,
+                               kmod_module_get_name(kmod));
+                       kmod_module_unref(kmod);
+               }
+               return 0;
+       }
+
        dbg(ctx, "%s inserting module: %s\n", devname,
                kmod_module_get_name(kmod));
        rc = kmod_module_probe_insert_module(kmod,
diff --git a/util/sysfs.c b/util/sysfs.c
index e027e38..eaf4b60 100644
--- a/util/sysfs.c
+++ b/util/sysfs.c
@@ -6,6 +6,7 @@
 #include <stdarg.h>
 #include <unistd.h>
 #include <errno.h>
+#include <limits.h>
 #include <string.h>
 #include <ctype.h>
 #include <fcntl.h>
@@ -168,6 +169,47 @@ struct kmod_module *__util_modalias_to_module(struct 
kmod_ctx *kmod_ctx,
        return mod;
 }
 
+bool __util_kmod_skip_probe_insert(struct kmod_module *module,
+                                  struct log_ctx *ctx, int *state_out)
+{
+       const char *name = kmod_module_get_name(module);
+       int state = kmod_module_get_initstate(module);
+       char path[PATH_MAX];
+       struct stat st;
+
+       if (state_out)
+               *state_out = state;
+
+       if (state == KMOD_MODULE_BUILTIN || state == KMOD_MODULE_LIVE)
+               return true;
+
+       /*
+        * When modules.builtin is missing (e.g. a kernel installed
+        * without modules_install), libkmod's sysfs fallback returns
+        * KMOD_MODULE_COMING for builtin drivers because /sys/module/<name>/
+        * exists but the initstate file does not. Treat that pattern as
+        * builtin to avoid a spurious "insert failure" message.
+        */
+       if (state != KMOD_MODULE_COMING)
+               return false;
+
+       if (snprintf(path, sizeof(path), "/sys/module/%s/initstate", name)
+                       >= (int)sizeof(path))
+               return false;
+       if (stat(path, &st) == 0 || errno != ENOENT)
+               return false;
+
+       if (snprintf(path, sizeof(path), "/sys/module/%s", name)
+                       >= (int)sizeof(path))
+               return false;
+       if (stat(path, &st) != 0 || !S_ISDIR(st.st_mode))
+               return false;
+
+       log_dbg(ctx, "module %s appears builtin (no modules.builtin index)\n",
+                       name);
+       return true;
+}
+
 int __util_bind(const char *devname, struct kmod_module *module,
                const char *bus, struct log_ctx *ctx)
 {
@@ -182,7 +224,7 @@ int __util_bind(const char *devname, struct kmod_module 
*module,
                return -EINVAL;
        }
 
-       if (module) {
+       if (module && !__util_kmod_skip_probe_insert(module, ctx, NULL)) {
                rc = kmod_module_probe_insert_module(module,
                                                     KMOD_PROBE_APPLY_BLACKLIST,
                                                     NULL, NULL, NULL, NULL);
diff --git a/util/sysfs.h b/util/sysfs.h
index 4c95c70..e4f6115 100644
--- a/util/sysfs.h
+++ b/util/sysfs.h
@@ -3,6 +3,7 @@
 #ifndef __UTIL_SYSFS_H__
 #define __UTIL_SYSFS_H__
 
+#include <stdbool.h>
 #include <string.h>
 
 typedef void *(*add_dev_fn)(void *parent, int id, const char *dev_path);
@@ -36,6 +37,21 @@ struct kmod_module *__util_modalias_to_module(struct 
kmod_ctx *kmod_ctx,
 #define util_modalias_to_module(ctx, buf)                                      
\
        __util_modalias_to_module((ctx)->kmod_ctx, buf, &(ctx)->ctx)
 
+/*
+ * __util_kmod_skip_probe_insert - true when kmod_module_probe_insert_module()
+ * should be skipped because @module is already part of the running kernel:
+ * KMOD_MODULE_BUILTIN, KMOD_MODULE_LIVE, or KMOD_MODULE_COMING with
+ * /sys/module/<name>/ existing but no initstate file (the fingerprint
+ * libkmod's sysfs fallback emits for builtin drivers when the
+ * modules.builtin index is missing). If @state_out is non-NULL, the
+ * libkmod state actually observed is stored there so callers can avoid
+ * an extra kmod_module_get_initstate() call.
+ */
+bool __util_kmod_skip_probe_insert(struct kmod_module *module,
+                                  struct log_ctx *ctx, int *state_out);
+#define util_kmod_skip_probe_insert(m, c, s)                                   
\
+       __util_kmod_skip_probe_insert((m), &(c)->ctx, (s))
+
 int __util_bind(const char *devname, struct kmod_module *module, const char 
*bus,
              struct log_ctx *ctx);
 #define util_bind(n, m, b, c) __util_bind(n, m, b, &(c)->ctx)
-- 
2.50.1


Reply via email to