-
Notifications
You must be signed in to change notification settings - Fork 103
/
Copy pathinterface.rs
99 lines (87 loc) · 3.05 KB
/
interface.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
// These constants are defined in the including code.
// const column_count: u64 = ...;
// const first_column_id: u64 = ...;
fn known_to_slice<'a>(known: *mut u32, len: u64) -> &'a mut [u32] {
let words_per_row = (column_count + 31) / 32;
let rows = len / column_count;
let known_len = rows * words_per_row;
unsafe { std::slice::from_raw_parts_mut(known, known_len as usize) }
}
fn index(global_offset: u64, local_offset: i32, column: u64) -> usize {
let column = column - first_column_id;
let row = (global_offset as i64 + local_offset as i64) as u64;
(row * column_count + column) as usize
}
fn index_known(global_offset: u64, local_offset: i32, column: u64) -> (u64, u64) {
let column = column - first_column_id;
let row = (global_offset as i64 + local_offset as i64) as u64;
let words_per_row = (column_count + 31) / 32;
(row * words_per_row + column / 32, column % 32)
}
fn get(data: &[FieldElement], global_offset: u64, local_offset: i32, column: u64) -> FieldElement {
data[index(global_offset, local_offset, column)]
}
fn set(
data: &mut [FieldElement],
global_offset: u64,
local_offset: i32,
column: u64,
value: FieldElement,
) {
let i = index(global_offset, local_offset, column);
data[i] = value;
}
fn set_known(known: &mut [u32], global_offset: u64, local_offset: i32, column: u64) {
let (known_idx, known_bit) = index_known(global_offset, local_offset, column);
known[known_idx as usize] |= 1 << (known_bit);
}
fn get_param(params: &[LookupCell<FieldElement>], i: usize) -> FieldElement {
match params[i] {
LookupCell::Input(v) => *v,
LookupCell::Output(_) => panic!("Output cell used as input"),
}
}
fn set_param(params: &mut [LookupCell<FieldElement>], i: usize, value: FieldElement) {
match &mut params[i] {
LookupCell::Input(_) => panic!("Input cell used as output"),
LookupCell::Output(v) => **v = value,
}
}
#[repr(C)]
enum LookupCell<'a, T> {
/// Value is known (i.e. an input)
Input(&'a T),
/// Value is not known (i.e. an output)
Output(&'a mut T),
}
#[repr(C)]
pub struct MutSlice<T> {
data: *mut T,
len: u64,
}
impl<T> From<&mut [T]> for MutSlice<T> {
fn from(slice: &mut [T]) -> Self {
MutSlice {
data: slice.as_mut_ptr(),
len: slice.len() as u64,
}
}
}
impl<T> MutSlice<T> {
fn to_mut_slice<'a>(self) -> &'a mut [T] {
unsafe { std::slice::from_raw_parts_mut(self.data, self.len as usize) }
}
}
#[repr(C)]
pub struct WitgenFunctionParams<'a, T: 'a> {
data: MutSlice<T>,
known: *mut u32,
row_offset: u64,
params: MutSlice<LookupCell<'a, T>>,
mutable_state: *const std::ffi::c_void,
call_machine: extern "C" fn(*const std::ffi::c_void, T, MutSlice<LookupCell<'_, T>>) -> bool,
fixed_data: *const std::ffi::c_void,
get_fixed_value: extern "C" fn(*const std::ffi::c_void, u64, u64) -> T,
input_from_channel: extern "C" fn(*const std::ffi::c_void, u32, u64) -> T,
output_to_channel: extern "C" fn(*const std::ffi::c_void, u32, T),
}