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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
use alloc::vec::Vec;
#[derive(Debug, PartialEq, Clone, Hash)]
pub struct BitString<'a> {
data: &'a [u8],
padding_bits: u8,
}
impl<'a> BitString<'a> {
pub fn new(data: &'a [u8], padding_bits: u8) -> Option<BitString<'a>> {
if padding_bits > 7 || (data.is_empty() && padding_bits != 0) {
return None;
}
if padding_bits > 0 && data[data.len() - 1] & ((1 << padding_bits) - 1) != 0 {
return None;
}
Some(BitString { data, padding_bits })
}
pub fn as_bytes(&self) -> &'a [u8] {
self.data
}
pub fn padding_bits(&self) -> u8 {
self.padding_bits
}
pub fn has_bit_set(&self, n: usize) -> bool {
let idx = n / 8;
let v = 1 << (7 - (n & 0x07));
if self.data.len() < (idx + 1) {
false
} else {
self.data[idx] & v != 0
}
}
}
#[derive(Debug, PartialEq, Clone, Hash)]
pub struct OwnedBitString {
data: Vec<u8>,
padding_bits: u8,
}
impl OwnedBitString {
pub fn new(data: Vec<u8>, padding_bits: u8) -> Option<OwnedBitString> {
BitString::new(&data, padding_bits)?;
Some(OwnedBitString { data, padding_bits })
}
pub fn as_bitstring(&self) -> BitString<'_> {
BitString::new(&self.data, self.padding_bits).unwrap()
}
}
#[cfg(test)]
mod tests {
use crate::{BitString, OwnedBitString};
#[test]
fn test_bitstring_new() {
assert_eq!(BitString::new(b"abc", 8), None);
assert_eq!(BitString::new(b"", 2), None);
assert_eq!(BitString::new(b"\xff", 1), None);
assert!(BitString::new(b"\xff", 0).is_some());
assert!(BitString::new(b"\xfe", 1).is_some());
}
#[test]
fn test_owned_bitstring_new() {
assert_eq!(OwnedBitString::new(vec![b'a', b'b', b'c'], 8), None);
assert_eq!(OwnedBitString::new(vec![], 2), None);
assert_eq!(OwnedBitString::new(vec![0xff], 1), None);
assert!(OwnedBitString::new(vec![0xff], 0).is_some());
assert!(OwnedBitString::new(vec![0xfe], 1).is_some());
}
#[test]
fn test_bitstring_as_bytes() {
let bs = BitString::new(b"\xfe", 1).unwrap();
assert_eq!(bs.as_bytes(), b"\xfe");
}
#[test]
fn test_bitstring_padding_bits() {
let bs = BitString::new(b"\xfe", 1).unwrap();
assert_eq!(bs.padding_bits(), 1);
let bs = BitString::new(b"\xe0", 5).unwrap();
assert_eq!(bs.padding_bits(), 5);
}
#[test]
fn test_bitstring_has_bit_set() {
let bs = BitString::new(b"\x80", 0).unwrap();
assert!(bs.has_bit_set(0));
assert!(!bs.has_bit_set(1));
assert!(!bs.has_bit_set(7));
assert!(!bs.has_bit_set(50));
let bs = BitString::new(b"\xc0", 4).unwrap();
assert!(bs.has_bit_set(0));
assert!(bs.has_bit_set(1));
assert!(!bs.has_bit_set(2));
assert!(!bs.has_bit_set(3));
assert!(!bs.has_bit_set(4));
assert!(!bs.has_bit_set(5));
assert!(!bs.has_bit_set(6));
assert!(!bs.has_bit_set(7));
}
}