forked from blackmagic-debug/bmputil
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathusb.rs
250 lines (217 loc) · 8.24 KB
/
usb.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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
// SPDX-License-Identifier: MIT OR Apache-2.0
// SPDX-FileCopyrightText: 2022-2023 1BitSquared <[email protected]>
// SPDX-FileContributor: Written by Mikaela Szekely <[email protected]>
use thiserror::Error;
/// Simple newtype struct for some clarity in function arguments and whatnot.
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
pub struct Vid(pub u16);
/// Simple newtype struct for some clarity in function arguments and whatnot.
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
pub struct Pid(pub u16);
/// Simple newtype struct for some clarity in function arguments and whatnot.
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
pub struct InterfaceClass(pub u8);
impl InterfaceClass
{
/// bInterfaceClass field in DFU-class interface descriptors.
///
/// \[[USB DFU Device Class Spec § 4.2.1, Table 4.1](https://usb.org/sites/default/files/DFU_1.1.pdf#page=12)
/// and [§ 4.2.3, Table 4.4](https://usb.org/sites/default/files/DFU_1.1.pdf#page=15)\]
pub const APPLICATION_SPECIFIC: Self = Self(0xFE);
}
/// Simple newtype struct for some clarity in function arguments and whatnot.
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
pub struct InterfaceSubClass(pub u8);
impl InterfaceSubClass
{
/// bInterfaceSubClass field in DFU-class interface descriptors.
///
/// \[[USB DFU Device Class Spec § 4.2.1, Table 4.1](https://usb.org/sites/default/files/DFU_1.1.pdf#page=12)
/// and [§ 4.2.3, Table 4.4](https://usb.org/sites/default/files/DFU_1.1.pdf#page=15)\]
pub const DFU: Self = Self(0x01);
}
/// Simple newtype struct for some clarity in function arguments and whatnot.
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
pub struct InterfaceProtocol(pub u8);
impl InterfaceProtocol
{
/// bInterfaceProtocol field in DFU-class interface descriptors while in runtime mode.
///
/// \[[USB DFU Device Class Spec § 4.2.1, Table 4.1](https://usb.org/sites/default/files/DFU_1.1.pdf#page=12)
/// and [§ 4.2.3, Table 4.4](https://usb.org/sites/default/files/DFU_1.1.pdf#page=15)\]
#[allow(dead_code)] // XXX
pub const DFU_RUNTIME_MODE: Self = Self(0x01);
/// bInterfaceProtocol field in DFU-class interface descriptors while in DFU mode.
///
/// \[[USB DFU Device Class Spec § 4.2.1, Table 4.1](https://usb.org/sites/default/files/DFU_1.1.pdf#page=12)
/// and [§ 4.2.3, Table 4.4](https://usb.org/sites/default/files/DFU_1.1.pdf#page=15)\]
#[allow(dead_code)] // XXX
pub const DFU_DFU_MODE: Self = Self(0x02);
}
/// Enum of request numbers for DFU class requests.
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
#[allow(dead_code)]
pub enum DfuRequest
{
Detach = 0,
Dnload = 1,
Upload = 2,
GetStatus = 3,
ClrStatus = 4,
GetState = 5,
Abort = 6,
}
/// Enum representing the two "modes" a DFU-class device can be in.
///
/// Runtime mode is the normal operation mode, in which a device does the things it's made for and
/// exposes all the necessary descriptors to do so.
/// DFU mode is limited operating mode used for firmware upgrade purposes *only*. Devices switch
/// into this mode at the host's request.
/// \[[USB DFU Device Class Spec § 4.1](https://usb.org/sites/default/files/DFU_1.1.pdf#page=11)
/// and [§ 4.2](https://usb.org/sites/default/files/DFU_1.1.pdf#page=14)\].
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
pub enum DfuOperatingMode
{
Runtime,
FirmwareUpgrade,
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct GenericDescriptorRef<'a>
{
pub raw: &'a [u8],
}
impl<'a> GenericDescriptorRef<'a>
{
/// Returns the first descriptor found.
///
/// Panics if `bytes.len()` < `bytes[0]`.
pub fn single_from_bytes(bytes: &'a [u8]) -> Self
{
let length = bytes[0] as usize;
Self {
raw: &bytes[0..length],
}
}
/// Panics if any descriptors have an invalid size.
pub fn multiple_from_bytes(bytes: &'a [u8]) -> Vec<Self>
{
let mut v: Vec<Self> = Vec::new();
let mut current_bytes = &bytes[0..];
loop {
let descriptor = Self::single_from_bytes(current_bytes);
let parsed_count = descriptor.length_usize();
let remaining = current_bytes.len() - parsed_count;
v.push(descriptor);
if remaining == 0 {
break;
} else if remaining > 2 {
current_bytes = ¤t_bytes[parsed_count..];
} else {
panic!("Descriptor seems to have an invalid size of {}!", remaining);
}
}
v
}
#[allow(dead_code)] // XXX
pub fn length(&self) -> u8
{
self.raw[0]
}
pub fn length_usize(&self) -> usize
{
self.raw[0] as usize
}
pub fn descriptor_type(&self) -> u8
{
self.raw[1]
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, Error)]
pub enum DescriptorConvertError
{
#[error(
"bLength field ({provided_length}) in provided data does not match the correct value\
({correct_length}) for this descriptor type"
)]
LengthFieldMismatch
{
provided_length: u8,
correct_length: u8,
},
#[error(
"bDescriptorType field ({provided_type}) in provided data does not match the correct\
value ({correct_type}) for this descriptor type"
)]
DescriptorTypeMismatch
{
provided_type: u8,
correct_type: u8,
},
}
/// Structure of the DFU-class functional descriptor.
///
/// Unfortunately, as this structure contains `u16`s at uneven offsets, making this struct
/// `repr(packed)` would allow you to easily create unaligned references, and thus this
/// struct does not match the memory layout of the data sent over the USB bus. Sadface indeed.
#[allow(non_snake_case)]
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
#[repr(C)]
pub struct DfuFunctionalDescriptor
{
pub bLength: u8, // Should be 0x09.
pub bDescriptorType: u8, // Should be 0x21.
pub bmAttributes: u8,
pub wDetachTimeOut: u16,
pub wTransferSize: u16,
pub bcdDFUVersion: u16,
}
impl DfuFunctionalDescriptor
{
pub const LENGTH: u8 = 0x09;
pub const TYPE: u8 = 0x21;
/// Constructs a [DfuFunctionalDescriptor] from a byte slice, via per-field copy.
pub fn copy_from_bytes(bytes: &[u8; 0x09]) -> Result<Self, DescriptorConvertError>
{
if bytes[0] != Self::LENGTH {
return Err(DescriptorConvertError::LengthFieldMismatch {
provided_length: bytes[0],
correct_length: Self::LENGTH,
});
}
if bytes[1] != Self::TYPE {
return Err(DescriptorConvertError::DescriptorTypeMismatch {
provided_type: bytes[0],
correct_type: Self::TYPE,
});
}
Ok(Self {
bLength: bytes[0],
bDescriptorType: bytes[1],
bmAttributes: bytes[2],
wDetachTimeOut: u16::from_le_bytes(bytes[3..=4].try_into().unwrap()),
wTransferSize: u16::from_le_bytes(bytes[5..=6].try_into().unwrap()),
bcdDFUVersion: u16::from_le_bytes(bytes[7..=8].try_into().unwrap()),
})
}
}
/// The libusb version against which error conditions have been checked from its source code.
pub(crate) const CHECKED_LIBUSB_VERSION: &str = "1.0.26";
/// libusb has an API function whose documentation state non-zero return codes indicate failure
/// (and thus the [`rusb`](https://docs.rs/rusb) equivalents for them return `Result<T>`),
/// but the source code has no possible logic paths for a return code of anything other than zero,
/// and the documentation states that currently no error is ever returned.
/// Thus, `.unwrap()` panicking should be unreachable. This macro serves both as in-code
/// documentation of these cases (as `.unwrap()`s tend to look sus), and as a fallback in case
/// libusb updates or there is some failure condition that was missed in reviewing libusb's
/// source code.
#[macro_export]
macro_rules! libusb_cannot_fail
{
($funcname:literal) => {
const_format::formatcp!(
"As of libusb {}, {} cannot fail. This should be unreachable, unless libusb has updated.",
$crate::usb::CHECKED_LIBUSB_VERSION,
$funcname,
)
};
}