175 lines
3.3 KiB
Rust
175 lines
3.3 KiB
Rust
//@ check-pass
|
|
// https://github.com/rust-lang/rust/issues/52873
|
|
#![crate_name="foo"]
|
|
|
|
// Regression test for #52873. We used to ICE due to unexpected
|
|
// overflows when checking for "blanket impl inclusion".
|
|
|
|
use std::marker::PhantomData;
|
|
use std::cmp::Ordering;
|
|
use std::ops::{Add, Mul};
|
|
|
|
pub type True = B1;
|
|
pub type False = B0;
|
|
pub type U0 = UTerm;
|
|
pub type U1 = UInt<UTerm, B1>;
|
|
|
|
pub trait NonZero {}
|
|
|
|
pub trait Bit {
|
|
}
|
|
|
|
pub trait Unsigned {
|
|
}
|
|
|
|
#[derive(Eq, PartialEq, Ord, PartialOrd, Clone, Copy, Hash, Debug, Default)]
|
|
pub struct B0;
|
|
|
|
impl B0 {
|
|
#[inline]
|
|
pub fn new() -> B0 {
|
|
B0
|
|
}
|
|
}
|
|
|
|
#[derive(Eq, PartialEq, Ord, PartialOrd, Clone, Copy, Hash, Debug, Default)]
|
|
pub struct B1;
|
|
|
|
impl B1 {
|
|
#[inline]
|
|
pub fn new() -> B1 {
|
|
B1
|
|
}
|
|
}
|
|
|
|
impl Bit for B0 {
|
|
}
|
|
|
|
impl Bit for B1 {
|
|
}
|
|
|
|
impl NonZero for B1 {}
|
|
|
|
pub trait PrivatePow<Y, N> {
|
|
type Output;
|
|
}
|
|
pub type PrivatePowOut<A, Y, N> = <A as PrivatePow<Y, N>>::Output;
|
|
|
|
pub type Add1<A> = <A as Add<::B1>>::Output;
|
|
pub type Prod<A, B> = <A as Mul<B>>::Output;
|
|
pub type Square<A> = <A as Mul>::Output;
|
|
pub type Sum<A, B> = <A as Add<B>>::Output;
|
|
|
|
#[derive(Eq, PartialEq, Ord, PartialOrd, Clone, Copy, Hash, Debug, Default)]
|
|
pub struct UTerm;
|
|
|
|
impl UTerm {
|
|
#[inline]
|
|
pub fn new() -> UTerm {
|
|
UTerm
|
|
}
|
|
}
|
|
|
|
impl Unsigned for UTerm {
|
|
}
|
|
|
|
#[derive(Eq, PartialEq, Ord, PartialOrd, Clone, Copy, Hash, Debug, Default)]
|
|
pub struct UInt<U, B> {
|
|
_marker: PhantomData<(U, B)>,
|
|
}
|
|
|
|
impl<U: Unsigned, B: Bit> UInt<U, B> {
|
|
#[inline]
|
|
pub fn new() -> UInt<U, B> {
|
|
UInt {
|
|
_marker: PhantomData,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<U: Unsigned, B: Bit> Unsigned for UInt<U, B> {
|
|
}
|
|
|
|
impl<U: Unsigned, B: Bit> NonZero for UInt<U, B> {}
|
|
|
|
impl Add<B0> for UTerm {
|
|
type Output = UTerm;
|
|
fn add(self, _: B0) -> Self::Output {
|
|
UTerm
|
|
}
|
|
}
|
|
|
|
impl<U: Unsigned, B: Bit> Add<B0> for UInt<U, B> {
|
|
type Output = UInt<U, B>;
|
|
fn add(self, _: B0) -> Self::Output {
|
|
UInt::new()
|
|
}
|
|
}
|
|
|
|
impl<U: Unsigned> Add<U> for UTerm {
|
|
type Output = U;
|
|
fn add(self, _: U) -> Self::Output {
|
|
unimplemented!()
|
|
}
|
|
}
|
|
|
|
impl<U: Unsigned, B: Bit> Mul<B0> for UInt<U, B> {
|
|
type Output = UTerm;
|
|
fn mul(self, _: B0) -> Self::Output {
|
|
UTerm
|
|
}
|
|
}
|
|
|
|
impl<U: Unsigned, B: Bit> Mul<B1> for UInt<U, B> {
|
|
type Output = UInt<U, B>;
|
|
fn mul(self, _: B1) -> Self::Output {
|
|
UInt::new()
|
|
}
|
|
}
|
|
|
|
impl<U: Unsigned> Mul<U> for UTerm {
|
|
type Output = UTerm;
|
|
fn mul(self, _: U) -> Self::Output {
|
|
UTerm
|
|
}
|
|
}
|
|
|
|
impl<Ul: Unsigned, B: Bit, Ur: Unsigned> Mul<UInt<Ur, B>> for UInt<Ul, B0>
|
|
where
|
|
Ul: Mul<UInt<Ur, B>>,
|
|
{
|
|
type Output = UInt<Prod<Ul, UInt<Ur, B>>, B0>;
|
|
fn mul(self, _: UInt<Ur, B>) -> Self::Output {
|
|
unimplemented!()
|
|
}
|
|
}
|
|
|
|
pub trait Pow<Exp> {
|
|
type Output;
|
|
}
|
|
|
|
impl<X: Unsigned, N: Unsigned> Pow<N> for X
|
|
where
|
|
X: PrivatePow<U1, N>,
|
|
{
|
|
type Output = PrivatePowOut<X, U1, N>;
|
|
}
|
|
|
|
impl<Y: Unsigned, X: Unsigned> PrivatePow<Y, U0> for X {
|
|
type Output = Y;
|
|
}
|
|
|
|
impl<Y: Unsigned, X: Unsigned> PrivatePow<Y, U1> for X
|
|
where
|
|
X: Mul<Y>,
|
|
{
|
|
type Output = Prod<X, Y>;
|
|
}
|
|
|
|
impl<Y: Unsigned, U: Unsigned, B: Bit, X: Unsigned> PrivatePow<Y, UInt<UInt<U, B>, B0>> for X
|
|
where
|
|
X: Mul,
|
|
Square<X>: PrivatePow<Y, UInt<U, B>>,
|
|
{
|
|
type Output = PrivatePowOut<Square<X>, Y, UInt<U, B>>;
|
|
}
|