--- /dev/null
+/* Copyright (C) 2016 National Instruments Corp.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+  * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ */
+#include <linux/leds.h>
+#include <linux/phy.h>
+#include <linux/netdevice.h>
+
+static struct phy_led_trigger *phy_speed_to_led_trigger(struct phy_device *phy,
+                                                       unsigned int speed)
+{
+       unsigned int i;
+
+       for (i = 0; i < phy->phy_num_led_triggers; i++) {
+               if (phy->phy_led_triggers[i].speed == speed)
+                       return &phy->phy_led_triggers[i];
+       }
+       return NULL;
+}
+
+void phy_led_trigger_change_speed(struct phy_device *phy)
+{
+       struct phy_led_trigger *plt;
+
+       if (!phy->link)
+               goto out_change_speed;
+
+       if (phy->speed == 0)
+               return;
+
+       plt = phy_speed_to_led_trigger(phy, phy->speed);
+       if (!plt) {
+               netdev_alert(phy->attached_dev,
+                            "No phy led trigger registered for speed(%d)\n",
+                            phy->speed);
+               goto out_change_speed;
+       }
+
+       if (plt != phy->last_triggered) {
+               led_trigger_event(&phy->last_triggered->trigger, LED_OFF);
+               led_trigger_event(&plt->trigger, LED_FULL);
+               phy->last_triggered = plt;
+       }
+       return;
+
+out_change_speed:
+       if (phy->last_triggered) {
+               led_trigger_event(&phy->last_triggered->trigger,
+                                 LED_OFF);
+               phy->last_triggered = NULL;
+       }
+}
+EXPORT_SYMBOL_GPL(phy_led_trigger_change_speed);
+
+static int phy_led_trigger_register(struct phy_device *phy,
+                                   struct phy_led_trigger *plt,
+                                   unsigned int speed)
+{
+       char name_suffix[PHY_LED_TRIGGER_SPEED_SUFFIX_SIZE];
+
+       plt->speed = speed;
+
+       if (speed < SPEED_1000)
+               snprintf(name_suffix, sizeof(name_suffix), "%dMbps", speed);
+       else if (speed == SPEED_2500)
+               snprintf(name_suffix, sizeof(name_suffix), "2.5Gbps");
+       else
+               snprintf(name_suffix, sizeof(name_suffix), "%dGbps",
+                        DIV_ROUND_CLOSEST(speed, 1000));
+
+       snprintf(plt->name, sizeof(plt->name), PHY_ID_FMT ":%s",
+                phy->mdio.bus->id, phy->mdio.addr, name_suffix);
+       plt->trigger.name = plt->name;
+
+       return led_trigger_register(&plt->trigger);
+}
+
+static void phy_led_trigger_unregister(struct phy_led_trigger *plt)
+{
+       led_trigger_unregister(&plt->trigger);
+}
+
+int phy_led_triggers_register(struct phy_device *phy)
+{
+       int i, err;
+       unsigned int speeds[50];
+
+       phy->phy_num_led_triggers = phy_supported_speeds(phy, speeds,
+                                                        ARRAY_SIZE(speeds));
+       if (!phy->phy_num_led_triggers)
+               return 0;
+
+       phy->phy_led_triggers = devm_kzalloc(&phy->mdio.dev,
+                                           sizeof(struct phy_led_trigger) *
+                                                  phy->phy_num_led_triggers,
+                                           GFP_KERNEL);
+       if (!phy->phy_led_triggers)
+               return -ENOMEM;
+
+       for (i = 0; i < phy->phy_num_led_triggers; i++) {
+               err = phy_led_trigger_register(phy, &phy->phy_led_triggers[i],
+                                              speeds[i]);
+               if (err)
+                       goto out_unreg;
+       }
+
+       phy->last_triggered = NULL;
+       phy_led_trigger_change_speed(phy);
+
+       return 0;
+out_unreg:
+       while (i--)
+               phy_led_trigger_unregister(&phy->phy_led_triggers[i]);
+       devm_kfree(&phy->mdio.dev, phy->phy_led_triggers);
+       return err;
+}
+EXPORT_SYMBOL_GPL(phy_led_triggers_register);
+
+void phy_led_triggers_unregister(struct phy_device *phy)
+{
+       int i;
+
+       for (i = 0; i < phy->phy_num_led_triggers; i++)
+               phy_led_trigger_unregister(&phy->phy_led_triggers[i]);
+
+       devm_kfree(&phy->mdio.dev, phy->phy_led_triggers);
+}
+EXPORT_SYMBOL_GPL(phy_led_triggers_unregister);
 
--- /dev/null
+/* Copyright (C) 2016 National Instruments Corp.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ */
+#ifndef __PHY_LED_TRIGGERS
+#define __PHY_LED_TRIGGERS
+
+struct phy_device;
+
+#ifdef CONFIG_LED_TRIGGER_PHY
+
+#include <linux/leds.h>
+
+#define PHY_LED_TRIGGER_SPEED_SUFFIX_SIZE      10
+#define PHY_MII_BUS_ID_SIZE    (20 - 3)
+
+#define PHY_LINK_LED_TRIGGER_NAME_SIZE (PHY_MII_BUS_ID_SIZE + \
+                                      FIELD_SIZEOF(struct mdio_device, addr)+\
+                                      PHY_LED_TRIGGER_SPEED_SUFFIX_SIZE)
+
+struct phy_led_trigger {
+       struct led_trigger trigger;
+       char name[PHY_LINK_LED_TRIGGER_NAME_SIZE];
+       unsigned int speed;
+};
+
+
+extern int phy_led_triggers_register(struct phy_device *phy);
+extern void phy_led_triggers_unregister(struct phy_device *phy);
+extern void phy_led_trigger_change_speed(struct phy_device *phy);
+
+#else
+
+static inline int phy_led_triggers_register(struct phy_device *phy)
+{
+       return 0;
+}
+static inline void phy_led_triggers_unregister(struct phy_device *phy) { }
+static inline void phy_led_trigger_change_speed(struct phy_device *phy) { }
+
+#endif
+
+#endif