diff options
| author | Dmitry Torokhov <[email protected]> | 2023-08-30 16:06:38 -0700 | 
|---|---|---|
| committer | Dmitry Torokhov <[email protected]> | 2023-08-30 16:06:38 -0700 | 
| commit | 1ac731c529cd4d6adbce134754b51ff7d822b145 (patch) | |
| tree | 143ab3f35ca5f3b69f583c84e6964b17139c2ec1 /drivers/net/phy/phy_device.c | |
| parent | 07b4c950f27bef0362dc6ad7ee713aab61d58149 (diff) | |
| parent | 54116d442e001e1b6bd482122043b1870998a1f3 (diff) | |
Merge branch 'next' into for-linus
Prepare input updates for 6.6 merge window.
Diffstat (limited to 'drivers/net/phy/phy_device.c')
| -rw-r--r-- | drivers/net/phy/phy_device.c | 127 | 
1 files changed, 123 insertions, 4 deletions
diff --git a/drivers/net/phy/phy_device.c b/drivers/net/phy/phy_device.c index 1785f1cead97..53598210be6c 100644 --- a/drivers/net/phy/phy_device.c +++ b/drivers/net/phy/phy_device.c @@ -19,10 +19,12 @@  #include <linux/interrupt.h>  #include <linux/io.h>  #include <linux/kernel.h> +#include <linux/list.h>  #include <linux/mdio.h>  #include <linux/mii.h>  #include <linux/mm.h>  #include <linux/module.h> +#include <linux/of.h>  #include <linux/netdevice.h>  #include <linux/phy.h>  #include <linux/phy_led_triggers.h> @@ -674,6 +676,7 @@ struct phy_device *phy_device_create(struct mii_bus *bus, int addr, u32 phy_id,  	device_initialize(&mdiodev->dev);  	dev->state = PHY_DOWN; +	INIT_LIST_HEAD(&dev->leds);  	mutex_init(&dev->lock);  	INIT_DELAYED_WORK(&dev->state_queue, phy_state_machine); @@ -2988,6 +2991,115 @@ static bool phy_drv_supports_irq(struct phy_driver *phydrv)  	return phydrv->config_intr && phydrv->handle_interrupt;  } +static int phy_led_set_brightness(struct led_classdev *led_cdev, +				  enum led_brightness value) +{ +	struct phy_led *phyled = to_phy_led(led_cdev); +	struct phy_device *phydev = phyled->phydev; +	int err; + +	mutex_lock(&phydev->lock); +	err = phydev->drv->led_brightness_set(phydev, phyled->index, value); +	mutex_unlock(&phydev->lock); + +	return err; +} + +static int phy_led_blink_set(struct led_classdev *led_cdev, +			     unsigned long *delay_on, +			     unsigned long *delay_off) +{ +	struct phy_led *phyled = to_phy_led(led_cdev); +	struct phy_device *phydev = phyled->phydev; +	int err; + +	mutex_lock(&phydev->lock); +	err = phydev->drv->led_blink_set(phydev, phyled->index, +					 delay_on, delay_off); +	mutex_unlock(&phydev->lock); + +	return err; +} + +static void phy_leds_unregister(struct phy_device *phydev) +{ +	struct phy_led *phyled; + +	list_for_each_entry(phyled, &phydev->leds, list) { +		led_classdev_unregister(&phyled->led_cdev); +	} +} + +static int of_phy_led(struct phy_device *phydev, +		      struct device_node *led) +{ +	struct device *dev = &phydev->mdio.dev; +	struct led_init_data init_data = {}; +	struct led_classdev *cdev; +	struct phy_led *phyled; +	u32 index; +	int err; + +	phyled = devm_kzalloc(dev, sizeof(*phyled), GFP_KERNEL); +	if (!phyled) +		return -ENOMEM; + +	cdev = &phyled->led_cdev; +	phyled->phydev = phydev; + +	err = of_property_read_u32(led, "reg", &index); +	if (err) +		return err; +	if (index > U8_MAX) +		return -EINVAL; + +	phyled->index = index; +	if (phydev->drv->led_brightness_set) +		cdev->brightness_set_blocking = phy_led_set_brightness; +	if (phydev->drv->led_blink_set) +		cdev->blink_set = phy_led_blink_set; +	cdev->max_brightness = 1; +	init_data.devicename = dev_name(&phydev->mdio.dev); +	init_data.fwnode = of_fwnode_handle(led); +	init_data.devname_mandatory = true; + +	err = led_classdev_register_ext(dev, cdev, &init_data); +	if (err) +		return err; + +	list_add(&phyled->list, &phydev->leds); + +	return 0; +} + +static int of_phy_leds(struct phy_device *phydev) +{ +	struct device_node *node = phydev->mdio.dev.of_node; +	struct device_node *leds, *led; +	int err; + +	if (!IS_ENABLED(CONFIG_OF_MDIO)) +		return 0; + +	if (!node) +		return 0; + +	leds = of_get_child_by_name(node, "leds"); +	if (!leds) +		return 0; + +	for_each_available_child_of_node(leds, led) { +		err = of_phy_led(phydev, led); +		if (err) { +			of_node_put(led); +			phy_leds_unregister(phydev); +			return err; +		} +	} + +	return 0; +} +  /**   * fwnode_mdio_find_device - Given a fwnode, find the mdio_device   * @fwnode: pointer to the mdio_device's fwnode @@ -3057,7 +3169,7 @@ EXPORT_SYMBOL_GPL(device_phy_find_device);   * and "phy-device" are not supported in ACPI. DT supports all the three   * named references to the phy node.   */ -struct fwnode_handle *fwnode_get_phy_node(struct fwnode_handle *fwnode) +struct fwnode_handle *fwnode_get_phy_node(const struct fwnode_handle *fwnode)  {  	struct fwnode_handle *phy_node; @@ -3076,9 +3188,7 @@ EXPORT_SYMBOL_GPL(fwnode_get_phy_node);   * phy_probe - probe and init a PHY device   * @dev: device to probe and init   * - * Description: Take care of setting up the phy_device structure, - *   set the state to READY (the driver's init function should - *   set it to STARTING if needed). + * Take care of setting up the phy_device structure, set the state to READY.   */  static int phy_probe(struct device *dev)  { @@ -3185,6 +3295,12 @@ static int phy_probe(struct device *dev)  	/* Set the state to READY by default */  	phydev->state = PHY_READY; +	/* Get the LEDs from the device tree, and instantiate standard +	 * LEDs for them. +	 */ +	if (IS_ENABLED(CONFIG_PHYLIB_LEDS)) +		err = of_phy_leds(phydev); +  out:  	/* Re-assert the reset signal on error */  	if (err) @@ -3199,6 +3315,9 @@ static int phy_remove(struct device *dev)  	cancel_delayed_work_sync(&phydev->state_queue); +	if (IS_ENABLED(CONFIG_PHYLIB_LEDS)) +		phy_leds_unregister(phydev); +  	phydev->state = PHY_DOWN;  	sfp_bus_del_upstream(phydev->sfp_bus);  |