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
use crate::{asn1::Any, Encodable, Encoder, Header, Length, Result, Tag};
use core::convert::TryFrom;
pub(super) fn decode_slice(any: Any<'_>) -> Result<&[u8]> {
let tag = any.tag().assert_eq(Tag::Integer)?;
let bytes = any.as_bytes();
match bytes {
[] => Err(tag.non_canonical_error()),
[0] => Ok(bytes),
[0, byte, ..] if *byte < 0x80 => Err(tag.non_canonical_error()),
[0, rest @ ..] => Ok(&rest),
[byte, ..] if *byte >= 0x80 => Err(tag.value_error()),
_ => Ok(bytes),
}
}
pub(super) fn decode_array<const N: usize>(any: Any<'_>) -> Result<[u8; N]> {
let input = decode_slice(any)?;
let mut output = [0u8; N];
output[N.saturating_sub(input.len())..].copy_from_slice(input);
Ok(output)
}
pub(super) fn encode(encoder: &mut Encoder<'_>, bytes: &[u8]) -> Result<()> {
let bytes = strip_leading_zeroes(&bytes);
let leading_zero = needs_leading_zero(bytes);
let len = (Length::try_from(bytes.len())? + leading_zero as u8)?;
Header::new(Tag::Integer, len)?.encode(encoder)?;
if leading_zero {
encoder.byte(0)?;
}
encoder.bytes(bytes)
}
#[inline]
pub(super) fn encoded_len(bytes: &[u8]) -> Result<Length> {
let bytes = strip_leading_zeroes(&bytes);
Length::try_from(bytes.len())? + needs_leading_zero(bytes) as u8
}
pub(super) fn strip_leading_zeroes(mut bytes: &[u8]) -> &[u8] {
while let Some((byte, rest)) = bytes.split_first() {
if *byte == 0 && !rest.is_empty() {
bytes = rest;
} else {
break;
}
}
bytes
}
fn needs_leading_zero(bytes: &[u8]) -> bool {
matches!(bytes.get(0), Some(byte) if *byte >= 0x80)
}