use crate::EndpointDescriptor;
use libusb1_sys as ffi;
use std::{fmt, slice};
pub struct Interface<'a> {
descriptors: &'a [ffi::libusb_interface_descriptor],
}
impl<'a> Interface<'a> {
pub fn number(&self) -> u8 {
self.descriptors[0].bInterfaceNumber
}
pub fn descriptors(&self) -> InterfaceDescriptors<'a> {
InterfaceDescriptors {
iter: self.descriptors.iter(),
}
}
}
impl<'a> From<&'a ffi::libusb_interface> for Interface<'a> {
fn from(interface: &'a ffi::libusb_interface) -> Self {
let descriptors = unsafe {
slice::from_raw_parts(interface.altsetting, interface.num_altsetting as usize)
};
debug_assert!(!descriptors.is_empty());
Interface { descriptors }
}
}
pub struct InterfaceDescriptors<'a> {
iter: slice::Iter<'a, ffi::libusb_interface_descriptor>,
}
impl<'a> Iterator for InterfaceDescriptors<'a> {
type Item = InterfaceDescriptor<'a>;
fn next(&mut self) -> Option<InterfaceDescriptor<'a>> {
self.iter.next().map(InterfaceDescriptor)
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.iter.size_hint()
}
}
pub struct InterfaceDescriptor<'a>(&'a ffi::libusb_interface_descriptor);
impl<'a> InterfaceDescriptor<'a> {
pub fn interface_number(&self) -> u8 {
self.0.bInterfaceNumber
}
pub fn setting_number(&self) -> u8 {
self.0.bAlternateSetting
}
pub fn class_code(&self) -> u8 {
self.0.bInterfaceClass
}
pub fn sub_class_code(&self) -> u8 {
self.0.bInterfaceSubClass
}
pub fn protocol_code(&self) -> u8 {
self.0.bInterfaceProtocol
}
pub fn description_string_index(&self) -> Option<u8> {
match self.0.iInterface {
0 => None,
n => Some(n),
}
}
pub fn num_endpoints(&self) -> u8 {
self.0.bNumEndpoints
}
pub fn endpoint_descriptors(&self) -> EndpointDescriptors<'a> {
let endpoints =
unsafe { slice::from_raw_parts(self.0.endpoint, self.0.bNumEndpoints as usize) };
EndpointDescriptors {
iter: endpoints.iter(),
}
}
pub fn extra(&self) -> &[u8] {
unsafe {
match self.0.extra_length {
len if len > 0 => slice::from_raw_parts(self.0.extra, len as usize),
_ => &[],
}
}
}
}
impl<'a> fmt::Debug for InterfaceDescriptor<'a> {
fn fmt(&self, fmt: &mut fmt::Formatter) -> Result<(), fmt::Error> {
let mut debug = fmt.debug_struct("InterfaceDescriptor");
debug.field("bLength", &self.0.bLength);
debug.field("bDescriptorType", &self.0.bDescriptorType);
debug.field("bInterfaceNumber", &self.0.bInterfaceNumber);
debug.field("bAlternateSetting", &self.0.bAlternateSetting);
debug.field("bNumEndpoints", &self.0.bNumEndpoints);
debug.field("bInterfaceClass", &self.0.bInterfaceClass);
debug.field("bInterfaceSubClass", &self.0.bInterfaceSubClass);
debug.field("bInterfaceProtocol", &self.0.bInterfaceProtocol);
debug.field("iInterface", &self.0.iInterface);
debug.finish()
}
}
pub struct EndpointDescriptors<'a> {
iter: slice::Iter<'a, ffi::libusb_endpoint_descriptor>,
}
impl<'a> Iterator for EndpointDescriptors<'a> {
type Item = EndpointDescriptor<'a>;
fn next(&mut self) -> Option<EndpointDescriptor<'a>> {
self.iter.next().map(EndpointDescriptor::from)
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.iter.size_hint()
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn it_has_interface_number() {
assert_eq!(
42,
Interface::from(&interface!(interface_descriptor!(bInterfaceNumber: 42))).number()
);
}
#[test]
fn it_has_interface_number_in_descriptor() {
assert_eq!(
vec!(42),
Interface::from(&interface!(interface_descriptor!(bInterfaceNumber: 42)))
.descriptors()
.map(|setting| setting.interface_number())
.collect::<Vec<_>>()
);
}
#[test]
fn it_has_alternate_setting_number() {
assert_eq!(
vec!(42),
Interface::from(&interface!(interface_descriptor!(bAlternateSetting: 42)))
.descriptors()
.map(|setting| setting.setting_number())
.collect::<Vec<_>>()
);
}
#[test]
fn it_has_class_code() {
assert_eq!(
vec!(42),
Interface::from(&interface!(interface_descriptor!(bInterfaceClass: 42)))
.descriptors()
.map(|setting| setting.class_code())
.collect::<Vec<_>>()
);
}
#[test]
fn it_has_sub_class_code() {
assert_eq!(
vec!(42),
Interface::from(&interface!(interface_descriptor!(bInterfaceSubClass: 42)))
.descriptors()
.map(|setting| setting.sub_class_code())
.collect::<Vec<_>>()
);
}
#[test]
fn it_has_protocol_code() {
assert_eq!(
vec!(42),
Interface::from(&interface!(interface_descriptor!(bInterfaceProtocol: 42)))
.descriptors()
.map(|setting| setting.protocol_code())
.collect::<Vec<_>>()
);
}
#[test]
fn it_has_description_string_index() {
assert_eq!(
vec!(Some(42)),
Interface::from(&interface!(interface_descriptor!(iInterface: 42)))
.descriptors()
.map(|setting| setting.description_string_index())
.collect::<Vec<_>>()
);
}
#[test]
fn it_handles_missing_description_string_index() {
assert_eq!(
vec!(None),
Interface::from(&interface!(interface_descriptor!(iInterface: 0)))
.descriptors()
.map(|setting| setting.description_string_index())
.collect::<Vec<_>>()
);
}
#[test]
fn it_has_num_endpoints() {
let endpoint1 = endpoint_descriptor!(bEndpointAddress: 0x81);
let endpoint2 = endpoint_descriptor!(bEndpointAddress: 0x01);
assert_eq!(
vec!(2),
Interface::from(&interface!(interface_descriptor!(endpoint1, endpoint2)))
.descriptors()
.map(|setting| setting.num_endpoints())
.collect::<Vec<_>>()
);
}
#[test]
fn it_has_endpoints() {
let libusb_interface = interface!(interface_descriptor!(
endpoint_descriptor!(bEndpointAddress: 0x87)
));
let interface = Interface::from(&libusb_interface);
let endpoint_addresses = interface
.descriptors()
.next()
.unwrap()
.endpoint_descriptors()
.map(|endpoint| endpoint.address())
.collect::<Vec<_>>();
assert_eq!(vec![0x87], endpoint_addresses);
}
}