-
Notifications
You must be signed in to change notification settings - Fork 13.9k
compiler: Allow extern "interrupt" fn() -> !
#143075
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
compiler: Allow extern "interrupt" fn() -> !
#143075
Conversation
This PR modifies cc @jieyouxu |
Now that the rules get more complex, I think the error on the return type should probably move to type Unit = ();
#[cfg(any(x64, i686))]
extern "x86-interrupt" fn x86_ret_unit() -> Unit {
unsafe { hint::unreachable_unchecked() }
} gives
The error message is also incorrect now: the function can have a return type (so long as it's |
765309e
to
b5ab966
Compare
I mean if I really want to get it right it should probably be even deeper, so we can check the actual layout. |
I don't think I want to do that level of correctness before we decide we actually want these interrupt ABIs, though, and I, at least, am not sold on them. |
This comment was marked as off-topic.
This comment was marked as off-topic.
Fair enough. Then the error message should at least be correct though right? (so, there can be a return type, just literally
The x86 one gets a surprising (to me) amount of use https://github.com/search?q=extern+%22x86-interrupt%22&type=code. But in theory those could now use naked functions and compile on stable. |
@andrewoffenden what. |
Yes, Phil Opp espoused using them, so they gained some popularity. But they should not be used by a real OS, because an OS cannot support any debugger APIs if it uses them. |
I think there are valid use-cases for the "x86-interrupt" ABI even in a "real OS". Sometimes you don't need to look at the registers of the interrupted thread, and in that case, using "x86-interrupt" is completely fine and can even result in better code than a hand-written interrupt handler. For example, I use it for handling timer interrupts that happen while the kernel is executing in ring 0. That said, there are, of course, also use cases where this absolutely will not work: If you're going to write an |
Apart from convenience, my main motivation for using the That being said, I also agree that the Thanks a lot for this PR by the way! I appreciate it that you try to avoid breakage, even though it's a nightly feature! |
(Not directly related to this PR, but to discussion about whether we want to keep the I think if we had support for LLVM's |
We do support PreserveMost already as I agree that we really do need that as a prerequisite to any alternatives. That way any assembly can be a minimal shim sufficient to do whatever operations are needed between the machine and the OS and then jump into the main handler, instead of saving everything in the assembly. |
I was wondering why my kernel couldn't compile yet x86_64 expected a return type of |
fn reject_params_or_return(&self, abi: ExternAbi, ident: &Ident, sig: &FnSig) { | ||
let mut spans: Vec<_> = sig.decl.inputs.iter().map(|p| p.span).collect(); | ||
if let FnRetTy::Ty(ref ret_ty) = sig.decl.output { | ||
if let FnRetTy::Ty(ref ret_ty) = sig.decl.output | ||
&& match &ret_ty.kind { | ||
TyKind::Never => false, | ||
TyKind::Tup(tup) if tup.is_empty() => false, | ||
_ => true, | ||
} | ||
{ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actually, @folkertdev, I think the biggest logic with moving this to hir_typeck
is I'm not actually sure if that's the right place for an ad-hoc validation check on the HIR. And I don't know if I should trust the previous places we put ABI checks. 😵💫
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I just know that it's where we now perform the checks for extern "custom"
and for naked functions there. They were previously in rustc_passes
but moving them into hir_typeck
was both shorter and faster.
r? compiler |
Commenting to follow this PR. This should fix a CI failure I started seeing a few days ago on my reference MSP430 repo. (I've been following #132841- the issue which was closed by #142633- since it has the Re: the interrupt CC proliferation, in terms of msp430-land, TI's ABI doc states (emphasis mine):
In other words, it's a generic "save all the registers that are used except those that are saved by the hardware (PC, SR) and/or have known use cases to modify (PC, SR's power-mode bits, SP). I think |
@folkertdev Ah yeah you're right, I should have known better, sadly this is allowed due to doing AST-level validation: #![feature(abi_x86_interrupt)]
use core::{mem, ptr};
type Alias<A> = unsafe extern "x86-interrupt" fn(A);
fn main() {
let ptr = unsafe { mem::transmute::<_, Alias<i32>>(ptr::dangling::<u8>()) };
} |
Yeah, sorry, I thought it was going to be just a couple days of work and then my computer broke which... lol? I'm fine with landing this as-is if others are happy. |
|
If I use this: extern "x86-interrupt" fn double_fault_handler(
stack_frame: InterruptStackFrame,
_error_code: u64,
) the compiler gives:
But if I use this: extern "x86-interrupt" fn double_fault_handler(
stack_frame: InterruptStackFrame,
_error_code: u64,
) -> ! the compiler gives:
It is contradictory. |
@Code-Coder-coder the first error arises because of the library you are using that provides the So, yes it is contradictory, but it is not related to the issue/PR. |
merging this PR will fix the regression, allowing the use of |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM, seems reasonable. hir_typeck
is probably the better place for these checks but if you want more time to think about the interrupt ABIs first then we can always move it later. It's a small addition anyway.
@bors r+ |
…rn-nevermore, r=davidtwco compiler: Allow `extern "interrupt" fn() -> !` While reviewing rust-lang#142633 I overlooked a few details because I was kind of excited. - Fixes rust-lang#143072
Rollup of 13 pull requests Successful merges: - #122661 (Change the desugaring of `assert!` for better error output) - #142372 (Improve `--remap-path-prefix` documentation) - #143075 (compiler: Allow `extern "interrupt" fn() -> !`) - #145005 (strip prefix of temporary file names when it exceeds filesystem name length limit) - #145065 (resolve: Introduce `RibKind::Block`) - #145120 (llvm: Accept new LLVM lifetime format) - #145189 (Weekly `cargo update`) - #145261 (Improve tracing in bootstrap) - #145266 (Reduce some queries around associated items) - #145299 (doc test: fix mpsc.rs try_send doc test) - #145331 (Make std use the edition 2024 prelude) - #145353 (bootstrap: Fix jemalloc 64K page support for aarch64 tools) - #145361 (Suppress wrapper suggestion when expected and actual ty are the same adt and the variant is unresolved) Failed merges: - #145324 (Rename and document `ONLY_HOSTS` in bootstrap) - #145372 (resolve: Miscellaneous cleanups) r? `@ghost` `@rustbot` modify labels: rollup
Rollup of 13 pull requests Successful merges: - #122661 (Change the desugaring of `assert!` for better error output) - #142372 (Improve `--remap-path-prefix` documentation) - #143075 (compiler: Allow `extern "interrupt" fn() -> !`) - #145005 (strip prefix of temporary file names when it exceeds filesystem name length limit) - #145065 (resolve: Introduce `RibKind::Block`) - #145120 (llvm: Accept new LLVM lifetime format) - #145189 (Weekly `cargo update`) - #145261 (Improve tracing in bootstrap) - #145266 (Reduce some queries around associated items) - #145299 (doc test: fix mpsc.rs try_send doc test) - #145331 (Make std use the edition 2024 prelude) - #145353 (bootstrap: Fix jemalloc 64K page support for aarch64 tools) - #145361 (Suppress wrapper suggestion when expected and actual ty are the same adt and the variant is unresolved) Failed merges: - #145324 (Rename and document `ONLY_HOSTS` in bootstrap) - #145372 (resolve: Miscellaneous cleanups) r? `@ghost` `@rustbot` modify labels: rollup
…rn-nevermore, r=davidtwco compiler: Allow `extern "interrupt" fn() -> !` While reviewing rust-lang#142633 I overlooked a few details because I was kind of excited. - Fixes rust-lang#143072
Rollup of 22 pull requests Successful merges: - #118087 (Add Ref/RefMut try_map method) - #122661 (Change the desugaring of `assert!` for better error output) - #140740 (Add `-Zindirect-branch-cs-prefix`) - #142640 (Implement autodiff using intrinsics) - #143075 (compiler: Allow `extern "interrupt" fn() -> !`) - #144865 (Fix tail calls to `#[track_caller]` functions) - #144944 (E0793: Clarify that it applies to unions as well) - #144947 (Fix description of unsigned `checked_exact_div`) - #145004 (Couple of minor cleanups) - #145005 (strip prefix of temporary file names when it exceeds filesystem name length limit) - #145012 (Tail call diagnostics to include lifetime info) - #145065 (resolve: Introduce `RibKind::Block`) - #145120 (llvm: Accept new LLVM lifetime format) - #145189 (Weekly `cargo update`) - #145235 (Minor `[const]` tweaks) - #145275 (fix(compiler/rustc_codegen_llvm): apply `target-cpu` attribute) - #145322 (Resolve the prelude import in `build_reduced_graph`) - #145331 (Make std use the edition 2024 prelude) - #145369 (Do not ICE on private type in field of unresolved struct) - #145378 (Add `FnContext` in parser for diagnostic) - #145389 ([rustdoc] Revert "rustdoc search: prefer stable items in search results") - #145392 (coverage: Remove intermediate data structures from mapping creation) r? `@ghost` `@rustbot` modify labels: rollup
…rn-nevermore, r=davidtwco compiler: Allow `extern "interrupt" fn() -> !` While reviewing rust-lang#142633 I overlooked a few details because I was kind of excited. - Fixes rust-lang#143072
Rollup of 21 pull requests Successful merges: - #118087 (Add Ref/RefMut try_map method) - #122661 (Change the desugaring of `assert!` for better error output) - #142640 (Implement autodiff using intrinsics) - #143075 (compiler: Allow `extern "interrupt" fn() -> !`) - #144865 (Fix tail calls to `#[track_caller]` functions) - #144944 (E0793: Clarify that it applies to unions as well) - #144947 (Fix description of unsigned `checked_exact_div`) - #145004 (Couple of minor cleanups) - #145005 (strip prefix of temporary file names when it exceeds filesystem name length limit) - #145012 (Tail call diagnostics to include lifetime info) - #145065 (resolve: Introduce `RibKind::Block`) - #145120 (llvm: Accept new LLVM lifetime format) - #145189 (Weekly `cargo update`) - #145235 (Minor `[const]` tweaks) - #145275 (fix(compiler/rustc_codegen_llvm): apply `target-cpu` attribute) - #145322 (Resolve the prelude import in `build_reduced_graph`) - #145331 (Make std use the edition 2024 prelude) - #145369 (Do not ICE on private type in field of unresolved struct) - #145378 (Add `FnContext` in parser for diagnostic) - #145389 ([rustdoc] Revert "rustdoc search: prefer stable items in search results") - #145392 (coverage: Remove intermediate data structures from mapping creation) r? `@ghost` `@rustbot` modify labels: rollup
Rollup of 21 pull requests Successful merges: - rust-lang/rust#118087 (Add Ref/RefMut try_map method) - rust-lang/rust#122661 (Change the desugaring of `assert!` for better error output) - rust-lang/rust#142640 (Implement autodiff using intrinsics) - rust-lang/rust#143075 (compiler: Allow `extern "interrupt" fn() -> !`) - rust-lang/rust#144865 (Fix tail calls to `#[track_caller]` functions) - rust-lang/rust#144944 (E0793: Clarify that it applies to unions as well) - rust-lang/rust#144947 (Fix description of unsigned `checked_exact_div`) - rust-lang/rust#145004 (Couple of minor cleanups) - rust-lang/rust#145005 (strip prefix of temporary file names when it exceeds filesystem name length limit) - rust-lang/rust#145012 (Tail call diagnostics to include lifetime info) - rust-lang/rust#145065 (resolve: Introduce `RibKind::Block`) - rust-lang/rust#145120 (llvm: Accept new LLVM lifetime format) - rust-lang/rust#145189 (Weekly `cargo update`) - rust-lang/rust#145235 (Minor `[const]` tweaks) - rust-lang/rust#145275 (fix(compiler/rustc_codegen_llvm): apply `target-cpu` attribute) - rust-lang/rust#145322 (Resolve the prelude import in `build_reduced_graph`) - rust-lang/rust#145331 (Make std use the edition 2024 prelude) - rust-lang/rust#145369 (Do not ICE on private type in field of unresolved struct) - rust-lang/rust#145378 (Add `FnContext` in parser for diagnostic) - rust-lang/rust#145389 ([rustdoc] Revert "rustdoc search: prefer stable items in search results") - rust-lang/rust#145392 (coverage: Remove intermediate data structures from mapping creation) r? `@ghost` `@rustbot` modify labels: rollup
While reviewing #142633 I overlooked a few details because I was kind of excited.