Skip to content

Commit ed68857

Browse files
added unchecked_rem for signed and unsigned types
fixed unused variables in unchecked_rem's precondition macro
1 parent c0b3ba1 commit ed68857

File tree

2 files changed

+68
-0
lines changed

2 files changed

+68
-0
lines changed

library/core/src/num/int_macros.rs

+35
Original file line numberDiff line numberDiff line change
@@ -1084,6 +1084,41 @@ macro_rules! int_impl {
10841084
}
10851085
}
10861086

1087+
/// Unchecked integer remainder. Computes `self % rhs`, assuming `rhs != 0`
1088+
/// or overflow cannot occur.
1089+
///
1090+
/// # Safety
1091+
///
1092+
/// This results in undefined behavior when
1093+
#[doc = concat!("`rhs == 0` or (`self ==", stringify!($SelfT), "::MIN` and `rhs == -1`)")]
1094+
/// i.e. when [`checked_rem`] would return `None`.
1095+
///
1096+
#[doc = concat!("[`checked_rem`]: ", stringify!($SelfT), "::checked_rem")]
1097+
#[unstable(
1098+
feature = "unchecked_div_rem",
1099+
reason = "consistency with other unchecked_* functions",
1100+
issue = "136716",
1101+
)]
1102+
#[must_use = "this returns the result of the operation, \
1103+
without modifying the original"]
1104+
#[inline(always)]
1105+
#[cfg_attr(miri, track_caller)] // even without panics, this helps for Miri backtraces
1106+
pub const unsafe fn unchecked_rem(self, rhs: Self) -> Self {
1107+
assert_unsafe_precondition!(
1108+
check_language_ub,
1109+
concat!(stringify!($SelfT), "::unchecked_rem cannot overflow or accept rhs as 0"),
1110+
(
1111+
lhs: $SelfT = self,
1112+
rhs: $SelfT = rhs
1113+
) => !lhs.overflowing_rem(rhs).1 || !(rhs == 0),
1114+
);
1115+
1116+
// SAFETY: this is guaranteed to be safe by the caller.
1117+
unsafe {
1118+
intrinsics::unchecked_rem(self, rhs)
1119+
}
1120+
}
1121+
10871122
/// Strict integer remainder. Computes `self % rhs`, panicking if
10881123
/// the division results in overflow.
10891124
///

library/core/src/num/uint_macros.rs

+33
Original file line numberDiff line numberDiff line change
@@ -1175,6 +1175,39 @@ macro_rules! uint_impl {
11751175
}
11761176
}
11771177

1178+
/// Unchecked integer remainder. Computes `self % rhs`, assuming `rhs != 0`
1179+
///
1180+
/// # Safety
1181+
///
1182+
/// This results in undefined behavior when
1183+
#[doc = concat!("`rhs == 0`")]
1184+
/// i.e. when [`checked_rem`] would return `None`.
1185+
///
1186+
#[doc = concat!("[`checked_rem`]: ", stringify!($SelfT), "::checked_rem")]
1187+
#[unstable(
1188+
feature = "unchecked_div_rem",
1189+
reason = "consistency with other unchecked_* functions",
1190+
issue = "136716",
1191+
)]
1192+
#[must_use = "this returns the result of the operation, \
1193+
without modifying the original"]
1194+
#[inline(always)]
1195+
#[cfg_attr(miri, track_caller)] // even without panics, this helps for Miri backtraces
1196+
pub const unsafe fn unchecked_rem(self, rhs: Self) -> Self {
1197+
assert_unsafe_precondition!(
1198+
check_language_ub,
1199+
concat!(stringify!($SelfT), "::unchecked_rem cannot accept rhs as 0"),
1200+
(
1201+
rhs: $SelfT = rhs
1202+
) => !(rhs == 0),
1203+
);
1204+
1205+
// SAFETY: this is guaranteed to be safe by the caller.
1206+
unsafe {
1207+
intrinsics::unchecked_rem(self, rhs)
1208+
}
1209+
}
1210+
11781211
/// Strict integer remainder. Computes `self % rhs`.
11791212
///
11801213
/// Strict remainder calculation on unsigned types is just the regular

0 commit comments

Comments
 (0)