forked from mirrors/linux
		
	 60495b6622
			
		
	
	
		60495b6622
		
	
	
	
	
		
			
			net/core/dev_ioctl.c (built-in code) will want to call phy_mii_ioctl()
for hardware timestamping purposes. This is not directly possible,
because phy_mii_ioctl() is a symbol provided under CONFIG_PHYLIB.
Do something similar to what was done in DSA in commit 5a17818682
("net: dsa: replace NETDEV_PRE_CHANGE_HWTSTAMP notifier with a stub"),
and arrange some indirect calls to phy_mii_ioctl() through a stub
structure containing function pointers, that's provided by phylib as
built-in even when CONFIG_PHYLIB=m, and which phy_init() populates at
runtime (module insertion).
Note: maybe the ownership of the ethtool_phy_ops singleton is backwards,
and the methods exposed by that should be later merged into phylib_stubs.
Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com>
Link: https://lore.kernel.org/r/20230801142824.1772134-12-vladimir.oltean@nxp.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
		
	
			
		
			
				
	
	
		
			68 lines
		
	
	
	
		
			1.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			68 lines
		
	
	
	
		
			1.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* SPDX-License-Identifier: GPL-2.0-or-later */
 | |
| /*
 | |
|  * Stubs for the Network PHY library
 | |
|  */
 | |
| 
 | |
| #include <linux/rtnetlink.h>
 | |
| 
 | |
| struct kernel_hwtstamp_config;
 | |
| struct netlink_ext_ack;
 | |
| struct phy_device;
 | |
| 
 | |
| #if IS_ENABLED(CONFIG_PHYLIB)
 | |
| 
 | |
| extern const struct phylib_stubs *phylib_stubs;
 | |
| 
 | |
| struct phylib_stubs {
 | |
| 	int (*hwtstamp_get)(struct phy_device *phydev,
 | |
| 			    struct kernel_hwtstamp_config *config);
 | |
| 	int (*hwtstamp_set)(struct phy_device *phydev,
 | |
| 			    struct kernel_hwtstamp_config *config,
 | |
| 			    struct netlink_ext_ack *extack);
 | |
| };
 | |
| 
 | |
| static inline int phy_hwtstamp_get(struct phy_device *phydev,
 | |
| 				   struct kernel_hwtstamp_config *config)
 | |
| {
 | |
| 	/* phylib_register_stubs() and phylib_unregister_stubs()
 | |
| 	 * also run under rtnl_lock().
 | |
| 	 */
 | |
| 	ASSERT_RTNL();
 | |
| 
 | |
| 	if (!phylib_stubs)
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	return phylib_stubs->hwtstamp_get(phydev, config);
 | |
| }
 | |
| 
 | |
| static inline int phy_hwtstamp_set(struct phy_device *phydev,
 | |
| 				   struct kernel_hwtstamp_config *config,
 | |
| 				   struct netlink_ext_ack *extack)
 | |
| {
 | |
| 	/* phylib_register_stubs() and phylib_unregister_stubs()
 | |
| 	 * also run under rtnl_lock().
 | |
| 	 */
 | |
| 	ASSERT_RTNL();
 | |
| 
 | |
| 	if (!phylib_stubs)
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	return phylib_stubs->hwtstamp_set(phydev, config, extack);
 | |
| }
 | |
| 
 | |
| #else
 | |
| 
 | |
| static inline int phy_hwtstamp_get(struct phy_device *phydev,
 | |
| 				   struct kernel_hwtstamp_config *config)
 | |
| {
 | |
| 	return -EOPNOTSUPP;
 | |
| }
 | |
| 
 | |
| static inline int phy_hwtstamp_set(struct phy_device *phydev,
 | |
| 				   struct kernel_hwtstamp_config *config,
 | |
| 				   struct netlink_ext_ack *extack)
 | |
| {
 | |
| 	return -EOPNOTSUPP;
 | |
| }
 | |
| 
 | |
| #endif
 |