1use super::Tag;
4use crate::{Error, ErrorKind, Result};
5use core::fmt;
6
7#[derive(Copy, Clone, Debug, Eq, PartialEq, PartialOrd, Ord)]
22pub struct TagNumber(pub(super) u8);
23
24impl TagNumber {
25 pub const N0: Self = Self(0);
27
28 pub const N1: Self = Self(1);
30
31 pub const N2: Self = Self(2);
33
34 pub const N3: Self = Self(3);
36
37 pub const N4: Self = Self(4);
39
40 pub const N5: Self = Self(5);
42
43 pub const N6: Self = Self(6);
45
46 pub const N7: Self = Self(7);
48
49 pub const N8: Self = Self(8);
51
52 pub const N9: Self = Self(9);
54
55 pub const N10: Self = Self(10);
57
58 pub const N11: Self = Self(11);
60
61 pub const N12: Self = Self(12);
63
64 pub const N13: Self = Self(13);
66
67 pub const N14: Self = Self(14);
69
70 pub const N15: Self = Self(15);
72
73 pub const N16: Self = Self(16);
75
76 pub const N17: Self = Self(17);
78
79 pub const N18: Self = Self(18);
81
82 pub const N19: Self = Self(19);
84
85 pub const N20: Self = Self(20);
87
88 pub const N21: Self = Self(21);
90
91 pub const N22: Self = Self(22);
93
94 pub const N23: Self = Self(23);
96
97 pub const N24: Self = Self(24);
99
100 pub const N25: Self = Self(25);
102
103 pub const N26: Self = Self(26);
105
106 pub const N27: Self = Self(27);
108
109 pub const N28: Self = Self(28);
111
112 pub const N29: Self = Self(29);
114
115 pub const N30: Self = Self(30);
117
118 pub(super) const MASK: u8 = 0b11111;
120
121 const MAX: u8 = 30;
123
124 pub const fn new(byte: u8) -> Self {
129 #[allow(clippy::panic)]
130 if byte > Self::MAX {
131 panic!("tag number out of range");
132 }
133
134 Self(byte)
135 }
136
137 pub fn application(self, constructed: bool) -> Tag {
139 Tag::Application {
140 constructed,
141 number: self,
142 }
143 }
144
145 pub fn context_specific(self, constructed: bool) -> Tag {
147 Tag::ContextSpecific {
148 constructed,
149 number: self,
150 }
151 }
152
153 pub fn private(self, constructed: bool) -> Tag {
155 Tag::Private {
156 constructed,
157 number: self,
158 }
159 }
160
161 pub fn value(self) -> u8 {
163 self.0
164 }
165}
166
167impl TryFrom<u8> for TagNumber {
168 type Error = Error;
169
170 fn try_from(byte: u8) -> Result<Self> {
171 match byte {
172 0..=Self::MAX => Ok(Self(byte)),
173 _ => Err(ErrorKind::TagNumberInvalid.into()),
174 }
175 }
176}
177
178impl From<TagNumber> for u8 {
179 fn from(tag_number: TagNumber) -> u8 {
180 tag_number.0
181 }
182}
183
184impl fmt::Display for TagNumber {
185 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
186 write!(f, "{}", self.0)
187 }
188}
189
190#[cfg(feature = "arbitrary")]
193impl<'a> arbitrary::Arbitrary<'a> for TagNumber {
194 fn arbitrary(u: &mut arbitrary::Unstructured<'a>) -> arbitrary::Result<Self> {
195 Ok(Self::new(u.int_in_range(0..=Self::MAX)?))
196 }
197
198 fn size_hint(depth: usize) -> (usize, Option<usize>) {
199 u8::size_hint(depth)
200 }
201}