From: LOLi Date: Mon, 24 Jul 2017 17:52:53 +0000 (+0200) Subject: Avoid WARN() from procfs on kstat collision X-Git-Tag: spl-0.7.12~52 X-Git-Url: https://git.proxmox.com/?a=commitdiff_plain;ds=inline;h=cd47801828c760efebbbb1fad7f7899369d8bce7;p=mirror_spl.git Avoid WARN() from procfs on kstat collision When we load a ZFS pool having spa_name equals to some existing kstat we would have to create a duplicate entry, which procfs doesn't like. For instance a ZFS pool named "zil" would have its kstat "txgs" (module "zfs/zil") intalled under "/proc/spl/kstat/zfs/zil": unfortunately we already have a kstat named "zil" (module "zfs") installed in the same procfs location. Avoid this issue by skipping the duplicate entry creation in procfs. Reviewed-by: Brian Behlendorf Signed-off-by: loli10K Closes #628 --- diff --git a/module/spl/spl-kstat.c b/module/spl/spl-kstat.c index e8917a3..1b6a7df 100644 --- a/module/spl/spl-kstat.c +++ b/module/spl/spl-kstat.c @@ -27,6 +27,7 @@ #include #include #include +#include #ifndef HAVE_PDE_DATA #define PDE_DATA(x) (PDE(x)->data) @@ -608,6 +609,29 @@ __kstat_create(const char *ks_module, int ks_instance, const char *ks_name, } EXPORT_SYMBOL(__kstat_create); +static int +kstat_detect_collision(kstat_t *ksp) +{ + kstat_module_t *module; + kstat_t *tmp; + char parent[KSTAT_STRLEN+1]; + char *cp; + + (void) strlcpy(parent, ksp->ks_module, sizeof(parent)); + + if ((cp = strrchr(parent, '/')) == NULL) + return (0); + + cp[0] = '\0'; + if ((module = kstat_find_module(parent)) != NULL) { + list_for_each_entry(tmp, &module->ksm_kstat_list, ks_list) + if (strncmp(tmp->ks_name, cp+1, KSTAT_STRLEN) == 0) + return (EEXIST); + } + + return (0); +} + void __kstat_install(kstat_t *ksp) { @@ -620,6 +644,11 @@ __kstat_install(kstat_t *ksp) module = kstat_find_module(ksp->ks_module); if (module == NULL) { + if (kstat_detect_collision(ksp) != 0) { + cmn_err(CE_WARN, "kstat_create('%s', '%s'): namespace" \ + " collision", ksp->ks_module, ksp->ks_name); + goto out; + } module = kstat_create_module(ksp->ks_module); if (module == NULL) goto out;