mirror of
https://github.com/torvalds/linux.git
synced 2026-04-23 17:15:46 -04:00
Register each CAN channel of the device as an devlink physical port.
This makes it easier to get device information for a given network
interface (i.e. can2).
Example output:
$ devlink dev
usb/1-1.3:1.0
$ devlink port
usb/1-1.3:1.0/0: type eth netdev can0 flavour physical port 0 splittable false
usb/1-1.3:1.0/1: type eth netdev can1 flavour physical port 1 splittable false
$ devlink port show can1
usb/1-1.3:1.0/1: type eth netdev can1 flavour physical port 0 splittable false
$ devlink dev info
usb/1-1.3:1.0:
driver kvaser_usb
serial_number 1020
versions:
fixed:
board.rev 1
board.id 7330130009653
running:
fw 3.22.527
$ ethtool -i can1
driver: kvaser_usb
version: 6.12.10-arch1-1
firmware-version: 3.22.527
expansion-rom-version:
bus-info: 1-1.3:1.0
supports-statistics: no
supports-test: no
supports-eeprom-access: no
supports-register-dump: no
supports-priv-flags: no
Reviewed-by: Vincent Mailhol <mailhol.vincent@wanadoo.fr>
Signed-off-by: Jimmy Assarsson <extja@kvaser.com>
Link: https://patch.msgid.link/20250725123452.41-11-extja@kvaser.com
Signed-off-by: Marc Kleine-Budde <mkl@pengutronix.de>
88 lines
2.1 KiB
C
88 lines
2.1 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/* kvaser_usb devlink functions
|
|
*
|
|
* Copyright (C) 2025 KVASER AB, Sweden. All rights reserved.
|
|
*/
|
|
#include "kvaser_usb.h"
|
|
|
|
#include <linux/netdevice.h>
|
|
#include <net/devlink.h>
|
|
|
|
#define KVASER_USB_EAN_MSB 0x00073301
|
|
|
|
static int kvaser_usb_devlink_info_get(struct devlink *devlink,
|
|
struct devlink_info_req *req,
|
|
struct netlink_ext_ack *extack)
|
|
{
|
|
struct kvaser_usb *dev = devlink_priv(devlink);
|
|
char buf[] = "73301XXXXXXXXXX";
|
|
int ret;
|
|
|
|
if (dev->serial_number) {
|
|
snprintf(buf, sizeof(buf), "%u", dev->serial_number);
|
|
ret = devlink_info_serial_number_put(req, buf);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
if (dev->fw_version.major) {
|
|
snprintf(buf, sizeof(buf), "%u.%u.%u",
|
|
dev->fw_version.major,
|
|
dev->fw_version.minor,
|
|
dev->fw_version.build);
|
|
ret = devlink_info_version_running_put(req,
|
|
DEVLINK_INFO_VERSION_GENERIC_FW,
|
|
buf);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
if (dev->hw_revision) {
|
|
snprintf(buf, sizeof(buf), "%u", dev->hw_revision);
|
|
ret = devlink_info_version_fixed_put(req,
|
|
DEVLINK_INFO_VERSION_GENERIC_BOARD_REV,
|
|
buf);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
if (dev->ean[1] == KVASER_USB_EAN_MSB) {
|
|
snprintf(buf, sizeof(buf), "%x%08x", dev->ean[1], dev->ean[0]);
|
|
ret = devlink_info_version_fixed_put(req,
|
|
DEVLINK_INFO_VERSION_GENERIC_BOARD_ID,
|
|
buf);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
const struct devlink_ops kvaser_usb_devlink_ops = {
|
|
.info_get = kvaser_usb_devlink_info_get,
|
|
};
|
|
|
|
int kvaser_usb_devlink_port_register(struct kvaser_usb_net_priv *priv)
|
|
{
|
|
int ret;
|
|
struct devlink_port_attrs attrs = {
|
|
.flavour = DEVLINK_PORT_FLAVOUR_PHYSICAL,
|
|
.phys.port_number = priv->channel,
|
|
};
|
|
devlink_port_attrs_set(&priv->devlink_port, &attrs);
|
|
|
|
ret = devlink_port_register(priv_to_devlink(priv->dev),
|
|
&priv->devlink_port, priv->channel);
|
|
if (ret)
|
|
return ret;
|
|
|
|
SET_NETDEV_DEVLINK_PORT(priv->netdev, &priv->devlink_port);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void kvaser_usb_devlink_port_unregister(struct kvaser_usb_net_priv *priv)
|
|
{
|
|
devlink_port_unregister(&priv->devlink_port);
|
|
}
|