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
use std::ffi::CString;
use bitvec::prelude::*;
use crate::ffi;
use crate::Datatype;
use crate::Error;
use crate::Offset;
use crate::Status;
pub struct Session {
pub handle: ffi::Session,
}
impl Session {
pub fn open(bitfile: &str, signature: &str, resource: &str) -> Result<Self, Error> {
let mut handle: ffi::Session = Default::default();
let c_bitfile = CString::new(bitfile).unwrap();
let c_signature = CString::new(signature).unwrap();
let c_resource = CString::new(resource).unwrap();
let status = Status::from(unsafe {
ffi::Open(
c_bitfile.as_ptr(),
c_signature.as_ptr(),
c_resource.as_ptr(),
0,
&mut handle as *mut ffi::Session,
)
});
match status {
Status::Success => Ok(Session { handle }),
_ => Err(Error::FPGA(status)),
}
}
pub fn read<T: Datatype>(&self, offset: Offset) -> Result<T, Error> {
let mut bv = BitVec::with_capacity(((T::SIZE_IN_BITS - 1) / 8 + 1) * 8);
unsafe {
bv.set_len(((T::SIZE_IN_BITS - 1) / 8 + 1) * 8);
}
let fpga_bits = bv.as_mut_bitslice();
let status = Status::from(unsafe {
ffi::ReadArrayU8(
self.handle,
offset,
fpga_bits.as_mut_slice().as_mut_ptr(),
(T::SIZE_IN_BITS - 1) / 8 + 1,
)
});
match status {
Status::Success => Ok(Datatype::unpack(
&fpga_bits[((T::SIZE_IN_BITS - 1) / 8 + 1) * 8 - T::SIZE_IN_BITS..],
)?),
_ => Err(Error::FPGA(status)),
}
}
pub fn write<T: Datatype>(&self, offset: Offset, data: &T) -> Result<(), Error> {
let mut bv = BitVec::with_capacity(((T::SIZE_IN_BITS - 1) / 8 + 1) * 8);
unsafe {
bv.set_len(((T::SIZE_IN_BITS - 1) / 8 + 1) * 8);
}
let fpga_bits = bv.as_mut_bitslice();
Datatype::pack(
&mut fpga_bits[((T::SIZE_IN_BITS - 1) / 8 + 1) * 8 - T::SIZE_IN_BITS..],
data,
)?;
let status = Status::from(unsafe {
ffi::WriteArrayU8(
self.handle,
offset,
fpga_bits.as_slice().as_ptr(),
(T::SIZE_IN_BITS - 1) / 8 + 1,
)
});
match status {
Status::Success => Ok(()),
_ => Err(Error::FPGA(status)),
}
}
}
impl Drop for Session {
fn drop(&mut self) {
unsafe { ffi::Close(self.handle, 0) };
}
}