aboutsummaryrefslogtreecommitdiffstats
path: root/kernel
diff options
context:
space:
mode:
Diffstat (limited to 'kernel')
-rw-r--r--kernel/module.c91
1 files changed, 59 insertions, 32 deletions
diff --git a/kernel/module.c b/kernel/module.c
index f99558e1945a..8c6b42840dd1 100644
--- a/kernel/module.c
+++ b/kernel/module.c
@@ -582,33 +582,26 @@ static int add_module_usage(struct module *a, struct module *b)
582} 582}
583 583
584/* Module a uses b: caller needs module_mutex() */ 584/* Module a uses b: caller needs module_mutex() */
585int use_module(struct module *a, struct module *b) 585int ref_module(struct module *a, struct module *b)
586{ 586{
587 int err; 587 int err;
588 588
589 if (b == NULL || already_uses(a, b)) return 1; 589 if (b == NULL || already_uses(a, b))
590
591 /* If we're interrupted or time out, we fail. */
592 if (wait_event_interruptible_timeout(
593 module_wq, (err = strong_try_module_get(b)) != -EBUSY,
594 30 * HZ) <= 0) {
595 printk("%s: gave up waiting for init of module %s.\n",
596 a->name, b->name);
597 return 0; 590 return 0;
598 }
599 591
600 /* If strong_try_module_get() returned a different error, we fail. */ 592 /* If module isn't available, we fail. */
593 err = strong_try_module_get(b);
601 if (err) 594 if (err)
602 return 0; 595 return err;
603 596
604 err = add_module_usage(a, b); 597 err = add_module_usage(a, b);
605 if (err) { 598 if (err) {
606 module_put(b); 599 module_put(b);
607 return 0; 600 return err;
608 } 601 }
609 return 1; 602 return 0;
610} 603}
611EXPORT_SYMBOL_GPL(use_module); 604EXPORT_SYMBOL_GPL(ref_module);
612 605
613/* Clear the unload stuff of the module. */ 606/* Clear the unload stuff of the module. */
614static void module_unload_free(struct module *mod) 607static void module_unload_free(struct module *mod)
@@ -893,11 +886,11 @@ static inline void module_unload_free(struct module *mod)
893{ 886{
894} 887}
895 888
896int use_module(struct module *a, struct module *b) 889int ref_module(struct module *a, struct module *b)
897{ 890{
898 return strong_try_module_get(b) == 0; 891 return strong_try_module_get(b);
899} 892}
900EXPORT_SYMBOL_GPL(use_module); 893EXPORT_SYMBOL_GPL(ref_module);
901 894
902static inline void module_unload_init(struct module *mod) 895static inline void module_unload_init(struct module *mod)
903{ 896{
@@ -1062,26 +1055,58 @@ static inline int same_magic(const char *amagic, const char *bmagic,
1062static const struct kernel_symbol *resolve_symbol(Elf_Shdr *sechdrs, 1055static const struct kernel_symbol *resolve_symbol(Elf_Shdr *sechdrs,
1063 unsigned int versindex, 1056 unsigned int versindex,
1064 const char *name, 1057 const char *name,
1065 struct module *mod) 1058 struct module *mod,
1059 char ownername[])
1066{ 1060{
1067 struct module *owner; 1061 struct module *owner;
1068 const struct kernel_symbol *sym; 1062 const struct kernel_symbol *sym;
1069 const unsigned long *crc; 1063 const unsigned long *crc;
1064 int err;
1070 1065
1071 mutex_lock(&module_mutex); 1066 mutex_lock(&module_mutex);
1072 sym = find_symbol(name, &owner, &crc, 1067 sym = find_symbol(name, &owner, &crc,
1073 !(mod->taints & (1 << TAINT_PROPRIETARY_MODULE)), true); 1068 !(mod->taints & (1 << TAINT_PROPRIETARY_MODULE)), true);
1074 /* use_module can fail due to OOM, 1069 if (!sym)
1075 or module initialization or unloading */ 1070 goto unlock;
1076 if (sym) { 1071
1077 if (!check_version(sechdrs, versindex, name, mod, crc, owner) 1072 if (!check_version(sechdrs, versindex, name, mod, crc, owner)) {
1078 || !use_module(mod, owner)) 1073 sym = ERR_PTR(-EINVAL);
1079 sym = NULL; 1074 goto getname;
1080 } 1075 }
1076
1077 err = ref_module(mod, owner);
1078 if (err) {
1079 sym = ERR_PTR(err);
1080 goto getname;
1081 }
1082
1083getname:
1084 /* We must make copy under the lock if we failed to get ref. */
1085 strncpy(ownername, module_name(owner), MODULE_NAME_LEN);
1086unlock:
1081 mutex_unlock(&module_mutex); 1087 mutex_unlock(&module_mutex);
1082 return sym; 1088 return sym;
1083} 1089}
1084 1090
1091static const struct kernel_symbol *resolve_symbol_wait(Elf_Shdr *sechdrs,
1092 unsigned int versindex,
1093 const char *name,
1094 struct module *mod)
1095{
1096 const struct kernel_symbol *ksym;
1097 char ownername[MODULE_NAME_LEN];
1098
1099 if (wait_event_interruptible_timeout(module_wq,
1100 !IS_ERR(ksym = resolve_symbol(sechdrs, versindex, name,
1101 mod, ownername)) ||
1102 PTR_ERR(ksym) != -EBUSY,
1103 30 * HZ) <= 0) {
1104 printk(KERN_WARNING "%s: gave up waiting for init of module %s.\n",
1105 mod->name, ownername);
1106 }
1107 return ksym;
1108}
1109
1085/* 1110/*
1086 * /sys/module/foo/sections stuff 1111 * /sys/module/foo/sections stuff
1087 * J. Corbet <corbet@lwn.net> 1112 * J. Corbet <corbet@lwn.net>
@@ -1638,21 +1663,23 @@ static int simplify_symbols(Elf_Shdr *sechdrs,
1638 break; 1663 break;
1639 1664
1640 case SHN_UNDEF: 1665 case SHN_UNDEF:
1641 ksym = resolve_symbol(sechdrs, versindex, 1666 ksym = resolve_symbol_wait(sechdrs, versindex,
1642 strtab + sym[i].st_name, mod); 1667 strtab + sym[i].st_name,
1668 mod);
1643 /* Ok if resolved. */ 1669 /* Ok if resolved. */
1644 if (ksym) { 1670 if (ksym && !IS_ERR(ksym)) {
1645 sym[i].st_value = ksym->value; 1671 sym[i].st_value = ksym->value;
1646 break; 1672 break;
1647 } 1673 }
1648 1674
1649 /* Ok if weak. */ 1675 /* Ok if weak. */
1650 if (ELF_ST_BIND(sym[i].st_info) == STB_WEAK) 1676 if (!ksym && ELF_ST_BIND(sym[i].st_info) == STB_WEAK)
1651 break; 1677 break;
1652 1678
1653 printk(KERN_WARNING "%s: Unknown symbol %s\n", 1679 printk(KERN_WARNING "%s: Unknown symbol %s (err %li)\n",
1654 mod->name, strtab + sym[i].st_name); 1680 mod->name, strtab + sym[i].st_name,
1655 ret = -ENOENT; 1681 PTR_ERR(ksym));
1682 ret = PTR_ERR(ksym) ?: -ENOENT;
1656 break; 1683 break;
1657 1684
1658 default: 1685 default: