// SPDX-License-Identifier: GPL-2.0 // Copyright (C) Tehuti Networks Ltd. // Copyright (C) 2024 FUJITA Tomonori //! Applied Micro Circuits Corporation QT2025 PHY driver //! //! This driver is based on the vendor driver `QT2025_phy.c`. This source //! and firmware can be downloaded on the EN-9320SFP+ support site. //! //! The QT2025 PHY integrates an Intel 8051 micro-controller. use kernel::c_str; use kernel::error::code; use kernel::firmware::Firmware; use kernel::net::phy::{ self, reg::{Mmd, C45}, Driver, }; use kernel::prelude::*; use kernel::sizes::{SZ_16K, SZ_8K}; kernel::module_phy_driver! { drivers: [PhyQT2025], device_table: [ phy::DeviceId::new_with_driver::(), ], name: "qt2025_phy", author: "FUJITA Tomonori ", description: "AMCC QT2025 PHY driver", license: "GPL", firmware: ["qt2025-2.0.3.3.fw"], } struct PhyQT2025; #[vtable] impl Driver for PhyQT2025 { const NAME: &'static CStr = c_str!("QT2025 10Gpbs SFP+"); const PHY_DEVICE_ID: phy::DeviceId = phy::DeviceId::new_with_exact_mask(0x0043a400); fn probe(dev: &mut phy::Device) -> Result<()> { // Check the hardware revision code. // Only 0x3b works with this driver and firmware. let hw_rev = dev.read(C45::new(Mmd::PMAPMD, 0xd001))?; if (hw_rev >> 8) != 0xb3 { return Err(code::ENODEV); } // `MICRO_RESETN`: hold the micro-controller in reset while configuring. dev.write(C45::new(Mmd::PMAPMD, 0xc300), 0x0000)?; // `SREFCLK_FREQ`: configure clock frequency of the micro-controller. dev.write(C45::new(Mmd::PMAPMD, 0xc302), 0x0004)?; // Non loopback mode. dev.write(C45::new(Mmd::PMAPMD, 0xc319), 0x0038)?; // `CUS_LAN_WAN_CONFIG`: select between LAN and WAN (WIS) mode. dev.write(C45::new(Mmd::PMAPMD, 0xc31a), 0x0098)?; // The following writes use standardized registers (3.38 through // 3.41 5/10/25GBASE-R PCS test pattern seed B) for something else. // We don't know what. dev.write(C45::new(Mmd::PCS, 0x0026), 0x0e00)?; dev.write(C45::new(Mmd::PCS, 0x0027), 0x0893)?; dev.write(C45::new(Mmd::PCS, 0x0028), 0xa528)?; dev.write(C45::new(Mmd::PCS, 0x0029), 0x0003)?; // Configure transmit and recovered clock. dev.write(C45::new(Mmd::PMAPMD, 0xa30a), 0x06e1)?; // `MICRO_RESETN`: release the micro-controller from the reset state. dev.write(C45::new(Mmd::PMAPMD, 0xc300), 0x0002)?; // The micro-controller will start running from the boot ROM. dev.write(C45::new(Mmd::PCS, 0xe854), 0x00c0)?; let fw = Firmware::request(c_str!("qt2025-2.0.3.3.fw"), dev.as_ref())?; if fw.data().len() > SZ_16K + SZ_8K { return Err(code::EFBIG); } // The 24kB of program memory space is accessible by MDIO. // The first 16kB of memory is located in the address range 3.8000h - 3.BFFFh. // The next 8kB of memory is located at 4.8000h - 4.9FFFh. let mut dst_offset = 0; let mut dst_mmd = Mmd::PCS; for (src_idx, val) in fw.data().iter().enumerate() { if src_idx == SZ_16K { // Start writing to the next register with no offset dst_offset = 0; dst_mmd = Mmd::PHYXS; } dev.write(C45::new(dst_mmd, 0x8000 + dst_offset), (*val).into())?; dst_offset += 1; } // The micro-controller will start running from SRAM. dev.write(C45::new(Mmd::PCS, 0xe854), 0x0040)?; // TODO: sleep here until the hw becomes ready. Ok(()) } fn read_status(dev: &mut phy::Device) -> Result { dev.genphy_read_status::() } }