mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 02:30:34 +02:00 
			
		
		
		
	Core
 ----
 
  - More core refactoring to reduce the RTNL lock contention,
    including preparatory work for the per-network namespace RTNL lock,
    replacing RTNL lock with a per device-one to protect NAPI-related
    net device data and moving synchronize_net() calls outside such
    lock.
 
  - Extend drop reasons usage, adding net scheduler, AF_UNIX, bridge and
    more specific TCP coverage.
 
  - Reduce network namespace tear-down time by removing per-subsystems
    synchronize_net() in tipc and sched.
 
  - Add flow label selector support for fib rules, allowing traffic
    redirection based on such header field.
 
 Netfilter
 ---------
 
  - Do not remove netdev basechain when last device is gone, allowing
    netdev basechains without devices.
 
  - Revisit the flowtable teardown strategy, dealing better with fin,
    reset and re-open events.
 
  - Scale-up IP-vs connection dumping by avoiding linear search on
    each restart.
 
 Protocols
 ---------
 
  - A significant XDP socket refactor, consolidating and optimizing
    several helpers into the core
 
  - Better scaling of ICMP rate-limiting, by removing false-sharing in
    inet peers handling.
 
  - Introduces netlink notifications for multicast IPv4 and IPv6
    address changes.
 
  - Add ipsec support for IP-TFS/AggFrag encapsulation, allowing
    aggregation and fragmentation of the inner IP.
 
  - Add sysctl to configure TIME-WAIT reuse delay for TCP sockets,
    to avoid local port exhaustion issues when the average connection
    lifetime is very short.
 
  - Support updating keys (re-keying) for connections using kernel
    TLS (for TLS 1.3 only).
 
  - Support ipv4-mapped ipv6 address clients in smc-r v2.
 
  - Add support for jumbo data packet transmission in RxRPC sockets,
    gluing multiple data packets in a single UDP packet.
 
  - Support RxRPC RACK-TLP to manage packet loss and retransmission in
    conjunction with the congestion control algorithm.
 
 Driver API
 ----------
 
  - Introduce a unified and structured interface for reporting PHY
    statistics, exposing consistent data across different H/W via
    ethtool.
 
  - Make timestamping selectable, allow the user to select the desired
    hwtstamp provider (PHY or MAC) administratively.
 
  - Add support for configuring a header-data-split threshold (HDS)
    value via ethtool, to deal with partial or buggy H/W implementation.
 
  - Consolidate DSA drivers Energy Efficiency Ethernet support.
 
  - Add EEE management to phylink, making use of the phylib
    implementation.
 
  - Add phylib support for in-band capabilities negotiation.
 
  - Simplify how phylib-enabled mac drivers expose the supported
    interfaces.
 
 Tests and tooling
 -----------------
 
  - Make the YNL tool package-friendly to make it easier to deploy it
    separately from the kernel.
 
  - Increase TCP selftest coverage importing several packetdrill
    test-cases.
 
  - Regenerate the ethtool uapi header from the YNL spec,
    to ease maintenance and future development.
 
  - Add YNL support for decoding the link types used in net
    self-tests, allowing a single build to run both net and
    drivers/net.
 
 Drivers
 -------
 
  - Ethernet high-speed NICs:
    - nVidia/Mellanox (mlx5):
      - add cross E-Switch QoS support
      - add SW Steering support for ConnectX-8
      - implement support for HW-Managed Flow Steering, improving the
        rule deletion/insertion rate
      - support for multi-host LAG
    - Intel (ixgbe, ice, igb):
      - ice: add support for devlink health events
      - ixgbe: add initial support for E610 chipset variant
      - igb: add support for AF_XDP zero-copy
    - Meta:
      - add support for basic RSS config
      - allow changing the number of channels
      - add hardware monitoring support
    - Broadcom (bnxt):
      - implement TCP data split and HDS threshold ethtool support,
        enabling Device Memory TCP.
    - Marvell Octeon:
      - implement egress ipsec offload support for the cn10k family
    - Hisilicon (HIBMC):
      - implement unicast MAC filtering
 
  - Ethernet NICs embedded and virtual:
    - Convert UDP tunnel drivers to NETDEV_PCPU_STAT_DSTATS, avoiding
      contented atomic operations for drop counters
    - Freescale:
      - quicc: phylink conversion
      - enetc: support Tx and Rx checksum offload and improve TSO
        performances
    - MediaTek:
      - airoha: introduce support for ETS and HTB Qdisc offload
    - Microchip:
      - lan78XX USB: preparation work for phylink conversion
    - Synopsys (stmmac):
      - support DWMAC IP on NXP Automotive SoCs S32G2xx/S32G3xx/S32R45
      - refactor EEE support to leverage the new driver API
      - optimize DMA and cache access to increase raw RX performances
        by 40%
    - TI:
      - icssg-prueth: add multicast filtering support for VLAN
        interface
    - netkit:
      - add ability to configure head/tailroom
    - VXLAN:
      - accepts packets with user-defined reserved bit
 
  - Ethernet switches:
    - Microchip:
      - lan969x: add RGMII support
      - lan969x: improve TX and RX performance using the FDMA engine
    - nVidia/Mellanox:
      - move Tx header handling to PCI driver, to ease XDP support
 
  - Ethernet PHYs:
    - Texas Instruments DP83822:
      - add support for GPIO2 clock output
    - Realtek:
      - 8169: add support for RTL8125D rev.b
      - rtl822x: add hwmon support for the temperature sensor
    - Microchip:
      - add support for RDS PTP hardware
      - consolidate periodic output signal generation
 
  - CAN:
    - several DT-bindings to DT schema conversions
    - tcan4x5x:
      - add HW standby support
      - support nWKRQ voltage selection
    - kvaser:
      - allowing Bus Error Reporting runtime configuration
 
  - WiFi:
    - the on-going Multi-Link Operation (MLO) effort continues, affecting
      both the stack and in drivers
    - mac80211/cfg80211:
      - Emergency Preparedness Communication Services (EPCS) station mode
        support
      - support for adding and removing station links for MLO
      - add support for WiFi 7/EHT mesh over 320 MHz channels
      - report Tx power info for each link
    - RealTek (rtw88):
      - enable USB Rx aggregation and USB 3 to improve performance
      - LED support
    - RealTek (rtw89):
      - refactor power save to support Multi-Link Operations
      - add support for RTL8922AE-VS variant
    - MediaTek (mt76):
      - single wiphy multiband support (preparation for MLO)
      - p2p device support
      - add TP-Link TXE50UH USB adapter support
    - Qualcomm (ath10k):
      - support for the QCA6698AQ IP core
    - Qualcomm (ath12k):
      - enable MLO for QCN9274
 
  - Bluetooth:
    - Allow sysfs to trigger hdev reset, to allow recovering devices
      not responsive from user-space
    - MediaTek: add support for MT7922, MT7925, MT7921e devices
    - Realtek: add support for RTL8851BE devices
    - Qualcomm: add support for WCN785x devices
    - ISO: allow BIG re-sync
 
 Signed-off-by: Paolo Abeni <pabeni@redhat.com>
 -----BEGIN PGP SIGNATURE-----
 
 iQJGBAABCAAwFiEEg1AjqC77wbdLX2LbKSR5jcyPE6QFAmePf5YSHHBhYmVuaUBy
 ZWRoYXQuY29tAAoJECkkeY3MjxOkUcMQALblhkGTxurnfT+yK+Bsuhn2LoHl2RPN
 4u2Kjkzm+2FYgcw6lS17cFXsnfAPlRIpmhnmKk1EBgsBdkuL29c+jtqnljA2bboD
 tIMhMgWiaLS3xgEMrLeKnseIo0G9mviQRphGeZPFTaLb4Ww/bd5LAp4ZGc5oij76
 tURatC3b6MuO4Lt5U+jWKnRwviXku8udHkVHXlvPdirawHCVinmx3tvce/BI/MaD
 eUOp6ZeJCPCOLtk7b8WEyxxvdY0f6D9ed82qfPDHjb94SJv+Vxb38RZtNuApIjn9
 S0KdlNih/4flDy17LDxGYSyFps78lUFRbpqmsUlnZkyLXpsph7/WTvAmMAFcrX0K
 UgQ/F/q5GAvcP5WZcCj5+tZaRmfKQraQirXMtYU/Uj50qCnSU7ssyACASt23GLZ8
 OF8tCLlm9lLOU1B6Ofkul1Dbo5f0Xpaghga4dFb0kzSfbm78fTUnqBNsJ7jIkWfi
 fD6dO+fg+p2ZMD0CACGo3CNxQuJmaQWg6BIDeno6God8kZ6qBMxY/sFr4qozrvFH
 x/FgQq8dgc8WLmaPejKiNIPkdQepXrIiv3T9jgMVyEjJnWB/LBfyWKSQOdTfnLs+
 rgr4YMV6XW4bx0fYqTI8B9jZ+FCWbG6sn4UtRTHITKcd3FSvd8Y+PHa5YyCUWvJM
 l8pePMGF0XVF
 =hrsp
 -----END PGP SIGNATURE-----
Merge tag 'net-next-6.14' of git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net-next
Pull networking updates from Paolo Abeni:
 "This is slightly smaller than usual, with the most interesting work
  being still around RTNL scope reduction.
  Core:
   - More core refactoring to reduce the RTNL lock contention, including
     preparatory work for the per-network namespace RTNL lock, replacing
     RTNL lock with a per device-one to protect NAPI-related net device
     data and moving synchronize_net() calls outside such lock.
   - Extend drop reasons usage, adding net scheduler, AF_UNIX, bridge
     and more specific TCP coverage.
   - Reduce network namespace tear-down time by removing per-subsystems
     synchronize_net() in tipc and sched.
   - Add flow label selector support for fib rules, allowing traffic
     redirection based on such header field.
  Netfilter:
   - Do not remove netdev basechain when last device is gone, allowing
     netdev basechains without devices.
   - Revisit the flowtable teardown strategy, dealing better with fin,
     reset and re-open events.
   - Scale-up IP-vs connection dumping by avoiding linear search on each
     restart.
  Protocols:
   - A significant XDP socket refactor, consolidating and optimizing
     several helpers into the core
   - Better scaling of ICMP rate-limiting, by removing false-sharing in
     inet peers handling.
   - Introduces netlink notifications for multicast IPv4 and IPv6
     address changes.
   - Add ipsec support for IP-TFS/AggFrag encapsulation, allowing
     aggregation and fragmentation of the inner IP.
   - Add sysctl to configure TIME-WAIT reuse delay for TCP sockets, to
     avoid local port exhaustion issues when the average connection
     lifetime is very short.
   - Support updating keys (re-keying) for connections using kernel TLS
     (for TLS 1.3 only).
   - Support ipv4-mapped ipv6 address clients in smc-r v2.
   - Add support for jumbo data packet transmission in RxRPC sockets,
     gluing multiple data packets in a single UDP packet.
   - Support RxRPC RACK-TLP to manage packet loss and retransmission in
     conjunction with the congestion control algorithm.
  Driver API:
   - Introduce a unified and structured interface for reporting PHY
     statistics, exposing consistent data across different H/W via
     ethtool.
   - Make timestamping selectable, allow the user to select the desired
     hwtstamp provider (PHY or MAC) administratively.
   - Add support for configuring a header-data-split threshold (HDS)
     value via ethtool, to deal with partial or buggy H/W
     implementation.
   - Consolidate DSA drivers Energy Efficiency Ethernet support.
   - Add EEE management to phylink, making use of the phylib
     implementation.
   - Add phylib support for in-band capabilities negotiation.
   - Simplify how phylib-enabled mac drivers expose the supported
     interfaces.
  Tests and tooling:
   - Make the YNL tool package-friendly to make it easier to deploy it
     separately from the kernel.
   - Increase TCP selftest coverage importing several packetdrill
     test-cases.
   - Regenerate the ethtool uapi header from the YNL spec, to ease
     maintenance and future development.
   - Add YNL support for decoding the link types used in net self-tests,
     allowing a single build to run both net and drivers/net.
  Drivers:
   - Ethernet high-speed NICs:
      - nVidia/Mellanox (mlx5):
         - add cross E-Switch QoS support
         - add SW Steering support for ConnectX-8
         - implement support for HW-Managed Flow Steering, improving the
           rule deletion/insertion rate
         - support for multi-host LAG
      - Intel (ixgbe, ice, igb):
         - ice: add support for devlink health events
         - ixgbe: add initial support for E610 chipset variant
         - igb: add support for AF_XDP zero-copy
      - Meta:
         - add support for basic RSS config
         - allow changing the number of channels
         - add hardware monitoring support
      - Broadcom (bnxt):
         - implement TCP data split and HDS threshold ethtool support,
           enabling Device Memory TCP.
      - Marvell Octeon:
         - implement egress ipsec offload support for the cn10k family
      - Hisilicon (HIBMC):
         - implement unicast MAC filtering
   - Ethernet NICs embedded and virtual:
      - Convert UDP tunnel drivers to NETDEV_PCPU_STAT_DSTATS, avoiding
        contented atomic operations for drop counters
      - Freescale:
         - quicc: phylink conversion
         - enetc: support Tx and Rx checksum offload and improve TSO
           performances
      - MediaTek:
         - airoha: introduce support for ETS and HTB Qdisc offload
      - Microchip:
         - lan78XX USB: preparation work for phylink conversion
      - Synopsys (stmmac):
         - support DWMAC IP on NXP Automotive SoCs S32G2xx/S32G3xx/S32R45
         - refactor EEE support to leverage the new driver API
         - optimize DMA and cache access to increase raw RX performances
           by 40%
      - TI:
         - icssg-prueth: add multicast filtering support for VLAN
           interface
      - netkit:
         - add ability to configure head/tailroom
      - VXLAN:
         - accepts packets with user-defined reserved bit
   - Ethernet switches:
      - Microchip:
         - lan969x: add RGMII support
         - lan969x: improve TX and RX performance using the FDMA engine
      - nVidia/Mellanox:
         - move Tx header handling to PCI driver, to ease XDP support
   - Ethernet PHYs:
      - Texas Instruments DP83822:
         - add support for GPIO2 clock output
      - Realtek:
         - 8169: add support for RTL8125D rev.b
         - rtl822x: add hwmon support for the temperature sensor
      - Microchip:
         - add support for RDS PTP hardware
         - consolidate periodic output signal generation
   - CAN:
      - several DT-bindings to DT schema conversions
      - tcan4x5x:
         - add HW standby support
         - support nWKRQ voltage selection
      - kvaser:
         - allowing Bus Error Reporting runtime configuration
   - WiFi:
      - the on-going Multi-Link Operation (MLO) effort continues,
        affecting both the stack and in drivers
      - mac80211/cfg80211:
         - Emergency Preparedness Communication Services (EPCS) station
           mode support
         - support for adding and removing station links for MLO
         - add support for WiFi 7/EHT mesh over 320 MHz channels
         - report Tx power info for each link
      - RealTek (rtw88):
         - enable USB Rx aggregation and USB 3 to improve performance
         - LED support
      - RealTek (rtw89):
         - refactor power save to support Multi-Link Operations
         - add support for RTL8922AE-VS variant
      - MediaTek (mt76):
         - single wiphy multiband support (preparation for MLO)
         - p2p device support
         - add TP-Link TXE50UH USB adapter support
      - Qualcomm (ath10k):
         - support for the QCA6698AQ IP core
      - Qualcomm (ath12k):
         - enable MLO for QCN9274
   - Bluetooth:
      - Allow sysfs to trigger hdev reset, to allow recovering devices
        not responsive from user-space
      - MediaTek: add support for MT7922, MT7925, MT7921e devices
      - Realtek: add support for RTL8851BE devices
      - Qualcomm: add support for WCN785x devices
      - ISO: allow BIG re-sync"
* tag 'net-next-6.14' of git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net-next: (1386 commits)
  net/rose: prevent integer overflows in rose_setsockopt()
  net: phylink: fix regression when binding a PHY
  net: ethernet: ti: am65-cpsw: streamline TX queue creation and cleanup
  net: ethernet: ti: am65-cpsw: streamline RX queue creation and cleanup
  net: ethernet: ti: am65-cpsw: ensure proper channel cleanup in error path
  ipv6: Convert inet6_rtm_deladdr() to per-netns RTNL.
  ipv6: Convert inet6_rtm_newaddr() to per-netns RTNL.
  ipv6: Move lifetime validation to inet6_rtm_newaddr().
  ipv6: Set cfg.ifa_flags before device lookup in inet6_rtm_newaddr().
  ipv6: Pass dev to inet6_addr_add().
  ipv6: Convert inet6_ioctl() to per-netns RTNL.
  ipv6: Hold rtnl_net_lock() in addrconf_init() and addrconf_cleanup().
  ipv6: Hold rtnl_net_lock() in addrconf_dad_work().
  ipv6: Hold rtnl_net_lock() in addrconf_verify_work().
  ipv6: Convert net.ipv6.conf.${DEV}.XXX sysctl to per-netns RTNL.
  ipv6: Add __in6_dev_get_rtnl_net().
  net: stmmac: Drop redundant skb_mark_for_recycle() for SKB frags
  net: mii: Fix the Speed display when the network cable is not connected
  sysctl net: Remove macro checks for CONFIG_SYSCTL
  eth: bnxt: update header sizing defaults
  ...
		
	
			
		
			
				
	
	
		
			921 lines
		
	
	
	
		
			31 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			921 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::{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 i32 };
 | 
						|
    }
 | 
						|
 | 
						|
    /// 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 as i32,
 | 
						|
            DuplexMode::Half => bindings::DUPLEX_HALF as i32,
 | 
						|
            DuplexMode::Unknown => bindings::DUPLEX_UNKNOWN as i32,
 | 
						|
        };
 | 
						|
        // SAFETY: The struct invariant ensures that we may access
 | 
						|
        // this field without additional synchronization.
 | 
						|
        unsafe { (*phydev).duplex = v };
 | 
						|
    }
 | 
						|
 | 
						|
    /// 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::as_ref(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,
 | 
						|
    ) -> crate::ffi::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) -> crate::ffi::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,
 | 
						|
    ) -> crate::ffi::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) -> crate::ffi::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) -> crate::ffi::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,
 | 
						|
    ) -> crate::ffi::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,
 | 
						|
    ) -> crate::ffi::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,
 | 
						|
    ) -> crate::ffi::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.
 | 
						|
pub struct DeviceId {
 | 
						|
    id: u32,
 | 
						|
    mask: DeviceMask,
 | 
						|
}
 | 
						|
 | 
						|
impl DeviceId {
 | 
						|
    /// Creates a new instance with the exact match mask.
 | 
						|
    pub const fn new_with_exact_mask(id: u32) -> Self {
 | 
						|
        DeviceId {
 | 
						|
            id,
 | 
						|
            mask: DeviceMask::Exact,
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /// Creates a new instance with the model match mask.
 | 
						|
    pub const fn new_with_model_mask(id: u32) -> Self {
 | 
						|
        DeviceId {
 | 
						|
            id,
 | 
						|
            mask: DeviceMask::Model,
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /// Creates a new instance with the vendor match mask.
 | 
						|
    pub const fn new_with_vendor_mask(id: u32) -> Self {
 | 
						|
        DeviceId {
 | 
						|
            id,
 | 
						|
            mask: DeviceMask::Vendor,
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /// Creates a new instance with a custom match mask.
 | 
						|
    pub const fn new_with_custom_mask(id: u32, mask: u32) -> Self {
 | 
						|
        DeviceId {
 | 
						|
            id,
 | 
						|
            mask: DeviceMask::Custom(mask),
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /// Creates a new instance from [`Driver`].
 | 
						|
    pub const fn new_with_driver<T: Driver>() -> Self {
 | 
						|
        T::PHY_DEVICE_ID
 | 
						|
    }
 | 
						|
 | 
						|
    /// Get a `mask` as u32.
 | 
						|
    pub const fn mask_as_int(&self) -> u32 {
 | 
						|
        self.mask.as_int()
 | 
						|
    }
 | 
						|
 | 
						|
    // macro use only
 | 
						|
    #[doc(hidden)]
 | 
						|
    pub const fn mdio_device_id(&self) -> bindings::mdio_device_id {
 | 
						|
        bindings::mdio_device_id {
 | 
						|
            phy_id: self.id,
 | 
						|
            phy_id_mask: self.mask.as_int(),
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
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",
 | 
						|
///     author: "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",
 | 
						|
///     author: "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 _DEVICE_TABLE: [::kernel::bindings::mdio_device_id; 2] = [
 | 
						|
///     ::kernel::bindings::mdio_device_id {
 | 
						|
///         phy_id: 0x00000001,
 | 
						|
///         phy_id_mask: 0xffffffff,
 | 
						|
///     },
 | 
						|
///     ::kernel::bindings::mdio_device_id {
 | 
						|
///         phy_id: 0,
 | 
						|
///         phy_id_mask: 0,
 | 
						|
///     },
 | 
						|
/// ];
 | 
						|
/// #[cfg(MODULE)]
 | 
						|
/// #[no_mangle]
 | 
						|
/// static __mod_device_table__mdio__phydev: [::kernel::bindings::mdio_device_id; 2] = _DEVICE_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),+]) => {
 | 
						|
        // SAFETY: C will not read off the end of this constant since the last element is zero.
 | 
						|
        const _DEVICE_TABLE: [$crate::bindings::mdio_device_id;
 | 
						|
            $crate::module_phy_driver!(@count_devices $($dev),+) + 1] = [
 | 
						|
            $($dev.mdio_device_id()),+,
 | 
						|
            $crate::bindings::mdio_device_id {
 | 
						|
                phy_id: 0,
 | 
						|
                phy_id_mask: 0
 | 
						|
            }
 | 
						|
        ];
 | 
						|
 | 
						|
        #[cfg(MODULE)]
 | 
						|
        #[no_mangle]
 | 
						|
        static __mod_device_table__mdio__phydev: [$crate::bindings::mdio_device_id;
 | 
						|
            $crate::module_phy_driver!(@count_devices $($dev),+) + 1] = _DEVICE_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),+]);
 | 
						|
    }
 | 
						|
}
 |