2019-09-06 11:32:33 +01:00
|
|
|
=================
|
|
|
|
Symbol Namespaces
|
|
|
|
=================
|
|
|
|
|
|
|
|
The following document describes how to use Symbol Namespaces to structure the
|
|
|
|
export surface of in-kernel symbols exported through the family of
|
|
|
|
EXPORT_SYMBOL() macros.
|
|
|
|
|
2025-05-22 16:17:22 +09:00
|
|
|
Introduction
|
|
|
|
============
|
2019-09-06 11:32:33 +01:00
|
|
|
|
|
|
|
Symbol Namespaces have been introduced as a means to structure the export
|
|
|
|
surface of the in-kernel API. It allows subsystem maintainers to partition
|
|
|
|
their exported symbols into separate namespaces. That is useful for
|
|
|
|
documentation purposes (think of the SUBSYSTEM_DEBUG namespace) as well as for
|
|
|
|
limiting the availability of a set of symbols for use in other parts of the
|
|
|
|
kernel. As of today, modules that make use of symbols exported into namespaces,
|
|
|
|
are required to import the namespace. Otherwise the kernel will, depending on
|
|
|
|
its configuration, reject loading the module or warn about a missing import.
|
|
|
|
|
2025-05-02 16:12:09 +02:00
|
|
|
Additionally, it is possible to put symbols into a module namespace, strictly
|
|
|
|
limiting which modules are allowed to use these symbols.
|
|
|
|
|
2025-05-22 16:17:22 +09:00
|
|
|
How to define Symbol Namespaces
|
|
|
|
===============================
|
2019-09-06 11:32:33 +01:00
|
|
|
|
|
|
|
Symbols can be exported into namespace using different methods. All of them are
|
|
|
|
changing the way EXPORT_SYMBOL and friends are instrumented to create ksymtab
|
|
|
|
entries.
|
|
|
|
|
2025-05-22 16:17:22 +09:00
|
|
|
Using the EXPORT_SYMBOL macros
|
|
|
|
------------------------------
|
2019-09-06 11:32:33 +01:00
|
|
|
|
|
|
|
In addition to the macros EXPORT_SYMBOL() and EXPORT_SYMBOL_GPL(), that allow
|
|
|
|
exporting of kernel symbols to the kernel symbol table, variants of these are
|
|
|
|
available to export symbols into a certain namespace: EXPORT_SYMBOL_NS() and
|
2024-12-30 15:23:59 +01:00
|
|
|
EXPORT_SYMBOL_NS_GPL(). They take one additional argument: the namespace as a
|
|
|
|
string constant. Note that this string must not contain whitespaces.
|
|
|
|
E.g. to export the symbol ``usb_stor_suspend`` into the
|
2021-04-28 18:07:20 +08:00
|
|
|
namespace ``USB_STORAGE``, use::
|
2019-09-06 11:32:33 +01:00
|
|
|
|
module: Convert symbol namespace to string literal
Clean up the existing export namespace code along the same lines of
commit 33def8498fdd ("treewide: Convert macro and uses of __section(foo)
to __section("foo")") and for the same reason, it is not desired for the
namespace argument to be a macro expansion itself.
Scripted using
git grep -l -e MODULE_IMPORT_NS -e EXPORT_SYMBOL_NS | while read file;
do
awk -i inplace '
/^#define EXPORT_SYMBOL_NS/ {
gsub(/__stringify\(ns\)/, "ns");
print;
next;
}
/^#define MODULE_IMPORT_NS/ {
gsub(/__stringify\(ns\)/, "ns");
print;
next;
}
/MODULE_IMPORT_NS/ {
$0 = gensub(/MODULE_IMPORT_NS\(([^)]*)\)/, "MODULE_IMPORT_NS(\"\\1\")", "g");
}
/EXPORT_SYMBOL_NS/ {
if ($0 ~ /(EXPORT_SYMBOL_NS[^(]*)\(([^,]+),/) {
if ($0 !~ /(EXPORT_SYMBOL_NS[^(]*)\(([^,]+), ([^)]+)\)/ &&
$0 !~ /(EXPORT_SYMBOL_NS[^(]*)\(\)/ &&
$0 !~ /^my/) {
getline line;
gsub(/[[:space:]]*\\$/, "");
gsub(/[[:space:]]/, "", line);
$0 = $0 " " line;
}
$0 = gensub(/(EXPORT_SYMBOL_NS[^(]*)\(([^,]+), ([^)]+)\)/,
"\\1(\\2, \"\\3\")", "g");
}
}
{ print }' $file;
done
Requested-by: Masahiro Yamada <masahiroy@kernel.org>
Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org>
Link: https://mail.google.com/mail/u/2/#inbox/FMfcgzQXKWgMmjdFwwdsfgxzKpVHWPlc
Acked-by: Greg KH <gregkh@linuxfoundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2024-12-02 15:59:47 +01:00
|
|
|
EXPORT_SYMBOL_NS(usb_stor_suspend, "USB_STORAGE");
|
2019-09-06 11:32:33 +01:00
|
|
|
|
2021-04-28 18:07:20 +08:00
|
|
|
The corresponding ksymtab entry struct ``kernel_symbol`` will have the member
|
|
|
|
``namespace`` set accordingly. A symbol that is exported without a namespace will
|
|
|
|
refer to ``NULL``. There is no default namespace if none is defined. ``modpost``
|
2022-06-15 06:33:03 +09:00
|
|
|
and kernel/module/main.c make use the namespace at build time or module load
|
|
|
|
time, respectively.
|
2019-09-06 11:32:33 +01:00
|
|
|
|
2025-05-22 16:17:22 +09:00
|
|
|
Using the DEFAULT_SYMBOL_NAMESPACE define
|
|
|
|
-----------------------------------------
|
2019-09-06 11:32:33 +01:00
|
|
|
|
|
|
|
Defining namespaces for all symbols of a subsystem can be very verbose and may
|
|
|
|
become hard to maintain. Therefore a default define (DEFAULT_SYMBOL_NAMESPACE)
|
|
|
|
is been provided, that, if set, will become the default for all EXPORT_SYMBOL()
|
|
|
|
and EXPORT_SYMBOL_GPL() macro expansions that do not specify a namespace.
|
|
|
|
|
|
|
|
There are multiple ways of specifying this define and it depends on the
|
|
|
|
subsystem and the maintainer's preference, which one to use. The first option
|
2021-04-28 18:07:20 +08:00
|
|
|
is to define the default namespace in the ``Makefile`` of the subsystem. E.g. to
|
2019-09-06 11:32:33 +01:00
|
|
|
export all symbols defined in usb-common into the namespace USB_COMMON, add a
|
|
|
|
line like this to drivers/usb/common/Makefile::
|
|
|
|
|
2024-12-03 19:21:07 +09:00
|
|
|
ccflags-y += -DDEFAULT_SYMBOL_NAMESPACE='"USB_COMMON"'
|
2019-09-06 11:32:33 +01:00
|
|
|
|
|
|
|
That will affect all EXPORT_SYMBOL() and EXPORT_SYMBOL_GPL() statements. A
|
|
|
|
symbol exported with EXPORT_SYMBOL_NS() while this definition is present, will
|
|
|
|
still be exported into the namespace that is passed as the namespace argument
|
|
|
|
as this argument has preference over a default symbol namespace.
|
|
|
|
|
|
|
|
A second option to define the default namespace is directly in the compilation
|
|
|
|
unit as preprocessor statement. The above example would then read::
|
|
|
|
|
2024-12-03 19:21:07 +09:00
|
|
|
#define DEFAULT_SYMBOL_NAMESPACE "USB_COMMON"
|
2019-09-06 11:32:33 +01:00
|
|
|
|
2024-12-30 15:24:00 +01:00
|
|
|
within the corresponding compilation unit before the #include for
|
|
|
|
<linux/export.h>. Typically it's placed before the first #include statement.
|
2019-09-06 11:32:33 +01:00
|
|
|
|
2025-05-22 16:17:22 +09:00
|
|
|
Using the EXPORT_SYMBOL_GPL_FOR_MODULES() macro
|
|
|
|
-----------------------------------------------
|
2025-05-02 16:12:09 +02:00
|
|
|
|
|
|
|
Symbols exported using this macro are put into a module namespace. This
|
|
|
|
namespace cannot be imported.
|
|
|
|
|
|
|
|
The macro takes a comma separated list of module names, allowing only those
|
|
|
|
modules to access this symbol. Simple tail-globs are supported.
|
|
|
|
|
2025-05-27 00:10:39 +03:00
|
|
|
For example::
|
2025-05-02 16:12:09 +02:00
|
|
|
|
|
|
|
EXPORT_SYMBOL_GPL_FOR_MODULES(preempt_notifier_inc, "kvm,kvm-*")
|
|
|
|
|
|
|
|
will limit usage of this symbol to modules whoes name matches the given
|
|
|
|
patterns.
|
|
|
|
|
2025-05-22 16:17:22 +09:00
|
|
|
How to use Symbols exported in Namespaces
|
|
|
|
=========================================
|
2019-09-06 11:32:33 +01:00
|
|
|
|
|
|
|
In order to use symbols that are exported into namespaces, kernel modules need
|
|
|
|
to explicitly import these namespaces. Otherwise the kernel might reject to
|
|
|
|
load the module. The module code is required to use the macro MODULE_IMPORT_NS
|
|
|
|
for the namespaces it uses symbols from. E.g. a module using the
|
|
|
|
usb_stor_suspend symbol from above, needs to import the namespace USB_STORAGE
|
|
|
|
using a statement like::
|
|
|
|
|
module: Convert symbol namespace to string literal
Clean up the existing export namespace code along the same lines of
commit 33def8498fdd ("treewide: Convert macro and uses of __section(foo)
to __section("foo")") and for the same reason, it is not desired for the
namespace argument to be a macro expansion itself.
Scripted using
git grep -l -e MODULE_IMPORT_NS -e EXPORT_SYMBOL_NS | while read file;
do
awk -i inplace '
/^#define EXPORT_SYMBOL_NS/ {
gsub(/__stringify\(ns\)/, "ns");
print;
next;
}
/^#define MODULE_IMPORT_NS/ {
gsub(/__stringify\(ns\)/, "ns");
print;
next;
}
/MODULE_IMPORT_NS/ {
$0 = gensub(/MODULE_IMPORT_NS\(([^)]*)\)/, "MODULE_IMPORT_NS(\"\\1\")", "g");
}
/EXPORT_SYMBOL_NS/ {
if ($0 ~ /(EXPORT_SYMBOL_NS[^(]*)\(([^,]+),/) {
if ($0 !~ /(EXPORT_SYMBOL_NS[^(]*)\(([^,]+), ([^)]+)\)/ &&
$0 !~ /(EXPORT_SYMBOL_NS[^(]*)\(\)/ &&
$0 !~ /^my/) {
getline line;
gsub(/[[:space:]]*\\$/, "");
gsub(/[[:space:]]/, "", line);
$0 = $0 " " line;
}
$0 = gensub(/(EXPORT_SYMBOL_NS[^(]*)\(([^,]+), ([^)]+)\)/,
"\\1(\\2, \"\\3\")", "g");
}
}
{ print }' $file;
done
Requested-by: Masahiro Yamada <masahiroy@kernel.org>
Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org>
Link: https://mail.google.com/mail/u/2/#inbox/FMfcgzQXKWgMmjdFwwdsfgxzKpVHWPlc
Acked-by: Greg KH <gregkh@linuxfoundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2024-12-02 15:59:47 +01:00
|
|
|
MODULE_IMPORT_NS("USB_STORAGE");
|
2019-09-06 11:32:33 +01:00
|
|
|
|
2021-04-28 18:07:20 +08:00
|
|
|
This will create a ``modinfo`` tag in the module for each imported namespace.
|
2019-09-06 11:32:33 +01:00
|
|
|
This has the side effect, that the imported namespaces of a module can be
|
|
|
|
inspected with modinfo::
|
|
|
|
|
|
|
|
$ modinfo drivers/usb/storage/ums-karma.ko
|
|
|
|
[...]
|
|
|
|
import_ns: USB_STORAGE
|
|
|
|
[...]
|
|
|
|
|
|
|
|
|
2024-12-03 19:21:06 +09:00
|
|
|
It is advisable to add the MODULE_IMPORT_NS() statement close to other module
|
2025-05-22 16:17:22 +09:00
|
|
|
metadata definitions like MODULE_AUTHOR() or MODULE_LICENSE().
|
2019-09-06 11:32:33 +01:00
|
|
|
|
2025-05-22 16:17:22 +09:00
|
|
|
Loading Modules that use namespaced Symbols
|
|
|
|
===========================================
|
2019-09-06 11:32:33 +01:00
|
|
|
|
2021-04-28 18:07:20 +08:00
|
|
|
At module loading time (e.g. ``insmod``), the kernel will check each symbol
|
2019-09-06 11:32:33 +01:00
|
|
|
referenced from the module for its availability and whether the namespace it
|
|
|
|
might be exported to has been imported by the module. The default behaviour of
|
|
|
|
the kernel is to reject loading modules that don't specify sufficient imports.
|
|
|
|
An error will be logged and loading will be failed with EINVAL. In order to
|
|
|
|
allow loading of modules that don't satisfy this precondition, a configuration
|
|
|
|
option is available: Setting MODULE_ALLOW_MISSING_NAMESPACE_IMPORTS=y will
|
|
|
|
enable loading regardless, but will emit a warning.
|
|
|
|
|
2025-05-22 16:17:22 +09:00
|
|
|
Automatically creating MODULE_IMPORT_NS statements
|
|
|
|
==================================================
|
2019-09-06 11:32:33 +01:00
|
|
|
|
|
|
|
Missing namespaces imports can easily be detected at build time. In fact,
|
|
|
|
modpost will emit a warning if a module uses a symbol from a namespace
|
|
|
|
without importing it.
|
2024-12-03 19:21:06 +09:00
|
|
|
MODULE_IMPORT_NS() statements will usually be added at a definite location
|
2019-09-06 11:32:33 +01:00
|
|
|
(along with other module meta data). To make the life of module authors (and
|
|
|
|
subsystem maintainers) easier, a script and make target is available to fixup
|
|
|
|
missing imports. Fixing missing imports can be done with::
|
|
|
|
|
|
|
|
$ make nsdeps
|
|
|
|
|
|
|
|
A typical scenario for module authors would be::
|
|
|
|
|
|
|
|
- write code that depends on a symbol from a not imported namespace
|
2021-04-28 18:07:20 +08:00
|
|
|
- ``make``
|
2019-09-06 11:32:33 +01:00
|
|
|
- notice the warning of modpost telling about a missing import
|
2021-04-28 18:07:20 +08:00
|
|
|
- run ``make nsdeps`` to add the import to the correct code location
|
2019-09-06 11:32:33 +01:00
|
|
|
|
|
|
|
For subsystem maintainers introducing a namespace, the steps are very similar.
|
2021-04-28 18:07:20 +08:00
|
|
|
Again, ``make nsdeps`` will eventually add the missing namespace imports for
|
2019-09-06 11:32:33 +01:00
|
|
|
in-tree modules::
|
|
|
|
|
|
|
|
- move or add symbols to a namespace (e.g. with EXPORT_SYMBOL_NS())
|
2021-04-28 18:07:20 +08:00
|
|
|
- ``make`` (preferably with an allmodconfig to cover all in-kernel
|
2019-09-06 11:32:33 +01:00
|
|
|
modules)
|
|
|
|
- notice the warning of modpost telling about a missing import
|
2021-04-28 18:07:20 +08:00
|
|
|
- run ``make nsdeps`` to add the import to the correct code location
|
2019-09-06 11:32:33 +01:00
|
|
|
|
2019-10-29 21:38:08 +09:00
|
|
|
You can also run nsdeps for external module builds. A typical usage is::
|
|
|
|
|
|
|
|
$ make -C <path_to_kernel_src> M=$PWD nsdeps
|
2025-05-02 16:12:09 +02:00
|
|
|
|
|
|
|
Note: it will happily generate an import statement for the module namespace;
|
|
|
|
which will not work and generates build and runtime failures.
|