Skip to content

Commit d257440

Browse files
committed
Auto merge of #64882 - ehuss:stabilize-bare-extern, r=eddyb
Stabilize --extern flag without a path. This stabilizes the `--extern` flag without a path, implemented in #54116. This flag is used to add a crate that may be found in the search path to the extern prelude. The intent of stabilizing this now is to change Cargo to emit this flag for `proc_macro` when building a proc-macro crate. This will allow the ability to elide `extern crate proc_macro;` for proc-macros, one of the few places where it is still necessary. It is intended that Cargo may also use this flag for other cases in the future as part of the [std-aware work](https://github.com/rust-lang/wg-cargo-std-aware/). There will likely be some kind of syntax where users may declare dependencies on other crates (such as `alloc`), and Cargo will use this flag so that they may be used like any other crate. At this time there are no short-term plans to use it for anything other than proc-macro. This will not help for non-proc-macro crates that use `proc_macro`, which I believe is not too common? An alternate approach for proc-macro is to use the `meta` crate, but from my inquiries there doesn't appear to be anyone interested in pushing that forward. The `meta` crate also doesn't help with things like `alloc` or `test`. cc #57288
2 parents e8f43b7 + ee459c6 commit d257440

File tree

17 files changed

+101
-29
lines changed

17 files changed

+101
-29
lines changed

src/doc/rustc/src/codegen-options/index.md

+5-1
Original file line numberDiff line numberDiff line change
@@ -197,7 +197,11 @@ in software.
197197
## prefer-dynamic
198198

199199
By default, `rustc` prefers to statically link dependencies. This option will
200-
make it use dynamic linking instead.
200+
indicate that dynamic linking should be used if possible if both a static and
201+
dynamic versions of a library are available. There is an internal algorithm
202+
for determining whether or not it is possible to statically or dynamically
203+
link with a dependency. For example, `cdylib` crate types may only use static
204+
linkage.
201205

202206
## no-integrated-as
203207

src/doc/rustc/src/command-line-arguments.md

+25-5
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,7 @@ to `#[cfg(verbose)]` and `#[cfg(feature = "serde")]` respectively.
2121
<a id="option-l-search-path"></a>
2222
## `-L`: add a directory to the library search path
2323

24-
When looking for external crates or libraries, a directory passed to this flag
25-
will be searched.
24+
The `-L` flag adds a path to search for external crates and libraries.
2625

2726
The kind of search path can optionally be specified with the form `-L
2827
KIND=PATH` where `KIND` may be one of:
@@ -262,9 +261,30 @@ This flag, when combined with other flags, makes them produce extra output.
262261
<a id="option-extern"></a>
263262
## `--extern`: specify where an external library is located
264263

265-
This flag allows you to pass the name and location of an external crate that
266-
will be linked into the crate you are building. This flag may be specified
267-
multiple times. The format of the value should be `CRATENAME=PATH`.
264+
This flag allows you to pass the name and location for an external crate of a
265+
direct dependency. Indirect dependencies (dependencies of dependencies) are
266+
located using the [`-L` flag](#option-l-search-path). The given crate name is
267+
added to the [extern prelude], which is the same as specifying `extern crate`
268+
within the root module. The given crate name does not need to match the name
269+
the library was built with.
270+
271+
This flag may be specified multiple times. This flag takes an argument with
272+
either of the following formats:
273+
274+
* `CRATENAME=PATH` — Indicates the given crate is found at the given path.
275+
* `CRATENAME` — Indicates the given crate may be found in the search path,
276+
such as within the sysroot or via the `-L` flag.
277+
278+
The same crate name may be specified multiple times for different crate types.
279+
If both an `rlib` and `dylib` are found, an internal algorithm is used to
280+
decide which to use for linking. The [`-C prefer-dynamic`
281+
flag][prefer-dynamic] may be used to influence which is used.
282+
283+
If the same crate name is specified with and without a path, the one with the
284+
path is used and the pathless flag has no effect.
285+
286+
[extern prelude]: ../reference/items/extern-crates.html#extern-prelude
287+
[prefer-dynamic]: codegen-options/index.md#prefer-dynamic
268288

269289
<a id="option-sysroot"></a>
270290
## `--sysroot`: Override the system root

src/librustc/session/config.rs

+3-14
Original file line numberDiff line numberDiff line change
@@ -1800,7 +1800,7 @@ pub fn rustc_optgroups() -> Vec<RustcOptGroup> {
18001800
"",
18011801
"extern",
18021802
"Specify where an external rust library is located",
1803-
"NAME=PATH",
1803+
"NAME[=PATH]",
18041804
),
18051805
opt::multi_s(
18061806
"",
@@ -2164,7 +2164,6 @@ fn collect_print_requests(
21642164
cg: &mut CodegenOptions,
21652165
dopts: &mut DebuggingOptions,
21662166
matches: &getopts::Matches,
2167-
is_unstable_enabled: bool,
21682167
error_format: ErrorOutputType,
21692168
) -> Vec<PrintRequest> {
21702169
let mut prints = Vec::<PrintRequest>::new();
@@ -2206,7 +2205,7 @@ fn collect_print_requests(
22062205
"tls-models" => PrintRequest::TlsModels,
22072206
"native-static-libs" => PrintRequest::NativeStaticLibs,
22082207
"target-spec-json" => {
2209-
if is_unstable_enabled {
2208+
if dopts.unstable_options {
22102209
PrintRequest::TargetSpec
22112210
} else {
22122211
early_error(
@@ -2370,7 +2369,6 @@ fn parse_externs(
23702369
matches: &getopts::Matches,
23712370
debugging_opts: &DebuggingOptions,
23722371
error_format: ErrorOutputType,
2373-
is_unstable_enabled: bool,
23742372
) -> Externs {
23752373
if matches.opt_present("extern-private") && !debugging_opts.unstable_options {
23762374
early_error(
@@ -2392,13 +2390,6 @@ fn parse_externs(
23922390
let name = parts.next().unwrap_or_else(||
23932391
early_error(error_format, "--extern value must not be empty"));
23942392
let location = parts.next().map(|s| s.to_string());
2395-
if location.is_none() && !is_unstable_enabled {
2396-
early_error(
2397-
error_format,
2398-
"the `-Z unstable-options` flag must also be passed to \
2399-
enable `--extern crate_name` without `=path`",
2400-
);
2401-
};
24022393

24032394
let entry = externs
24042395
.entry(name.to_owned())
@@ -2483,12 +2474,10 @@ pub fn build_session_options(matches: &getopts::Matches) -> Options {
24832474
);
24842475
}
24852476

2486-
let is_unstable_enabled = nightly_options::is_unstable_enabled(matches);
24872477
let prints = collect_print_requests(
24882478
&mut cg,
24892479
&mut debugging_opts,
24902480
matches,
2491-
is_unstable_enabled,
24922481
error_format,
24932482
);
24942483

@@ -2521,7 +2510,7 @@ pub fn build_session_options(matches: &getopts::Matches) -> Options {
25212510
);
25222511
}
25232512

2524-
let externs = parse_externs(matches, &debugging_opts, error_format, is_unstable_enabled);
2513+
let externs = parse_externs(matches, &debugging_opts, error_format);
25252514

25262515
let crate_name = matches.opt_str("crate-name");
25272516

src/librustdoc/config.rs

-4
Original file line numberDiff line numberDiff line change
@@ -615,10 +615,6 @@ fn parse_externs(matches: &getopts::Matches) -> Result<Externs, String> {
615615
let mut parts = arg.splitn(2, '=');
616616
let name = parts.next().ok_or("--extern value must not be empty".to_string())?;
617617
let location = parts.next().map(|s| s.to_string());
618-
if location.is_none() && !nightly_options::is_unstable_enabled(matches) {
619-
return Err("the `-Z unstable-options` flag must also be passed to \
620-
enable `--extern crate_name` without `=path`".to_string());
621-
}
622618
let name = name.to_string();
623619
// For Rustdoc purposes, we can treat all externs as public
624620
externs.entry(name)

src/librustdoc/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -140,7 +140,7 @@ fn opts() -> Vec<RustcOptGroup> {
140140
}),
141141
stable("cfg", |o| o.optmulti("", "cfg", "pass a --cfg to rustc", "")),
142142
stable("extern", |o| {
143-
o.optmulti("", "extern", "pass an --extern to rustc", "NAME=PATH")
143+
o.optmulti("", "extern", "pass an --extern to rustc", "NAME[=PATH]")
144144
}),
145145
unstable("extern-html-root-url", |o| {
146146
o.optmulti("", "extern-html-root-url",

src/test/run-make-fulldeps/extern-flag-fun/Makefile

+3-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,6 @@ all:
44
$(RUSTC) bar.rs --crate-type=rlib
55
$(RUSTC) bar.rs --crate-type=rlib -C extra-filename=-a
66
$(RUSTC) bar-alt.rs --crate-type=rlib
7-
$(RUSTC) foo.rs --extern hello && exit 1 || exit 0
87
$(RUSTC) foo.rs --extern bar=no-exist && exit 1 || exit 0
98
$(RUSTC) foo.rs --extern bar=foo.rs && exit 1 || exit 0
109
$(RUSTC) foo.rs \
@@ -15,3 +14,6 @@ all:
1514
--extern bar=$(TMPDIR)/libbar.rlib \
1615
--extern bar=$(TMPDIR)/libbar-a.rlib
1716
$(RUSTC) foo.rs --extern bar=$(TMPDIR)/libbar.rlib
17+
# Try to be sneaky and load a private crate from with a non-private name.
18+
$(RUSTC) rustc.rs -Zforce-unstable-if-unmarked --crate-type=rlib
19+
$(RUSTC) gated_unstable.rs --extern alloc=$(TMPDIR)/librustc.rlib 2>&1 | $(CGREP) 'rustc_private'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
extern crate alloc;
2+
3+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
pub fn foo() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
-include ../tools.mk
2+
3+
# Test mixing pathless --extern with paths.
4+
5+
all:
6+
$(RUSTC) bar-static.rs --crate-name=bar --crate-type=rlib
7+
$(RUSTC) bar-dynamic.rs --crate-name=bar --crate-type=dylib -C prefer-dynamic
8+
# rlib preferred over dylib
9+
$(RUSTC) foo.rs --extern bar
10+
$(call RUN,foo) | $(CGREP) 'static'
11+
$(RUSTC) foo.rs --extern bar=$(TMPDIR)/libbar.rlib --extern bar
12+
$(call RUN,foo) | $(CGREP) 'static'
13+
# explicit --extern overrides pathless
14+
$(RUSTC) foo.rs --extern bar=$(call DYLIB,bar) --extern bar
15+
$(call RUN,foo) | $(CGREP) 'dynamic'
16+
# prefer-dynamic does what it says
17+
$(RUSTC) foo.rs --extern bar -C prefer-dynamic
18+
$(call RUN,foo) | $(CGREP) 'dynamic'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
pub fn f() {
2+
println!("dynamic");
3+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
pub fn f() {
2+
println!("static");
3+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
fn main() {
2+
bar::f();
3+
}

src/test/run-make-fulldeps/save-analysis-rfc2126/Makefile

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,7 @@
11
-include ../tools.mk
22

33
all: extern_absolute_paths.rs krate2
4-
$(RUSTC) extern_absolute_paths.rs -Zsave-analysis --edition=2018 \
5-
-Z unstable-options --extern krate2
4+
$(RUSTC) extern_absolute_paths.rs -Zsave-analysis --edition=2018 --extern krate2
65
cat $(TMPDIR)/save-analysis/extern_absolute_paths.json | "$(PYTHON)" validate_json.py
76

87
krate2: krate2.rs

src/test/rustdoc/inline_cross/use_crate.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
// aux-build:use_crate_2.rs
33
// build-aux-docs
44
// edition:2018
5-
// compile-flags:--extern use_crate --extern use_crate_2 -Z unstable-options
5+
// compile-flags:--extern use_crate --extern use_crate_2
66

77
// During the buildup to Rust 2018, rustdoc would eagerly inline `pub use some_crate;` as if it
88
// were a module, so we changed it to make `pub use`ing crate roots remain as a `pub use` statement
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
// ignore-stage1
2+
// edition:2018
3+
// compile-flags:--extern rustc
4+
5+
// Test that `--extern rustc` fails with `rustc_private`.
6+
7+
pub use rustc;
8+
//~^ ERROR use of unstable library feature 'rustc_private'
9+
10+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
error[E0658]: use of unstable library feature 'rustc_private': this crate is being loaded from the sysroot, an unstable location; did you mean to load this crate from crates.io via `Cargo.toml` instead?
2+
--> $DIR/pathless-extern-unstable.rs:7:9
3+
|
4+
LL | pub use rustc;
5+
| ^^^^^
6+
|
7+
= note: for more information, see https://github.com/rust-lang/rust/issues/27812
8+
= help: add `#![feature(rustc_private)]` to the crate attributes to enable
9+
10+
error: aborting due to previous error
11+
12+
For more information about this error, try `rustc --explain E0658`.

src/test/ui/pathless-extern-ok.rs

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
// edition:2018
2+
// compile-flags:--extern alloc
3+
// build-pass
4+
5+
// Test that `--extern alloc` will load from the sysroot without error.
6+
7+
fn main() {
8+
let _: Vec<i32> = alloc::vec::Vec::new();
9+
}

0 commit comments

Comments
 (0)