Skip to content

Commit fd1c9c2

Browse files
authored
Add contracts for Layout and Alignment (#33)
These contracts seek to capture what is described in documentation and debug assertions.
1 parent 5d8ee62 commit fd1c9c2

File tree

2 files changed

+29
-0
lines changed

2 files changed

+29
-0
lines changed

library/core/src/alloc/layout.rs

+23
Original file line numberDiff line numberDiff line change
@@ -4,12 +4,16 @@
44
// collections, resulting in having to optimize down excess IR multiple times.
55
// Your performance intuition is useless. Run perf.
66

7+
use safety::requires;
78
use crate::cmp;
89
use crate::error::Error;
910
use crate::fmt;
1011
use crate::mem;
1112
use crate::ptr::{Alignment, NonNull};
1213

14+
#[cfg(kani)]
15+
use crate::kani;
16+
1317
// While this function is used in one place and its implementation
1418
// could be inlined, the previous attempts to do so made rustc
1519
// slower:
@@ -117,6 +121,7 @@ impl Layout {
117121
#[must_use]
118122
#[inline]
119123
#[rustc_allow_const_fn_unstable(ptr_alignment_type)]
124+
#[requires(Layout::from_size_align(size, align).is_ok())]
120125
pub const unsafe fn from_size_align_unchecked(size: usize, align: usize) -> Self {
121126
// SAFETY: the caller is required to uphold the preconditions.
122127
unsafe { Layout { size, align: Alignment::new_unchecked(align) } }
@@ -492,3 +497,21 @@ impl fmt::Display for LayoutError {
492497
f.write_str("invalid parameters to Layout::from_size_align")
493498
}
494499
}
500+
501+
#[cfg(kani)]
502+
#[unstable(feature="kani", issue="none")]
503+
mod verify {
504+
use super::*;
505+
506+
#[kani::proof_for_contract(Layout::from_size_align_unchecked)]
507+
pub fn check_from_size_align_unchecked() {
508+
let s = kani::any::<usize>();
509+
let a = kani::any::<usize>();
510+
511+
unsafe {
512+
let layout = Layout::from_size_align_unchecked(s, a);
513+
assert_eq!(layout.size(), s);
514+
assert_eq!(layout.align(), a);
515+
}
516+
}
517+
}

library/core/src/ptr/alignment.rs

+6
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,12 @@
1+
use safety::requires;
12
use crate::num::NonZero;
23
#[cfg(debug_assertions)]
34
use crate::ub_checks::assert_unsafe_precondition;
45
use crate::{cmp, fmt, hash, mem, num};
56

7+
#[cfg(kani)]
8+
use crate::kani;
9+
610
/// A type storing a `usize` which is a power of two, and thus
711
/// represents a possible alignment in the Rust abstract machine.
812
///
@@ -76,6 +80,8 @@ impl Alignment {
7680
#[unstable(feature = "ptr_alignment_type", issue = "102070")]
7781
#[rustc_const_unstable(feature = "ptr_alignment_type", issue = "102070")]
7882
#[inline]
83+
#[requires(align > 0)]
84+
#[requires((align & (align - 1)) == 0)]
7985
pub const unsafe fn new_unchecked(align: usize) -> Self {
8086
#[cfg(debug_assertions)]
8187
assert_unsafe_precondition!(

0 commit comments

Comments
 (0)