123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505 |
- /*! Low-level packet access and construction.
- The `wire` module deals with the packet *representation*. It provides two levels
- of functionality.
- * First, it provides functions to extract fields from sequences of octets,
- and to insert fields into sequences of octets. This happens `Packet` family of
- structures, e.g. [EthernetFrame] or [Ipv4Packet].
- * Second, in cases where the space of valid field values is much smaller than the space
- of possible field values, it provides a compact, high-level representation
- of packet data that can be parsed from and emitted into a sequence of octets.
- This happens through the `Repr` family of structs and enums, e.g. [ArpRepr] or [Ipv4Repr].
- [EthernetFrame]: struct.EthernetFrame.html
- [Ipv4Packet]: struct.Ipv4Packet.html
- [ArpRepr]: enum.ArpRepr.html
- [Ipv4Repr]: struct.Ipv4Repr.html
- The functions in the `wire` module are designed for use together with `-Cpanic=abort`.
- The `Packet` family of data structures guarantees that, if the `Packet::check_len()` method
- returned `Ok(())`, then no accessor or setter method will panic; however, the guarantee
- provided by `Packet::check_len()` may no longer hold after changing certain fields,
- which are listed in the documentation for the specific packet.
- The `Packet::new_checked` method is a shorthand for a combination of `Packet::new_unchecked`
- and `Packet::check_len`.
- When parsing untrusted input, it is *necessary* to use `Packet::new_checked()`;
- so long as the buffer is not modified, no accessor will fail.
- When emitting output, though, it is *incorrect* to use `Packet::new_checked()`;
- the length check is likely to succeed on a zeroed buffer, but fail on a buffer
- filled with data from a previous packet, such as when reusing buffers, resulting
- in nondeterministic panics with some network devices but not others.
- The buffer length for emission is not calculated by the `Packet` layer.
- In the `Repr` family of data structures, the `Repr::parse()` method never panics
- as long as `Packet::new_checked()` (or `Packet::check_len()`) has succeeded, and
- the `Repr::emit()` method never panics as long as the underlying buffer is exactly
- `Repr::buffer_len()` octets long.
- # Examples
- To emit an IP packet header into an octet buffer, and then parse it back:
- ```rust
- # #[cfg(feature = "proto-ipv4")]
- # {
- use smoltcp::phy::ChecksumCapabilities;
- use smoltcp::wire::*;
- let repr = Ipv4Repr {
- src_addr: Ipv4Address::new(10, 0, 0, 1),
- dst_addr: Ipv4Address::new(10, 0, 0, 2),
- next_header: IpProtocol::Tcp,
- payload_len: 10,
- hop_limit: 64,
- };
- let mut buffer = vec![0; repr.buffer_len() + repr.payload_len];
- { // emission
- let mut packet = Ipv4Packet::new_unchecked(&mut buffer);
- repr.emit(&mut packet, &ChecksumCapabilities::default());
- }
- { // parsing
- let packet = Ipv4Packet::new_checked(&buffer)
- .expect("truncated packet");
- let parsed = Ipv4Repr::parse(&packet, &ChecksumCapabilities::default())
- .expect("malformed packet");
- assert_eq!(repr, parsed);
- }
- # }
- ```
- */
- mod field {
- pub type Field = ::core::ops::Range<usize>;
- pub type Rest = ::core::ops::RangeFrom<usize>;
- }
- pub mod pretty_print;
- #[cfg(all(feature = "proto-ipv4", feature = "medium-ethernet"))]
- mod arp;
- #[cfg(feature = "proto-dhcpv4")]
- pub(crate) mod dhcpv4;
- #[cfg(feature = "proto-dns")]
- pub(crate) mod dns;
- #[cfg(feature = "medium-ethernet")]
- mod ethernet;
- #[cfg(any(feature = "proto-ipv4", feature = "proto-ipv6"))]
- mod icmp;
- #[cfg(feature = "proto-ipv4")]
- mod icmpv4;
- #[cfg(feature = "proto-ipv6")]
- mod icmpv6;
- #[cfg(feature = "medium-ieee802154")]
- pub mod ieee802154;
- #[cfg(feature = "proto-igmp")]
- mod igmp;
- pub(crate) mod ip;
- #[cfg(feature = "proto-ipv4")]
- mod ipv4;
- #[cfg(feature = "proto-ipv6")]
- mod ipv6;
- #[cfg(feature = "proto-ipv6")]
- mod ipv6fragment;
- #[cfg(feature = "proto-ipv6")]
- mod ipv6hopbyhop;
- #[cfg(feature = "proto-ipv6")]
- mod ipv6option;
- #[cfg(feature = "proto-ipv6")]
- mod ipv6routing;
- #[cfg(feature = "proto-ipv6")]
- mod mld;
- #[cfg(all(
- feature = "proto-ipv6",
- any(feature = "medium-ethernet", feature = "medium-ieee802154")
- ))]
- mod ndisc;
- #[cfg(all(
- feature = "proto-ipv6",
- any(feature = "medium-ethernet", feature = "medium-ieee802154")
- ))]
- mod ndiscoption;
- #[cfg(feature = "proto-rpl")]
- mod rpl;
- #[cfg(all(feature = "proto-sixlowpan", feature = "medium-ieee802154"))]
- mod sixlowpan;
- mod tcp;
- mod udp;
- use core::fmt;
- use crate::phy::Medium;
- pub use self::pretty_print::PrettyPrinter;
- #[cfg(feature = "medium-ethernet")]
- pub use self::ethernet::{
- Address as EthernetAddress, EtherType as EthernetProtocol, Frame as EthernetFrame,
- Repr as EthernetRepr, HEADER_LEN as ETHERNET_HEADER_LEN,
- };
- #[cfg(all(feature = "proto-ipv4", feature = "medium-ethernet"))]
- pub use self::arp::{
- Hardware as ArpHardware, Operation as ArpOperation, Packet as ArpPacket, Repr as ArpRepr,
- };
- #[cfg(feature = "proto-rpl")]
- pub use self::rpl::{
- data::HopByHopOption as RplHopByHopRepr, data::Packet as RplHopByHopPacket,
- options::Packet as RplOptionPacket, options::Repr as RplOptionRepr,
- InstanceId as RplInstanceId, Repr as RplRepr,
- };
- #[cfg(all(feature = "proto-sixlowpan", feature = "medium-ieee802154"))]
- pub use self::sixlowpan::{
- frag::{Key as SixlowpanFragKey, Packet as SixlowpanFragPacket, Repr as SixlowpanFragRepr},
- iphc::{Packet as SixlowpanIphcPacket, Repr as SixlowpanIphcRepr},
- nhc::{
- ExtHeaderPacket as SixlowpanExtHeaderPacket, ExtHeaderRepr as SixlowpanExtHeaderRepr,
- NhcPacket as SixlowpanNhcPacket, UdpNhcPacket as SixlowpanUdpNhcPacket,
- UdpNhcRepr as SixlowpanUdpNhcRepr,
- },
- AddressContext as SixlowpanAddressContext, NextHeader as SixlowpanNextHeader, SixlowpanPacket,
- };
- #[cfg(feature = "medium-ieee802154")]
- pub use self::ieee802154::{
- Address as Ieee802154Address, AddressingMode as Ieee802154AddressingMode,
- Frame as Ieee802154Frame, FrameType as Ieee802154FrameType,
- FrameVersion as Ieee802154FrameVersion, Pan as Ieee802154Pan, Repr as Ieee802154Repr,
- };
- pub use self::ip::{
- Address as IpAddress, Cidr as IpCidr, Endpoint as IpEndpoint,
- ListenEndpoint as IpListenEndpoint, Protocol as IpProtocol, Repr as IpRepr,
- Version as IpVersion,
- };
- #[cfg(feature = "proto-ipv4")]
- pub use self::ipv4::{
- Address as Ipv4Address, Cidr as Ipv4Cidr, Key as Ipv4FragKey, Packet as Ipv4Packet,
- Repr as Ipv4Repr, HEADER_LEN as IPV4_HEADER_LEN, MIN_MTU as IPV4_MIN_MTU,
- };
- #[cfg(feature = "proto-ipv6")]
- pub use self::ipv6::{
- Address as Ipv6Address, Cidr as Ipv6Cidr, Packet as Ipv6Packet, Repr as Ipv6Repr,
- HEADER_LEN as IPV6_HEADER_LEN, MIN_MTU as IPV6_MIN_MTU,
- };
- #[cfg(feature = "proto-ipv6")]
- pub use self::ipv6option::{
- FailureType as Ipv6OptionFailureType, Ipv6Option, Repr as Ipv6OptionRepr,
- Type as Ipv6OptionType,
- };
- #[cfg(feature = "proto-ipv6")]
- pub use self::ipv6hopbyhop::{Header as Ipv6HopByHopHeader, Repr as Ipv6HopByHopRepr};
- #[cfg(feature = "proto-ipv6")]
- pub use self::ipv6fragment::{Header as Ipv6FragmentHeader, Repr as Ipv6FragmentRepr};
- #[cfg(feature = "proto-ipv6")]
- pub use self::ipv6routing::{
- Header as Ipv6RoutingHeader, Repr as Ipv6RoutingRepr, Type as Ipv6RoutingType,
- };
- #[cfg(feature = "proto-ipv4")]
- pub use self::icmpv4::{
- DstUnreachable as Icmpv4DstUnreachable, Message as Icmpv4Message, Packet as Icmpv4Packet,
- ParamProblem as Icmpv4ParamProblem, Redirect as Icmpv4Redirect, Repr as Icmpv4Repr,
- TimeExceeded as Icmpv4TimeExceeded,
- };
- #[cfg(feature = "proto-igmp")]
- pub use self::igmp::{IgmpVersion, Packet as IgmpPacket, Repr as IgmpRepr};
- #[cfg(feature = "proto-ipv6")]
- pub use self::icmpv6::{
- DstUnreachable as Icmpv6DstUnreachable, Message as Icmpv6Message, Packet as Icmpv6Packet,
- ParamProblem as Icmpv6ParamProblem, Repr as Icmpv6Repr, TimeExceeded as Icmpv6TimeExceeded,
- };
- #[cfg(any(feature = "proto-ipv4", feature = "proto-ipv6"))]
- pub use self::icmp::Repr as IcmpRepr;
- #[cfg(all(
- feature = "proto-ipv6",
- any(feature = "medium-ethernet", feature = "medium-ieee802154")
- ))]
- pub use self::ndisc::{
- NeighborFlags as NdiscNeighborFlags, Repr as NdiscRepr, RouterFlags as NdiscRouterFlags,
- };
- #[cfg(all(
- feature = "proto-ipv6",
- any(feature = "medium-ethernet", feature = "medium-ieee802154")
- ))]
- pub use self::ndiscoption::{
- NdiscOption, PrefixInfoFlags as NdiscPrefixInfoFlags,
- PrefixInformation as NdiscPrefixInformation, RedirectedHeader as NdiscRedirectedHeader,
- Repr as NdiscOptionRepr, Type as NdiscOptionType,
- };
- #[cfg(feature = "proto-ipv6")]
- pub use self::mld::{AddressRecord as MldAddressRecord, Repr as MldRepr};
- pub use self::udp::{Packet as UdpPacket, Repr as UdpRepr, HEADER_LEN as UDP_HEADER_LEN};
- pub use self::tcp::{
- Control as TcpControl, Packet as TcpPacket, Repr as TcpRepr, SeqNumber as TcpSeqNumber,
- TcpOption, HEADER_LEN as TCP_HEADER_LEN,
- };
- #[cfg(feature = "proto-dhcpv4")]
- pub use self::dhcpv4::{
- DhcpOption, DhcpOptionWriter, MessageType as DhcpMessageType, Packet as DhcpPacket,
- Repr as DhcpRepr, CLIENT_PORT as DHCP_CLIENT_PORT,
- MAX_DNS_SERVER_COUNT as DHCP_MAX_DNS_SERVER_COUNT, SERVER_PORT as DHCP_SERVER_PORT,
- };
- #[cfg(feature = "proto-dns")]
- pub use self::dns::{
- Flags as DnsFlags, Opcode as DnsOpcode, Packet as DnsPacket, Repr as DnsRepr,
- Type as DnsQueryType,
- };
- /// Parsing a packet failed.
- ///
- /// Either it is malformed, or it is not supported by smoltcp.
- #[derive(Debug, Clone, Copy, PartialEq, Eq)]
- #[cfg_attr(feature = "defmt", derive(defmt::Format))]
- pub struct Error;
- #[cfg(feature = "std")]
- impl std::error::Error for Error {}
- impl fmt::Display for Error {
- fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
- write!(f, "wire::Error")
- }
- }
- pub type Result<T> = core::result::Result<T, Error>;
- /// Representation of an hardware address, such as an Ethernet address or an IEEE802.15.4 address.
- #[cfg(any(
- feature = "medium-ip",
- feature = "medium-ethernet",
- feature = "medium-ieee802154"
- ))]
- #[derive(Debug, Clone, Copy, PartialEq, Eq)]
- #[cfg_attr(feature = "defmt", derive(defmt::Format))]
- pub enum HardwareAddress {
- #[cfg(feature = "medium-ip")]
- Ip,
- #[cfg(feature = "medium-ethernet")]
- Ethernet(EthernetAddress),
- #[cfg(feature = "medium-ieee802154")]
- Ieee802154(Ieee802154Address),
- }
- #[cfg(any(
- feature = "medium-ip",
- feature = "medium-ethernet",
- feature = "medium-ieee802154"
- ))]
- impl HardwareAddress {
- pub const fn as_bytes(&self) -> &[u8] {
- match self {
- #[cfg(feature = "medium-ip")]
- HardwareAddress::Ip => unreachable!(),
- #[cfg(feature = "medium-ethernet")]
- HardwareAddress::Ethernet(addr) => addr.as_bytes(),
- #[cfg(feature = "medium-ieee802154")]
- HardwareAddress::Ieee802154(addr) => addr.as_bytes(),
- }
- }
- /// Query wether the address is an unicast address.
- pub fn is_unicast(&self) -> bool {
- match self {
- #[cfg(feature = "medium-ip")]
- HardwareAddress::Ip => unreachable!(),
- #[cfg(feature = "medium-ethernet")]
- HardwareAddress::Ethernet(addr) => addr.is_unicast(),
- #[cfg(feature = "medium-ieee802154")]
- HardwareAddress::Ieee802154(addr) => addr.is_unicast(),
- }
- }
- /// Query wether the address is a broadcast address.
- pub fn is_broadcast(&self) -> bool {
- match self {
- #[cfg(feature = "medium-ip")]
- HardwareAddress::Ip => unreachable!(),
- #[cfg(feature = "medium-ethernet")]
- HardwareAddress::Ethernet(addr) => addr.is_broadcast(),
- #[cfg(feature = "medium-ieee802154")]
- HardwareAddress::Ieee802154(addr) => addr.is_broadcast(),
- }
- }
- #[cfg(feature = "medium-ethernet")]
- pub(crate) fn ethernet_or_panic(&self) -> EthernetAddress {
- match self {
- HardwareAddress::Ethernet(addr) => *addr,
- #[allow(unreachable_patterns)]
- _ => panic!("HardwareAddress is not Ethernet."),
- }
- }
- #[cfg(feature = "medium-ieee802154")]
- pub(crate) fn ieee802154_or_panic(&self) -> Ieee802154Address {
- match self {
- HardwareAddress::Ieee802154(addr) => *addr,
- #[allow(unreachable_patterns)]
- _ => panic!("HardwareAddress is not Ethernet."),
- }
- }
- #[inline]
- pub(crate) fn medium(&self) -> Medium {
- match self {
- #[cfg(feature = "medium-ip")]
- HardwareAddress::Ip => Medium::Ip,
- #[cfg(feature = "medium-ethernet")]
- HardwareAddress::Ethernet(_) => Medium::Ethernet,
- #[cfg(feature = "medium-ieee802154")]
- HardwareAddress::Ieee802154(_) => Medium::Ieee802154,
- }
- }
- }
- #[cfg(any(
- feature = "medium-ip",
- feature = "medium-ethernet",
- feature = "medium-ieee802154"
- ))]
- impl core::fmt::Display for HardwareAddress {
- fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
- match self {
- #[cfg(feature = "medium-ip")]
- HardwareAddress::Ip => write!(f, "no hardware addr"),
- #[cfg(feature = "medium-ethernet")]
- HardwareAddress::Ethernet(addr) => write!(f, "{addr}"),
- #[cfg(feature = "medium-ieee802154")]
- HardwareAddress::Ieee802154(addr) => write!(f, "{addr}"),
- }
- }
- }
- #[cfg(feature = "medium-ethernet")]
- impl From<EthernetAddress> for HardwareAddress {
- fn from(addr: EthernetAddress) -> Self {
- HardwareAddress::Ethernet(addr)
- }
- }
- #[cfg(feature = "medium-ieee802154")]
- impl From<Ieee802154Address> for HardwareAddress {
- fn from(addr: Ieee802154Address) -> Self {
- HardwareAddress::Ieee802154(addr)
- }
- }
- #[cfg(not(feature = "medium-ieee802154"))]
- pub const MAX_HARDWARE_ADDRESS_LEN: usize = 6;
- #[cfg(feature = "medium-ieee802154")]
- pub const MAX_HARDWARE_ADDRESS_LEN: usize = 8;
- /// Unparsed hardware address.
- ///
- /// Used to make NDISC parsing agnostic of the hardware medium in use.
- #[cfg(any(feature = "medium-ethernet", feature = "medium-ieee802154"))]
- #[derive(Debug, PartialEq, Eq, Clone, Copy)]
- #[cfg_attr(feature = "defmt", derive(defmt::Format))]
- pub struct RawHardwareAddress {
- len: u8,
- data: [u8; MAX_HARDWARE_ADDRESS_LEN],
- }
- #[cfg(any(feature = "medium-ethernet", feature = "medium-ieee802154"))]
- impl RawHardwareAddress {
- pub fn from_bytes(addr: &[u8]) -> Self {
- let mut data = [0u8; MAX_HARDWARE_ADDRESS_LEN];
- data[..addr.len()].copy_from_slice(addr);
- Self {
- len: addr.len() as u8,
- data,
- }
- }
- pub fn as_bytes(&self) -> &[u8] {
- &self.data[..self.len as usize]
- }
- pub const fn len(&self) -> usize {
- self.len as usize
- }
- pub const fn is_empty(&self) -> bool {
- self.len == 0
- }
- pub fn parse(&self, medium: Medium) -> Result<HardwareAddress> {
- match medium {
- #[cfg(feature = "medium-ethernet")]
- Medium::Ethernet => {
- if self.len() < 6 {
- return Err(Error);
- }
- Ok(HardwareAddress::Ethernet(EthernetAddress::from_bytes(
- self.as_bytes(),
- )))
- }
- #[cfg(feature = "medium-ieee802154")]
- Medium::Ieee802154 => {
- if self.len() < 8 {
- return Err(Error);
- }
- Ok(HardwareAddress::Ieee802154(Ieee802154Address::from_bytes(
- self.as_bytes(),
- )))
- }
- #[cfg(feature = "medium-ip")]
- Medium::Ip => unreachable!(),
- }
- }
- }
- #[cfg(any(feature = "medium-ethernet", feature = "medium-ieee802154"))]
- impl core::fmt::Display for RawHardwareAddress {
- fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
- for (i, &b) in self.as_bytes().iter().enumerate() {
- if i != 0 {
- write!(f, ":")?;
- }
- write!(f, "{b:02x}")?;
- }
- Ok(())
- }
- }
- #[cfg(feature = "medium-ethernet")]
- impl From<EthernetAddress> for RawHardwareAddress {
- fn from(addr: EthernetAddress) -> Self {
- Self::from_bytes(addr.as_bytes())
- }
- }
- #[cfg(feature = "medium-ieee802154")]
- impl From<Ieee802154Address> for RawHardwareAddress {
- fn from(addr: Ieee802154Address) -> Self {
- Self::from_bytes(addr.as_bytes())
- }
- }
- #[cfg(any(feature = "medium-ethernet", feature = "medium-ieee802154"))]
- impl From<HardwareAddress> for RawHardwareAddress {
- fn from(addr: HardwareAddress) -> Self {
- Self::from_bytes(addr.as_bytes())
- }
- }
|