Skip to content

Commit

Permalink
Rollup merge of rust-lang#110141 - petrochenkov:cratecfg2, r=WaffleLa…
Browse files Browse the repository at this point in the history
…pkin

expand: Change how `#![cfg(FALSE)]` behaves on crate root

Previously it removed all other attributes from the crate root.
Now it removes only attributes below itself (during both regular expansion and pre-configuration).

So it becomes possible to configure some global crate properties even for fully unconfigured crates.

Fixes rust-lang#104633
Part of rust-lang#110082
  • Loading branch information
GuillaumeGomez authored Jun 9, 2023
2 parents 343ad6f + bfc21b1 commit 69cf0f9
Show file tree
Hide file tree
Showing 10 changed files with 63 additions and 31 deletions.
8 changes: 5 additions & 3 deletions compiler/rustc_expand/src/config.rs
Original file line number Diff line number Diff line change
Expand Up @@ -197,9 +197,11 @@ pub fn pre_configure_attrs(sess: &Session, attrs: &[Attribute]) -> ast::AttrVec
config_tokens: false,
lint_node_id: ast::CRATE_NODE_ID,
};
let attrs: ast::AttrVec =
attrs.iter().flat_map(|attr| strip_unconfigured.process_cfg_attr(attr)).collect();
if strip_unconfigured.in_cfg(&attrs) { attrs } else { ast::AttrVec::new() }
attrs
.iter()
.flat_map(|attr| strip_unconfigured.process_cfg_attr(attr))
.take_while(|attr| !is_cfg(attr) || strip_unconfigured.cfg_true(attr))
.collect()
}

#[macro_export]
Expand Down
20 changes: 16 additions & 4 deletions compiler/rustc_expand/src/expand.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1039,7 +1039,12 @@ trait InvocationCollectorNode: HasAttrs + HasNodeId + Sized {
) -> Result<Self::OutputTy, Self> {
Ok(noop_flat_map(node, collector))
}
fn expand_cfg_false(&mut self, collector: &mut InvocationCollector<'_, '_>, span: Span) {
fn expand_cfg_false(
&mut self,
collector: &mut InvocationCollector<'_, '_>,
_pos: usize,
span: Span,
) {
collector.cx.emit_err(RemoveNodeNotSupported { span, descr: Self::descr() });
}

Expand Down Expand Up @@ -1409,8 +1414,15 @@ impl InvocationCollectorNode for ast::Crate {
fn noop_visit<V: MutVisitor>(&mut self, visitor: &mut V) {
noop_visit_crate(self, visitor)
}
fn expand_cfg_false(&mut self, collector: &mut InvocationCollector<'_, '_>, _span: Span) {
self.attrs.clear();
fn expand_cfg_false(
&mut self,
collector: &mut InvocationCollector<'_, '_>,
pos: usize,
_span: Span,
) {
// Attributes above `cfg(FALSE)` are left in place, because we may want to configure
// some global crate properties even on fully unconfigured crates.
self.attrs.truncate(pos);
// Standard prelude imports are left in the crate for backward compatibility.
self.items.truncate(collector.cx.num_standard_library_imports);
}
Expand Down Expand Up @@ -1804,7 +1816,7 @@ impl<'a, 'b> InvocationCollector<'a, 'b> {
continue;
}

node.expand_cfg_false(self, span);
node.expand_cfg_false(self, pos, span);
continue;
}
sym::cfg_attr => {
Expand Down
6 changes: 2 additions & 4 deletions tests/ui/cfg/auxiliary/cfg_false_lib.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,4 @@
// It is unclear whether a fully unconfigured crate should link to standard library,
// or what its `no_std`/`no_core`/`compiler_builtins` status, more precisely.
// Currently the usual standard library prelude is added to such crates,
// and therefore they link to libstd.
// `#![no_std]` on a fully unconfigured crate is respected if it's placed before `cfg(FALSE)`.
// This crate has no such attribute, therefore this crate does link to libstd.

#![cfg(FALSE)]
5 changes: 5 additions & 0 deletions tests/ui/cfg/auxiliary/cfg_false_lib_no_std_after.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
// `#![no_std]` on a fully unconfigured crate is respected if it's placed before `cfg(FALSE)`.
// Therefore this crate does link to libstd.

#![cfg(FALSE)]
#![no_std]
8 changes: 8 additions & 0 deletions tests/ui/cfg/auxiliary/cfg_false_lib_no_std_before.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
// `#![no_std]` on a fully unconfigured crate is respected if it's placed before `cfg(FALSE)`.
// Therefore this crate doesn't link to libstd.

// no-prefer-dynamic

#![no_std]
#![crate_type = "lib"]
#![cfg(FALSE)]
6 changes: 2 additions & 4 deletions tests/ui/cfg/cfg-false-feature.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
// It is unclear which features should be in effect in a fully unconfigured crate (issue #104633).
// Currently none on the features are in effect, so we get the feature gates reported.
// Features above `cfg(FALSE)` are in effect in a fully unconfigured crate (issue #104633).

// check-pass
// compile-flags: --crate-type lib
Expand All @@ -8,8 +7,7 @@
#![cfg(FALSE)]
#![feature(box_syntax)]

macro mac() {} //~ WARN `macro` is experimental
//~| WARN unstable syntax can change at any point in the future
macro mac() {} // OK

trait A = Clone; //~ WARN trait aliases are experimental
//~| WARN unstable syntax can change at any point in the future
Expand Down
17 changes: 3 additions & 14 deletions tests/ui/cfg/cfg-false-feature.stderr
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
warning: trait aliases are experimental
--> $DIR/cfg-false-feature.rs:14:1
--> $DIR/cfg-false-feature.rs:12:1
|
LL | trait A = Clone;
| ^^^^^^^^^^^^^^^^
Expand All @@ -9,19 +9,8 @@ LL | trait A = Clone;
= warning: unstable syntax can change at any point in the future, causing a hard error!
= note: for more information, see issue #65860 <https://github.com/rust-lang/rust/issues/65860>

warning: `macro` is experimental
--> $DIR/cfg-false-feature.rs:11:1
|
LL | macro mac() {}
| ^^^^^^^^^^^^^^
|
= note: see issue #39412 <https://github.com/rust-lang/rust/issues/39412> for more information
= help: add `#![feature(decl_macro)]` to the crate attributes to enable
= warning: unstable syntax can change at any point in the future, causing a hard error!
= note: for more information, see issue #65860 <https://github.com/rust-lang/rust/issues/65860>

warning: box pattern syntax is experimental
--> $DIR/cfg-false-feature.rs:18:9
--> $DIR/cfg-false-feature.rs:16:9
|
LL | let box _ = Box::new(0);
| ^^^^^
Expand All @@ -31,5 +20,5 @@ LL | let box _ = Box::new(0);
= warning: unstable syntax can change at any point in the future, causing a hard error!
= note: for more information, see issue #65860 <https://github.com/rust-lang/rust/issues/65860>

warning: 3 warnings emitted
warning: 2 warnings emitted

10 changes: 10 additions & 0 deletions tests/ui/cfg/cfg_false_no_std-1.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
// No error, panic handler is supplied by libstd linked though the empty library.

// check-pass
// aux-build: cfg_false_lib_no_std_after.rs

#![no_std]

extern crate cfg_false_lib_no_std_after as _;

fn main() {}
11 changes: 11 additions & 0 deletions tests/ui/cfg/cfg_false_no_std-2.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
// Error, the linked empty library is `no_std` and doesn't provide a panic handler.

// dont-check-compiler-stderr
// error-pattern: `#[panic_handler]` function required, but not found
// aux-build: cfg_false_lib_no_std_before.rs

#![no_std]

extern crate cfg_false_lib_no_std_before as _;

fn main() {}
3 changes: 1 addition & 2 deletions tests/ui/cfg/cfg_false_no_std.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
// Currently no error because the panic handler is supplied by libstd linked though the empty
// library, but the desirable behavior is unclear (see comments in cfg_false_lib.rs).
// No error, panic handler is supplied by libstd linked though the empty library.

// check-pass
// aux-build: cfg_false_lib.rs
Expand Down

0 comments on commit 69cf0f9

Please sign in to comment.