19505a83fSJimmy Assarsson // SPDX-License-Identifier: GPL-2.0
29505a83fSJimmy Assarsson /* kvaser_usb devlink functions
39505a83fSJimmy Assarsson *
49505a83fSJimmy Assarsson * Copyright (C) 2025 KVASER AB, Sweden. All rights reserved.
59505a83fSJimmy Assarsson */
69505a83fSJimmy Assarsson #include "kvaser_usb.h"
79505a83fSJimmy Assarsson
8*aa6a5c99SJimmy Assarsson #include <linux/netdevice.h>
99505a83fSJimmy Assarsson #include <net/devlink.h>
109505a83fSJimmy Assarsson
118720aed9SJimmy Assarsson #define KVASER_USB_EAN_MSB 0x00073301
128720aed9SJimmy Assarsson
kvaser_usb_devlink_info_get(struct devlink * devlink,struct devlink_info_req * req,struct netlink_ext_ack * extack)138720aed9SJimmy Assarsson static int kvaser_usb_devlink_info_get(struct devlink *devlink,
148720aed9SJimmy Assarsson struct devlink_info_req *req,
158720aed9SJimmy Assarsson struct netlink_ext_ack *extack)
168720aed9SJimmy Assarsson {
178720aed9SJimmy Assarsson struct kvaser_usb *dev = devlink_priv(devlink);
188720aed9SJimmy Assarsson char buf[] = "73301XXXXXXXXXX";
198720aed9SJimmy Assarsson int ret;
208720aed9SJimmy Assarsson
218720aed9SJimmy Assarsson if (dev->serial_number) {
228720aed9SJimmy Assarsson snprintf(buf, sizeof(buf), "%u", dev->serial_number);
238720aed9SJimmy Assarsson ret = devlink_info_serial_number_put(req, buf);
248720aed9SJimmy Assarsson if (ret)
258720aed9SJimmy Assarsson return ret;
268720aed9SJimmy Assarsson }
278720aed9SJimmy Assarsson
288720aed9SJimmy Assarsson if (dev->fw_version.major) {
298720aed9SJimmy Assarsson snprintf(buf, sizeof(buf), "%u.%u.%u",
308720aed9SJimmy Assarsson dev->fw_version.major,
318720aed9SJimmy Assarsson dev->fw_version.minor,
328720aed9SJimmy Assarsson dev->fw_version.build);
338720aed9SJimmy Assarsson ret = devlink_info_version_running_put(req,
348720aed9SJimmy Assarsson DEVLINK_INFO_VERSION_GENERIC_FW,
358720aed9SJimmy Assarsson buf);
368720aed9SJimmy Assarsson if (ret)
378720aed9SJimmy Assarsson return ret;
388720aed9SJimmy Assarsson }
398720aed9SJimmy Assarsson
408720aed9SJimmy Assarsson if (dev->hw_revision) {
418720aed9SJimmy Assarsson snprintf(buf, sizeof(buf), "%u", dev->hw_revision);
428720aed9SJimmy Assarsson ret = devlink_info_version_fixed_put(req,
438720aed9SJimmy Assarsson DEVLINK_INFO_VERSION_GENERIC_BOARD_REV,
448720aed9SJimmy Assarsson buf);
458720aed9SJimmy Assarsson if (ret)
468720aed9SJimmy Assarsson return ret;
478720aed9SJimmy Assarsson }
488720aed9SJimmy Assarsson
498720aed9SJimmy Assarsson if (dev->ean[1] == KVASER_USB_EAN_MSB) {
508720aed9SJimmy Assarsson snprintf(buf, sizeof(buf), "%x%08x", dev->ean[1], dev->ean[0]);
518720aed9SJimmy Assarsson ret = devlink_info_version_fixed_put(req,
528720aed9SJimmy Assarsson DEVLINK_INFO_VERSION_GENERIC_BOARD_ID,
538720aed9SJimmy Assarsson buf);
548720aed9SJimmy Assarsson if (ret)
558720aed9SJimmy Assarsson return ret;
568720aed9SJimmy Assarsson }
578720aed9SJimmy Assarsson
588720aed9SJimmy Assarsson return 0;
598720aed9SJimmy Assarsson }
608720aed9SJimmy Assarsson
619505a83fSJimmy Assarsson const struct devlink_ops kvaser_usb_devlink_ops = {
628720aed9SJimmy Assarsson .info_get = kvaser_usb_devlink_info_get,
639505a83fSJimmy Assarsson };
64*aa6a5c99SJimmy Assarsson
kvaser_usb_devlink_port_register(struct kvaser_usb_net_priv * priv)65*aa6a5c99SJimmy Assarsson int kvaser_usb_devlink_port_register(struct kvaser_usb_net_priv *priv)
66*aa6a5c99SJimmy Assarsson {
67*aa6a5c99SJimmy Assarsson int ret;
68*aa6a5c99SJimmy Assarsson struct devlink_port_attrs attrs = {
69*aa6a5c99SJimmy Assarsson .flavour = DEVLINK_PORT_FLAVOUR_PHYSICAL,
70*aa6a5c99SJimmy Assarsson .phys.port_number = priv->channel,
71*aa6a5c99SJimmy Assarsson };
72*aa6a5c99SJimmy Assarsson devlink_port_attrs_set(&priv->devlink_port, &attrs);
73*aa6a5c99SJimmy Assarsson
74*aa6a5c99SJimmy Assarsson ret = devlink_port_register(priv_to_devlink(priv->dev),
75*aa6a5c99SJimmy Assarsson &priv->devlink_port, priv->channel);
76*aa6a5c99SJimmy Assarsson if (ret)
77*aa6a5c99SJimmy Assarsson return ret;
78*aa6a5c99SJimmy Assarsson
79*aa6a5c99SJimmy Assarsson SET_NETDEV_DEVLINK_PORT(priv->netdev, &priv->devlink_port);
80*aa6a5c99SJimmy Assarsson
81*aa6a5c99SJimmy Assarsson return 0;
82*aa6a5c99SJimmy Assarsson }
83*aa6a5c99SJimmy Assarsson
kvaser_usb_devlink_port_unregister(struct kvaser_usb_net_priv * priv)84*aa6a5c99SJimmy Assarsson void kvaser_usb_devlink_port_unregister(struct kvaser_usb_net_priv *priv)
85*aa6a5c99SJimmy Assarsson {
86*aa6a5c99SJimmy Assarsson devlink_port_unregister(&priv->devlink_port);
87*aa6a5c99SJimmy Assarsson }
88