Skip to content

Commit 387e7a5

Browse files
committedJan 9, 2024
Auto merge of #119760 - matthiaskrgr:rollup-ti2xpp7, r=matthiaskrgr
Rollup of 7 pull requests Successful merges: - #117744 (Add -Zuse-sync-unwind) - #118649 (Make inductive cycles in coherence ambiguous always) - #118979 (Use `assert_unsafe_precondition` for `char::from_u32_unchecked`) - #119619 (mir-opt and custom target fixes) - #119632 (Fix broken build for ESP IDF due to #119026) - #119712 (Adding alignment to the cases to test for specific error messages.) - #119734 (Miri subtree update) r? `@ghost` `@rustbot` modify labels: rollup
2 parents a399117 + 91fcc17 commit 387e7a5

File tree

38 files changed

+1002
-737
lines changed

38 files changed

+1002
-737
lines changed
 

‎Cargo.lock

+3-3
Original file line numberDiff line numberDiff line change
@@ -2465,6 +2465,7 @@ dependencies = [
24652465
"ctrlc",
24662466
"env_logger",
24672467
"getrandom",
2468+
"jemalloc-sys",
24682469
"lazy_static",
24692470
"libc",
24702471
"libffi",
@@ -2474,7 +2475,6 @@ dependencies = [
24742475
"rand",
24752476
"regex",
24762477
"rustc_version",
2477-
"serde",
24782478
"smallvec",
24792479
"ui_test",
24802480
]
@@ -3280,9 +3280,9 @@ dependencies = [
32803280

32813281
[[package]]
32823282
name = "rustc-build-sysroot"
3283-
version = "0.4.2"
3283+
version = "0.4.4"
32843284
source = "registry+https://github.com/rust-lang/crates.io-index"
3285-
checksum = "8ed2a90dfa5232ed5ff21d53d4df655f315ab316ea06fc508f1c74bcedb1ce6c"
3285+
checksum = "39dcf8d82b1f79a179bdb284dc44db440a9666eefa5a6df5ef282d6db930d544"
32863286
dependencies = [
32873287
"anyhow",
32883288
"rustc_version",

‎compiler/rustc_codegen_llvm/src/allocator.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -134,7 +134,8 @@ fn create_wrapper_function(
134134
llvm::LLVMRustSetVisibility(llfn, llvm::Visibility::Hidden);
135135
}
136136
if tcx.sess.must_emit_unwind_tables() {
137-
let uwtable = attributes::uwtable_attr(llcx);
137+
let uwtable =
138+
attributes::uwtable_attr(llcx, tcx.sess.opts.unstable_opts.use_sync_unwind);
138139
attributes::apply_to_llfn(llfn, llvm::AttributePlace::Function, &[uwtable]);
139140
}
140141

‎compiler/rustc_codegen_llvm/src/attributes.rs

+4-3
Original file line numberDiff line numberDiff line change
@@ -95,11 +95,12 @@ pub fn sanitize_attrs<'ll>(
9595

9696
/// Tell LLVM to emit or not emit the information necessary to unwind the stack for the function.
9797
#[inline]
98-
pub fn uwtable_attr(llcx: &llvm::Context) -> &Attribute {
98+
pub fn uwtable_attr(llcx: &llvm::Context, use_sync_unwind: Option<bool>) -> &Attribute {
9999
// NOTE: We should determine if we even need async unwind tables, as they
100100
// take have more overhead and if we can use sync unwind tables we
101101
// probably should.
102-
llvm::CreateUWTableAttr(llcx, true)
102+
let async_unwind = !use_sync_unwind.unwrap_or(false);
103+
llvm::CreateUWTableAttr(llcx, async_unwind)
103104
}
104105

105106
pub fn frame_pointer_type_attr<'ll>(cx: &CodegenCx<'ll, '_>) -> Option<&'ll Attribute> {
@@ -333,7 +334,7 @@ pub fn from_fn_attrs<'ll, 'tcx>(
333334
// You can also find more info on why Windows always requires uwtables here:
334335
// https://bugzilla.mozilla.org/show_bug.cgi?id=1302078
335336
if cx.sess().must_emit_unwind_tables() {
336-
to_add.push(uwtable_attr(cx.llcx));
337+
to_add.push(uwtable_attr(cx.llcx, cx.sess().opts.unstable_opts.use_sync_unwind));
337338
}
338339

339340
if cx.sess().opts.unstable_opts.profile_sample_use.is_some() {

‎compiler/rustc_lint/src/lib.rs

+5
Original file line numberDiff line numberDiff line change
@@ -513,6 +513,11 @@ fn register_builtins(store: &mut LintStore) {
513513
"converted into hard error, see PR #117984 \
514514
<https://github.com/rust-lang/rust/pull/117984> for more information",
515515
);
516+
store.register_removed(
517+
"coinductive_overlap_in_coherence",
518+
"converted into hard error, see PR #118649 \
519+
<https://github.com/rust-lang/rust/pull/118649> for more information",
520+
);
516521
}
517522

518523
fn register_internals(store: &mut LintStore) {

‎compiler/rustc_lint_defs/src/builtin.rs

-40
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,6 @@ declare_lint_pass! {
2626
BYTE_SLICE_IN_PACKED_STRUCT_WITH_DERIVE,
2727
CENUM_IMPL_DROP_CAST,
2828
COHERENCE_LEAK_CHECK,
29-
COINDUCTIVE_OVERLAP_IN_COHERENCE,
3029
CONFLICTING_REPR_HINTS,
3130
CONST_EVALUATABLE_UNCHECKED,
3231
CONST_ITEM_MUTATION,
@@ -4367,45 +4366,6 @@ declare_lint! {
43674366
@feature_gate = sym::type_privacy_lints;
43684367
}
43694368

4370-
declare_lint! {
4371-
/// The `coinductive_overlap_in_coherence` lint detects impls which are currently
4372-
/// considered not overlapping, but may be considered to overlap if support for
4373-
/// coinduction is added to the trait solver.
4374-
///
4375-
/// ### Example
4376-
///
4377-
/// ```rust,compile_fail
4378-
/// #![deny(coinductive_overlap_in_coherence)]
4379-
///
4380-
/// trait CyclicTrait {}
4381-
/// impl<T: CyclicTrait> CyclicTrait for T {}
4382-
///
4383-
/// trait Trait {}
4384-
/// impl<T: CyclicTrait> Trait for T {}
4385-
/// // conflicting impl with the above
4386-
/// impl Trait for u8 {}
4387-
/// ```
4388-
///
4389-
/// {{produces}}
4390-
///
4391-
/// ### Explanation
4392-
///
4393-
/// We have two choices for impl which satisfy `u8: Trait`: the blanket impl
4394-
/// for generic `T`, and the direct impl for `u8`. These two impls nominally
4395-
/// overlap, since we can infer `T = u8` in the former impl, but since the where
4396-
/// clause `u8: CyclicTrait` would end up resulting in a cycle (since it depends
4397-
/// on itself), the blanket impl is not considered to hold for `u8`. This will
4398-
/// change in a future release.
4399-
pub COINDUCTIVE_OVERLAP_IN_COHERENCE,
4400-
Deny,
4401-
"impls that are not considered to overlap may be considered to \
4402-
overlap in the future",
4403-
@future_incompatible = FutureIncompatibleInfo {
4404-
reason: FutureIncompatibilityReason::FutureReleaseErrorReportInDeps,
4405-
reference: "issue #114040 <https://github.com/rust-lang/rust/issues/114040>",
4406-
};
4407-
}
4408-
44094369
declare_lint! {
44104370
/// The `unknown_or_malformed_diagnostic_attributes` lint detects unrecognized or otherwise malformed
44114371
/// diagnostic attributes.

‎compiler/rustc_parse_format/src/lib.rs

+22-6
Original file line numberDiff line numberDiff line change
@@ -289,10 +289,10 @@ impl<'a> Iterator for Parser<'a> {
289289
}
290290
} else {
291291
if let Some(&(_, maybe)) = self.cur.peek() {
292-
if maybe == '?' {
293-
self.suggest_format();
294-
} else {
295-
self.suggest_positional_arg_instead_of_captured_arg(arg);
292+
match maybe {
293+
'?' => self.suggest_format_debug(),
294+
'<' | '^' | '>' => self.suggest_format_align(maybe),
295+
_ => self.suggest_positional_arg_instead_of_captured_arg(arg),
296296
}
297297
}
298298
}
@@ -868,10 +868,9 @@ impl<'a> Parser<'a> {
868868
found.then_some(cur)
869869
}
870870

871-
fn suggest_format(&mut self) {
871+
fn suggest_format_debug(&mut self) {
872872
if let (Some(pos), Some(_)) = (self.consume_pos('?'), self.consume_pos(':')) {
873873
let word = self.word();
874-
let _end = self.current_pos();
875874
let pos = self.to_span_index(pos);
876875
self.errors.insert(
877876
0,
@@ -887,6 +886,23 @@ impl<'a> Parser<'a> {
887886
}
888887
}
889888

889+
fn suggest_format_align(&mut self, alignment: char) {
890+
if let Some(pos) = self.consume_pos(alignment) {
891+
let pos = self.to_span_index(pos);
892+
self.errors.insert(
893+
0,
894+
ParseError {
895+
description: "expected format parameter to occur after `:`".to_owned(),
896+
note: None,
897+
label: format!("expected `{}` to occur after `:`", alignment).to_owned(),
898+
span: pos.to(pos),
899+
secondary_label: None,
900+
suggestion: Suggestion::None,
901+
},
902+
);
903+
}
904+
}
905+
890906
fn suggest_positional_arg_instead_of_captured_arg(&mut self, arg: Argument<'a>) {
891907
if let Some(end) = self.consume_pos('.') {
892908
let byte_pos = self.to_span_index(end);

‎compiler/rustc_session/src/options.rs

+2
Original file line numberDiff line numberDiff line change
@@ -1960,6 +1960,8 @@ written to standard error output)"),
19601960
"adds unstable command line options to rustc interface (default: no)"),
19611961
use_ctors_section: Option<bool> = (None, parse_opt_bool, [TRACKED],
19621962
"use legacy .ctors section for initializers rather than .init_array"),
1963+
use_sync_unwind: Option<bool> = (None, parse_opt_bool, [TRACKED],
1964+
"Generate sync unwind tables instead of async unwind tables (default: no)"),
19631965
validate_mir: bool = (false, parse_bool, [UNTRACKED],
19641966
"validate MIR after each transformation"),
19651967
#[rustc_lint_opt_deny_field_access("use `Session::verbose_internals` instead of this field")]

‎compiler/rustc_trait_selection/src/traits/coherence.rs

+6-58
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ use crate::solve::inspect::{InspectGoal, ProofTreeInferCtxtExt, ProofTreeVisitor
1010
use crate::solve::{deeply_normalize_for_diagnostics, inspect};
1111
use crate::traits::engine::TraitEngineExt;
1212
use crate::traits::query::evaluate_obligation::InferCtxtExt;
13-
use crate::traits::select::{IntercrateAmbiguityCause, TreatInductiveCycleAs};
13+
use crate::traits::select::IntercrateAmbiguityCause;
1414
use crate::traits::structural_normalize::StructurallyNormalizeExt;
1515
use crate::traits::NormalizeExt;
1616
use crate::traits::SkipLeakCheck;
@@ -31,7 +31,6 @@ use rustc_middle::traits::DefiningAnchor;
3131
use rustc_middle::ty::fast_reject::{DeepRejectCtxt, TreatParams};
3232
use rustc_middle::ty::visit::{TypeVisitable, TypeVisitableExt};
3333
use rustc_middle::ty::{self, Ty, TyCtxt, TypeSuperVisitable, TypeVisitor};
34-
use rustc_session::lint::builtin::COINDUCTIVE_OVERLAP_IN_COHERENCE;
3534
use rustc_span::symbol::sym;
3635
use rustc_span::DUMMY_SP;
3736
use std::fmt::Debug;
@@ -197,7 +196,7 @@ fn overlap<'tcx>(
197196
.intercrate(true)
198197
.with_next_trait_solver(tcx.next_trait_solver_in_coherence())
199198
.build();
200-
let selcx = &mut SelectionContext::new(&infcx);
199+
let selcx = &mut SelectionContext::with_treat_inductive_cycle_as_ambig(&infcx);
201200
if track_ambiguity_causes.is_yes() {
202201
selcx.enable_tracking_intercrate_ambiguity_causes();
203202
}
@@ -224,61 +223,10 @@ fn overlap<'tcx>(
224223
);
225224

226225
if overlap_mode.use_implicit_negative() {
227-
for mode in [TreatInductiveCycleAs::Ambig, TreatInductiveCycleAs::Recur] {
228-
if let Some(failing_obligation) = selcx.with_treat_inductive_cycle_as(mode, |selcx| {
229-
impl_intersection_has_impossible_obligation(selcx, &obligations)
230-
}) {
231-
if matches!(mode, TreatInductiveCycleAs::Recur) {
232-
let first_local_impl = impl1_header
233-
.impl_def_id
234-
.as_local()
235-
.or(impl2_header.impl_def_id.as_local())
236-
.expect("expected one of the impls to be local");
237-
infcx.tcx.struct_span_lint_hir(
238-
COINDUCTIVE_OVERLAP_IN_COHERENCE,
239-
infcx.tcx.local_def_id_to_hir_id(first_local_impl),
240-
infcx.tcx.def_span(first_local_impl),
241-
format!(
242-
"implementations {} will conflict in the future",
243-
match impl1_header.trait_ref {
244-
Some(trait_ref) => {
245-
let trait_ref = infcx.resolve_vars_if_possible(trait_ref);
246-
format!(
247-
"of `{}` for `{}`",
248-
trait_ref.print_trait_sugared(),
249-
trait_ref.self_ty()
250-
)
251-
}
252-
None => format!(
253-
"for `{}`",
254-
infcx.resolve_vars_if_possible(impl1_header.self_ty)
255-
),
256-
},
257-
),
258-
|lint| {
259-
lint.note(
260-
"impls that are not considered to overlap may be considered to \
261-
overlap in the future",
262-
)
263-
.span_label(
264-
infcx.tcx.def_span(impl1_header.impl_def_id),
265-
"the first impl is here",
266-
)
267-
.span_label(
268-
infcx.tcx.def_span(impl2_header.impl_def_id),
269-
"the second impl is here",
270-
);
271-
lint.note(format!(
272-
"`{}` may be considered to hold in future releases, \
273-
causing the impls to overlap",
274-
infcx.resolve_vars_if_possible(failing_obligation.predicate)
275-
));
276-
},
277-
);
278-
}
279-
280-
return None;
281-
}
226+
if let Some(_failing_obligation) =
227+
impl_intersection_has_impossible_obligation(selcx, &obligations)
228+
{
229+
return None;
282230
}
283231
}
284232

‎compiler/rustc_trait_selection/src/traits/select/mod.rs

+8-12
Original file line numberDiff line numberDiff line change
@@ -239,20 +239,16 @@ impl<'cx, 'tcx> SelectionContext<'cx, 'tcx> {
239239
}
240240
}
241241

242-
// Sets the `TreatInductiveCycleAs` mode temporarily in the selection context
243-
pub fn with_treat_inductive_cycle_as<T>(
244-
&mut self,
245-
treat_inductive_cycle: TreatInductiveCycleAs,
246-
f: impl FnOnce(&mut Self) -> T,
247-
) -> T {
242+
pub fn with_treat_inductive_cycle_as_ambig(
243+
infcx: &'cx InferCtxt<'tcx>,
244+
) -> SelectionContext<'cx, 'tcx> {
248245
// Should be executed in a context where caching is disabled,
249246
// otherwise the cache is poisoned with the temporary result.
250-
assert!(self.is_intercrate());
251-
let treat_inductive_cycle =
252-
std::mem::replace(&mut self.treat_inductive_cycle, treat_inductive_cycle);
253-
let value = f(self);
254-
self.treat_inductive_cycle = treat_inductive_cycle;
255-
value
247+
assert!(infcx.intercrate);
248+
SelectionContext {
249+
treat_inductive_cycle: TreatInductiveCycleAs::Ambig,
250+
..SelectionContext::new(infcx)
251+
}
256252
}
257253

258254
pub fn with_query_mode(

‎library/core/src/char/convert.rs

+8-1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ use crate::char::TryFromCharError;
44
use crate::convert::TryFrom;
55
use crate::error::Error;
66
use crate::fmt;
7+
use crate::intrinsics::assert_unsafe_precondition;
78
use crate::mem::transmute;
89
use crate::str::FromStr;
910

@@ -23,7 +24,13 @@ pub(super) const fn from_u32(i: u32) -> Option<char> {
2324
#[must_use]
2425
pub(super) const unsafe fn from_u32_unchecked(i: u32) -> char {
2526
// SAFETY: the caller must guarantee that `i` is a valid char value.
26-
if cfg!(debug_assertions) { char::from_u32(i).unwrap() } else { unsafe { transmute(i) } }
27+
unsafe {
28+
assert_unsafe_precondition!(
29+
"invalid value for `char`",
30+
(i: u32) => char_try_from_u32(i).is_ok()
31+
);
32+
transmute(i)
33+
}
2734
}
2835

2936
#[stable(feature = "char_convert", since = "1.13.0")]

‎library/std/src/os/unix/net/listener.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,7 @@ impl UnixListener {
7373
unsafe {
7474
let inner = Socket::new_raw(libc::AF_UNIX, libc::SOCK_STREAM)?;
7575
let (addr, len) = sockaddr_un(path.as_ref())?;
76-
#[cfg(any(target_os = "windows", target_os = "redox"))]
76+
#[cfg(any(target_os = "windows", target_os = "redox", target_os = "espidf"))]
7777
const backlog: libc::c_int = 128;
7878
#[cfg(any(target_os = "linux", target_os = "freebsd", target_os = "openbsd"))]
7979
const backlog: libc::c_int = -1;
@@ -82,7 +82,8 @@ impl UnixListener {
8282
target_os = "redox",
8383
target_os = "linux",
8484
target_os = "freebsd",
85-
target_os = "openbsd"
85+
target_os = "openbsd",
86+
target_os = "espidf"
8687
)))]
8788
const backlog: libc::c_int = libc::SOMAXCONN;
8889

‎src/bootstrap/src/core/build_steps/synthetic_targets.rs

+5
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,11 @@ fn create_synthetic_target(
5959
let mut cmd = Command::new(builder.rustc(compiler));
6060
cmd.arg("--target").arg(base.rustc_target_arg());
6161
cmd.args(["-Zunstable-options", "--print", "target-spec-json"]);
62+
63+
// If `rust.channel` is set to either beta or stable, rustc will complain that
64+
// we cannot use nightly features. So `RUSTC_BOOTSTRAP` is needed here.
65+
cmd.env("RUSTC_BOOTSTRAP", "1");
66+
6267
cmd.stdout(Stdio::piped());
6368

6469
let output = cmd.spawn().unwrap().wait_with_output().unwrap();

‎src/bootstrap/src/core/build_steps/test.rs

+7-2
Original file line numberDiff line numberDiff line change
@@ -1596,8 +1596,13 @@ NOTE: if you're sure you want to do this, please open an issue as to why. In the
15961596
// NOTE: Only stage 1 is special cased because we need the rustc_private artifacts to match the
15971597
// running compiler in stage 2 when plugins run.
15981598
let stage_id = if suite == "ui-fulldeps" && compiler.stage == 1 {
1599-
compiler = builder.compiler(compiler.stage - 1, target);
1600-
format!("stage{}-{}", compiler.stage + 1, target)
1599+
// At stage 0 (stage - 1) we are using the beta compiler. Using `self.target` can lead finding
1600+
// an incorrect compiler path on cross-targets, as the stage 0 beta compiler is always equal
1601+
// to `build.build` in the configuration.
1602+
let build = builder.build.build;
1603+
1604+
compiler = builder.compiler(compiler.stage - 1, build);
1605+
format!("stage{}-{}", compiler.stage + 1, build)
16011606
} else {
16021607
format!("stage{}-{}", compiler.stage, target)
16031608
};

‎src/tools/compiletest/src/common.rs

+33-12
Original file line numberDiff line numberDiff line change
@@ -483,6 +483,7 @@ impl TargetCfgs {
483483
let mut targets: HashMap<String, TargetCfg> = serde_json::from_str(&rustc_output(
484484
config,
485485
&["--print=all-target-specs-json", "-Zunstable-options"],
486+
Default::default(),
486487
))
487488
.unwrap();
488489

@@ -495,16 +496,33 @@ impl TargetCfgs {
495496
let mut all_families = HashSet::new();
496497
let mut all_pointer_widths = HashSet::new();
497498

498-
// Handle custom target specs, which are not included in `--print=all-target-specs-json`.
499-
if config.target.ends_with(".json") {
500-
targets.insert(
501-
config.target.clone(),
502-
serde_json::from_str(&rustc_output(
503-
config,
504-
&["--print=target-spec-json", "-Zunstable-options", "--target", &config.target],
505-
))
506-
.unwrap(),
507-
);
499+
// If current target is not included in the `--print=all-target-specs-json` output,
500+
// we check whether it is a custom target from the user or a synthetic target from bootstrap.
501+
if !targets.contains_key(&config.target) {
502+
let mut envs: HashMap<String, String> = HashMap::new();
503+
504+
if let Ok(t) = std::env::var("RUST_TARGET_PATH") {
505+
envs.insert("RUST_TARGET_PATH".into(), t);
506+
}
507+
508+
// This returns false only when the target is neither a synthetic target
509+
// nor a custom target from the user, indicating it is most likely invalid.
510+
if config.target.ends_with(".json") || !envs.is_empty() {
511+
targets.insert(
512+
config.target.clone(),
513+
serde_json::from_str(&rustc_output(
514+
config,
515+
&[
516+
"--print=target-spec-json",
517+
"-Zunstable-options",
518+
"--target",
519+
&config.target,
520+
],
521+
envs,
522+
))
523+
.unwrap(),
524+
);
525+
}
508526
}
509527

510528
for (target, cfg) in targets.iter() {
@@ -549,7 +567,9 @@ impl TargetCfgs {
549567
// code below extracts them from `--print=cfg`: make sure to only override fields that can
550568
// actually be changed with `-C` flags.
551569
for config in
552-
rustc_output(config, &["--print=cfg", "--target", &config.target]).trim().lines()
570+
rustc_output(config, &["--print=cfg", "--target", &config.target], Default::default())
571+
.trim()
572+
.lines()
553573
{
554574
let (name, value) = config
555575
.split_once("=\"")
@@ -628,11 +648,12 @@ pub enum Endian {
628648
Big,
629649
}
630650

631-
fn rustc_output(config: &Config, args: &[&str]) -> String {
651+
fn rustc_output(config: &Config, args: &[&str], envs: HashMap<String, String>) -> String {
632652
let mut command = Command::new(&config.rustc_path);
633653
add_dylib_path(&mut command, iter::once(&config.compile_lib_path));
634654
command.args(&config.target_rustcflags).args(args);
635655
command.env("RUSTC_BOOTSTRAP", "1");
656+
command.envs(envs);
636657

637658
let output = match command.output() {
638659
Ok(output) => output,

‎src/tools/miri/.github/workflows/ci.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,7 @@ jobs:
7373
cargo -V
7474
7575
- name: Test
76-
run: ./ci.sh
76+
run: ./ci/ci.sh
7777

7878
style:
7979
name: style checks
@@ -169,7 +169,7 @@ jobs:
169169
--message 'Dear @*T-miri*,
170170
171171
It would appear that the [Miri cron job build]('"https://github.com/$GITHUB_REPOSITORY/actions/runs/$GITHUB_RUN_ID"') failed.
172-
172+
173173
This likely means that rustc changed the miri directory and
174174
we now need to do a [`./miri rustc-pull`](https://github.com/rust-lang/miri/blob/master/CONTRIBUTING.md#importing-changes-from-the-rustc-repo).
175175
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
name: Tier 2 sysroots
2+
3+
on: push
4+
# schedule:
5+
# - cron: '44 4 * * *' # At 4:44 UTC every day.
6+
7+
defaults:
8+
run:
9+
shell: bash
10+
11+
jobs:
12+
sysroots:
13+
name: Build the sysroots
14+
runs-on: ubuntu-latest
15+
steps:
16+
- uses: actions/checkout@v3
17+
- name: Build the sysroots
18+
run: |
19+
cargo install -f rustup-toolchain-install-master
20+
./miri toolchain -c rust-docs # Docs are the only place targets are separated by tier
21+
./miri install
22+
python3 -m pip install beautifulsoup4
23+
./ci/build-all-targets.sh
24+
25+
sysroots-cron-fail-notify:
26+
name: sysroots cronjob failure notification
27+
runs-on: ubuntu-latest
28+
needs: [sysroots]
29+
if: failure() || cancelled()
30+
steps:
31+
# Send a Zulip notification
32+
- name: Install zulip-send
33+
run: pip3 install zulip
34+
- name: Send Zulip notification
35+
env:
36+
ZULIP_BOT_EMAIL: ${{ secrets.ZULIP_BOT_EMAIL }}
37+
ZULIP_API_TOKEN: ${{ secrets.ZULIP_API_TOKEN }}
38+
run: |
39+
~/.local/bin/zulip-send --user $ZULIP_BOT_EMAIL --api-key $ZULIP_API_TOKEN --site https://rust-lang.zulipchat.com \
40+
--stream miri --subject "Cron Job Failure (miri, $(date -u +%Y-%m))" \
41+
--message 'Dear @*T-miri*,
42+
43+
It would appear that the [Miri sysroots cron job build]('"https://github.com/$GITHUB_REPOSITORY/actions/runs/$GITHUB_RUN_ID"') failed.
44+
45+
Would you mind investigating this issue?
46+
47+
Thanks in advance!
48+
Sincerely,
49+
The Miri Cronjobs Bot'

‎src/tools/miri/Cargo.lock

+187-135
Large diffs are not rendered by default.

‎src/tools/miri/Cargo.toml

+8-5
Original file line numberDiff line numberDiff line change
@@ -24,10 +24,16 @@ log = "0.4"
2424
rand = "0.8"
2525
smallvec = "1.7"
2626
aes = { version = "0.8.3", features = ["hazmat"] }
27-
2827
measureme = "10.0.0"
2928
ctrlc = "3.2.5"
3029

30+
# Copied from `compiler/rustc/Cargo.toml`.
31+
# But only for some targets, it fails for others. Rustc configures this in its CI, but we can't
32+
# easily use that since we support of-tree builds.
33+
[target.'cfg(any(target_os = "linux", target_os = "macos"))'.dependencies.jemalloc-sys]
34+
version = "0.5.0"
35+
features = ['unprefixed_malloc_on_supported_platforms']
36+
3137
[target.'cfg(unix)'.dependencies]
3238
libc = "0.2"
3339

@@ -39,11 +45,8 @@ libloading = "0.8"
3945
colored = "2"
4046
ui_test = "0.21.1"
4147
rustc_version = "0.4"
42-
# Features chosen to match those required by env_logger, to avoid rebuilds
43-
regex = { version = "1.5.5", default-features = false, features = ["perf", "std"] }
48+
regex = "1.5.5"
4449
lazy_static = "1.4.0"
45-
# Require a version of serde without intransparent unreproducible binary blobs.
46-
serde = { version = "1.0.185", features = ["derive"] }
4750

4851
[package.metadata.rust-analyzer]
4952
# This crate uses #[feature(rustc_private)].

‎src/tools/miri/cargo-miri/Cargo.lock

+140-92
Large diffs are not rendered by default.
+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
#!/bin/bash
2+
3+
set -eu
4+
set -o pipefail
5+
6+
FAILS_DIR=failures
7+
8+
rm -rf $FAILS_DIR
9+
mkdir $FAILS_DIR
10+
11+
PLATFORM_SUPPORT_FILE=$(rustc +miri --print sysroot)/share/doc/rust/html/rustc/platform-support.html
12+
13+
for target in $(python3 ci/scrape-targets.py $PLATFORM_SUPPORT_FILE); do
14+
# Wipe the cache before every build to minimize disk usage
15+
rm -rf ~/.cache/miri
16+
if cargo +miri miri setup --target $target 2>&1 | tee failures/$target; then
17+
# If the build succeeds, delete its output. If we have output, a build failed.
18+
rm $FAILS_DIR/$target
19+
fi
20+
done
21+
22+
# If the sysroot for any target fails to build, we will have a file in FAILS_DIR.
23+
if [[ $(ls failures | wc -l) -ne 0 ]]; then
24+
echo "Sysroots for the following targets failed to build:"
25+
ls $FAILS_DIR
26+
exit 1
27+
fi
File renamed without changes.

‎src/tools/miri/ci/scrape-targets.py

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
import sys
2+
from bs4 import BeautifulSoup
3+
4+
html = open(sys.argv[1], 'r').read()
5+
soup = BeautifulSoup(html, features="html.parser")
6+
# The tables are:
7+
# Tier 1 <-- this is already checked by main CI, so we ignore it here
8+
# Tier 2 with host tools <-- we want this one
9+
# Tier 2 without host tools <-- and also this
10+
# Tier 3
11+
for table in soup.find_all("table")[1:3]:
12+
for row in table.find_all('tr'):
13+
code = row.find('code')
14+
if code is not None:
15+
print(code.text)

‎src/tools/miri/miri-script/Cargo.lock

+125-68
Original file line numberDiff line numberDiff line change

‎src/tools/miri/miri-script/src/util.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -111,9 +111,11 @@ impl MiriEnv {
111111
) -> Result<()> {
112112
let MiriEnv { toolchain, cargo_extra_flags, .. } = self;
113113
let quiet_flag = if quiet { Some("--quiet") } else { None };
114+
// We build the tests as well, (a) to avoid having rebuilds when building the tests later
115+
// and (b) to have more parallelism during the build of Miri and its tests.
114116
let mut cmd = cmd!(
115117
self.sh,
116-
"cargo +{toolchain} build {cargo_extra_flags...} --manifest-path {manifest_path} {quiet_flag...} {args...}"
118+
"cargo +{toolchain} build --bins --tests {cargo_extra_flags...} --manifest-path {manifest_path} {quiet_flag...} {args...}"
117119
);
118120
cmd.set_quiet(quiet);
119121
cmd.run()?;

‎src/tools/miri/rust-version

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
2271c26e4a8e062bb00d709d0ccb5846e0c341b9
1+
5bcd86d89b2b7b6a490f7e075dd4eb346deb5f98

‎src/tools/miri/src/bin/miri.rs

+39-1
Original file line numberDiff line numberDiff line change
@@ -293,13 +293,51 @@ fn run_compiler(
293293
}
294294

295295
/// Parses a comma separated list of `T` from the given string:
296-
///
297296
/// `<value1>,<value2>,<value3>,...`
298297
fn parse_comma_list<T: FromStr>(input: &str) -> Result<Vec<T>, T::Err> {
299298
input.split(',').map(str::parse::<T>).collect()
300299
}
301300

301+
#[cfg(any(target_os = "linux", target_os = "macos"))]
302+
fn jemalloc_magic() {
303+
// These magic runes are copied from
304+
// <https://github.com/rust-lang/rust/blob/e89bd9428f621545c979c0ec686addc6563a394e/compiler/rustc/src/main.rs#L39>.
305+
// See there for further comments.
306+
use std::os::raw::{c_int, c_void};
307+
308+
#[used]
309+
static _F1: unsafe extern "C" fn(usize, usize) -> *mut c_void = jemalloc_sys::calloc;
310+
#[used]
311+
static _F2: unsafe extern "C" fn(*mut *mut c_void, usize, usize) -> c_int =
312+
jemalloc_sys::posix_memalign;
313+
#[used]
314+
static _F3: unsafe extern "C" fn(usize, usize) -> *mut c_void = jemalloc_sys::aligned_alloc;
315+
#[used]
316+
static _F4: unsafe extern "C" fn(usize) -> *mut c_void = jemalloc_sys::malloc;
317+
#[used]
318+
static _F5: unsafe extern "C" fn(*mut c_void, usize) -> *mut c_void = jemalloc_sys::realloc;
319+
#[used]
320+
static _F6: unsafe extern "C" fn(*mut c_void) = jemalloc_sys::free;
321+
322+
// On OSX, jemalloc doesn't directly override malloc/free, but instead
323+
// registers itself with the allocator's zone APIs in a ctor. However,
324+
// the linker doesn't seem to consider ctors as "used" when statically
325+
// linking, so we need to explicitly depend on the function.
326+
#[cfg(target_os = "macos")]
327+
{
328+
extern "C" {
329+
fn _rjem_je_zone_register();
330+
}
331+
332+
#[used]
333+
static _F7: unsafe extern "C" fn() = _rjem_je_zone_register;
334+
}
335+
}
336+
302337
fn main() {
338+
#[cfg(any(target_os = "linux", target_os = "macos"))]
339+
jemalloc_magic();
340+
303341
let early_dcx = EarlyDiagCtxt::new(ErrorOutputType::default());
304342

305343
// Snapshot a copy of the environment before `rustc` starts messing with it.

‎src/tools/miri/src/borrow_tracker/tree_borrows/tree.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -612,7 +612,7 @@ impl<'tcx> Tree {
612612
is_foreign: rel_pos.is_foreign(),
613613
access_cause,
614614
access_range,
615-
transition_range: perms_range.clone(),
615+
transition_range: perms_range,
616616
span,
617617
});
618618
}

‎src/tools/miri/src/shims/intrinsics/mod.rs

+30-22
Original file line numberDiff line numberDiff line change
@@ -145,17 +145,30 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriInterpCxExt<'mir, 'tcx> {
145145
"fabsf32" => {
146146
let [f] = check_arg_count(args)?;
147147
let f = this.read_scalar(f)?.to_f32()?;
148-
// Can be implemented in soft-floats.
149148
// This is a "bitwise" operation, so there's no NaN non-determinism.
150149
this.write_scalar(Scalar::from_f32(f.abs()), dest)?;
151150
}
152151
"fabsf64" => {
153152
let [f] = check_arg_count(args)?;
154153
let f = this.read_scalar(f)?.to_f64()?;
155-
// Can be implemented in soft-floats.
156154
// This is a "bitwise" operation, so there's no NaN non-determinism.
157155
this.write_scalar(Scalar::from_f64(f.abs()), dest)?;
158156
}
157+
"floorf32" | "ceilf32" | "truncf32" | "roundf32" | "rintf32" => {
158+
let [f] = check_arg_count(args)?;
159+
let f = this.read_scalar(f)?.to_f32()?;
160+
let mode = match intrinsic_name {
161+
"floorf32" => Round::TowardNegative,
162+
"ceilf32" => Round::TowardPositive,
163+
"truncf32" => Round::TowardZero,
164+
"roundf32" => Round::NearestTiesToAway,
165+
"rintf32" => Round::NearestTiesToEven,
166+
_ => bug!(),
167+
};
168+
let res = f.round_to_integral(mode).value;
169+
let res = this.adjust_nan(res, &[f]);
170+
this.write_scalar(res, dest)?;
171+
}
159172
#[rustfmt::skip]
160173
| "sinf32"
161174
| "cosf32"
@@ -165,11 +178,6 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriInterpCxExt<'mir, 'tcx> {
165178
| "logf32"
166179
| "log10f32"
167180
| "log2f32"
168-
| "floorf32"
169-
| "ceilf32"
170-
| "truncf32"
171-
| "roundf32"
172-
| "rintf32"
173181
=> {
174182
let [f] = check_arg_count(args)?;
175183
let f = this.read_scalar(f)?.to_f32()?;
@@ -184,18 +192,28 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriInterpCxExt<'mir, 'tcx> {
184192
"logf32" => f_host.ln(),
185193
"log10f32" => f_host.log10(),
186194
"log2f32" => f_host.log2(),
187-
"floorf32" => f_host.floor(),
188-
"ceilf32" => f_host.ceil(),
189-
"truncf32" => f_host.trunc(),
190-
"roundf32" => f_host.round(),
191-
"rintf32" => f_host.round_ties_even(),
192195
_ => bug!(),
193196
};
194197
let res = res.to_soft();
195198
let res = this.adjust_nan(res, &[f]);
196199
this.write_scalar(res, dest)?;
197200
}
198201

202+
"floorf64" | "ceilf64" | "truncf64" | "roundf64" | "rintf64" => {
203+
let [f] = check_arg_count(args)?;
204+
let f = this.read_scalar(f)?.to_f64()?;
205+
let mode = match intrinsic_name {
206+
"floorf64" => Round::TowardNegative,
207+
"ceilf64" => Round::TowardPositive,
208+
"truncf64" => Round::TowardZero,
209+
"roundf64" => Round::NearestTiesToAway,
210+
"rintf64" => Round::NearestTiesToEven,
211+
_ => bug!(),
212+
};
213+
let res = f.round_to_integral(mode).value;
214+
let res = this.adjust_nan(res, &[f]);
215+
this.write_scalar(res, dest)?;
216+
}
199217
#[rustfmt::skip]
200218
| "sinf64"
201219
| "cosf64"
@@ -205,11 +223,6 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriInterpCxExt<'mir, 'tcx> {
205223
| "logf64"
206224
| "log10f64"
207225
| "log2f64"
208-
| "floorf64"
209-
| "ceilf64"
210-
| "truncf64"
211-
| "roundf64"
212-
| "rintf64"
213226
=> {
214227
let [f] = check_arg_count(args)?;
215228
let f = this.read_scalar(f)?.to_f64()?;
@@ -224,11 +237,6 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriInterpCxExt<'mir, 'tcx> {
224237
"logf64" => f_host.ln(),
225238
"log10f64" => f_host.log10(),
226239
"log2f64" => f_host.log2(),
227-
"floorf64" => f_host.floor(),
228-
"ceilf64" => f_host.ceil(),
229-
"truncf64" => f_host.trunc(),
230-
"roundf64" => f_host.round(),
231-
"rintf64" => f_host.round_ties_even(),
232240
_ => bug!(),
233241
};
234242
let res = res.to_soft();

‎src/tools/miri/test_dependencies/Cargo.lock

+54-63
Original file line numberDiff line numberDiff line change

‎src/tools/miri/tests/fail/function_pointers/abi_mismatch_array_vs_struct.rs

+6-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,12 @@
33
// Some targets treat arrays and structs very differently. We would probably catch that on those
44
// targets since we check the `PassMode`; here we ensure that we catch it on *all* targets
55
// (in particular, on x86-64 the pass mode is `Indirect` for both of these).
6-
struct S(#[allow(dead_code)] i32, #[allow(dead_code)] i32, #[allow(dead_code)] i32, #[allow(dead_code)] i32);
6+
struct S(
7+
#[allow(dead_code)] i32,
8+
#[allow(dead_code)] i32,
9+
#[allow(dead_code)] i32,
10+
#[allow(dead_code)] i32,
11+
);
712
type A = [i32; 4];
813

914
fn main() {

‎src/tools/miri/tests/pass/float.rs

+161-2
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,18 @@
11
#![feature(stmt_expr_attributes)]
22
#![feature(round_ties_even)]
3+
#![feature(float_gamma)]
34
#![allow(arithmetic_overflow)]
5+
46
use std::fmt::Debug;
57
use std::hint::black_box;
8+
use std::{f32, f64};
9+
10+
macro_rules! assert_approx_eq {
11+
($a:expr, $b:expr) => {{
12+
let (a, b) = (&$a, &$b);
13+
assert!((*a - *b).abs() < 1.0e-6, "{} is not approximately equal to {}", *a, *b);
14+
}};
15+
}
616

717
fn main() {
818
basic();
@@ -11,6 +21,8 @@ fn main() {
1121
ops();
1222
nan_casts();
1323
rounding();
24+
mul_add();
25+
libm();
1426
}
1527

1628
// Helper function to avoid promotion so that this tests "run-time" casts, not CTFE.
@@ -148,8 +160,6 @@ fn basic() {
148160
assert_ne!({ 5.0_f32 / 0.0 }, { -5.0_f32 / 0.0 });
149161
assert!((5.0_f64 / 0.0).is_infinite());
150162
assert_ne!({ 5.0_f64 / 0.0 }, { 5.0_f64 / -0.0 });
151-
assert!((-5.0_f32).sqrt().is_nan());
152-
assert!((-5.0_f64).sqrt().is_nan());
153163
assert_ne!(f32::NAN, f32::NAN);
154164
assert_ne!(f64::NAN, f64::NAN);
155165
// negative zero
@@ -178,6 +188,9 @@ fn basic() {
178188
assert!((black_box(1.0f64) % -1.0).is_sign_positive());
179189
assert!((black_box(-1.0f64) % 1.0).is_sign_negative());
180190
assert!((black_box(-1.0f64) % -1.0).is_sign_negative());
191+
192+
assert_eq!((-1.0f32).abs(), 1.0f32);
193+
assert_eq!(34.2f64.abs(), 34.2f64);
181194
}
182195

183196
/// Many of these test values are taken from
@@ -592,4 +605,150 @@ fn rounding() {
592605
assert_eq((-1.3f64).round_ties_even(), -1.0f64);
593606
assert_eq((-1.5f64).round_ties_even(), -2.0f64);
594607
assert_eq((-1.7f64).round_ties_even(), -2.0f64);
608+
609+
assert_eq!(3.8f32.floor(), 3.0f32);
610+
assert_eq!((-1.1f64).floor(), -2.0f64);
611+
612+
assert_eq!((-2.3f32).ceil(), -2.0f32);
613+
assert_eq!(3.8f64.ceil(), 4.0f64);
614+
615+
assert_eq!(0.1f32.trunc(), 0.0f32);
616+
assert_eq!((-0.1f64).trunc(), 0.0f64);
617+
618+
assert_eq!(3.3_f32.round(), 3.0);
619+
assert_eq!(2.5_f32.round(), 3.0);
620+
assert_eq!(3.9_f64.round(), 4.0);
621+
assert_eq!(2.5_f64.round(), 3.0);
622+
}
623+
624+
fn mul_add() {
625+
assert_eq!(3.0f32.mul_add(2.0f32, 5.0f32), 11.0);
626+
assert_eq!(0.0f32.mul_add(-2.0, f32::consts::E), f32::consts::E);
627+
assert_eq!(3.0f64.mul_add(2.0, 5.0), 11.0);
628+
assert_eq!(0.0f64.mul_add(-2.0f64, f64::consts::E), f64::consts::E);
629+
assert_eq!((-3.2f32).mul_add(2.4, f32::NEG_INFINITY), f32::NEG_INFINITY);
630+
assert_eq!((-3.2f64).mul_add(2.4, f64::NEG_INFINITY), f64::NEG_INFINITY);
631+
632+
let f = f32::mul_add(
633+
-0.000000000000000000000000000000000000014728589,
634+
0.0000037105144,
635+
0.000000000000000000000000000000000000000000055,
636+
);
637+
assert_eq!(f.to_bits(), f32::to_bits(-0.0));
638+
}
639+
640+
pub fn libm() {
641+
fn ldexp(a: f64, b: i32) -> f64 {
642+
extern "C" {
643+
fn ldexp(x: f64, n: i32) -> f64;
644+
}
645+
unsafe { ldexp(a, b) }
646+
}
647+
648+
assert_approx_eq!(64f32.sqrt(), 8f32);
649+
assert_approx_eq!(64f64.sqrt(), 8f64);
650+
assert!((-5.0_f32).sqrt().is_nan());
651+
assert!((-5.0_f64).sqrt().is_nan());
652+
653+
assert_approx_eq!(25f32.powi(-2), 0.0016f32);
654+
assert_approx_eq!(23.2f64.powi(2), 538.24f64);
655+
656+
assert_approx_eq!(25f32.powf(-2f32), 0.0016f32);
657+
assert_approx_eq!(400f64.powf(0.5f64), 20f64);
658+
659+
assert_approx_eq!(1f32.exp(), f32::consts::E);
660+
assert_approx_eq!(1f64.exp(), f64::consts::E);
661+
662+
assert_approx_eq!(1f32.exp_m1(), f32::consts::E - 1.0);
663+
assert_approx_eq!(1f64.exp_m1(), f64::consts::E - 1.0);
664+
665+
assert_approx_eq!(10f32.exp2(), 1024f32);
666+
assert_approx_eq!(50f64.exp2(), 1125899906842624f64);
667+
668+
assert_approx_eq!(f32::consts::E.ln(), 1f32);
669+
assert_approx_eq!(1f64.ln(), 0f64);
670+
671+
assert_approx_eq!(0f32.ln_1p(), 0f32);
672+
assert_approx_eq!(0f64.ln_1p(), 0f64);
673+
674+
assert_approx_eq!(10f32.log10(), 1f32);
675+
assert_approx_eq!(f64::consts::E.log10(), f64::consts::LOG10_E);
676+
677+
assert_approx_eq!(8f32.log2(), 3f32);
678+
assert_approx_eq!(f64::consts::E.log2(), f64::consts::LOG2_E);
679+
680+
#[allow(deprecated)]
681+
{
682+
assert_approx_eq!(5.0f32.abs_sub(3.0), 2.0);
683+
assert_approx_eq!(3.0f64.abs_sub(5.0), 0.0);
684+
}
685+
686+
assert_approx_eq!(27.0f32.cbrt(), 3.0f32);
687+
assert_approx_eq!(27.0f64.cbrt(), 3.0f64);
688+
689+
assert_approx_eq!(3.0f32.hypot(4.0f32), 5.0f32);
690+
assert_approx_eq!(3.0f64.hypot(4.0f64), 5.0f64);
691+
692+
assert_eq!(ldexp(0.65f64, 3i32), 5.2f64);
693+
assert_eq!(ldexp(1.42, 0xFFFF), f64::INFINITY);
694+
assert_eq!(ldexp(1.42, -0xFFFF), 0f64);
695+
696+
// Trigonometric functions.
697+
698+
assert_approx_eq!(0f32.sin(), 0f32);
699+
assert_approx_eq!((f64::consts::PI / 2f64).sin(), 1f64);
700+
assert_approx_eq!(f32::consts::FRAC_PI_6.sin(), 0.5);
701+
assert_approx_eq!(f64::consts::FRAC_PI_6.sin(), 0.5);
702+
assert_approx_eq!(f32::consts::FRAC_PI_4.sin().asin(), f32::consts::FRAC_PI_4);
703+
assert_approx_eq!(f64::consts::FRAC_PI_4.sin().asin(), f64::consts::FRAC_PI_4);
704+
705+
assert_approx_eq!(1.0f32.sinh(), 1.1752012f32);
706+
assert_approx_eq!(1.0f64.sinh(), 1.1752012f64);
707+
assert_approx_eq!(2.0f32.asinh(), 1.443635475178810342493276740273105f32);
708+
assert_approx_eq!((-2.0f64).asinh(), -1.443635475178810342493276740273105f64);
709+
710+
assert_approx_eq!(0f32.cos(), 1f32);
711+
assert_approx_eq!((f64::consts::PI * 2f64).cos(), 1f64);
712+
assert_approx_eq!(f32::consts::FRAC_PI_3.cos(), 0.5);
713+
assert_approx_eq!(f64::consts::FRAC_PI_3.cos(), 0.5);
714+
assert_approx_eq!(f32::consts::FRAC_PI_4.cos().acos(), f32::consts::FRAC_PI_4);
715+
assert_approx_eq!(f64::consts::FRAC_PI_4.cos().acos(), f64::consts::FRAC_PI_4);
716+
717+
assert_approx_eq!(1.0f32.cosh(), 1.54308f32);
718+
assert_approx_eq!(1.0f64.cosh(), 1.54308f64);
719+
assert_approx_eq!(2.0f32.acosh(), 1.31695789692481670862504634730796844f32);
720+
assert_approx_eq!(3.0f64.acosh(), 1.76274717403908605046521864995958461f64);
721+
722+
assert_approx_eq!(1.0f32.tan(), 1.557408f32);
723+
assert_approx_eq!(1.0f64.tan(), 1.557408f64);
724+
assert_approx_eq!(1.0_f32, 1.0_f32.tan().atan());
725+
assert_approx_eq!(1.0_f64, 1.0_f64.tan().atan());
726+
assert_approx_eq!(1.0f32.atan2(2.0f32), 0.46364761f32);
727+
assert_approx_eq!(1.0f32.atan2(2.0f32), 0.46364761f32);
728+
729+
assert_approx_eq!(
730+
1.0f32.tanh(),
731+
(1.0 - f32::consts::E.powi(-2)) / (1.0 + f32::consts::E.powi(-2))
732+
);
733+
assert_approx_eq!(
734+
1.0f64.tanh(),
735+
(1.0 - f64::consts::E.powi(-2)) / (1.0 + f64::consts::E.powi(-2))
736+
);
737+
assert_approx_eq!(0.5f32.atanh(), 0.54930614433405484569762261846126285f32);
738+
assert_approx_eq!(0.5f64.atanh(), 0.54930614433405484569762261846126285f64);
739+
740+
assert_approx_eq!(5.0f32.gamma(), 24.0);
741+
assert_approx_eq!(5.0f64.gamma(), 24.0);
742+
assert_approx_eq!((-0.5f32).gamma(), (-2.0) * f32::consts::PI.sqrt());
743+
assert_approx_eq!((-0.5f64).gamma(), (-2.0) * f64::consts::PI.sqrt());
744+
745+
assert_eq!(2.0f32.ln_gamma(), (0.0, 1));
746+
assert_eq!(2.0f64.ln_gamma(), (0.0, 1));
747+
// Gamma(-0.5) = -2*sqrt(π)
748+
let (val, sign) = (-0.5f32).ln_gamma();
749+
assert_approx_eq!(val, (2.0 * f32::consts::PI.sqrt()).ln());
750+
assert_eq!(sign, -1);
751+
let (val, sign) = (-0.5f64).ln_gamma();
752+
assert_approx_eq!(val, (2.0 * f64::consts::PI.sqrt()).ln());
753+
assert_eq!(sign, -1);
595754
}

‎src/tools/miri/tests/pass/float_nan.rs

+8
Original file line numberDiff line numberDiff line change
@@ -264,6 +264,10 @@ fn test_f32() {
264264
HashSet::from_iter([F32::nan(Pos, Quiet, 0), F32::nan(Neg, Quiet, 0)]),
265265
|| F32::from(f32::min(nan, nan)),
266266
);
267+
check_all_outcomes(
268+
HashSet::from_iter([F32::nan(Pos, Quiet, 0), F32::nan(Neg, Quiet, 0)]),
269+
|| F32::from(nan.floor()),
270+
);
267271
check_all_outcomes(
268272
HashSet::from_iter([F32::nan(Pos, Quiet, 0), F32::nan(Neg, Quiet, 0)]),
269273
|| F32::from(nan.sin()),
@@ -376,6 +380,10 @@ fn test_f64() {
376380
HashSet::from_iter([F64::nan(Pos, Quiet, 0), F64::nan(Neg, Quiet, 0)]),
377381
|| F64::from(f64::min(nan, nan)),
378382
);
383+
check_all_outcomes(
384+
HashSet::from_iter([F64::nan(Pos, Quiet, 0), F64::nan(Neg, Quiet, 0)]),
385+
|| F64::from(nan.floor()),
386+
);
379387
check_all_outcomes(
380388
HashSet::from_iter([F64::nan(Pos, Quiet, 0), F64::nan(Neg, Quiet, 0)]),
381389
|| F64::from(nan.sin()),

‎src/tools/miri/tests/pass/intrinsics-math.rs

-156
This file was deleted.

‎src/tools/miri/tests/pass/stacked-borrows/non_scalar_field_retagging.rs

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,10 @@
11
//@compile-flags: -Zmiri-retag-fields=scalar
22

3-
struct Newtype<'a>(#[allow(dead_code)] &'a mut i32, #[allow(dead_code)] i32, #[allow(dead_code)] i32);
3+
struct Newtype<'a>(
4+
#[allow(dead_code)] &'a mut i32,
5+
#[allow(dead_code)] i32,
6+
#[allow(dead_code)] i32,
7+
);
48

59
fn dealloc_while_running(_n: Newtype<'_>, dealloc: impl FnOnce()) {
610
dealloc();

‎tests/ui/coherence/warn-when-cycle-is-error-in-coherence.rs

+2-5
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,15 @@
1-
#![deny(coinductive_overlap_in_coherence)]
2-
31
use std::borrow::Borrow;
42
use std::cmp::Ordering;
53
use std::marker::PhantomData;
64

75
#[derive(PartialEq, Default)]
6+
//~^ ERROR conflicting implementations of trait `PartialEq<Interval<_>>` for type `Interval<_>`
87
pub(crate) struct Interval<T>(PhantomData<T>);
98

109
// This impl overlaps with the `derive` unless we reject the nested
1110
// `Interval<?1>: PartialOrd<Interval<?1>>` candidate which results
12-
// in a - currently inductive - cycle.
11+
// in a -- currently inductive -- cycle.
1312
impl<T, Q> PartialEq<Q> for Interval<T>
14-
//~^ ERROR implementations of `PartialEq<Interval<_>>` for `Interval<_>` will conflict in the future
15-
//~| WARN this was previously accepted by the compiler but is being phased out
1613
where
1714
T: Borrow<Q>,
1815
Q: ?Sized + PartialOrd,
Original file line numberDiff line numberDiff line change
@@ -1,51 +1,17 @@
1-
error: implementations of `PartialEq<Interval<_>>` for `Interval<_>` will conflict in the future
2-
--> $DIR/warn-when-cycle-is-error-in-coherence.rs:13:1
1+
error[E0119]: conflicting implementations of trait `PartialEq<Interval<_>>` for type `Interval<_>`
2+
--> $DIR/warn-when-cycle-is-error-in-coherence.rs:5:10
33
|
44
LL | #[derive(PartialEq, Default)]
5-
| --------- the second impl is here
5+
| ^^^^^^^^^ conflicting implementation for `Interval<_>`
66
...
77
LL | / impl<T, Q> PartialEq<Q> for Interval<T>
8-
LL | |
9-
LL | |
108
LL | | where
119
LL | | T: Borrow<Q>,
1210
LL | | Q: ?Sized + PartialOrd,
13-
| |___________________________^ the first impl is here
11+
| |___________________________- first implementation here
1412
|
15-
= warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in a future release!
16-
= note: for more information, see issue #114040 <https://github.com/rust-lang/rust/issues/114040>
17-
= note: impls that are not considered to overlap may be considered to overlap in the future
18-
= note: `Interval<_>: PartialOrd` may be considered to hold in future releases, causing the impls to overlap
19-
note: the lint level is defined here
20-
--> $DIR/warn-when-cycle-is-error-in-coherence.rs:1:9
21-
|
22-
LL | #![deny(coinductive_overlap_in_coherence)]
23-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
13+
= note: this error originates in the derive macro `PartialEq` (in Nightly builds, run with -Z macro-backtrace for more info)
2414

2515
error: aborting due to 1 previous error
2616

27-
Future incompatibility report: Future breakage diagnostic:
28-
error: implementations of `PartialEq<Interval<_>>` for `Interval<_>` will conflict in the future
29-
--> $DIR/warn-when-cycle-is-error-in-coherence.rs:13:1
30-
|
31-
LL | #[derive(PartialEq, Default)]
32-
| --------- the second impl is here
33-
...
34-
LL | / impl<T, Q> PartialEq<Q> for Interval<T>
35-
LL | |
36-
LL | |
37-
LL | | where
38-
LL | | T: Borrow<Q>,
39-
LL | | Q: ?Sized + PartialOrd,
40-
| |___________________________^ the first impl is here
41-
|
42-
= warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in a future release!
43-
= note: for more information, see issue #114040 <https://github.com/rust-lang/rust/issues/114040>
44-
= note: impls that are not considered to overlap may be considered to overlap in the future
45-
= note: `Interval<_>: PartialOrd` may be considered to hold in future releases, causing the impls to overlap
46-
note: the lint level is defined here
47-
--> $DIR/warn-when-cycle-is-error-in-coherence.rs:1:9
48-
|
49-
LL | #![deny(coinductive_overlap_in_coherence)]
50-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
51-
17+
For more information about this error, try `rustc --explain E0119`.

‎tests/ui/fmt/format-string-wrong-order.rs

+6
Original file line numberDiff line numberDiff line change
@@ -12,4 +12,10 @@ fn main() {
1212
//~^ ERROR invalid format string: expected `'}'`, found `'?'`
1313
format!("{?:#?}", bar);
1414
//~^ ERROR invalid format string: expected format parameter to occur after `:`
15+
format!("Hello {<5:}!", "x");
16+
//~^ ERROR invalid format string: expected format parameter to occur after `:`
17+
format!("Hello {^5:}!", "x");
18+
//~^ ERROR invalid format string: expected format parameter to occur after `:`
19+
format!("Hello {>5:}!", "x");
20+
//~^ ERROR invalid format string: expected format parameter to occur after `:`
1521
}

‎tests/ui/fmt/format-string-wrong-order.stderr

+19-1
Original file line numberDiff line numberDiff line change
@@ -50,5 +50,23 @@ LL | format!("{?:#?}", bar);
5050
|
5151
= note: `?` comes after `:`, try `:?` instead
5252

53-
error: aborting due to 6 previous errors
53+
error: invalid format string: expected format parameter to occur after `:`
54+
--> $DIR/format-string-wrong-order.rs:15:21
55+
|
56+
LL | format!("Hello {<5:}!", "x");
57+
| ^ expected `<` to occur after `:` in format string
58+
59+
error: invalid format string: expected format parameter to occur after `:`
60+
--> $DIR/format-string-wrong-order.rs:17:21
61+
|
62+
LL | format!("Hello {^5:}!", "x");
63+
| ^ expected `^` to occur after `:` in format string
64+
65+
error: invalid format string: expected format parameter to occur after `:`
66+
--> $DIR/format-string-wrong-order.rs:19:21
67+
|
68+
LL | format!("Hello {>5:}!", "x");
69+
| ^ expected `>` to occur after `:` in format string
70+
71+
error: aborting due to 9 previous errors
5472

0 commit comments

Comments
 (0)
Please sign in to comment.