mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-09-04 20:19:47 +08:00

Core & protocols ---------------- - Wrap datapath globals into net_aligned_data, to avoid false sharing. - Preserve MSG_ZEROCOPY in forwarding (e.g. out of a container). - Add SO_INQ and SCM_INQ support to AF_UNIX. - Add SIOCINQ support to AF_VSOCK. - Add TCP_MAXSEG sockopt to MPTCP. - Add IPv6 force_forwarding sysctl to enable forwarding per interface. - Make TCP validation of whether packet fully fits in the receive window and the rcv_buf more strict. With increased use of HW aggregation a single "packet" can be multiple 100s of kB. - Add MSG_MORE flag to optimize large TCP transmissions via sockmap, improves latency up to 33% for sockmap users. - Convert TCP send queue handling from tasklet to BH workque. - Improve BPF iteration over TCP sockets to see each socket exactly once. - Remove obsolete and unused TCP RFC3517/RFC6675 loss recovery code. - Support enabling kernel threads for NAPI processing on per-NAPI instance basis rather than a whole device. Fully stop the kernel NAPI thread when threaded NAPI gets disabled. Previously thread would stick around until ifdown due to tricky synchronization. - Allow multicast routing to take effect on locally-generated packets. - Add output interface argument for End.X in segment routing. - MCTP: add support for gateway routing, improve bind() handling. - Don't require rtnl_lock when fetching an IPv6 neighbor over Netlink. - Add a new neighbor flag ("extern_valid"), which cedes refresh responsibilities to userspace. This is needed for EVPN multi-homing where a neighbor entry for a multi-homed host needs to be synced across all the VTEPs among which the host is multi-homed. - Support NUD_PERMANENT for proxy neighbor entries. - Add a new queuing discipline for IETF RFC9332 DualQ Coupled AQM. - Add sequence numbers to netconsole messages. Unregister netconsole's console when all net targets are removed. Code refactoring. Add a number of selftests. - Align IPSec inbound SA lookup to RFC 4301. Only SPI and protocol should be used for an inbound SA lookup. - Support inspecting ref_tracker state via DebugFS. - Don't force bonding advertisement frames tx to ~333 ms boundaries. Add broadcast_neighbor option to send ARP/ND on all bonded links. - Allow providing upcall pid for the 'execute' command in openvswitch. - Remove DCCP support from Netfilter's conntrack. - Disallow multiple packet duplications in the queuing layer. - Prevent use of deprecated iptables code on PREEMPT_RT. Driver API ---------- - Support RSS and hashing configuration over ethtool Netlink. - Add dedicated ethtool callbacks for getting and setting hashing fields. - Add support for power budget evaluation strategy in PSE / Power-over-Ethernet. Generate Netlink events for overcurrent etc. - Support DPLL phase offset monitoring across all device inputs. Support providing clock reference and SYNC over separate DPLL inputs. - Support traffic classes in devlink rate API for bandwidth management. - Remove rtnl_lock dependency from UDP tunnel port configuration. Device drivers -------------- - Add a new Broadcom driver for 800G Ethernet (bnge). - Add a standalone driver for Microchip ZL3073x DPLL. - Remove IBM's NETIUCV device driver. - Ethernet high-speed NICs: - Broadcom (bnxt): - support zero-copy Tx of DMABUF memory - take page size into account for page pool recycling rings - Intel (100G, ice, idpf): - idpf: XDP and AF_XDP support preparations - idpf: add flow steering - add link_down_events statistic - clean up the TSPLL code - preparations for live VM migration - nVidia/Mellanox: - support zero-copy Rx/Tx interfaces (DMABUF and io_uring) - optimize context memory usage for matchers - expose serial numbers in devlink info - support PCIe congestion metrics - Meta (fbnic): - add 25G, 50G, and 100G link modes to phylink - support dumping FW logs - Marvell/Cavium: - support for CN20K generation of the Octeon chips - Amazon: - add HW clock (without timestamping, just hypervisor time access) - Ethernet virtual: - VirtIO net: - support segmentation of UDP-tunnel-encapsulated packets - Google (gve): - support packet timestamping and clock synchronization - Microsoft vNIC: - add handler for device-originated servicing events - allow dynamic MSI-X vector allocation - support Tx bandwidth clamping - Ethernet NICs consumer, and embedded: - AMD: - amd-xgbe: hardware timestamping and PTP clock support - Broadcom integrated MACs (bcmgenet, bcmasp): - use napi_complete_done() return value to support NAPI polling - add support for re-starting auto-negotiation - Broadcom switches (b53): - support BCM5325 switches - add bcm63xx EPHY power control - Synopsys (stmmac): - lots of code refactoring and cleanups - TI: - icssg-prueth: read firmware-names from device tree - icssg: PRP offload support - Microchip: - lan78xx: convert to PHYLINK for improved PHY and MAC management - ksz: add KSZ8463 switch support - Intel: - support similar queue priority scheme in multi-queue and time-sensitive networking (taprio) - support packet pre-emption in both - RealTek (r8169): - enable EEE at 5Gbps on RTL8126 - Airoha: - add PPPoE offload support - MDIO bus controller for Airoha AN7583 - Ethernet PHYs: - support for the IPQ5018 internal GE PHY - micrel KSZ9477 switch-integrated PHYs: - add MDI/MDI-X control support - add RX error counters - add cable test support - add Signal Quality Indicator (SQI) reporting - dp83tg720: improve reset handling and reduce link recovery time - support bcm54811 (and its MII-Lite interface type) - air_en8811h: support resume/suspend - support PHY counters for QCA807x and QCA808x - support WoL for QCA807x - CAN drivers: - rcar_canfd: support for Transceiver Delay Compensation - kvaser: report FW versions via devlink dev info - WiFi: - extended regulatory info support (6 GHz) - add statistics and beacon monitor for Multi-Link Operation (MLO) - support S1G aggregation, improve S1G support - add Radio Measurement action fields - support per-radio RTS threshold - some work around how FIPS affects wifi, which was wrong (RC4 is used by TKIP, not only WEP) - improvements for unsolicited probe response handling - WiFi drivers: - RealTek (rtw88): - IBSS mode for SDIO devices - RealTek (rtw89): - BT coexistence for MLO/WiFi7 - concurrent station + P2P support - support for USB devices RTL8851BU/RTL8852BU - Intel (iwlwifi): - use embedded PNVM in (to be released) FW images to fix compatibility issues - many cleanups (unused FW APIs, PCIe code, WoWLAN) - some FIPS interoperability - MediaTek (mt76): - firmware recovery improvements - more MLO work - Qualcomm/Atheros (ath12k): - fix scan on multi-radio devices - more EHT/Wi-Fi 7 features - encapsulation/decapsulation offload - Broadcom (brcm80211): - support SDIO 43751 device - Bluetooth: - hci_event: add support for handling LE BIG Sync Lost event - ISO: add socket option to report packet seqnum via CMSG - ISO: support SCM_TIMESTAMPING for ISO TS - Bluetooth drivers: - intel_pcie: support Function Level Reset - nxpuart: add support for 4M baudrate - nxpuart: implement powerup sequence, reset, FW dump, and FW loading Signed-off-by: Jakub Kicinski <kuba@kernel.org> -----BEGIN PGP SIGNATURE----- iQIzBAABCgAdFiEE6jPA+I1ugmIBA4hXMUZtbf5SIrsFAmiFgLgACgkQMUZtbf5S IrvafxAAnQRwYBoIG+piCILx6z5pRvBGHkmEQ4AQgSCFuq2eO3ubwMFIqEybfma1 5+QFjUZAV3OgGgKRBS2KGWxtSzdiF+/JGV1VOIN67sX3Mm0a2QgjA4n5CgKL0FPr o6BEzjX5XwG1zvGcBNQ5BZ19xUUKjoZQgTtnea8sZ57Fsp5RtRgmYRqoewNvNk/n uImh0NFsDVb0UeOpSzC34VD9l1dJvLGdui4zJAjno/vpvmT1DkXjoK419J/r52SS X+5WgsfJ6DkjHqVN1tIhhK34yWqBOcwGFZJgEnWHMkFIl2FqRfFKMHyqtfLlVnLA mnIpSyz8Sq2AHtx0TlgZ3At/Ri8p5+yYJgHOXcDKyABa8y8Zf4wrycmr6cV9JLuL z54nLEVnJuvfDVDVJjsLYdJXyhMpZFq6+uAItdxKaw8Ugp/QqG4QtoRj+XIHz4ZW z6OohkCiCzTwEISFK+pSTxPS30eOxq43kCspcvuLiwCCStJBRkRb5GdZA4dm7LA+ 1Od4ADAkHjyrFtBqTyyC2scX8UJ33DlAIpAYyIeS6w9Cj9EXxtp1z33IAAAZ03MW jJwIaJuc8bK2fWKMmiG7ucIXjPo4t//KiWlpkwwqLhPbjZgfDAcxq1AC2TLoqHBL y4EOgKpHDCMAghSyiFIAn2JprGcEt8dp+11B0JRXIn4Pm/eYDH8= =lqbe -----END PGP SIGNATURE----- Merge tag 'net-next-6.17' of git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net-next Pull networking updates from Jakub Kicinski: "Core & protocols: - Wrap datapath globals into net_aligned_data, to avoid false sharing - Preserve MSG_ZEROCOPY in forwarding (e.g. out of a container) - Add SO_INQ and SCM_INQ support to AF_UNIX - Add SIOCINQ support to AF_VSOCK - Add TCP_MAXSEG sockopt to MPTCP - Add IPv6 force_forwarding sysctl to enable forwarding per interface - Make TCP validation of whether packet fully fits in the receive window and the rcv_buf more strict. With increased use of HW aggregation a single "packet" can be multiple 100s of kB - Add MSG_MORE flag to optimize large TCP transmissions via sockmap, improves latency up to 33% for sockmap users - Convert TCP send queue handling from tasklet to BH workque - Improve BPF iteration over TCP sockets to see each socket exactly once - Remove obsolete and unused TCP RFC3517/RFC6675 loss recovery code - Support enabling kernel threads for NAPI processing on per-NAPI instance basis rather than a whole device. Fully stop the kernel NAPI thread when threaded NAPI gets disabled. Previously thread would stick around until ifdown due to tricky synchronization - Allow multicast routing to take effect on locally-generated packets - Add output interface argument for End.X in segment routing - MCTP: add support for gateway routing, improve bind() handling - Don't require rtnl_lock when fetching an IPv6 neighbor over Netlink - Add a new neighbor flag ("extern_valid"), which cedes refresh responsibilities to userspace. This is needed for EVPN multi-homing where a neighbor entry for a multi-homed host needs to be synced across all the VTEPs among which the host is multi-homed - Support NUD_PERMANENT for proxy neighbor entries - Add a new queuing discipline for IETF RFC9332 DualQ Coupled AQM - Add sequence numbers to netconsole messages. Unregister netconsole's console when all net targets are removed. Code refactoring. Add a number of selftests - Align IPSec inbound SA lookup to RFC 4301. Only SPI and protocol should be used for an inbound SA lookup - Support inspecting ref_tracker state via DebugFS - Don't force bonding advertisement frames tx to ~333 ms boundaries. Add broadcast_neighbor option to send ARP/ND on all bonded links - Allow providing upcall pid for the 'execute' command in openvswitch - Remove DCCP support from Netfilter's conntrack - Disallow multiple packet duplications in the queuing layer - Prevent use of deprecated iptables code on PREEMPT_RT Driver API: - Support RSS and hashing configuration over ethtool Netlink - Add dedicated ethtool callbacks for getting and setting hashing fields - Add support for power budget evaluation strategy in PSE / Power-over-Ethernet. Generate Netlink events for overcurrent etc - Support DPLL phase offset monitoring across all device inputs. Support providing clock reference and SYNC over separate DPLL inputs - Support traffic classes in devlink rate API for bandwidth management - Remove rtnl_lock dependency from UDP tunnel port configuration Device drivers: - Add a new Broadcom driver for 800G Ethernet (bnge) - Add a standalone driver for Microchip ZL3073x DPLL - Remove IBM's NETIUCV device driver - Ethernet high-speed NICs: - Broadcom (bnxt): - support zero-copy Tx of DMABUF memory - take page size into account for page pool recycling rings - Intel (100G, ice, idpf): - idpf: XDP and AF_XDP support preparations - idpf: add flow steering - add link_down_events statistic - clean up the TSPLL code - preparations for live VM migration - nVidia/Mellanox: - support zero-copy Rx/Tx interfaces (DMABUF and io_uring) - optimize context memory usage for matchers - expose serial numbers in devlink info - support PCIe congestion metrics - Meta (fbnic): - add 25G, 50G, and 100G link modes to phylink - support dumping FW logs - Marvell/Cavium: - support for CN20K generation of the Octeon chips - Amazon: - add HW clock (without timestamping, just hypervisor time access) - Ethernet virtual: - VirtIO net: - support segmentation of UDP-tunnel-encapsulated packets - Google (gve): - support packet timestamping and clock synchronization - Microsoft vNIC: - add handler for device-originated servicing events - allow dynamic MSI-X vector allocation - support Tx bandwidth clamping - Ethernet NICs consumer, and embedded: - AMD: - amd-xgbe: hardware timestamping and PTP clock support - Broadcom integrated MACs (bcmgenet, bcmasp): - use napi_complete_done() return value to support NAPI polling - add support for re-starting auto-negotiation - Broadcom switches (b53): - support BCM5325 switches - add bcm63xx EPHY power control - Synopsys (stmmac): - lots of code refactoring and cleanups - TI: - icssg-prueth: read firmware-names from device tree - icssg: PRP offload support - Microchip: - lan78xx: convert to PHYLINK for improved PHY and MAC management - ksz: add KSZ8463 switch support - Intel: - support similar queue priority scheme in multi-queue and time-sensitive networking (taprio) - support packet pre-emption in both - RealTek (r8169): - enable EEE at 5Gbps on RTL8126 - Airoha: - add PPPoE offload support - MDIO bus controller for Airoha AN7583 - Ethernet PHYs: - support for the IPQ5018 internal GE PHY - micrel KSZ9477 switch-integrated PHYs: - add MDI/MDI-X control support - add RX error counters - add cable test support - add Signal Quality Indicator (SQI) reporting - dp83tg720: improve reset handling and reduce link recovery time - support bcm54811 (and its MII-Lite interface type) - air_en8811h: support resume/suspend - support PHY counters for QCA807x and QCA808x - support WoL for QCA807x - CAN drivers: - rcar_canfd: support for Transceiver Delay Compensation - kvaser: report FW versions via devlink dev info - WiFi: - extended regulatory info support (6 GHz) - add statistics and beacon monitor for Multi-Link Operation (MLO) - support S1G aggregation, improve S1G support - add Radio Measurement action fields - support per-radio RTS threshold - some work around how FIPS affects wifi, which was wrong (RC4 is used by TKIP, not only WEP) - improvements for unsolicited probe response handling - WiFi drivers: - RealTek (rtw88): - IBSS mode for SDIO devices - RealTek (rtw89): - BT coexistence for MLO/WiFi7 - concurrent station + P2P support - support for USB devices RTL8851BU/RTL8852BU - Intel (iwlwifi): - use embedded PNVM in (to be released) FW images to fix compatibility issues - many cleanups (unused FW APIs, PCIe code, WoWLAN) - some FIPS interoperability - MediaTek (mt76): - firmware recovery improvements - more MLO work - Qualcomm/Atheros (ath12k): - fix scan on multi-radio devices - more EHT/Wi-Fi 7 features - encapsulation/decapsulation offload - Broadcom (brcm80211): - support SDIO 43751 device - Bluetooth: - hci_event: add support for handling LE BIG Sync Lost event - ISO: add socket option to report packet seqnum via CMSG - ISO: support SCM_TIMESTAMPING for ISO TS - Bluetooth drivers: - intel_pcie: support Function Level Reset - nxpuart: add support for 4M baudrate - nxpuart: implement powerup sequence, reset, FW dump, and FW loading" * tag 'net-next-6.17' of git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net-next: (1742 commits) dpll: zl3073x: Fix build failure selftests: bpf: fix legacy netfilter options ipv6: annotate data-races around rt->fib6_nsiblings ipv6: fix possible infinite loop in fib6_info_uses_dev() ipv6: prevent infinite loop in rt6_nlmsg_size() ipv6: add a retry logic in net6_rt_notify() vrf: Drop existing dst reference in vrf_ip6_input_dst net/sched: taprio: align entry index attr validation with mqprio net: fsl_pq_mdio: use dev_err_probe selftests: rtnetlink.sh: remove esp4_offload after test vsock: remove unnecessary null check in vsock_getname() igb: xsk: solve negative overflow of nb_pkts in zerocopy mode stmmac: xsk: fix negative overflow of budget in zerocopy mode dt-bindings: ieee802154: Convert at86rf230.txt yaml format net: dsa: microchip: Disable PTP function of KSZ8463 net: dsa: microchip: Setup fiber ports for KSZ8463 net: dsa: microchip: Write switch MAC address differently for KSZ8463 net: dsa: microchip: Use different registers for KSZ8463 net: dsa: microchip: Add KSZ8463 switch support to KSZ DSA driver dt-bindings: net: dsa: microchip: Add KSZ8463 switch support ...
913 lines
31 KiB
Rust
913 lines
31 KiB
Rust
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
// Copyright (C) 2023 FUJITA Tomonori <fujita.tomonori@gmail.com>
|
|
|
|
//! Network PHY device.
|
|
//!
|
|
//! C headers: [`include/linux/phy.h`](srctree/include/linux/phy.h).
|
|
|
|
use crate::{device_id::RawDeviceId, error::*, prelude::*, types::Opaque};
|
|
use core::{marker::PhantomData, ptr::addr_of_mut};
|
|
|
|
pub mod reg;
|
|
|
|
/// PHY state machine states.
|
|
///
|
|
/// Corresponds to the kernel's [`enum phy_state`].
|
|
///
|
|
/// Some of PHY drivers access to the state of PHY's software state machine.
|
|
///
|
|
/// [`enum phy_state`]: srctree/include/linux/phy.h
|
|
#[derive(PartialEq, Eq)]
|
|
pub enum DeviceState {
|
|
/// PHY device and driver are not ready for anything.
|
|
Down,
|
|
/// PHY is ready to send and receive packets.
|
|
Ready,
|
|
/// PHY is up, but no polling or interrupts are done.
|
|
Halted,
|
|
/// PHY is up, but is in an error state.
|
|
Error,
|
|
/// PHY and attached device are ready to do work.
|
|
Up,
|
|
/// PHY is currently running.
|
|
Running,
|
|
/// PHY is up, but not currently plugged in.
|
|
NoLink,
|
|
/// PHY is performing a cable test.
|
|
CableTest,
|
|
}
|
|
|
|
/// A mode of Ethernet communication.
|
|
///
|
|
/// PHY drivers get duplex information from hardware and update the current state.
|
|
pub enum DuplexMode {
|
|
/// PHY is in full-duplex mode.
|
|
Full,
|
|
/// PHY is in half-duplex mode.
|
|
Half,
|
|
/// PHY is in unknown duplex mode.
|
|
Unknown,
|
|
}
|
|
|
|
/// An instance of a PHY device.
|
|
///
|
|
/// Wraps the kernel's [`struct phy_device`].
|
|
///
|
|
/// A [`Device`] instance is created when a callback in [`Driver`] is executed. A PHY driver
|
|
/// executes [`Driver`]'s methods during the callback.
|
|
///
|
|
/// # Invariants
|
|
///
|
|
/// - Referencing a `phy_device` using this struct asserts that you are in
|
|
/// a context where all methods defined on this struct are safe to call.
|
|
/// - This struct always has a valid `self.0.mdio.dev`.
|
|
///
|
|
/// [`struct phy_device`]: srctree/include/linux/phy.h
|
|
// During the calls to most functions in [`Driver`], the C side (`PHYLIB`) holds a lock that is
|
|
// unique for every instance of [`Device`]. `PHYLIB` uses a different serialization technique for
|
|
// [`Driver::resume`] and [`Driver::suspend`]: `PHYLIB` updates `phy_device`'s state with
|
|
// the lock held, thus guaranteeing that [`Driver::resume`] has exclusive access to the instance.
|
|
// [`Driver::resume`] and [`Driver::suspend`] also are called where only one thread can access
|
|
// to the instance.
|
|
#[repr(transparent)]
|
|
pub struct Device(Opaque<bindings::phy_device>);
|
|
|
|
impl Device {
|
|
/// Creates a new [`Device`] instance from a raw pointer.
|
|
///
|
|
/// # Safety
|
|
///
|
|
/// For the duration of `'a`,
|
|
/// - the pointer must point at a valid `phy_device`, and the caller
|
|
/// must be in a context where all methods defined on this struct
|
|
/// are safe to call.
|
|
/// - `(*ptr).mdio.dev` must be a valid.
|
|
unsafe fn from_raw<'a>(ptr: *mut bindings::phy_device) -> &'a mut Self {
|
|
// CAST: `Self` is a `repr(transparent)` wrapper around `bindings::phy_device`.
|
|
let ptr = ptr.cast::<Self>();
|
|
// SAFETY: by the function requirements the pointer is valid and we have unique access for
|
|
// the duration of `'a`.
|
|
unsafe { &mut *ptr }
|
|
}
|
|
|
|
/// Gets the id of the PHY.
|
|
pub fn phy_id(&self) -> u32 {
|
|
let phydev = self.0.get();
|
|
// SAFETY: The struct invariant ensures that we may access
|
|
// this field without additional synchronization.
|
|
unsafe { (*phydev).phy_id }
|
|
}
|
|
|
|
/// Gets the state of PHY state machine states.
|
|
pub fn state(&self) -> DeviceState {
|
|
let phydev = self.0.get();
|
|
// SAFETY: The struct invariant ensures that we may access
|
|
// this field without additional synchronization.
|
|
let state = unsafe { (*phydev).state };
|
|
// TODO: this conversion code will be replaced with automatically generated code by bindgen
|
|
// when it becomes possible.
|
|
match state {
|
|
bindings::phy_state_PHY_DOWN => DeviceState::Down,
|
|
bindings::phy_state_PHY_READY => DeviceState::Ready,
|
|
bindings::phy_state_PHY_HALTED => DeviceState::Halted,
|
|
bindings::phy_state_PHY_ERROR => DeviceState::Error,
|
|
bindings::phy_state_PHY_UP => DeviceState::Up,
|
|
bindings::phy_state_PHY_RUNNING => DeviceState::Running,
|
|
bindings::phy_state_PHY_NOLINK => DeviceState::NoLink,
|
|
bindings::phy_state_PHY_CABLETEST => DeviceState::CableTest,
|
|
_ => DeviceState::Error,
|
|
}
|
|
}
|
|
|
|
/// Gets the current link state.
|
|
///
|
|
/// It returns true if the link is up.
|
|
pub fn is_link_up(&self) -> bool {
|
|
const LINK_IS_UP: u64 = 1;
|
|
// TODO: the code to access to the bit field will be replaced with automatically
|
|
// generated code by bindgen when it becomes possible.
|
|
// SAFETY: The struct invariant ensures that we may access
|
|
// this field without additional synchronization.
|
|
let bit_field = unsafe { &(*self.0.get())._bitfield_1 };
|
|
bit_field.get(14, 1) == LINK_IS_UP
|
|
}
|
|
|
|
/// Gets the current auto-negotiation configuration.
|
|
///
|
|
/// It returns true if auto-negotiation is enabled.
|
|
pub fn is_autoneg_enabled(&self) -> bool {
|
|
// TODO: the code to access to the bit field will be replaced with automatically
|
|
// generated code by bindgen when it becomes possible.
|
|
// SAFETY: The struct invariant ensures that we may access
|
|
// this field without additional synchronization.
|
|
let bit_field = unsafe { &(*self.0.get())._bitfield_1 };
|
|
bit_field.get(13, 1) == bindings::AUTONEG_ENABLE as u64
|
|
}
|
|
|
|
/// Gets the current auto-negotiation state.
|
|
///
|
|
/// It returns true if auto-negotiation is completed.
|
|
pub fn is_autoneg_completed(&self) -> bool {
|
|
const AUTONEG_COMPLETED: u64 = 1;
|
|
// TODO: the code to access to the bit field will be replaced with automatically
|
|
// generated code by bindgen when it becomes possible.
|
|
// SAFETY: The struct invariant ensures that we may access
|
|
// this field without additional synchronization.
|
|
let bit_field = unsafe { &(*self.0.get())._bitfield_1 };
|
|
bit_field.get(15, 1) == AUTONEG_COMPLETED
|
|
}
|
|
|
|
/// Sets the speed of the PHY.
|
|
pub fn set_speed(&mut self, speed: u32) {
|
|
let phydev = self.0.get();
|
|
// SAFETY: The struct invariant ensures that we may access
|
|
// this field without additional synchronization.
|
|
unsafe { (*phydev).speed = speed as c_int };
|
|
}
|
|
|
|
/// Sets duplex mode.
|
|
pub fn set_duplex(&mut self, mode: DuplexMode) {
|
|
let phydev = self.0.get();
|
|
let v = match mode {
|
|
DuplexMode::Full => bindings::DUPLEX_FULL,
|
|
DuplexMode::Half => bindings::DUPLEX_HALF,
|
|
DuplexMode::Unknown => bindings::DUPLEX_UNKNOWN,
|
|
};
|
|
// SAFETY: The struct invariant ensures that we may access
|
|
// this field without additional synchronization.
|
|
unsafe { (*phydev).duplex = v as c_int };
|
|
}
|
|
|
|
/// Reads a PHY register.
|
|
// This function reads a hardware register and updates the stats so takes `&mut self`.
|
|
pub fn read<R: reg::Register>(&mut self, reg: R) -> Result<u16> {
|
|
reg.read(self)
|
|
}
|
|
|
|
/// Writes a PHY register.
|
|
pub fn write<R: reg::Register>(&mut self, reg: R, val: u16) -> Result {
|
|
reg.write(self, val)
|
|
}
|
|
|
|
/// Reads a paged register.
|
|
pub fn read_paged(&mut self, page: u16, regnum: u16) -> Result<u16> {
|
|
let phydev = self.0.get();
|
|
// SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
|
|
// So it's just an FFI call.
|
|
let ret = unsafe { bindings::phy_read_paged(phydev, page.into(), regnum.into()) };
|
|
if ret < 0 {
|
|
Err(Error::from_errno(ret))
|
|
} else {
|
|
Ok(ret as u16)
|
|
}
|
|
}
|
|
|
|
/// Resolves the advertisements into PHY settings.
|
|
pub fn resolve_aneg_linkmode(&mut self) {
|
|
let phydev = self.0.get();
|
|
// SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
|
|
// So it's just an FFI call.
|
|
unsafe { bindings::phy_resolve_aneg_linkmode(phydev) };
|
|
}
|
|
|
|
/// Executes software reset the PHY via `BMCR_RESET` bit.
|
|
pub fn genphy_soft_reset(&mut self) -> Result {
|
|
let phydev = self.0.get();
|
|
// SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
|
|
// So it's just an FFI call.
|
|
to_result(unsafe { bindings::genphy_soft_reset(phydev) })
|
|
}
|
|
|
|
/// Initializes the PHY.
|
|
pub fn init_hw(&mut self) -> Result {
|
|
let phydev = self.0.get();
|
|
// SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
|
|
// So it's just an FFI call.
|
|
to_result(unsafe { bindings::phy_init_hw(phydev) })
|
|
}
|
|
|
|
/// Starts auto-negotiation.
|
|
pub fn start_aneg(&mut self) -> Result {
|
|
let phydev = self.0.get();
|
|
// SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
|
|
// So it's just an FFI call.
|
|
to_result(unsafe { bindings::_phy_start_aneg(phydev) })
|
|
}
|
|
|
|
/// Resumes the PHY via `BMCR_PDOWN` bit.
|
|
pub fn genphy_resume(&mut self) -> Result {
|
|
let phydev = self.0.get();
|
|
// SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
|
|
// So it's just an FFI call.
|
|
to_result(unsafe { bindings::genphy_resume(phydev) })
|
|
}
|
|
|
|
/// Suspends the PHY via `BMCR_PDOWN` bit.
|
|
pub fn genphy_suspend(&mut self) -> Result {
|
|
let phydev = self.0.get();
|
|
// SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
|
|
// So it's just an FFI call.
|
|
to_result(unsafe { bindings::genphy_suspend(phydev) })
|
|
}
|
|
|
|
/// Checks the link status and updates current link state.
|
|
pub fn genphy_read_status<R: reg::Register>(&mut self) -> Result<u16> {
|
|
R::read_status(self)
|
|
}
|
|
|
|
/// Updates the link status.
|
|
pub fn genphy_update_link(&mut self) -> Result {
|
|
let phydev = self.0.get();
|
|
// SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
|
|
// So it's just an FFI call.
|
|
to_result(unsafe { bindings::genphy_update_link(phydev) })
|
|
}
|
|
|
|
/// Reads link partner ability.
|
|
pub fn genphy_read_lpa(&mut self) -> Result {
|
|
let phydev = self.0.get();
|
|
// SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
|
|
// So it's just an FFI call.
|
|
to_result(unsafe { bindings::genphy_read_lpa(phydev) })
|
|
}
|
|
|
|
/// Reads PHY abilities.
|
|
pub fn genphy_read_abilities(&mut self) -> Result {
|
|
let phydev = self.0.get();
|
|
// SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
|
|
// So it's just an FFI call.
|
|
to_result(unsafe { bindings::genphy_read_abilities(phydev) })
|
|
}
|
|
}
|
|
|
|
impl AsRef<kernel::device::Device> for Device {
|
|
fn as_ref(&self) -> &kernel::device::Device {
|
|
let phydev = self.0.get();
|
|
// SAFETY: The struct invariant ensures that `mdio.dev` is valid.
|
|
unsafe { kernel::device::Device::from_raw(addr_of_mut!((*phydev).mdio.dev)) }
|
|
}
|
|
}
|
|
|
|
/// Defines certain other features this PHY supports (like interrupts).
|
|
///
|
|
/// These flag values are used in [`Driver::FLAGS`].
|
|
pub mod flags {
|
|
/// PHY is internal.
|
|
pub const IS_INTERNAL: u32 = bindings::PHY_IS_INTERNAL;
|
|
/// PHY needs to be reset after the refclk is enabled.
|
|
pub const RST_AFTER_CLK_EN: u32 = bindings::PHY_RST_AFTER_CLK_EN;
|
|
/// Polling is used to detect PHY status changes.
|
|
pub const POLL_CABLE_TEST: u32 = bindings::PHY_POLL_CABLE_TEST;
|
|
/// Don't suspend.
|
|
pub const ALWAYS_CALL_SUSPEND: u32 = bindings::PHY_ALWAYS_CALL_SUSPEND;
|
|
}
|
|
|
|
/// An adapter for the registration of a PHY driver.
|
|
struct Adapter<T: Driver> {
|
|
_p: PhantomData<T>,
|
|
}
|
|
|
|
impl<T: Driver> Adapter<T> {
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn soft_reset_callback(phydev: *mut bindings::phy_device) -> c_int {
|
|
from_result(|| {
|
|
// SAFETY: This callback is called only in contexts
|
|
// where we hold `phy_device->lock`, so the accessors on
|
|
// `Device` are okay to call.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
T::soft_reset(dev)?;
|
|
Ok(0)
|
|
})
|
|
}
|
|
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn probe_callback(phydev: *mut bindings::phy_device) -> c_int {
|
|
from_result(|| {
|
|
// SAFETY: This callback is called only in contexts
|
|
// where we can exclusively access `phy_device` because
|
|
// it's not published yet, so the accessors on `Device` are okay
|
|
// to call.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
T::probe(dev)?;
|
|
Ok(0)
|
|
})
|
|
}
|
|
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn get_features_callback(phydev: *mut bindings::phy_device) -> c_int {
|
|
from_result(|| {
|
|
// SAFETY: This callback is called only in contexts
|
|
// where we hold `phy_device->lock`, so the accessors on
|
|
// `Device` are okay to call.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
T::get_features(dev)?;
|
|
Ok(0)
|
|
})
|
|
}
|
|
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn suspend_callback(phydev: *mut bindings::phy_device) -> c_int {
|
|
from_result(|| {
|
|
// SAFETY: The C core code ensures that the accessors on
|
|
// `Device` are okay to call even though `phy_device->lock`
|
|
// might not be held.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
T::suspend(dev)?;
|
|
Ok(0)
|
|
})
|
|
}
|
|
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn resume_callback(phydev: *mut bindings::phy_device) -> c_int {
|
|
from_result(|| {
|
|
// SAFETY: The C core code ensures that the accessors on
|
|
// `Device` are okay to call even though `phy_device->lock`
|
|
// might not be held.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
T::resume(dev)?;
|
|
Ok(0)
|
|
})
|
|
}
|
|
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn config_aneg_callback(phydev: *mut bindings::phy_device) -> c_int {
|
|
from_result(|| {
|
|
// SAFETY: This callback is called only in contexts
|
|
// where we hold `phy_device->lock`, so the accessors on
|
|
// `Device` are okay to call.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
T::config_aneg(dev)?;
|
|
Ok(0)
|
|
})
|
|
}
|
|
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn read_status_callback(phydev: *mut bindings::phy_device) -> c_int {
|
|
from_result(|| {
|
|
// SAFETY: This callback is called only in contexts
|
|
// where we hold `phy_device->lock`, so the accessors on
|
|
// `Device` are okay to call.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
T::read_status(dev)?;
|
|
Ok(0)
|
|
})
|
|
}
|
|
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn match_phy_device_callback(
|
|
phydev: *mut bindings::phy_device,
|
|
_phydrv: *const bindings::phy_driver,
|
|
) -> c_int {
|
|
// SAFETY: This callback is called only in contexts
|
|
// where we hold `phy_device->lock`, so the accessors on
|
|
// `Device` are okay to call.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
T::match_phy_device(dev) as i32
|
|
}
|
|
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn read_mmd_callback(
|
|
phydev: *mut bindings::phy_device,
|
|
devnum: i32,
|
|
regnum: u16,
|
|
) -> i32 {
|
|
from_result(|| {
|
|
// SAFETY: This callback is called only in contexts
|
|
// where we hold `phy_device->lock`, so the accessors on
|
|
// `Device` are okay to call.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
// CAST: the C side verifies devnum < 32.
|
|
let ret = T::read_mmd(dev, devnum as u8, regnum)?;
|
|
Ok(ret.into())
|
|
})
|
|
}
|
|
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn write_mmd_callback(
|
|
phydev: *mut bindings::phy_device,
|
|
devnum: i32,
|
|
regnum: u16,
|
|
val: u16,
|
|
) -> i32 {
|
|
from_result(|| {
|
|
// SAFETY: This callback is called only in contexts
|
|
// where we hold `phy_device->lock`, so the accessors on
|
|
// `Device` are okay to call.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
T::write_mmd(dev, devnum as u8, regnum, val)?;
|
|
Ok(0)
|
|
})
|
|
}
|
|
|
|
/// # Safety
|
|
///
|
|
/// `phydev` must be passed by the corresponding callback in `phy_driver`.
|
|
unsafe extern "C" fn link_change_notify_callback(phydev: *mut bindings::phy_device) {
|
|
// SAFETY: This callback is called only in contexts
|
|
// where we hold `phy_device->lock`, so the accessors on
|
|
// `Device` are okay to call.
|
|
let dev = unsafe { Device::from_raw(phydev) };
|
|
T::link_change_notify(dev);
|
|
}
|
|
}
|
|
|
|
/// Driver structure for a particular PHY type.
|
|
///
|
|
/// Wraps the kernel's [`struct phy_driver`].
|
|
/// This is used to register a driver for a particular PHY type with the kernel.
|
|
///
|
|
/// # Invariants
|
|
///
|
|
/// `self.0` is always in a valid state.
|
|
///
|
|
/// [`struct phy_driver`]: srctree/include/linux/phy.h
|
|
#[repr(transparent)]
|
|
pub struct DriverVTable(Opaque<bindings::phy_driver>);
|
|
|
|
// SAFETY: `DriverVTable` doesn't expose any &self method to access internal data, so it's safe to
|
|
// share `&DriverVTable` across execution context boundaries.
|
|
unsafe impl Sync for DriverVTable {}
|
|
|
|
/// Creates a [`DriverVTable`] instance from [`Driver`].
|
|
///
|
|
/// This is used by [`module_phy_driver`] macro to create a static array of `phy_driver`.
|
|
///
|
|
/// [`module_phy_driver`]: crate::module_phy_driver
|
|
pub const fn create_phy_driver<T: Driver>() -> DriverVTable {
|
|
// INVARIANT: All the fields of `struct phy_driver` are initialized properly.
|
|
DriverVTable(Opaque::new(bindings::phy_driver {
|
|
name: T::NAME.as_char_ptr().cast_mut(),
|
|
flags: T::FLAGS,
|
|
phy_id: T::PHY_DEVICE_ID.id(),
|
|
phy_id_mask: T::PHY_DEVICE_ID.mask_as_int(),
|
|
soft_reset: if T::HAS_SOFT_RESET {
|
|
Some(Adapter::<T>::soft_reset_callback)
|
|
} else {
|
|
None
|
|
},
|
|
probe: if T::HAS_PROBE {
|
|
Some(Adapter::<T>::probe_callback)
|
|
} else {
|
|
None
|
|
},
|
|
get_features: if T::HAS_GET_FEATURES {
|
|
Some(Adapter::<T>::get_features_callback)
|
|
} else {
|
|
None
|
|
},
|
|
match_phy_device: if T::HAS_MATCH_PHY_DEVICE {
|
|
Some(Adapter::<T>::match_phy_device_callback)
|
|
} else {
|
|
None
|
|
},
|
|
suspend: if T::HAS_SUSPEND {
|
|
Some(Adapter::<T>::suspend_callback)
|
|
} else {
|
|
None
|
|
},
|
|
resume: if T::HAS_RESUME {
|
|
Some(Adapter::<T>::resume_callback)
|
|
} else {
|
|
None
|
|
},
|
|
config_aneg: if T::HAS_CONFIG_ANEG {
|
|
Some(Adapter::<T>::config_aneg_callback)
|
|
} else {
|
|
None
|
|
},
|
|
read_status: if T::HAS_READ_STATUS {
|
|
Some(Adapter::<T>::read_status_callback)
|
|
} else {
|
|
None
|
|
},
|
|
read_mmd: if T::HAS_READ_MMD {
|
|
Some(Adapter::<T>::read_mmd_callback)
|
|
} else {
|
|
None
|
|
},
|
|
write_mmd: if T::HAS_WRITE_MMD {
|
|
Some(Adapter::<T>::write_mmd_callback)
|
|
} else {
|
|
None
|
|
},
|
|
link_change_notify: if T::HAS_LINK_CHANGE_NOTIFY {
|
|
Some(Adapter::<T>::link_change_notify_callback)
|
|
} else {
|
|
None
|
|
},
|
|
// SAFETY: The rest is zeroed out to initialize `struct phy_driver`,
|
|
// sets `Option<&F>` to be `None`.
|
|
..unsafe { core::mem::MaybeUninit::<bindings::phy_driver>::zeroed().assume_init() }
|
|
}))
|
|
}
|
|
|
|
/// Driver implementation for a particular PHY type.
|
|
///
|
|
/// This trait is used to create a [`DriverVTable`].
|
|
#[vtable]
|
|
pub trait Driver {
|
|
/// Defines certain other features this PHY supports.
|
|
/// It is a combination of the flags in the [`flags`] module.
|
|
const FLAGS: u32 = 0;
|
|
|
|
/// The friendly name of this PHY type.
|
|
const NAME: &'static CStr;
|
|
|
|
/// This driver only works for PHYs with IDs which match this field.
|
|
/// The default id and mask are zero.
|
|
const PHY_DEVICE_ID: DeviceId = DeviceId::new_with_custom_mask(0, 0);
|
|
|
|
/// Issues a PHY software reset.
|
|
fn soft_reset(_dev: &mut Device) -> Result {
|
|
build_error!(VTABLE_DEFAULT_ERROR)
|
|
}
|
|
|
|
/// Sets up device-specific structures during discovery.
|
|
fn probe(_dev: &mut Device) -> Result {
|
|
build_error!(VTABLE_DEFAULT_ERROR)
|
|
}
|
|
|
|
/// Probes the hardware to determine what abilities it has.
|
|
fn get_features(_dev: &mut Device) -> Result {
|
|
build_error!(VTABLE_DEFAULT_ERROR)
|
|
}
|
|
|
|
/// Returns true if this is a suitable driver for the given phydev.
|
|
/// If not implemented, matching is based on [`Driver::PHY_DEVICE_ID`].
|
|
fn match_phy_device(_dev: &Device) -> bool {
|
|
false
|
|
}
|
|
|
|
/// Configures the advertisement and resets auto-negotiation
|
|
/// if auto-negotiation is enabled.
|
|
fn config_aneg(_dev: &mut Device) -> Result {
|
|
build_error!(VTABLE_DEFAULT_ERROR)
|
|
}
|
|
|
|
/// Determines the negotiated speed and duplex.
|
|
fn read_status(_dev: &mut Device) -> Result<u16> {
|
|
build_error!(VTABLE_DEFAULT_ERROR)
|
|
}
|
|
|
|
/// Suspends the hardware, saving state if needed.
|
|
fn suspend(_dev: &mut Device) -> Result {
|
|
build_error!(VTABLE_DEFAULT_ERROR)
|
|
}
|
|
|
|
/// Resumes the hardware, restoring state if needed.
|
|
fn resume(_dev: &mut Device) -> Result {
|
|
build_error!(VTABLE_DEFAULT_ERROR)
|
|
}
|
|
|
|
/// Overrides the default MMD read function for reading a MMD register.
|
|
fn read_mmd(_dev: &mut Device, _devnum: u8, _regnum: u16) -> Result<u16> {
|
|
build_error!(VTABLE_DEFAULT_ERROR)
|
|
}
|
|
|
|
/// Overrides the default MMD write function for writing a MMD register.
|
|
fn write_mmd(_dev: &mut Device, _devnum: u8, _regnum: u16, _val: u16) -> Result {
|
|
build_error!(VTABLE_DEFAULT_ERROR)
|
|
}
|
|
|
|
/// Callback for notification of link change.
|
|
fn link_change_notify(_dev: &mut Device) {}
|
|
}
|
|
|
|
/// Registration structure for PHY drivers.
|
|
///
|
|
/// Registers [`DriverVTable`] instances with the kernel. They will be unregistered when dropped.
|
|
///
|
|
/// # Invariants
|
|
///
|
|
/// The `drivers` slice are currently registered to the kernel via `phy_drivers_register`.
|
|
pub struct Registration {
|
|
drivers: Pin<&'static mut [DriverVTable]>,
|
|
}
|
|
|
|
// SAFETY: The only action allowed in a `Registration` instance is dropping it, which is safe to do
|
|
// from any thread because `phy_drivers_unregister` can be called from any thread context.
|
|
unsafe impl Send for Registration {}
|
|
|
|
impl Registration {
|
|
/// Registers a PHY driver.
|
|
pub fn register(
|
|
module: &'static crate::ThisModule,
|
|
drivers: Pin<&'static mut [DriverVTable]>,
|
|
) -> Result<Self> {
|
|
if drivers.is_empty() {
|
|
return Err(code::EINVAL);
|
|
}
|
|
// SAFETY: The type invariants of [`DriverVTable`] ensure that all elements of
|
|
// the `drivers` slice are initialized properly. `drivers` will not be moved.
|
|
// So it's just an FFI call.
|
|
to_result(unsafe {
|
|
bindings::phy_drivers_register(drivers[0].0.get(), drivers.len().try_into()?, module.0)
|
|
})?;
|
|
// INVARIANT: The `drivers` slice is successfully registered to the kernel via `phy_drivers_register`.
|
|
Ok(Registration { drivers })
|
|
}
|
|
}
|
|
|
|
impl Drop for Registration {
|
|
fn drop(&mut self) {
|
|
// SAFETY: The type invariants guarantee that `self.drivers` is valid.
|
|
// So it's just an FFI call.
|
|
unsafe {
|
|
bindings::phy_drivers_unregister(self.drivers[0].0.get(), self.drivers.len() as i32)
|
|
};
|
|
}
|
|
}
|
|
|
|
/// An identifier for PHY devices on an MDIO/MII bus.
|
|
///
|
|
/// Represents the kernel's `struct mdio_device_id`. This is used to find an appropriate
|
|
/// PHY driver.
|
|
#[repr(transparent)]
|
|
#[derive(Clone, Copy)]
|
|
pub struct DeviceId(bindings::mdio_device_id);
|
|
|
|
impl DeviceId {
|
|
/// Creates a new instance with the exact match mask.
|
|
pub const fn new_with_exact_mask(id: u32) -> Self {
|
|
Self(bindings::mdio_device_id {
|
|
phy_id: id,
|
|
phy_id_mask: DeviceMask::Exact.as_int(),
|
|
})
|
|
}
|
|
|
|
/// Creates a new instance with the model match mask.
|
|
pub const fn new_with_model_mask(id: u32) -> Self {
|
|
Self(bindings::mdio_device_id {
|
|
phy_id: id,
|
|
phy_id_mask: DeviceMask::Model.as_int(),
|
|
})
|
|
}
|
|
|
|
/// Creates a new instance with the vendor match mask.
|
|
pub const fn new_with_vendor_mask(id: u32) -> Self {
|
|
Self(bindings::mdio_device_id {
|
|
phy_id: id,
|
|
phy_id_mask: DeviceMask::Vendor.as_int(),
|
|
})
|
|
}
|
|
|
|
/// Creates a new instance with a custom match mask.
|
|
pub const fn new_with_custom_mask(id: u32, mask: u32) -> Self {
|
|
Self(bindings::mdio_device_id {
|
|
phy_id: id,
|
|
phy_id_mask: DeviceMask::Custom(mask).as_int(),
|
|
})
|
|
}
|
|
|
|
/// Creates a new instance from [`Driver`].
|
|
pub const fn new_with_driver<T: Driver>() -> Self {
|
|
T::PHY_DEVICE_ID
|
|
}
|
|
|
|
/// Get the MDIO device's PHY ID.
|
|
pub const fn id(&self) -> u32 {
|
|
self.0.phy_id
|
|
}
|
|
|
|
/// Get the MDIO device's match mask.
|
|
pub const fn mask_as_int(&self) -> u32 {
|
|
self.0.phy_id_mask
|
|
}
|
|
|
|
// macro use only
|
|
#[doc(hidden)]
|
|
pub const fn mdio_device_id(&self) -> bindings::mdio_device_id {
|
|
self.0
|
|
}
|
|
}
|
|
|
|
// SAFETY: `DeviceId` is a `#[repr(transparent)]` wrapper of `struct mdio_device_id`
|
|
// and does not add additional invariants, so it's safe to transmute to `RawType`.
|
|
unsafe impl RawDeviceId for DeviceId {
|
|
type RawType = bindings::mdio_device_id;
|
|
}
|
|
|
|
enum DeviceMask {
|
|
Exact,
|
|
Model,
|
|
Vendor,
|
|
Custom(u32),
|
|
}
|
|
|
|
impl DeviceMask {
|
|
const MASK_EXACT: u32 = !0;
|
|
const MASK_MODEL: u32 = !0 << 4;
|
|
const MASK_VENDOR: u32 = !0 << 10;
|
|
|
|
const fn as_int(&self) -> u32 {
|
|
match self {
|
|
DeviceMask::Exact => Self::MASK_EXACT,
|
|
DeviceMask::Model => Self::MASK_MODEL,
|
|
DeviceMask::Vendor => Self::MASK_VENDOR,
|
|
DeviceMask::Custom(mask) => *mask,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Declares a kernel module for PHYs drivers.
|
|
///
|
|
/// This creates a static array of kernel's `struct phy_driver` and registers it.
|
|
/// This also corresponds to the kernel's `MODULE_DEVICE_TABLE` macro, which embeds the information
|
|
/// for module loading into the module binary file. Every driver needs an entry in `device_table`.
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```
|
|
/// # mod module_phy_driver_sample {
|
|
/// use kernel::c_str;
|
|
/// use kernel::net::phy::{self, DeviceId};
|
|
/// use kernel::prelude::*;
|
|
///
|
|
/// kernel::module_phy_driver! {
|
|
/// drivers: [PhySample],
|
|
/// device_table: [
|
|
/// DeviceId::new_with_driver::<PhySample>()
|
|
/// ],
|
|
/// name: "rust_sample_phy",
|
|
/// authors: ["Rust for Linux Contributors"],
|
|
/// description: "Rust sample PHYs driver",
|
|
/// license: "GPL",
|
|
/// }
|
|
///
|
|
/// struct PhySample;
|
|
///
|
|
/// #[vtable]
|
|
/// impl phy::Driver for PhySample {
|
|
/// const NAME: &'static CStr = c_str!("PhySample");
|
|
/// const PHY_DEVICE_ID: phy::DeviceId = phy::DeviceId::new_with_exact_mask(0x00000001);
|
|
/// }
|
|
/// # }
|
|
/// ```
|
|
///
|
|
/// This expands to the following code:
|
|
///
|
|
/// ```ignore
|
|
/// use kernel::c_str;
|
|
/// use kernel::net::phy::{self, DeviceId};
|
|
/// use kernel::prelude::*;
|
|
///
|
|
/// struct Module {
|
|
/// _reg: ::kernel::net::phy::Registration,
|
|
/// }
|
|
///
|
|
/// module! {
|
|
/// type: Module,
|
|
/// name: "rust_sample_phy",
|
|
/// authors: ["Rust for Linux Contributors"],
|
|
/// description: "Rust sample PHYs driver",
|
|
/// license: "GPL",
|
|
/// }
|
|
///
|
|
/// struct PhySample;
|
|
///
|
|
/// #[vtable]
|
|
/// impl phy::Driver for PhySample {
|
|
/// const NAME: &'static CStr = c_str!("PhySample");
|
|
/// const PHY_DEVICE_ID: phy::DeviceId = phy::DeviceId::new_with_exact_mask(0x00000001);
|
|
/// }
|
|
///
|
|
/// const _: () = {
|
|
/// static mut DRIVERS: [::kernel::net::phy::DriverVTable; 1] =
|
|
/// [::kernel::net::phy::create_phy_driver::<PhySample>()];
|
|
///
|
|
/// impl ::kernel::Module for Module {
|
|
/// fn init(module: &'static ::kernel::ThisModule) -> Result<Self> {
|
|
/// let drivers = unsafe { &mut DRIVERS };
|
|
/// let mut reg = ::kernel::net::phy::Registration::register(
|
|
/// module,
|
|
/// ::core::pin::Pin::static_mut(drivers),
|
|
/// )?;
|
|
/// Ok(Module { _reg: reg })
|
|
/// }
|
|
/// }
|
|
/// };
|
|
///
|
|
/// const N: usize = 1;
|
|
///
|
|
/// const TABLE: ::kernel::device_id::IdArray<::kernel::net::phy::DeviceId, (), N> =
|
|
/// ::kernel::device_id::IdArray::new_without_index([
|
|
/// ::kernel::net::phy::DeviceId(
|
|
/// ::kernel::bindings::mdio_device_id {
|
|
/// phy_id: 0x00000001,
|
|
/// phy_id_mask: 0xffffffff,
|
|
/// }),
|
|
/// ]);
|
|
///
|
|
/// ::kernel::module_device_table!("mdio", phydev, TABLE);
|
|
/// ```
|
|
#[macro_export]
|
|
macro_rules! module_phy_driver {
|
|
(@replace_expr $_t:tt $sub:expr) => {$sub};
|
|
|
|
(@count_devices $($x:expr),*) => {
|
|
0usize $(+ $crate::module_phy_driver!(@replace_expr $x 1usize))*
|
|
};
|
|
|
|
(@device_table [$($dev:expr),+]) => {
|
|
const N: usize = $crate::module_phy_driver!(@count_devices $($dev),+);
|
|
|
|
const TABLE: $crate::device_id::IdArray<$crate::net::phy::DeviceId, (), N> =
|
|
$crate::device_id::IdArray::new_without_index([ $(($dev,())),+, ]);
|
|
|
|
$crate::module_device_table!("mdio", phydev, TABLE);
|
|
};
|
|
|
|
(drivers: [$($driver:ident),+ $(,)?], device_table: [$($dev:expr),+ $(,)?], $($f:tt)*) => {
|
|
struct Module {
|
|
_reg: $crate::net::phy::Registration,
|
|
}
|
|
|
|
$crate::prelude::module! {
|
|
type: Module,
|
|
$($f)*
|
|
}
|
|
|
|
const _: () = {
|
|
static mut DRIVERS: [$crate::net::phy::DriverVTable;
|
|
$crate::module_phy_driver!(@count_devices $($driver),+)] =
|
|
[$($crate::net::phy::create_phy_driver::<$driver>()),+];
|
|
|
|
impl $crate::Module for Module {
|
|
fn init(module: &'static $crate::ThisModule) -> Result<Self> {
|
|
// SAFETY: The anonymous constant guarantees that nobody else can access
|
|
// the `DRIVERS` static. The array is used only in the C side.
|
|
let drivers = unsafe { &mut DRIVERS };
|
|
let mut reg = $crate::net::phy::Registration::register(
|
|
module,
|
|
::core::pin::Pin::static_mut(drivers),
|
|
)?;
|
|
Ok(Module { _reg: reg })
|
|
}
|
|
}
|
|
};
|
|
|
|
$crate::module_phy_driver!(@device_table [$($dev),+]);
|
|
}
|
|
}
|