1 // SPDX-License-Identifier: GPL-2.0
3 // Copyright (C) 2023 FUJITA Tomonori <fujita.tomonori@gmail.com>
5 //! Network PHY device.
7 //! C headers: [`include/linux/phy.h`](../../../../../../../include/linux/phy.h).
9 use crate::{bindings, error::*, prelude::*, str::CStr, types::Opaque};
11 use core::marker::PhantomData;
13 /// PHY state machine states.
15 /// Corresponds to the kernel's [`enum phy_state`].
17 /// Some of PHY drivers access to the state of PHY's software state machine.
19 /// [`enum phy_state`]: ../../../../../../../include/linux/phy.h
20 #[derive(PartialEq, Eq)]
21 pub enum DeviceState {
22 /// PHY device and driver are not ready for anything.
24 /// PHY is ready to send and receive packets.
26 /// PHY is up, but no polling or interrupts are done.
28 /// PHY is up, but is in an error state.
30 /// PHY and attached device are ready to do work.
32 /// PHY is currently running.
34 /// PHY is up, but not currently plugged in.
36 /// PHY is performing a cable test.
40 /// A mode of Ethernet communication.
42 /// PHY drivers get duplex information from hardware and update the current state.
44 /// PHY is in full-duplex mode.
46 /// PHY is in half-duplex mode.
48 /// PHY is in unknown duplex mode.
52 /// An instance of a PHY device.
54 /// Wraps the kernel's [`struct phy_device`].
56 /// A [`Device`] instance is created when a callback in [`Driver`] is executed. A PHY driver
57 /// executes [`Driver`]'s methods during the callback.
61 /// Referencing a `phy_device` using this struct asserts that you are in
62 /// a context where all methods defined on this struct are safe to call.
64 /// [`struct phy_device`]: ../../../../../../../include/linux/phy.h
65 // During the calls to most functions in [`Driver`], the C side (`PHYLIB`) holds a lock that is
66 // unique for every instance of [`Device`]. `PHYLIB` uses a different serialization technique for
67 // [`Driver::resume`] and [`Driver::suspend`]: `PHYLIB` updates `phy_device`'s state with
68 // the lock held, thus guaranteeing that [`Driver::resume`] has exclusive access to the instance.
69 // [`Driver::resume`] and [`Driver::suspend`] also are called where only one thread can access
72 pub struct Device(Opaque<bindings::phy_device>);
75 /// Creates a new [`Device`] instance from a raw pointer.
79 /// For the duration of 'a, the pointer must point at a valid `phy_device`,
80 /// and the caller must be in a context where all methods defined on this struct
82 unsafe fn from_raw<'a>(ptr: *mut bindings::phy_device) -> &'a mut Self {
83 // CAST: `Self` is a `repr(transparent)` wrapper around `bindings::phy_device`.
84 let ptr = ptr.cast::<Self>();
85 // SAFETY: by the function requirements the pointer is valid and we have unique access for
86 // the duration of `'a`.
90 /// Gets the id of the PHY.
91 pub fn phy_id(&self) -> u32 {
92 let phydev = self.0.get();
93 // SAFETY: The struct invariant ensures that we may access
94 // this field without additional synchronization.
95 unsafe { (*phydev).phy_id }
98 /// Gets the state of PHY state machine states.
99 pub fn state(&self) -> DeviceState {
100 let phydev = self.0.get();
101 // SAFETY: The struct invariant ensures that we may access
102 // this field without additional synchronization.
103 let state = unsafe { (*phydev).state };
104 // TODO: this conversion code will be replaced with automatically generated code by bindgen
105 // when it becomes possible.
107 bindings::phy_state_PHY_DOWN => DeviceState::Down,
108 bindings::phy_state_PHY_READY => DeviceState::Ready,
109 bindings::phy_state_PHY_HALTED => DeviceState::Halted,
110 bindings::phy_state_PHY_ERROR => DeviceState::Error,
111 bindings::phy_state_PHY_UP => DeviceState::Up,
112 bindings::phy_state_PHY_RUNNING => DeviceState::Running,
113 bindings::phy_state_PHY_NOLINK => DeviceState::NoLink,
114 bindings::phy_state_PHY_CABLETEST => DeviceState::CableTest,
115 _ => DeviceState::Error,
119 /// Gets the current link state.
121 /// It returns true if the link is up.
122 pub fn is_link_up(&self) -> bool {
123 const LINK_IS_UP: u64 = 1;
124 // TODO: the code to access to the bit field will be replaced with automatically
125 // generated code by bindgen when it becomes possible.
126 // SAFETY: The struct invariant ensures that we may access
127 // this field without additional synchronization.
128 let bit_field = unsafe { &(*self.0.get())._bitfield_1 };
129 bit_field.get(14, 1) == LINK_IS_UP
132 /// Gets the current auto-negotiation configuration.
134 /// It returns true if auto-negotiation is enabled.
135 pub fn is_autoneg_enabled(&self) -> bool {
136 // TODO: the code to access to the bit field will be replaced with automatically
137 // generated code by bindgen when it becomes possible.
138 // SAFETY: The struct invariant ensures that we may access
139 // this field without additional synchronization.
140 let bit_field = unsafe { &(*self.0.get())._bitfield_1 };
141 bit_field.get(13, 1) == bindings::AUTONEG_ENABLE as u64
144 /// Gets the current auto-negotiation state.
146 /// It returns true if auto-negotiation is completed.
147 pub fn is_autoneg_completed(&self) -> bool {
148 const AUTONEG_COMPLETED: u64 = 1;
149 // TODO: the code to access to the bit field will be replaced with automatically
150 // generated code by bindgen when it becomes possible.
151 // SAFETY: The struct invariant ensures that we may access
152 // this field without additional synchronization.
153 let bit_field = unsafe { &(*self.0.get())._bitfield_1 };
154 bit_field.get(15, 1) == AUTONEG_COMPLETED
157 /// Sets the speed of the PHY.
158 pub fn set_speed(&mut self, speed: u32) {
159 let phydev = self.0.get();
160 // SAFETY: The struct invariant ensures that we may access
161 // this field without additional synchronization.
162 unsafe { (*phydev).speed = speed as i32 };
165 /// Sets duplex mode.
166 pub fn set_duplex(&mut self, mode: DuplexMode) {
167 let phydev = self.0.get();
169 DuplexMode::Full => bindings::DUPLEX_FULL as i32,
170 DuplexMode::Half => bindings::DUPLEX_HALF as i32,
171 DuplexMode::Unknown => bindings::DUPLEX_UNKNOWN as i32,
173 // SAFETY: The struct invariant ensures that we may access
174 // this field without additional synchronization.
175 unsafe { (*phydev).duplex = v };
178 /// Reads a given C22 PHY register.
179 // This function reads a hardware register and updates the stats so takes `&mut self`.
180 pub fn read(&mut self, regnum: u16) -> Result<u16> {
181 let phydev = self.0.get();
182 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
183 // So it's just an FFI call, open code of `phy_read()` with a valid `phy_device` pointer
186 bindings::mdiobus_read((*phydev).mdio.bus, (*phydev).mdio.addr, regnum.into())
189 Err(Error::from_errno(ret))
195 /// Writes a given C22 PHY register.
196 pub fn write(&mut self, regnum: u16, val: u16) -> Result {
197 let phydev = self.0.get();
198 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
199 // So it's just an FFI call, open code of `phy_write()` with a valid `phy_device` pointer
202 bindings::mdiobus_write((*phydev).mdio.bus, (*phydev).mdio.addr, regnum.into(), val)
206 /// Reads a paged register.
207 pub fn read_paged(&mut self, page: u16, regnum: u16) -> Result<u16> {
208 let phydev = self.0.get();
209 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
210 // So it's just an FFI call.
211 let ret = unsafe { bindings::phy_read_paged(phydev, page.into(), regnum.into()) };
213 Err(Error::from_errno(ret))
219 /// Resolves the advertisements into PHY settings.
220 pub fn resolve_aneg_linkmode(&mut self) {
221 let phydev = self.0.get();
222 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
223 // So it's just an FFI call.
224 unsafe { bindings::phy_resolve_aneg_linkmode(phydev) };
227 /// Executes software reset the PHY via `BMCR_RESET` bit.
228 pub fn genphy_soft_reset(&mut self) -> Result {
229 let phydev = self.0.get();
230 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
231 // So it's just an FFI call.
232 to_result(unsafe { bindings::genphy_soft_reset(phydev) })
235 /// Initializes the PHY.
236 pub fn init_hw(&mut self) -> Result {
237 let phydev = self.0.get();
238 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
239 // So it's just an FFI call.
240 to_result(unsafe { bindings::phy_init_hw(phydev) })
243 /// Starts auto-negotiation.
244 pub fn start_aneg(&mut self) -> Result {
245 let phydev = self.0.get();
246 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
247 // So it's just an FFI call.
248 to_result(unsafe { bindings::_phy_start_aneg(phydev) })
251 /// Resumes the PHY via `BMCR_PDOWN` bit.
252 pub fn genphy_resume(&mut self) -> Result {
253 let phydev = self.0.get();
254 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
255 // So it's just an FFI call.
256 to_result(unsafe { bindings::genphy_resume(phydev) })
259 /// Suspends the PHY via `BMCR_PDOWN` bit.
260 pub fn genphy_suspend(&mut self) -> Result {
261 let phydev = self.0.get();
262 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
263 // So it's just an FFI call.
264 to_result(unsafe { bindings::genphy_suspend(phydev) })
267 /// Checks the link status and updates current link state.
268 pub fn genphy_read_status(&mut self) -> Result<u16> {
269 let phydev = self.0.get();
270 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
271 // So it's just an FFI call.
272 let ret = unsafe { bindings::genphy_read_status(phydev) };
274 Err(Error::from_errno(ret))
280 /// Updates the link status.
281 pub fn genphy_update_link(&mut self) -> Result {
282 let phydev = self.0.get();
283 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
284 // So it's just an FFI call.
285 to_result(unsafe { bindings::genphy_update_link(phydev) })
288 /// Reads link partner ability.
289 pub fn genphy_read_lpa(&mut self) -> Result {
290 let phydev = self.0.get();
291 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
292 // So it's just an FFI call.
293 to_result(unsafe { bindings::genphy_read_lpa(phydev) })
296 /// Reads PHY abilities.
297 pub fn genphy_read_abilities(&mut self) -> Result {
298 let phydev = self.0.get();
299 // SAFETY: `phydev` is pointing to a valid object by the type invariant of `Self`.
300 // So it's just an FFI call.
301 to_result(unsafe { bindings::genphy_read_abilities(phydev) })
305 /// Defines certain other features this PHY supports (like interrupts).
307 /// These flag values are used in [`Driver::FLAGS`].
310 pub const IS_INTERNAL: u32 = bindings::PHY_IS_INTERNAL;
311 /// PHY needs to be reset after the refclk is enabled.
312 pub const RST_AFTER_CLK_EN: u32 = bindings::PHY_RST_AFTER_CLK_EN;
313 /// Polling is used to detect PHY status changes.
314 pub const POLL_CABLE_TEST: u32 = bindings::PHY_POLL_CABLE_TEST;
316 pub const ALWAYS_CALL_SUSPEND: u32 = bindings::PHY_ALWAYS_CALL_SUSPEND;
319 /// An adapter for the registration of a PHY driver.
320 struct Adapter<T: Driver> {
324 impl<T: Driver> Adapter<T> {
327 /// `phydev` must be passed by the corresponding callback in `phy_driver`.
328 unsafe extern "C" fn soft_reset_callback(
329 phydev: *mut bindings::phy_device,
330 ) -> core::ffi::c_int {
332 // SAFETY: This callback is called only in contexts
333 // where we hold `phy_device->lock`, so the accessors on
334 // `Device` are okay to call.
335 let dev = unsafe { Device::from_raw(phydev) };
343 /// `phydev` must be passed by the corresponding callback in `phy_driver`.
344 unsafe extern "C" fn get_features_callback(
345 phydev: *mut bindings::phy_device,
346 ) -> core::ffi::c_int {
348 // SAFETY: This callback is called only in contexts
349 // where we hold `phy_device->lock`, so the accessors on
350 // `Device` are okay to call.
351 let dev = unsafe { Device::from_raw(phydev) };
352 T::get_features(dev)?;
359 /// `phydev` must be passed by the corresponding callback in `phy_driver`.
360 unsafe extern "C" fn suspend_callback(phydev: *mut bindings::phy_device) -> core::ffi::c_int {
362 // SAFETY: The C core code ensures that the accessors on
363 // `Device` are okay to call even though `phy_device->lock`
364 // might not be held.
365 let dev = unsafe { Device::from_raw(phydev) };
373 /// `phydev` must be passed by the corresponding callback in `phy_driver`.
374 unsafe extern "C" fn resume_callback(phydev: *mut bindings::phy_device) -> core::ffi::c_int {
376 // SAFETY: The C core code ensures that the accessors on
377 // `Device` are okay to call even though `phy_device->lock`
378 // might not be held.
379 let dev = unsafe { Device::from_raw(phydev) };
387 /// `phydev` must be passed by the corresponding callback in `phy_driver`.
388 unsafe extern "C" fn config_aneg_callback(
389 phydev: *mut bindings::phy_device,
390 ) -> core::ffi::c_int {
392 // SAFETY: This callback is called only in contexts
393 // where we hold `phy_device->lock`, so the accessors on
394 // `Device` are okay to call.
395 let dev = unsafe { Device::from_raw(phydev) };
396 T::config_aneg(dev)?;
403 /// `phydev` must be passed by the corresponding callback in `phy_driver`.
404 unsafe extern "C" fn read_status_callback(
405 phydev: *mut bindings::phy_device,
406 ) -> core::ffi::c_int {
408 // SAFETY: This callback is called only in contexts
409 // where we hold `phy_device->lock`, so the accessors on
410 // `Device` are okay to call.
411 let dev = unsafe { Device::from_raw(phydev) };
412 T::read_status(dev)?;
419 /// `phydev` must be passed by the corresponding callback in `phy_driver`.
420 unsafe extern "C" fn match_phy_device_callback(
421 phydev: *mut bindings::phy_device,
422 ) -> core::ffi::c_int {
423 // SAFETY: This callback is called only in contexts
424 // where we hold `phy_device->lock`, so the accessors on
425 // `Device` are okay to call.
426 let dev = unsafe { Device::from_raw(phydev) };
427 T::match_phy_device(dev) as i32
432 /// `phydev` must be passed by the corresponding callback in `phy_driver`.
433 unsafe extern "C" fn read_mmd_callback(
434 phydev: *mut bindings::phy_device,
439 // SAFETY: This callback is called only in contexts
440 // where we hold `phy_device->lock`, so the accessors on
441 // `Device` are okay to call.
442 let dev = unsafe { Device::from_raw(phydev) };
443 // CAST: the C side verifies devnum < 32.
444 let ret = T::read_mmd(dev, devnum as u8, regnum)?;
451 /// `phydev` must be passed by the corresponding callback in `phy_driver`.
452 unsafe extern "C" fn write_mmd_callback(
453 phydev: *mut bindings::phy_device,
459 // SAFETY: This callback is called only in contexts
460 // where we hold `phy_device->lock`, so the accessors on
461 // `Device` are okay to call.
462 let dev = unsafe { Device::from_raw(phydev) };
463 T::write_mmd(dev, devnum as u8, regnum, val)?;
470 /// `phydev` must be passed by the corresponding callback in `phy_driver`.
471 unsafe extern "C" fn link_change_notify_callback(phydev: *mut bindings::phy_device) {
472 // SAFETY: This callback is called only in contexts
473 // where we hold `phy_device->lock`, so the accessors on
474 // `Device` are okay to call.
475 let dev = unsafe { Device::from_raw(phydev) };
476 T::link_change_notify(dev);
480 /// Driver structure for a particular PHY type.
482 /// Wraps the kernel's [`struct phy_driver`].
483 /// This is used to register a driver for a particular PHY type with the kernel.
487 /// `self.0` is always in a valid state.
489 /// [`struct phy_driver`]: ../../../../../../../include/linux/phy.h
491 pub struct DriverVTable(Opaque<bindings::phy_driver>);
493 // SAFETY: `DriverVTable` doesn't expose any &self method to access internal data, so it's safe to
494 // share `&DriverVTable` across execution context boundries.
495 unsafe impl Sync for DriverVTable {}
497 /// Creates a [`DriverVTable`] instance from [`Driver`].
499 /// This is used by [`module_phy_driver`] macro to create a static array of `phy_driver`.
501 /// [`module_phy_driver`]: crate::module_phy_driver
502 pub const fn create_phy_driver<T: Driver>() -> DriverVTable {
503 // INVARIANT: All the fields of `struct phy_driver` are initialized properly.
504 DriverVTable(Opaque::new(bindings::phy_driver {
505 name: T::NAME.as_char_ptr().cast_mut(),
507 phy_id: T::PHY_DEVICE_ID.id,
508 phy_id_mask: T::PHY_DEVICE_ID.mask_as_int(),
509 soft_reset: if T::HAS_SOFT_RESET {
510 Some(Adapter::<T>::soft_reset_callback)
514 get_features: if T::HAS_GET_FEATURES {
515 Some(Adapter::<T>::get_features_callback)
519 match_phy_device: if T::HAS_MATCH_PHY_DEVICE {
520 Some(Adapter::<T>::match_phy_device_callback)
524 suspend: if T::HAS_SUSPEND {
525 Some(Adapter::<T>::suspend_callback)
529 resume: if T::HAS_RESUME {
530 Some(Adapter::<T>::resume_callback)
534 config_aneg: if T::HAS_CONFIG_ANEG {
535 Some(Adapter::<T>::config_aneg_callback)
539 read_status: if T::HAS_READ_STATUS {
540 Some(Adapter::<T>::read_status_callback)
544 read_mmd: if T::HAS_READ_MMD {
545 Some(Adapter::<T>::read_mmd_callback)
549 write_mmd: if T::HAS_WRITE_MMD {
550 Some(Adapter::<T>::write_mmd_callback)
554 link_change_notify: if T::HAS_LINK_CHANGE_NOTIFY {
555 Some(Adapter::<T>::link_change_notify_callback)
559 // SAFETY: The rest is zeroed out to initialize `struct phy_driver`,
560 // sets `Option<&F>` to be `None`.
561 ..unsafe { core::mem::MaybeUninit::<bindings::phy_driver>::zeroed().assume_init() }
565 /// Driver implementation for a particular PHY type.
567 /// This trait is used to create a [`DriverVTable`].
570 /// Defines certain other features this PHY supports.
571 /// It is a combination of the flags in the [`flags`] module.
572 const FLAGS: u32 = 0;
574 /// The friendly name of this PHY type.
575 const NAME: &'static CStr;
577 /// This driver only works for PHYs with IDs which match this field.
578 /// The default id and mask are zero.
579 const PHY_DEVICE_ID: DeviceId = DeviceId::new_with_custom_mask(0, 0);
581 /// Issues a PHY software reset.
582 fn soft_reset(_dev: &mut Device) -> Result {
586 /// Probes the hardware to determine what abilities it has.
587 fn get_features(_dev: &mut Device) -> Result {
591 /// Returns true if this is a suitable driver for the given phydev.
592 /// If not implemented, matching is based on [`Driver::PHY_DEVICE_ID`].
593 fn match_phy_device(_dev: &Device) -> bool {
597 /// Configures the advertisement and resets auto-negotiation
598 /// if auto-negotiation is enabled.
599 fn config_aneg(_dev: &mut Device) -> Result {
603 /// Determines the negotiated speed and duplex.
604 fn read_status(_dev: &mut Device) -> Result<u16> {
608 /// Suspends the hardware, saving state if needed.
609 fn suspend(_dev: &mut Device) -> Result {
613 /// Resumes the hardware, restoring state if needed.
614 fn resume(_dev: &mut Device) -> Result {
618 /// Overrides the default MMD read function for reading a MMD register.
619 fn read_mmd(_dev: &mut Device, _devnum: u8, _regnum: u16) -> Result<u16> {
623 /// Overrides the default MMD write function for writing a MMD register.
624 fn write_mmd(_dev: &mut Device, _devnum: u8, _regnum: u16, _val: u16) -> Result {
628 /// Callback for notification of link change.
629 fn link_change_notify(_dev: &mut Device) {}
632 /// Registration structure for PHY drivers.
634 /// Registers [`DriverVTable`] instances with the kernel. They will be unregistered when dropped.
638 /// The `drivers` slice are currently registered to the kernel via `phy_drivers_register`.
639 pub struct Registration {
640 drivers: Pin<&'static mut [DriverVTable]>,
644 /// Registers a PHY driver.
646 module: &'static crate::ThisModule,
647 drivers: Pin<&'static mut [DriverVTable]>,
649 if drivers.is_empty() {
650 return Err(code::EINVAL);
652 // SAFETY: The type invariants of [`DriverVTable`] ensure that all elements of
653 // the `drivers` slice are initialized properly. `drivers` will not be moved.
654 // So it's just an FFI call.
656 bindings::phy_drivers_register(drivers[0].0.get(), drivers.len().try_into()?, module.0)
658 // INVARIANT: The `drivers` slice is successfully registered to the kernel via `phy_drivers_register`.
659 Ok(Registration { drivers })
663 impl Drop for Registration {
665 // SAFETY: The type invariants guarantee that `self.drivers` is valid.
666 // So it's just an FFI call.
668 bindings::phy_drivers_unregister(self.drivers[0].0.get(), self.drivers.len() as i32)
673 /// An identifier for PHY devices on an MDIO/MII bus.
675 /// Represents the kernel's `struct mdio_device_id`. This is used to find an appropriate
677 pub struct DeviceId {
683 /// Creates a new instance with the exact match mask.
684 pub const fn new_with_exact_mask(id: u32) -> Self {
687 mask: DeviceMask::Exact,
691 /// Creates a new instance with the model match mask.
692 pub const fn new_with_model_mask(id: u32) -> Self {
695 mask: DeviceMask::Model,
699 /// Creates a new instance with the vendor match mask.
700 pub const fn new_with_vendor_mask(id: u32) -> Self {
703 mask: DeviceMask::Vendor,
707 /// Creates a new instance with a custom match mask.
708 pub const fn new_with_custom_mask(id: u32, mask: u32) -> Self {
711 mask: DeviceMask::Custom(mask),
715 /// Creates a new instance from [`Driver`].
716 pub const fn new_with_driver<T: Driver>() -> Self {
720 /// Get a `mask` as u32.
721 pub const fn mask_as_int(&self) -> u32 {
727 pub const fn mdio_device_id(&self) -> bindings::mdio_device_id {
728 bindings::mdio_device_id {
730 phy_id_mask: self.mask.as_int(),
743 const MASK_EXACT: u32 = !0;
744 const MASK_MODEL: u32 = !0 << 4;
745 const MASK_VENDOR: u32 = !0 << 10;
747 const fn as_int(&self) -> u32 {
749 DeviceMask::Exact => Self::MASK_EXACT,
750 DeviceMask::Model => Self::MASK_MODEL,
751 DeviceMask::Vendor => Self::MASK_VENDOR,
752 DeviceMask::Custom(mask) => *mask,
757 /// Declares a kernel module for PHYs drivers.
759 /// This creates a static array of kernel's `struct phy_driver` and registers it.
760 /// This also corresponds to the kernel's `MODULE_DEVICE_TABLE` macro, which embeds the information
761 /// for module loading into the module binary file. Every driver needs an entry in `device_table`.
766 /// # mod module_phy_driver_sample {
767 /// use kernel::c_str;
768 /// use kernel::net::phy::{self, DeviceId};
769 /// use kernel::prelude::*;
771 /// kernel::module_phy_driver! {
772 /// drivers: [PhySample],
774 /// DeviceId::new_with_driver::<PhySample>()
776 /// name: "rust_sample_phy",
777 /// author: "Rust for Linux Contributors",
778 /// description: "Rust sample PHYs driver",
782 /// struct PhySample;
785 /// impl phy::Driver for PhySample {
786 /// const NAME: &'static CStr = c_str!("PhySample");
787 /// const PHY_DEVICE_ID: phy::DeviceId = phy::DeviceId::new_with_exact_mask(0x00000001);
792 /// This expands to the following code:
795 /// use kernel::c_str;
796 /// use kernel::net::phy::{self, DeviceId};
797 /// use kernel::prelude::*;
800 /// _reg: ::kernel::net::phy::Registration,
805 /// name: "rust_sample_phy",
806 /// author: "Rust for Linux Contributors",
807 /// description: "Rust sample PHYs driver",
811 /// struct PhySample;
814 /// impl phy::Driver for PhySample {
815 /// const NAME: &'static CStr = c_str!("PhySample");
816 /// const PHY_DEVICE_ID: phy::DeviceId = phy::DeviceId::new_with_exact_mask(0x00000001);
820 /// static mut DRIVERS: [::kernel::net::phy::DriverVTable; 1] =
821 /// [::kernel::net::phy::create_phy_driver::<PhySample>()];
823 /// impl ::kernel::Module for Module {
824 /// fn init(module: &'static ThisModule) -> Result<Self> {
825 /// let drivers = unsafe { &mut DRIVERS };
826 /// let mut reg = ::kernel::net::phy::Registration::register(
828 /// ::core::pin::Pin::static_mut(drivers),
830 /// Ok(Module { _reg: reg })
837 /// static __mod_mdio__phydev_device_table: [::kernel::bindings::mdio_device_id; 2] = [
838 /// ::kernel::bindings::mdio_device_id {
839 /// phy_id: 0x00000001,
840 /// phy_id_mask: 0xffffffff,
842 /// ::kernel::bindings::mdio_device_id {
849 macro_rules! module_phy_driver {
850 (@replace_expr $_t:tt $sub:expr) => {$sub};
852 (@count_devices $($x:expr),*) => {
853 0usize $(+ $crate::module_phy_driver!(@replace_expr $x 1usize))*
856 (@device_table [$($dev:expr),+]) => {
857 // SAFETY: C will not read off the end of this constant since the last element is zero.
860 static __mod_mdio__phydev_device_table: [$crate::bindings::mdio_device_id;
861 $crate::module_phy_driver!(@count_devices $($dev),+) + 1] = [
862 $($dev.mdio_device_id()),+,
863 $crate::bindings::mdio_device_id {
870 (drivers: [$($driver:ident),+ $(,)?], device_table: [$($dev:expr),+ $(,)?], $($f:tt)*) => {
872 _reg: $crate::net::phy::Registration,
875 $crate::prelude::module! {
881 static mut DRIVERS: [$crate::net::phy::DriverVTable;
882 $crate::module_phy_driver!(@count_devices $($driver),+)] =
883 [$($crate::net::phy::create_phy_driver::<$driver>()),+];
885 impl $crate::Module for Module {
886 fn init(module: &'static ThisModule) -> Result<Self> {
887 // SAFETY: The anonymous constant guarantees that nobody else can access
888 // the `DRIVERS` static. The array is used only in the C side.
889 let drivers = unsafe { &mut DRIVERS };
890 let mut reg = $crate::net::phy::Registration::register(
892 ::core::pin::Pin::static_mut(drivers),
894 Ok(Module { _reg: reg })
899 $crate::module_phy_driver!(@device_table [$($dev),+]);