1use std::fmt;
2use std::num::NonZero;
3
4use rustc_abi::Size;
5use rustc_apfloat::Float;
6use rustc_apfloat::ieee::{Double, Half, Quad, Single};
7use rustc_data_structures::stable_hasher::{StableHash, StableHashCtxt};
8use rustc_serialize::{Decodable, Decoder, Encodable, Encoder};
9
10use crate::ty::TyCtxt;
11
12#[derive(#[automatically_derived]
impl ::core::marker::Copy for ConstInt { }Copy, #[automatically_derived]
impl ::core::clone::Clone for ConstInt {
#[inline]
fn clone(&self) -> ConstInt {
let _: ::core::clone::AssertParamIsClone<ScalarInt>;
let _: ::core::clone::AssertParamIsClone<bool>;
*self
}
}Clone)]
13pub struct ConstInt {
15 int: ScalarInt,
17 signed: bool,
19 is_ptr_sized_integral: bool,
21}
22
23impl ConstInt {
24 pub fn new(int: ScalarInt, signed: bool, is_ptr_sized_integral: bool) -> Self {
25 Self { int, signed, is_ptr_sized_integral }
26 }
27}
28
29#[derive(#[automatically_derived]
impl ::core::fmt::Debug for AtomicOrdering {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
AtomicOrdering::Relaxed => "Relaxed",
AtomicOrdering::Release => "Release",
AtomicOrdering::Acquire => "Acquire",
AtomicOrdering::AcqRel => "AcqRel",
AtomicOrdering::SeqCst => "SeqCst",
})
}
}Debug, #[automatically_derived]
impl ::core::marker::Copy for AtomicOrdering { }Copy, #[automatically_derived]
impl ::core::clone::Clone for AtomicOrdering {
#[inline]
fn clone(&self) -> AtomicOrdering { *self }
}Clone)]
33pub enum AtomicOrdering {
34 Relaxed = 0,
36 Release = 1,
37 Acquire = 2,
38 AcqRel = 3,
39 SeqCst = 4,
40}
41
42#[derive(#[automatically_derived]
impl ::core::fmt::Debug for SimdAlign {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
SimdAlign::Unaligned => "Unaligned",
SimdAlign::Element => "Element",
SimdAlign::Vector => "Vector",
})
}
}Debug, #[automatically_derived]
impl ::core::marker::Copy for SimdAlign { }Copy, #[automatically_derived]
impl ::core::clone::Clone for SimdAlign {
#[inline]
fn clone(&self) -> SimdAlign { *self }
}Clone)]
44pub enum SimdAlign {
45 Unaligned = 0,
47 Element = 1,
48 Vector = 2,
49}
50
51impl std::fmt::Debug for ConstInt {
52 fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
53 let Self { int, signed, is_ptr_sized_integral } = *self;
54 let size = int.size().bytes();
55 let raw = int.data;
56 if signed {
57 let bit_size = size * 8;
58 let min = 1u128 << (bit_size - 1);
59 let max = min - 1;
60 if raw == min {
61 match (size, is_ptr_sized_integral) {
62 (_, true) => fmt.write_fmt(format_args!("isize::MIN"))write!(fmt, "isize::MIN"),
63 (1, _) => fmt.write_fmt(format_args!("i8::MIN"))write!(fmt, "i8::MIN"),
64 (2, _) => fmt.write_fmt(format_args!("i16::MIN"))write!(fmt, "i16::MIN"),
65 (4, _) => fmt.write_fmt(format_args!("i32::MIN"))write!(fmt, "i32::MIN"),
66 (8, _) => fmt.write_fmt(format_args!("i64::MIN"))write!(fmt, "i64::MIN"),
67 (16, _) => fmt.write_fmt(format_args!("i128::MIN"))write!(fmt, "i128::MIN"),
68 _ => crate::util::bug::bug_fmt(format_args!("ConstInt 0x{0:x} with size = {1} and signed = {2}",
raw, size, signed))bug!("ConstInt 0x{:x} with size = {} and signed = {}", raw, size, signed),
69 }
70 } else if raw == max {
71 match (size, is_ptr_sized_integral) {
72 (_, true) => fmt.write_fmt(format_args!("isize::MAX"))write!(fmt, "isize::MAX"),
73 (1, _) => fmt.write_fmt(format_args!("i8::MAX"))write!(fmt, "i8::MAX"),
74 (2, _) => fmt.write_fmt(format_args!("i16::MAX"))write!(fmt, "i16::MAX"),
75 (4, _) => fmt.write_fmt(format_args!("i32::MAX"))write!(fmt, "i32::MAX"),
76 (8, _) => fmt.write_fmt(format_args!("i64::MAX"))write!(fmt, "i64::MAX"),
77 (16, _) => fmt.write_fmt(format_args!("i128::MAX"))write!(fmt, "i128::MAX"),
78 _ => crate::util::bug::bug_fmt(format_args!("ConstInt 0x{0:x} with size = {1} and signed = {2}",
raw, size, signed))bug!("ConstInt 0x{:x} with size = {} and signed = {}", raw, size, signed),
79 }
80 } else {
81 match size {
82 1 => fmt.write_fmt(format_args!("{0}", raw as i8))write!(fmt, "{}", raw as i8)?,
83 2 => fmt.write_fmt(format_args!("{0}", raw as i16))write!(fmt, "{}", raw as i16)?,
84 4 => fmt.write_fmt(format_args!("{0}", raw as i32))write!(fmt, "{}", raw as i32)?,
85 8 => fmt.write_fmt(format_args!("{0}", raw as i64))write!(fmt, "{}", raw as i64)?,
86 16 => fmt.write_fmt(format_args!("{0}", raw as i128))write!(fmt, "{}", raw as i128)?,
87 _ => crate::util::bug::bug_fmt(format_args!("ConstInt 0x{0:x} with size = {1} and signed = {2}",
raw, size, signed))bug!("ConstInt 0x{:x} with size = {} and signed = {}", raw, size, signed),
88 }
89 if fmt.alternate() {
90 match (size, is_ptr_sized_integral) {
91 (_, true) => fmt.write_fmt(format_args!("_isize"))write!(fmt, "_isize")?,
92 (1, _) => fmt.write_fmt(format_args!("_i8"))write!(fmt, "_i8")?,
93 (2, _) => fmt.write_fmt(format_args!("_i16"))write!(fmt, "_i16")?,
94 (4, _) => fmt.write_fmt(format_args!("_i32"))write!(fmt, "_i32")?,
95 (8, _) => fmt.write_fmt(format_args!("_i64"))write!(fmt, "_i64")?,
96 (16, _) => fmt.write_fmt(format_args!("_i128"))write!(fmt, "_i128")?,
97 (sz, _) => crate::util::bug::bug_fmt(format_args!("unexpected int size i{0}", sz))bug!("unexpected int size i{sz}"),
98 }
99 }
100 Ok(())
101 }
102 } else {
103 let max = Size::from_bytes(size).truncate(u128::MAX);
104 if raw == max {
105 match (size, is_ptr_sized_integral) {
106 (_, true) => fmt.write_fmt(format_args!("usize::MAX"))write!(fmt, "usize::MAX"),
107 (1, _) => fmt.write_fmt(format_args!("u8::MAX"))write!(fmt, "u8::MAX"),
108 (2, _) => fmt.write_fmt(format_args!("u16::MAX"))write!(fmt, "u16::MAX"),
109 (4, _) => fmt.write_fmt(format_args!("u32::MAX"))write!(fmt, "u32::MAX"),
110 (8, _) => fmt.write_fmt(format_args!("u64::MAX"))write!(fmt, "u64::MAX"),
111 (16, _) => fmt.write_fmt(format_args!("u128::MAX"))write!(fmt, "u128::MAX"),
112 _ => crate::util::bug::bug_fmt(format_args!("ConstInt 0x{0:x} with size = {1} and signed = {2}",
raw, size, signed))bug!("ConstInt 0x{:x} with size = {} and signed = {}", raw, size, signed),
113 }
114 } else {
115 match size {
116 1 => fmt.write_fmt(format_args!("{0}", raw as u8))write!(fmt, "{}", raw as u8)?,
117 2 => fmt.write_fmt(format_args!("{0}", raw as u16))write!(fmt, "{}", raw as u16)?,
118 4 => fmt.write_fmt(format_args!("{0}", raw as u32))write!(fmt, "{}", raw as u32)?,
119 8 => fmt.write_fmt(format_args!("{0}", raw as u64))write!(fmt, "{}", raw as u64)?,
120 16 => fmt.write_fmt(format_args!("{0}", raw as u128))write!(fmt, "{}", raw as u128)?,
121 _ => crate::util::bug::bug_fmt(format_args!("ConstInt 0x{0:x} with size = {1} and signed = {2}",
raw, size, signed))bug!("ConstInt 0x{:x} with size = {} and signed = {}", raw, size, signed),
122 }
123 if fmt.alternate() {
124 match (size, is_ptr_sized_integral) {
125 (_, true) => fmt.write_fmt(format_args!("_usize"))write!(fmt, "_usize")?,
126 (1, _) => fmt.write_fmt(format_args!("_u8"))write!(fmt, "_u8")?,
127 (2, _) => fmt.write_fmt(format_args!("_u16"))write!(fmt, "_u16")?,
128 (4, _) => fmt.write_fmt(format_args!("_u32"))write!(fmt, "_u32")?,
129 (8, _) => fmt.write_fmt(format_args!("_u64"))write!(fmt, "_u64")?,
130 (16, _) => fmt.write_fmt(format_args!("_u128"))write!(fmt, "_u128")?,
131 (sz, _) => crate::util::bug::bug_fmt(format_args!("unexpected unsigned int size u{0}",
sz))bug!("unexpected unsigned int size u{sz}"),
132 }
133 }
134 Ok(())
135 }
136 }
137 }
138}
139
140#[derive(#[automatically_derived]
impl ::core::clone::Clone for ScalarInt {
#[inline]
fn clone(&self) -> ScalarInt {
let _: ::core::clone::AssertParamIsClone<u128>;
let _: ::core::clone::AssertParamIsClone<NonZero<u8>>;
*self
}
}Clone, #[automatically_derived]
impl ::core::marker::Copy for ScalarInt { }Copy, #[automatically_derived]
impl ::core::cmp::Eq for ScalarInt {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<u128>;
let _: ::core::cmp::AssertParamIsEq<NonZero<u8>>;
}
}Eq, #[automatically_derived]
impl ::core::cmp::PartialEq for ScalarInt {
#[inline]
fn eq(&self, other: &ScalarInt) -> bool {
({ self.data }) == ({ other.data }) &&
({ self.size }) == ({ other.size })
}
}PartialEq, #[automatically_derived]
impl ::core::hash::Hash for ScalarInt {
#[inline]
fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
::core::hash::Hash::hash(&{ self.data }, state);
::core::hash::Hash::hash(&{ self.size }, state)
}
}Hash)]
145#[repr(packed)]
146pub struct ScalarInt {
147 data: u128,
150 size: NonZero<u8>,
151}
152
153impl StableHash for ScalarInt {
156 fn stable_hash<Hcx: StableHashCtxt>(
157 &self,
158 hcx: &mut Hcx,
159 hasher: &mut crate::ty::StableHasher,
160 ) {
161 { self.data }.stable_hash(hcx, hasher);
166 self.size.get().stable_hash(hcx, hasher);
167 }
168}
169
170impl<S: Encoder> Encodable<S> for ScalarInt {
171 fn encode(&self, s: &mut S) {
172 let size = self.size.get();
173 s.emit_u8(size);
174 s.emit_raw_bytes(&self.data.to_le_bytes()[..size as usize]);
175 }
176}
177
178impl<D: Decoder> Decodable<D> for ScalarInt {
179 fn decode(d: &mut D) -> ScalarInt {
180 let mut data = [0u8; 16];
181 let size = d.read_u8();
182 data[..size as usize].copy_from_slice(d.read_raw_bytes(size as usize));
183 ScalarInt { data: u128::from_le_bytes(data), size: NonZero::new(size).unwrap() }
184 }
185}
186
187impl ScalarInt {
188 pub const TRUE: ScalarInt = ScalarInt { data: 1_u128, size: NonZero::new(1).unwrap() };
189 pub const FALSE: ScalarInt = ScalarInt { data: 0_u128, size: NonZero::new(1).unwrap() };
190
191 fn raw(data: u128, size: Size) -> Self {
192 Self { data, size: NonZero::new(size.bytes() as u8).unwrap() }
193 }
194
195 #[inline]
196 pub fn size(self) -> Size {
197 Size::from_bytes(self.size.get())
198 }
199
200 #[inline(always)]
204 fn check_data(self) {
205 if true {
match (&self.size().truncate(self.data), &{ self.data }) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val,
::core::option::Option::Some(format_args!("Scalar value {0:#x} exceeds size of {1} bytes",
{ self.data }, self.size)));
}
}
};
};debug_assert_eq!(
211 self.size().truncate(self.data),
212 { self.data },
213 "Scalar value {:#x} exceeds size of {} bytes",
214 { self.data },
215 self.size
216 );
217 }
218
219 #[inline]
220 pub fn null(size: Size) -> Self {
221 Self::raw(0, size)
222 }
223
224 #[inline]
225 pub fn is_null(self) -> bool {
226 self.data == 0
227 }
228
229 #[inline]
230 pub fn try_from_uint(i: impl Into<u128>, size: Size) -> Option<Self> {
231 let (r, overflow) = Self::truncate_from_uint(i, size);
232 if overflow { None } else { Some(r) }
233 }
234
235 #[inline]
237 pub fn truncate_from_uint(i: impl Into<u128>, size: Size) -> (Self, bool) {
238 let data = i.into();
239 let r = Self::raw(size.truncate(data), size);
240 (r, r.data != data)
241 }
242
243 #[inline]
244 pub fn try_from_int(i: impl Into<i128>, size: Size) -> Option<Self> {
245 let (r, overflow) = Self::truncate_from_int(i, size);
246 if overflow { None } else { Some(r) }
247 }
248
249 #[inline]
251 pub fn truncate_from_int(i: impl Into<i128>, size: Size) -> (Self, bool) {
252 let data = i.into();
253 let r = Self::raw(size.truncate(data as u128), size);
255 (r, size.sign_extend(r.data) != data)
256 }
257
258 #[inline]
259 pub fn try_from_target_usize(i: impl Into<u128>, tcx: TyCtxt<'_>) -> Option<Self> {
260 Self::try_from_uint(i, tcx.data_layout.pointer_size())
261 }
262
263 #[inline]
268 pub fn try_to_bits(self, target_size: Size) -> Result<u128, Size> {
269 match (&(target_size.bytes()), &(0)) {
(left_val, right_val) => {
if *left_val == *right_val {
let kind = ::core::panicking::AssertKind::Ne;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::Some(format_args!("you should never look at the bits of a ZST")));
}
}
};assert_ne!(target_size.bytes(), 0, "you should never look at the bits of a ZST");
270 if target_size.bytes() == u64::from(self.size.get()) {
271 self.check_data();
272 Ok(self.data)
273 } else {
274 Err(self.size())
275 }
276 }
277
278 #[inline]
279 pub fn to_bits(self, target_size: Size) -> u128 {
280 self.try_to_bits(target_size).unwrap_or_else(|size| {
281 crate::util::bug::bug_fmt(format_args!("expected int of size {0}, but got size {1}",
target_size.bytes(), size.bytes()))bug!("expected int of size {}, but got size {}", target_size.bytes(), size.bytes())
282 })
283 }
284
285 #[inline]
287 pub fn to_bits_unchecked(self) -> u128 {
288 self.check_data();
289 self.data
290 }
291
292 #[inline]
295 pub fn to_uint(self, size: Size) -> u128 {
296 self.to_bits(size)
297 }
298
299 #[inline]
302 pub fn to_u8(self) -> u8 {
303 self.to_uint(Size::from_bits(8)).try_into().unwrap()
304 }
305
306 #[inline]
309 pub fn to_u16(self) -> u16 {
310 self.to_uint(Size::from_bits(16)).try_into().unwrap()
311 }
312
313 #[inline]
316 pub fn to_u32(self) -> u32 {
317 self.to_uint(Size::from_bits(32)).try_into().unwrap()
318 }
319
320 #[inline]
323 pub fn to_u64(self) -> u64 {
324 self.to_uint(Size::from_bits(64)).try_into().unwrap()
325 }
326
327 #[inline]
330 pub fn to_u128(self) -> u128 {
331 self.to_uint(Size::from_bits(128))
332 }
333
334 #[inline]
335 pub fn to_target_usize(&self, tcx: TyCtxt<'_>) -> u64 {
336 self.to_uint(tcx.data_layout.pointer_size()).try_into().unwrap()
337 }
338
339 #[inline]
340 pub fn to_atomic_ordering(self) -> AtomicOrdering {
341 use AtomicOrdering::*;
342 let val = self.to_u32();
343 if val == Relaxed as u32 {
344 Relaxed
345 } else if val == Release as u32 {
346 Release
347 } else if val == Acquire as u32 {
348 Acquire
349 } else if val == AcqRel as u32 {
350 AcqRel
351 } else if val == SeqCst as u32 {
352 SeqCst
353 } else {
354 { ::core::panicking::panic_fmt(format_args!("not a valid atomic ordering")); }panic!("not a valid atomic ordering")
355 }
356 }
357
358 #[inline]
359 pub fn to_simd_alignment(self) -> SimdAlign {
360 use SimdAlign::*;
361 let val = self.to_u32();
362 if val == Unaligned as u32 {
363 Unaligned
364 } else if val == Element as u32 {
365 Element
366 } else if val == Vector as u32 {
367 Vector
368 } else {
369 { ::core::panicking::panic_fmt(format_args!("not a valid simd alignment")); }panic!("not a valid simd alignment")
370 }
371 }
372
373 #[inline]
377 pub fn try_to_bool(self) -> Result<bool, ()> {
378 match self.to_u8() {
379 0 => Ok(false),
380 1 => Ok(true),
381 _ => Err(()),
382 }
383 }
384
385 #[inline]
388 pub fn to_int(self, size: Size) -> i128 {
389 let b = self.to_bits(size);
390 size.sign_extend(b)
391 }
392
393 pub fn to_i8(self) -> i8 {
396 self.to_int(Size::from_bits(8)).try_into().unwrap()
397 }
398
399 pub fn to_i16(self) -> i16 {
402 self.to_int(Size::from_bits(16)).try_into().unwrap()
403 }
404
405 pub fn to_i32(self) -> i32 {
408 self.to_int(Size::from_bits(32)).try_into().unwrap()
409 }
410
411 pub fn to_i64(self) -> i64 {
414 self.to_int(Size::from_bits(64)).try_into().unwrap()
415 }
416
417 pub fn to_i128(self) -> i128 {
420 self.to_int(Size::from_bits(128))
421 }
422
423 #[inline]
424 pub fn to_target_isize(&self, tcx: TyCtxt<'_>) -> i64 {
425 self.to_int(tcx.data_layout.pointer_size()).try_into().unwrap()
426 }
427
428 #[inline]
429 pub fn to_float<F: Float>(self) -> F {
430 F::from_bits(self.to_bits(Size::from_bits(F::BITS)))
432 }
433
434 #[inline]
435 pub fn to_f16(self) -> Half {
436 self.to_float()
437 }
438
439 #[inline]
440 pub fn to_f32(self) -> Single {
441 self.to_float()
442 }
443
444 #[inline]
445 pub fn to_f64(self) -> Double {
446 self.to_float()
447 }
448
449 #[inline]
450 pub fn to_f128(self) -> Quad {
451 self.to_float()
452 }
453}
454
455macro_rules! from_x_for_scalar_int {
456 ($($ty:ty),*) => {
457 $(
458 impl From<$ty> for ScalarInt {
459 #[inline]
460 fn from(u: $ty) -> Self {
461 Self {
462 data: u128::from(u),
463 size: NonZero::new(size_of::<$ty>() as u8).unwrap(),
464 }
465 }
466 }
467 )*
468 }
469}
470
471macro_rules! from_scalar_int_for_x {
472 ($($ty:ty),*) => {
473 $(
474 impl From<ScalarInt> for $ty {
475 #[inline]
476 fn from(int: ScalarInt) -> Self {
477 int.to_uint(Size::from_bytes(size_of::<$ty>()))
480 .try_into().unwrap()
481 }
482 }
483 )*
484 }
485}
486
487impl From<bool> for ScalarInt {
#[inline]
fn from(u: bool) -> Self {
Self {
data: u128::from(u),
size: NonZero::new(size_of::<bool>() as u8).unwrap(),
}
}
}from_x_for_scalar_int!(u8, u16, u32, u64, u128, bool);
488impl From<ScalarInt> for u128 {
#[inline]
fn from(int: ScalarInt) -> Self {
int.to_uint(Size::from_bytes(size_of::<u128>())).try_into().unwrap()
}
}from_scalar_int_for_x!(u8, u16, u32, u64, u128);
489
490impl TryFrom<ScalarInt> for bool {
491 type Error = ();
492 #[inline]
493 fn try_from(int: ScalarInt) -> Result<Self, ()> {
494 int.try_to_bool()
495 }
496}
497
498impl From<char> for ScalarInt {
499 #[inline]
500 fn from(c: char) -> Self {
501 (c as u32).into()
502 }
503}
504
505macro_rules! from_x_for_scalar_int_signed {
506 ($($ty:ty),*) => {
507 $(
508 impl From<$ty> for ScalarInt {
509 #[inline]
510 fn from(u: $ty) -> Self {
511 Self {
512 data: u128::from(u.cast_unsigned()), size: NonZero::new(size_of::<$ty>() as u8).unwrap(),
514 }
515 }
516 }
517 )*
518 }
519}
520
521macro_rules! from_scalar_int_for_x_signed {
522 ($($ty:ty),*) => {
523 $(
524 impl From<ScalarInt> for $ty {
525 #[inline]
526 fn from(int: ScalarInt) -> Self {
527 int.to_int(Size::from_bytes(size_of::<$ty>()))
530 .try_into().unwrap()
531 }
532 }
533 )*
534 }
535}
536
537impl From<i128> for ScalarInt {
#[inline]
fn from(u: i128) -> Self {
Self {
data: u128::from(u.cast_unsigned()),
size: NonZero::new(size_of::<i128>() as u8).unwrap(),
}
}
}from_x_for_scalar_int_signed!(i8, i16, i32, i64, i128);
538impl From<ScalarInt> for i128 {
#[inline]
fn from(int: ScalarInt) -> Self {
int.to_int(Size::from_bytes(size_of::<i128>())).try_into().unwrap()
}
}from_scalar_int_for_x_signed!(i8, i16, i32, i64, i128);
539
540impl From<std::cmp::Ordering> for ScalarInt {
541 #[inline]
542 fn from(c: std::cmp::Ordering) -> Self {
543 ScalarInt::from(c as i8)
545 }
546}
547
548#[derive(#[automatically_derived]
impl ::core::fmt::Debug for CharTryFromScalarInt {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f, "CharTryFromScalarInt")
}
}Debug)]
550pub struct CharTryFromScalarInt;
551
552impl TryFrom<ScalarInt> for char {
553 type Error = CharTryFromScalarInt;
554
555 #[inline]
556 fn try_from(int: ScalarInt) -> Result<Self, Self::Error> {
557 match char::from_u32(int.to_u32()) {
558 Some(c) => Ok(c),
559 None => Err(CharTryFromScalarInt),
560 }
561 }
562}
563
564impl From<Half> for ScalarInt {
565 #[inline]
566 fn from(f: Half) -> Self {
567 Self { data: f.to_bits(), size: NonZero::new((Half::BITS / 8) as u8).unwrap() }
569 }
570}
571
572impl From<ScalarInt> for Half {
573 #[inline]
574 fn from(int: ScalarInt) -> Self {
575 Self::from_bits(int.to_bits(Size::from_bytes(2)))
576 }
577}
578
579impl From<Single> for ScalarInt {
580 #[inline]
581 fn from(f: Single) -> Self {
582 Self { data: f.to_bits(), size: NonZero::new((Single::BITS / 8) as u8).unwrap() }
584 }
585}
586
587impl From<ScalarInt> for Single {
588 #[inline]
589 fn from(int: ScalarInt) -> Self {
590 Self::from_bits(int.to_bits(Size::from_bytes(4)))
591 }
592}
593
594impl From<Double> for ScalarInt {
595 #[inline]
596 fn from(f: Double) -> Self {
597 Self { data: f.to_bits(), size: NonZero::new((Double::BITS / 8) as u8).unwrap() }
599 }
600}
601
602impl From<ScalarInt> for Double {
603 #[inline]
604 fn from(int: ScalarInt) -> Self {
605 Self::from_bits(int.to_bits(Size::from_bytes(8)))
606 }
607}
608
609impl From<Quad> for ScalarInt {
610 #[inline]
611 fn from(f: Quad) -> Self {
612 Self { data: f.to_bits(), size: NonZero::new((Quad::BITS / 8) as u8).unwrap() }
614 }
615}
616
617impl From<ScalarInt> for Quad {
618 #[inline]
619 fn from(int: ScalarInt) -> Self {
620 Self::from_bits(int.to_bits(Size::from_bytes(16)))
621 }
622}
623
624impl fmt::Debug for ScalarInt {
625 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
626 f.write_fmt(format_args!("0x{0:x}", self))write!(f, "0x{self:x}")
628 }
629}
630
631impl fmt::LowerHex for ScalarInt {
632 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
633 self.check_data();
634 if f.alternate() {
635 f.write_fmt(format_args!("0x"))write!(f, "0x")?;
637 }
638 f.write_fmt(format_args!("{0:01$x}", { self.data },
self.size.get() as usize * 2))write!(f, "{:01$x}", { self.data }, self.size.get() as usize * 2)
646 }
647}
648
649impl fmt::UpperHex for ScalarInt {
650 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
651 self.check_data();
652 f.write_fmt(format_args!("{0:01$X}", { self.data },
self.size.get() as usize * 2))write!(f, "{:01$X}", { self.data }, self.size.get() as usize * 2)
660 }
661}
662
663impl fmt::Display for ScalarInt {
664 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
665 self.check_data();
666 f.write_fmt(format_args!("{0}", { self.data }))write!(f, "{}", { self.data })
667 }
668}