xref: /linux/drivers/net/can/usb/kvaser_usb/kvaser_usb_devlink.c (revision 8be4d31cb8aaeea27bde4b7ddb26e28a89062ebf)
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