Skip to content

Commit 062dea0

Browse files
committed
Auto merge of rust-lang#71258 - Mark-Simulacrum:beta-next, r=Mark-Simulacrum
[beta] backports This includes: * Do not reuse post LTO products when exports change rust-lang#71131 * macro_rules: `NtLifetime` cannot start with an identifier rust-lang#70768 * Update RELEASES.md for 1.43.0 rust-lang#70354 r? @ghost
2 parents 934ae77 + 6d0fbd1 commit 062dea0

File tree

7 files changed

+285
-27
lines changed

7 files changed

+285
-27
lines changed

RELEASES.md

+149-3
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,152 @@
1+
Version 1.43.0 (2020-04-23)
2+
==========================
3+
4+
Language
5+
--------
6+
- [Fixed using binary operations with `&{number}` (e.g. `&1.0`) not having
7+
the type inferred correctly.][68129]
8+
- [Attributes such as `#[cfg()]` can now be used on `if` expressions.][69201]
9+
10+
**Syntax only changes**
11+
- [Allow `type Foo: Ord` syntactically.][69361]
12+
- [Fuse associated and extern items up to defaultness.][69194]
13+
- [Syntactically allow `self` in all `fn` contexts.][68764]
14+
- [Merge `fn` syntax + cleanup item parsing.][68728]
15+
- [`item` macro fragments can be interpolated into `trait`s, `impl`s, and `extern` blocks.][69366]
16+
For example, you may now write:
17+
```rust
18+
macro_rules! mac_trait {
19+
($i:item) => {
20+
trait T { $i }
21+
}
22+
}
23+
mac_trait! {
24+
fn foo() {}
25+
}
26+
```
27+
28+
These are still rejected *semantically*, so you will likely receive an error but
29+
these changes can be seen and parsed by macros and
30+
conditional compilation.
31+
32+
33+
Compiler
34+
--------
35+
- [You can now pass multiple lint flags to rustc to override the previous
36+
flags.][67885] For example; `rustc -D unused -A unused-variables` denies
37+
everything in the `unused` lint group except `unused-variables` which
38+
is explicitly allowed. However, passing `rustc -A unused-variables -D unused` denies
39+
everything in the `unused` lint group **including** `unused-variables` since
40+
the allow flag is specified before the deny flag (and therefore overridden).
41+
- [rustc will now prefer your system MinGW libraries over its bundled libraries
42+
if they are available on `windows-gnu`.][67429]
43+
- [rustc now buffers errors/warnings printed in JSON.][69227]
44+
45+
Libraries
46+
---------
47+
- [`Arc<[T; N]>`, `Box<[T; N]>`, and `Rc<[T; N]>`, now implement
48+
`TryFrom<Arc<[T]>>`,`TryFrom<Box<[T]>>`, and `TryFrom<Rc<[T]>>`
49+
respectively.][69538] **Note** These conversions are only available when `N`
50+
is `0..=32`.
51+
- [You can now use associated constants on floats and integers directly, rather
52+
than having to import the module.][68952] e.g. You can now write `u32::MAX` or
53+
`f32::NAN` with no imports.
54+
- [`u8::is_ascii` is now `const`.][68984]
55+
- [`String` now implements `AsMut<str>`.][68742]
56+
- [Added the `primitive` module to `std` and `core`.][67637] This module
57+
reexports Rust's primitive types. This is mainly useful in macros
58+
where you want avoid these types being shadowed.
59+
- [Relaxed some of the trait bounds on `HashMap` and `HashSet`.][67642]
60+
- [`string::FromUtf8Error` now implements `Clone + Eq`.][68738]
61+
62+
Stabilized APIs
63+
---------------
64+
- [`Once::is_completed`]
65+
- [`f32::LOG10_2`]
66+
- [`f32::LOG2_10`]
67+
- [`f64::LOG10_2`]
68+
- [`f64::LOG2_10`]
69+
- [`iter::once_with`]
70+
71+
Cargo
72+
-----
73+
- [You can now set config `[profile]`s in your `.cargo/config`, or through
74+
your environment.][cargo/7823]
75+
- [Cargo will now set `CARGO_BIN_EXE_<name>` pointing to a binary's
76+
executable path when running integration tests or benchmarks.][cargo/7697]
77+
`<name>` is the name of your binary as-is e.g. If you wanted the executable
78+
path for a binary named `my-program`you would use `env!("CARGO_BIN_EXE_my-program")`.
79+
80+
Misc
81+
----
82+
- [Certain checks in the `const_err` lint were deemed unrelated to const
83+
evaluation][69185], and have been moved to the `unconditional_panic` and
84+
`arithmetic_overflow` lints.
85+
86+
Compatibility Notes
87+
-------------------
88+
89+
- [Having trailing syntax in the `assert!` macro is now a hard error.][69548] This
90+
has been a warning since 1.36.0.
91+
- [Fixed `Self` not having the correctly inferred type.][69340] This incorrectly
92+
led to some instances being accepted, and now correctly emits a hard error.
93+
94+
[69340]: https://github.com/rust-lang/rust/pull/69340
95+
96+
Internal Only
97+
-------------
98+
These changes provide no direct user facing benefits, but represent significant
99+
improvements to the internals and overall performance of `rustc` and
100+
related tools.
101+
102+
- [All components are now built with `opt-level=3` instead of `2`.][67878]
103+
- [Improved how rustc generates drop code.][67332]
104+
- [Improved performance from `#[inline]`-ing certain hot functions.][69256]
105+
- [traits: preallocate 2 Vecs of known initial size][69022]
106+
- [Avoid exponential behaviour when relating types][68772]
107+
- [Skip `Drop` terminators for enum variants without drop glue][68943]
108+
- [Improve performance of coherence checks][68966]
109+
- [Deduplicate types in the generator witness][68672]
110+
- [Invert control in struct_lint_level.][68725]
111+
112+
[67332]: https://github.com/rust-lang/rust/pull/67332/
113+
[67429]: https://github.com/rust-lang/rust/pull/67429/
114+
[67637]: https://github.com/rust-lang/rust/pull/67637/
115+
[67642]: https://github.com/rust-lang/rust/pull/67642/
116+
[67878]: https://github.com/rust-lang/rust/pull/67878/
117+
[67885]: https://github.com/rust-lang/rust/pull/67885/
118+
[68129]: https://github.com/rust-lang/rust/pull/68129/
119+
[68672]: https://github.com/rust-lang/rust/pull/68672/
120+
[68725]: https://github.com/rust-lang/rust/pull/68725/
121+
[68728]: https://github.com/rust-lang/rust/pull/68728/
122+
[68738]: https://github.com/rust-lang/rust/pull/68738/
123+
[68742]: https://github.com/rust-lang/rust/pull/68742/
124+
[68764]: https://github.com/rust-lang/rust/pull/68764/
125+
[68772]: https://github.com/rust-lang/rust/pull/68772/
126+
[68943]: https://github.com/rust-lang/rust/pull/68943/
127+
[68952]: https://github.com/rust-lang/rust/pull/68952/
128+
[68966]: https://github.com/rust-lang/rust/pull/68966/
129+
[68984]: https://github.com/rust-lang/rust/pull/68984/
130+
[69022]: https://github.com/rust-lang/rust/pull/69022/
131+
[69185]: https://github.com/rust-lang/rust/pull/69185/
132+
[69194]: https://github.com/rust-lang/rust/pull/69194/
133+
[69201]: https://github.com/rust-lang/rust/pull/69201/
134+
[69227]: https://github.com/rust-lang/rust/pull/69227/
135+
[69548]: https://github.com/rust-lang/rust/pull/69548/
136+
[69256]: https://github.com/rust-lang/rust/pull/69256/
137+
[69361]: https://github.com/rust-lang/rust/pull/69361/
138+
[69366]: https://github.com/rust-lang/rust/pull/69366/
139+
[69538]: https://github.com/rust-lang/rust/pull/69538/
140+
[cargo/7823]: https://github.com/rust-lang/cargo/pull/7823
141+
[cargo/7697]: https://github.com/rust-lang/cargo/pull/7697
142+
[`Once::is_completed`]: https://doc.rust-lang.org/std/sync/struct.Once.html#method.is_completed
143+
[`f32::LOG10_2`]: https://doc.rust-lang.org/std/f32/consts/constant.LOG10_2.html
144+
[`f32::LOG2_10`]: https://doc.rust-lang.org/std/f32/consts/constant.LOG2_10.html
145+
[`f64::LOG10_2`]: https://doc.rust-lang.org/std/f64/consts/constant.LOG10_2.html
146+
[`f64::LOG2_10`]: https://doc.rust-lang.org/std/f64/consts/constant.LOG2_10.html
147+
[`iter::once_with`]: https://doc.rust-lang.org/std/iter/fn.once_with.html
148+
149+
1150
Version 1.42.0 (2020-03-12)
2151
==========================
3152

@@ -4838,7 +4987,6 @@ Version 1.11.0 (2016-08-18)
48384987
Language
48394988
--------
48404989

4841-
* [`cfg_attr` works on `path` attributes](https://github.com/rust-lang/rust/pull/34546)
48424990
* [Support nested `cfg_attr` attributes](https://github.com/rust-lang/rust/pull/34216)
48434991
* [Allow statement-generating braced macro invocations at the end of blocks](https://github.com/rust-lang/rust/pull/34436)
48444992
* [Macros can be expanded inside of trait definitions](https://github.com/rust-lang/rust/pull/34213)
@@ -4957,8 +5105,6 @@ Version 1.10.0 (2016-07-07)
49575105
Language
49585106
--------
49595107

4960-
* [Allow `concat_idents!` in type positions as well as in expression
4961-
positions](https://github.com/rust-lang/rust/pull/33735).
49625108
* [`Copy` types are required to have a trivial implementation of `Clone`](https://github.com/rust-lang/rust/pull/33420).
49635109
[RFC 1521](https://github.com/rust-lang/rfcs/blob/master/text/1521-copy-clone-semantics.md).
49645110
* [Single-variant enums support the `#[repr(..)]` attribute](https://github.com/rust-lang/rust/pull/33355).

src/librustc_codegen_llvm/back/lto.rs

+69-22
Original file line numberDiff line numberDiff line change
@@ -463,15 +463,18 @@ fn thin_lto(
463463
// If previous imports have been deleted, or we get an IO error
464464
// reading the file storing them, then we'll just use `None` as the
465465
// prev_import_map, which will force the code to be recompiled.
466-
let prev =
467-
if path.exists() { ThinLTOImports::load_from_file(&path).ok() } else { None };
468-
let curr = ThinLTOImports::from_thin_lto_data(data);
466+
let prev = if path.exists() {
467+
ThinLTOImportMaps::load_from_file(&path).ok()
468+
} else {
469+
None
470+
};
471+
let curr = ThinLTOImportMaps::from_thin_lto_data(data);
469472
(Some(path), prev, curr)
470473
} else {
471474
// If we don't compile incrementally, we don't need to load the
472475
// import data from LLVM.
473476
assert!(green_modules.is_empty());
474-
let curr = ThinLTOImports::default();
477+
let curr = ThinLTOImportMaps::default();
475478
(None, None, curr)
476479
};
477480
info!("thin LTO import map loaded");
@@ -497,10 +500,14 @@ fn thin_lto(
497500
let module_name = module_name_to_str(module_name);
498501

499502
// If (1.) the module hasn't changed, and (2.) none of the modules
500-
// it imports from has changed, *and* (3.) the import-set itself has
501-
// not changed from the previous compile when it was last
502-
// ThinLTO'ed, then we can re-use the post-ThinLTO version of the
503-
// module. Otherwise, freshly perform LTO optimization.
503+
// it imports from nor exports to have changed, *and* (3.) the
504+
// import and export sets themselves have not changed from the
505+
// previous compile when it was last ThinLTO'ed, then we can re-use
506+
// the post-ThinLTO version of the module. Otherwise, freshly
507+
// perform LTO optimization.
508+
//
509+
// (Note that globally, the export set is just the inverse of the
510+
// import set.)
504511
//
505512
// This strategy means we can always save the computed imports as
506513
// canon: when we reuse the post-ThinLTO version, condition (3.)
@@ -509,19 +516,30 @@ fn thin_lto(
509516
// version, the current import set *is* the correct one, since we
510517
// are doing the ThinLTO in this current compilation cycle.)
511518
//
512-
// See rust-lang/rust#59535.
519+
// For more discussion, see rust-lang/rust#59535 (where the import
520+
// issue was discovered) and rust-lang/rust#69798 (where the
521+
// analogous export issue was discovered).
513522
if let (Some(prev_import_map), true) =
514523
(prev_import_map.as_ref(), green_modules.contains_key(module_name))
515524
{
516525
assert!(cgcx.incr_comp_session_dir.is_some());
517526

518-
let prev_imports = prev_import_map.modules_imported_by(module_name);
519-
let curr_imports = curr_import_map.modules_imported_by(module_name);
527+
let prev_imports = prev_import_map.imports_of(module_name);
528+
let curr_imports = curr_import_map.imports_of(module_name);
529+
let prev_exports = prev_import_map.exports_of(module_name);
530+
let curr_exports = curr_import_map.exports_of(module_name);
520531
let imports_all_green = curr_imports
521532
.iter()
522533
.all(|imported_module| green_modules.contains_key(imported_module));
534+
let exports_all_green = curr_exports
535+
.iter()
536+
.all(|exported_module| green_modules.contains_key(exported_module));
523537

524-
if imports_all_green && equivalent_as_sets(prev_imports, curr_imports) {
538+
if imports_all_green
539+
&& equivalent_as_sets(prev_imports, curr_imports)
540+
&& exports_all_green
541+
&& equivalent_as_sets(prev_exports, curr_exports)
542+
{
525543
let work_product = green_modules[module_name].clone();
526544
copy_jobs.push(work_product);
527545
info!(" - {}: re-used", module_name);
@@ -881,17 +899,32 @@ pub unsafe fn optimize_thin_module(
881899
Ok(module)
882900
}
883901

902+
/// Summarizes module import/export relationships used by LLVM's ThinLTO pass.
903+
///
904+
/// Note that we tend to have two such instances of `ThinLTOImportMaps` in use:
905+
/// one loaded from a file that represents the relationships used during the
906+
/// compilation associated with the incremetnal build artifacts we are
907+
/// attempting to reuse, and another constructed via `from_thin_lto_data`, which
908+
/// captures the relationships of ThinLTO in the current compilation.
884909
#[derive(Debug, Default)]
885-
pub struct ThinLTOImports {
910+
pub struct ThinLTOImportMaps {
886911
// key = llvm name of importing module, value = list of modules it imports from
887912
imports: FxHashMap<String, Vec<String>>,
913+
// key = llvm name of exporting module, value = list of modules it exports to
914+
exports: FxHashMap<String, Vec<String>>,
888915
}
889916

890-
impl ThinLTOImports {
891-
fn modules_imported_by(&self, llvm_module_name: &str) -> &[String] {
917+
impl ThinLTOImportMaps {
918+
/// Returns modules imported by `llvm_module_name` during some ThinLTO pass.
919+
fn imports_of(&self, llvm_module_name: &str) -> &[String] {
892920
self.imports.get(llvm_module_name).map(|v| &v[..]).unwrap_or(&[])
893921
}
894922

923+
/// Returns modules exported by `llvm_module_name` during some ThinLTO pass.
924+
fn exports_of(&self, llvm_module_name: &str) -> &[String] {
925+
self.exports.get(llvm_module_name).map(|v| &v[..]).unwrap_or(&[])
926+
}
927+
895928
fn save_to_file(&self, path: &Path) -> io::Result<()> {
896929
use std::io::Write;
897930
let file = File::create(path)?;
@@ -906,16 +939,20 @@ impl ThinLTOImports {
906939
Ok(())
907940
}
908941

909-
fn load_from_file(path: &Path) -> io::Result<ThinLTOImports> {
942+
fn load_from_file(path: &Path) -> io::Result<ThinLTOImportMaps> {
910943
use std::io::BufRead;
911944
let mut imports = FxHashMap::default();
912-
let mut current_module = None;
913-
let mut current_imports = vec![];
945+
let mut exports: FxHashMap<_, Vec<_>> = FxHashMap::default();
946+
let mut current_module: Option<String> = None;
947+
let mut current_imports: Vec<String> = vec![];
914948
let file = File::open(path)?;
915949
for line in io::BufReader::new(file).lines() {
916950
let line = line?;
917951
if line.is_empty() {
918952
let importing_module = current_module.take().expect("Importing module not set");
953+
for imported in &current_imports {
954+
exports.entry(imported.clone()).or_default().push(importing_module.clone());
955+
}
919956
imports.insert(importing_module, mem::replace(&mut current_imports, vec![]));
920957
} else if line.starts_with(' ') {
921958
// Space marks an imported module
@@ -927,17 +964,17 @@ impl ThinLTOImports {
927964
current_module = Some(line.trim().to_string());
928965
}
929966
}
930-
Ok(ThinLTOImports { imports })
967+
Ok(ThinLTOImportMaps { imports, exports })
931968
}
932969

933970
/// Loads the ThinLTO import map from ThinLTOData.
934-
unsafe fn from_thin_lto_data(data: *const llvm::ThinLTOData) -> ThinLTOImports {
971+
unsafe fn from_thin_lto_data(data: *const llvm::ThinLTOData) -> ThinLTOImportMaps {
935972
unsafe extern "C" fn imported_module_callback(
936973
payload: *mut libc::c_void,
937974
importing_module_name: *const libc::c_char,
938975
imported_module_name: *const libc::c_char,
939976
) {
940-
let map = &mut *(payload as *mut ThinLTOImports);
977+
let map = &mut *(payload as *mut ThinLTOImportMaps);
941978
let importing_module_name = CStr::from_ptr(importing_module_name);
942979
let importing_module_name = module_name_to_str(&importing_module_name);
943980
let imported_module_name = CStr::from_ptr(imported_module_name);
@@ -951,8 +988,18 @@ impl ThinLTOImports {
951988
.get_mut(importing_module_name)
952989
.unwrap()
953990
.push(imported_module_name.to_owned());
991+
992+
if !map.exports.contains_key(imported_module_name) {
993+
map.exports.insert(imported_module_name.to_owned(), vec![]);
994+
}
995+
996+
map.exports
997+
.get_mut(imported_module_name)
998+
.unwrap()
999+
.push(importing_module_name.to_owned());
9541000
}
955-
let mut map = ThinLTOImports::default();
1001+
1002+
let mut map = ThinLTOImportMaps::default();
9561003
llvm::LLVMRustGetThinLTOModuleImports(
9571004
data,
9581005
imported_module_callback,

src/librustc_expand/mbe/macro_parser.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -762,7 +762,7 @@ fn may_begin_with(token: &Token, name: Name) -> bool {
762762
/// Checks whether the non-terminal may contain a single (non-keyword) identifier.
763763
fn may_be_ident(nt: &token::Nonterminal) -> bool {
764764
match *nt {
765-
token::NtItem(_) | token::NtBlock(_) | token::NtVis(_) => false,
765+
token::NtItem(_) | token::NtBlock(_) | token::NtVis(_) | token::NtLifetime(_) => false,
766766
_ => true,
767767
}
768768
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
// revisions: cfail1 cfail2
2+
// build-pass
3+
4+
// rust-lang/rust#69798:
5+
//
6+
// This is analgous to cgu_invalidated_when_import_added, but it covers a
7+
// problem uncovered where a change to the *export* set caused a link failure
8+
// when reusing post-LTO optimized object code.
9+
10+
pub struct Foo {}
11+
impl Drop for Foo {
12+
fn drop(&mut self) {
13+
println!("Dropping Foo");
14+
}
15+
}
16+
#[no_mangle]
17+
pub extern "C" fn run() {
18+
thread_local! { pub static FOO : Foo = Foo { } ; }
19+
20+
#[cfg(cfail2)]
21+
{
22+
FOO.with(|_f| ())
23+
}
24+
}
25+
26+
pub fn main() { run() }

0 commit comments

Comments
 (0)