mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-16 18:08:20 +00:00
Rust changes for v6.12
Toolchain and infrastructure: - Support 'MITIGATION_{RETHUNK,RETPOLINE,SLS}' (which cleans up objtool warnings), teach objtool about 'noreturn' Rust symbols and mimic '___ADDRESSABLE()' for 'module_{init,exit}'. With that, we should be objtool-warning-free, so enable it to run for all Rust object files. - KASAN (no 'SW_TAGS'), KCFI and shadow call sanitizer support. - Support 'RUSTC_VERSION', including re-config and re-build on change. - Split helpers file into several files in a folder, to avoid conflicts in it. Eventually those files will be moved to the right places with the new build system. In addition, remove the need to manually export the symbols defined there, reusing existing machinery for that. - Relax restriction on configurations with Rust + GCC plugins to just the RANDSTRUCT plugin. 'kernel' crate: - New 'list' module: doubly-linked linked list for use with reference counted values, which is heavily used by the upcoming Rust Binder. This includes 'ListArc' (a wrapper around 'Arc' that is guaranteed unique for the given ID), 'AtomicTracker' (tracks whether a 'ListArc' exists using an atomic), 'ListLinks' (the prev/next pointers for an item in a linked list), 'List' (the linked list itself), 'Iter' (an iterator over a 'List'), 'Cursor' (a cursor into a 'List' that allows to remove elements), 'ListArcField' (a field exclusively owned by a 'ListArc'), as well as support for heterogeneous lists. - New 'rbtree' module: red-black tree abstractions used by the upcoming Rust Binder. This includes 'RBTree' (the red-black tree itself), 'RBTreeNode' (a node), 'RBTreeNodeReservation' (a memory reservation for a node), 'Iter' and 'IterMut' (immutable and mutable iterators), 'Cursor' (bidirectional cursor that allows to remove elements), as well as an entry API similar to the Rust standard library one. - 'init' module: add 'write_[pin_]init' methods and the 'InPlaceWrite' trait. Add the 'assert_pinned!' macro. - 'sync' module: implement the 'InPlaceInit' trait for 'Arc' by introducing an associated type in the trait. - 'alloc' module: add 'drop_contents' method to 'BoxExt'. - 'types' module: implement the 'ForeignOwnable' trait for 'Pin<Box<T>>' and improve the trait's documentation. In addition, add the 'into_raw' method to the 'ARef' type. - 'error' module: in preparation for the upcoming Rust support for 32-bit architectures, like arm, locally allow Clippy lint for those. Documentation: - https://rust.docs.kernel.org has been announced, so link to it. - Enable rustdoc's "jump to definition" feature, making its output a bit closer to the experience in a cross-referencer. - Debian Testing now also provides recent Rust releases (outside of the freeze period), so add it to the list. MAINTAINERS: - Trevor is joining as reviewer of the "RUST" entry. And a few other small bits. -----BEGIN PGP SIGNATURE----- iQIzBAABCgAdFiEEPjU5OPd5QIZ9jqqOGXyLc2htIW0FAmbzNz4ACgkQGXyLc2ht IW3muA/9HcPL0QqVB5+SqSRqcatmrFU/wq8Oaa6Z/No0JaynqyikK+R1WNokUd/5 WpQi4PC1OYV+ekyAuWdkooKmaSqagH5r53XlezNw+cM5zo8y7p0otVlbepQ0t3Ky pVEmfDRIeSFXsKrg91BJUKyJf70TQlgSggDVCExlanfOjPz88C1+s3EcJ/XWYGKQ cRk/XDdbF5eNaldp2MriVF0fw7XktgIrmVzxt/z0lb4PE7RaCAnO6gSQI+90Vb2d zvyOYKS4AkqE3suFvDIIUlPUv+8XbACj0c4wvBZHH5uZGTbgWUffqygJ45GqChEt c4fS/+E8VaM1z0EvxNczC0nQkfLwkTc1mgbP+sG3VZJMPVCJ2zQan1/ond7GqCpw pt6uQaGvDsAvllm7sbiAIVaAY81icqyYWKfNBXLLEL7DhY5je5Wq+E83XQ8d5u5F EuapnZhW3y12d6UCsSe9bD8W45NFoWHPXky1TzT+whTxnX1yH9YsPXbJceGSbbgd Lw3GmUtZx2bVAMToVjNFD2lPA3OmPY1e2lk0jwzTuQrEXfnZYuzbjqs3YUijb7xR AlsWfIb0IHBwHWpB7da24ezqWP2VD4eaDdD8/+LmDSj6XLngxMNWRLKmXT000eTW vIFP9GJrvag2R3YFPhrurgGpRsp8HUTLtvcZROxp2JVQGQ7Z4Ww= =52BN -----END PGP SIGNATURE----- Merge tag 'rust-6.12' of https://github.com/Rust-for-Linux/linux Pull Rust updates from Miguel Ojeda: "Toolchain and infrastructure: - Support 'MITIGATION_{RETHUNK,RETPOLINE,SLS}' (which cleans up objtool warnings), teach objtool about 'noreturn' Rust symbols and mimic '___ADDRESSABLE()' for 'module_{init,exit}'. With that, we should be objtool-warning-free, so enable it to run for all Rust object files. - KASAN (no 'SW_TAGS'), KCFI and shadow call sanitizer support. - Support 'RUSTC_VERSION', including re-config and re-build on change. - Split helpers file into several files in a folder, to avoid conflicts in it. Eventually those files will be moved to the right places with the new build system. In addition, remove the need to manually export the symbols defined there, reusing existing machinery for that. - Relax restriction on configurations with Rust + GCC plugins to just the RANDSTRUCT plugin. 'kernel' crate: - New 'list' module: doubly-linked linked list for use with reference counted values, which is heavily used by the upcoming Rust Binder. This includes 'ListArc' (a wrapper around 'Arc' that is guaranteed unique for the given ID), 'AtomicTracker' (tracks whether a 'ListArc' exists using an atomic), 'ListLinks' (the prev/next pointers for an item in a linked list), 'List' (the linked list itself), 'Iter' (an iterator over a 'List'), 'Cursor' (a cursor into a 'List' that allows to remove elements), 'ListArcField' (a field exclusively owned by a 'ListArc'), as well as support for heterogeneous lists. - New 'rbtree' module: red-black tree abstractions used by the upcoming Rust Binder. This includes 'RBTree' (the red-black tree itself), 'RBTreeNode' (a node), 'RBTreeNodeReservation' (a memory reservation for a node), 'Iter' and 'IterMut' (immutable and mutable iterators), 'Cursor' (bidirectional cursor that allows to remove elements), as well as an entry API similar to the Rust standard library one. - 'init' module: add 'write_[pin_]init' methods and the 'InPlaceWrite' trait. Add the 'assert_pinned!' macro. - 'sync' module: implement the 'InPlaceInit' trait for 'Arc' by introducing an associated type in the trait. - 'alloc' module: add 'drop_contents' method to 'BoxExt'. - 'types' module: implement the 'ForeignOwnable' trait for 'Pin<Box<T>>' and improve the trait's documentation. In addition, add the 'into_raw' method to the 'ARef' type. - 'error' module: in preparation for the upcoming Rust support for 32-bit architectures, like arm, locally allow Clippy lint for those. Documentation: - https://rust.docs.kernel.org has been announced, so link to it. - Enable rustdoc's "jump to definition" feature, making its output a bit closer to the experience in a cross-referencer. - Debian Testing now also provides recent Rust releases (outside of the freeze period), so add it to the list. MAINTAINERS: - Trevor is joining as reviewer of the "RUST" entry. And a few other small bits" * tag 'rust-6.12' of https://github.com/Rust-for-Linux/linux: (54 commits) kasan: rust: Add KASAN smoke test via UAF kbuild: rust: Enable KASAN support rust: kasan: Rust does not support KHWASAN kbuild: rust: Define probing macros for rustc kasan: simplify and clarify Makefile rust: cfi: add support for CFI_CLANG with Rust cfi: add CONFIG_CFI_ICALL_NORMALIZE_INTEGERS rust: support for shadow call stack sanitizer docs: rust: include other expressions in conditional compilation section kbuild: rust: replace proc macros dependency on `core.o` with the version text kbuild: rust: rebuild if the version text changes kbuild: rust: re-run Kconfig if the version text changes kbuild: rust: add `CONFIG_RUSTC_VERSION` rust: avoid `box_uninit_write` feature MAINTAINERS: add Trevor Gross as Rust reviewer rust: rbtree: add `RBTree::entry` rust: rbtree: add cursor rust: rbtree: add mutable iterator rust: rbtree: add iterator rust: rbtree: add red-black tree implementation backed by the C version ...
This commit is contained in:
commit
5701725692
@ -15,6 +15,8 @@ but not `std <https://doc.rust-lang.org/std/>`_. Crates for use in the
|
|||||||
kernel must opt into this behavior using the ``#![no_std]`` attribute.
|
kernel must opt into this behavior using the ``#![no_std]`` attribute.
|
||||||
|
|
||||||
|
|
||||||
|
.. _rust_code_documentation:
|
||||||
|
|
||||||
Code documentation
|
Code documentation
|
||||||
------------------
|
------------------
|
||||||
|
|
||||||
@ -22,10 +24,17 @@ Rust kernel code is documented using ``rustdoc``, its built-in documentation
|
|||||||
generator.
|
generator.
|
||||||
|
|
||||||
The generated HTML docs include integrated search, linked items (e.g. types,
|
The generated HTML docs include integrated search, linked items (e.g. types,
|
||||||
functions, constants), source code, etc. They may be read at (TODO: link when
|
functions, constants), source code, etc. They may be read at:
|
||||||
in mainline and generated alongside the rest of the documentation):
|
|
||||||
|
|
||||||
http://kernel.org/
|
https://rust.docs.kernel.org
|
||||||
|
|
||||||
|
For linux-next, please see:
|
||||||
|
|
||||||
|
https://rust.docs.kernel.org/next/
|
||||||
|
|
||||||
|
There are also tags for each main release, e.g.:
|
||||||
|
|
||||||
|
https://rust.docs.kernel.org/6.10/
|
||||||
|
|
||||||
The docs can also be easily generated and read locally. This is quite fast
|
The docs can also be easily generated and read locally. This is quite fast
|
||||||
(same order as compiling the code itself) and no special tools or environment
|
(same order as compiling the code itself) and no special tools or environment
|
||||||
@ -75,7 +84,7 @@ should provide as-safe-as-possible abstractions as needed.
|
|||||||
.. code-block::
|
.. code-block::
|
||||||
|
|
||||||
rust/bindings/
|
rust/bindings/
|
||||||
(rust/helpers.c)
|
(rust/helpers/)
|
||||||
|
|
||||||
include/ -----+ <-+
|
include/ -----+ <-+
|
||||||
| |
|
| |
|
||||||
@ -112,7 +121,7 @@ output files in the ``rust/bindings/`` directory.
|
|||||||
|
|
||||||
For parts of the C header that ``bindgen`` does not auto generate, e.g. C
|
For parts of the C header that ``bindgen`` does not auto generate, e.g. C
|
||||||
``inline`` functions or non-trivial macros, it is acceptable to add a small
|
``inline`` functions or non-trivial macros, it is acceptable to add a small
|
||||||
wrapper function to ``rust/helpers.c`` to make it available for the Rust side as
|
wrapper function to ``rust/helpers/`` to make it available for the Rust side as
|
||||||
well.
|
well.
|
||||||
|
|
||||||
Abstractions
|
Abstractions
|
||||||
@ -142,3 +151,11 @@ configuration:
|
|||||||
#[cfg(CONFIG_X="y")] // Enabled as a built-in (`y`)
|
#[cfg(CONFIG_X="y")] // Enabled as a built-in (`y`)
|
||||||
#[cfg(CONFIG_X="m")] // Enabled as a module (`m`)
|
#[cfg(CONFIG_X="m")] // Enabled as a module (`m`)
|
||||||
#[cfg(not(CONFIG_X))] // Disabled
|
#[cfg(not(CONFIG_X))] // Disabled
|
||||||
|
|
||||||
|
For other predicates that Rust's ``cfg`` does not support, e.g. expressions with
|
||||||
|
numerical comparisons, one may define a new Kconfig symbol:
|
||||||
|
|
||||||
|
.. code-block:: kconfig
|
||||||
|
|
||||||
|
config RUSTC_VERSION_MIN_107900
|
||||||
|
def_bool y if RUSTC_VERSION >= 107900
|
||||||
|
@ -25,13 +25,27 @@ support is still in development/experimental, especially for certain kernel
|
|||||||
configurations.
|
configurations.
|
||||||
|
|
||||||
|
|
||||||
|
Code documentation
|
||||||
|
------------------
|
||||||
|
|
||||||
|
Given a kernel configuration, the kernel may generate Rust code documentation,
|
||||||
|
i.e. HTML rendered by the ``rustdoc`` tool.
|
||||||
|
|
||||||
.. only:: rustdoc and html
|
.. only:: rustdoc and html
|
||||||
|
|
||||||
You can also browse `rustdoc documentation <rustdoc/kernel/index.html>`_.
|
This kernel documentation was built with `Rust code documentation
|
||||||
|
<rustdoc/kernel/index.html>`_.
|
||||||
|
|
||||||
.. only:: not rustdoc and html
|
.. only:: not rustdoc and html
|
||||||
|
|
||||||
This documentation does not include rustdoc generated information.
|
This kernel documentation was not built with Rust code documentation.
|
||||||
|
|
||||||
|
A pregenerated version is provided at:
|
||||||
|
|
||||||
|
https://rust.docs.kernel.org
|
||||||
|
|
||||||
|
Please see the :ref:`Code documentation <rust_code_documentation>` section for
|
||||||
|
more details.
|
||||||
|
|
||||||
.. toctree::
|
.. toctree::
|
||||||
:maxdepth: 1
|
:maxdepth: 1
|
||||||
|
@ -39,8 +39,8 @@ of the box, e.g.::
|
|||||||
Debian
|
Debian
|
||||||
******
|
******
|
||||||
|
|
||||||
Debian Unstable (Sid), outside of the freeze period, provides recent Rust
|
Debian Testing and Debian Unstable (Sid), outside of the freeze period, provide
|
||||||
releases and thus it should generally work out of the box, e.g.::
|
recent Rust releases and thus they should generally work out of the box, e.g.::
|
||||||
|
|
||||||
apt install rustc rust-src bindgen rustfmt rust-clippy
|
apt install rustc rust-src bindgen rustfmt rust-clippy
|
||||||
|
|
||||||
|
@ -20144,6 +20144,7 @@ R: Björn Roy Baron <bjorn3_gh@protonmail.com>
|
|||||||
R: Benno Lossin <benno.lossin@proton.me>
|
R: Benno Lossin <benno.lossin@proton.me>
|
||||||
R: Andreas Hindborg <a.hindborg@kernel.org>
|
R: Andreas Hindborg <a.hindborg@kernel.org>
|
||||||
R: Alice Ryhl <aliceryhl@google.com>
|
R: Alice Ryhl <aliceryhl@google.com>
|
||||||
|
R: Trevor Gross <tmgross@umich.edu>
|
||||||
L: rust-for-linux@vger.kernel.org
|
L: rust-for-linux@vger.kernel.org
|
||||||
S: Supported
|
S: Supported
|
||||||
W: https://rust-for-linux.com
|
W: https://rust-for-linux.com
|
||||||
|
19
Makefile
19
Makefile
@ -645,9 +645,11 @@ endif
|
|||||||
|
|
||||||
# The expansion should be delayed until arch/$(SRCARCH)/Makefile is included.
|
# The expansion should be delayed until arch/$(SRCARCH)/Makefile is included.
|
||||||
# Some architectures define CROSS_COMPILE in arch/$(SRCARCH)/Makefile.
|
# Some architectures define CROSS_COMPILE in arch/$(SRCARCH)/Makefile.
|
||||||
# CC_VERSION_TEXT is referenced from Kconfig (so it needs export),
|
# CC_VERSION_TEXT and RUSTC_VERSION_TEXT are referenced from Kconfig (so they
|
||||||
# and from include/config/auto.conf.cmd to detect the compiler upgrade.
|
# need export), and from include/config/auto.conf.cmd to detect the compiler
|
||||||
|
# upgrade.
|
||||||
CC_VERSION_TEXT = $(subst $(pound),,$(shell LC_ALL=C $(CC) --version 2>/dev/null | head -n 1))
|
CC_VERSION_TEXT = $(subst $(pound),,$(shell LC_ALL=C $(CC) --version 2>/dev/null | head -n 1))
|
||||||
|
RUSTC_VERSION_TEXT = $(subst $(pound),,$(shell $(RUSTC) --version 2>/dev/null))
|
||||||
|
|
||||||
ifneq ($(findstring clang,$(CC_VERSION_TEXT)),)
|
ifneq ($(findstring clang,$(CC_VERSION_TEXT)),)
|
||||||
include $(srctree)/scripts/Makefile.clang
|
include $(srctree)/scripts/Makefile.clang
|
||||||
@ -668,7 +670,7 @@ ifdef config-build
|
|||||||
# KBUILD_DEFCONFIG may point out an alternative default configuration
|
# KBUILD_DEFCONFIG may point out an alternative default configuration
|
||||||
# used for 'make defconfig'
|
# used for 'make defconfig'
|
||||||
include $(srctree)/arch/$(SRCARCH)/Makefile
|
include $(srctree)/arch/$(SRCARCH)/Makefile
|
||||||
export KBUILD_DEFCONFIG KBUILD_KCONFIG CC_VERSION_TEXT
|
export KBUILD_DEFCONFIG KBUILD_KCONFIG CC_VERSION_TEXT RUSTC_VERSION_TEXT
|
||||||
|
|
||||||
config: outputmakefile scripts_basic FORCE
|
config: outputmakefile scripts_basic FORCE
|
||||||
$(Q)$(MAKE) $(build)=scripts/kconfig $@
|
$(Q)$(MAKE) $(build)=scripts/kconfig $@
|
||||||
@ -924,6 +926,7 @@ ifdef CONFIG_SHADOW_CALL_STACK
|
|||||||
ifndef CONFIG_DYNAMIC_SCS
|
ifndef CONFIG_DYNAMIC_SCS
|
||||||
CC_FLAGS_SCS := -fsanitize=shadow-call-stack
|
CC_FLAGS_SCS := -fsanitize=shadow-call-stack
|
||||||
KBUILD_CFLAGS += $(CC_FLAGS_SCS)
|
KBUILD_CFLAGS += $(CC_FLAGS_SCS)
|
||||||
|
KBUILD_RUSTFLAGS += -Zsanitizer=shadow-call-stack
|
||||||
endif
|
endif
|
||||||
export CC_FLAGS_SCS
|
export CC_FLAGS_SCS
|
||||||
endif
|
endif
|
||||||
@ -948,6 +951,16 @@ endif
|
|||||||
|
|
||||||
ifdef CONFIG_CFI_CLANG
|
ifdef CONFIG_CFI_CLANG
|
||||||
CC_FLAGS_CFI := -fsanitize=kcfi
|
CC_FLAGS_CFI := -fsanitize=kcfi
|
||||||
|
ifdef CONFIG_CFI_ICALL_NORMALIZE_INTEGERS
|
||||||
|
CC_FLAGS_CFI += -fsanitize-cfi-icall-experimental-normalize-integers
|
||||||
|
endif
|
||||||
|
ifdef CONFIG_RUST
|
||||||
|
# Always pass -Zsanitizer-cfi-normalize-integers as CONFIG_RUST selects
|
||||||
|
# CONFIG_CFI_ICALL_NORMALIZE_INTEGERS.
|
||||||
|
RUSTC_FLAGS_CFI := -Zsanitizer=kcfi -Zsanitizer-cfi-normalize-integers
|
||||||
|
KBUILD_RUSTFLAGS += $(RUSTC_FLAGS_CFI)
|
||||||
|
export RUSTC_FLAGS_CFI
|
||||||
|
endif
|
||||||
KBUILD_CFLAGS += $(CC_FLAGS_CFI)
|
KBUILD_CFLAGS += $(CC_FLAGS_CFI)
|
||||||
export CC_FLAGS_CFI
|
export CC_FLAGS_CFI
|
||||||
endif
|
endif
|
||||||
|
16
arch/Kconfig
16
arch/Kconfig
@ -835,6 +835,22 @@ config CFI_CLANG
|
|||||||
|
|
||||||
https://clang.llvm.org/docs/ControlFlowIntegrity.html
|
https://clang.llvm.org/docs/ControlFlowIntegrity.html
|
||||||
|
|
||||||
|
config CFI_ICALL_NORMALIZE_INTEGERS
|
||||||
|
bool "Normalize CFI tags for integers"
|
||||||
|
depends on CFI_CLANG
|
||||||
|
depends on $(cc-option,-fsanitize=kcfi -fsanitize-cfi-icall-experimental-normalize-integers)
|
||||||
|
help
|
||||||
|
This option normalizes the CFI tags for integer types so that all
|
||||||
|
integer types of the same size and signedness receive the same CFI
|
||||||
|
tag.
|
||||||
|
|
||||||
|
The option is separate from CONFIG_RUST because it affects the ABI.
|
||||||
|
When working with build systems that care about the ABI, it is
|
||||||
|
convenient to be able to turn on this flag first, before Rust is
|
||||||
|
turned on.
|
||||||
|
|
||||||
|
This option is necessary for using CFI with Rust. If unsure, say N.
|
||||||
|
|
||||||
config CFI_PERMISSIVE
|
config CFI_PERMISSIVE
|
||||||
bool "Use CFI in permissive mode"
|
bool "Use CFI in permissive mode"
|
||||||
depends on CFI_CLANG
|
depends on CFI_CLANG
|
||||||
|
@ -235,7 +235,7 @@ config ARM64
|
|||||||
select HAVE_FUNCTION_ARG_ACCESS_API
|
select HAVE_FUNCTION_ARG_ACCESS_API
|
||||||
select MMU_GATHER_RCU_TABLE_FREE
|
select MMU_GATHER_RCU_TABLE_FREE
|
||||||
select HAVE_RSEQ
|
select HAVE_RSEQ
|
||||||
select HAVE_RUST if CPU_LITTLE_ENDIAN
|
select HAVE_RUST if RUSTC_SUPPORTS_ARM64
|
||||||
select HAVE_STACKPROTECTOR
|
select HAVE_STACKPROTECTOR
|
||||||
select HAVE_SYSCALL_TRACEPOINTS
|
select HAVE_SYSCALL_TRACEPOINTS
|
||||||
select HAVE_KPROBES
|
select HAVE_KPROBES
|
||||||
@ -270,6 +270,18 @@ config ARM64
|
|||||||
help
|
help
|
||||||
ARM 64-bit (AArch64) Linux support.
|
ARM 64-bit (AArch64) Linux support.
|
||||||
|
|
||||||
|
config RUSTC_SUPPORTS_ARM64
|
||||||
|
def_bool y
|
||||||
|
depends on CPU_LITTLE_ENDIAN
|
||||||
|
# Shadow call stack is only supported on certain rustc versions.
|
||||||
|
#
|
||||||
|
# When using the UNWIND_PATCH_PAC_INTO_SCS option, rustc version 1.80+ is
|
||||||
|
# required due to use of the -Zfixed-x18 flag.
|
||||||
|
#
|
||||||
|
# Otherwise, rustc version 1.82+ is required due to use of the
|
||||||
|
# -Zsanitizer=shadow-call-stack flag.
|
||||||
|
depends on !SHADOW_CALL_STACK || RUSTC_VERSION >= 108200 || RUSTC_VERSION >= 108000 && UNWIND_PATCH_PAC_INTO_SCS
|
||||||
|
|
||||||
config CLANG_SUPPORTS_DYNAMIC_FTRACE_WITH_ARGS
|
config CLANG_SUPPORTS_DYNAMIC_FTRACE_WITH_ARGS
|
||||||
def_bool CC_IS_CLANG
|
def_bool CC_IS_CLANG
|
||||||
# https://github.com/ClangBuiltLinux/linux/issues/1507
|
# https://github.com/ClangBuiltLinux/linux/issues/1507
|
||||||
|
@ -57,9 +57,11 @@ KBUILD_AFLAGS += $(call cc-option,-mabi=lp64)
|
|||||||
ifneq ($(CONFIG_UNWIND_TABLES),y)
|
ifneq ($(CONFIG_UNWIND_TABLES),y)
|
||||||
KBUILD_CFLAGS += -fno-asynchronous-unwind-tables -fno-unwind-tables
|
KBUILD_CFLAGS += -fno-asynchronous-unwind-tables -fno-unwind-tables
|
||||||
KBUILD_AFLAGS += -fno-asynchronous-unwind-tables -fno-unwind-tables
|
KBUILD_AFLAGS += -fno-asynchronous-unwind-tables -fno-unwind-tables
|
||||||
|
KBUILD_RUSTFLAGS += -Cforce-unwind-tables=n
|
||||||
else
|
else
|
||||||
KBUILD_CFLAGS += -fasynchronous-unwind-tables
|
KBUILD_CFLAGS += -fasynchronous-unwind-tables
|
||||||
KBUILD_AFLAGS += -fasynchronous-unwind-tables
|
KBUILD_AFLAGS += -fasynchronous-unwind-tables
|
||||||
|
KBUILD_RUSTFLAGS += -Cforce-unwind-tables=y -Zuse-sync-unwind=n
|
||||||
endif
|
endif
|
||||||
|
|
||||||
ifeq ($(CONFIG_STACKPROTECTOR_PER_TASK),y)
|
ifeq ($(CONFIG_STACKPROTECTOR_PER_TASK),y)
|
||||||
@ -114,6 +116,7 @@ endif
|
|||||||
|
|
||||||
ifeq ($(CONFIG_SHADOW_CALL_STACK), y)
|
ifeq ($(CONFIG_SHADOW_CALL_STACK), y)
|
||||||
KBUILD_CFLAGS += -ffixed-x18
|
KBUILD_CFLAGS += -ffixed-x18
|
||||||
|
KBUILD_RUSTFLAGS += -Zfixed-x18
|
||||||
endif
|
endif
|
||||||
|
|
||||||
ifeq ($(CONFIG_CPU_BIG_ENDIAN), y)
|
ifeq ($(CONFIG_CPU_BIG_ENDIAN), y)
|
||||||
|
@ -177,7 +177,7 @@ config RISCV
|
|||||||
select HAVE_REGS_AND_STACK_ACCESS_API
|
select HAVE_REGS_AND_STACK_ACCESS_API
|
||||||
select HAVE_RETHOOK if !XIP_KERNEL
|
select HAVE_RETHOOK if !XIP_KERNEL
|
||||||
select HAVE_RSEQ
|
select HAVE_RSEQ
|
||||||
select HAVE_RUST if 64BIT
|
select HAVE_RUST if RUSTC_SUPPORTS_RISCV
|
||||||
select HAVE_SAMPLE_FTRACE_DIRECT
|
select HAVE_SAMPLE_FTRACE_DIRECT
|
||||||
select HAVE_SAMPLE_FTRACE_DIRECT_MULTI
|
select HAVE_SAMPLE_FTRACE_DIRECT_MULTI
|
||||||
select HAVE_STACKPROTECTOR
|
select HAVE_STACKPROTECTOR
|
||||||
@ -209,6 +209,13 @@ config RISCV
|
|||||||
select USER_STACKTRACE_SUPPORT
|
select USER_STACKTRACE_SUPPORT
|
||||||
select ZONE_DMA32 if 64BIT
|
select ZONE_DMA32 if 64BIT
|
||||||
|
|
||||||
|
config RUSTC_SUPPORTS_RISCV
|
||||||
|
def_bool y
|
||||||
|
depends on 64BIT
|
||||||
|
# Shadow call stack requires rustc version 1.82+ due to use of the
|
||||||
|
# -Zsanitizer=shadow-call-stack flag.
|
||||||
|
depends on !SHADOW_CALL_STACK || RUSTC_VERSION >= 108200
|
||||||
|
|
||||||
config CLANG_SUPPORTS_DYNAMIC_FTRACE
|
config CLANG_SUPPORTS_DYNAMIC_FTRACE
|
||||||
def_bool CC_IS_CLANG
|
def_bool CC_IS_CLANG
|
||||||
# https://github.com/ClangBuiltLinux/linux/issues/1817
|
# https://github.com/ClangBuiltLinux/linux/issues/1817
|
||||||
|
@ -24,11 +24,15 @@ RETPOLINE_CFLAGS += $(call cc-option,-mindirect-branch-cs-prefix)
|
|||||||
|
|
||||||
ifdef CONFIG_MITIGATION_RETHUNK
|
ifdef CONFIG_MITIGATION_RETHUNK
|
||||||
RETHUNK_CFLAGS := -mfunction-return=thunk-extern
|
RETHUNK_CFLAGS := -mfunction-return=thunk-extern
|
||||||
|
RETHUNK_RUSTFLAGS := -Zfunction-return=thunk-extern
|
||||||
RETPOLINE_CFLAGS += $(RETHUNK_CFLAGS)
|
RETPOLINE_CFLAGS += $(RETHUNK_CFLAGS)
|
||||||
|
RETPOLINE_RUSTFLAGS += $(RETHUNK_RUSTFLAGS)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
export RETHUNK_CFLAGS
|
export RETHUNK_CFLAGS
|
||||||
|
export RETHUNK_RUSTFLAGS
|
||||||
export RETPOLINE_CFLAGS
|
export RETPOLINE_CFLAGS
|
||||||
|
export RETPOLINE_RUSTFLAGS
|
||||||
export RETPOLINE_VDSO_CFLAGS
|
export RETPOLINE_VDSO_CFLAGS
|
||||||
|
|
||||||
# For gcc stack alignment is specified with -mpreferred-stack-boundary,
|
# For gcc stack alignment is specified with -mpreferred-stack-boundary,
|
||||||
@ -218,9 +222,10 @@ KBUILD_CFLAGS += -fno-asynchronous-unwind-tables
|
|||||||
# Avoid indirect branches in kernel to deal with Spectre
|
# Avoid indirect branches in kernel to deal with Spectre
|
||||||
ifdef CONFIG_MITIGATION_RETPOLINE
|
ifdef CONFIG_MITIGATION_RETPOLINE
|
||||||
KBUILD_CFLAGS += $(RETPOLINE_CFLAGS)
|
KBUILD_CFLAGS += $(RETPOLINE_CFLAGS)
|
||||||
|
KBUILD_RUSTFLAGS += $(RETPOLINE_RUSTFLAGS)
|
||||||
# Additionally, avoid generating expensive indirect jumps which
|
# Additionally, avoid generating expensive indirect jumps which
|
||||||
# are subject to retpolines for small number of switch cases.
|
# are subject to retpolines for small number of switch cases.
|
||||||
# clang turns off jump table generation by default when under
|
# LLVM turns off jump table generation by default when under
|
||||||
# retpoline builds, however, gcc does not for x86. This has
|
# retpoline builds, however, gcc does not for x86. This has
|
||||||
# only been fixed starting from gcc stable version 8.4.0 and
|
# only been fixed starting from gcc stable version 8.4.0 and
|
||||||
# onwards, but not for older ones. See gcc bug #86952.
|
# onwards, but not for older ones. See gcc bug #86952.
|
||||||
@ -237,6 +242,10 @@ ifdef CONFIG_CALL_PADDING
|
|||||||
PADDING_CFLAGS := -fpatchable-function-entry=$(CONFIG_FUNCTION_PADDING_BYTES),$(CONFIG_FUNCTION_PADDING_BYTES)
|
PADDING_CFLAGS := -fpatchable-function-entry=$(CONFIG_FUNCTION_PADDING_BYTES),$(CONFIG_FUNCTION_PADDING_BYTES)
|
||||||
KBUILD_CFLAGS += $(PADDING_CFLAGS)
|
KBUILD_CFLAGS += $(PADDING_CFLAGS)
|
||||||
export PADDING_CFLAGS
|
export PADDING_CFLAGS
|
||||||
|
|
||||||
|
PADDING_RUSTFLAGS := -Zpatchable-function-entry=$(CONFIG_FUNCTION_PADDING_BYTES),$(CONFIG_FUNCTION_PADDING_BYTES)
|
||||||
|
KBUILD_RUSTFLAGS += $(PADDING_RUSTFLAGS)
|
||||||
|
export PADDING_RUSTFLAGS
|
||||||
endif
|
endif
|
||||||
|
|
||||||
KBUILD_LDFLAGS += -m elf_$(UTS_MACHINE)
|
KBUILD_LDFLAGS += -m elf_$(UTS_MACHINE)
|
||||||
|
19
init/Kconfig
19
init/Kconfig
@ -60,6 +60,13 @@ config LLD_VERSION
|
|||||||
default $(ld-version) if LD_IS_LLD
|
default $(ld-version) if LD_IS_LLD
|
||||||
default 0
|
default 0
|
||||||
|
|
||||||
|
config RUSTC_VERSION
|
||||||
|
int
|
||||||
|
default $(shell,$(srctree)/scripts/rustc-version.sh $(RUSTC))
|
||||||
|
help
|
||||||
|
It does not depend on `RUST` since that one may need to use the version
|
||||||
|
in a `depends on`.
|
||||||
|
|
||||||
config RUST_IS_AVAILABLE
|
config RUST_IS_AVAILABLE
|
||||||
def_bool $(success,$(srctree)/scripts/rust_is_available.sh)
|
def_bool $(success,$(srctree)/scripts/rust_is_available.sh)
|
||||||
help
|
help
|
||||||
@ -1935,12 +1942,14 @@ config RUST
|
|||||||
bool "Rust support"
|
bool "Rust support"
|
||||||
depends on HAVE_RUST
|
depends on HAVE_RUST
|
||||||
depends on RUST_IS_AVAILABLE
|
depends on RUST_IS_AVAILABLE
|
||||||
depends on !CFI_CLANG
|
|
||||||
depends on !MODVERSIONS
|
depends on !MODVERSIONS
|
||||||
depends on !GCC_PLUGINS
|
depends on !GCC_PLUGIN_RANDSTRUCT
|
||||||
depends on !RANDSTRUCT
|
depends on !RANDSTRUCT
|
||||||
depends on !SHADOW_CALL_STACK
|
|
||||||
depends on !DEBUG_INFO_BTF || PAHOLE_HAS_LANG_EXCLUDE
|
depends on !DEBUG_INFO_BTF || PAHOLE_HAS_LANG_EXCLUDE
|
||||||
|
depends on !CFI_CLANG || RUSTC_VERSION >= 107900 && $(cc-option,-fsanitize=kcfi -fsanitize-cfi-icall-experimental-normalize-integers)
|
||||||
|
select CFI_ICALL_NORMALIZE_INTEGERS if CFI_CLANG
|
||||||
|
depends on !CALL_PADDING || RUSTC_VERSION >= 108000
|
||||||
|
depends on !KASAN_SW_TAGS
|
||||||
help
|
help
|
||||||
Enables Rust support in the kernel.
|
Enables Rust support in the kernel.
|
||||||
|
|
||||||
@ -1957,7 +1966,9 @@ config RUST
|
|||||||
config RUSTC_VERSION_TEXT
|
config RUSTC_VERSION_TEXT
|
||||||
string
|
string
|
||||||
depends on RUST
|
depends on RUST
|
||||||
default "$(shell,$(RUSTC) --version 2>/dev/null)"
|
default "$(RUSTC_VERSION_TEXT)"
|
||||||
|
help
|
||||||
|
See `CC_VERSION_TEXT`.
|
||||||
|
|
||||||
config BINDGEN_VERSION_TEXT
|
config BINDGEN_VERSION_TEXT
|
||||||
string
|
string
|
||||||
|
@ -44,7 +44,8 @@ ifndef CONFIG_CC_HAS_KASAN_MEMINTRINSIC_PREFIX
|
|||||||
CFLAGS_KASAN_TEST += -fno-builtin
|
CFLAGS_KASAN_TEST += -fno-builtin
|
||||||
endif
|
endif
|
||||||
|
|
||||||
CFLAGS_kasan_test.o := $(CFLAGS_KASAN_TEST)
|
CFLAGS_kasan_test_c.o := $(CFLAGS_KASAN_TEST)
|
||||||
|
RUSTFLAGS_kasan_test_rust.o := $(RUSTFLAGS_KASAN)
|
||||||
CFLAGS_kasan_test_module.o := $(CFLAGS_KASAN_TEST)
|
CFLAGS_kasan_test_module.o := $(CFLAGS_KASAN_TEST)
|
||||||
|
|
||||||
obj-y := common.o report.o
|
obj-y := common.o report.o
|
||||||
@ -52,5 +53,10 @@ obj-$(CONFIG_KASAN_GENERIC) += init.o generic.o report_generic.o shadow.o quaran
|
|||||||
obj-$(CONFIG_KASAN_HW_TAGS) += hw_tags.o report_hw_tags.o tags.o report_tags.o
|
obj-$(CONFIG_KASAN_HW_TAGS) += hw_tags.o report_hw_tags.o tags.o report_tags.o
|
||||||
obj-$(CONFIG_KASAN_SW_TAGS) += init.o report_sw_tags.o shadow.o sw_tags.o tags.o report_tags.o
|
obj-$(CONFIG_KASAN_SW_TAGS) += init.o report_sw_tags.o shadow.o sw_tags.o tags.o report_tags.o
|
||||||
|
|
||||||
|
kasan_test-objs := kasan_test_c.o
|
||||||
|
ifdef CONFIG_RUST
|
||||||
|
kasan_test-objs += kasan_test_rust.o
|
||||||
|
endif
|
||||||
|
|
||||||
obj-$(CONFIG_KASAN_KUNIT_TEST) += kasan_test.o
|
obj-$(CONFIG_KASAN_KUNIT_TEST) += kasan_test.o
|
||||||
obj-$(CONFIG_KASAN_MODULE_TEST) += kasan_test_module.o
|
obj-$(CONFIG_KASAN_MODULE_TEST) += kasan_test_module.o
|
||||||
|
@ -555,6 +555,12 @@ static inline bool kasan_arch_is_ready(void) { return true; }
|
|||||||
void kasan_kunit_test_suite_start(void);
|
void kasan_kunit_test_suite_start(void);
|
||||||
void kasan_kunit_test_suite_end(void);
|
void kasan_kunit_test_suite_end(void);
|
||||||
|
|
||||||
|
#ifdef CONFIG_RUST
|
||||||
|
char kasan_test_rust_uaf(void);
|
||||||
|
#else
|
||||||
|
static inline char kasan_test_rust_uaf(void) { return '\0'; }
|
||||||
|
#endif
|
||||||
|
|
||||||
#else /* CONFIG_KASAN_KUNIT_TEST */
|
#else /* CONFIG_KASAN_KUNIT_TEST */
|
||||||
|
|
||||||
static inline void kasan_kunit_test_suite_start(void) { }
|
static inline void kasan_kunit_test_suite_start(void) { }
|
||||||
|
@ -1944,6 +1944,16 @@ static void match_all_mem_tag(struct kunit *test)
|
|||||||
kfree(ptr);
|
kfree(ptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check that Rust performing a use-after-free using `unsafe` is detected.
|
||||||
|
* This is a smoke test to make sure that Rust is being sanitized properly.
|
||||||
|
*/
|
||||||
|
static void rust_uaf(struct kunit *test)
|
||||||
|
{
|
||||||
|
KASAN_TEST_NEEDS_CONFIG_ON(test, CONFIG_RUST);
|
||||||
|
KUNIT_EXPECT_KASAN_FAIL(test, kasan_test_rust_uaf());
|
||||||
|
}
|
||||||
|
|
||||||
static struct kunit_case kasan_kunit_test_cases[] = {
|
static struct kunit_case kasan_kunit_test_cases[] = {
|
||||||
KUNIT_CASE(kmalloc_oob_right),
|
KUNIT_CASE(kmalloc_oob_right),
|
||||||
KUNIT_CASE(kmalloc_oob_left),
|
KUNIT_CASE(kmalloc_oob_left),
|
||||||
@ -2017,6 +2027,7 @@ static struct kunit_case kasan_kunit_test_cases[] = {
|
|||||||
KUNIT_CASE(match_all_not_assigned),
|
KUNIT_CASE(match_all_not_assigned),
|
||||||
KUNIT_CASE(match_all_ptr_tag),
|
KUNIT_CASE(match_all_ptr_tag),
|
||||||
KUNIT_CASE(match_all_mem_tag),
|
KUNIT_CASE(match_all_mem_tag),
|
||||||
|
KUNIT_CASE(rust_uaf),
|
||||||
{}
|
{}
|
||||||
};
|
};
|
||||||
|
|
21
mm/kasan/kasan_test_rust.rs
Normal file
21
mm/kasan/kasan_test_rust.rs
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
//! Helper crate for KASAN testing.
|
||||||
|
//!
|
||||||
|
//! Provides behavior to check the sanitization of Rust code.
|
||||||
|
|
||||||
|
use core::ptr::addr_of_mut;
|
||||||
|
use kernel::prelude::*;
|
||||||
|
|
||||||
|
/// Trivial UAF - allocate a big vector, grab a pointer partway through,
|
||||||
|
/// drop the vector, and touch it.
|
||||||
|
#[no_mangle]
|
||||||
|
pub extern "C" fn kasan_test_rust_uaf() -> u8 {
|
||||||
|
let mut v: Vec<u8> = Vec::new();
|
||||||
|
for _ in 0..4096 {
|
||||||
|
v.push(0x42, GFP_KERNEL).unwrap();
|
||||||
|
}
|
||||||
|
let ptr: *mut u8 = addr_of_mut!(v[2048]);
|
||||||
|
drop(v);
|
||||||
|
unsafe { *ptr }
|
||||||
|
}
|
@ -8,16 +8,16 @@ always-$(CONFIG_RUST) += exports_core_generated.h
|
|||||||
|
|
||||||
# Missing prototypes are expected in the helpers since these are exported
|
# Missing prototypes are expected in the helpers since these are exported
|
||||||
# for Rust only, thus there is no header nor prototypes.
|
# for Rust only, thus there is no header nor prototypes.
|
||||||
obj-$(CONFIG_RUST) += helpers.o
|
obj-$(CONFIG_RUST) += helpers/helpers.o
|
||||||
CFLAGS_REMOVE_helpers.o = -Wmissing-prototypes -Wmissing-declarations
|
CFLAGS_REMOVE_helpers/helpers.o = -Wmissing-prototypes -Wmissing-declarations
|
||||||
|
|
||||||
always-$(CONFIG_RUST) += libmacros.so
|
always-$(CONFIG_RUST) += libmacros.so
|
||||||
no-clean-files += libmacros.so
|
no-clean-files += libmacros.so
|
||||||
|
|
||||||
always-$(CONFIG_RUST) += bindings/bindings_generated.rs bindings/bindings_helpers_generated.rs
|
always-$(CONFIG_RUST) += bindings/bindings_generated.rs bindings/bindings_helpers_generated.rs
|
||||||
obj-$(CONFIG_RUST) += alloc.o bindings.o kernel.o
|
obj-$(CONFIG_RUST) += alloc.o bindings.o kernel.o
|
||||||
always-$(CONFIG_RUST) += exports_alloc_generated.h exports_bindings_generated.h \
|
always-$(CONFIG_RUST) += exports_alloc_generated.h exports_helpers_generated.h \
|
||||||
exports_kernel_generated.h
|
exports_bindings_generated.h exports_kernel_generated.h
|
||||||
|
|
||||||
always-$(CONFIG_RUST) += uapi/uapi_generated.rs
|
always-$(CONFIG_RUST) += uapi/uapi_generated.rs
|
||||||
obj-$(CONFIG_RUST) += uapi.o
|
obj-$(CONFIG_RUST) += uapi.o
|
||||||
@ -63,6 +63,7 @@ quiet_cmd_rustdoc = RUSTDOC $(if $(rustdoc_host),H, ) $<
|
|||||||
OBJTREE=$(abspath $(objtree)) \
|
OBJTREE=$(abspath $(objtree)) \
|
||||||
$(RUSTDOC) $(if $(rustdoc_host),$(rust_common_flags),$(rust_flags)) \
|
$(RUSTDOC) $(if $(rustdoc_host),$(rust_common_flags),$(rust_flags)) \
|
||||||
$(rustc_target_flags) -L$(objtree)/$(obj) \
|
$(rustc_target_flags) -L$(objtree)/$(obj) \
|
||||||
|
-Zunstable-options --generate-link-to-definition \
|
||||||
--output $(rustdoc_output) \
|
--output $(rustdoc_output) \
|
||||||
--crate-name $(subst rustdoc-,,$@) \
|
--crate-name $(subst rustdoc-,,$@) \
|
||||||
$(if $(rustdoc_host),,--sysroot=/dev/null) \
|
$(if $(rustdoc_host),,--sysroot=/dev/null) \
|
||||||
@ -270,7 +271,7 @@ quiet_cmd_bindgen = BINDGEN $@
|
|||||||
cmd_bindgen = \
|
cmd_bindgen = \
|
||||||
$(BINDGEN) $< $(bindgen_target_flags) \
|
$(BINDGEN) $< $(bindgen_target_flags) \
|
||||||
--use-core --with-derive-default --ctypes-prefix core::ffi --no-layout-tests \
|
--use-core --with-derive-default --ctypes-prefix core::ffi --no-layout-tests \
|
||||||
--no-debug '.*' \
|
--no-debug '.*' --enable-function-attribute-detection \
|
||||||
-o $@ -- $(bindgen_c_flags_final) -DMODULE \
|
-o $@ -- $(bindgen_c_flags_final) -DMODULE \
|
||||||
$(bindgen_target_cflags) $(bindgen_target_extra)
|
$(bindgen_target_cflags) $(bindgen_target_extra)
|
||||||
|
|
||||||
@ -299,13 +300,13 @@ $(obj)/bindings/bindings_helpers_generated.rs: private bindgen_target_cflags = \
|
|||||||
-I$(objtree)/$(obj) -Wno-missing-prototypes -Wno-missing-declarations
|
-I$(objtree)/$(obj) -Wno-missing-prototypes -Wno-missing-declarations
|
||||||
$(obj)/bindings/bindings_helpers_generated.rs: private bindgen_target_extra = ; \
|
$(obj)/bindings/bindings_helpers_generated.rs: private bindgen_target_extra = ; \
|
||||||
sed -Ei 's/pub fn rust_helper_([a-zA-Z0-9_]*)/#[link_name="rust_helper_\1"]\n pub fn \1/g' $@
|
sed -Ei 's/pub fn rust_helper_([a-zA-Z0-9_]*)/#[link_name="rust_helper_\1"]\n pub fn \1/g' $@
|
||||||
$(obj)/bindings/bindings_helpers_generated.rs: $(src)/helpers.c FORCE
|
$(obj)/bindings/bindings_helpers_generated.rs: $(src)/helpers/helpers.c FORCE
|
||||||
$(call if_changed_dep,bindgen)
|
$(call if_changed_dep,bindgen)
|
||||||
|
|
||||||
quiet_cmd_exports = EXPORTS $@
|
quiet_cmd_exports = EXPORTS $@
|
||||||
cmd_exports = \
|
cmd_exports = \
|
||||||
$(NM) -p --defined-only $< \
|
$(NM) -p --defined-only $< \
|
||||||
| awk '/ (T|R|D|B) / {printf "EXPORT_SYMBOL_RUST_GPL(%s);\n",$$3}' > $@
|
| awk '$$2~/(T|R|D|B)/ && $$3!~/__cfi/ {printf "EXPORT_SYMBOL_RUST_GPL(%s);\n",$$3}' > $@
|
||||||
|
|
||||||
$(obj)/exports_core_generated.h: $(obj)/core.o FORCE
|
$(obj)/exports_core_generated.h: $(obj)/core.o FORCE
|
||||||
$(call if_changed,exports)
|
$(call if_changed,exports)
|
||||||
@ -313,6 +314,18 @@ $(obj)/exports_core_generated.h: $(obj)/core.o FORCE
|
|||||||
$(obj)/exports_alloc_generated.h: $(obj)/alloc.o FORCE
|
$(obj)/exports_alloc_generated.h: $(obj)/alloc.o FORCE
|
||||||
$(call if_changed,exports)
|
$(call if_changed,exports)
|
||||||
|
|
||||||
|
# Even though Rust kernel modules should never use the bindings directly,
|
||||||
|
# symbols from the `bindings` crate and the C helpers need to be exported
|
||||||
|
# because Rust generics and inlined functions may not get their code generated
|
||||||
|
# in the crate where they are defined. Other helpers, called from non-inline
|
||||||
|
# functions, may not be exported, in principle. However, in general, the Rust
|
||||||
|
# compiler does not guarantee codegen will be performed for a non-inline
|
||||||
|
# function either. Therefore, we export all symbols from helpers and bindings.
|
||||||
|
# In the future, this may be revisited to reduce the number of exports after
|
||||||
|
# the compiler is informed about the places codegen is required.
|
||||||
|
$(obj)/exports_helpers_generated.h: $(obj)/helpers/helpers.o FORCE
|
||||||
|
$(call if_changed,exports)
|
||||||
|
|
||||||
$(obj)/exports_bindings_generated.h: $(obj)/bindings.o FORCE
|
$(obj)/exports_bindings_generated.h: $(obj)/bindings.o FORCE
|
||||||
$(call if_changed,exports)
|
$(call if_changed,exports)
|
||||||
|
|
||||||
@ -329,9 +342,7 @@ quiet_cmd_rustc_procmacro = $(RUSTC_OR_CLIPPY_QUIET) P $@
|
|||||||
--crate-name $(patsubst lib%.so,%,$(notdir $@)) $<
|
--crate-name $(patsubst lib%.so,%,$(notdir $@)) $<
|
||||||
|
|
||||||
# Procedural macros can only be used with the `rustc` that compiled it.
|
# Procedural macros can only be used with the `rustc` that compiled it.
|
||||||
# Therefore, to get `libmacros.so` automatically recompiled when the compiler
|
$(obj)/libmacros.so: $(src)/macros/lib.rs FORCE
|
||||||
# version changes, we add `core.o` as a dependency (even if it is not needed).
|
|
||||||
$(obj)/libmacros.so: $(src)/macros/lib.rs $(obj)/core.o FORCE
|
|
||||||
+$(call if_changed_dep,rustc_procmacro)
|
+$(call if_changed_dep,rustc_procmacro)
|
||||||
|
|
||||||
quiet_cmd_rustc_library = $(if $(skip_clippy),RUSTC,$(RUSTC_OR_CLIPPY_QUIET)) L $@
|
quiet_cmd_rustc_library = $(if $(skip_clippy),RUSTC,$(RUSTC_OR_CLIPPY_QUIET)) L $@
|
||||||
@ -344,7 +355,8 @@ quiet_cmd_rustc_library = $(if $(skip_clippy),RUSTC,$(RUSTC_OR_CLIPPY_QUIET)) L
|
|||||||
--crate-type rlib -L$(objtree)/$(obj) \
|
--crate-type rlib -L$(objtree)/$(obj) \
|
||||||
--crate-name $(patsubst %.o,%,$(notdir $@)) $< \
|
--crate-name $(patsubst %.o,%,$(notdir $@)) $< \
|
||||||
--sysroot=/dev/null \
|
--sysroot=/dev/null \
|
||||||
$(if $(rustc_objcopy),;$(OBJCOPY) $(rustc_objcopy) $@)
|
$(if $(rustc_objcopy),;$(OBJCOPY) $(rustc_objcopy) $@) \
|
||||||
|
$(cmd_objtool)
|
||||||
|
|
||||||
rust-analyzer:
|
rust-analyzer:
|
||||||
$(Q)$(srctree)/scripts/generate_rust_analyzer.py \
|
$(Q)$(srctree)/scripts/generate_rust_analyzer.py \
|
||||||
@ -366,44 +378,50 @@ ifneq ($(or $(CONFIG_ARM64),$(and $(CONFIG_RISCV),$(CONFIG_64BIT))),)
|
|||||||
__ashlti3 __lshrti3
|
__ashlti3 __lshrti3
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
define rule_rustc_library
|
||||||
|
$(call cmd_and_fixdep,rustc_library)
|
||||||
|
$(call cmd,gen_objtooldep)
|
||||||
|
endef
|
||||||
|
|
||||||
$(obj)/core.o: private skip_clippy = 1
|
$(obj)/core.o: private skip_clippy = 1
|
||||||
$(obj)/core.o: private skip_flags = -Wunreachable_pub
|
$(obj)/core.o: private skip_flags = -Wunreachable_pub
|
||||||
$(obj)/core.o: private rustc_objcopy = $(foreach sym,$(redirect-intrinsics),--redefine-sym $(sym)=__rust$(sym))
|
$(obj)/core.o: private rustc_objcopy = $(foreach sym,$(redirect-intrinsics),--redefine-sym $(sym)=__rust$(sym))
|
||||||
$(obj)/core.o: private rustc_target_flags = $(core-cfgs)
|
$(obj)/core.o: private rustc_target_flags = $(core-cfgs)
|
||||||
$(obj)/core.o: $(RUST_LIB_SRC)/core/src/lib.rs FORCE
|
$(obj)/core.o: $(RUST_LIB_SRC)/core/src/lib.rs \
|
||||||
+$(call if_changed_dep,rustc_library)
|
$(wildcard $(objtree)/include/config/RUSTC_VERSION_TEXT) FORCE
|
||||||
|
+$(call if_changed_rule,rustc_library)
|
||||||
ifneq ($(or $(CONFIG_X86_64),$(CONFIG_X86_32)),)
|
ifneq ($(or $(CONFIG_X86_64),$(CONFIG_X86_32)),)
|
||||||
$(obj)/core.o: scripts/target.json
|
$(obj)/core.o: scripts/target.json
|
||||||
endif
|
endif
|
||||||
|
|
||||||
$(obj)/compiler_builtins.o: private rustc_objcopy = -w -W '__*'
|
$(obj)/compiler_builtins.o: private rustc_objcopy = -w -W '__*'
|
||||||
$(obj)/compiler_builtins.o: $(src)/compiler_builtins.rs $(obj)/core.o FORCE
|
$(obj)/compiler_builtins.o: $(src)/compiler_builtins.rs $(obj)/core.o FORCE
|
||||||
+$(call if_changed_dep,rustc_library)
|
+$(call if_changed_rule,rustc_library)
|
||||||
|
|
||||||
$(obj)/alloc.o: private skip_clippy = 1
|
$(obj)/alloc.o: private skip_clippy = 1
|
||||||
$(obj)/alloc.o: private skip_flags = -Wunreachable_pub
|
$(obj)/alloc.o: private skip_flags = -Wunreachable_pub
|
||||||
$(obj)/alloc.o: private rustc_target_flags = $(alloc-cfgs)
|
$(obj)/alloc.o: private rustc_target_flags = $(alloc-cfgs)
|
||||||
$(obj)/alloc.o: $(RUST_LIB_SRC)/alloc/src/lib.rs $(obj)/compiler_builtins.o FORCE
|
$(obj)/alloc.o: $(RUST_LIB_SRC)/alloc/src/lib.rs $(obj)/compiler_builtins.o FORCE
|
||||||
+$(call if_changed_dep,rustc_library)
|
+$(call if_changed_rule,rustc_library)
|
||||||
|
|
||||||
$(obj)/build_error.o: $(src)/build_error.rs $(obj)/compiler_builtins.o FORCE
|
$(obj)/build_error.o: $(src)/build_error.rs $(obj)/compiler_builtins.o FORCE
|
||||||
+$(call if_changed_dep,rustc_library)
|
+$(call if_changed_rule,rustc_library)
|
||||||
|
|
||||||
$(obj)/bindings.o: $(src)/bindings/lib.rs \
|
$(obj)/bindings.o: $(src)/bindings/lib.rs \
|
||||||
$(obj)/compiler_builtins.o \
|
$(obj)/compiler_builtins.o \
|
||||||
$(obj)/bindings/bindings_generated.rs \
|
$(obj)/bindings/bindings_generated.rs \
|
||||||
$(obj)/bindings/bindings_helpers_generated.rs FORCE
|
$(obj)/bindings/bindings_helpers_generated.rs FORCE
|
||||||
+$(call if_changed_dep,rustc_library)
|
+$(call if_changed_rule,rustc_library)
|
||||||
|
|
||||||
$(obj)/uapi.o: $(src)/uapi/lib.rs \
|
$(obj)/uapi.o: $(src)/uapi/lib.rs \
|
||||||
$(obj)/compiler_builtins.o \
|
$(obj)/compiler_builtins.o \
|
||||||
$(obj)/uapi/uapi_generated.rs FORCE
|
$(obj)/uapi/uapi_generated.rs FORCE
|
||||||
+$(call if_changed_dep,rustc_library)
|
+$(call if_changed_rule,rustc_library)
|
||||||
|
|
||||||
$(obj)/kernel.o: private rustc_target_flags = --extern alloc \
|
$(obj)/kernel.o: private rustc_target_flags = --extern alloc \
|
||||||
--extern build_error --extern macros --extern bindings --extern uapi
|
--extern build_error --extern macros --extern bindings --extern uapi
|
||||||
$(obj)/kernel.o: $(src)/kernel/lib.rs $(obj)/alloc.o $(obj)/build_error.o \
|
$(obj)/kernel.o: $(src)/kernel/lib.rs $(obj)/alloc.o $(obj)/build_error.o \
|
||||||
$(obj)/libmacros.so $(obj)/bindings.o $(obj)/uapi.o FORCE
|
$(obj)/libmacros.so $(obj)/bindings.o $(obj)/uapi.o FORCE
|
||||||
+$(call if_changed_dep,rustc_library)
|
+$(call if_changed_rule,rustc_library)
|
||||||
|
|
||||||
endif # CONFIG_RUST
|
endif # CONFIG_RUST
|
||||||
|
@ -7,8 +7,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <kunit/test.h>
|
#include <kunit/test.h>
|
||||||
#include <linux/blk_types.h>
|
|
||||||
#include <linux/blk-mq.h>
|
#include <linux/blk-mq.h>
|
||||||
|
#include <linux/blk_types.h>
|
||||||
#include <linux/blkdev.h>
|
#include <linux/blkdev.h>
|
||||||
#include <linux/errname.h>
|
#include <linux/errname.h>
|
||||||
#include <linux/ethtool.h>
|
#include <linux/ethtool.h>
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
|
|
||||||
#include "exports_core_generated.h"
|
#include "exports_core_generated.h"
|
||||||
#include "exports_alloc_generated.h"
|
#include "exports_alloc_generated.h"
|
||||||
|
#include "exports_helpers_generated.h"
|
||||||
#include "exports_bindings_generated.h"
|
#include "exports_bindings_generated.h"
|
||||||
#include "exports_kernel_generated.h"
|
#include "exports_kernel_generated.h"
|
||||||
|
|
||||||
|
239
rust/helpers.c
239
rust/helpers.c
@ -1,239 +0,0 @@
|
|||||||
// SPDX-License-Identifier: GPL-2.0
|
|
||||||
/*
|
|
||||||
* Non-trivial C macros cannot be used in Rust. Similarly, inlined C functions
|
|
||||||
* cannot be called either. This file explicitly creates functions ("helpers")
|
|
||||||
* that wrap those so that they can be called from Rust.
|
|
||||||
*
|
|
||||||
* Even though Rust kernel modules should never use the bindings directly, some
|
|
||||||
* of these helpers need to be exported because Rust generics and inlined
|
|
||||||
* functions may not get their code generated in the crate where they are
|
|
||||||
* defined. Other helpers, called from non-inline functions, may not be
|
|
||||||
* exported, in principle. However, in general, the Rust compiler does not
|
|
||||||
* guarantee codegen will be performed for a non-inline function either.
|
|
||||||
* Therefore, this file exports all the helpers. In the future, this may be
|
|
||||||
* revisited to reduce the number of exports after the compiler is informed
|
|
||||||
* about the places codegen is required.
|
|
||||||
*
|
|
||||||
* All symbols are exported as GPL-only to guarantee no GPL-only feature is
|
|
||||||
* accidentally exposed.
|
|
||||||
*
|
|
||||||
* Sorted alphabetically.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <kunit/test-bug.h>
|
|
||||||
#include <linux/bug.h>
|
|
||||||
#include <linux/build_bug.h>
|
|
||||||
#include <linux/device.h>
|
|
||||||
#include <linux/err.h>
|
|
||||||
#include <linux/errname.h>
|
|
||||||
#include <linux/gfp.h>
|
|
||||||
#include <linux/highmem.h>
|
|
||||||
#include <linux/mutex.h>
|
|
||||||
#include <linux/refcount.h>
|
|
||||||
#include <linux/sched/signal.h>
|
|
||||||
#include <linux/slab.h>
|
|
||||||
#include <linux/spinlock.h>
|
|
||||||
#include <linux/wait.h>
|
|
||||||
#include <linux/workqueue.h>
|
|
||||||
|
|
||||||
__noreturn void rust_helper_BUG(void)
|
|
||||||
{
|
|
||||||
BUG();
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_BUG);
|
|
||||||
|
|
||||||
unsigned long rust_helper_copy_from_user(void *to, const void __user *from,
|
|
||||||
unsigned long n)
|
|
||||||
{
|
|
||||||
return copy_from_user(to, from, n);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_copy_from_user);
|
|
||||||
|
|
||||||
unsigned long rust_helper_copy_to_user(void __user *to, const void *from,
|
|
||||||
unsigned long n)
|
|
||||||
{
|
|
||||||
return copy_to_user(to, from, n);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_copy_to_user);
|
|
||||||
|
|
||||||
void rust_helper_mutex_lock(struct mutex *lock)
|
|
||||||
{
|
|
||||||
mutex_lock(lock);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_mutex_lock);
|
|
||||||
|
|
||||||
void rust_helper___spin_lock_init(spinlock_t *lock, const char *name,
|
|
||||||
struct lock_class_key *key)
|
|
||||||
{
|
|
||||||
#ifdef CONFIG_DEBUG_SPINLOCK
|
|
||||||
__raw_spin_lock_init(spinlock_check(lock), name, key, LD_WAIT_CONFIG);
|
|
||||||
#else
|
|
||||||
spin_lock_init(lock);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper___spin_lock_init);
|
|
||||||
|
|
||||||
void rust_helper_spin_lock(spinlock_t *lock)
|
|
||||||
{
|
|
||||||
spin_lock(lock);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_spin_lock);
|
|
||||||
|
|
||||||
void rust_helper_spin_unlock(spinlock_t *lock)
|
|
||||||
{
|
|
||||||
spin_unlock(lock);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_spin_unlock);
|
|
||||||
|
|
||||||
void rust_helper_init_wait(struct wait_queue_entry *wq_entry)
|
|
||||||
{
|
|
||||||
init_wait(wq_entry);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_init_wait);
|
|
||||||
|
|
||||||
int rust_helper_signal_pending(struct task_struct *t)
|
|
||||||
{
|
|
||||||
return signal_pending(t);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_signal_pending);
|
|
||||||
|
|
||||||
struct page *rust_helper_alloc_pages(gfp_t gfp_mask, unsigned int order)
|
|
||||||
{
|
|
||||||
return alloc_pages(gfp_mask, order);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_alloc_pages);
|
|
||||||
|
|
||||||
void *rust_helper_kmap_local_page(struct page *page)
|
|
||||||
{
|
|
||||||
return kmap_local_page(page);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_kmap_local_page);
|
|
||||||
|
|
||||||
void rust_helper_kunmap_local(const void *addr)
|
|
||||||
{
|
|
||||||
kunmap_local(addr);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_kunmap_local);
|
|
||||||
|
|
||||||
refcount_t rust_helper_REFCOUNT_INIT(int n)
|
|
||||||
{
|
|
||||||
return (refcount_t)REFCOUNT_INIT(n);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_REFCOUNT_INIT);
|
|
||||||
|
|
||||||
void rust_helper_refcount_inc(refcount_t *r)
|
|
||||||
{
|
|
||||||
refcount_inc(r);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_refcount_inc);
|
|
||||||
|
|
||||||
bool rust_helper_refcount_dec_and_test(refcount_t *r)
|
|
||||||
{
|
|
||||||
return refcount_dec_and_test(r);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_refcount_dec_and_test);
|
|
||||||
|
|
||||||
__force void *rust_helper_ERR_PTR(long err)
|
|
||||||
{
|
|
||||||
return ERR_PTR(err);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_ERR_PTR);
|
|
||||||
|
|
||||||
bool rust_helper_IS_ERR(__force const void *ptr)
|
|
||||||
{
|
|
||||||
return IS_ERR(ptr);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_IS_ERR);
|
|
||||||
|
|
||||||
long rust_helper_PTR_ERR(__force const void *ptr)
|
|
||||||
{
|
|
||||||
return PTR_ERR(ptr);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_PTR_ERR);
|
|
||||||
|
|
||||||
const char *rust_helper_errname(int err)
|
|
||||||
{
|
|
||||||
return errname(err);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_errname);
|
|
||||||
|
|
||||||
struct task_struct *rust_helper_get_current(void)
|
|
||||||
{
|
|
||||||
return current;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_get_current);
|
|
||||||
|
|
||||||
void rust_helper_get_task_struct(struct task_struct *t)
|
|
||||||
{
|
|
||||||
get_task_struct(t);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_get_task_struct);
|
|
||||||
|
|
||||||
void rust_helper_put_task_struct(struct task_struct *t)
|
|
||||||
{
|
|
||||||
put_task_struct(t);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_put_task_struct);
|
|
||||||
|
|
||||||
struct kunit *rust_helper_kunit_get_current_test(void)
|
|
||||||
{
|
|
||||||
return kunit_get_current_test();
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_kunit_get_current_test);
|
|
||||||
|
|
||||||
void rust_helper_init_work_with_key(struct work_struct *work, work_func_t func,
|
|
||||||
bool onstack, const char *name,
|
|
||||||
struct lock_class_key *key)
|
|
||||||
{
|
|
||||||
__init_work(work, onstack);
|
|
||||||
work->data = (atomic_long_t)WORK_DATA_INIT();
|
|
||||||
lockdep_init_map(&work->lockdep_map, name, key, 0);
|
|
||||||
INIT_LIST_HEAD(&work->entry);
|
|
||||||
work->func = func;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_init_work_with_key);
|
|
||||||
|
|
||||||
void * __must_check __realloc_size(2)
|
|
||||||
rust_helper_krealloc(const void *objp, size_t new_size, gfp_t flags)
|
|
||||||
{
|
|
||||||
return krealloc(objp, new_size, flags);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_krealloc);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* `bindgen` binds the C `size_t` type as the Rust `usize` type, so we can
|
|
||||||
* use it in contexts where Rust expects a `usize` like slice (array) indices.
|
|
||||||
* `usize` is defined to be the same as C's `uintptr_t` type (can hold any
|
|
||||||
* pointer) but not necessarily the same as `size_t` (can hold the size of any
|
|
||||||
* single object). Most modern platforms use the same concrete integer type for
|
|
||||||
* both of them, but in case we find ourselves on a platform where
|
|
||||||
* that's not true, fail early instead of risking ABI or
|
|
||||||
* integer-overflow issues.
|
|
||||||
*
|
|
||||||
* If your platform fails this assertion, it means that you are in
|
|
||||||
* danger of integer-overflow bugs (even if you attempt to add
|
|
||||||
* `--no-size_t-is-usize`). It may be easiest to change the kernel ABI on
|
|
||||||
* your platform such that `size_t` matches `uintptr_t` (i.e., to increase
|
|
||||||
* `size_t`, because `uintptr_t` has to be at least as big as `size_t`).
|
|
||||||
*/
|
|
||||||
static_assert(
|
|
||||||
sizeof(size_t) == sizeof(uintptr_t) &&
|
|
||||||
__alignof__(size_t) == __alignof__(uintptr_t),
|
|
||||||
"Rust code expects C `size_t` to match Rust `usize`"
|
|
||||||
);
|
|
||||||
|
|
||||||
// This will soon be moved to a separate file, so no need to merge with above.
|
|
||||||
#include <linux/blk-mq.h>
|
|
||||||
#include <linux/blkdev.h>
|
|
||||||
|
|
||||||
void *rust_helper_blk_mq_rq_to_pdu(struct request *rq)
|
|
||||||
{
|
|
||||||
return blk_mq_rq_to_pdu(rq);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_blk_mq_rq_to_pdu);
|
|
||||||
|
|
||||||
struct request *rust_helper_blk_mq_rq_from_pdu(void *pdu)
|
|
||||||
{
|
|
||||||
return blk_mq_rq_from_pdu(pdu);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(rust_helper_blk_mq_rq_from_pdu);
|
|
14
rust/helpers/blk.c
Normal file
14
rust/helpers/blk.c
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/blk-mq.h>
|
||||||
|
#include <linux/blkdev.h>
|
||||||
|
|
||||||
|
void *rust_helper_blk_mq_rq_to_pdu(struct request *rq)
|
||||||
|
{
|
||||||
|
return blk_mq_rq_to_pdu(rq);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct request *rust_helper_blk_mq_rq_from_pdu(void *pdu)
|
||||||
|
{
|
||||||
|
return blk_mq_rq_from_pdu(pdu);
|
||||||
|
}
|
8
rust/helpers/bug.c
Normal file
8
rust/helpers/bug.c
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/bug.h>
|
||||||
|
|
||||||
|
__noreturn void rust_helper_BUG(void)
|
||||||
|
{
|
||||||
|
BUG();
|
||||||
|
}
|
25
rust/helpers/build_assert.c
Normal file
25
rust/helpers/build_assert.c
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/build_bug.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
* `bindgen` binds the C `size_t` type as the Rust `usize` type, so we can
|
||||||
|
* use it in contexts where Rust expects a `usize` like slice (array) indices.
|
||||||
|
* `usize` is defined to be the same as C's `uintptr_t` type (can hold any
|
||||||
|
* pointer) but not necessarily the same as `size_t` (can hold the size of any
|
||||||
|
* single object). Most modern platforms use the same concrete integer type for
|
||||||
|
* both of them, but in case we find ourselves on a platform where
|
||||||
|
* that's not true, fail early instead of risking ABI or
|
||||||
|
* integer-overflow issues.
|
||||||
|
*
|
||||||
|
* If your platform fails this assertion, it means that you are in
|
||||||
|
* danger of integer-overflow bugs (even if you attempt to add
|
||||||
|
* `--no-size_t-is-usize`). It may be easiest to change the kernel ABI on
|
||||||
|
* your platform such that `size_t` matches `uintptr_t` (i.e., to increase
|
||||||
|
* `size_t`, because `uintptr_t` has to be at least as big as `size_t`).
|
||||||
|
*/
|
||||||
|
static_assert(
|
||||||
|
sizeof(size_t) == sizeof(uintptr_t) &&
|
||||||
|
__alignof__(size_t) == __alignof__(uintptr_t),
|
||||||
|
"Rust code expects C `size_t` to match Rust `usize`"
|
||||||
|
);
|
9
rust/helpers/build_bug.c
Normal file
9
rust/helpers/build_bug.c
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/export.h>
|
||||||
|
#include <linux/errname.h>
|
||||||
|
|
||||||
|
const char *rust_helper_errname(int err)
|
||||||
|
{
|
||||||
|
return errname(err);
|
||||||
|
}
|
19
rust/helpers/err.c
Normal file
19
rust/helpers/err.c
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/err.h>
|
||||||
|
#include <linux/export.h>
|
||||||
|
|
||||||
|
__force void *rust_helper_ERR_PTR(long err)
|
||||||
|
{
|
||||||
|
return ERR_PTR(err);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool rust_helper_IS_ERR(__force const void *ptr)
|
||||||
|
{
|
||||||
|
return IS_ERR(ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
long rust_helper_PTR_ERR(__force const void *ptr)
|
||||||
|
{
|
||||||
|
return PTR_ERR(ptr);
|
||||||
|
}
|
26
rust/helpers/helpers.c
Normal file
26
rust/helpers/helpers.c
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
/*
|
||||||
|
* Non-trivial C macros cannot be used in Rust. Similarly, inlined C functions
|
||||||
|
* cannot be called either. This file explicitly creates functions ("helpers")
|
||||||
|
* that wrap those so that they can be called from Rust.
|
||||||
|
*
|
||||||
|
* Sorted alphabetically.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "blk.c"
|
||||||
|
#include "bug.c"
|
||||||
|
#include "build_assert.c"
|
||||||
|
#include "build_bug.c"
|
||||||
|
#include "err.c"
|
||||||
|
#include "kunit.c"
|
||||||
|
#include "mutex.c"
|
||||||
|
#include "page.c"
|
||||||
|
#include "rbtree.c"
|
||||||
|
#include "refcount.c"
|
||||||
|
#include "signal.c"
|
||||||
|
#include "slab.c"
|
||||||
|
#include "spinlock.c"
|
||||||
|
#include "task.c"
|
||||||
|
#include "uaccess.c"
|
||||||
|
#include "wait.c"
|
||||||
|
#include "workqueue.c"
|
9
rust/helpers/kunit.c
Normal file
9
rust/helpers/kunit.c
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <kunit/test-bug.h>
|
||||||
|
#include <linux/export.h>
|
||||||
|
|
||||||
|
struct kunit *rust_helper_kunit_get_current_test(void)
|
||||||
|
{
|
||||||
|
return kunit_get_current_test();
|
||||||
|
}
|
9
rust/helpers/mutex.c
Normal file
9
rust/helpers/mutex.c
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/export.h>
|
||||||
|
#include <linux/mutex.h>
|
||||||
|
|
||||||
|
void rust_helper_mutex_lock(struct mutex *lock)
|
||||||
|
{
|
||||||
|
mutex_lock(lock);
|
||||||
|
}
|
19
rust/helpers/page.c
Normal file
19
rust/helpers/page.c
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/gfp.h>
|
||||||
|
#include <linux/highmem.h>
|
||||||
|
|
||||||
|
struct page *rust_helper_alloc_pages(gfp_t gfp_mask, unsigned int order)
|
||||||
|
{
|
||||||
|
return alloc_pages(gfp_mask, order);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *rust_helper_kmap_local_page(struct page *page)
|
||||||
|
{
|
||||||
|
return kmap_local_page(page);
|
||||||
|
}
|
||||||
|
|
||||||
|
void rust_helper_kunmap_local(const void *addr)
|
||||||
|
{
|
||||||
|
kunmap_local(addr);
|
||||||
|
}
|
9
rust/helpers/rbtree.c
Normal file
9
rust/helpers/rbtree.c
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/rbtree.h>
|
||||||
|
|
||||||
|
void rust_helper_rb_link_node(struct rb_node *node, struct rb_node *parent,
|
||||||
|
struct rb_node **rb_link)
|
||||||
|
{
|
||||||
|
rb_link_node(node, parent, rb_link);
|
||||||
|
}
|
19
rust/helpers/refcount.c
Normal file
19
rust/helpers/refcount.c
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/export.h>
|
||||||
|
#include <linux/refcount.h>
|
||||||
|
|
||||||
|
refcount_t rust_helper_REFCOUNT_INIT(int n)
|
||||||
|
{
|
||||||
|
return (refcount_t)REFCOUNT_INIT(n);
|
||||||
|
}
|
||||||
|
|
||||||
|
void rust_helper_refcount_inc(refcount_t *r)
|
||||||
|
{
|
||||||
|
refcount_inc(r);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool rust_helper_refcount_dec_and_test(refcount_t *r)
|
||||||
|
{
|
||||||
|
return refcount_dec_and_test(r);
|
||||||
|
}
|
9
rust/helpers/signal.c
Normal file
9
rust/helpers/signal.c
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/export.h>
|
||||||
|
#include <linux/sched/signal.h>
|
||||||
|
|
||||||
|
int rust_helper_signal_pending(struct task_struct *t)
|
||||||
|
{
|
||||||
|
return signal_pending(t);
|
||||||
|
}
|
9
rust/helpers/slab.c
Normal file
9
rust/helpers/slab.c
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/slab.h>
|
||||||
|
|
||||||
|
void * __must_check __realloc_size(2)
|
||||||
|
rust_helper_krealloc(const void *objp, size_t new_size, gfp_t flags)
|
||||||
|
{
|
||||||
|
return krealloc(objp, new_size, flags);
|
||||||
|
}
|
24
rust/helpers/spinlock.c
Normal file
24
rust/helpers/spinlock.c
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/export.h>
|
||||||
|
#include <linux/spinlock.h>
|
||||||
|
|
||||||
|
void rust_helper___spin_lock_init(spinlock_t *lock, const char *name,
|
||||||
|
struct lock_class_key *key)
|
||||||
|
{
|
||||||
|
#ifdef CONFIG_DEBUG_SPINLOCK
|
||||||
|
__raw_spin_lock_init(spinlock_check(lock), name, key, LD_WAIT_CONFIG);
|
||||||
|
#else
|
||||||
|
spin_lock_init(lock);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void rust_helper_spin_lock(spinlock_t *lock)
|
||||||
|
{
|
||||||
|
spin_lock(lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
void rust_helper_spin_unlock(spinlock_t *lock)
|
||||||
|
{
|
||||||
|
spin_unlock(lock);
|
||||||
|
}
|
19
rust/helpers/task.c
Normal file
19
rust/helpers/task.c
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/export.h>
|
||||||
|
#include <linux/sched/task.h>
|
||||||
|
|
||||||
|
struct task_struct *rust_helper_get_current(void)
|
||||||
|
{
|
||||||
|
return current;
|
||||||
|
}
|
||||||
|
|
||||||
|
void rust_helper_get_task_struct(struct task_struct *t)
|
||||||
|
{
|
||||||
|
get_task_struct(t);
|
||||||
|
}
|
||||||
|
|
||||||
|
void rust_helper_put_task_struct(struct task_struct *t)
|
||||||
|
{
|
||||||
|
put_task_struct(t);
|
||||||
|
}
|
15
rust/helpers/uaccess.c
Normal file
15
rust/helpers/uaccess.c
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/uaccess.h>
|
||||||
|
|
||||||
|
unsigned long rust_helper_copy_from_user(void *to, const void __user *from,
|
||||||
|
unsigned long n)
|
||||||
|
{
|
||||||
|
return copy_from_user(to, from, n);
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned long rust_helper_copy_to_user(void __user *to, const void *from,
|
||||||
|
unsigned long n)
|
||||||
|
{
|
||||||
|
return copy_to_user(to, from, n);
|
||||||
|
}
|
9
rust/helpers/wait.c
Normal file
9
rust/helpers/wait.c
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/export.h>
|
||||||
|
#include <linux/wait.h>
|
||||||
|
|
||||||
|
void rust_helper_init_wait(struct wait_queue_entry *wq_entry)
|
||||||
|
{
|
||||||
|
init_wait(wq_entry);
|
||||||
|
}
|
15
rust/helpers/workqueue.c
Normal file
15
rust/helpers/workqueue.c
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
#include <linux/export.h>
|
||||||
|
#include <linux/workqueue.h>
|
||||||
|
|
||||||
|
void rust_helper_init_work_with_key(struct work_struct *work, work_func_t func,
|
||||||
|
bool onstack, const char *name,
|
||||||
|
struct lock_class_key *key)
|
||||||
|
{
|
||||||
|
__init_work(work, onstack);
|
||||||
|
work->data = (atomic_long_t)WORK_DATA_INIT();
|
||||||
|
lockdep_init_map(&work->lockdep_map, name, key, 0);
|
||||||
|
INIT_LIST_HEAD(&work->entry);
|
||||||
|
work->func = func;
|
||||||
|
}
|
@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
use super::{AllocError, Flags};
|
use super::{AllocError, Flags};
|
||||||
use alloc::boxed::Box;
|
use alloc::boxed::Box;
|
||||||
use core::mem::MaybeUninit;
|
use core::{mem::MaybeUninit, ptr, result::Result};
|
||||||
|
|
||||||
/// Extensions to [`Box`].
|
/// Extensions to [`Box`].
|
||||||
pub trait BoxExt<T>: Sized {
|
pub trait BoxExt<T>: Sized {
|
||||||
@ -17,6 +17,24 @@ pub trait BoxExt<T>: Sized {
|
|||||||
///
|
///
|
||||||
/// The allocation may fail, in which case an error is returned.
|
/// The allocation may fail, in which case an error is returned.
|
||||||
fn new_uninit(flags: Flags) -> Result<Box<MaybeUninit<T>>, AllocError>;
|
fn new_uninit(flags: Flags) -> Result<Box<MaybeUninit<T>>, AllocError>;
|
||||||
|
|
||||||
|
/// Drops the contents, but keeps the allocation.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// use kernel::alloc::{flags, box_ext::BoxExt};
|
||||||
|
/// let value = Box::new([0; 32], flags::GFP_KERNEL)?;
|
||||||
|
/// assert_eq!(*value, [0; 32]);
|
||||||
|
/// let mut value = Box::drop_contents(value);
|
||||||
|
/// // Now we can re-use `value`:
|
||||||
|
/// value.write([1; 32]);
|
||||||
|
/// // SAFETY: We just wrote to it.
|
||||||
|
/// let value = unsafe { value.assume_init() };
|
||||||
|
/// assert_eq!(*value, [1; 32]);
|
||||||
|
/// # Ok::<(), Error>(())
|
||||||
|
/// ```
|
||||||
|
fn drop_contents(this: Self) -> Box<MaybeUninit<T>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> BoxExt<T> for Box<T> {
|
impl<T> BoxExt<T> for Box<T> {
|
||||||
@ -55,4 +73,17 @@ impl<T> BoxExt<T> for Box<T> {
|
|||||||
// zero-sized types, we use `NonNull::dangling`.
|
// zero-sized types, we use `NonNull::dangling`.
|
||||||
Ok(unsafe { Box::from_raw(ptr) })
|
Ok(unsafe { Box::from_raw(ptr) })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn drop_contents(this: Self) -> Box<MaybeUninit<T>> {
|
||||||
|
let ptr = Box::into_raw(this);
|
||||||
|
// SAFETY: `ptr` is valid, because it came from `Box::into_raw`.
|
||||||
|
unsafe { ptr::drop_in_place(ptr) };
|
||||||
|
|
||||||
|
// CAST: `MaybeUninit<T>` is a transparent wrapper of `T`.
|
||||||
|
let ptr = ptr.cast::<MaybeUninit<T>>();
|
||||||
|
|
||||||
|
// SAFETY: `ptr` is valid for writes, because it came from `Box::into_raw` and it is valid for
|
||||||
|
// reads, since the pointer came from `Box::into_raw` and the type is `MaybeUninit<T>`.
|
||||||
|
unsafe { Box::from_raw(ptr) }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -135,8 +135,11 @@ impl Error {
|
|||||||
/// Returns the error encoded as a pointer.
|
/// Returns the error encoded as a pointer.
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
pub(crate) fn to_ptr<T>(self) -> *mut T {
|
pub(crate) fn to_ptr<T>(self) -> *mut T {
|
||||||
|
#[cfg_attr(target_pointer_width = "32", allow(clippy::useless_conversion))]
|
||||||
// SAFETY: `self.0` is a valid error due to its invariant.
|
// SAFETY: `self.0` is a valid error due to its invariant.
|
||||||
unsafe { bindings::ERR_PTR(self.0.into()) as *mut _ }
|
unsafe {
|
||||||
|
bindings::ERR_PTR(self.0.into()) as *mut _
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns a string representing the error, if one exists.
|
/// Returns a string representing the error, if one exists.
|
||||||
|
@ -213,6 +213,7 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
alloc::{box_ext::BoxExt, AllocError, Flags},
|
alloc::{box_ext::BoxExt, AllocError, Flags},
|
||||||
error::{self, Error},
|
error::{self, Error},
|
||||||
|
sync::Arc,
|
||||||
sync::UniqueArc,
|
sync::UniqueArc,
|
||||||
types::{Opaque, ScopeGuard},
|
types::{Opaque, ScopeGuard},
|
||||||
};
|
};
|
||||||
@ -742,6 +743,74 @@ macro_rules! try_init {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Asserts that a field on a struct using `#[pin_data]` is marked with `#[pin]` ie. that it is
|
||||||
|
/// structurally pinned.
|
||||||
|
///
|
||||||
|
/// # Example
|
||||||
|
///
|
||||||
|
/// This will succeed:
|
||||||
|
/// ```
|
||||||
|
/// use kernel::assert_pinned;
|
||||||
|
/// #[pin_data]
|
||||||
|
/// struct MyStruct {
|
||||||
|
/// #[pin]
|
||||||
|
/// some_field: u64,
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// assert_pinned!(MyStruct, some_field, u64);
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// This will fail:
|
||||||
|
// TODO: replace with `compile_fail` when supported.
|
||||||
|
/// ```ignore
|
||||||
|
/// use kernel::assert_pinned;
|
||||||
|
/// #[pin_data]
|
||||||
|
/// struct MyStruct {
|
||||||
|
/// some_field: u64,
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// assert_pinned!(MyStruct, some_field, u64);
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// Some uses of the macro may trigger the `can't use generic parameters from outer item` error. To
|
||||||
|
/// work around this, you may pass the `inline` parameter to the macro. The `inline` parameter can
|
||||||
|
/// only be used when the macro is invoked from a function body.
|
||||||
|
/// ```
|
||||||
|
/// use kernel::assert_pinned;
|
||||||
|
/// #[pin_data]
|
||||||
|
/// struct Foo<T> {
|
||||||
|
/// #[pin]
|
||||||
|
/// elem: T,
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// impl<T> Foo<T> {
|
||||||
|
/// fn project(self: Pin<&mut Self>) -> Pin<&mut T> {
|
||||||
|
/// assert_pinned!(Foo<T>, elem, T, inline);
|
||||||
|
///
|
||||||
|
/// // SAFETY: The field is structurally pinned.
|
||||||
|
/// unsafe { self.map_unchecked_mut(|me| &mut me.elem) }
|
||||||
|
/// }
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! assert_pinned {
|
||||||
|
($ty:ty, $field:ident, $field_ty:ty, inline) => {
|
||||||
|
let _ = move |ptr: *mut $field_ty| {
|
||||||
|
// SAFETY: This code is unreachable.
|
||||||
|
let data = unsafe { <$ty as $crate::init::__internal::HasPinData>::__pin_data() };
|
||||||
|
let init = $crate::init::__internal::AlwaysFail::<$field_ty>::new();
|
||||||
|
// SAFETY: This code is unreachable.
|
||||||
|
unsafe { data.$field(ptr, init) }.ok();
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
($ty:ty, $field:ident, $field_ty:ty) => {
|
||||||
|
const _: () = {
|
||||||
|
$crate::assert_pinned!($ty, $field, $field_ty, inline);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
/// A pin-initializer for the type `T`.
|
/// A pin-initializer for the type `T`.
|
||||||
///
|
///
|
||||||
/// To use this initializer, you will need a suitable memory location that can hold a `T`. This can
|
/// To use this initializer, you will need a suitable memory location that can hold a `T`. This can
|
||||||
@ -1107,11 +1176,17 @@ unsafe impl<T, E> PinInit<T, E> for T {
|
|||||||
|
|
||||||
/// Smart pointer that can initialize memory in-place.
|
/// Smart pointer that can initialize memory in-place.
|
||||||
pub trait InPlaceInit<T>: Sized {
|
pub trait InPlaceInit<T>: Sized {
|
||||||
|
/// Pinned version of `Self`.
|
||||||
|
///
|
||||||
|
/// If a type already implicitly pins its pointee, `Pin<Self>` is unnecessary. In this case use
|
||||||
|
/// `Self`, otherwise just use `Pin<Self>`.
|
||||||
|
type PinnedSelf;
|
||||||
|
|
||||||
/// Use the given pin-initializer to pin-initialize a `T` inside of a new smart pointer of this
|
/// Use the given pin-initializer to pin-initialize a `T` inside of a new smart pointer of this
|
||||||
/// type.
|
/// type.
|
||||||
///
|
///
|
||||||
/// If `T: !Unpin` it will not be able to move afterwards.
|
/// If `T: !Unpin` it will not be able to move afterwards.
|
||||||
fn try_pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> Result<Pin<Self>, E>
|
fn try_pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> Result<Self::PinnedSelf, E>
|
||||||
where
|
where
|
||||||
E: From<AllocError>;
|
E: From<AllocError>;
|
||||||
|
|
||||||
@ -1119,7 +1194,7 @@ pub trait InPlaceInit<T>: Sized {
|
|||||||
/// type.
|
/// type.
|
||||||
///
|
///
|
||||||
/// If `T: !Unpin` it will not be able to move afterwards.
|
/// If `T: !Unpin` it will not be able to move afterwards.
|
||||||
fn pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> error::Result<Pin<Self>>
|
fn pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> error::Result<Self::PinnedSelf>
|
||||||
where
|
where
|
||||||
Error: From<E>,
|
Error: From<E>,
|
||||||
{
|
{
|
||||||
@ -1148,19 +1223,15 @@ pub trait InPlaceInit<T>: Sized {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> InPlaceInit<T> for Box<T> {
|
impl<T> InPlaceInit<T> for Arc<T> {
|
||||||
|
type PinnedSelf = Self;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn try_pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> Result<Pin<Self>, E>
|
fn try_pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> Result<Self::PinnedSelf, E>
|
||||||
where
|
where
|
||||||
E: From<AllocError>,
|
E: From<AllocError>,
|
||||||
{
|
{
|
||||||
let mut this = <Box<_> as BoxExt<_>>::new_uninit(flags)?;
|
UniqueArc::try_pin_init(init, flags).map(|u| u.into())
|
||||||
let slot = this.as_mut_ptr();
|
|
||||||
// SAFETY: When init errors/panics, slot will get deallocated but not dropped,
|
|
||||||
// slot is valid and will not be moved, because we pin it later.
|
|
||||||
unsafe { init.__pinned_init(slot)? };
|
|
||||||
// SAFETY: All fields have been initialized.
|
|
||||||
Ok(unsafe { this.assume_init() }.into())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
@ -1168,29 +1239,39 @@ impl<T> InPlaceInit<T> for Box<T> {
|
|||||||
where
|
where
|
||||||
E: From<AllocError>,
|
E: From<AllocError>,
|
||||||
{
|
{
|
||||||
let mut this = <Box<_> as BoxExt<_>>::new_uninit(flags)?;
|
UniqueArc::try_init(init, flags).map(|u| u.into())
|
||||||
let slot = this.as_mut_ptr();
|
}
|
||||||
// SAFETY: When init errors/panics, slot will get deallocated but not dropped,
|
}
|
||||||
// slot is valid.
|
|
||||||
unsafe { init.__init(slot)? };
|
impl<T> InPlaceInit<T> for Box<T> {
|
||||||
// SAFETY: All fields have been initialized.
|
type PinnedSelf = Pin<Self>;
|
||||||
Ok(unsafe { this.assume_init() })
|
|
||||||
|
#[inline]
|
||||||
|
fn try_pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> Result<Self::PinnedSelf, E>
|
||||||
|
where
|
||||||
|
E: From<AllocError>,
|
||||||
|
{
|
||||||
|
<Box<_> as BoxExt<_>>::new_uninit(flags)?.write_pin_init(init)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn try_init<E>(init: impl Init<T, E>, flags: Flags) -> Result<Self, E>
|
||||||
|
where
|
||||||
|
E: From<AllocError>,
|
||||||
|
{
|
||||||
|
<Box<_> as BoxExt<_>>::new_uninit(flags)?.write_init(init)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> InPlaceInit<T> for UniqueArc<T> {
|
impl<T> InPlaceInit<T> for UniqueArc<T> {
|
||||||
|
type PinnedSelf = Pin<Self>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn try_pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> Result<Pin<Self>, E>
|
fn try_pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> Result<Self::PinnedSelf, E>
|
||||||
where
|
where
|
||||||
E: From<AllocError>,
|
E: From<AllocError>,
|
||||||
{
|
{
|
||||||
let mut this = UniqueArc::new_uninit(flags)?;
|
UniqueArc::new_uninit(flags)?.write_pin_init(init)
|
||||||
let slot = this.as_mut_ptr();
|
|
||||||
// SAFETY: When init errors/panics, slot will get deallocated but not dropped,
|
|
||||||
// slot is valid and will not be moved, because we pin it later.
|
|
||||||
unsafe { init.__pinned_init(slot)? };
|
|
||||||
// SAFETY: All fields have been initialized.
|
|
||||||
Ok(unsafe { this.assume_init() }.into())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
@ -1198,13 +1279,67 @@ impl<T> InPlaceInit<T> for UniqueArc<T> {
|
|||||||
where
|
where
|
||||||
E: From<AllocError>,
|
E: From<AllocError>,
|
||||||
{
|
{
|
||||||
let mut this = UniqueArc::new_uninit(flags)?;
|
UniqueArc::new_uninit(flags)?.write_init(init)
|
||||||
let slot = this.as_mut_ptr();
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Smart pointer containing uninitialized memory and that can write a value.
|
||||||
|
pub trait InPlaceWrite<T> {
|
||||||
|
/// The type `Self` turns into when the contents are initialized.
|
||||||
|
type Initialized;
|
||||||
|
|
||||||
|
/// Use the given initializer to write a value into `self`.
|
||||||
|
///
|
||||||
|
/// Does not drop the current value and considers it as uninitialized memory.
|
||||||
|
fn write_init<E>(self, init: impl Init<T, E>) -> Result<Self::Initialized, E>;
|
||||||
|
|
||||||
|
/// Use the given pin-initializer to write a value into `self`.
|
||||||
|
///
|
||||||
|
/// Does not drop the current value and considers it as uninitialized memory.
|
||||||
|
fn write_pin_init<E>(self, init: impl PinInit<T, E>) -> Result<Pin<Self::Initialized>, E>;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> InPlaceWrite<T> for Box<MaybeUninit<T>> {
|
||||||
|
type Initialized = Box<T>;
|
||||||
|
|
||||||
|
fn write_init<E>(mut self, init: impl Init<T, E>) -> Result<Self::Initialized, E> {
|
||||||
|
let slot = self.as_mut_ptr();
|
||||||
// SAFETY: When init errors/panics, slot will get deallocated but not dropped,
|
// SAFETY: When init errors/panics, slot will get deallocated but not dropped,
|
||||||
// slot is valid.
|
// slot is valid.
|
||||||
unsafe { init.__init(slot)? };
|
unsafe { init.__init(slot)? };
|
||||||
// SAFETY: All fields have been initialized.
|
// SAFETY: All fields have been initialized.
|
||||||
Ok(unsafe { this.assume_init() })
|
Ok(unsafe { self.assume_init() })
|
||||||
|
}
|
||||||
|
|
||||||
|
fn write_pin_init<E>(mut self, init: impl PinInit<T, E>) -> Result<Pin<Self::Initialized>, E> {
|
||||||
|
let slot = self.as_mut_ptr();
|
||||||
|
// SAFETY: When init errors/panics, slot will get deallocated but not dropped,
|
||||||
|
// slot is valid and will not be moved, because we pin it later.
|
||||||
|
unsafe { init.__pinned_init(slot)? };
|
||||||
|
// SAFETY: All fields have been initialized.
|
||||||
|
Ok(unsafe { self.assume_init() }.into())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> InPlaceWrite<T> for UniqueArc<MaybeUninit<T>> {
|
||||||
|
type Initialized = UniqueArc<T>;
|
||||||
|
|
||||||
|
fn write_init<E>(mut self, init: impl Init<T, E>) -> Result<Self::Initialized, E> {
|
||||||
|
let slot = self.as_mut_ptr();
|
||||||
|
// SAFETY: When init errors/panics, slot will get deallocated but not dropped,
|
||||||
|
// slot is valid.
|
||||||
|
unsafe { init.__init(slot)? };
|
||||||
|
// SAFETY: All fields have been initialized.
|
||||||
|
Ok(unsafe { self.assume_init() })
|
||||||
|
}
|
||||||
|
|
||||||
|
fn write_pin_init<E>(mut self, init: impl PinInit<T, E>) -> Result<Pin<Self::Initialized>, E> {
|
||||||
|
let slot = self.as_mut_ptr();
|
||||||
|
// SAFETY: When init errors/panics, slot will get deallocated but not dropped,
|
||||||
|
// slot is valid and will not be moved, because we pin it later.
|
||||||
|
unsafe { init.__pinned_init(slot)? };
|
||||||
|
// SAFETY: All fields have been initialized.
|
||||||
|
Ok(unsafe { self.assume_init() }.into())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -228,3 +228,32 @@ impl OnlyCallFromDrop {
|
|||||||
Self(())
|
Self(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Initializer that always fails.
|
||||||
|
///
|
||||||
|
/// Used by [`assert_pinned!`].
|
||||||
|
///
|
||||||
|
/// [`assert_pinned!`]: crate::assert_pinned
|
||||||
|
pub struct AlwaysFail<T: ?Sized> {
|
||||||
|
_t: PhantomData<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: ?Sized> AlwaysFail<T> {
|
||||||
|
/// Creates a new initializer that always fails.
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self { _t: PhantomData }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: ?Sized> Default for AlwaysFail<T> {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::new()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: `__pinned_init` always fails, which is always okay.
|
||||||
|
unsafe impl<T: ?Sized> PinInit<T, ()> for AlwaysFail<T> {
|
||||||
|
unsafe fn __pinned_init(self, _slot: *mut T) -> Result<(), ()> {
|
||||||
|
Err(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -38,12 +38,14 @@ pub mod init;
|
|||||||
pub mod ioctl;
|
pub mod ioctl;
|
||||||
#[cfg(CONFIG_KUNIT)]
|
#[cfg(CONFIG_KUNIT)]
|
||||||
pub mod kunit;
|
pub mod kunit;
|
||||||
|
pub mod list;
|
||||||
#[cfg(CONFIG_NET)]
|
#[cfg(CONFIG_NET)]
|
||||||
pub mod net;
|
pub mod net;
|
||||||
pub mod page;
|
pub mod page;
|
||||||
pub mod prelude;
|
pub mod prelude;
|
||||||
pub mod print;
|
pub mod print;
|
||||||
pub mod sizes;
|
pub mod sizes;
|
||||||
|
pub mod rbtree;
|
||||||
mod static_assert;
|
mod static_assert;
|
||||||
#[doc(hidden)]
|
#[doc(hidden)]
|
||||||
pub mod std_vendor;
|
pub mod std_vendor;
|
||||||
|
686
rust/kernel/list.rs
Normal file
686
rust/kernel/list.rs
Normal file
@ -0,0 +1,686 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
// Copyright (C) 2024 Google LLC.
|
||||||
|
|
||||||
|
//! A linked list implementation.
|
||||||
|
|
||||||
|
use crate::init::PinInit;
|
||||||
|
use crate::sync::ArcBorrow;
|
||||||
|
use crate::types::Opaque;
|
||||||
|
use core::iter::{DoubleEndedIterator, FusedIterator};
|
||||||
|
use core::marker::PhantomData;
|
||||||
|
use core::ptr;
|
||||||
|
|
||||||
|
mod impl_list_item_mod;
|
||||||
|
pub use self::impl_list_item_mod::{
|
||||||
|
impl_has_list_links, impl_has_list_links_self_ptr, impl_list_item, HasListLinks, HasSelfPtr,
|
||||||
|
};
|
||||||
|
|
||||||
|
mod arc;
|
||||||
|
pub use self::arc::{impl_list_arc_safe, AtomicTracker, ListArc, ListArcSafe, TryNewListArc};
|
||||||
|
|
||||||
|
mod arc_field;
|
||||||
|
pub use self::arc_field::{define_list_arc_field_getter, ListArcField};
|
||||||
|
|
||||||
|
/// A linked list.
|
||||||
|
///
|
||||||
|
/// All elements in this linked list will be [`ListArc`] references to the value. Since a value can
|
||||||
|
/// only have one `ListArc` (for each pair of prev/next pointers), this ensures that the same
|
||||||
|
/// prev/next pointers are not used for several linked lists.
|
||||||
|
///
|
||||||
|
/// # Invariants
|
||||||
|
///
|
||||||
|
/// * If the list is empty, then `first` is null. Otherwise, `first` points at the `ListLinks`
|
||||||
|
/// field of the first element in the list.
|
||||||
|
/// * All prev/next pointers in `ListLinks` fields of items in the list are valid and form a cycle.
|
||||||
|
/// * For every item in the list, the list owns the associated [`ListArc`] reference and has
|
||||||
|
/// exclusive access to the `ListLinks` field.
|
||||||
|
pub struct List<T: ?Sized + ListItem<ID>, const ID: u64 = 0> {
|
||||||
|
first: *mut ListLinksFields,
|
||||||
|
_ty: PhantomData<ListArc<T, ID>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: This is a container of `ListArc<T, ID>`, and access to the container allows the same
|
||||||
|
// type of access to the `ListArc<T, ID>` elements.
|
||||||
|
unsafe impl<T, const ID: u64> Send for List<T, ID>
|
||||||
|
where
|
||||||
|
ListArc<T, ID>: Send,
|
||||||
|
T: ?Sized + ListItem<ID>,
|
||||||
|
{
|
||||||
|
}
|
||||||
|
// SAFETY: This is a container of `ListArc<T, ID>`, and access to the container allows the same
|
||||||
|
// type of access to the `ListArc<T, ID>` elements.
|
||||||
|
unsafe impl<T, const ID: u64> Sync for List<T, ID>
|
||||||
|
where
|
||||||
|
ListArc<T, ID>: Sync,
|
||||||
|
T: ?Sized + ListItem<ID>,
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Implemented by types where a [`ListArc<Self>`] can be inserted into a [`List`].
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// Implementers must ensure that they provide the guarantees documented on methods provided by
|
||||||
|
/// this trait.
|
||||||
|
///
|
||||||
|
/// [`ListArc<Self>`]: ListArc
|
||||||
|
pub unsafe trait ListItem<const ID: u64 = 0>: ListArcSafe<ID> {
|
||||||
|
/// Views the [`ListLinks`] for this value.
|
||||||
|
///
|
||||||
|
/// # Guarantees
|
||||||
|
///
|
||||||
|
/// If there is a previous call to `prepare_to_insert` and there is no call to `post_remove`
|
||||||
|
/// since the most recent such call, then this returns the same pointer as the one returned by
|
||||||
|
/// the most recent call to `prepare_to_insert`.
|
||||||
|
///
|
||||||
|
/// Otherwise, the returned pointer points at a read-only [`ListLinks`] with two null pointers.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// The provided pointer must point at a valid value. (It need not be in an `Arc`.)
|
||||||
|
unsafe fn view_links(me: *const Self) -> *mut ListLinks<ID>;
|
||||||
|
|
||||||
|
/// View the full value given its [`ListLinks`] field.
|
||||||
|
///
|
||||||
|
/// Can only be used when the value is in a list.
|
||||||
|
///
|
||||||
|
/// # Guarantees
|
||||||
|
///
|
||||||
|
/// * Returns the same pointer as the one passed to the most recent call to `prepare_to_insert`.
|
||||||
|
/// * The returned pointer is valid until the next call to `post_remove`.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// * The provided pointer must originate from the most recent call to `prepare_to_insert`, or
|
||||||
|
/// from a call to `view_links` that happened after the most recent call to
|
||||||
|
/// `prepare_to_insert`.
|
||||||
|
/// * Since the most recent call to `prepare_to_insert`, the `post_remove` method must not have
|
||||||
|
/// been called.
|
||||||
|
unsafe fn view_value(me: *mut ListLinks<ID>) -> *const Self;
|
||||||
|
|
||||||
|
/// This is called when an item is inserted into a [`List`].
|
||||||
|
///
|
||||||
|
/// # Guarantees
|
||||||
|
///
|
||||||
|
/// The caller is granted exclusive access to the returned [`ListLinks`] until `post_remove` is
|
||||||
|
/// called.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// * The provided pointer must point at a valid value in an [`Arc`].
|
||||||
|
/// * Calls to `prepare_to_insert` and `post_remove` on the same value must alternate.
|
||||||
|
/// * The caller must own the [`ListArc`] for this value.
|
||||||
|
/// * The caller must not give up ownership of the [`ListArc`] unless `post_remove` has been
|
||||||
|
/// called after this call to `prepare_to_insert`.
|
||||||
|
///
|
||||||
|
/// [`Arc`]: crate::sync::Arc
|
||||||
|
unsafe fn prepare_to_insert(me: *const Self) -> *mut ListLinks<ID>;
|
||||||
|
|
||||||
|
/// This undoes a previous call to `prepare_to_insert`.
|
||||||
|
///
|
||||||
|
/// # Guarantees
|
||||||
|
///
|
||||||
|
/// The returned pointer is the pointer that was originally passed to `prepare_to_insert`.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// The provided pointer must be the pointer returned by the most recent call to
|
||||||
|
/// `prepare_to_insert`.
|
||||||
|
unsafe fn post_remove(me: *mut ListLinks<ID>) -> *const Self;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[repr(C)]
|
||||||
|
#[derive(Copy, Clone)]
|
||||||
|
struct ListLinksFields {
|
||||||
|
next: *mut ListLinksFields,
|
||||||
|
prev: *mut ListLinksFields,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The prev/next pointers for an item in a linked list.
|
||||||
|
///
|
||||||
|
/// # Invariants
|
||||||
|
///
|
||||||
|
/// The fields are null if and only if this item is not in a list.
|
||||||
|
#[repr(transparent)]
|
||||||
|
pub struct ListLinks<const ID: u64 = 0> {
|
||||||
|
// This type is `!Unpin` for aliasing reasons as the pointers are part of an intrusive linked
|
||||||
|
// list.
|
||||||
|
inner: Opaque<ListLinksFields>,
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: The only way to access/modify the pointers inside of `ListLinks<ID>` is via holding the
|
||||||
|
// associated `ListArc<T, ID>`. Since that type correctly implements `Send`, it is impossible to
|
||||||
|
// move this an instance of this type to a different thread if the pointees are `!Send`.
|
||||||
|
unsafe impl<const ID: u64> Send for ListLinks<ID> {}
|
||||||
|
// SAFETY: The type is opaque so immutable references to a ListLinks are useless. Therefore, it's
|
||||||
|
// okay to have immutable access to a ListLinks from several threads at once.
|
||||||
|
unsafe impl<const ID: u64> Sync for ListLinks<ID> {}
|
||||||
|
|
||||||
|
impl<const ID: u64> ListLinks<ID> {
|
||||||
|
/// Creates a new initializer for this type.
|
||||||
|
pub fn new() -> impl PinInit<Self> {
|
||||||
|
// INVARIANT: Pin-init initializers can't be used on an existing `Arc`, so this value will
|
||||||
|
// not be constructed in an `Arc` that already has a `ListArc`.
|
||||||
|
ListLinks {
|
||||||
|
inner: Opaque::new(ListLinksFields {
|
||||||
|
prev: ptr::null_mut(),
|
||||||
|
next: ptr::null_mut(),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// `me` must be dereferenceable.
|
||||||
|
#[inline]
|
||||||
|
unsafe fn fields(me: *mut Self) -> *mut ListLinksFields {
|
||||||
|
// SAFETY: The caller promises that the pointer is valid.
|
||||||
|
unsafe { Opaque::raw_get(ptr::addr_of!((*me).inner)) }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// `me` must be dereferenceable.
|
||||||
|
#[inline]
|
||||||
|
unsafe fn from_fields(me: *mut ListLinksFields) -> *mut Self {
|
||||||
|
me.cast()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Similar to [`ListLinks`], but also contains a pointer to the full value.
|
||||||
|
///
|
||||||
|
/// This type can be used instead of [`ListLinks`] to support lists with trait objects.
|
||||||
|
#[repr(C)]
|
||||||
|
pub struct ListLinksSelfPtr<T: ?Sized, const ID: u64 = 0> {
|
||||||
|
/// The `ListLinks` field inside this value.
|
||||||
|
///
|
||||||
|
/// This is public so that it can be used with `impl_has_list_links!`.
|
||||||
|
pub inner: ListLinks<ID>,
|
||||||
|
// UnsafeCell is not enough here because we use `Opaque::uninit` as a dummy value, and
|
||||||
|
// `ptr::null()` doesn't work for `T: ?Sized`.
|
||||||
|
self_ptr: Opaque<*const T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: The fields of a ListLinksSelfPtr can be moved across thread boundaries.
|
||||||
|
unsafe impl<T: ?Sized + Send, const ID: u64> Send for ListLinksSelfPtr<T, ID> {}
|
||||||
|
// SAFETY: The type is opaque so immutable references to a ListLinksSelfPtr are useless. Therefore,
|
||||||
|
// it's okay to have immutable access to a ListLinks from several threads at once.
|
||||||
|
//
|
||||||
|
// Note that `inner` being a public field does not prevent this type from being opaque, since
|
||||||
|
// `inner` is a opaque type.
|
||||||
|
unsafe impl<T: ?Sized + Sync, const ID: u64> Sync for ListLinksSelfPtr<T, ID> {}
|
||||||
|
|
||||||
|
impl<T: ?Sized, const ID: u64> ListLinksSelfPtr<T, ID> {
|
||||||
|
/// The offset from the [`ListLinks`] to the self pointer field.
|
||||||
|
pub const LIST_LINKS_SELF_PTR_OFFSET: usize = core::mem::offset_of!(Self, self_ptr);
|
||||||
|
|
||||||
|
/// Creates a new initializer for this type.
|
||||||
|
pub fn new() -> impl PinInit<Self> {
|
||||||
|
// INVARIANT: Pin-init initializers can't be used on an existing `Arc`, so this value will
|
||||||
|
// not be constructed in an `Arc` that already has a `ListArc`.
|
||||||
|
Self {
|
||||||
|
inner: ListLinks {
|
||||||
|
inner: Opaque::new(ListLinksFields {
|
||||||
|
prev: ptr::null_mut(),
|
||||||
|
next: ptr::null_mut(),
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
self_ptr: Opaque::uninit(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: ?Sized + ListItem<ID>, const ID: u64> List<T, ID> {
|
||||||
|
/// Creates a new empty list.
|
||||||
|
pub const fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
first: ptr::null_mut(),
|
||||||
|
_ty: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns whether this list is empty.
|
||||||
|
pub fn is_empty(&self) -> bool {
|
||||||
|
self.first.is_null()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Add the provided item to the back of the list.
|
||||||
|
pub fn push_back(&mut self, item: ListArc<T, ID>) {
|
||||||
|
let raw_item = ListArc::into_raw(item);
|
||||||
|
// SAFETY:
|
||||||
|
// * We just got `raw_item` from a `ListArc`, so it's in an `Arc`.
|
||||||
|
// * Since we have ownership of the `ListArc`, `post_remove` must have been called after
|
||||||
|
// the most recent call to `prepare_to_insert`, if any.
|
||||||
|
// * We own the `ListArc`.
|
||||||
|
// * Removing items from this list is always done using `remove_internal_inner`, which
|
||||||
|
// calls `post_remove` before giving up ownership.
|
||||||
|
let list_links = unsafe { T::prepare_to_insert(raw_item) };
|
||||||
|
// SAFETY: We have not yet called `post_remove`, so `list_links` is still valid.
|
||||||
|
let item = unsafe { ListLinks::fields(list_links) };
|
||||||
|
|
||||||
|
if self.first.is_null() {
|
||||||
|
self.first = item;
|
||||||
|
// SAFETY: The caller just gave us ownership of these fields.
|
||||||
|
// INVARIANT: A linked list with one item should be cyclic.
|
||||||
|
unsafe {
|
||||||
|
(*item).next = item;
|
||||||
|
(*item).prev = item;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let next = self.first;
|
||||||
|
// SAFETY: By the type invariant, this pointer is valid or null. We just checked that
|
||||||
|
// it's not null, so it must be valid.
|
||||||
|
let prev = unsafe { (*next).prev };
|
||||||
|
// SAFETY: Pointers in a linked list are never dangling, and the caller just gave us
|
||||||
|
// ownership of the fields on `item`.
|
||||||
|
// INVARIANT: This correctly inserts `item` between `prev` and `next`.
|
||||||
|
unsafe {
|
||||||
|
(*item).next = next;
|
||||||
|
(*item).prev = prev;
|
||||||
|
(*prev).next = item;
|
||||||
|
(*next).prev = item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Add the provided item to the front of the list.
|
||||||
|
pub fn push_front(&mut self, item: ListArc<T, ID>) {
|
||||||
|
let raw_item = ListArc::into_raw(item);
|
||||||
|
// SAFETY:
|
||||||
|
// * We just got `raw_item` from a `ListArc`, so it's in an `Arc`.
|
||||||
|
// * If this requirement is violated, then the previous caller of `prepare_to_insert`
|
||||||
|
// violated the safety requirement that they can't give up ownership of the `ListArc`
|
||||||
|
// until they call `post_remove`.
|
||||||
|
// * We own the `ListArc`.
|
||||||
|
// * Removing items] from this list is always done using `remove_internal_inner`, which
|
||||||
|
// calls `post_remove` before giving up ownership.
|
||||||
|
let list_links = unsafe { T::prepare_to_insert(raw_item) };
|
||||||
|
// SAFETY: We have not yet called `post_remove`, so `list_links` is still valid.
|
||||||
|
let item = unsafe { ListLinks::fields(list_links) };
|
||||||
|
|
||||||
|
if self.first.is_null() {
|
||||||
|
// SAFETY: The caller just gave us ownership of these fields.
|
||||||
|
// INVARIANT: A linked list with one item should be cyclic.
|
||||||
|
unsafe {
|
||||||
|
(*item).next = item;
|
||||||
|
(*item).prev = item;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let next = self.first;
|
||||||
|
// SAFETY: We just checked that `next` is non-null.
|
||||||
|
let prev = unsafe { (*next).prev };
|
||||||
|
// SAFETY: Pointers in a linked list are never dangling, and the caller just gave us
|
||||||
|
// ownership of the fields on `item`.
|
||||||
|
// INVARIANT: This correctly inserts `item` between `prev` and `next`.
|
||||||
|
unsafe {
|
||||||
|
(*item).next = next;
|
||||||
|
(*item).prev = prev;
|
||||||
|
(*prev).next = item;
|
||||||
|
(*next).prev = item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self.first = item;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Removes the last item from this list.
|
||||||
|
pub fn pop_back(&mut self) -> Option<ListArc<T, ID>> {
|
||||||
|
if self.first.is_null() {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: We just checked that the list is not empty.
|
||||||
|
let last = unsafe { (*self.first).prev };
|
||||||
|
// SAFETY: The last item of this list is in this list.
|
||||||
|
Some(unsafe { self.remove_internal(last) })
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Removes the first item from this list.
|
||||||
|
pub fn pop_front(&mut self) -> Option<ListArc<T, ID>> {
|
||||||
|
if self.first.is_null() {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: The first item of this list is in this list.
|
||||||
|
Some(unsafe { self.remove_internal(self.first) })
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Removes the provided item from this list and returns it.
|
||||||
|
///
|
||||||
|
/// This returns `None` if the item is not in the list. (Note that by the safety requirements,
|
||||||
|
/// this means that the item is not in any list.)
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// `item` must not be in a different linked list (with the same id).
|
||||||
|
pub unsafe fn remove(&mut self, item: &T) -> Option<ListArc<T, ID>> {
|
||||||
|
let mut item = unsafe { ListLinks::fields(T::view_links(item)) };
|
||||||
|
// SAFETY: The user provided a reference, and reference are never dangling.
|
||||||
|
//
|
||||||
|
// As for why this is not a data race, there are two cases:
|
||||||
|
//
|
||||||
|
// * If `item` is not in any list, then these fields are read-only and null.
|
||||||
|
// * If `item` is in this list, then we have exclusive access to these fields since we
|
||||||
|
// have a mutable reference to the list.
|
||||||
|
//
|
||||||
|
// In either case, there's no race.
|
||||||
|
let ListLinksFields { next, prev } = unsafe { *item };
|
||||||
|
|
||||||
|
debug_assert_eq!(next.is_null(), prev.is_null());
|
||||||
|
if !next.is_null() {
|
||||||
|
// This is really a no-op, but this ensures that `item` is a raw pointer that was
|
||||||
|
// obtained without going through a pointer->reference->pointer conversion roundtrip.
|
||||||
|
// This ensures that the list is valid under the more restrictive strict provenance
|
||||||
|
// ruleset.
|
||||||
|
//
|
||||||
|
// SAFETY: We just checked that `next` is not null, and it's not dangling by the
|
||||||
|
// list invariants.
|
||||||
|
unsafe {
|
||||||
|
debug_assert_eq!(item, (*next).prev);
|
||||||
|
item = (*next).prev;
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: We just checked that `item` is in a list, so the caller guarantees that it
|
||||||
|
// is in this list. The pointers are in the right order.
|
||||||
|
Some(unsafe { self.remove_internal_inner(item, next, prev) })
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Removes the provided item from the list.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// `item` must point at an item in this list.
|
||||||
|
unsafe fn remove_internal(&mut self, item: *mut ListLinksFields) -> ListArc<T, ID> {
|
||||||
|
// SAFETY: The caller promises that this pointer is not dangling, and there's no data race
|
||||||
|
// since we have a mutable reference to the list containing `item`.
|
||||||
|
let ListLinksFields { next, prev } = unsafe { *item };
|
||||||
|
// SAFETY: The pointers are ok and in the right order.
|
||||||
|
unsafe { self.remove_internal_inner(item, next, prev) }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Removes the provided item from the list.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// The `item` pointer must point at an item in this list, and we must have `(*item).next ==
|
||||||
|
/// next` and `(*item).prev == prev`.
|
||||||
|
unsafe fn remove_internal_inner(
|
||||||
|
&mut self,
|
||||||
|
item: *mut ListLinksFields,
|
||||||
|
next: *mut ListLinksFields,
|
||||||
|
prev: *mut ListLinksFields,
|
||||||
|
) -> ListArc<T, ID> {
|
||||||
|
// SAFETY: We have exclusive access to the pointers of items in the list, and the prev/next
|
||||||
|
// pointers are always valid for items in a list.
|
||||||
|
//
|
||||||
|
// INVARIANT: There are three cases:
|
||||||
|
// * If the list has at least three items, then after removing the item, `prev` and `next`
|
||||||
|
// will be next to each other.
|
||||||
|
// * If the list has two items, then the remaining item will point at itself.
|
||||||
|
// * If the list has one item, then `next == prev == item`, so these writes have no
|
||||||
|
// effect. The list remains unchanged and `item` is still in the list for now.
|
||||||
|
unsafe {
|
||||||
|
(*next).prev = prev;
|
||||||
|
(*prev).next = next;
|
||||||
|
}
|
||||||
|
// SAFETY: We have exclusive access to items in the list.
|
||||||
|
// INVARIANT: `item` is being removed, so the pointers should be null.
|
||||||
|
unsafe {
|
||||||
|
(*item).prev = ptr::null_mut();
|
||||||
|
(*item).next = ptr::null_mut();
|
||||||
|
}
|
||||||
|
// INVARIANT: There are three cases:
|
||||||
|
// * If `item` was not the first item, then `self.first` should remain unchanged.
|
||||||
|
// * If `item` was the first item and there is another item, then we just updated
|
||||||
|
// `prev->next` to `next`, which is the new first item, and setting `item->next` to null
|
||||||
|
// did not modify `prev->next`.
|
||||||
|
// * If `item` was the only item in the list, then `prev == item`, and we just set
|
||||||
|
// `item->next` to null, so this correctly sets `first` to null now that the list is
|
||||||
|
// empty.
|
||||||
|
if self.first == item {
|
||||||
|
// SAFETY: The `prev` pointer is the value that `item->prev` had when it was in this
|
||||||
|
// list, so it must be valid. There is no race since `prev` is still in the list and we
|
||||||
|
// still have exclusive access to the list.
|
||||||
|
self.first = unsafe { (*prev).next };
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: `item` used to be in the list, so it is dereferenceable by the type invariants
|
||||||
|
// of `List`.
|
||||||
|
let list_links = unsafe { ListLinks::from_fields(item) };
|
||||||
|
// SAFETY: Any pointer in the list originates from a `prepare_to_insert` call.
|
||||||
|
let raw_item = unsafe { T::post_remove(list_links) };
|
||||||
|
// SAFETY: The above call to `post_remove` guarantees that we can recreate the `ListArc`.
|
||||||
|
unsafe { ListArc::from_raw(raw_item) }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Moves all items from `other` into `self`.
|
||||||
|
///
|
||||||
|
/// The items of `other` are added to the back of `self`, so the last item of `other` becomes
|
||||||
|
/// the last item of `self`.
|
||||||
|
pub fn push_all_back(&mut self, other: &mut List<T, ID>) {
|
||||||
|
// First, we insert the elements into `self`. At the end, we make `other` empty.
|
||||||
|
if self.is_empty() {
|
||||||
|
// INVARIANT: All of the elements in `other` become elements of `self`.
|
||||||
|
self.first = other.first;
|
||||||
|
} else if !other.is_empty() {
|
||||||
|
let other_first = other.first;
|
||||||
|
// SAFETY: The other list is not empty, so this pointer is valid.
|
||||||
|
let other_last = unsafe { (*other_first).prev };
|
||||||
|
let self_first = self.first;
|
||||||
|
// SAFETY: The self list is not empty, so this pointer is valid.
|
||||||
|
let self_last = unsafe { (*self_first).prev };
|
||||||
|
|
||||||
|
// SAFETY: We have exclusive access to both lists, so we can update the pointers.
|
||||||
|
// INVARIANT: This correctly sets the pointers to merge both lists. We do not need to
|
||||||
|
// update `self.first` because the first element of `self` does not change.
|
||||||
|
unsafe {
|
||||||
|
(*self_first).prev = other_last;
|
||||||
|
(*other_last).next = self_first;
|
||||||
|
(*self_last).next = other_first;
|
||||||
|
(*other_first).prev = self_last;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// INVARIANT: The other list is now empty, so update its pointer.
|
||||||
|
other.first = ptr::null_mut();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a cursor to the first element of the list.
|
||||||
|
///
|
||||||
|
/// If the list is empty, this returns `None`.
|
||||||
|
pub fn cursor_front(&mut self) -> Option<Cursor<'_, T, ID>> {
|
||||||
|
if self.first.is_null() {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
Some(Cursor {
|
||||||
|
current: self.first,
|
||||||
|
list: self,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates an iterator over the list.
|
||||||
|
pub fn iter(&self) -> Iter<'_, T, ID> {
|
||||||
|
// INVARIANT: If the list is empty, both pointers are null. Otherwise, both pointers point
|
||||||
|
// at the first element of the same list.
|
||||||
|
Iter {
|
||||||
|
current: self.first,
|
||||||
|
stop: self.first,
|
||||||
|
_ty: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: ?Sized + ListItem<ID>, const ID: u64> Default for List<T, ID> {
|
||||||
|
fn default() -> Self {
|
||||||
|
List::new()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: ?Sized + ListItem<ID>, const ID: u64> Drop for List<T, ID> {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
while let Some(item) = self.pop_front() {
|
||||||
|
drop(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// An iterator over a [`List`].
|
||||||
|
///
|
||||||
|
/// # Invariants
|
||||||
|
///
|
||||||
|
/// * There must be a [`List`] that is immutably borrowed for the duration of `'a`.
|
||||||
|
/// * The `current` pointer is null or points at a value in that [`List`].
|
||||||
|
/// * The `stop` pointer is equal to the `first` field of that [`List`].
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct Iter<'a, T: ?Sized + ListItem<ID>, const ID: u64 = 0> {
|
||||||
|
current: *mut ListLinksFields,
|
||||||
|
stop: *mut ListLinksFields,
|
||||||
|
_ty: PhantomData<&'a ListArc<T, ID>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, T: ?Sized + ListItem<ID>, const ID: u64> Iterator for Iter<'a, T, ID> {
|
||||||
|
type Item = ArcBorrow<'a, T>;
|
||||||
|
|
||||||
|
fn next(&mut self) -> Option<ArcBorrow<'a, T>> {
|
||||||
|
if self.current.is_null() {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let current = self.current;
|
||||||
|
|
||||||
|
// SAFETY: We just checked that `current` is not null, so it is in a list, and hence not
|
||||||
|
// dangling. There's no race because the iterator holds an immutable borrow to the list.
|
||||||
|
let next = unsafe { (*current).next };
|
||||||
|
// INVARIANT: If `current` was the last element of the list, then this updates it to null.
|
||||||
|
// Otherwise, we update it to the next element.
|
||||||
|
self.current = if next != self.stop {
|
||||||
|
next
|
||||||
|
} else {
|
||||||
|
ptr::null_mut()
|
||||||
|
};
|
||||||
|
|
||||||
|
// SAFETY: The `current` pointer points at a value in the list.
|
||||||
|
let item = unsafe { T::view_value(ListLinks::from_fields(current)) };
|
||||||
|
// SAFETY:
|
||||||
|
// * All values in a list are stored in an `Arc`.
|
||||||
|
// * The value cannot be removed from the list for the duration of the lifetime annotated
|
||||||
|
// on the returned `ArcBorrow`, because removing it from the list would require mutable
|
||||||
|
// access to the list. However, the `ArcBorrow` is annotated with the iterator's
|
||||||
|
// lifetime, and the list is immutably borrowed for that lifetime.
|
||||||
|
// * Values in a list never have a `UniqueArc` reference.
|
||||||
|
Some(unsafe { ArcBorrow::from_raw(item) })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A cursor into a [`List`].
|
||||||
|
///
|
||||||
|
/// # Invariants
|
||||||
|
///
|
||||||
|
/// The `current` pointer points a value in `list`.
|
||||||
|
pub struct Cursor<'a, T: ?Sized + ListItem<ID>, const ID: u64 = 0> {
|
||||||
|
current: *mut ListLinksFields,
|
||||||
|
list: &'a mut List<T, ID>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, T: ?Sized + ListItem<ID>, const ID: u64> Cursor<'a, T, ID> {
|
||||||
|
/// Access the current element of this cursor.
|
||||||
|
pub fn current(&self) -> ArcBorrow<'_, T> {
|
||||||
|
// SAFETY: The `current` pointer points a value in the list.
|
||||||
|
let me = unsafe { T::view_value(ListLinks::from_fields(self.current)) };
|
||||||
|
// SAFETY:
|
||||||
|
// * All values in a list are stored in an `Arc`.
|
||||||
|
// * The value cannot be removed from the list for the duration of the lifetime annotated
|
||||||
|
// on the returned `ArcBorrow`, because removing it from the list would require mutable
|
||||||
|
// access to the cursor or the list. However, the `ArcBorrow` holds an immutable borrow
|
||||||
|
// on the cursor, which in turn holds a mutable borrow on the list, so any such
|
||||||
|
// mutable access requires first releasing the immutable borrow on the cursor.
|
||||||
|
// * Values in a list never have a `UniqueArc` reference, because the list has a `ListArc`
|
||||||
|
// reference, and `UniqueArc` references must be unique.
|
||||||
|
unsafe { ArcBorrow::from_raw(me) }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Move the cursor to the next element.
|
||||||
|
pub fn next(self) -> Option<Cursor<'a, T, ID>> {
|
||||||
|
// SAFETY: The `current` field is always in a list.
|
||||||
|
let next = unsafe { (*self.current).next };
|
||||||
|
|
||||||
|
if next == self.list.first {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
// INVARIANT: Since `self.current` is in the `list`, its `next` pointer is also in the
|
||||||
|
// `list`.
|
||||||
|
Some(Cursor {
|
||||||
|
current: next,
|
||||||
|
list: self.list,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Move the cursor to the previous element.
|
||||||
|
pub fn prev(self) -> Option<Cursor<'a, T, ID>> {
|
||||||
|
// SAFETY: The `current` field is always in a list.
|
||||||
|
let prev = unsafe { (*self.current).prev };
|
||||||
|
|
||||||
|
if self.current == self.list.first {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
// INVARIANT: Since `self.current` is in the `list`, its `prev` pointer is also in the
|
||||||
|
// `list`.
|
||||||
|
Some(Cursor {
|
||||||
|
current: prev,
|
||||||
|
list: self.list,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Remove the current element from the list.
|
||||||
|
pub fn remove(self) -> ListArc<T, ID> {
|
||||||
|
// SAFETY: The `current` pointer always points at a member of the list.
|
||||||
|
unsafe { self.list.remove_internal(self.current) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, T: ?Sized + ListItem<ID>, const ID: u64> FusedIterator for Iter<'a, T, ID> {}
|
||||||
|
|
||||||
|
impl<'a, T: ?Sized + ListItem<ID>, const ID: u64> IntoIterator for &'a List<T, ID> {
|
||||||
|
type IntoIter = Iter<'a, T, ID>;
|
||||||
|
type Item = ArcBorrow<'a, T>;
|
||||||
|
|
||||||
|
fn into_iter(self) -> Iter<'a, T, ID> {
|
||||||
|
self.iter()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// An owning iterator into a [`List`].
|
||||||
|
pub struct IntoIter<T: ?Sized + ListItem<ID>, const ID: u64 = 0> {
|
||||||
|
list: List<T, ID>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: ?Sized + ListItem<ID>, const ID: u64> Iterator for IntoIter<T, ID> {
|
||||||
|
type Item = ListArc<T, ID>;
|
||||||
|
|
||||||
|
fn next(&mut self) -> Option<ListArc<T, ID>> {
|
||||||
|
self.list.pop_front()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: ?Sized + ListItem<ID>, const ID: u64> FusedIterator for IntoIter<T, ID> {}
|
||||||
|
|
||||||
|
impl<T: ?Sized + ListItem<ID>, const ID: u64> DoubleEndedIterator for IntoIter<T, ID> {
|
||||||
|
fn next_back(&mut self) -> Option<ListArc<T, ID>> {
|
||||||
|
self.list.pop_back()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: ?Sized + ListItem<ID>, const ID: u64> IntoIterator for List<T, ID> {
|
||||||
|
type IntoIter = IntoIter<T, ID>;
|
||||||
|
type Item = ListArc<T, ID>;
|
||||||
|
|
||||||
|
fn into_iter(self) -> IntoIter<T, ID> {
|
||||||
|
IntoIter { list: self }
|
||||||
|
}
|
||||||
|
}
|
521
rust/kernel/list/arc.rs
Normal file
521
rust/kernel/list/arc.rs
Normal file
@ -0,0 +1,521 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
// Copyright (C) 2024 Google LLC.
|
||||||
|
|
||||||
|
//! A wrapper around `Arc` for linked lists.
|
||||||
|
|
||||||
|
use crate::alloc::{AllocError, Flags};
|
||||||
|
use crate::prelude::*;
|
||||||
|
use crate::sync::{Arc, ArcBorrow, UniqueArc};
|
||||||
|
use core::marker::{PhantomPinned, Unsize};
|
||||||
|
use core::ops::Deref;
|
||||||
|
use core::pin::Pin;
|
||||||
|
use core::sync::atomic::{AtomicBool, Ordering};
|
||||||
|
|
||||||
|
/// Declares that this type has some way to ensure that there is exactly one `ListArc` instance for
|
||||||
|
/// this id.
|
||||||
|
///
|
||||||
|
/// Types that implement this trait should include some kind of logic for keeping track of whether
|
||||||
|
/// a [`ListArc`] exists or not. We refer to this logic as "the tracking inside `T`".
|
||||||
|
///
|
||||||
|
/// We allow the case where the tracking inside `T` thinks that a [`ListArc`] exists, but actually,
|
||||||
|
/// there isn't a [`ListArc`]. However, we do not allow the opposite situation where a [`ListArc`]
|
||||||
|
/// exists, but the tracking thinks it doesn't. This is because the former can at most result in us
|
||||||
|
/// failing to create a [`ListArc`] when the operation could succeed, whereas the latter can result
|
||||||
|
/// in the creation of two [`ListArc`] references. Only the latter situation can lead to memory
|
||||||
|
/// safety issues.
|
||||||
|
///
|
||||||
|
/// A consequence of the above is that you may implement the tracking inside `T` by not actually
|
||||||
|
/// keeping track of anything. To do this, you always claim that a [`ListArc`] exists, even if
|
||||||
|
/// there isn't one. This implementation is allowed by the above rule, but it means that
|
||||||
|
/// [`ListArc`] references can only be created if you have ownership of *all* references to the
|
||||||
|
/// refcounted object, as you otherwise have no way of knowing whether a [`ListArc`] exists.
|
||||||
|
pub trait ListArcSafe<const ID: u64 = 0> {
|
||||||
|
/// Informs the tracking inside this type that it now has a [`ListArc`] reference.
|
||||||
|
///
|
||||||
|
/// This method may be called even if the tracking inside this type thinks that a `ListArc`
|
||||||
|
/// reference exists. (But only if that's not actually the case.)
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// Must not be called if a [`ListArc`] already exist for this value.
|
||||||
|
unsafe fn on_create_list_arc_from_unique(self: Pin<&mut Self>);
|
||||||
|
|
||||||
|
/// Informs the tracking inside this type that there is no [`ListArc`] reference anymore.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// Must only be called if there is no [`ListArc`] reference, but the tracking thinks there is.
|
||||||
|
unsafe fn on_drop_list_arc(&self);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Declares that this type is able to safely attempt to create `ListArc`s at any time.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// The guarantees of `try_new_list_arc` must be upheld.
|
||||||
|
pub unsafe trait TryNewListArc<const ID: u64 = 0>: ListArcSafe<ID> {
|
||||||
|
/// Attempts to convert an `Arc<Self>` into an `ListArc<Self>`. Returns `true` if the
|
||||||
|
/// conversion was successful.
|
||||||
|
///
|
||||||
|
/// This method should not be called directly. Use [`ListArc::try_from_arc`] instead.
|
||||||
|
///
|
||||||
|
/// # Guarantees
|
||||||
|
///
|
||||||
|
/// If this call returns `true`, then there is no [`ListArc`] pointing to this value.
|
||||||
|
/// Additionally, this call will have transitioned the tracking inside `Self` from not thinking
|
||||||
|
/// that a [`ListArc`] exists, to thinking that a [`ListArc`] exists.
|
||||||
|
fn try_new_list_arc(&self) -> bool;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Declares that this type supports [`ListArc`].
|
||||||
|
///
|
||||||
|
/// This macro supports a few different strategies for implementing the tracking inside the type:
|
||||||
|
///
|
||||||
|
/// * The `untracked` strategy does not actually keep track of whether a [`ListArc`] exists. When
|
||||||
|
/// using this strategy, the only way to create a [`ListArc`] is using a [`UniqueArc`].
|
||||||
|
/// * The `tracked_by` strategy defers the tracking to a field of the struct. The user much specify
|
||||||
|
/// which field to defer the tracking to. The field must implement [`ListArcSafe`]. If the field
|
||||||
|
/// implements [`TryNewListArc`], then the type will also implement [`TryNewListArc`].
|
||||||
|
///
|
||||||
|
/// The `tracked_by` strategy is usually used by deferring to a field of type
|
||||||
|
/// [`AtomicTracker`]. However, it is also possible to defer the tracking to another struct
|
||||||
|
/// using also using this macro.
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! impl_list_arc_safe {
|
||||||
|
(impl$({$($generics:tt)*})? ListArcSafe<$num:tt> for $t:ty { untracked; } $($rest:tt)*) => {
|
||||||
|
impl$(<$($generics)*>)? $crate::list::ListArcSafe<$num> for $t {
|
||||||
|
unsafe fn on_create_list_arc_from_unique(self: ::core::pin::Pin<&mut Self>) {}
|
||||||
|
unsafe fn on_drop_list_arc(&self) {}
|
||||||
|
}
|
||||||
|
$crate::list::impl_list_arc_safe! { $($rest)* }
|
||||||
|
};
|
||||||
|
|
||||||
|
(impl$({$($generics:tt)*})? ListArcSafe<$num:tt> for $t:ty {
|
||||||
|
tracked_by $field:ident : $fty:ty;
|
||||||
|
} $($rest:tt)*) => {
|
||||||
|
impl$(<$($generics)*>)? $crate::list::ListArcSafe<$num> for $t {
|
||||||
|
unsafe fn on_create_list_arc_from_unique(self: ::core::pin::Pin<&mut Self>) {
|
||||||
|
$crate::assert_pinned!($t, $field, $fty, inline);
|
||||||
|
|
||||||
|
// SAFETY: This field is structurally pinned as per the above assertion.
|
||||||
|
let field = unsafe {
|
||||||
|
::core::pin::Pin::map_unchecked_mut(self, |me| &mut me.$field)
|
||||||
|
};
|
||||||
|
// SAFETY: The caller promises that there is no `ListArc`.
|
||||||
|
unsafe {
|
||||||
|
<$fty as $crate::list::ListArcSafe<$num>>::on_create_list_arc_from_unique(field)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
unsafe fn on_drop_list_arc(&self) {
|
||||||
|
// SAFETY: The caller promises that there is no `ListArc` reference, and also
|
||||||
|
// promises that the tracking thinks there is a `ListArc` reference.
|
||||||
|
unsafe { <$fty as $crate::list::ListArcSafe<$num>>::on_drop_list_arc(&self.$field) };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
unsafe impl$(<$($generics)*>)? $crate::list::TryNewListArc<$num> for $t
|
||||||
|
where
|
||||||
|
$fty: TryNewListArc<$num>,
|
||||||
|
{
|
||||||
|
fn try_new_list_arc(&self) -> bool {
|
||||||
|
<$fty as $crate::list::TryNewListArc<$num>>::try_new_list_arc(&self.$field)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$crate::list::impl_list_arc_safe! { $($rest)* }
|
||||||
|
};
|
||||||
|
|
||||||
|
() => {};
|
||||||
|
}
|
||||||
|
pub use impl_list_arc_safe;
|
||||||
|
|
||||||
|
/// A wrapper around [`Arc`] that's guaranteed unique for the given id.
|
||||||
|
///
|
||||||
|
/// The `ListArc` type can be thought of as a special reference to a refcounted object that owns the
|
||||||
|
/// permission to manipulate the `next`/`prev` pointers stored in the refcounted object. By ensuring
|
||||||
|
/// that each object has only one `ListArc` reference, the owner of that reference is assured
|
||||||
|
/// exclusive access to the `next`/`prev` pointers. When a `ListArc` is inserted into a [`List`],
|
||||||
|
/// the [`List`] takes ownership of the `ListArc` reference.
|
||||||
|
///
|
||||||
|
/// There are various strategies to ensuring that a value has only one `ListArc` reference. The
|
||||||
|
/// simplest is to convert a [`UniqueArc`] into a `ListArc`. However, the refcounted object could
|
||||||
|
/// also keep track of whether a `ListArc` exists using a boolean, which could allow for the
|
||||||
|
/// creation of new `ListArc` references from an [`Arc`] reference. Whatever strategy is used, the
|
||||||
|
/// relevant tracking is referred to as "the tracking inside `T`", and the [`ListArcSafe`] trait
|
||||||
|
/// (and its subtraits) are used to update the tracking when a `ListArc` is created or destroyed.
|
||||||
|
///
|
||||||
|
/// Note that we allow the case where the tracking inside `T` thinks that a `ListArc` exists, but
|
||||||
|
/// actually, there isn't a `ListArc`. However, we do not allow the opposite situation where a
|
||||||
|
/// `ListArc` exists, but the tracking thinks it doesn't. This is because the former can at most
|
||||||
|
/// result in us failing to create a `ListArc` when the operation could succeed, whereas the latter
|
||||||
|
/// can result in the creation of two `ListArc` references.
|
||||||
|
///
|
||||||
|
/// While this `ListArc` is unique for the given id, there still might exist normal `Arc`
|
||||||
|
/// references to the object.
|
||||||
|
///
|
||||||
|
/// # Invariants
|
||||||
|
///
|
||||||
|
/// * Each reference counted object has at most one `ListArc` for each value of `ID`.
|
||||||
|
/// * The tracking inside `T` is aware that a `ListArc` reference exists.
|
||||||
|
///
|
||||||
|
/// [`List`]: crate::list::List
|
||||||
|
#[repr(transparent)]
|
||||||
|
pub struct ListArc<T, const ID: u64 = 0>
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID> + ?Sized,
|
||||||
|
{
|
||||||
|
arc: Arc<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: ListArcSafe<ID>, const ID: u64> ListArc<T, ID> {
|
||||||
|
/// Constructs a new reference counted instance of `T`.
|
||||||
|
#[inline]
|
||||||
|
pub fn new(contents: T, flags: Flags) -> Result<Self, AllocError> {
|
||||||
|
Ok(Self::from(UniqueArc::new(contents, flags)?))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Use the given initializer to in-place initialize a `T`.
|
||||||
|
///
|
||||||
|
/// If `T: !Unpin` it will not be able to move afterwards.
|
||||||
|
// We don't implement `InPlaceInit` because `ListArc` is implicitly pinned. This is similar to
|
||||||
|
// what we do for `Arc`.
|
||||||
|
#[inline]
|
||||||
|
pub fn pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> Result<Self, E>
|
||||||
|
where
|
||||||
|
E: From<AllocError>,
|
||||||
|
{
|
||||||
|
Ok(Self::from(UniqueArc::try_pin_init(init, flags)?))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Use the given initializer to in-place initialize a `T`.
|
||||||
|
///
|
||||||
|
/// This is equivalent to [`ListArc<T>::pin_init`], since a [`ListArc`] is always pinned.
|
||||||
|
#[inline]
|
||||||
|
pub fn init<E>(init: impl Init<T, E>, flags: Flags) -> Result<Self, E>
|
||||||
|
where
|
||||||
|
E: From<AllocError>,
|
||||||
|
{
|
||||||
|
Ok(Self::from(UniqueArc::try_init(init, flags)?))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T, const ID: u64> From<UniqueArc<T>> for ListArc<T, ID>
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID> + ?Sized,
|
||||||
|
{
|
||||||
|
/// Convert a [`UniqueArc`] into a [`ListArc`].
|
||||||
|
#[inline]
|
||||||
|
fn from(unique: UniqueArc<T>) -> Self {
|
||||||
|
Self::from(Pin::from(unique))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T, const ID: u64> From<Pin<UniqueArc<T>>> for ListArc<T, ID>
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID> + ?Sized,
|
||||||
|
{
|
||||||
|
/// Convert a pinned [`UniqueArc`] into a [`ListArc`].
|
||||||
|
#[inline]
|
||||||
|
fn from(mut unique: Pin<UniqueArc<T>>) -> Self {
|
||||||
|
// SAFETY: We have a `UniqueArc`, so there is no `ListArc`.
|
||||||
|
unsafe { T::on_create_list_arc_from_unique(unique.as_mut()) };
|
||||||
|
let arc = Arc::from(unique);
|
||||||
|
// SAFETY: We just called `on_create_list_arc_from_unique` on an arc without a `ListArc`,
|
||||||
|
// so we can create a `ListArc`.
|
||||||
|
unsafe { Self::transmute_from_arc(arc) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T, const ID: u64> ListArc<T, ID>
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID> + ?Sized,
|
||||||
|
{
|
||||||
|
/// Creates two `ListArc`s from a [`UniqueArc`].
|
||||||
|
///
|
||||||
|
/// The two ids must be different.
|
||||||
|
#[inline]
|
||||||
|
pub fn pair_from_unique<const ID2: u64>(unique: UniqueArc<T>) -> (Self, ListArc<T, ID2>)
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID2>,
|
||||||
|
{
|
||||||
|
Self::pair_from_pin_unique(Pin::from(unique))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates two `ListArc`s from a pinned [`UniqueArc`].
|
||||||
|
///
|
||||||
|
/// The two ids must be different.
|
||||||
|
#[inline]
|
||||||
|
pub fn pair_from_pin_unique<const ID2: u64>(
|
||||||
|
mut unique: Pin<UniqueArc<T>>,
|
||||||
|
) -> (Self, ListArc<T, ID2>)
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID2>,
|
||||||
|
{
|
||||||
|
build_assert!(ID != ID2);
|
||||||
|
|
||||||
|
// SAFETY: We have a `UniqueArc`, so there is no `ListArc`.
|
||||||
|
unsafe { <T as ListArcSafe<ID>>::on_create_list_arc_from_unique(unique.as_mut()) };
|
||||||
|
// SAFETY: We have a `UniqueArc`, so there is no `ListArc`.
|
||||||
|
unsafe { <T as ListArcSafe<ID2>>::on_create_list_arc_from_unique(unique.as_mut()) };
|
||||||
|
|
||||||
|
let arc1 = Arc::from(unique);
|
||||||
|
let arc2 = Arc::clone(&arc1);
|
||||||
|
|
||||||
|
// SAFETY: We just called `on_create_list_arc_from_unique` on an arc without a `ListArc`
|
||||||
|
// for both IDs (which are different), so we can create two `ListArc`s.
|
||||||
|
unsafe {
|
||||||
|
(
|
||||||
|
Self::transmute_from_arc(arc1),
|
||||||
|
ListArc::transmute_from_arc(arc2),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Try to create a new `ListArc`.
|
||||||
|
///
|
||||||
|
/// This fails if this value already has a `ListArc`.
|
||||||
|
pub fn try_from_arc(arc: Arc<T>) -> Result<Self, Arc<T>>
|
||||||
|
where
|
||||||
|
T: TryNewListArc<ID>,
|
||||||
|
{
|
||||||
|
if arc.try_new_list_arc() {
|
||||||
|
// SAFETY: The `try_new_list_arc` method returned true, so we made the tracking think
|
||||||
|
// that a `ListArc` exists. This lets us create a `ListArc`.
|
||||||
|
Ok(unsafe { Self::transmute_from_arc(arc) })
|
||||||
|
} else {
|
||||||
|
Err(arc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Try to create a new `ListArc`.
|
||||||
|
///
|
||||||
|
/// This fails if this value already has a `ListArc`.
|
||||||
|
pub fn try_from_arc_borrow(arc: ArcBorrow<'_, T>) -> Option<Self>
|
||||||
|
where
|
||||||
|
T: TryNewListArc<ID>,
|
||||||
|
{
|
||||||
|
if arc.try_new_list_arc() {
|
||||||
|
// SAFETY: The `try_new_list_arc` method returned true, so we made the tracking think
|
||||||
|
// that a `ListArc` exists. This lets us create a `ListArc`.
|
||||||
|
Some(unsafe { Self::transmute_from_arc(Arc::from(arc)) })
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Try to create a new `ListArc`.
|
||||||
|
///
|
||||||
|
/// If it's not possible to create a new `ListArc`, then the `Arc` is dropped. This will never
|
||||||
|
/// run the destructor of the value.
|
||||||
|
pub fn try_from_arc_or_drop(arc: Arc<T>) -> Option<Self>
|
||||||
|
where
|
||||||
|
T: TryNewListArc<ID>,
|
||||||
|
{
|
||||||
|
match Self::try_from_arc(arc) {
|
||||||
|
Ok(list_arc) => Some(list_arc),
|
||||||
|
Err(arc) => Arc::into_unique_or_drop(arc).map(Self::from),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Transmutes an [`Arc`] into a `ListArc` without updating the tracking inside `T`.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// * The value must not already have a `ListArc` reference.
|
||||||
|
/// * The tracking inside `T` must think that there is a `ListArc` reference.
|
||||||
|
#[inline]
|
||||||
|
unsafe fn transmute_from_arc(arc: Arc<T>) -> Self {
|
||||||
|
// INVARIANT: By the safety requirements, the invariants on `ListArc` are satisfied.
|
||||||
|
Self { arc }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Transmutes a `ListArc` into an [`Arc`] without updating the tracking inside `T`.
|
||||||
|
///
|
||||||
|
/// After this call, the tracking inside `T` will still think that there is a `ListArc`
|
||||||
|
/// reference.
|
||||||
|
#[inline]
|
||||||
|
fn transmute_to_arc(self) -> Arc<T> {
|
||||||
|
// Use a transmute to skip destructor.
|
||||||
|
//
|
||||||
|
// SAFETY: ListArc is repr(transparent).
|
||||||
|
unsafe { core::mem::transmute(self) }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Convert ownership of this `ListArc` into a raw pointer.
|
||||||
|
///
|
||||||
|
/// The returned pointer is indistinguishable from pointers returned by [`Arc::into_raw`]. The
|
||||||
|
/// tracking inside `T` will still think that a `ListArc` exists after this call.
|
||||||
|
#[inline]
|
||||||
|
pub fn into_raw(self) -> *const T {
|
||||||
|
Arc::into_raw(Self::transmute_to_arc(self))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Take ownership of the `ListArc` from a raw pointer.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// * `ptr` must satisfy the safety requirements of [`Arc::from_raw`].
|
||||||
|
/// * The value must not already have a `ListArc` reference.
|
||||||
|
/// * The tracking inside `T` must think that there is a `ListArc` reference.
|
||||||
|
#[inline]
|
||||||
|
pub unsafe fn from_raw(ptr: *const T) -> Self {
|
||||||
|
// SAFETY: The pointer satisfies the safety requirements for `Arc::from_raw`.
|
||||||
|
let arc = unsafe { Arc::from_raw(ptr) };
|
||||||
|
// SAFETY: The value doesn't already have a `ListArc` reference, but the tracking thinks it
|
||||||
|
// does.
|
||||||
|
unsafe { Self::transmute_from_arc(arc) }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Converts the `ListArc` into an [`Arc`].
|
||||||
|
#[inline]
|
||||||
|
pub fn into_arc(self) -> Arc<T> {
|
||||||
|
let arc = Self::transmute_to_arc(self);
|
||||||
|
// SAFETY: There is no longer a `ListArc`, but the tracking thinks there is.
|
||||||
|
unsafe { T::on_drop_list_arc(&arc) };
|
||||||
|
arc
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Clone a `ListArc` into an [`Arc`].
|
||||||
|
#[inline]
|
||||||
|
pub fn clone_arc(&self) -> Arc<T> {
|
||||||
|
self.arc.clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a reference to an [`Arc`] from the given [`ListArc`].
|
||||||
|
///
|
||||||
|
/// This is useful when the argument of a function call is an [`&Arc`] (e.g., in a method
|
||||||
|
/// receiver), but we have a [`ListArc`] instead.
|
||||||
|
///
|
||||||
|
/// [`&Arc`]: Arc
|
||||||
|
#[inline]
|
||||||
|
pub fn as_arc(&self) -> &Arc<T> {
|
||||||
|
&self.arc
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns an [`ArcBorrow`] from the given [`ListArc`].
|
||||||
|
///
|
||||||
|
/// This is useful when the argument of a function call is an [`ArcBorrow`] (e.g., in a method
|
||||||
|
/// receiver), but we have an [`Arc`] instead. Getting an [`ArcBorrow`] is free when optimised.
|
||||||
|
#[inline]
|
||||||
|
pub fn as_arc_borrow(&self) -> ArcBorrow<'_, T> {
|
||||||
|
self.arc.as_arc_borrow()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Compare whether two [`ListArc`] pointers reference the same underlying object.
|
||||||
|
#[inline]
|
||||||
|
pub fn ptr_eq(this: &Self, other: &Self) -> bool {
|
||||||
|
Arc::ptr_eq(&this.arc, &other.arc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T, const ID: u64> Deref for ListArc<T, ID>
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID> + ?Sized,
|
||||||
|
{
|
||||||
|
type Target = T;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
self.arc.deref()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T, const ID: u64> Drop for ListArc<T, ID>
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID> + ?Sized,
|
||||||
|
{
|
||||||
|
#[inline]
|
||||||
|
fn drop(&mut self) {
|
||||||
|
// SAFETY: There is no longer a `ListArc`, but the tracking thinks there is by the type
|
||||||
|
// invariants on `Self`.
|
||||||
|
unsafe { T::on_drop_list_arc(&self.arc) };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T, const ID: u64> AsRef<Arc<T>> for ListArc<T, ID>
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID> + ?Sized,
|
||||||
|
{
|
||||||
|
#[inline]
|
||||||
|
fn as_ref(&self) -> &Arc<T> {
|
||||||
|
self.as_arc()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// This is to allow [`ListArc`] (and variants) to be used as the type of `self`.
|
||||||
|
impl<T, const ID: u64> core::ops::Receiver for ListArc<T, ID> where T: ListArcSafe<ID> + ?Sized {}
|
||||||
|
|
||||||
|
// This is to allow coercion from `ListArc<T>` to `ListArc<U>` if `T` can be converted to the
|
||||||
|
// dynamically-sized type (DST) `U`.
|
||||||
|
impl<T, U, const ID: u64> core::ops::CoerceUnsized<ListArc<U, ID>> for ListArc<T, ID>
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID> + Unsize<U> + ?Sized,
|
||||||
|
U: ListArcSafe<ID> + ?Sized,
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
// This is to allow `ListArc<U>` to be dispatched on when `ListArc<T>` can be coerced into
|
||||||
|
// `ListArc<U>`.
|
||||||
|
impl<T, U, const ID: u64> core::ops::DispatchFromDyn<ListArc<U, ID>> for ListArc<T, ID>
|
||||||
|
where
|
||||||
|
T: ListArcSafe<ID> + Unsize<U> + ?Sized,
|
||||||
|
U: ListArcSafe<ID> + ?Sized,
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A utility for tracking whether a [`ListArc`] exists using an atomic.
|
||||||
|
///
|
||||||
|
/// # Invariant
|
||||||
|
///
|
||||||
|
/// If the boolean is `false`, then there is no [`ListArc`] for this value.
|
||||||
|
#[repr(transparent)]
|
||||||
|
pub struct AtomicTracker<const ID: u64 = 0> {
|
||||||
|
inner: AtomicBool,
|
||||||
|
// This value needs to be pinned to justify the INVARIANT: comment in `AtomicTracker::new`.
|
||||||
|
_pin: PhantomPinned,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<const ID: u64> AtomicTracker<ID> {
|
||||||
|
/// Creates a new initializer for this type.
|
||||||
|
pub fn new() -> impl PinInit<Self> {
|
||||||
|
// INVARIANT: Pin-init initializers can't be used on an existing `Arc`, so this value will
|
||||||
|
// not be constructed in an `Arc` that already has a `ListArc`.
|
||||||
|
Self {
|
||||||
|
inner: AtomicBool::new(false),
|
||||||
|
_pin: PhantomPinned,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn project_inner(self: Pin<&mut Self>) -> &mut AtomicBool {
|
||||||
|
// SAFETY: The `inner` field is not structurally pinned, so we may obtain a mutable
|
||||||
|
// reference to it even if we only have a pinned reference to `self`.
|
||||||
|
unsafe { &mut Pin::into_inner_unchecked(self).inner }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<const ID: u64> ListArcSafe<ID> for AtomicTracker<ID> {
|
||||||
|
unsafe fn on_create_list_arc_from_unique(self: Pin<&mut Self>) {
|
||||||
|
// INVARIANT: We just created a ListArc, so the boolean should be true.
|
||||||
|
*self.project_inner().get_mut() = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe fn on_drop_list_arc(&self) {
|
||||||
|
// INVARIANT: We just dropped a ListArc, so the boolean should be false.
|
||||||
|
self.inner.store(false, Ordering::Release);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: If this method returns `true`, then by the type invariant there is no `ListArc` before
|
||||||
|
// this call, so it is okay to create a new `ListArc`.
|
||||||
|
//
|
||||||
|
// The acquire ordering will synchronize with the release store from the destruction of any
|
||||||
|
// previous `ListArc`, so if there was a previous `ListArc`, then the destruction of the previous
|
||||||
|
// `ListArc` happens-before the creation of the new `ListArc`.
|
||||||
|
unsafe impl<const ID: u64> TryNewListArc<ID> for AtomicTracker<ID> {
|
||||||
|
fn try_new_list_arc(&self) -> bool {
|
||||||
|
// INVARIANT: If this method returns true, then the boolean used to be false, and is no
|
||||||
|
// longer false, so it is okay for the caller to create a new [`ListArc`].
|
||||||
|
self.inner
|
||||||
|
.compare_exchange(false, true, Ordering::Acquire, Ordering::Relaxed)
|
||||||
|
.is_ok()
|
||||||
|
}
|
||||||
|
}
|
96
rust/kernel/list/arc_field.rs
Normal file
96
rust/kernel/list/arc_field.rs
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
// Copyright (C) 2024 Google LLC.
|
||||||
|
|
||||||
|
//! A field that is exclusively owned by a [`ListArc`].
|
||||||
|
//!
|
||||||
|
//! This can be used to have reference counted struct where one of the reference counted pointers
|
||||||
|
//! has exclusive access to a field of the struct.
|
||||||
|
//!
|
||||||
|
//! [`ListArc`]: crate::list::ListArc
|
||||||
|
|
||||||
|
use core::cell::UnsafeCell;
|
||||||
|
|
||||||
|
/// A field owned by a specific [`ListArc`].
|
||||||
|
///
|
||||||
|
/// [`ListArc`]: crate::list::ListArc
|
||||||
|
pub struct ListArcField<T, const ID: u64 = 0> {
|
||||||
|
value: UnsafeCell<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: If the inner type is thread-safe, then it's also okay for `ListArc` to be thread-safe.
|
||||||
|
unsafe impl<T: Send + Sync, const ID: u64> Send for ListArcField<T, ID> {}
|
||||||
|
// SAFETY: If the inner type is thread-safe, then it's also okay for `ListArc` to be thread-safe.
|
||||||
|
unsafe impl<T: Send + Sync, const ID: u64> Sync for ListArcField<T, ID> {}
|
||||||
|
|
||||||
|
impl<T, const ID: u64> ListArcField<T, ID> {
|
||||||
|
/// Creates a new `ListArcField`.
|
||||||
|
pub fn new(value: T) -> Self {
|
||||||
|
Self {
|
||||||
|
value: UnsafeCell::new(value),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Access the value when we have exclusive access to the `ListArcField`.
|
||||||
|
///
|
||||||
|
/// This allows access to the field using an `UniqueArc` instead of a `ListArc`.
|
||||||
|
pub fn get_mut(&mut self) -> &mut T {
|
||||||
|
self.value.get_mut()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Unsafely assert that you have shared access to the `ListArc` for this field.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// The caller must have shared access to the `ListArc<ID>` containing the struct with this
|
||||||
|
/// field for the duration of the returned reference.
|
||||||
|
pub unsafe fn assert_ref(&self) -> &T {
|
||||||
|
// SAFETY: The caller has shared access to the `ListArc`, so they also have shared access
|
||||||
|
// to this field.
|
||||||
|
unsafe { &*self.value.get() }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Unsafely assert that you have mutable access to the `ListArc` for this field.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// The caller must have mutable access to the `ListArc<ID>` containing the struct with this
|
||||||
|
/// field for the duration of the returned reference.
|
||||||
|
#[allow(clippy::mut_from_ref)]
|
||||||
|
pub unsafe fn assert_mut(&self) -> &mut T {
|
||||||
|
// SAFETY: The caller has exclusive access to the `ListArc`, so they also have exclusive
|
||||||
|
// access to this field.
|
||||||
|
unsafe { &mut *self.value.get() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Defines getters for a [`ListArcField`].
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! define_list_arc_field_getter {
|
||||||
|
($pub:vis fn $name:ident(&self $(<$id:tt>)?) -> &$typ:ty { $field:ident }
|
||||||
|
$($rest:tt)*
|
||||||
|
) => {
|
||||||
|
$pub fn $name<'a>(self: &'a $crate::list::ListArc<Self $(, $id)?>) -> &'a $typ {
|
||||||
|
let field = &(&**self).$field;
|
||||||
|
// SAFETY: We have a shared reference to the `ListArc`.
|
||||||
|
unsafe { $crate::list::ListArcField::<$typ $(, $id)?>::assert_ref(field) }
|
||||||
|
}
|
||||||
|
|
||||||
|
$crate::list::define_list_arc_field_getter!($($rest)*);
|
||||||
|
};
|
||||||
|
|
||||||
|
($pub:vis fn $name:ident(&mut self $(<$id:tt>)?) -> &mut $typ:ty { $field:ident }
|
||||||
|
$($rest:tt)*
|
||||||
|
) => {
|
||||||
|
$pub fn $name<'a>(self: &'a mut $crate::list::ListArc<Self $(, $id)?>) -> &'a mut $typ {
|
||||||
|
let field = &(&**self).$field;
|
||||||
|
// SAFETY: We have a mutable reference to the `ListArc`.
|
||||||
|
unsafe { $crate::list::ListArcField::<$typ $(, $id)?>::assert_mut(field) }
|
||||||
|
}
|
||||||
|
|
||||||
|
$crate::list::define_list_arc_field_getter!($($rest)*);
|
||||||
|
};
|
||||||
|
|
||||||
|
() => {};
|
||||||
|
}
|
||||||
|
pub use define_list_arc_field_getter;
|
274
rust/kernel/list/impl_list_item_mod.rs
Normal file
274
rust/kernel/list/impl_list_item_mod.rs
Normal file
@ -0,0 +1,274 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
|
||||||
|
// Copyright (C) 2024 Google LLC.
|
||||||
|
|
||||||
|
//! Helpers for implementing list traits safely.
|
||||||
|
|
||||||
|
use crate::list::ListLinks;
|
||||||
|
|
||||||
|
/// Declares that this type has a `ListLinks<ID>` field at a fixed offset.
|
||||||
|
///
|
||||||
|
/// This trait is only used to help implement `ListItem` safely. If `ListItem` is implemented
|
||||||
|
/// manually, then this trait is not needed. Use the [`impl_has_list_links!`] macro to implement
|
||||||
|
/// this trait.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// All values of this type must have a `ListLinks<ID>` field at the given offset.
|
||||||
|
///
|
||||||
|
/// The behavior of `raw_get_list_links` must not be changed.
|
||||||
|
pub unsafe trait HasListLinks<const ID: u64 = 0> {
|
||||||
|
/// The offset of the `ListLinks` field.
|
||||||
|
const OFFSET: usize;
|
||||||
|
|
||||||
|
/// Returns a pointer to the [`ListLinks<T, ID>`] field.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// The provided pointer must point at a valid struct of type `Self`.
|
||||||
|
///
|
||||||
|
/// [`ListLinks<T, ID>`]: ListLinks
|
||||||
|
// We don't really need this method, but it's necessary for the implementation of
|
||||||
|
// `impl_has_list_links!` to be correct.
|
||||||
|
#[inline]
|
||||||
|
unsafe fn raw_get_list_links(ptr: *mut Self) -> *mut ListLinks<ID> {
|
||||||
|
// SAFETY: The caller promises that the pointer is valid. The implementer promises that the
|
||||||
|
// `OFFSET` constant is correct.
|
||||||
|
unsafe { (ptr as *mut u8).add(Self::OFFSET) as *mut ListLinks<ID> }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Implements the [`HasListLinks`] trait for the given type.
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! impl_has_list_links {
|
||||||
|
($(impl$(<$($implarg:ident),*>)?
|
||||||
|
HasListLinks$(<$id:tt>)?
|
||||||
|
for $self:ident $(<$($selfarg:ty),*>)?
|
||||||
|
{ self$(.$field:ident)* }
|
||||||
|
)*) => {$(
|
||||||
|
// SAFETY: The implementation of `raw_get_list_links` only compiles if the field has the
|
||||||
|
// right type.
|
||||||
|
//
|
||||||
|
// The behavior of `raw_get_list_links` is not changed since the `addr_of_mut!` macro is
|
||||||
|
// equivalent to the pointer offset operation in the trait definition.
|
||||||
|
unsafe impl$(<$($implarg),*>)? $crate::list::HasListLinks$(<$id>)? for
|
||||||
|
$self $(<$($selfarg),*>)?
|
||||||
|
{
|
||||||
|
const OFFSET: usize = ::core::mem::offset_of!(Self, $($field).*) as usize;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
unsafe fn raw_get_list_links(ptr: *mut Self) -> *mut $crate::list::ListLinks$(<$id>)? {
|
||||||
|
// SAFETY: The caller promises that the pointer is not dangling. We know that this
|
||||||
|
// expression doesn't follow any pointers, as the `offset_of!` invocation above
|
||||||
|
// would otherwise not compile.
|
||||||
|
unsafe { ::core::ptr::addr_of_mut!((*ptr)$(.$field)*) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)*};
|
||||||
|
}
|
||||||
|
pub use impl_has_list_links;
|
||||||
|
|
||||||
|
/// Declares that the `ListLinks<ID>` field in this struct is inside a `ListLinksSelfPtr<T, ID>`.
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// The `ListLinks<ID>` field of this struct at the offset `HasListLinks<ID>::OFFSET` must be
|
||||||
|
/// inside a `ListLinksSelfPtr<T, ID>`.
|
||||||
|
pub unsafe trait HasSelfPtr<T: ?Sized, const ID: u64 = 0>
|
||||||
|
where
|
||||||
|
Self: HasListLinks<ID>,
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Implements the [`HasListLinks`] and [`HasSelfPtr`] traits for the given type.
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! impl_has_list_links_self_ptr {
|
||||||
|
($(impl$({$($implarg:tt)*})?
|
||||||
|
HasSelfPtr<$item_type:ty $(, $id:tt)?>
|
||||||
|
for $self:ident $(<$($selfarg:ty),*>)?
|
||||||
|
{ self.$field:ident }
|
||||||
|
)*) => {$(
|
||||||
|
// SAFETY: The implementation of `raw_get_list_links` only compiles if the field has the
|
||||||
|
// right type.
|
||||||
|
unsafe impl$(<$($implarg)*>)? $crate::list::HasSelfPtr<$item_type $(, $id)?> for
|
||||||
|
$self $(<$($selfarg),*>)?
|
||||||
|
{}
|
||||||
|
|
||||||
|
unsafe impl$(<$($implarg)*>)? $crate::list::HasListLinks$(<$id>)? for
|
||||||
|
$self $(<$($selfarg),*>)?
|
||||||
|
{
|
||||||
|
const OFFSET: usize = ::core::mem::offset_of!(Self, $field) as usize;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
unsafe fn raw_get_list_links(ptr: *mut Self) -> *mut $crate::list::ListLinks$(<$id>)? {
|
||||||
|
// SAFETY: The caller promises that the pointer is not dangling.
|
||||||
|
let ptr: *mut $crate::list::ListLinksSelfPtr<$item_type $(, $id)?> =
|
||||||
|
unsafe { ::core::ptr::addr_of_mut!((*ptr).$field) };
|
||||||
|
ptr.cast()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)*};
|
||||||
|
}
|
||||||
|
pub use impl_has_list_links_self_ptr;
|
||||||
|
|
||||||
|
/// Implements the [`ListItem`] trait for the given type.
|
||||||
|
///
|
||||||
|
/// Requires that the type implements [`HasListLinks`]. Use the [`impl_has_list_links!`] macro to
|
||||||
|
/// implement that trait.
|
||||||
|
///
|
||||||
|
/// [`ListItem`]: crate::list::ListItem
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! impl_list_item {
|
||||||
|
(
|
||||||
|
$(impl$({$($generics:tt)*})? ListItem<$num:tt> for $t:ty {
|
||||||
|
using ListLinks;
|
||||||
|
})*
|
||||||
|
) => {$(
|
||||||
|
// SAFETY: See GUARANTEES comment on each method.
|
||||||
|
unsafe impl$(<$($generics)*>)? $crate::list::ListItem<$num> for $t {
|
||||||
|
// GUARANTEES:
|
||||||
|
// * This returns the same pointer as `prepare_to_insert` because `prepare_to_insert`
|
||||||
|
// is implemented in terms of `view_links`.
|
||||||
|
// * By the type invariants of `ListLinks`, the `ListLinks` has two null pointers when
|
||||||
|
// this value is not in a list.
|
||||||
|
unsafe fn view_links(me: *const Self) -> *mut $crate::list::ListLinks<$num> {
|
||||||
|
// SAFETY: The caller guarantees that `me` points at a valid value of type `Self`.
|
||||||
|
unsafe {
|
||||||
|
<Self as $crate::list::HasListLinks<$num>>::raw_get_list_links(me.cast_mut())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GUARANTEES:
|
||||||
|
// * `me` originates from the most recent call to `prepare_to_insert`, which just added
|
||||||
|
// `offset` to the pointer passed to `prepare_to_insert`. This method subtracts
|
||||||
|
// `offset` from `me` so it returns the pointer originally passed to
|
||||||
|
// `prepare_to_insert`.
|
||||||
|
// * The pointer remains valid until the next call to `post_remove` because the caller
|
||||||
|
// of the most recent call to `prepare_to_insert` promised to retain ownership of the
|
||||||
|
// `ListArc` containing `Self` until the next call to `post_remove`. The value cannot
|
||||||
|
// be destroyed while a `ListArc` reference exists.
|
||||||
|
unsafe fn view_value(me: *mut $crate::list::ListLinks<$num>) -> *const Self {
|
||||||
|
let offset = <Self as $crate::list::HasListLinks<$num>>::OFFSET;
|
||||||
|
// SAFETY: `me` originates from the most recent call to `prepare_to_insert`, so it
|
||||||
|
// points at the field at offset `offset` in a value of type `Self`. Thus,
|
||||||
|
// subtracting `offset` from `me` is still in-bounds of the allocation.
|
||||||
|
unsafe { (me as *const u8).sub(offset) as *const Self }
|
||||||
|
}
|
||||||
|
|
||||||
|
// GUARANTEES:
|
||||||
|
// This implementation of `ListItem` will not give out exclusive access to the same
|
||||||
|
// `ListLinks` several times because calls to `prepare_to_insert` and `post_remove`
|
||||||
|
// must alternate and exclusive access is given up when `post_remove` is called.
|
||||||
|
//
|
||||||
|
// Other invocations of `impl_list_item!` also cannot give out exclusive access to the
|
||||||
|
// same `ListLinks` because you can only implement `ListItem` once for each value of
|
||||||
|
// `ID`, and the `ListLinks` fields only work with the specified `ID`.
|
||||||
|
unsafe fn prepare_to_insert(me: *const Self) -> *mut $crate::list::ListLinks<$num> {
|
||||||
|
// SAFETY: The caller promises that `me` points at a valid value.
|
||||||
|
unsafe { <Self as $crate::list::ListItem<$num>>::view_links(me) }
|
||||||
|
}
|
||||||
|
|
||||||
|
// GUARANTEES:
|
||||||
|
// * `me` originates from the most recent call to `prepare_to_insert`, which just added
|
||||||
|
// `offset` to the pointer passed to `prepare_to_insert`. This method subtracts
|
||||||
|
// `offset` from `me` so it returns the pointer originally passed to
|
||||||
|
// `prepare_to_insert`.
|
||||||
|
unsafe fn post_remove(me: *mut $crate::list::ListLinks<$num>) -> *const Self {
|
||||||
|
let offset = <Self as $crate::list::HasListLinks<$num>>::OFFSET;
|
||||||
|
// SAFETY: `me` originates from the most recent call to `prepare_to_insert`, so it
|
||||||
|
// points at the field at offset `offset` in a value of type `Self`. Thus,
|
||||||
|
// subtracting `offset` from `me` is still in-bounds of the allocation.
|
||||||
|
unsafe { (me as *const u8).sub(offset) as *const Self }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)*};
|
||||||
|
|
||||||
|
(
|
||||||
|
$(impl$({$($generics:tt)*})? ListItem<$num:tt> for $t:ty {
|
||||||
|
using ListLinksSelfPtr;
|
||||||
|
})*
|
||||||
|
) => {$(
|
||||||
|
// SAFETY: See GUARANTEES comment on each method.
|
||||||
|
unsafe impl$(<$($generics)*>)? $crate::list::ListItem<$num> for $t {
|
||||||
|
// GUARANTEES:
|
||||||
|
// This implementation of `ListItem` will not give out exclusive access to the same
|
||||||
|
// `ListLinks` several times because calls to `prepare_to_insert` and `post_remove`
|
||||||
|
// must alternate and exclusive access is given up when `post_remove` is called.
|
||||||
|
//
|
||||||
|
// Other invocations of `impl_list_item!` also cannot give out exclusive access to the
|
||||||
|
// same `ListLinks` because you can only implement `ListItem` once for each value of
|
||||||
|
// `ID`, and the `ListLinks` fields only work with the specified `ID`.
|
||||||
|
unsafe fn prepare_to_insert(me: *const Self) -> *mut $crate::list::ListLinks<$num> {
|
||||||
|
// SAFETY: The caller promises that `me` points at a valid value of type `Self`.
|
||||||
|
let links_field = unsafe { <Self as $crate::list::ListItem<$num>>::view_links(me) };
|
||||||
|
|
||||||
|
let spoff = $crate::list::ListLinksSelfPtr::<Self, $num>::LIST_LINKS_SELF_PTR_OFFSET;
|
||||||
|
// Goes via the offset as the field is private.
|
||||||
|
//
|
||||||
|
// SAFETY: The constant is equal to `offset_of!(ListLinksSelfPtr, self_ptr)`, so
|
||||||
|
// the pointer stays in bounds of the allocation.
|
||||||
|
let self_ptr = unsafe { (links_field as *const u8).add(spoff) }
|
||||||
|
as *const $crate::types::Opaque<*const Self>;
|
||||||
|
let cell_inner = $crate::types::Opaque::raw_get(self_ptr);
|
||||||
|
|
||||||
|
// SAFETY: This value is not accessed in any other places than `prepare_to_insert`,
|
||||||
|
// `post_remove`, or `view_value`. By the safety requirements of those methods,
|
||||||
|
// none of these three methods may be called in parallel with this call to
|
||||||
|
// `prepare_to_insert`, so this write will not race with any other access to the
|
||||||
|
// value.
|
||||||
|
unsafe { ::core::ptr::write(cell_inner, me) };
|
||||||
|
|
||||||
|
links_field
|
||||||
|
}
|
||||||
|
|
||||||
|
// GUARANTEES:
|
||||||
|
// * This returns the same pointer as `prepare_to_insert` because `prepare_to_insert`
|
||||||
|
// returns the return value of `view_links`.
|
||||||
|
// * By the type invariants of `ListLinks`, the `ListLinks` has two null pointers when
|
||||||
|
// this value is not in a list.
|
||||||
|
unsafe fn view_links(me: *const Self) -> *mut $crate::list::ListLinks<$num> {
|
||||||
|
// SAFETY: The caller promises that `me` points at a valid value of type `Self`.
|
||||||
|
unsafe { <Self as HasListLinks<$num>>::raw_get_list_links(me.cast_mut()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
// This function is also used as the implementation of `post_remove`, so the caller
|
||||||
|
// may choose to satisfy the safety requirements of `post_remove` instead of the safety
|
||||||
|
// requirements for `view_value`.
|
||||||
|
//
|
||||||
|
// GUARANTEES: (always)
|
||||||
|
// * This returns the same pointer as the one passed to the most recent call to
|
||||||
|
// `prepare_to_insert` since that call wrote that pointer to this location. The value
|
||||||
|
// is only modified in `prepare_to_insert`, so it has not been modified since the
|
||||||
|
// most recent call.
|
||||||
|
//
|
||||||
|
// GUARANTEES: (only when using the `view_value` safety requirements)
|
||||||
|
// * The pointer remains valid until the next call to `post_remove` because the caller
|
||||||
|
// of the most recent call to `prepare_to_insert` promised to retain ownership of the
|
||||||
|
// `ListArc` containing `Self` until the next call to `post_remove`. The value cannot
|
||||||
|
// be destroyed while a `ListArc` reference exists.
|
||||||
|
unsafe fn view_value(links_field: *mut $crate::list::ListLinks<$num>) -> *const Self {
|
||||||
|
let spoff = $crate::list::ListLinksSelfPtr::<Self, $num>::LIST_LINKS_SELF_PTR_OFFSET;
|
||||||
|
// SAFETY: The constant is equal to `offset_of!(ListLinksSelfPtr, self_ptr)`, so
|
||||||
|
// the pointer stays in bounds of the allocation.
|
||||||
|
let self_ptr = unsafe { (links_field as *const u8).add(spoff) }
|
||||||
|
as *const ::core::cell::UnsafeCell<*const Self>;
|
||||||
|
let cell_inner = ::core::cell::UnsafeCell::raw_get(self_ptr);
|
||||||
|
// SAFETY: This is not a data race, because the only function that writes to this
|
||||||
|
// value is `prepare_to_insert`, but by the safety requirements the
|
||||||
|
// `prepare_to_insert` method may not be called in parallel with `view_value` or
|
||||||
|
// `post_remove`.
|
||||||
|
unsafe { ::core::ptr::read(cell_inner) }
|
||||||
|
}
|
||||||
|
|
||||||
|
// GUARANTEES:
|
||||||
|
// The first guarantee of `view_value` is exactly what `post_remove` guarantees.
|
||||||
|
unsafe fn post_remove(me: *mut $crate::list::ListLinks<$num>) -> *const Self {
|
||||||
|
// SAFETY: This specific implementation of `view_value` allows the caller to
|
||||||
|
// promise the safety requirements of `post_remove` instead of the safety
|
||||||
|
// requirements for `view_value`.
|
||||||
|
unsafe { <Self as $crate::list::ListItem<$num>>::view_value(me) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)*};
|
||||||
|
}
|
||||||
|
pub use impl_list_item;
|
@ -37,6 +37,6 @@ pub use super::error::{code::*, Error, Result};
|
|||||||
|
|
||||||
pub use super::{str::CStr, ThisModule};
|
pub use super::{str::CStr, ThisModule};
|
||||||
|
|
||||||
pub use super::init::{InPlaceInit, Init, PinInit};
|
pub use super::init::{InPlaceInit, InPlaceWrite, Init, PinInit};
|
||||||
|
|
||||||
pub use super::current;
|
pub use super::current;
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
//!
|
//!
|
||||||
//! C header: [`include/linux/printk.h`](srctree/include/linux/printk.h)
|
//! C header: [`include/linux/printk.h`](srctree/include/linux/printk.h)
|
||||||
//!
|
//!
|
||||||
//! Reference: <https://www.kernel.org/doc/html/latest/core-api/printk-basics.html>
|
//! Reference: <https://docs.kernel.org/core-api/printk-basics.html>
|
||||||
|
|
||||||
use core::{
|
use core::{
|
||||||
ffi::{c_char, c_void},
|
ffi::{c_char, c_void},
|
||||||
@ -197,7 +197,7 @@ macro_rules! print_macro (
|
|||||||
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
||||||
/// `alloc::format!` for information about the formatting syntax.
|
/// `alloc::format!` for information about the formatting syntax.
|
||||||
///
|
///
|
||||||
/// [`pr_emerg`]: https://www.kernel.org/doc/html/latest/core-api/printk-basics.html#c.pr_emerg
|
/// [`pr_emerg`]: https://docs.kernel.org/core-api/printk-basics.html#c.pr_emerg
|
||||||
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
@ -221,7 +221,7 @@ macro_rules! pr_emerg (
|
|||||||
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
||||||
/// `alloc::format!` for information about the formatting syntax.
|
/// `alloc::format!` for information about the formatting syntax.
|
||||||
///
|
///
|
||||||
/// [`pr_alert`]: https://www.kernel.org/doc/html/latest/core-api/printk-basics.html#c.pr_alert
|
/// [`pr_alert`]: https://docs.kernel.org/core-api/printk-basics.html#c.pr_alert
|
||||||
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
@ -245,7 +245,7 @@ macro_rules! pr_alert (
|
|||||||
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
||||||
/// `alloc::format!` for information about the formatting syntax.
|
/// `alloc::format!` for information about the formatting syntax.
|
||||||
///
|
///
|
||||||
/// [`pr_crit`]: https://www.kernel.org/doc/html/latest/core-api/printk-basics.html#c.pr_crit
|
/// [`pr_crit`]: https://docs.kernel.org/core-api/printk-basics.html#c.pr_crit
|
||||||
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
@ -269,7 +269,7 @@ macro_rules! pr_crit (
|
|||||||
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
||||||
/// `alloc::format!` for information about the formatting syntax.
|
/// `alloc::format!` for information about the formatting syntax.
|
||||||
///
|
///
|
||||||
/// [`pr_err`]: https://www.kernel.org/doc/html/latest/core-api/printk-basics.html#c.pr_err
|
/// [`pr_err`]: https://docs.kernel.org/core-api/printk-basics.html#c.pr_err
|
||||||
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
@ -293,7 +293,7 @@ macro_rules! pr_err (
|
|||||||
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
||||||
/// `alloc::format!` for information about the formatting syntax.
|
/// `alloc::format!` for information about the formatting syntax.
|
||||||
///
|
///
|
||||||
/// [`pr_warn`]: https://www.kernel.org/doc/html/latest/core-api/printk-basics.html#c.pr_warn
|
/// [`pr_warn`]: https://docs.kernel.org/core-api/printk-basics.html#c.pr_warn
|
||||||
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
@ -317,7 +317,7 @@ macro_rules! pr_warn (
|
|||||||
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
||||||
/// `alloc::format!` for information about the formatting syntax.
|
/// `alloc::format!` for information about the formatting syntax.
|
||||||
///
|
///
|
||||||
/// [`pr_notice`]: https://www.kernel.org/doc/html/latest/core-api/printk-basics.html#c.pr_notice
|
/// [`pr_notice`]: https://docs.kernel.org/core-api/printk-basics.html#c.pr_notice
|
||||||
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
@ -341,7 +341,7 @@ macro_rules! pr_notice (
|
|||||||
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
||||||
/// `alloc::format!` for information about the formatting syntax.
|
/// `alloc::format!` for information about the formatting syntax.
|
||||||
///
|
///
|
||||||
/// [`pr_info`]: https://www.kernel.org/doc/html/latest/core-api/printk-basics.html#c.pr_info
|
/// [`pr_info`]: https://docs.kernel.org/core-api/printk-basics.html#c.pr_info
|
||||||
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
@ -367,7 +367,7 @@ macro_rules! pr_info (
|
|||||||
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
/// Mimics the interface of [`std::print!`]. See [`core::fmt`] and
|
||||||
/// `alloc::format!` for information about the formatting syntax.
|
/// `alloc::format!` for information about the formatting syntax.
|
||||||
///
|
///
|
||||||
/// [`pr_debug`]: https://www.kernel.org/doc/html/latest/core-api/printk-basics.html#c.pr_debug
|
/// [`pr_debug`]: https://docs.kernel.org/core-api/printk-basics.html#c.pr_debug
|
||||||
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
@ -395,7 +395,7 @@ macro_rules! pr_debug (
|
|||||||
/// `alloc::format!` for information about the formatting syntax.
|
/// `alloc::format!` for information about the formatting syntax.
|
||||||
///
|
///
|
||||||
/// [`pr_info!`]: crate::pr_info!
|
/// [`pr_info!`]: crate::pr_info!
|
||||||
/// [`pr_cont`]: https://www.kernel.org/doc/html/latest/core-api/printk-basics.html#c.pr_cont
|
/// [`pr_cont`]: https://docs.kernel.org/core-api/printk-basics.html#c.pr_cont
|
||||||
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
/// [`std::print!`]: https://doc.rust-lang.org/std/macro.print.html
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
|
1278
rust/kernel/rbtree.rs
Normal file
1278
rust/kernel/rbtree.rs
Normal file
File diff suppressed because it is too large
Load Diff
@ -136,7 +136,7 @@
|
|||||||
///
|
///
|
||||||
/// [`std::dbg`]: https://doc.rust-lang.org/std/macro.dbg.html
|
/// [`std::dbg`]: https://doc.rust-lang.org/std/macro.dbg.html
|
||||||
/// [`eprintln`]: https://doc.rust-lang.org/std/macro.eprintln.html
|
/// [`eprintln`]: https://doc.rust-lang.org/std/macro.eprintln.html
|
||||||
/// [`printk`]: https://www.kernel.org/doc/html/latest/core-api/printk-basics.html
|
/// [`printk`]: https://docs.kernel.org/core-api/printk-basics.html
|
||||||
/// [`pr_info`]: crate::pr_info!
|
/// [`pr_info`]: crate::pr_info!
|
||||||
/// [`pr_debug`]: crate::pr_debug!
|
/// [`pr_debug`]: crate::pr_debug!
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
|
@ -12,12 +12,13 @@
|
|||||||
//! 2. It does not support weak references, which allows it to be half the size.
|
//! 2. It does not support weak references, which allows it to be half the size.
|
||||||
//! 3. It saturates the reference count instead of aborting when it goes over a threshold.
|
//! 3. It saturates the reference count instead of aborting when it goes over a threshold.
|
||||||
//! 4. It does not provide a `get_mut` method, so the ref counted object is pinned.
|
//! 4. It does not provide a `get_mut` method, so the ref counted object is pinned.
|
||||||
|
//! 5. The object in [`Arc`] is pinned implicitly.
|
||||||
//!
|
//!
|
||||||
//! [`Arc`]: https://doc.rust-lang.org/std/sync/struct.Arc.html
|
//! [`Arc`]: https://doc.rust-lang.org/std/sync/struct.Arc.html
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
alloc::{box_ext::BoxExt, AllocError, Flags},
|
alloc::{box_ext::BoxExt, AllocError, Flags},
|
||||||
error::{self, Error},
|
bindings,
|
||||||
init::{self, InPlaceInit, Init, PinInit},
|
init::{self, InPlaceInit, Init, PinInit},
|
||||||
try_init,
|
try_init,
|
||||||
types::{ForeignOwnable, Opaque},
|
types::{ForeignOwnable, Opaque},
|
||||||
@ -209,28 +210,6 @@ impl<T> Arc<T> {
|
|||||||
// `Arc` object.
|
// `Arc` object.
|
||||||
Ok(unsafe { Self::from_inner(Box::leak(inner).into()) })
|
Ok(unsafe { Self::from_inner(Box::leak(inner).into()) })
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Use the given initializer to in-place initialize a `T`.
|
|
||||||
///
|
|
||||||
/// If `T: !Unpin` it will not be able to move afterwards.
|
|
||||||
#[inline]
|
|
||||||
pub fn pin_init<E>(init: impl PinInit<T, E>, flags: Flags) -> error::Result<Self>
|
|
||||||
where
|
|
||||||
Error: From<E>,
|
|
||||||
{
|
|
||||||
UniqueArc::pin_init(init, flags).map(|u| u.into())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Use the given initializer to in-place initialize a `T`.
|
|
||||||
///
|
|
||||||
/// This is equivalent to [`Arc<T>::pin_init`], since an [`Arc`] is always pinned.
|
|
||||||
#[inline]
|
|
||||||
pub fn init<E>(init: impl Init<T, E>, flags: Flags) -> error::Result<Self>
|
|
||||||
where
|
|
||||||
Error: From<E>,
|
|
||||||
{
|
|
||||||
UniqueArc::init(init, flags).map(|u| u.into())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: ?Sized> Arc<T> {
|
impl<T: ?Sized> Arc<T> {
|
||||||
|
@ -7,8 +7,9 @@ use alloc::boxed::Box;
|
|||||||
use core::{
|
use core::{
|
||||||
cell::UnsafeCell,
|
cell::UnsafeCell,
|
||||||
marker::{PhantomData, PhantomPinned},
|
marker::{PhantomData, PhantomPinned},
|
||||||
mem::MaybeUninit,
|
mem::{ManuallyDrop, MaybeUninit},
|
||||||
ops::{Deref, DerefMut},
|
ops::{Deref, DerefMut},
|
||||||
|
pin::Pin,
|
||||||
ptr::NonNull,
|
ptr::NonNull,
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -26,7 +27,10 @@ pub trait ForeignOwnable: Sized {
|
|||||||
|
|
||||||
/// Converts a Rust-owned object to a foreign-owned one.
|
/// Converts a Rust-owned object to a foreign-owned one.
|
||||||
///
|
///
|
||||||
/// The foreign representation is a pointer to void.
|
/// The foreign representation is a pointer to void. There are no guarantees for this pointer.
|
||||||
|
/// For example, it might be invalid, dangling or pointing to uninitialized memory. Using it in
|
||||||
|
/// any way except for [`ForeignOwnable::from_foreign`], [`ForeignOwnable::borrow`],
|
||||||
|
/// [`ForeignOwnable::try_from_foreign`] can result in undefined behavior.
|
||||||
fn into_foreign(self) -> *const core::ffi::c_void;
|
fn into_foreign(self) -> *const core::ffi::c_void;
|
||||||
|
|
||||||
/// Borrows a foreign-owned object.
|
/// Borrows a foreign-owned object.
|
||||||
@ -89,6 +93,32 @@ impl<T: 'static> ForeignOwnable for Box<T> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T: 'static> ForeignOwnable for Pin<Box<T>> {
|
||||||
|
type Borrowed<'a> = Pin<&'a T>;
|
||||||
|
|
||||||
|
fn into_foreign(self) -> *const core::ffi::c_void {
|
||||||
|
// SAFETY: We are still treating the box as pinned.
|
||||||
|
Box::into_raw(unsafe { Pin::into_inner_unchecked(self) }) as _
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe fn borrow<'a>(ptr: *const core::ffi::c_void) -> Pin<&'a T> {
|
||||||
|
// SAFETY: The safety requirements for this function ensure that the object is still alive,
|
||||||
|
// so it is safe to dereference the raw pointer.
|
||||||
|
// The safety requirements of `from_foreign` also ensure that the object remains alive for
|
||||||
|
// the lifetime of the returned value.
|
||||||
|
let r = unsafe { &*ptr.cast() };
|
||||||
|
|
||||||
|
// SAFETY: This pointer originates from a `Pin<Box<T>>`.
|
||||||
|
unsafe { Pin::new_unchecked(r) }
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe fn from_foreign(ptr: *const core::ffi::c_void) -> Self {
|
||||||
|
// SAFETY: The safety requirements of this function ensure that `ptr` comes from a previous
|
||||||
|
// call to `Self::into_foreign`.
|
||||||
|
unsafe { Pin::new_unchecked(Box::from_raw(ptr as _)) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl ForeignOwnable for () {
|
impl ForeignOwnable for () {
|
||||||
type Borrowed<'a> = ();
|
type Borrowed<'a> = ();
|
||||||
|
|
||||||
@ -366,6 +396,35 @@ impl<T: AlwaysRefCounted> ARef<T> {
|
|||||||
_p: PhantomData,
|
_p: PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Consumes the `ARef`, returning a raw pointer.
|
||||||
|
///
|
||||||
|
/// This function does not change the refcount. After calling this function, the caller is
|
||||||
|
/// responsible for the refcount previously managed by the `ARef`.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// use core::ptr::NonNull;
|
||||||
|
/// use kernel::types::{ARef, AlwaysRefCounted};
|
||||||
|
///
|
||||||
|
/// struct Empty {}
|
||||||
|
///
|
||||||
|
/// unsafe impl AlwaysRefCounted for Empty {
|
||||||
|
/// fn inc_ref(&self) {}
|
||||||
|
/// unsafe fn dec_ref(_obj: NonNull<Self>) {}
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// let mut data = Empty {};
|
||||||
|
/// let ptr = NonNull::<Empty>::new(&mut data as *mut _).unwrap();
|
||||||
|
/// let data_ref: ARef<Empty> = unsafe { ARef::from_raw(ptr) };
|
||||||
|
/// let raw_ptr: NonNull<Empty> = ARef::into_raw(data_ref);
|
||||||
|
///
|
||||||
|
/// assert_eq!(ptr, raw_ptr);
|
||||||
|
/// ```
|
||||||
|
pub fn into_raw(me: Self) -> NonNull<T> {
|
||||||
|
ManuallyDrop::new(me).ptr
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: AlwaysRefCounted> Clone for ARef<T> {
|
impl<T: AlwaysRefCounted> Clone for ARef<T> {
|
||||||
|
@ -2,6 +2,10 @@
|
|||||||
|
|
||||||
//! Crate for all kernel procedural macros.
|
//! Crate for all kernel procedural macros.
|
||||||
|
|
||||||
|
// When fixdep scans this, it will find this string `CONFIG_RUSTC_VERSION_TEXT`
|
||||||
|
// and thus add a dependency on `include/config/RUSTC_VERSION_TEXT`, which is
|
||||||
|
// touched by Kconfig when the version string from the compiler changes.
|
||||||
|
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
mod quote;
|
mod quote;
|
||||||
mod concat_idents;
|
mod concat_idents;
|
||||||
|
@ -260,6 +260,12 @@ pub(crate) fn module(ts: TokenStream) -> TokenStream {
|
|||||||
unsafe {{ __init() }}
|
unsafe {{ __init() }}
|
||||||
}}
|
}}
|
||||||
|
|
||||||
|
#[cfg(MODULE)]
|
||||||
|
#[doc(hidden)]
|
||||||
|
#[used]
|
||||||
|
#[link_section = \".init.data\"]
|
||||||
|
static __UNIQUE_ID___addressable_init_module: unsafe extern \"C\" fn() -> i32 = init_module;
|
||||||
|
|
||||||
#[cfg(MODULE)]
|
#[cfg(MODULE)]
|
||||||
#[doc(hidden)]
|
#[doc(hidden)]
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
@ -273,6 +279,12 @@ pub(crate) fn module(ts: TokenStream) -> TokenStream {
|
|||||||
unsafe {{ __exit() }}
|
unsafe {{ __exit() }}
|
||||||
}}
|
}}
|
||||||
|
|
||||||
|
#[cfg(MODULE)]
|
||||||
|
#[doc(hidden)]
|
||||||
|
#[used]
|
||||||
|
#[link_section = \".exit.data\"]
|
||||||
|
static __UNIQUE_ID___addressable_cleanup_module: extern \"C\" fn() = cleanup_module;
|
||||||
|
|
||||||
// Built-in modules are initialized through an initcall pointer
|
// Built-in modules are initialized through an initcall pointer
|
||||||
// and the identifiers need to be unique.
|
// and the identifiers need to be unique.
|
||||||
#[cfg(not(MODULE))]
|
#[cfg(not(MODULE))]
|
||||||
|
@ -64,3 +64,11 @@ ld-version := $(shell,set -- $(ld-info) && echo $2)
|
|||||||
cc-option-bit = $(if-success,$(CC) -Werror $(1) -E -x c /dev/null -o /dev/null,$(1))
|
cc-option-bit = $(if-success,$(CC) -Werror $(1) -E -x c /dev/null -o /dev/null,$(1))
|
||||||
m32-flag := $(cc-option-bit,-m32)
|
m32-flag := $(cc-option-bit,-m32)
|
||||||
m64-flag := $(cc-option-bit,-m64)
|
m64-flag := $(cc-option-bit,-m64)
|
||||||
|
|
||||||
|
# $(rustc-option,<flag>)
|
||||||
|
# Return y if the Rust compiler supports <flag>, n otherwise
|
||||||
|
# Calls to this should be guarded so that they are not evaluated if
|
||||||
|
# CONFIG_RUST_IS_AVAILABLE is not set.
|
||||||
|
# If you are testing for unstable features, consider testing RUSTC_VERSION
|
||||||
|
# instead, as features may have different completeness while available.
|
||||||
|
rustc-option = $(success,trap "rm -rf .tmp_$$" EXIT; mkdir .tmp_$$; $(RUSTC) $(1) --crate-type=rlib /dev/null --out-dir=.tmp_$$ -o .tmp_$$/tmp.rlib)
|
||||||
|
@ -273,10 +273,15 @@ rust_common_cmd = \
|
|||||||
# would not match each other.
|
# would not match each other.
|
||||||
|
|
||||||
quiet_cmd_rustc_o_rs = $(RUSTC_OR_CLIPPY_QUIET) $(quiet_modtag) $@
|
quiet_cmd_rustc_o_rs = $(RUSTC_OR_CLIPPY_QUIET) $(quiet_modtag) $@
|
||||||
cmd_rustc_o_rs = $(rust_common_cmd) --emit=obj=$@ $<
|
cmd_rustc_o_rs = $(rust_common_cmd) --emit=obj=$@ $< $(cmd_objtool)
|
||||||
|
|
||||||
|
define rule_rustc_o_rs
|
||||||
|
$(call cmd_and_fixdep,rustc_o_rs)
|
||||||
|
$(call cmd,gen_objtooldep)
|
||||||
|
endef
|
||||||
|
|
||||||
$(obj)/%.o: $(obj)/%.rs FORCE
|
$(obj)/%.o: $(obj)/%.rs FORCE
|
||||||
+$(call if_changed_dep,rustc_o_rs)
|
+$(call if_changed_rule,rustc_o_rs)
|
||||||
|
|
||||||
quiet_cmd_rustc_rsi_rs = $(RUSTC_OR_CLIPPY_QUIET) $(quiet_modtag) $@
|
quiet_cmd_rustc_rsi_rs = $(RUSTC_OR_CLIPPY_QUIET) $(quiet_modtag) $@
|
||||||
cmd_rustc_rsi_rs = \
|
cmd_rustc_rsi_rs = \
|
||||||
|
@ -72,3 +72,18 @@ clang-min-version = $(call test-ge, $(CONFIG_CLANG_VERSION), $1)
|
|||||||
# ld-option
|
# ld-option
|
||||||
# Usage: KBUILD_LDFLAGS += $(call ld-option, -X, -Y)
|
# Usage: KBUILD_LDFLAGS += $(call ld-option, -X, -Y)
|
||||||
ld-option = $(call try-run, $(LD) $(KBUILD_LDFLAGS) $(1) -v,$(1),$(2),$(3))
|
ld-option = $(call try-run, $(LD) $(KBUILD_LDFLAGS) $(1) -v,$(1),$(2),$(3))
|
||||||
|
|
||||||
|
# __rustc-option
|
||||||
|
# Usage: MY_RUSTFLAGS += $(call __rustc-option,$(RUSTC),$(MY_RUSTFLAGS),-Cinstrument-coverage,-Zinstrument-coverage)
|
||||||
|
__rustc-option = $(call try-run,\
|
||||||
|
$(1) $(2) $(3) --crate-type=rlib /dev/null --out-dir=$$TMPOUT -o "$$TMP",$(3),$(4))
|
||||||
|
|
||||||
|
# rustc-option
|
||||||
|
# Usage: rustflags-y += $(call rustc-option,-Cinstrument-coverage,-Zinstrument-coverage)
|
||||||
|
rustc-option = $(call __rustc-option, $(RUSTC),\
|
||||||
|
$(KBUILD_RUSTFLAGS),$(1),$(2))
|
||||||
|
|
||||||
|
# rustc-option-yn
|
||||||
|
# Usage: flag := $(call rustc-option-yn,-Cinstrument-coverage)
|
||||||
|
rustc-option-yn = $(call try-run,\
|
||||||
|
$(RUSTC) $(KBUILD_RUSTFLAGS) $(1) --crate-type=rlib /dev/null --out-dir=$$TMPOUT -o "$$TMP",y,n)
|
||||||
|
@ -12,6 +12,11 @@ endif
|
|||||||
KASAN_SHADOW_OFFSET ?= $(CONFIG_KASAN_SHADOW_OFFSET)
|
KASAN_SHADOW_OFFSET ?= $(CONFIG_KASAN_SHADOW_OFFSET)
|
||||||
|
|
||||||
cc-param = $(call cc-option, -mllvm -$(1), $(call cc-option, --param $(1)))
|
cc-param = $(call cc-option, -mllvm -$(1), $(call cc-option, --param $(1)))
|
||||||
|
rustc-param = $(call rustc-option, -Cllvm-args=-$(1),)
|
||||||
|
|
||||||
|
check-args = $(foreach arg,$(2),$(call $(1),$(arg)))
|
||||||
|
|
||||||
|
kasan_params :=
|
||||||
|
|
||||||
ifdef CONFIG_KASAN_STACK
|
ifdef CONFIG_KASAN_STACK
|
||||||
stack_enable := 1
|
stack_enable := 1
|
||||||
@ -41,39 +46,59 @@ CFLAGS_KASAN := $(call cc-option, -fsanitize=kernel-address \
|
|||||||
$(call cc-option, -fsanitize=kernel-address \
|
$(call cc-option, -fsanitize=kernel-address \
|
||||||
-mllvm -asan-mapping-offset=$(KASAN_SHADOW_OFFSET)))
|
-mllvm -asan-mapping-offset=$(KASAN_SHADOW_OFFSET)))
|
||||||
|
|
||||||
# Now, add other parameters enabled similarly in both GCC and Clang.
|
# The minimum supported `rustc` version has a minimum supported LLVM
|
||||||
# As some of them are not supported by older compilers, use cc-param.
|
# version late enough that we can assume support for -asan-mapping-offset.
|
||||||
CFLAGS_KASAN += $(call cc-param,asan-instrumentation-with-call-threshold=$(call_threshold)) \
|
RUSTFLAGS_KASAN := -Zsanitizer=kernel-address \
|
||||||
$(call cc-param,asan-stack=$(stack_enable)) \
|
-Zsanitizer-recover=kernel-address \
|
||||||
$(call cc-param,asan-instrument-allocas=1) \
|
-Cllvm-args=-asan-mapping-offset=$(KASAN_SHADOW_OFFSET)
|
||||||
$(call cc-param,asan-globals=1)
|
|
||||||
|
# Now, add other parameters enabled similarly in GCC, Clang, and rustc.
|
||||||
|
# As some of them are not supported by older compilers, these will be filtered
|
||||||
|
# through `cc-param` or `rust-param` as applicable.
|
||||||
|
kasan_params += asan-instrumentation-with-call-threshold=$(call_threshold) \
|
||||||
|
asan-stack=$(stack_enable) \
|
||||||
|
asan-instrument-allocas=1 \
|
||||||
|
asan-globals=1
|
||||||
|
|
||||||
# Instrument memcpy/memset/memmove calls by using instrumented __asan_mem*()
|
# Instrument memcpy/memset/memmove calls by using instrumented __asan_mem*()
|
||||||
# instead. With compilers that don't support this option, compiler-inserted
|
# instead. With compilers that don't support this option, compiler-inserted
|
||||||
# memintrinsics won't be checked by KASAN on GENERIC_ENTRY architectures.
|
# memintrinsics won't be checked by KASAN on GENERIC_ENTRY architectures.
|
||||||
CFLAGS_KASAN += $(call cc-param,asan-kernel-mem-intrinsic-prefix=1)
|
kasan_params += asan-kernel-mem-intrinsic-prefix=1
|
||||||
|
|
||||||
endif # CONFIG_KASAN_GENERIC
|
endif # CONFIG_KASAN_GENERIC
|
||||||
|
|
||||||
ifdef CONFIG_KASAN_SW_TAGS
|
ifdef CONFIG_KASAN_SW_TAGS
|
||||||
|
|
||||||
|
CFLAGS_KASAN := -fsanitize=kernel-hwaddress
|
||||||
|
|
||||||
|
# This sets flags that will enable SW_TAGS KASAN once enabled in Rust. These
|
||||||
|
# will not work today, and is guarded against in dependencies for CONFIG_RUST.
|
||||||
|
RUSTFLAGS_KASAN := -Zsanitizer=kernel-hwaddress \
|
||||||
|
-Zsanitizer-recover=kernel-hwaddress
|
||||||
|
|
||||||
ifdef CONFIG_KASAN_INLINE
|
ifdef CONFIG_KASAN_INLINE
|
||||||
instrumentation_flags := $(call cc-param,hwasan-mapping-offset=$(KASAN_SHADOW_OFFSET))
|
kasan_params += hwasan-mapping-offset=$(KASAN_SHADOW_OFFSET)
|
||||||
else
|
else
|
||||||
instrumentation_flags := $(call cc-param,hwasan-instrument-with-calls=1)
|
kasan_params += hwasan-instrument-with-calls=1
|
||||||
endif
|
endif
|
||||||
|
|
||||||
CFLAGS_KASAN := -fsanitize=kernel-hwaddress \
|
kasan_params += hwasan-instrument-stack=$(stack_enable) \
|
||||||
$(call cc-param,hwasan-instrument-stack=$(stack_enable)) \
|
hwasan-use-short-granules=0 \
|
||||||
$(call cc-param,hwasan-use-short-granules=0) \
|
hwasan-inline-all-checks=0
|
||||||
$(call cc-param,hwasan-inline-all-checks=0) \
|
|
||||||
$(instrumentation_flags)
|
|
||||||
|
|
||||||
# Instrument memcpy/memset/memmove calls by using instrumented __hwasan_mem*().
|
# Instrument memcpy/memset/memmove calls by using instrumented __hwasan_mem*().
|
||||||
ifeq ($(call clang-min-version, 150000)$(call gcc-min-version, 130000),y)
|
ifeq ($(call clang-min-version, 150000)$(call gcc-min-version, 130000),y)
|
||||||
CFLAGS_KASAN += $(call cc-param,hwasan-kernel-mem-intrinsic-prefix=1)
|
kasan_params += hwasan-kernel-mem-intrinsic-prefix=1
|
||||||
endif
|
endif
|
||||||
|
|
||||||
endif # CONFIG_KASAN_SW_TAGS
|
endif # CONFIG_KASAN_SW_TAGS
|
||||||
|
|
||||||
export CFLAGS_KASAN CFLAGS_KASAN_NOSANITIZE
|
# Add all as-supported KASAN LLVM parameters requested by the configuration.
|
||||||
|
CFLAGS_KASAN += $(call check-args, cc-param, $(kasan_params))
|
||||||
|
|
||||||
|
ifdef CONFIG_RUST
|
||||||
|
# Avoid calling `rustc-param` unless Rust is enabled.
|
||||||
|
RUSTFLAGS_KASAN += $(call check-args, rustc-param, $(kasan_params))
|
||||||
|
endif # CONFIG_RUST
|
||||||
|
|
||||||
|
export CFLAGS_KASAN CFLAGS_KASAN_NOSANITIZE RUSTFLAGS_KASAN
|
||||||
|
@ -146,6 +146,9 @@ ifneq ($(CONFIG_KASAN_HW_TAGS),y)
|
|||||||
_c_flags += $(if $(patsubst n%,, \
|
_c_flags += $(if $(patsubst n%,, \
|
||||||
$(KASAN_SANITIZE_$(target-stem).o)$(KASAN_SANITIZE)$(is-kernel-object)), \
|
$(KASAN_SANITIZE_$(target-stem).o)$(KASAN_SANITIZE)$(is-kernel-object)), \
|
||||||
$(CFLAGS_KASAN), $(CFLAGS_KASAN_NOSANITIZE))
|
$(CFLAGS_KASAN), $(CFLAGS_KASAN_NOSANITIZE))
|
||||||
|
_rust_flags += $(if $(patsubst n%,, \
|
||||||
|
$(KASAN_SANITIZE_$(target-stem).o)$(KASAN_SANITIZE)$(is-kernel-object)), \
|
||||||
|
$(RUSTFLAGS_KASAN))
|
||||||
endif
|
endif
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
@ -20,12 +20,28 @@ enum Value {
|
|||||||
Boolean(bool),
|
Boolean(bool),
|
||||||
Number(i32),
|
Number(i32),
|
||||||
String(String),
|
String(String),
|
||||||
|
Array(Vec<Value>),
|
||||||
Object(Object),
|
Object(Object),
|
||||||
}
|
}
|
||||||
|
|
||||||
type Object = Vec<(String, Value)>;
|
type Object = Vec<(String, Value)>;
|
||||||
|
|
||||||
/// Minimal "almost JSON" generator (e.g. no `null`s, no arrays, no escaping),
|
fn comma_sep<T>(
|
||||||
|
seq: &[T],
|
||||||
|
formatter: &mut Formatter<'_>,
|
||||||
|
f: impl Fn(&mut Formatter<'_>, &T) -> Result,
|
||||||
|
) -> Result {
|
||||||
|
if let [ref rest @ .., ref last] = seq[..] {
|
||||||
|
for v in rest {
|
||||||
|
f(formatter, v)?;
|
||||||
|
formatter.write_str(",")?;
|
||||||
|
}
|
||||||
|
f(formatter, last)?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Minimal "almost JSON" generator (e.g. no `null`s, no escaping),
|
||||||
/// enough for this purpose.
|
/// enough for this purpose.
|
||||||
impl Display for Value {
|
impl Display for Value {
|
||||||
fn fmt(&self, formatter: &mut Formatter<'_>) -> Result {
|
fn fmt(&self, formatter: &mut Formatter<'_>) -> Result {
|
||||||
@ -33,59 +49,67 @@ impl Display for Value {
|
|||||||
Value::Boolean(boolean) => write!(formatter, "{}", boolean),
|
Value::Boolean(boolean) => write!(formatter, "{}", boolean),
|
||||||
Value::Number(number) => write!(formatter, "{}", number),
|
Value::Number(number) => write!(formatter, "{}", number),
|
||||||
Value::String(string) => write!(formatter, "\"{}\"", string),
|
Value::String(string) => write!(formatter, "\"{}\"", string),
|
||||||
|
Value::Array(values) => {
|
||||||
|
formatter.write_str("[")?;
|
||||||
|
comma_sep(&values[..], formatter, |formatter, v| v.fmt(formatter))?;
|
||||||
|
formatter.write_str("]")
|
||||||
|
}
|
||||||
Value::Object(object) => {
|
Value::Object(object) => {
|
||||||
formatter.write_str("{")?;
|
formatter.write_str("{")?;
|
||||||
if let [ref rest @ .., ref last] = object[..] {
|
comma_sep(&object[..], formatter, |formatter, v| {
|
||||||
for (key, value) in rest {
|
write!(formatter, "\"{}\": {}", v.0, v.1)
|
||||||
write!(formatter, "\"{}\": {},", key, value)?;
|
})?;
|
||||||
}
|
|
||||||
write!(formatter, "\"{}\": {}", last.0, last.1)?;
|
|
||||||
}
|
|
||||||
formatter.write_str("}")
|
formatter.write_str("}")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl From<bool> for Value {
|
||||||
|
fn from(value: bool) -> Self {
|
||||||
|
Self::Boolean(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<i32> for Value {
|
||||||
|
fn from(value: i32) -> Self {
|
||||||
|
Self::Number(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<String> for Value {
|
||||||
|
fn from(value: String) -> Self {
|
||||||
|
Self::String(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<&str> for Value {
|
||||||
|
fn from(value: &str) -> Self {
|
||||||
|
Self::String(value.to_string())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Object> for Value {
|
||||||
|
fn from(object: Object) -> Self {
|
||||||
|
Self::Object(object)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: Into<Value>, const N: usize> From<[T; N]> for Value {
|
||||||
|
fn from(i: [T; N]) -> Self {
|
||||||
|
Self::Array(i.into_iter().map(|v| v.into()).collect())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
struct TargetSpec(Object);
|
struct TargetSpec(Object);
|
||||||
|
|
||||||
impl TargetSpec {
|
impl TargetSpec {
|
||||||
fn new() -> TargetSpec {
|
fn new() -> TargetSpec {
|
||||||
TargetSpec(Vec::new())
|
TargetSpec(Vec::new())
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
trait Push<T> {
|
fn push(&mut self, key: &str, value: impl Into<Value>) {
|
||||||
fn push(&mut self, key: &str, value: T);
|
self.0.push((key.to_string(), value.into()));
|
||||||
}
|
|
||||||
|
|
||||||
impl Push<bool> for TargetSpec {
|
|
||||||
fn push(&mut self, key: &str, value: bool) {
|
|
||||||
self.0.push((key.to_string(), Value::Boolean(value)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Push<i32> for TargetSpec {
|
|
||||||
fn push(&mut self, key: &str, value: i32) {
|
|
||||||
self.0.push((key.to_string(), Value::Number(value)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Push<String> for TargetSpec {
|
|
||||||
fn push(&mut self, key: &str, value: String) {
|
|
||||||
self.0.push((key.to_string(), Value::String(value)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Push<&str> for TargetSpec {
|
|
||||||
fn push(&mut self, key: &str, value: &str) {
|
|
||||||
self.push(key, value.to_string());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Push<Object> for TargetSpec {
|
|
||||||
fn push(&mut self, key: &str, value: Object) {
|
|
||||||
self.0.push((key.to_string(), Value::Object(value)));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -164,10 +188,26 @@ fn main() {
|
|||||||
);
|
);
|
||||||
let mut features = "-mmx,+soft-float".to_string();
|
let mut features = "-mmx,+soft-float".to_string();
|
||||||
if cfg.has("MITIGATION_RETPOLINE") {
|
if cfg.has("MITIGATION_RETPOLINE") {
|
||||||
|
// The kernel uses `-mretpoline-external-thunk` (for Clang), which Clang maps to the
|
||||||
|
// target feature of the same name plus the other two target features in
|
||||||
|
// `clang/lib/Driver/ToolChains/Arch/X86.cpp`. These should be eventually enabled via
|
||||||
|
// `-Ctarget-feature` when `rustc` starts recognizing them (or via a new dedicated
|
||||||
|
// flag); see https://github.com/rust-lang/rust/issues/116852.
|
||||||
features += ",+retpoline-external-thunk";
|
features += ",+retpoline-external-thunk";
|
||||||
|
features += ",+retpoline-indirect-branches";
|
||||||
|
features += ",+retpoline-indirect-calls";
|
||||||
|
}
|
||||||
|
if cfg.has("MITIGATION_SLS") {
|
||||||
|
// The kernel uses `-mharden-sls=all`, which Clang maps to both these target features in
|
||||||
|
// `clang/lib/Driver/ToolChains/Arch/X86.cpp`. These should be eventually enabled via
|
||||||
|
// `-Ctarget-feature` when `rustc` starts recognizing them (or via a new dedicated
|
||||||
|
// flag); see https://github.com/rust-lang/rust/issues/116851.
|
||||||
|
features += ",+harden-sls-ijmp";
|
||||||
|
features += ",+harden-sls-ret";
|
||||||
}
|
}
|
||||||
ts.push("features", features);
|
ts.push("features", features);
|
||||||
ts.push("llvm-target", "x86_64-linux-gnu");
|
ts.push("llvm-target", "x86_64-linux-gnu");
|
||||||
|
ts.push("supported-sanitizers", ["kcfi", "kernel-address"]);
|
||||||
ts.push("target-pointer-width", "64");
|
ts.push("target-pointer-width", "64");
|
||||||
} else if cfg.has("X86_32") {
|
} else if cfg.has("X86_32") {
|
||||||
// This only works on UML, as i386 otherwise needs regparm support in rustc
|
// This only works on UML, as i386 otherwise needs regparm support in rustc
|
||||||
|
26
scripts/rustc-version.sh
Executable file
26
scripts/rustc-version.sh
Executable file
@ -0,0 +1,26 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# SPDX-License-Identifier: GPL-2.0
|
||||||
|
#
|
||||||
|
# Usage: $ ./rustc-version.sh rustc
|
||||||
|
#
|
||||||
|
# Print the Rust compiler version in a 6 or 7-digit form.
|
||||||
|
|
||||||
|
# Convert the version string x.y.z to a canonical up-to-7-digits form.
|
||||||
|
#
|
||||||
|
# Note that this function uses one more digit (compared to other
|
||||||
|
# instances in other version scripts) to give a bit more space to
|
||||||
|
# `rustc` since it will reach 1.100.0 in late 2026.
|
||||||
|
get_canonical_version()
|
||||||
|
{
|
||||||
|
IFS=.
|
||||||
|
set -- $1
|
||||||
|
echo $((100000 * $1 + 100 * $2 + $3))
|
||||||
|
}
|
||||||
|
|
||||||
|
if output=$("$@" --version 2>/dev/null); then
|
||||||
|
set -- $output
|
||||||
|
get_canonical_version $2
|
||||||
|
else
|
||||||
|
echo 0
|
||||||
|
exit 1
|
||||||
|
fi
|
@ -177,6 +177,52 @@ static bool is_sibling_call(struct instruction *insn)
|
|||||||
return (is_static_jump(insn) && insn_call_dest(insn));
|
return (is_static_jump(insn) && insn_call_dest(insn));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Checks if a string ends with another.
|
||||||
|
*/
|
||||||
|
static bool str_ends_with(const char *s, const char *sub)
|
||||||
|
{
|
||||||
|
const int slen = strlen(s);
|
||||||
|
const int sublen = strlen(sub);
|
||||||
|
|
||||||
|
if (sublen > slen)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return !memcmp(s + slen - sublen, sub, sublen);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Checks if a function is a Rust "noreturn" one.
|
||||||
|
*/
|
||||||
|
static bool is_rust_noreturn(const struct symbol *func)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* If it does not start with "_R", then it is not a Rust symbol.
|
||||||
|
*/
|
||||||
|
if (strncmp(func->name, "_R", 2))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* These are just heuristics -- we do not control the precise symbol
|
||||||
|
* name, due to the crate disambiguators (which depend on the compiler)
|
||||||
|
* as well as changes to the source code itself between versions (since
|
||||||
|
* these come from the Rust standard library).
|
||||||
|
*/
|
||||||
|
return str_ends_with(func->name, "_4core5sliceSp15copy_from_slice17len_mismatch_fail") ||
|
||||||
|
str_ends_with(func->name, "_4core6option13unwrap_failed") ||
|
||||||
|
str_ends_with(func->name, "_4core6result13unwrap_failed") ||
|
||||||
|
str_ends_with(func->name, "_4core9panicking5panic") ||
|
||||||
|
str_ends_with(func->name, "_4core9panicking9panic_fmt") ||
|
||||||
|
str_ends_with(func->name, "_4core9panicking14panic_explicit") ||
|
||||||
|
str_ends_with(func->name, "_4core9panicking14panic_nounwind") ||
|
||||||
|
str_ends_with(func->name, "_4core9panicking18panic_bounds_check") ||
|
||||||
|
str_ends_with(func->name, "_4core9panicking19assert_failed_inner") ||
|
||||||
|
str_ends_with(func->name, "_4core9panicking36panic_misaligned_pointer_dereference") ||
|
||||||
|
strstr(func->name, "_4core9panicking11panic_const24panic_const_") ||
|
||||||
|
(strstr(func->name, "_4core5slice5index24slice_") &&
|
||||||
|
str_ends_with(func->name, "_fail"));
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This checks to see if the given function is a "noreturn" function.
|
* This checks to see if the given function is a "noreturn" function.
|
||||||
*
|
*
|
||||||
@ -202,10 +248,14 @@ static bool __dead_end_function(struct objtool_file *file, struct symbol *func,
|
|||||||
if (!func)
|
if (!func)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (func->bind == STB_GLOBAL || func->bind == STB_WEAK)
|
if (func->bind == STB_GLOBAL || func->bind == STB_WEAK) {
|
||||||
|
if (is_rust_noreturn(func))
|
||||||
|
return true;
|
||||||
|
|
||||||
for (i = 0; i < ARRAY_SIZE(global_noreturns); i++)
|
for (i = 0; i < ARRAY_SIZE(global_noreturns); i++)
|
||||||
if (!strcmp(func->name, global_noreturns[i]))
|
if (!strcmp(func->name, global_noreturns[i]))
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (func->bind == STB_WEAK)
|
if (func->bind == STB_WEAK)
|
||||||
return false;
|
return false;
|
||||||
|
@ -39,6 +39,8 @@ NORETURN(panic)
|
|||||||
NORETURN(panic_smp_self_stop)
|
NORETURN(panic_smp_self_stop)
|
||||||
NORETURN(rest_init)
|
NORETURN(rest_init)
|
||||||
NORETURN(rewind_stack_and_make_dead)
|
NORETURN(rewind_stack_and_make_dead)
|
||||||
|
NORETURN(rust_begin_unwind)
|
||||||
|
NORETURN(rust_helper_BUG)
|
||||||
NORETURN(sev_es_terminate)
|
NORETURN(sev_es_terminate)
|
||||||
NORETURN(snp_abort)
|
NORETURN(snp_abort)
|
||||||
NORETURN(start_kernel)
|
NORETURN(start_kernel)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user