mirror of
https://github.com/openwrt/openwrt.git
synced 2025-01-12 16:03:13 +00:00
cddd459140
Add patches for linux-5.4. The patches are from NXP LSDK-20.04 release which was tagged LSDK-20.04-V5.4. https://source.codeaurora.org/external/qoriq/qoriq-components/linux/ For boards LS1021A-IOT, and Traverse-LS1043 which are not involved in LSDK, port the dts patches from 4.14. The patches are sorted into the following categories: 301-arch-xxxx 302-dts-xxxx 303-core-xxxx 701-net-xxxx 801-audio-xxxx 802-can-xxxx 803-clock-xxxx 804-crypto-xxxx 805-display-xxxx 806-dma-xxxx 807-gpio-xxxx 808-i2c-xxxx 809-jailhouse-xxxx 810-keys-xxxx 811-kvm-xxxx 812-pcie-xxxx 813-pm-xxxx 814-qe-xxxx 815-sata-xxxx 816-sdhc-xxxx 817-spi-xxxx 818-thermal-xxxx 819-uart-xxxx 820-usb-xxxx 821-vfio-xxxx Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
5114 lines
132 KiB
Diff
5114 lines
132 KiB
Diff
From e478ab518612f1a821968e1bb5b08b01b10085b0 Mon Sep 17 00:00:00 2001
|
|
From: Po Liu <Po.Liu@nxp.com>
|
|
Date: Tue, 15 Oct 2019 16:11:40 +0800
|
|
Subject: [PATCH] net:tsn: netlink interface for APP layer to config TSN
|
|
capability hardware ports
|
|
|
|
This patch provids netlink method to configure the TSN protocols hardwares.
|
|
TSN guaranteed packet transport with bounded low latency, low packet delay
|
|
variation, and low packet loss by hardware and software methods.
|
|
|
|
The three basic components of TSN are:
|
|
|
|
1. Time synchronization: This was implement by 8021AS which base on the
|
|
IEEE1588 precision Time Protocol. This is configured by the other way
|
|
in kernel.
|
|
8021AS not included in this patch.
|
|
|
|
2. Scheduling and traffic shaping and per-stream filter policing:
|
|
This patch support Qbv/Qci/Qbu/8021CB/Qav etc.
|
|
|
|
3. Selection of communication paths:
|
|
This patch not support the pure software only TSN protocols(like Qcc)
|
|
but hardware related configuration.
|
|
|
|
TSN Protocols supports by this patch: Qbv/Qci/Qbu/Credit-base Shaper(Qav).
|
|
This patch verified on NXP ls1028ardb board.
|
|
|
|
Signed-off-by: Po Liu <Po.Liu@nxp.com>
|
|
---
|
|
include/net/tsn.h | 114 ++
|
|
include/uapi/linux/tsn.h | 1207 +++++++++++++++
|
|
net/Kconfig | 1 +
|
|
net/Makefile | 3 +
|
|
net/tsn/Kconfig | 15 +
|
|
net/tsn/Makefile | 1 +
|
|
net/tsn/genl_tsn.c | 3696 ++++++++++++++++++++++++++++++++++++++++++++++
|
|
7 files changed, 5037 insertions(+)
|
|
create mode 100644 include/net/tsn.h
|
|
create mode 100644 include/uapi/linux/tsn.h
|
|
create mode 100644 net/tsn/Kconfig
|
|
create mode 100644 net/tsn/Makefile
|
|
create mode 100644 net/tsn/genl_tsn.c
|
|
|
|
--- /dev/null
|
|
+++ b/include/net/tsn.h
|
|
@@ -0,0 +1,114 @@
|
|
+/* SPDX-License-Identifier: (GPL-2.0+ OR BSD-3-Clause) */
|
|
+/* Copyright 2017-2019 NXP */
|
|
+
|
|
+#ifndef __TSN_H__
|
|
+#define __TSN_H__
|
|
+
|
|
+#include <linux/notifier.h>
|
|
+#include <uapi/linux/tsn.h>
|
|
+
|
|
+enum tsn_notifier_type {
|
|
+ TSN_QBV_CONFIGCHANGETIME_ARRIVE = 1,
|
|
+};
|
|
+
|
|
+struct tsn_notifier_info {
|
|
+ struct net_device *dev;
|
|
+ union {
|
|
+ struct tsn_qbv_conf qbv_notify;
|
|
+ struct tsn_qci_psfp_sgi_conf qci_notify;
|
|
+ } ntdata;
|
|
+};
|
|
+
|
|
+static inline struct net_device *
|
|
+tsn_notifier_info_to_dev(const struct tsn_notifier_info *info)
|
|
+{
|
|
+ return info->dev;
|
|
+}
|
|
+
|
|
+struct tsn_ops {
|
|
+ void (*device_init)(struct net_device *ndev);
|
|
+ void (*device_deinit)(struct net_device *ndev);
|
|
+ u32 (*get_capability)(struct net_device *ndev);
|
|
+ /* Qbv standard */
|
|
+ int (*qbv_set)(struct net_device *ndev, struct tsn_qbv_conf *qbvconf);
|
|
+ int (*qbv_get)(struct net_device *ndev, struct tsn_qbv_conf *qbvconf);
|
|
+ int (*qbv_get_status)(struct net_device *ndev,
|
|
+ struct tsn_qbv_status *qbvstat);
|
|
+ int (*cb_streamid_set)(struct net_device *ndev, u32 index,
|
|
+ bool enable, struct tsn_cb_streamid *sid);
|
|
+ int (*cb_streamid_get)(struct net_device *ndev, u32 index,
|
|
+ struct tsn_cb_streamid *sid);
|
|
+ int (*cb_streamid_counters_get)(struct net_device *ndev, u32 index,
|
|
+ struct tsn_cb_streamid_counters *sidcounter);
|
|
+ int (*qci_get_maxcap)(struct net_device *ndev,
|
|
+ struct tsn_qci_psfp_stream_param *qcicapa);
|
|
+ int (*qci_sfi_set)(struct net_device *ndev, u32 index, bool enable,
|
|
+ struct tsn_qci_psfp_sfi_conf *sficonf);
|
|
+ /* return: 0 stream filter instance not valid
|
|
+ * 1 stream filter instance valid
|
|
+ * -1 error happened
|
|
+ */
|
|
+ int (*qci_sfi_get)(struct net_device *ndev, u32 index,
|
|
+ struct tsn_qci_psfp_sfi_conf *sficonf);
|
|
+ int (*qci_sfi_counters_get)(struct net_device *ndev, u32 index,
|
|
+ struct tsn_qci_psfp_sfi_counters *sficounter);
|
|
+ int (*qci_sgi_set)(struct net_device *ndev, u32 index,
|
|
+ struct tsn_qci_psfp_sgi_conf *sgiconf);
|
|
+ int (*qci_sgi_get)(struct net_device *ndev, u32 index,
|
|
+ struct tsn_qci_psfp_sgi_conf *sgiconf);
|
|
+ int (*qci_sgi_status_get)(struct net_device *ndev, u16 index,
|
|
+ struct tsn_psfp_sgi_status *sgistat);
|
|
+ int (*qci_fmi_set)(struct net_device *ndev, u32 index, bool enable,
|
|
+ struct tsn_qci_psfp_fmi *fmi);
|
|
+ int (*qci_fmi_get)(struct net_device *ndev, u32 index,
|
|
+ struct tsn_qci_psfp_fmi *fmi,
|
|
+ struct tsn_qci_psfp_fmi_counters *counters);
|
|
+ int (*cbs_set)(struct net_device *ndev, u8 tc, u8 bw);
|
|
+ int (*cbs_get)(struct net_device *ndev, u8 tc);
|
|
+ /* To set a 8 bits vector shows 8 traffic classes
|
|
+ * preemtable(1) or express(0)
|
|
+ */
|
|
+ int (*qbu_set)(struct net_device *ndev, u8 ptvector);
|
|
+ /* To get port preemtion status */
|
|
+ int (*qbu_get)(struct net_device *ndev,
|
|
+ struct tsn_preempt_status *preemptstat);
|
|
+ int (*tsd_set)(struct net_device *ndev, struct tsn_tsd *tsd);
|
|
+ int (*tsd_get)(struct net_device *ndev, struct tsn_tsd_status *stats);
|
|
+ int (*ct_set)(struct net_device *ndev, u8 cut_thru);
|
|
+ int (*cbgen_set)(struct net_device *ndev, u32 index,
|
|
+ struct tsn_seq_gen_conf *seqgen);
|
|
+ int (*cbrec_set)(struct net_device *ndev, u32 index,
|
|
+ struct tsn_seq_rec_conf *seqrec);
|
|
+ int (*cb_get)(struct net_device *ndev, u32 index,
|
|
+ struct tsn_cb_status *c);
|
|
+ int (*dscp_set)(struct net_device *ndev, bool enable,
|
|
+ const u8 dscp_ix,
|
|
+ struct tsn_qos_switch_dscp_conf *c);
|
|
+};
|
|
+
|
|
+enum ethdev_type {
|
|
+ TSN_SWITCH,
|
|
+ TSN_ENDPOINT,
|
|
+};
|
|
+
|
|
+#define GROUP_OFFSET_SWITCH 256
|
|
+
|
|
+struct tsn_port {
|
|
+ u16 groupid;
|
|
+ struct tsn_ops *tsnops;
|
|
+ struct net_device *netdev;
|
|
+ struct list_head list;
|
|
+ enum ethdev_type type;
|
|
+ u8 tc_nums;
|
|
+ struct tsn_notifier_info nd;
|
|
+};
|
|
+
|
|
+struct tsn_port *tsn_get_port(struct net_device *ndev);
|
|
+int register_tsn_notifier(struct notifier_block *nb);
|
|
+int unregister_tsn_notifier(struct notifier_block *nb);
|
|
+int call_tsn_notifiers(unsigned long val, struct net_device *dev,
|
|
+ struct tsn_notifier_info *info);
|
|
+int tsn_port_register(struct net_device *netdev,
|
|
+ struct tsn_ops *tsnops, u16 groupid);
|
|
+void tsn_port_unregister(struct net_device *netdev);
|
|
+#endif
|
|
--- /dev/null
|
|
+++ b/include/uapi/linux/tsn.h
|
|
@@ -0,0 +1,1207 @@
|
|
+/* SPDX-License-Identifier: (GPL-2.0+ OR BSD-3-Clause) */
|
|
+/* Copyright 2017-2019 NXP */
|
|
+
|
|
+#ifndef __UAPI_GENL_TSN_H
|
|
+#define __UAPI_GENL_TSN_H
|
|
+
|
|
+#define TSN_GENL_NAME "TSN_GEN_CTRL"
|
|
+#define TSN_GENL_VERSION 0x1
|
|
+
|
|
+#define MAX_USER_SIZE 0
|
|
+#define MAX_ATTR_SIZE 3072
|
|
+#define MAX_TOTAL_MSG_SIZE (MAX_USER_SIZE + MAX_ATTR_SIZE)
|
|
+#define MAX_ENTRY_SIZE 2048
|
|
+#define MAX_ENTRY_NUMBER 128
|
|
+#define MAX_IFNAME_COUNT 64
|
|
+
|
|
+#define TSN_MULTICAST_GROUP_QBV "qbv"
|
|
+#define TSN_MULTICAST_GROUP_QCI "qci"
|
|
+
|
|
+/* multicast groups */
|
|
+enum tsn_multicast_groups {
|
|
+ TSN_MCGRP_QBV,
|
|
+ TSN_MCGRP_QCI,
|
|
+ TSN_MCGRP_MAX
|
|
+};
|
|
+
|
|
+enum tsn_capability {
|
|
+ TSN_CAP_QBV = 0x1,
|
|
+ TSN_CAP_QCI = 0x2,
|
|
+ TSN_CAP_QBU = 0x4,
|
|
+ TSN_CAP_CBS = 0x8, /* Credit-based Shapter Qav */
|
|
+ TSN_CAP_CB = 0x10, /* 8021CB redundancy and replication */
|
|
+ TSN_CAP_TBS = 0x20, /* Time Based schedule */
|
|
+ TSN_CAP_CTH = 0x40, /* cut through */
|
|
+};
|
|
+
|
|
+/*
|
|
+ * Commands sent from userspace
|
|
+ * Not versioned. New commands should only be inserted at the enum's end
|
|
+ * prior to __TSN_CMD_MAX
|
|
+ */
|
|
+
|
|
+enum {
|
|
+ TSN_CMD_UNSPEC = 0, /* Reserved */
|
|
+ TSN_CMD_QBV_SET,
|
|
+ TSN_CMD_QBV_GET,
|
|
+ TSN_CMD_QBV_GET_STATUS,
|
|
+ TSN_CMD_CB_STREAMID_SET,
|
|
+ TSN_CMD_CB_STREAMID_GET,
|
|
+ TSN_CMD_CB_STREAMID_GET_COUNTS,
|
|
+ TSN_CMD_QCI_CAP_GET, /* Qci capability get length capability get */
|
|
+ TSN_CMD_QCI_SFI_SET,
|
|
+ TSN_CMD_QCI_SFI_GET,
|
|
+ TSN_CMD_QCI_SFI_GET_COUNTS,
|
|
+ TSN_CMD_QCI_SGI_SET,
|
|
+ TSN_CMD_QCI_SGI_GET,
|
|
+ TSN_CMD_QCI_SGI_GET_STATUS,
|
|
+ TSN_CMD_QCI_FMI_SET,
|
|
+ TSN_CMD_QCI_FMI_GET,
|
|
+ TSN_CMD_CBS_SET,
|
|
+ TSN_CMD_CBS_GET,
|
|
+ TSN_CMD_QBU_SET,
|
|
+ TSN_CMD_QBU_GET_STATUS,
|
|
+ TSN_CMD_QAV_SET_CBS,
|
|
+ TSN_CMD_QAV_GET_CBS,
|
|
+ TSN_CMD_TSD_SET,
|
|
+ TSN_CMD_TSD_GET,
|
|
+ TSN_CMD_CT_SET,
|
|
+ TSN_CMD_CBGEN_SET,
|
|
+ TSN_CMD_CBREC_SET,
|
|
+ TSN_CMD_CBSTAT_GET,
|
|
+ TSN_CMD_PCPMAP_SET_UNUSE,
|
|
+ TSN_CMD_DSCP_SET,
|
|
+ TSN_CMD_ECHO, /* user->kernel request/get-response */
|
|
+ TSN_CMD_REPLY, /* kernel->user event */
|
|
+ TSN_CMD_CAP_GET,
|
|
+ __TSN_CMD_MAX,
|
|
+};
|
|
+#define TSN_CMD_MAX (__TSN_CMD_MAX - 1)
|
|
+
|
|
+
|
|
+enum {
|
|
+ TSN_CMD_ATTR_UNSPEC = 0,
|
|
+ TSN_CMD_ATTR_MESG, /* demo message */
|
|
+ TSN_CMD_ATTR_DATA, /* demo data */
|
|
+ TSN_ATTR_IFNAME,
|
|
+ TSN_ATTR_PORT_NUMBER,
|
|
+ TSN_ATTR_QBV,
|
|
+ TSN_ATTR_STREAM_IDENTIFY, /* stream identify */
|
|
+ TSN_ATTR_QCI_SP, /* psfp port capbility parameters */
|
|
+ TSN_ATTR_QCI_SFI, /* psfp stream filter instance */
|
|
+ TSN_ATTR_QCI_SGI, /* psfp stream gate instance */
|
|
+ TSN_ATTR_QCI_FMI, /* psfp flow meter instance */
|
|
+ TSN_ATTR_CBS, /* credit-based shaper */
|
|
+ TSN_ATTR_TSD, /* Time Specific Departure */
|
|
+ TSN_ATTR_QBU, /* preemption */
|
|
+ TSN_ATTR_CT, /* cut through */
|
|
+ TSN_ATTR_CBGEN, /* 802.1CB sequence generate */
|
|
+ TSN_ATTR_CBREC, /* 802.1CB sequence recover */
|
|
+ TSN_ATTR_CBSTAT, /* 802.1CB status */
|
|
+ TSN_ATTR_PCPMAP_UNUSE,
|
|
+ TSN_ATTR_DSCP,
|
|
+ TSN_ATTR_CAP, /* TSN capbility */
|
|
+ __TSN_CMD_ATTR_MAX,
|
|
+};
|
|
+#define TSN_CMD_ATTR_MAX (__TSN_CMD_ATTR_MAX - 1)
|
|
+
|
|
+enum {
|
|
+ TSN_CAP_ATTR_UNSPEC,
|
|
+ TSN_CAP_ATTR_QBV,
|
|
+ TSN_CAP_ATTR_QCI,
|
|
+ TSN_CAP_ATTR_QBU,
|
|
+ TSN_CAP_ATTR_CBS,
|
|
+ TSN_CAP_ATTR_CB,
|
|
+ TSN_CAP_ATTR_TBS,
|
|
+ TSN_CAP_ATTR_CTH,
|
|
+ __TSN_CAP_ATTR_MAX,
|
|
+ TSN_CAP_ATTR_MAX = __TSN_CAP_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_QBU_ATTR_UNSPEC,
|
|
+ TSN_QBU_ATTR_ADMIN_STATE,
|
|
+ TSN_QBU_ATTR_HOLD_ADVANCE,
|
|
+ TSN_QBU_ATTR_RELEASE_ADVANCE,
|
|
+ TSN_QBU_ATTR_ACTIVE,
|
|
+ TSN_QBU_ATTR_HOLD_REQUEST,
|
|
+ __TSN_QBU_ATTR_MAX,
|
|
+ TSN_QBU_ATTR_MAX = __TSN_QBU_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_CBS_ATTR_UNSPEC,
|
|
+ TSN_CBS_ATTR_TC_INDEX,
|
|
+ TSN_CBS_ATTR_BW,
|
|
+ __TSN_CBS_ATTR_MAX,
|
|
+ TSN_CBS_ATTR_MAX = __TSN_CBS_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_TSD_ATTR_UNSPEC,
|
|
+ TSN_TSD_ATTR_DISABLE,
|
|
+ TSN_TSD_ATTR_ENABLE,
|
|
+ TSN_TSD_ATTR_PERIOD,
|
|
+ TSN_TSD_ATTR_MAX_FRM_NUM,
|
|
+ TSN_TSD_ATTR_CYCLE_NUM,
|
|
+ TSN_TSD_ATTR_LOSS_STEPS,
|
|
+ TSN_TSD_ATTR_SYN_IMME,
|
|
+ __TSN_TSD_ATTR_MAX,
|
|
+ TSN_TSD_ATTR_MAX = __TSN_TSD_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_STREAMID_ATTR_UNSPEC,
|
|
+ TSN_STREAMID_ATTR_INDEX,
|
|
+ TSN_STREAMID_ATTR_ENABLE,
|
|
+ TSN_STREAMID_ATTR_DISABLE,
|
|
+ TSN_STREAMID_ATTR_STREAM_HANDLE,
|
|
+ TSN_STREAMID_ATTR_IFOP,
|
|
+ TSN_STREAMID_ATTR_OFOP,
|
|
+ TSN_STREAMID_ATTR_IFIP,
|
|
+ TSN_STREAMID_ATTR_OFIP,
|
|
+ TSN_STREAMID_ATTR_TYPE,
|
|
+ TSN_STREAMID_ATTR_NDMAC,
|
|
+ TSN_STREAMID_ATTR_NTAGGED,
|
|
+ TSN_STREAMID_ATTR_NVID,
|
|
+ TSN_STREAMID_ATTR_SMAC,
|
|
+ TSN_STREAMID_ATTR_STAGGED,
|
|
+ TSN_STREAMID_ATTR_SVID,
|
|
+ TSN_STREAMID_ATTR_COUNTERS_PSI,
|
|
+ TSN_STREAMID_ATTR_COUNTERS_PSO,
|
|
+ TSN_STREAMID_ATTR_COUNTERS_PSPPI,
|
|
+ TSN_STREAMID_ATTR_COUNTERS_PSPPO,
|
|
+ __TSN_STREAMID_ATTR_MAX,
|
|
+ TSN_STREAMID_ATTR_MAX = __TSN_STREAMID_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_QCI_STREAM_ATTR_UNSPEC = 0,
|
|
+ TSN_QCI_STREAM_ATTR_MAX_SFI,
|
|
+ TSN_QCI_STREAM_ATTR_MAX_SGI,
|
|
+ TSN_QCI_STREAM_ATTR_MAX_FMI,
|
|
+ TSN_QCI_STREAM_ATTR_SLM,
|
|
+ __TSN_QCI_STREAM_ATTR_MAX,
|
|
+ TSN_QCI_STREAM_ATTR_MAX = __TSN_QCI_STREAM_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_QCI_SFI_ATTR_UNSPEC = 0,
|
|
+ TSN_QCI_SFI_ATTR_INDEX,
|
|
+ TSN_QCI_SFI_ATTR_ENABLE,
|
|
+ TSN_QCI_SFI_ATTR_DISABLE,
|
|
+ TSN_QCI_SFI_ATTR_STREAM_HANDLE,
|
|
+ TSN_QCI_SFI_ATTR_PRIO_SPEC,
|
|
+ TSN_QCI_SFI_ATTR_GATE_ID,
|
|
+ TSN_QCI_SFI_ATTR_FILTER_TYPE,
|
|
+ TSN_QCI_SFI_ATTR_FLOW_ID,
|
|
+ TSN_QCI_SFI_ATTR_MAXSDU,
|
|
+ TSN_QCI_SFI_ATTR_COUNTERS,
|
|
+ TSN_QCI_SFI_ATTR_OVERSIZE_ENABLE,
|
|
+ TSN_QCI_SFI_ATTR_OVERSIZE,
|
|
+ __TSN_QCI_SFI_ATTR_MAX,
|
|
+ TSN_QCI_SFI_ATTR_MAX = __TSN_QCI_SFI_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_QCI_SFI_ATTR_COUNTERS_UNSPEC = 0,
|
|
+ TSN_QCI_SFI_ATTR_MATCH,
|
|
+ TSN_QCI_SFI_ATTR_PASS,
|
|
+ TSN_QCI_SFI_ATTR_DROP,
|
|
+ TSN_QCI_SFI_ATTR_SDU_DROP,
|
|
+ TSN_QCI_SFI_ATTR_SDU_PASS,
|
|
+ TSN_QCI_SFI_ATTR_RED,
|
|
+ __TSN_QCI_SFI_ATTR_COUNT_MAX,
|
|
+ TSN_QCI_SFI_ATTR_COUNT_MAX = __TSN_QCI_SFI_ATTR_COUNT_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_QCI_SGI_ATTR_UNSPEC = 0,
|
|
+ TSN_QCI_SGI_ATTR_INDEX,
|
|
+ TSN_QCI_SGI_ATTR_ENABLE,
|
|
+ TSN_QCI_SGI_ATTR_DISABLE,
|
|
+ TSN_QCI_SGI_ATTR_CONFCHANGE,
|
|
+ TSN_QCI_SGI_ATTR_IRXEN, /* Invalid rx enable*/
|
|
+ TSN_QCI_SGI_ATTR_IRX,
|
|
+ TSN_QCI_SGI_ATTR_OEXEN, /* Octet exceed enable */
|
|
+ TSN_QCI_SGI_ATTR_OEX,
|
|
+ TSN_QCI_SGI_ATTR_ADMINENTRY,
|
|
+ TSN_QCI_SGI_ATTR_OPERENTRY,
|
|
+ TSN_QCI_SGI_ATTR_CCTIME, /* config change time */
|
|
+ TSN_QCI_SGI_ATTR_TICKG,
|
|
+ TSN_QCI_SGI_ATTR_CUTIME,
|
|
+ TSN_QCI_SGI_ATTR_CPENDING,
|
|
+ TSN_QCI_SGI_ATTR_CCERROR,
|
|
+ __TSN_QCI_SGI_ATTR_MAX,
|
|
+ TSN_QCI_SGI_ATTR_MAX = __TSN_QCI_SGI_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_SGI_ATTR_CTRL_UNSPEC = 0,
|
|
+ TSN_SGI_ATTR_CTRL_INITSTATE,
|
|
+ TSN_SGI_ATTR_CTRL_LEN,
|
|
+ TSN_SGI_ATTR_CTRL_CYTIME,
|
|
+ TSN_SGI_ATTR_CTRL_CYTIMEEX,
|
|
+ TSN_SGI_ATTR_CTRL_BTIME,
|
|
+ TSN_SGI_ATTR_CTRL_INITIPV,
|
|
+ TSN_SGI_ATTR_CTRL_GCLENTRY,
|
|
+ __TSN_SGI_ATTR_CTRL_MAX,
|
|
+ TSN_SGI_ATTR_CTRL_MAX = __TSN_SGI_ATTR_CTRL_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_SGI_ATTR_GCL_UNSPEC = 0,
|
|
+ TSN_SGI_ATTR_GCL_GATESTATE,
|
|
+ TSN_SGI_ATTR_GCL_IPV,
|
|
+ TSN_SGI_ATTR_GCL_INTERVAL,
|
|
+ TSN_SGI_ATTR_GCL_OCTMAX,
|
|
+ __TSN_SGI_ATTR_GCL_MAX,
|
|
+ TSN_SGI_ATTR_GCL_MAX = __TSN_SGI_ATTR_GCL_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_QCI_FMI_ATTR_UNSPEC = 0,
|
|
+ TSN_QCI_FMI_ATTR_INDEX,
|
|
+ TSN_QCI_FMI_ATTR_ENABLE,
|
|
+ TSN_QCI_FMI_ATTR_DISABLE,
|
|
+ TSN_QCI_FMI_ATTR_CIR,
|
|
+ TSN_QCI_FMI_ATTR_CBS,
|
|
+ TSN_QCI_FMI_ATTR_EIR,
|
|
+ TSN_QCI_FMI_ATTR_EBS,
|
|
+ TSN_QCI_FMI_ATTR_CF,
|
|
+ TSN_QCI_FMI_ATTR_CM,
|
|
+ TSN_QCI_FMI_ATTR_DROPYL,
|
|
+ TSN_QCI_FMI_ATTR_MAREDEN,
|
|
+ TSN_QCI_FMI_ATTR_MARED,
|
|
+ TSN_QCI_FMI_ATTR_COUNTERS,
|
|
+ __TSN_QCI_FMI_ATTR_MAX,
|
|
+ TSN_QCI_FMI_ATTR_MAX = __TSN_QCI_FMI_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_QBV_ATTR_UNSPEC,
|
|
+ TSN_QBV_ATTR_ENABLE,
|
|
+ TSN_QBV_ATTR_DISABLE,
|
|
+ TSN_QBV_ATTR_CONFIGCHANGE,
|
|
+ TSN_QBV_ATTR_CONFIGCHANGETIME,
|
|
+ TSN_QBV_ATTR_MAXSDU,
|
|
+ TSN_QBV_ATTR_GRANULARITY,
|
|
+ TSN_QBV_ATTR_CURRENTTIME,
|
|
+ TSN_QBV_ATTR_CONFIGPENDING,
|
|
+ TSN_QBV_ATTR_CONFIGCHANGEERROR,
|
|
+ TSN_QBV_ATTR_ADMINENTRY,
|
|
+ TSN_QBV_ATTR_OPERENTRY,
|
|
+ TSN_QBV_ATTR_LISTMAX,
|
|
+ __TSN_QBV_ATTR_MAX,
|
|
+ TSN_QBV_ATTR_MAX = __TSN_QBV_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_QBV_ATTR_CTRL_UNSPEC,
|
|
+ TSN_QBV_ATTR_CTRL_LISTCOUNT,
|
|
+ TSN_QBV_ATTR_CTRL_GATESTATE,
|
|
+ TSN_QBV_ATTR_CTRL_CYCLETIME,
|
|
+ TSN_QBV_ATTR_CTRL_CYCLETIMEEXT,
|
|
+ TSN_QBV_ATTR_CTRL_BASETIME,
|
|
+ TSN_QBV_ATTR_CTRL_LISTENTRY,
|
|
+ __TSN_QBV_ATTR_CTRL_MAX,
|
|
+ TSN_QBV_ATTR_CTRL_MAX = __TSN_QBV_ATTR_CTRL_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_QBV_ATTR_ENTRY_UNSPEC,
|
|
+ TSN_QBV_ATTR_ENTRY_ID,
|
|
+ TSN_QBV_ATTR_ENTRY_GC,
|
|
+ TSN_QBV_ATTR_ENTRY_TM,
|
|
+ __TSN_QBV_ATTR_ENTRY_MAX,
|
|
+ TSN_QBV_ATTR_ENTRY_MAX = __TSN_QBV_ATTR_ENTRY_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_CT_ATTR_UNSPEC,
|
|
+ TSN_CT_ATTR_QUEUE_STATE,
|
|
+ __TSN_CT_ATTR_MAX,
|
|
+ TSN_CT_ATTR_MAX = __TSN_CT_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_CBGEN_ATTR_UNSPEC,
|
|
+ TSN_CBGEN_ATTR_INDEX,
|
|
+ TSN_CBGEN_ATTR_PORT_MASK,
|
|
+ TSN_CBGEN_ATTR_SPLIT_MASK,
|
|
+ TSN_CBGEN_ATTR_SEQ_LEN,
|
|
+ TSN_CBGEN_ATTR_SEQ_NUM,
|
|
+ __TSN_CBGEN_ATTR_MAX,
|
|
+ TSN_CBGEN_ATTR_MAX = __TSN_CBGEN_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_CBREC_ATTR_UNSPEC,
|
|
+ TSN_CBREC_ATTR_INDEX,
|
|
+ TSN_CBREC_ATTR_SEQ_LEN,
|
|
+ TSN_CBREC_ATTR_HIS_LEN,
|
|
+ TSN_CBREC_ATTR_TAG_POP_EN,
|
|
+ __TSN_CBREC_ATTR_MAX,
|
|
+ TSN_CBREC_ATTR_MAX = __TSN_CBREC_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_CBSTAT_ATTR_UNSPEC,
|
|
+ TSN_CBSTAT_ATTR_INDEX,
|
|
+ TSN_CBSTAT_ATTR_GEN_REC,
|
|
+ TSN_CBSTAT_ATTR_ERR,
|
|
+ TSN_CBSTAT_ATTR_SEQ_NUM,
|
|
+ TSN_CBSTAT_ATTR_SEQ_LEN,
|
|
+ TSN_CBSTAT_ATTR_SPLIT_MASK,
|
|
+ TSN_CBSTAT_ATTR_PORT_MASK,
|
|
+ TSN_CBSTAT_ATTR_HIS_LEN,
|
|
+ TSN_CBSTAT_ATTR_SEQ_HIS,
|
|
+ __TSN_CBSTAT_ATTR_MAX,
|
|
+ TSN_CBSTAT_ATTR_MAX = __TSN_CBSTAT_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+enum {
|
|
+ TSN_DSCP_ATTR_UNSPEC,
|
|
+ TSN_DSCP_ATTR_DISABLE,
|
|
+ TSN_DSCP_ATTR_INDEX,
|
|
+ TSN_DSCP_ATTR_COS,
|
|
+ TSN_DSCP_ATTR_DPL,
|
|
+ __TSN_DSCP_ATTR_MAX,
|
|
+ TSN_DSCP_ATTR_MAX = __TSN_DSCP_ATTR_MAX - 1,
|
|
+};
|
|
+
|
|
+#define ptptime_t __u64
|
|
+
|
|
+#define MAX_QUEUE_CNT 8
|
|
+
|
|
+struct tsn_preempt_status {
|
|
+ /* The value of admin_state shows a 8-bits vector value for showing
|
|
+ * the framePreemptionAdminStatus parameter and PreemptionPriority
|
|
+ * for the traffic class. Bit-7 is the highest priority traffic class
|
|
+ * and the bit-0 is the lowest priority traffic class.
|
|
+ * The bit is express (0) and is preemptible (1).
|
|
+ */
|
|
+ __u8 admin_state;
|
|
+ /* The value of the holdAdvance parameter for the port in nanoseconds.
|
|
+ * There is no default value; the holdAdvance is a property of the
|
|
+ * underlying MAC." This parameter corresponds to the holdAdvance
|
|
+ * parameter in 802.1Qbu.
|
|
+ */
|
|
+ __u32 hold_advance;
|
|
+
|
|
+ /* The value of the releaseAdvance parameter for the port in
|
|
+ * nanoseconds. There is no default value; the releaseAdvance is a
|
|
+ * property of the underlying MAC." This parameter corresponds to the
|
|
+ * releaseAdvance parameter in 802.1Qbu.
|
|
+ */
|
|
+ __u32 release_advance;
|
|
+
|
|
+ /* The value is active (TRUE) when preemption is operationally active
|
|
+ * for the port, and idle (FALSE) otherwise. This parameter corresponds
|
|
+ * to the preemptionActive parameter in 802.1Qbu.
|
|
+ */
|
|
+ __u8 preemption_active;
|
|
+
|
|
+ /* The value is hold (1) when the sequence of gate operations for
|
|
+ * the port has executed a Set-And-Hold-MAC operation, and release
|
|
+ * (2) when the sequence of gate operations has executed a
|
|
+ * Set-And-Release-MAC operation. The value of this object is release
|
|
+ * (FALSE) on system initialization. This parameter corresponds to the
|
|
+ * holdRequest parameter in 802.1Qbu.
|
|
+ */
|
|
+ __u8 hold_request;
|
|
+};
|
|
+
|
|
+enum tsn_tx_mode {
|
|
+ TX_MODE_STRICT,
|
|
+ TX_MODE_CBS,
|
|
+ TX_MODE_ETS,
|
|
+ TX_MODE_VENDOR_DEFINE = 255,
|
|
+};
|
|
+
|
|
+#define QUEUE_TX_MASK ((1 << TX_MODE_STRICT) | (1 << TX_MODE_CBS) \
|
|
+ | (1 << TX_MODE_ETS) | (1 << TX_MODE_VENDOR_DEFINE))
|
|
+
|
|
+struct cbs_status {
|
|
+ __u8 delta_bw; /* percentage, 0~100 */
|
|
+ __u32 idleslope;
|
|
+ __s32 sendslope;
|
|
+ __u32 maxframesize;
|
|
+ __u32 hicredit;
|
|
+ __s32 locredit;
|
|
+ __u32 maxninference;
|
|
+};
|
|
+
|
|
+struct tx_queue {
|
|
+ /* tx_queue_capbility shows the queue's capability mask.
|
|
+ * refer the enum tsn_tx_mode
|
|
+ */
|
|
+ __u8 capability;
|
|
+
|
|
+ /* tx_queue_mode is current queue working mode */
|
|
+ __u8 mode;
|
|
+
|
|
+ /* prio is showing the queue priority */
|
|
+ __u8 prio;
|
|
+
|
|
+ /* mstat shows the status data of cbs or priority */
|
|
+ union {
|
|
+ struct cbs_status cbs;
|
|
+ };
|
|
+};
|
|
+
|
|
+struct port_status {
|
|
+ /* txqueue_cnt shows how many queues in this port */
|
|
+ __u8 queue_cnt;
|
|
+
|
|
+ /* max_rate(Mbit/s) is the port transmit rate current port is setting */
|
|
+ __u32 max_rate;
|
|
+
|
|
+ /* tsn_capability mask the tsn capability */
|
|
+ __u32 tsn_capability;
|
|
+};
|
|
+
|
|
+enum tsn_cb_streamid_type {
|
|
+ STREAMID_RESERVED = 0,
|
|
+ /* Null Stream identification */
|
|
+ STREAMID_NULL,
|
|
+ /* Source MAC and VLAN Stream identification */
|
|
+ STREAMID_SMAC_VLAN,
|
|
+ /* Active Destination MAC and VLAN stream identification */
|
|
+ STREAMID_DMAC_VLAN,
|
|
+ /* IP stream identification */
|
|
+ STREAMID_IP,
|
|
+};
|
|
+
|
|
+/* When instantiating an instance of the Null Stream identification function
|
|
+ * 8021CB(6.4) for a particular input Stream, the managed objects in the
|
|
+ * following subsections serve as the tsnStreamIdParameters managed object
|
|
+ * 8021CB claus(9.1.1.7).
|
|
+ */
|
|
+struct tsn_cb_null_streamid {
|
|
+ /* tsnCpeNullDownDestMac. Specifies the destination_address that
|
|
+ * identifies a packet in an Enhanced Internal Sublayer Service (EISS)
|
|
+ * indication primitive, to the Null Stream identification function.
|
|
+ */
|
|
+ __u64 dmac;
|
|
+
|
|
+ /* tsnCpeNullDownTagged. It can take the following values:
|
|
+ * 1 tagged: A frame must have a VLAN tag to be recognized as belonging
|
|
+ * to the Stream.
|
|
+ * 2 priority: A frame must be untagged, or have a VLAN tag with a VLAN
|
|
+ * ID = 0 to be recognized as belonging to the Stream.
|
|
+ * 3 all: A frame is recognized as belonging to the Stream whether
|
|
+ * tagged or not.
|
|
+ */
|
|
+ __u8 tagged;
|
|
+
|
|
+ /* tsnCpeNullDownVlan. Specifies the vlan_identifier parameter that
|
|
+ * identifies a packet in an EISS indication primitive to the Null
|
|
+ * Stream identification function. A value of 0 indicates that the vlan
|
|
+ * _identifier parameter is ignored on EISS indication primitives.
|
|
+ */
|
|
+ __u16 vid;
|
|
+};
|
|
+
|
|
+struct tsn_cb_source_streamid {
|
|
+ __u64 smac;
|
|
+ __u8 tagged;
|
|
+ __u16 vid;
|
|
+};
|
|
+
|
|
+struct tsn_cb_dest_streamid {
|
|
+ __u64 down_dmac;
|
|
+ __u8 down_tagged;
|
|
+ __u16 down_vid;
|
|
+ __u8 down_prio;
|
|
+ __u64 up_dmac;
|
|
+ __u8 up_tagged;
|
|
+ __u16 up_vid;
|
|
+ __u8 up_prio;
|
|
+};
|
|
+
|
|
+struct tsn_cb_ip_streamid {
|
|
+ __u64 dmac;
|
|
+ __u8 tagged;
|
|
+ __u16 vid;
|
|
+ __u64 siph;
|
|
+ __u64 sipl;
|
|
+ __u64 diph;
|
|
+ __u64 dipl;
|
|
+ __u8 dscp;
|
|
+ __u8 npt;
|
|
+ __u16 sport;
|
|
+ __u16 dport;
|
|
+};
|
|
+
|
|
+/* 802.1CB stream identify table clause 9.1 */
|
|
+struct tsn_cb_streamid {
|
|
+ /* The objects in a given entry of the Stream identity table are used
|
|
+ * to control packets whose stream_handle subparameter is equal to the
|
|
+ * entry tsnStreamIdHandle object.
|
|
+ */
|
|
+ __s32 handle;
|
|
+
|
|
+ /* The list of ports on which an in-facing Stream identification
|
|
+ * function in the output (towards the system forwarding function)
|
|
+ * direction Only Active Destination MAC and VLAN Stream identification
|
|
+ * (or nothing) can be configured.
|
|
+ */
|
|
+ __u32 ifac_oport;
|
|
+
|
|
+ /* The list of ports on which an out-facing Stream identification
|
|
+ * function in the output (towards the physical interface) direction.
|
|
+ * Only Active Destination MAC and VLAN Stream identification
|
|
+ * (or nothing) can be configured.
|
|
+ */
|
|
+ __u32 ofac_oport;
|
|
+
|
|
+ /* The list of ports on which an in-facing Stream identification
|
|
+ * function in the input (coming from the system forwarding function)
|
|
+ * direction
|
|
+ */
|
|
+ __u32 ifac_iport;
|
|
+
|
|
+ /* The list of ports on which an out-facing Stream identification
|
|
+ * function in the input (coming from the physical interface) direction
|
|
+ * .
|
|
+ */
|
|
+ __u32 ofac_iport;
|
|
+
|
|
+ /* An enumerated value indicating the method used to identify packets
|
|
+ * belonging to the Stream.
|
|
+ * The Organizationally Unique Identifier (OUI) or Company Identifier
|
|
+ * (CID) to identify the organization defining the enumerated type
|
|
+ * should be: 00-80-C2
|
|
+ * 1: null stream identification
|
|
+ * 2: source mac and vlan stream identification
|
|
+ * 3: activ destination mac and vlan stream identification
|
|
+ * 4: ip stream identifaciton
|
|
+ */
|
|
+ __u8 type;
|
|
+
|
|
+ /* tsnStreamIdParameters The number of controlling parameters for a
|
|
+ * Stream identification method, their types and values, are specific
|
|
+ * to the tsnStreamIdIdentificationType
|
|
+ */
|
|
+ union {
|
|
+ struct tsn_cb_null_streamid nid;
|
|
+ struct tsn_cb_source_streamid sid;
|
|
+ struct tsn_cb_dest_streamid did;
|
|
+ struct tsn_cb_ip_streamid iid;
|
|
+ } para;
|
|
+};
|
|
+
|
|
+/* Following counters are instantiated for each port on which the Stream
|
|
+ * identification function (6.2) is configured. The counters are indexed by
|
|
+ * port number, facing (in-facing or out-facing), and stream_handle value
|
|
+ * (tsnStreamIdHandle, 9.1.1.1).
|
|
+ */
|
|
+struct tsn_cb_streamid_counters {
|
|
+ struct {
|
|
+ __u64 input;
|
|
+ __u64 output;
|
|
+ } per_stream;
|
|
+
|
|
+ struct {
|
|
+ __u64 input;
|
|
+ __u64 output;
|
|
+ } per_streamport[32];
|
|
+};
|
|
+
|
|
+/* 802.1Qci Stream Parameter Table, read from port */
|
|
+struct tsn_qci_psfp_stream_param {
|
|
+ /* MaxStreamFilterInstances.
|
|
+ * The maximum number of Stream Filter instances supported by this
|
|
+ * Bridge component.
|
|
+ */
|
|
+ __s32 max_sf_instance;
|
|
+
|
|
+ /* MaxStreamGateInstances
|
|
+ * The maximum number of Stream Gate instances supported by this Bridge
|
|
+ * component.
|
|
+ */
|
|
+ __s32 max_sg_instance;
|
|
+
|
|
+ /* MaxFlowMeterInstances
|
|
+ * The maximum number of Flow Meter instances supported by this Bridge
|
|
+ * component.
|
|
+ */
|
|
+ __s32 max_fm_instance;
|
|
+
|
|
+ /* SupportedListMax
|
|
+ * The maximum value supported by this Bridge component of the
|
|
+ * AdminControlListLength and OperControlListLength parameters.
|
|
+ */
|
|
+ __s32 supported_list_max;
|
|
+};
|
|
+
|
|
+/* 802.1Qci Stream Filter Instance Table, counters part only. */
|
|
+struct tsn_qci_psfp_sfi_counters {
|
|
+ /* The MatchingFramesCount counter counts received frames that match
|
|
+ * this stream filter.
|
|
+ */
|
|
+ __u64 matching_frames_count;
|
|
+
|
|
+ /* The PassingFramesCount counter counts received frames that pass the
|
|
+ * gate associated with this stream filter.
|
|
+ */
|
|
+ __u64 passing_frames_count;
|
|
+
|
|
+ /* The NotPassingFramesCount counter counts received frames that do not
|
|
+ * pass the gate associated with this stream filter.
|
|
+ */
|
|
+ __u64 not_passing_frames_count;
|
|
+
|
|
+ /* The PassingSDUCount counter counts received frames that pass the SDU
|
|
+ * size filter specification associated with this stream filter.
|
|
+ */
|
|
+ __u64 passing_sdu_count;
|
|
+
|
|
+ /* The NotPassingSDUCount counter counts received frames that do not
|
|
+ * pass the SDU size filter specification associated with this stream
|
|
+ * filter.
|
|
+ */
|
|
+ __u64 not_passing_sdu_count;
|
|
+
|
|
+ /* The REDFramesCount counter counts received random early detection
|
|
+ * (RED) frames associated with this stream filter.
|
|
+ */
|
|
+ __u64 red_frames_count;
|
|
+};
|
|
+
|
|
+/* 802.1Qci Stream Filter Instance Table, configuration part only. */
|
|
+struct tsn_qci_psfp_sfi_conf {
|
|
+
|
|
+ /* The StreamHandleSpec parameter contains a stream identifier
|
|
+ * specification value. A value of -1 denotes the wild card value; zero
|
|
+ * or positive values denote stream identifier values.
|
|
+ */
|
|
+ __s32 stream_handle_spec;
|
|
+
|
|
+ /* The PrioritySpec parameter contains a priority specification value.
|
|
+ * A value of -1 denotes the wild card value; zero or positive values
|
|
+ * denote priority values.
|
|
+ */
|
|
+ __s8 priority_spec;
|
|
+
|
|
+ /* The StreamGateInstanceID parameter contains the index of an entry in
|
|
+ * the Stream Gate Table.
|
|
+ */
|
|
+ __u32 stream_gate_instance_id;
|
|
+
|
|
+ /* The filter specifications. The actions specified in a filter
|
|
+ * specification can result in a frame passing or failing the specified
|
|
+ * filter. Frames that fail a filter are discarded.
|
|
+ */
|
|
+ struct {
|
|
+ /* The MaximumSDUSize parameter specifies the maximum allowed
|
|
+ * frame size for the stream. Any frame exceeding this value
|
|
+ * will be dropped. A value of 0 denote that the MaximumSDUSize
|
|
+ * filter is disabled for this stream.
|
|
+ */
|
|
+ __u16 maximum_sdu_size;
|
|
+
|
|
+ /* The FlowMeterInstanceID parameter contains the index of an
|
|
+ * entry in the Flow Meter Table. A value of -1 denotes that
|
|
+ * no flow meter is assigned; zero or positive values denote
|
|
+ * flow meter IDs.
|
|
+ */
|
|
+ __s32 flow_meter_instance_id;
|
|
+ } stream_filter;
|
|
+
|
|
+ /* The StreamBlockedDueToOversizeFrameEnable object contains a Boolean
|
|
+ * value that indicates whether the StreamBlockedDueToOversizeFrame
|
|
+ * function is enabled (TRUE) or disabled (FALSE).
|
|
+ */
|
|
+ __u8 block_oversize_enable;
|
|
+
|
|
+ /* The StreamBlockedDueToOversizeFrame object contains a Boolean value
|
|
+ * that indicates whether, if the StreamBlockedDueToOversizeFrame
|
|
+ * function is enabled, all frames are to be discarded (TRUE) or not
|
|
+ * (FALSE).
|
|
+ */
|
|
+ __u8 block_oversize;
|
|
+};
|
|
+
|
|
+/* 802.1Qci Stream Gate Control List Entry. */
|
|
+struct tsn_qci_psfp_gcl {
|
|
+ /* The GateState parameter specifies a desired state, open (true) or
|
|
+ * closed (false), for the stream gate.
|
|
+ */
|
|
+ __u8 gate_state;
|
|
+
|
|
+ /* An IPV is encoded as a signed integer. A negative denotes the null
|
|
+ * value; zero or positive values denote internal priority values.
|
|
+ */
|
|
+ __s8 ipv;
|
|
+
|
|
+ /* A TimeInterval is encoded in 4 octets as a 32-bit unsigned integer,
|
|
+ * representing a number of nanoseconds.
|
|
+ */
|
|
+ __u32 time_interval;
|
|
+
|
|
+ /* The maximum number of octets that are permitted to pass the gate
|
|
+ * during the specified TimeInterval. If zero, there is no maximum.
|
|
+ */
|
|
+ __u32 octet_max;
|
|
+
|
|
+};
|
|
+
|
|
+/* 802.1Qci Stream Gate Admin/Operation common list control parameters */
|
|
+struct tsn_qci_sg_control {
|
|
+ /* The administrative/operation value of the GateStates parameter
|
|
+ * for the stream gate. A value of false indicates closed;
|
|
+ * a value of true indicates open.
|
|
+ */
|
|
+ __u8 gate_states;
|
|
+
|
|
+ /* The administrative/operation value of the ListMax parameter for the
|
|
+ * gate. The integer value indicates the number of entries (TLVs) in
|
|
+ * the AdminControlList/OperControlList.
|
|
+ */
|
|
+ __u8 control_list_length;
|
|
+
|
|
+ /* The administrative/operation value of the CycleTime parameter for
|
|
+ * the gate. The value is an unsigned integer number of nanoseconds.
|
|
+ */
|
|
+ __u32 cycle_time;
|
|
+
|
|
+ /* The administrative/operation value of the CycleTimeExtension
|
|
+ * parameter for the gate. The value is an unsigned integer number
|
|
+ * of nanoseconds.
|
|
+ */
|
|
+ __u32 cycle_time_extension;
|
|
+
|
|
+ /* The administrative/operation value of the BaseTime parameter for the
|
|
+ * gate. The value is a representation of a PTPtime value, consisting
|
|
+ * of a 48-bit integer number of seconds and a 32-bit integer number of
|
|
+ * nanoseconds.
|
|
+ */
|
|
+ ptptime_t base_time;
|
|
+
|
|
+ /* The administrative/operation value of the IPV parameter for the gate.
|
|
+ * A value of -1 denotes the null value; zero or positive values denote
|
|
+ * internal priority values.
|
|
+ */
|
|
+ __s8 init_ipv;
|
|
+
|
|
+ /* control_list contend the gate control list of
|
|
+ * administrative/operation
|
|
+ */
|
|
+ struct tsn_qci_psfp_gcl *gcl;
|
|
+};
|
|
+
|
|
+/* 802.1Qci Stream Gate Instance Table, configuration part only. */
|
|
+struct tsn_qci_psfp_sgi_conf {
|
|
+ /* The GateEnabled parameter determines whether the stream gate is
|
|
+ * active (true) or inactive (false).
|
|
+ */
|
|
+ __u8 gate_enabled;
|
|
+
|
|
+ /* The ConfigChange parameter signals the start of a configuration
|
|
+ * change when it is set to TRUE. This should only be done when the
|
|
+ * various administrative parameters are all set to appropriate values.
|
|
+ */
|
|
+ __u8 config_change;
|
|
+
|
|
+ /* admin control parameters with admin control list */
|
|
+ struct tsn_qci_sg_control admin;
|
|
+
|
|
+ /* The GateClosedDueToInvalidRxEnable object contains a Boolean value
|
|
+ * that indicates whether the GateClosedDueToInvalidRx function is
|
|
+ * enabled (TRUE) or disabled (FALSE).
|
|
+ */
|
|
+ __u8 block_invalid_rx_enable;
|
|
+
|
|
+ /* The GateClosedDueToInvalidRx object contains a Boolean value that
|
|
+ * indicates whether, if the GateClosedDueToInvalidRx function is
|
|
+ * enabled, all frames are to be discarded (TRUE) or not (FALSE).
|
|
+ */
|
|
+ __u8 block_invalid_rx;
|
|
+
|
|
+ /* The GateClosedDueToOctetsExceededEnable object contains a Boolean
|
|
+ * value that indicates whether the GateClosedDueToOctetsExceeded
|
|
+ * function is enabled (TRUE) or disabled (FALSE).
|
|
+ */
|
|
+ __u8 block_octets_exceeded_enable;
|
|
+
|
|
+ /* The GateClosedDueToOctetsExceeded object contains a Boolean value
|
|
+ * that indicates whether, if the GateClosedDueToOctetsExceeded
|
|
+ * function is enabled, all frames are to be discarded (TRUE) or not
|
|
+ * (FALSE).
|
|
+ */
|
|
+ __u8 block_octets_exceeded;
|
|
+};
|
|
+
|
|
+/* 802.1Qci Stream Gate Instance Table, status part only. */
|
|
+struct tsn_psfp_sgi_status {
|
|
+
|
|
+ /* admin control parameters with admin control list */
|
|
+ struct tsn_qci_sg_control oper;
|
|
+
|
|
+ /* The PTPtime at which the next config change is scheduled to occur.
|
|
+ * The value is a representation of a PTPtime value, consisting of a
|
|
+ * 48-bit integer number of seconds and a 32-bit integer number of
|
|
+ * nanoseconds.
|
|
+ */
|
|
+ ptptime_t config_change_time;
|
|
+
|
|
+ /* The granularity of the cycle time clock, represented as an unsigned
|
|
+ * number of tenths of nanoseconds.
|
|
+ */
|
|
+ __u32 tick_granularity;
|
|
+
|
|
+ /* The current time, in PTPtime, as maintained by the local system.
|
|
+ * The value is a representation of a PTPtime value, consisting of a
|
|
+ * 48-bit integer number of seconds and a 32-bit integer number of
|
|
+ * nanoseconds.
|
|
+ */
|
|
+ ptptime_t current_time;
|
|
+
|
|
+ /* The value of the ConfigPending state machine variable. The value is
|
|
+ * TRUE if a configuration change is in progress but has not yet
|
|
+ * completed.
|
|
+ */
|
|
+ __u8 config_pending;
|
|
+
|
|
+ /* A counter of the number of times that a re-configuration of the
|
|
+ * traffic schedule has been requested with the old schedule still
|
|
+ * running and the requested base time was in the past.
|
|
+ */
|
|
+ __u64 config_change_error;
|
|
+
|
|
+};
|
|
+
|
|
+/* 802.1Qci Flow Meter Instance Table. */
|
|
+struct tsn_qci_psfp_fmi {
|
|
+ /* The FlowMeterCIR parameter contains an integer value that represents
|
|
+ * the CIR value for the flow meter, in kbit/s.
|
|
+ */
|
|
+ __u32 cir;
|
|
+
|
|
+ /* The FlowMeterCBS parameter contains an integer value that represents
|
|
+ * the CBS value for the flow meter, in octets.
|
|
+ */
|
|
+ __u32 cbs;
|
|
+
|
|
+ /* The FlowMeterEIR parameter contains an integer value that represents
|
|
+ * the EIR value for the flow meter, in kbit/s.
|
|
+ */
|
|
+ __u32 eir;
|
|
+
|
|
+ /* The FlowMeterEBS parameter contains an integer value that represents
|
|
+ * the EBS value for the flow meter, in octets.
|
|
+ */
|
|
+ __u32 ebs;
|
|
+
|
|
+ /* The FlowMeterCF parameter contains a Boolean value that represents
|
|
+ * the CF value for the flow meter, as a Boolean value indicating no
|
|
+ * coupling (FALSE) or coupling (TRUE).
|
|
+ */
|
|
+ __u8 cf;
|
|
+
|
|
+ /* The FlowMeterCM parameter contains a Boolean value that represents
|
|
+ * the CM value for the flow meter, as a Boolean value indicating
|
|
+ * colorBlind (FALSE) or colorAware (TRUE).
|
|
+ */
|
|
+ __u8 cm;
|
|
+
|
|
+ /* The FlowMeterDropOnYellow parameter contains a Boolean value that
|
|
+ * indicates whether yellow frames are dropped (TRUE) or have
|
|
+ * drop_eligible set to TRUE (FALSE).
|
|
+ */
|
|
+ __u8 drop_on_yellow;
|
|
+
|
|
+ /* The FlowMeterMarkAllFramesRedEnable parameter contains a Boolean
|
|
+ * value that indicates whether the MarkAllFramesRed function
|
|
+ * is enabled (TRUE) or disabled (FALSE).
|
|
+ */
|
|
+ __u8 mark_red_enable;
|
|
+
|
|
+ /* The FlowMeterMarkAllFramesRed parameter contains a Boolean value
|
|
+ * that indicates whether, if the MarkAllFramesRed function is enabled,
|
|
+ * all frames are to be discarded (TRUE) or not (FALSE).
|
|
+ */
|
|
+ __u8 mark_red;
|
|
+};
|
|
+
|
|
+struct tsn_qci_psfp_fmi_counters {
|
|
+ __u64 bytecount;
|
|
+ __u64 drop;
|
|
+ __u64 dr0_green;
|
|
+ __u64 dr1_green;
|
|
+ __u64 dr2_yellow;
|
|
+ __u64 remark_yellow;
|
|
+ __u64 dr3_red;
|
|
+ __u64 remark_red;
|
|
+};
|
|
+
|
|
+/* 802.1cb */
|
|
+struct tsn_seq_gen_conf {
|
|
+
|
|
+ /* The InputPortMask parameter contains a port mask.
|
|
+ * If the packet is from input port belonging to this
|
|
+ * port mask then it's on known stream and sequence
|
|
+ * generation parameters can be applied.
|
|
+ */
|
|
+ __u8 iport_mask;
|
|
+
|
|
+ /* The SplitMask parameter contains a output port mask
|
|
+ * used to add redundant paths.
|
|
+ */
|
|
+ __u8 split_mask;
|
|
+
|
|
+ /* The SequenceSpaceLenLog parameter is a value to specifies
|
|
+ * number of bits to be used for sequence number.
|
|
+ */
|
|
+ __u8 seq_len;
|
|
+
|
|
+ /* The SequenceNumber parameter is a value to used for
|
|
+ * outgoing packet's sequence number generation.
|
|
+ */
|
|
+ __u32 seq_num;
|
|
+};
|
|
+
|
|
+struct tsn_seq_rec_conf {
|
|
+
|
|
+ /* The SequenceSpaceLenLog parameter is a value to specifies
|
|
+ * number of bits to be used for sequence number.
|
|
+ */
|
|
+ __u8 seq_len;
|
|
+
|
|
+ /* The HistorySpaceLenLog parameter is a value to specifies
|
|
+ * number of bits to be used for history register.
|
|
+ */
|
|
+ __u8 his_len;
|
|
+
|
|
+ /* The RTagPopEnable parameter contains a __u8 to enable removal
|
|
+ * of redundancy tag from the packet.
|
|
+ */
|
|
+ __u8 rtag_pop_en;
|
|
+};
|
|
+
|
|
+struct tsn_cb_status {
|
|
+
|
|
+ /* The GenRecover parameter contains a value specifies type
|
|
+ * of stream sequence parameters:
|
|
+ * 0: Stream sequence parameters are for generation.
|
|
+ * 1: Stream sequence parameters are for recovery.
|
|
+ */
|
|
+ __u8 gen_rec;
|
|
+
|
|
+ /* The ErrStatus parameter indicates stream's error status
|
|
+ * 1: This switch is expected to sequence the stream,
|
|
+ * but the incoming packet has sequence number.
|
|
+ * 2: This switch is expected to recover the stream,
|
|
+ * but the incoming packet is NONSEQ.
|
|
+ */
|
|
+ __u8 err;
|
|
+
|
|
+ /* The SequenceNumber parameter is a value to used for
|
|
+ * outgoing packet's sequence number generation.
|
|
+ */
|
|
+ __u32 seq_num;
|
|
+
|
|
+ /* The SequenceSpaceLenLog parameter is a value to specifies
|
|
+ * number of bits to be used for sequence number.
|
|
+ */
|
|
+ __u8 seq_len;
|
|
+
|
|
+ /* The SplitMask parameter contains a output port mask
|
|
+ * used to add redundant paths.
|
|
+ */
|
|
+ __u8 split_mask;
|
|
+
|
|
+ /* The InputPortMask parameter contains a port mask.
|
|
+ * If the packet is from input port belonging to this
|
|
+ * port mask then it's on known stream and sequence
|
|
+ * generation parameters can be applied.
|
|
+ */
|
|
+ __u8 iport_mask;
|
|
+
|
|
+ /* The HistorySpaceLenLog parameter is a value to specifies
|
|
+ * number of bits to be used for history register.
|
|
+ */
|
|
+ __u8 his_len;
|
|
+
|
|
+ /* The SequenceHistory parameter Maintains history of sequence
|
|
+ * numbers of received packets.
|
|
+ */
|
|
+ __u32 seq_his;
|
|
+};
|
|
+
|
|
+/* An entry for gate control list */
|
|
+struct tsn_qbv_entry {
|
|
+ /* Octet represent the gate states for the corresponding traffic
|
|
+ * classes.
|
|
+ * The MS bit corresponds to traffic class 7.
|
|
+ * The LS bit to traffic class 0.
|
|
+ * A bit value of 0 indicates closed;
|
|
+ * A bit value of 1 indicates open.
|
|
+ */
|
|
+ __u8 gate_state;
|
|
+
|
|
+ /* A TimeInterval is encoded in 4 octets as a 32-bit unsigned integer,
|
|
+ * representing a number of nanoseconds.
|
|
+ */
|
|
+ __u32 time_interval;
|
|
+};
|
|
+
|
|
+/* The administrative/operation time and gate list */
|
|
+struct tsn_qbv_basic {
|
|
+ /* The administrative/operation value of the GateStates parameter for
|
|
+ * the Port.
|
|
+ * The bits of the octet represent the gate states for the
|
|
+ * corresponding traffic classes; the MS bit corresponds to traffic
|
|
+ * class 7, the LS bit to traffic class 0. A bit value of 0 indicates
|
|
+ * closed; a bit value of 1 indicates open.
|
|
+ * The value of this object MUST be retained
|
|
+ * across reinitializations of the management system.
|
|
+ */
|
|
+ __u8 gate_states;
|
|
+
|
|
+ /* The administrative/operation value of the ListMax parameter for the
|
|
+ * port. The integer value indicates the number of entries (TLVs) in
|
|
+ * the AdminControlList. The value of this object MUST be retained
|
|
+ * across reinitializations of the management system.
|
|
+ */
|
|
+ __u32 control_list_length;
|
|
+
|
|
+ /* The administrative/operation value of the AdminCycleTime
|
|
+ * parameter for the Port. The numerator and denominator together
|
|
+ * represent the cycle time as a rational number of seconds. The value
|
|
+ * of this object MUST be retained across reinitializations of the
|
|
+ * management system.
|
|
+ */
|
|
+ __u32 cycle_time;
|
|
+
|
|
+ /* The administrative/operation value of the CycleTimeExtension
|
|
+ * parameter for the Port. The value is an unsigned integer number of
|
|
+ * nanoseconds.
|
|
+ * The value of this object MUST be retained across reinitializations
|
|
+ * of the management system.
|
|
+ */
|
|
+
|
|
+ __u32 cycle_time_extension;
|
|
+
|
|
+ /* The administrative/operation value of the BaseTime parameter for the
|
|
+ * Port. The value is a representation of a PTPtime value, consisting
|
|
+ * of a 48-bit integer number of seconds and a 32-bit integer number of
|
|
+ * nanoseconds.
|
|
+ * The value of this object MUST be retained across reinitializations of
|
|
+ * the management system.
|
|
+ */
|
|
+ ptptime_t base_time;
|
|
+
|
|
+ /* admin_control_list represent the AdminControlList/OperControlList.
|
|
+ * The administrative version of the gate control list for the Port.
|
|
+ */
|
|
+ struct tsn_qbv_entry *control_list;
|
|
+};
|
|
+
|
|
+struct tsn_qbv_conf {
|
|
+ /* The GateEnabled parameter determines whether traffic scheduling is
|
|
+ * active (true) or inactive (false). The value of this object MUST be
|
|
+ * retained across reinitializations of the management system.
|
|
+ */
|
|
+ __u8 gate_enabled;
|
|
+
|
|
+ /* The maxsdu parameter denoting the maximum SDU size supported by the
|
|
+ * queue.
|
|
+ */
|
|
+ __u32 maxsdu;
|
|
+
|
|
+ /* The ConfigChange parameter signals the start of a configuration
|
|
+ * change when it is set to TRUE. This should only be done when the
|
|
+ * various administrative parameters are all set to appropriate values.
|
|
+ */
|
|
+ __u8 config_change;
|
|
+
|
|
+ /* The admin parameter signals the admin relate cycletime, basictime,
|
|
+ * gatelist paraters.
|
|
+ */
|
|
+ struct tsn_qbv_basic admin;
|
|
+};
|
|
+
|
|
+/* 802.1Qbv (Time Aware Shaper) port status */
|
|
+struct tsn_qbv_status {
|
|
+ /* The PTPtime at which the next config change is scheduled to occur.
|
|
+ * The value is a representation of a PTPtime value, consisting of a
|
|
+ * 48-bit integer number of seconds and a 32-bit integer number of
|
|
+ * nanoseconds. The value of this object MUST be retained across
|
|
+ * reinitializations of the management system.
|
|
+ */
|
|
+ ptptime_t config_change_time;
|
|
+
|
|
+ /* The granularity of the cycle time clock, represented as an unsigned
|
|
+ * number of tenths of nanoseconds. The value of this object MUST be
|
|
+ * retained across reinitializations of the management system.
|
|
+ */
|
|
+ __u32 tick_granularity;
|
|
+
|
|
+ /* The current time, in PTPtime, as maintained by the local system.
|
|
+ * The value is a representation of a PTPtime value, consisting of a
|
|
+ * 48-bit integer number of seconds and a 32-bit integer number of
|
|
+ * nanoseconds.
|
|
+ */
|
|
+ ptptime_t current_time;
|
|
+
|
|
+ /* The value of the ConfigPending state machine variable. The value is
|
|
+ * TRUE if a configuration change is in progress but has not yet
|
|
+ * completed.
|
|
+ */
|
|
+ __u8 config_pending;
|
|
+
|
|
+ /* A counter of the number of times that a re-configuration of the
|
|
+ * traffic schedule has been requested with the old schedule still
|
|
+ * running and the requested base time was in the past.
|
|
+ */
|
|
+ __u64 config_change_error;
|
|
+
|
|
+ /* The maximum value supported by this Port of the
|
|
+ * AdminControlListLength and OperControlListLength parameters.
|
|
+ */
|
|
+ __u32 supported_list_max;
|
|
+
|
|
+ /* Operation settings parameters and Oper gate list */
|
|
+ struct tsn_qbv_basic oper;
|
|
+};
|
|
+
|
|
+/* Time Specific Departure parameters */
|
|
+struct tsn_tsd {
|
|
+ __u8 enable;
|
|
+
|
|
+ /* The cycle time, in units of microsecond(us)*/
|
|
+ __u32 period;
|
|
+
|
|
+ /* The maximum number of frames which could be transmitted on one cycle
|
|
+ * The exceeding frames will be transmitted on next cycle.
|
|
+ */
|
|
+ __u32 maxFrameNum;
|
|
+
|
|
+ /* Specify the time of the first cycle begins.
|
|
+ * 1: begin when the queue get the first frame to transmit.
|
|
+ * 2: begin immediately at the end of setting function.
|
|
+ */
|
|
+ __u32 syn_flag;
|
|
+};
|
|
+
|
|
+struct tsn_tsd_status {
|
|
+ __u8 enable;
|
|
+ __u32 period;
|
|
+ __u32 maxFrameNum;
|
|
+ __u32 flag;
|
|
+ __u32 cycleNum;
|
|
+ __u32 loss_steps;
|
|
+};
|
|
+
|
|
+struct tsn_qos_switch_dscp_conf {
|
|
+ __u8 trust;
|
|
+ __u8 cos;
|
|
+ __u8 dpl;
|
|
+ __u8 remark;
|
|
+ __u8 dscp; /* New ingress translated DSCP value */
|
|
+};
|
|
+
|
|
+#endif /* _UAPI_GENL_TSN_H */
|
|
--- a/net/Kconfig
|
|
+++ b/net/Kconfig
|
|
@@ -240,6 +240,7 @@ source "net/ieee802154/Kconfig"
|
|
source "net/mac802154/Kconfig"
|
|
source "net/sched/Kconfig"
|
|
source "net/dcb/Kconfig"
|
|
+source "net/tsn/Kconfig"
|
|
source "net/dns_resolver/Kconfig"
|
|
source "net/batman-adv/Kconfig"
|
|
source "net/openvswitch/Kconfig"
|
|
--- a/net/Makefile
|
|
+++ b/net/Makefile
|
|
@@ -59,6 +59,9 @@ obj-$(CONFIG_CAIF) += caif/
|
|
ifneq ($(CONFIG_DCB),)
|
|
obj-y += dcb/
|
|
endif
|
|
+ifneq ($(CONFIG_TSN),)
|
|
+obj-y += tsn/
|
|
+endif
|
|
obj-$(CONFIG_6LOWPAN) += 6lowpan/
|
|
obj-$(CONFIG_IEEE802154) += ieee802154/
|
|
obj-$(CONFIG_MAC802154) += mac802154/
|
|
--- /dev/null
|
|
+++ b/net/tsn/Kconfig
|
|
@@ -0,0 +1,15 @@
|
|
+config TSN
|
|
+ bool "802.1 Time-Sensitive Networking support"
|
|
+ default n
|
|
+ depends on VLAN_8021Q && PTP_1588_CLOCK
|
|
+ help
|
|
+ This enables support for TSN(time sensitive networking)
|
|
+ TSN features include:
|
|
+ 802.1Qav:
|
|
+ 802.1Qbv:
|
|
+ 802.1Qci:
|
|
+ 802.1Qbu:
|
|
+ 802.1AS:
|
|
+ 802.1CB:
|
|
+
|
|
+ If unsure, say N.
|
|
--- /dev/null
|
|
+++ b/net/tsn/Makefile
|
|
@@ -0,0 +1 @@
|
|
+obj-$(CONFIG_TSN) += genl_tsn.o
|
|
--- /dev/null
|
|
+++ b/net/tsn/genl_tsn.c
|
|
@@ -0,0 +1,3696 @@
|
|
+// SPDX-License-Identifier: (GPL-2.0+ OR BSD-3-Clause)
|
|
+/* Copyright 2017-2019 NXP */
|
|
+
|
|
+#include <linux/module.h>
|
|
+#include <linux/kernel.h>
|
|
+#include <linux/init.h>
|
|
+#include <linux/netdevice.h>
|
|
+#include <linux/if_vlan.h>
|
|
+#include <net/genetlink.h>
|
|
+#include <net/netlink.h>
|
|
+#include <linux/version.h>
|
|
+#include <net/tsn.h>
|
|
+
|
|
+#define NLA_PARSE_NESTED(a, b, c, d) \
|
|
+ nla_parse_nested_deprecated(a, b, c, d, NULL)
|
|
+#define NLA_PUT_U64(a, b, c) nla_put_u64_64bit(a, b, c, NLA_U64)
|
|
+
|
|
+static struct genl_family tsn_family;
|
|
+
|
|
+LIST_HEAD(port_list);
|
|
+
|
|
+static const struct nla_policy tsn_cmd_policy[TSN_CMD_ATTR_MAX + 1] = {
|
|
+ [TSN_CMD_ATTR_MESG] = { .type = NLA_STRING },
|
|
+ [TSN_CMD_ATTR_DATA] = { .type = NLA_S32 },
|
|
+ [TSN_ATTR_IFNAME] = { .type = NLA_STRING },
|
|
+ [TSN_ATTR_PORT_NUMBER] = { .type = NLA_U8 },
|
|
+ [TSN_ATTR_CAP] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_QBV] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_STREAM_IDENTIFY] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_QCI_SP] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_QCI_SFI] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_QCI_SGI] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_QCI_FMI] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_CBS] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_TSD] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_QBU] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_CT] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_CBGEN] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_CBREC] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_CBSTAT] = { .type = NLA_NESTED },
|
|
+ [TSN_ATTR_DSCP] = { .type = NLA_NESTED },
|
|
+};
|
|
+
|
|
+static const struct nla_policy tsn_cap_policy[TSN_CAP_ATTR_MAX + 1] = {
|
|
+ [TSN_CAP_ATTR_QBV] = { .type = NLA_FLAG },
|
|
+ [TSN_CAP_ATTR_QCI] = { .type = NLA_FLAG },
|
|
+ [TSN_CAP_ATTR_QBU] = { .type = NLA_FLAG },
|
|
+ [TSN_CAP_ATTR_CBS] = { .type = NLA_FLAG },
|
|
+ [TSN_CAP_ATTR_CB] = { .type = NLA_FLAG },
|
|
+ [TSN_CAP_ATTR_TBS] = { .type = NLA_FLAG },
|
|
+ [TSN_CAP_ATTR_CTH] = { .type = NLA_FLAG },
|
|
+};
|
|
+
|
|
+static const struct nla_policy qci_cap_policy[TSN_QCI_STREAM_ATTR_MAX + 1] = {
|
|
+ [TSN_QCI_STREAM_ATTR_MAX_SFI] = { .type = NLA_U32 },
|
|
+ [TSN_QCI_STREAM_ATTR_MAX_SGI] = { .type = NLA_U32 },
|
|
+ [TSN_QCI_STREAM_ATTR_MAX_FMI] = { .type = NLA_U32 },
|
|
+ [TSN_QCI_STREAM_ATTR_SLM] = { .type = NLA_U32 },
|
|
+};
|
|
+
|
|
+static const struct nla_policy ct_policy[TSN_CT_ATTR_MAX + 1] = {
|
|
+ [TSN_CT_ATTR_QUEUE_STATE] = { .type = NLA_U8 }
|
|
+};
|
|
+
|
|
+static const struct nla_policy cbgen_policy[TSN_CBGEN_ATTR_MAX + 1] = {
|
|
+ [TSN_CBGEN_ATTR_INDEX] = { .type = NLA_U32 },
|
|
+ [TSN_CBGEN_ATTR_PORT_MASK] = { .type = NLA_U8 },
|
|
+ [TSN_CBGEN_ATTR_SPLIT_MASK] = { .type = NLA_U8 },
|
|
+ [TSN_CBGEN_ATTR_SEQ_LEN] = { .type = NLA_U8 },
|
|
+ [TSN_CBGEN_ATTR_SEQ_NUM] = { .type = NLA_U32 },
|
|
+};
|
|
+
|
|
+static const struct nla_policy cbrec_policy[TSN_CBREC_ATTR_MAX + 1] = {
|
|
+ [TSN_CBREC_ATTR_INDEX] = { .type = NLA_U32 },
|
|
+ [TSN_CBREC_ATTR_SEQ_LEN] = { .type = NLA_U8 },
|
|
+ [TSN_CBREC_ATTR_HIS_LEN] = { .type = NLA_U8 },
|
|
+ [TSN_CBREC_ATTR_TAG_POP_EN] = { .type = NLA_FLAG },
|
|
+};
|
|
+
|
|
+static const struct nla_policy cbstat_policy[TSN_CBSTAT_ATTR_MAX + 1] = {
|
|
+ [TSN_CBSTAT_ATTR_INDEX] = { .type = NLA_U32 },
|
|
+ [TSN_CBSTAT_ATTR_GEN_REC] = { .type = NLA_U8 },
|
|
+ [TSN_CBSTAT_ATTR_ERR] = { .type = NLA_U8 },
|
|
+ [TSN_CBSTAT_ATTR_SEQ_NUM] = { .type = NLA_U32 },
|
|
+ [TSN_CBSTAT_ATTR_SEQ_LEN] = { .type = NLA_U8 },
|
|
+ [TSN_CBSTAT_ATTR_SPLIT_MASK] = { .type = NLA_U8 },
|
|
+ [TSN_CBSTAT_ATTR_PORT_MASK] = { .type = NLA_U8 },
|
|
+ [TSN_CBSTAT_ATTR_HIS_LEN] = { .type = NLA_U8 },
|
|
+ [TSN_CBSTAT_ATTR_SEQ_HIS] = { .type = NLA_U32 },
|
|
+};
|
|
+
|
|
+static const struct nla_policy qbu_policy[TSN_QBU_ATTR_MAX + 1] = {
|
|
+ [TSN_QBU_ATTR_ADMIN_STATE] = { .type = NLA_U8 },
|
|
+ [TSN_QBU_ATTR_HOLD_ADVANCE] = { .type = NLA_U32},
|
|
+ [TSN_QBU_ATTR_RELEASE_ADVANCE] = { .type = NLA_U32},
|
|
+ [TSN_QBU_ATTR_ACTIVE] = { .type = NLA_FLAG},
|
|
+ [TSN_QBU_ATTR_HOLD_REQUEST] = { .type = NLA_U8},
|
|
+};
|
|
+
|
|
+static const struct nla_policy cbs_policy[TSN_CBS_ATTR_MAX + 1] = {
|
|
+ [TSN_CBS_ATTR_TC_INDEX] = { .type = NLA_U8},
|
|
+ [TSN_CBS_ATTR_BW] = { .type = NLA_U8},
|
|
+};
|
|
+
|
|
+static const struct nla_policy tsd_policy[TSN_TSD_ATTR_MAX + 1] = {
|
|
+ [TSN_TSD_ATTR_ENABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_TSD_ATTR_DISABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_TSD_ATTR_PERIOD] = { .type = NLA_U32},
|
|
+ [TSN_TSD_ATTR_MAX_FRM_NUM] = { .type = NLA_U32},
|
|
+ [TSN_TSD_ATTR_CYCLE_NUM] = { .type = NLA_U32},
|
|
+ [TSN_TSD_ATTR_LOSS_STEPS] = { .type = NLA_U32},
|
|
+ [TSN_TSD_ATTR_SYN_IMME] = { .type = NLA_FLAG},
|
|
+};
|
|
+
|
|
+static const struct nla_policy qbv_policy[TSN_QBV_ATTR_MAX + 1] = {
|
|
+ [TSN_QBV_ATTR_ADMINENTRY] = { .type = NLA_NESTED},
|
|
+ [TSN_QBV_ATTR_OPERENTRY] = { .type = NLA_NESTED},
|
|
+ [TSN_QBV_ATTR_ENABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_QBV_ATTR_DISABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_QBV_ATTR_CONFIGCHANGE] = { .type = NLA_FLAG},
|
|
+ [TSN_QBV_ATTR_CONFIGCHANGETIME] = { .type = NLA_U64},
|
|
+ [TSN_QBV_ATTR_MAXSDU] = { .type = NLA_U32},
|
|
+ [TSN_QBV_ATTR_GRANULARITY] = { .type = NLA_U32},
|
|
+ [TSN_QBV_ATTR_CURRENTTIME] = { .type = NLA_U64},
|
|
+ [TSN_QBV_ATTR_CONFIGPENDING] = {.type = NLA_FLAG},
|
|
+ [TSN_QBV_ATTR_CONFIGCHANGEERROR] = { .type = NLA_U64},
|
|
+ [TSN_QBV_ATTR_LISTMAX] = { .type = NLA_U32},
|
|
+};
|
|
+
|
|
+static const struct nla_policy qbv_ctrl_policy[TSN_QBV_ATTR_CTRL_MAX + 1] = {
|
|
+ [TSN_QBV_ATTR_CTRL_LISTCOUNT] = { .type = NLA_U32},
|
|
+ [TSN_QBV_ATTR_CTRL_GATESTATE] = { .type = NLA_U8},
|
|
+ [TSN_QBV_ATTR_CTRL_CYCLETIME] = { .type = NLA_U32},
|
|
+ [TSN_QBV_ATTR_CTRL_CYCLETIMEEXT] = { .type = NLA_U32},
|
|
+ [TSN_QBV_ATTR_CTRL_BASETIME] = { .type = NLA_U64},
|
|
+ [TSN_QBV_ATTR_CTRL_LISTENTRY] = { .type = NLA_NESTED},
|
|
+};
|
|
+
|
|
+static const struct nla_policy qbv_entry_policy[TSN_QBV_ATTR_ENTRY_MAX + 1] = {
|
|
+ [TSN_QBV_ATTR_ENTRY_ID] = { .type = NLA_U32},
|
|
+ [TSN_QBV_ATTR_ENTRY_GC] = { .type = NLA_U8},
|
|
+ [TSN_QBV_ATTR_ENTRY_TM] = { .type = NLA_U32},
|
|
+};
|
|
+
|
|
+static const struct nla_policy cb_streamid_policy[TSN_STREAMID_ATTR_MAX + 1] = {
|
|
+ [TSN_STREAMID_ATTR_INDEX] = { .type = NLA_U32},
|
|
+ [TSN_STREAMID_ATTR_ENABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_STREAMID_ATTR_DISABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_STREAMID_ATTR_STREAM_HANDLE] = { .type = NLA_S32},
|
|
+ [TSN_STREAMID_ATTR_IFOP] = { .type = NLA_U32},
|
|
+ [TSN_STREAMID_ATTR_OFOP] = { .type = NLA_U32},
|
|
+ [TSN_STREAMID_ATTR_IFIP] = { .type = NLA_U32},
|
|
+ [TSN_STREAMID_ATTR_OFIP] = { .type = NLA_U32},
|
|
+ [TSN_STREAMID_ATTR_TYPE] = { .type = NLA_U8},
|
|
+ [TSN_STREAMID_ATTR_NDMAC] = { .type = NLA_U64},
|
|
+ [TSN_STREAMID_ATTR_NTAGGED] = { .type = NLA_U8},
|
|
+ [TSN_STREAMID_ATTR_NVID] = { .type = NLA_U16},
|
|
+ [TSN_STREAMID_ATTR_SMAC] = { .type = NLA_U64},
|
|
+ [TSN_STREAMID_ATTR_STAGGED] = { .type = NLA_U8},
|
|
+ [TSN_STREAMID_ATTR_SVID] = { .type = NLA_U16},
|
|
+ [TSN_STREAMID_ATTR_COUNTERS_PSI] = { .type = NLA_U64},
|
|
+ [TSN_STREAMID_ATTR_COUNTERS_PSO] = { .type = NLA_U64},
|
|
+ [TSN_STREAMID_ATTR_COUNTERS_PSPPI] = { .type = NLA_U64},
|
|
+ [TSN_STREAMID_ATTR_COUNTERS_PSPPO] = { .type = NLA_U64},
|
|
+};
|
|
+
|
|
+static const struct nla_policy qci_sfi_policy[TSN_QCI_SFI_ATTR_MAX + 1] = {
|
|
+ [TSN_QCI_SFI_ATTR_INDEX] = { .type = NLA_U32},
|
|
+ [TSN_QCI_SFI_ATTR_ENABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SFI_ATTR_DISABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SFI_ATTR_STREAM_HANDLE] = { .type = NLA_S32},
|
|
+ [TSN_QCI_SFI_ATTR_PRIO_SPEC] = { .type = NLA_S8},
|
|
+ [TSN_QCI_SFI_ATTR_GATE_ID] = { .type = NLA_U32},
|
|
+ [TSN_QCI_SFI_ATTR_FILTER_TYPE] = { .type = NLA_U8},
|
|
+ [TSN_QCI_SFI_ATTR_FLOW_ID] = { .type = NLA_S32},
|
|
+ [TSN_QCI_SFI_ATTR_MAXSDU] = { .type = NLA_U16},
|
|
+ [TSN_QCI_SFI_ATTR_COUNTERS] = {
|
|
+ .len = sizeof(struct tsn_qci_psfp_sfi_counters)},
|
|
+ [TSN_QCI_SFI_ATTR_OVERSIZE_ENABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SFI_ATTR_OVERSIZE] = { .type = NLA_FLAG},
|
|
+};
|
|
+
|
|
+static const struct nla_policy qci_sgi_policy[] = {
|
|
+ [TSN_QCI_SGI_ATTR_INDEX] = { .type = NLA_U32},
|
|
+ [TSN_QCI_SGI_ATTR_ENABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SGI_ATTR_DISABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SGI_ATTR_CONFCHANGE] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SGI_ATTR_IRXEN] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SGI_ATTR_IRX] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SGI_ATTR_OEXEN] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SGI_ATTR_OEX] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SGI_ATTR_ADMINENTRY] = { .type = NLA_NESTED},
|
|
+ [TSN_QCI_SGI_ATTR_OPERENTRY] = { .type = NLA_NESTED},
|
|
+ [TSN_QCI_SGI_ATTR_CCTIME] = { .type = NLA_U64},
|
|
+ [TSN_QCI_SGI_ATTR_TICKG] = { .type = NLA_U32},
|
|
+ [TSN_QCI_SGI_ATTR_CUTIME] = { .type = NLA_U64},
|
|
+ [TSN_QCI_SGI_ATTR_CPENDING] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_SGI_ATTR_CCERROR] = { .type = NLA_U64},
|
|
+};
|
|
+
|
|
+static const struct nla_policy qci_sgi_ctrl_policy[] = {
|
|
+ [TSN_SGI_ATTR_CTRL_INITSTATE] = { .type = NLA_FLAG},
|
|
+ [TSN_SGI_ATTR_CTRL_LEN] = { .type = NLA_U8},
|
|
+ [TSN_SGI_ATTR_CTRL_CYTIME] = { .type = NLA_U32},
|
|
+ [TSN_SGI_ATTR_CTRL_CYTIMEEX] = { .type = NLA_U32},
|
|
+ [TSN_SGI_ATTR_CTRL_BTIME] = { .type = NLA_U64},
|
|
+ [TSN_SGI_ATTR_CTRL_INITIPV] = { .type = NLA_S8},
|
|
+ [TSN_SGI_ATTR_CTRL_GCLENTRY] = { .type = NLA_NESTED},
|
|
+};
|
|
+
|
|
+static const struct nla_policy qci_sgi_gcl_policy[] = {
|
|
+ [TSN_SGI_ATTR_GCL_GATESTATE] = { .type = NLA_FLAG},
|
|
+ [TSN_SGI_ATTR_GCL_IPV] = { .type = NLA_S8},
|
|
+ [TSN_SGI_ATTR_GCL_INTERVAL] = { .type = NLA_U32},
|
|
+ [TSN_SGI_ATTR_GCL_OCTMAX] = { .type = NLA_U32},
|
|
+};
|
|
+
|
|
+static const struct nla_policy qci_fmi_policy[] = {
|
|
+ [TSN_QCI_FMI_ATTR_INDEX] = { .type = NLA_U32},
|
|
+ [TSN_QCI_FMI_ATTR_ENABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_FMI_ATTR_DISABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_FMI_ATTR_CIR] = { .type = NLA_U32},
|
|
+ [TSN_QCI_FMI_ATTR_CBS] = { .type = NLA_U32},
|
|
+ [TSN_QCI_FMI_ATTR_EIR] = { .type = NLA_U32},
|
|
+ [TSN_QCI_FMI_ATTR_EBS] = { .type = NLA_U32},
|
|
+ [TSN_QCI_FMI_ATTR_CF] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_FMI_ATTR_CM] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_FMI_ATTR_DROPYL] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_FMI_ATTR_MAREDEN] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_FMI_ATTR_MARED] = { .type = NLA_FLAG},
|
|
+ [TSN_QCI_FMI_ATTR_COUNTERS] = {
|
|
+ .len = sizeof(struct tsn_qci_psfp_fmi_counters)},
|
|
+};
|
|
+
|
|
+static const struct nla_policy dscp_policy[] = {
|
|
+ [TSN_DSCP_ATTR_INDEX] = { .type = NLA_U32},
|
|
+ [TSN_DSCP_ATTR_DISABLE] = { .type = NLA_FLAG},
|
|
+ [TSN_DSCP_ATTR_COS] = { .type = NLA_U8},
|
|
+ [TSN_DSCP_ATTR_DPL] = { .type = NLA_U8},
|
|
+};
|
|
+
|
|
+static ATOMIC_NOTIFIER_HEAD(tsn_notif_chain);
|
|
+
|
|
+/**
|
|
+ * register_tsn_notifier - Register notifier
|
|
+ * @nb: notifier_block
|
|
+ *
|
|
+ * Register switch device notifier.
|
|
+ */
|
|
+int register_tsn_notifier(struct notifier_block *nb)
|
|
+{
|
|
+ return atomic_notifier_chain_register(&tsn_notif_chain, nb);
|
|
+}
|
|
+EXPORT_SYMBOL_GPL(register_tsn_notifier);
|
|
+
|
|
+/**
|
|
+ * unregister_tsn_notifier - Unregister notifier
|
|
+ * @nb: notifier_block
|
|
+ *
|
|
+ * Unregister switch device notifier.
|
|
+ */
|
|
+int unregister_tsn_notifier(struct notifier_block *nb)
|
|
+{
|
|
+ return atomic_notifier_chain_unregister(&tsn_notif_chain, nb);
|
|
+}
|
|
+EXPORT_SYMBOL_GPL(unregister_tsn_notifier);
|
|
+
|
|
+/**
|
|
+ * call_tsn_notifiers - Call notifiers
|
|
+ * @val: value passed unmodified to notifier function
|
|
+ * @dev: port device
|
|
+ * @info: notifier information data
|
|
+ *
|
|
+ * Call all network notifier blocks.
|
|
+ */
|
|
+int call_tsn_notifiers(unsigned long val, struct net_device *dev,
|
|
+ struct tsn_notifier_info *info)
|
|
+{
|
|
+ info->dev = dev;
|
|
+ return atomic_notifier_call_chain(&tsn_notif_chain, val, info);
|
|
+}
|
|
+EXPORT_SYMBOL_GPL(call_tsn_notifiers);
|
|
+
|
|
+struct tsn_port *tsn_get_port(struct net_device *ndev)
|
|
+{
|
|
+ struct tsn_port *port;
|
|
+ bool tsn_found = false;
|
|
+
|
|
+ list_for_each_entry(port, &port_list, list) {
|
|
+ if (port->netdev == ndev) {
|
|
+ tsn_found = true;
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (!tsn_found)
|
|
+ return NULL;
|
|
+
|
|
+ return port;
|
|
+}
|
|
+EXPORT_SYMBOL_GPL(tsn_get_port);
|
|
+
|
|
+static int tsn_prepare_reply(struct genl_info *info, u8 cmd,
|
|
+ struct sk_buff **skbp, size_t size)
|
|
+{
|
|
+ struct sk_buff *skb;
|
|
+ void *reply;
|
|
+
|
|
+ /* If new attributes are added, please revisit this allocation
|
|
+ */
|
|
+ skb = genlmsg_new(size, GFP_KERNEL);
|
|
+ if (!skb)
|
|
+ return -ENOMEM;
|
|
+
|
|
+ if (!info) {
|
|
+ nlmsg_free(skb);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ reply = genlmsg_put_reply(skb, info, &tsn_family, 0, cmd);
|
|
+ if (!reply) {
|
|
+ nlmsg_free(skb);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ *skbp = skb;
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_mk_reply(struct sk_buff *skb, int aggr, void *data, int len)
|
|
+{
|
|
+ /* add a netlink attribute to a socket buffer */
|
|
+ return nla_put(skb, aggr, len, data);
|
|
+}
|
|
+
|
|
+static int tsn_send_reply(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct genlmsghdr *genlhdr = nlmsg_data(nlmsg_hdr(skb));
|
|
+ void *reply = genlmsg_data(genlhdr);
|
|
+
|
|
+ genlmsg_end(skb, reply);
|
|
+
|
|
+ return genlmsg_reply(skb, info);
|
|
+}
|
|
+
|
|
+static int cmd_attr_echo_message(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ char *msg;
|
|
+ struct sk_buff *rep_skb;
|
|
+ size_t size;
|
|
+ int ret;
|
|
+
|
|
+ na = info->attrs[TSN_CMD_ATTR_MESG];
|
|
+ if (!na)
|
|
+ return -EINVAL;
|
|
+
|
|
+ msg = (char *)nla_data(na);
|
|
+ pr_info("tsn generic netlink receive echo mesg %s\n", msg);
|
|
+
|
|
+ size = nla_total_size(strlen(msg) + 1);
|
|
+
|
|
+ ret = tsn_prepare_reply(info, TSN_CMD_REPLY, &rep_skb,
|
|
+ size + NLMSG_ALIGN(MAX_USER_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ ret = tsn_mk_reply(rep_skb, TSN_CMD_ATTR_MESG, msg, size);
|
|
+ if (ret < 0)
|
|
+ goto err;
|
|
+
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+
|
|
+err:
|
|
+ nlmsg_free(rep_skb);
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int cmd_attr_echo_data(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ s32 data;
|
|
+ struct sk_buff *rep_skb;
|
|
+ size_t size;
|
|
+ int ret;
|
|
+
|
|
+ /*read data */
|
|
+ na = info->attrs[TSN_CMD_ATTR_DATA];
|
|
+ if (!na)
|
|
+ return -EINVAL;
|
|
+
|
|
+ data = nla_get_s32(info->attrs[TSN_CMD_ATTR_DATA]);
|
|
+ pr_info("tsn generic netlink receive echo data %d\n", data);
|
|
+
|
|
+ /* send back */
|
|
+ size = nla_total_size(sizeof(s32));
|
|
+
|
|
+ ret = tsn_prepare_reply(info, TSN_CMD_REPLY, &rep_skb,
|
|
+ size + NLMSG_ALIGN(MAX_USER_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ /* netlink lib func */
|
|
+ ret = nla_put_s32(rep_skb, TSN_CMD_ATTR_DATA, data);
|
|
+ if (ret < 0)
|
|
+ goto err;
|
|
+
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+
|
|
+err:
|
|
+ nlmsg_free(rep_skb);
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int tsn_echo_cmd(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_CMD_ATTR_MESG])
|
|
+ return cmd_attr_echo_message(info);
|
|
+ else if (info->attrs[TSN_CMD_ATTR_DATA])
|
|
+ return cmd_attr_echo_data(info);
|
|
+
|
|
+ return -EINVAL;
|
|
+}
|
|
+
|
|
+static int tsn_simple_reply(struct genl_info *info, u32 cmd,
|
|
+ char *portname, s32 retvalue)
|
|
+{
|
|
+ struct sk_buff *rep_skb;
|
|
+ size_t size;
|
|
+ int ret;
|
|
+
|
|
+ /* send back */
|
|
+ size = nla_total_size(strlen(portname) + 1);
|
|
+ size += nla_total_size(sizeof(s32));
|
|
+
|
|
+ ret = tsn_prepare_reply(info, cmd,
|
|
+ &rep_skb, size + NLMSG_ALIGN(MAX_USER_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ /* netlink lib func */
|
|
+ ret = nla_put_string(rep_skb, TSN_ATTR_IFNAME, portname);
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ ret = nla_put_s32(rep_skb, TSN_CMD_ATTR_DATA, retvalue);
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+}
|
|
+
|
|
+struct tsn_port *tsn_init_check(struct genl_info *info,
|
|
+ struct net_device **ndev)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ char *portname;
|
|
+ struct net_device *netdev;
|
|
+ struct tsn_port *port;
|
|
+ bool tsn_found = false;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ "no portname", -EINVAL);
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_IFNAME];
|
|
+
|
|
+ portname = (char *)nla_data(na);
|
|
+
|
|
+ netdev = __dev_get_by_name(genl_info_net(info), portname);
|
|
+ if (!netdev) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ "error device", -ENODEV);
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ list_for_each_entry(port, &port_list, list) {
|
|
+ if (port->netdev == netdev) {
|
|
+ tsn_found = true;
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (!tsn_found) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -ENODEV);
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ *ndev = netdev;
|
|
+
|
|
+ return port;
|
|
+}
|
|
+
|
|
+static int tsn_cap_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct sk_buff *rep_skb;
|
|
+ struct nlattr *tsn_cap_attr;
|
|
+ int ret;
|
|
+ u32 cap = 0;
|
|
+ struct net_device *netdev;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port) {
|
|
+ ret = -ENODEV;
|
|
+ goto out;
|
|
+ }
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+ genlhdr = info->genlhdr;
|
|
+ if (!tsnops->get_capability) {
|
|
+ ret = -EOPNOTSUPP;
|
|
+ goto out;
|
|
+ }
|
|
+
|
|
+ cap = tsnops->get_capability(netdev);
|
|
+ if (cap < 0) {
|
|
+ ret = cap;
|
|
+ goto out;
|
|
+ }
|
|
+
|
|
+ /* Pad netlink reply data */
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd,
|
|
+ &rep_skb, NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ goto out;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name)) {
|
|
+ ret = -EMSGSIZE;
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ tsn_cap_attr = nla_nest_start_noflag(rep_skb, TSN_ATTR_CAP);
|
|
+ if (!tsn_cap_attr) {
|
|
+ ret = -EMSGSIZE;
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (cap & TSN_CAP_QBV) {
|
|
+ if (nla_put_flag(rep_skb, TSN_CAP_ATTR_QBV))
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (cap & TSN_CAP_QCI) {
|
|
+ if (nla_put_flag(rep_skb, TSN_CAP_ATTR_QCI))
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (cap & TSN_CAP_QBU) {
|
|
+ if (nla_put_flag(rep_skb, TSN_CAP_ATTR_QBU))
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (cap & TSN_CAP_CBS) {
|
|
+ if (nla_put_flag(rep_skb, TSN_CAP_ATTR_CBS))
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (cap & TSN_CAP_CB) {
|
|
+ if (nla_put_flag(rep_skb, TSN_CAP_ATTR_CB))
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (cap & TSN_CAP_TBS) {
|
|
+ if (nla_put_flag(rep_skb, TSN_CAP_ATTR_TBS))
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (cap & TSN_CAP_CTH) {
|
|
+ if (nla_put_flag(rep_skb, TSN_CAP_ATTR_CTH))
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ nla_nest_end(rep_skb, tsn_cap_attr);
|
|
+
|
|
+ tsn_send_reply(rep_skb, info);
|
|
+ return 0;
|
|
+err:
|
|
+ nlmsg_free(rep_skb);
|
|
+out:
|
|
+ if (ret < 0)
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, ret);
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int cmd_cb_streamid_set(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *sid[TSN_STREAMID_ATTR_MAX + 1];
|
|
+ u32 sid_index;
|
|
+ u8 iden_type = 1;
|
|
+ bool enable;
|
|
+ int ret;
|
|
+ struct net_device *netdev;
|
|
+ struct tsn_cb_streamid sidconf;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ memset(&sidconf, 0, sizeof(struct tsn_cb_streamid));
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_STREAM_IDENTIFY])
|
|
+ return -EINVAL;
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_STREAM_IDENTIFY];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(sid, TSN_STREAMID_ATTR_MAX,
|
|
+ na, cb_streamid_policy);
|
|
+ if (ret)
|
|
+ return -EINVAL;
|
|
+
|
|
+ if (!sid[TSN_STREAMID_ATTR_INDEX])
|
|
+ return -EINVAL;
|
|
+
|
|
+ sid_index = nla_get_u32(sid[TSN_STREAMID_ATTR_INDEX]);
|
|
+
|
|
+ if (sid[TSN_STREAMID_ATTR_ENABLE])
|
|
+ enable = true;
|
|
+ else if (sid[TSN_STREAMID_ATTR_DISABLE])
|
|
+ enable = false;
|
|
+ else
|
|
+ return -EINVAL;
|
|
+
|
|
+ if (!enable)
|
|
+ goto loaddev;
|
|
+
|
|
+ if (sid[TSN_STREAMID_ATTR_TYPE])
|
|
+ iden_type = nla_get_u8(sid[TSN_STREAMID_ATTR_TYPE]);
|
|
+ else
|
|
+ return -EINVAL;
|
|
+
|
|
+ sidconf.type = iden_type;
|
|
+ switch (iden_type) {
|
|
+ case STREAMID_NULL:
|
|
+ if (!sid[TSN_STREAMID_ATTR_NDMAC] ||
|
|
+ !sid[TSN_STREAMID_ATTR_NTAGGED] ||
|
|
+ !sid[TSN_STREAMID_ATTR_NVID]) {
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ sidconf.para.nid.dmac =
|
|
+ nla_get_u64(sid[TSN_STREAMID_ATTR_NDMAC]);
|
|
+ sidconf.para.nid.tagged =
|
|
+ nla_get_u8(sid[TSN_STREAMID_ATTR_NTAGGED]);
|
|
+ sidconf.para.nid.vid =
|
|
+ nla_get_u16(sid[TSN_STREAMID_ATTR_NVID]);
|
|
+ break;
|
|
+ case STREAMID_SMAC_VLAN:
|
|
+ /* TODO: not supportted yet */
|
|
+ if (!sid[TSN_STREAMID_ATTR_SMAC] ||
|
|
+ !sid[TSN_STREAMID_ATTR_STAGGED] ||
|
|
+ !sid[TSN_STREAMID_ATTR_SVID]) {
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ sidconf.para.sid.smac =
|
|
+ nla_get_u64(sid[TSN_STREAMID_ATTR_SMAC]);
|
|
+ sidconf.para.sid.tagged =
|
|
+ nla_get_u8(sid[TSN_STREAMID_ATTR_STAGGED]);
|
|
+ sidconf.para.sid.vid =
|
|
+ nla_get_u16(sid[TSN_STREAMID_ATTR_SVID]);
|
|
+ break;
|
|
+ case STREAMID_DMAC_VLAN:
|
|
+
|
|
+ case STREAMID_IP:
|
|
+
|
|
+ default:
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (sid[TSN_STREAMID_ATTR_STREAM_HANDLE])
|
|
+ sidconf.handle =
|
|
+ nla_get_s32(sid[TSN_STREAMID_ATTR_STREAM_HANDLE]);
|
|
+
|
|
+ if (sid[TSN_STREAMID_ATTR_IFOP])
|
|
+ sidconf.ifac_oport = nla_get_u32(sid[TSN_STREAMID_ATTR_IFOP]);
|
|
+ if (sid[TSN_STREAMID_ATTR_OFOP])
|
|
+ sidconf.ofac_oport = nla_get_u32(sid[TSN_STREAMID_ATTR_OFOP]);
|
|
+ if (sid[TSN_STREAMID_ATTR_IFIP])
|
|
+ sidconf.ifac_iport = nla_get_u32(sid[TSN_STREAMID_ATTR_IFIP]);
|
|
+ if (sid[TSN_STREAMID_ATTR_OFIP])
|
|
+ sidconf.ofac_iport = nla_get_u32(sid[TSN_STREAMID_ATTR_OFIP]);
|
|
+
|
|
+loaddev:
|
|
+ if (!tsnops->cb_streamid_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -EOPNOTSUPP;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->cb_streamid_set(netdev, sid_index, enable, &sidconf);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ /* simple reply here. To be continue */
|
|
+ if (tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, 0))
|
|
+ return -1;
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_cb_streamid_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_cb_streamid_set(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_cb_streamid_get(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *sidattr, *sid[TSN_STREAMID_ATTR_MAX + 1];
|
|
+ u32 sid_index;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret, i;
|
|
+ int valid;
|
|
+ struct net_device *netdev;
|
|
+ struct tsn_cb_streamid sidconf;
|
|
+ struct tsn_cb_streamid_counters sidcounts;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ memset(&sidconf, 0, sizeof(struct tsn_cb_streamid));
|
|
+ memset(&sidcounts, 0, sizeof(struct tsn_cb_streamid_counters));
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_STREAM_IDENTIFY])
|
|
+ return -EINVAL;
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_STREAM_IDENTIFY];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(sid, TSN_STREAMID_ATTR_MAX,
|
|
+ na, cb_streamid_policy);
|
|
+ if (ret)
|
|
+ return -EINVAL;
|
|
+
|
|
+ if (!sid[TSN_STREAMID_ATTR_INDEX])
|
|
+ return -EINVAL;
|
|
+
|
|
+ sid_index = nla_get_u32(sid[TSN_STREAMID_ATTR_INDEX]);
|
|
+
|
|
+ if (!tsnops->cb_streamid_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ ret = -EINVAL;
|
|
+ goto exit;
|
|
+ } else {
|
|
+ valid = tsnops->cb_streamid_get(netdev, sid_index, &sidconf);
|
|
+ if (valid < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, valid);
|
|
+ return valid;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ /* send back */
|
|
+ genlhdr = info->genlhdr;
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd, &rep_skb,
|
|
+ NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ /* input netlink the parameters */
|
|
+ sidattr = nla_nest_start_noflag(rep_skb, TSN_ATTR_STREAM_IDENTIFY);
|
|
+ if (!sidattr) {
|
|
+ ret = -EINVAL;
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_STREAMID_ATTR_INDEX, sid_index))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (valid == 1) {
|
|
+ nla_put_flag(rep_skb, TSN_STREAMID_ATTR_ENABLE);
|
|
+ } else if (valid == 0) {
|
|
+ nla_put_flag(rep_skb, TSN_STREAMID_ATTR_DISABLE);
|
|
+ } else {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (nla_put_s32(rep_skb,
|
|
+ TSN_STREAMID_ATTR_STREAM_HANDLE, sidconf.handle) ||
|
|
+ nla_put_u32(rep_skb, TSN_STREAMID_ATTR_IFOP, sidconf.ifac_oport) ||
|
|
+ nla_put_u32(rep_skb, TSN_STREAMID_ATTR_OFOP, sidconf.ofac_oport) ||
|
|
+ nla_put_u32(rep_skb, TSN_STREAMID_ATTR_IFIP, sidconf.ifac_iport) ||
|
|
+ nla_put_u32(rep_skb, TSN_STREAMID_ATTR_OFIP, sidconf.ofac_iport) ||
|
|
+ nla_put_u8(rep_skb, TSN_STREAMID_ATTR_TYPE, sidconf.type))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ switch (sidconf.type) {
|
|
+ case STREAMID_NULL:
|
|
+ if (NLA_PUT_U64(rep_skb, TSN_STREAMID_ATTR_NDMAC,
|
|
+ sidconf.para.nid.dmac) ||
|
|
+ nla_put_u16(rep_skb, TSN_STREAMID_ATTR_NVID,
|
|
+ sidconf.para.nid.vid) ||
|
|
+ nla_put_u8(rep_skb, TSN_STREAMID_ATTR_NTAGGED,
|
|
+ sidconf.para.nid.tagged))
|
|
+ return -EMSGSIZE;
|
|
+ break;
|
|
+ case STREAMID_SMAC_VLAN:
|
|
+ if (NLA_PUT_U64(rep_skb, TSN_STREAMID_ATTR_SMAC,
|
|
+ sidconf.para.sid.smac) ||
|
|
+ nla_put_u16(rep_skb, TSN_STREAMID_ATTR_SVID,
|
|
+ sidconf.para.sid.vid) ||
|
|
+ nla_put_u8(rep_skb, TSN_STREAMID_ATTR_STAGGED,
|
|
+ sidconf.para.sid.tagged))
|
|
+ return -EMSGSIZE;
|
|
+ break;
|
|
+ case STREAMID_DMAC_VLAN:
|
|
+ case STREAMID_IP:
|
|
+ default:
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (!tsnops->cb_streamid_counters_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ goto err;
|
|
+ } else {
|
|
+ ret = tsnops->cb_streamid_counters_get(netdev,
|
|
+ sid_index,
|
|
+ &sidcounts);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ goto err;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (NLA_PUT_U64(rep_skb, TSN_STREAMID_ATTR_COUNTERS_PSI,
|
|
+ sidcounts.per_stream.input) ||
|
|
+ NLA_PUT_U64(rep_skb, TSN_STREAMID_ATTR_COUNTERS_PSO,
|
|
+ sidcounts.per_stream.output))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ for (i = 0; i < 32; i++) {
|
|
+ if (NLA_PUT_U64(rep_skb, TSN_STREAMID_ATTR_COUNTERS_PSPPI,
|
|
+ sidcounts.per_streamport[i].input) ||
|
|
+ NLA_PUT_U64(rep_skb, TSN_STREAMID_ATTR_COUNTERS_PSPPO,
|
|
+ sidcounts.per_streamport[i].output))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ nla_nest_end(rep_skb, sidattr);
|
|
+ /* end netlink input the parameters */
|
|
+
|
|
+ /* netlink lib func */
|
|
+ ret = nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name);
|
|
+ if (ret < 0)
|
|
+ goto err;
|
|
+
|
|
+ ret = nla_put_s32(rep_skb, TSN_CMD_ATTR_DATA, 0);
|
|
+ if (ret < 0)
|
|
+ goto err;
|
|
+
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+
|
|
+err:
|
|
+ nlmsg_free(rep_skb);
|
|
+exit:
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int tsn_cb_streamid_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_cb_streamid_get(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmb_cb_streamid_counters_get(struct genl_info *info)
|
|
+{
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_cb_streamid_counters_get(struct sk_buff *skb,
|
|
+ struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmb_cb_streamid_counters_get(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int tsn_qci_cap_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *qci_cap;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ struct net_device *netdev;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_qci_psfp_stream_param qci_cap_status;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port) {
|
|
+ ret = -EINVAL;
|
|
+ goto out;
|
|
+ }
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ memset(&qci_cap_status, 0, sizeof(qci_cap_status));
|
|
+
|
|
+ if (!tsnops->qci_get_maxcap) {
|
|
+ ret = -EOPNOTSUPP;
|
|
+ goto out;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qci_get_maxcap(netdev, &qci_cap_status);
|
|
+ if (ret < 0)
|
|
+ goto out;
|
|
+
|
|
+ /* Pad netlink reply data */
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd,
|
|
+ &rep_skb, NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ goto out;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name)) {
|
|
+ ret = -EMSGSIZE;
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ qci_cap = nla_nest_start_noflag(rep_skb, TSN_ATTR_QCI_SP);
|
|
+ if (!qci_cap) {
|
|
+ ret = -EMSGSIZE;
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_QCI_STREAM_ATTR_MAX_SFI,
|
|
+ qci_cap_status.max_sf_instance) ||
|
|
+ nla_put_u32(rep_skb, TSN_QCI_STREAM_ATTR_MAX_SGI,
|
|
+ qci_cap_status.max_sg_instance) ||
|
|
+ nla_put_u32(rep_skb, TSN_QCI_STREAM_ATTR_MAX_FMI,
|
|
+ qci_cap_status.max_fm_instance) ||
|
|
+ nla_put_u32(rep_skb, TSN_QCI_STREAM_ATTR_SLM,
|
|
+ qci_cap_status.supported_list_max)) {
|
|
+ ret = -EMSGSIZE;
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ nla_nest_end(rep_skb, qci_cap);
|
|
+
|
|
+ tsn_send_reply(rep_skb, info);
|
|
+
|
|
+ return 0;
|
|
+err:
|
|
+ nlmsg_free(rep_skb);
|
|
+out:
|
|
+ if (ret < 0)
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, ret);
|
|
+
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int cmd_qci_sfi_set(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *sfi[TSN_QCI_SFI_ATTR_MAX + 1];
|
|
+ u32 sfi_handle;
|
|
+ bool enable;
|
|
+ int ret;
|
|
+ struct net_device *netdev;
|
|
+ struct tsn_qci_psfp_sfi_conf sficonf;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ memset(&sficonf, 0, sizeof(struct tsn_qci_psfp_sfi_conf));
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_QCI_SFI])
|
|
+ return -EINVAL;
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_QCI_SFI];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(sfi, TSN_QCI_SFI_ATTR_MAX, na, qci_sfi_policy);
|
|
+ if (ret) {
|
|
+ pr_info("tsn: parse value TSN_QCI_SFI_ATTR_MAX error.");
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (!sfi[TSN_QCI_SFI_ATTR_INDEX])
|
|
+ return -EINVAL;
|
|
+
|
|
+ sfi_handle = nla_get_u32(sfi[TSN_QCI_SFI_ATTR_INDEX]);
|
|
+
|
|
+ if (sfi[TSN_QCI_SFI_ATTR_ENABLE]) {
|
|
+ enable = true;
|
|
+ } else if (sfi[TSN_QCI_SFI_ATTR_DISABLE]) {
|
|
+ enable = false;
|
|
+ goto loaddrive;
|
|
+ } else {
|
|
+ pr_err("tsn: must provde ENABLE or DISABLE attribute.\n");
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (!sfi[TSN_QCI_SFI_ATTR_GATE_ID]) {
|
|
+ pr_err("tsn: must provide stream gate index\n");
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (!sfi[TSN_QCI_SFI_ATTR_STREAM_HANDLE])
|
|
+ sficonf.stream_handle_spec = -1;
|
|
+ else
|
|
+ sficonf.stream_handle_spec =
|
|
+ nla_get_s32(sfi[TSN_QCI_SFI_ATTR_STREAM_HANDLE]);
|
|
+
|
|
+ if (!sfi[TSN_QCI_SFI_ATTR_PRIO_SPEC])
|
|
+ sficonf.priority_spec = -1;
|
|
+ else
|
|
+ sficonf.priority_spec =
|
|
+ nla_get_s8(sfi[TSN_QCI_SFI_ATTR_PRIO_SPEC]);
|
|
+
|
|
+ sficonf.stream_gate_instance_id =
|
|
+ nla_get_u32(sfi[TSN_QCI_SFI_ATTR_GATE_ID]);
|
|
+
|
|
+ if (sfi[TSN_QCI_SFI_ATTR_MAXSDU])
|
|
+ sficonf.stream_filter.maximum_sdu_size =
|
|
+ nla_get_u16(sfi[TSN_QCI_SFI_ATTR_MAXSDU]);
|
|
+ else
|
|
+ sficonf.stream_filter.maximum_sdu_size = 0;
|
|
+
|
|
+ if (sfi[TSN_QCI_SFI_ATTR_FLOW_ID])
|
|
+ sficonf.stream_filter.flow_meter_instance_id =
|
|
+ nla_get_s32(sfi[TSN_QCI_SFI_ATTR_FLOW_ID]);
|
|
+ else
|
|
+ sficonf.stream_filter.flow_meter_instance_id = -1;
|
|
+
|
|
+ if (sfi[TSN_QCI_SFI_ATTR_OVERSIZE_ENABLE])
|
|
+ sficonf.block_oversize_enable = true;
|
|
+
|
|
+ if (sfi[TSN_QCI_SFI_ATTR_OVERSIZE])
|
|
+ sficonf.block_oversize = true;
|
|
+
|
|
+loaddrive:
|
|
+ if (!tsnops->qci_sfi_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qci_sfi_set(netdev, sfi_handle, enable, &sficonf);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ ret = tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, 0);
|
|
+
|
|
+ if (ret)
|
|
+ return ret;
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_qci_sfi_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_qci_sfi_set(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_qci_sfi_get(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *sfiattr;
|
|
+ struct nlattr *sfi[TSN_QCI_SFI_ATTR_MAX + 1];
|
|
+ u32 sfi_handle;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret, valid = 0;
|
|
+ struct net_device *netdev;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_qci_psfp_sfi_conf sficonf;
|
|
+ struct tsn_qci_psfp_sfi_counters sficount;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_QCI_SFI])
|
|
+ return -EINVAL;
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_QCI_SFI];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(sfi, TSN_QCI_SFI_ATTR_MAX,
|
|
+ na, qci_sfi_policy);
|
|
+ if (ret)
|
|
+ return -EINVAL;
|
|
+
|
|
+ if (!sfi[TSN_QCI_SFI_ATTR_INDEX])
|
|
+ return -EINVAL;
|
|
+
|
|
+ sfi_handle = nla_get_u32(sfi[TSN_QCI_SFI_ATTR_INDEX]);
|
|
+
|
|
+ memset(&sficonf, 0, sizeof(struct tsn_qci_psfp_sfi_conf));
|
|
+ memset(&sficount, 0, sizeof(struct tsn_qci_psfp_sfi_counters));
|
|
+
|
|
+ if (!tsnops->qci_sfi_get || !tsnops->qci_sfi_counters_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ ret = -EINVAL;
|
|
+ goto exit;
|
|
+ } else {
|
|
+ valid = tsnops->qci_sfi_get(netdev, sfi_handle, &sficonf);
|
|
+ if (valid < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, valid);
|
|
+ return valid;
|
|
+ }
|
|
+
|
|
+ valid = tsnops->qci_sfi_counters_get(netdev, sfi_handle,
|
|
+ &sficount);
|
|
+ if (valid < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, valid);
|
|
+ return valid;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd,
|
|
+ &rep_skb, NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ goto err;
|
|
+
|
|
+ sfiattr = nla_nest_start_noflag(rep_skb, TSN_ATTR_QCI_SFI);
|
|
+ if (!sfiattr) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ ret = -EINVAL;
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_QCI_SFI_ATTR_INDEX, sfi_handle))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (valid) {
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SFI_ATTR_ENABLE))
|
|
+ return -EMSGSIZE;
|
|
+ } else {
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SFI_ATTR_DISABLE))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (nla_put_s32(rep_skb, TSN_QCI_SFI_ATTR_STREAM_HANDLE,
|
|
+ sficonf.stream_handle_spec) ||
|
|
+ nla_put_s8(rep_skb, TSN_QCI_SFI_ATTR_PRIO_SPEC,
|
|
+ sficonf.priority_spec) ||
|
|
+ nla_put_u32(rep_skb, TSN_QCI_SFI_ATTR_GATE_ID,
|
|
+ sficonf.stream_gate_instance_id))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (sficonf.stream_filter.maximum_sdu_size)
|
|
+ if (nla_put_u16(rep_skb, TSN_QCI_SFI_ATTR_MAXSDU,
|
|
+ sficonf.stream_filter.maximum_sdu_size))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (sficonf.stream_filter.flow_meter_instance_id >= 0)
|
|
+ if (nla_put_s32(rep_skb, TSN_QCI_SFI_ATTR_FLOW_ID,
|
|
+ sficonf.stream_filter.flow_meter_instance_id))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (sficonf.block_oversize_enable)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SFI_ATTR_OVERSIZE_ENABLE))
|
|
+ return -EMSGSIZE;
|
|
+ if (sficonf.block_oversize)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SFI_ATTR_OVERSIZE))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put(rep_skb, TSN_QCI_SFI_ATTR_COUNTERS,
|
|
+ sizeof(struct tsn_qci_psfp_sfi_counters), &sficount))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ nla_nest_end(rep_skb, sfiattr);
|
|
+
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+err:
|
|
+ nlmsg_free(rep_skb);
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+exit:
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int tsn_qci_sfi_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_qci_sfi_get(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_qci_sfi_counters_get(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *sfiattr;
|
|
+ struct nlattr *sfi[TSN_QCI_SFI_ATTR_MAX + 1];
|
|
+ u32 sfi_handle;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ struct net_device *netdev;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_qci_psfp_sfi_counters sficount;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_QCI_SFI])
|
|
+ return -EINVAL;
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_QCI_SFI];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(sfi, TSN_QCI_SFI_ATTR_MAX,
|
|
+ na, qci_sfi_policy);
|
|
+ if (ret)
|
|
+ return -EINVAL;
|
|
+
|
|
+ if (!sfi[TSN_QCI_SFI_ATTR_INDEX])
|
|
+ return -EINVAL;
|
|
+
|
|
+ sfi_handle = nla_get_u32(sfi[TSN_QCI_SFI_ATTR_INDEX]);
|
|
+
|
|
+ memset(&sficount, 0, sizeof(struct tsn_qci_psfp_sfi_counters));
|
|
+ if (!tsnops->qci_sfi_counters_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qci_sfi_counters_get(netdev, sfi_handle, &sficount);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd, &rep_skb,
|
|
+ NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ goto err;
|
|
+
|
|
+ sfiattr = nla_nest_start_noflag(rep_skb, TSN_ATTR_QCI_SFI);
|
|
+ if (!sfiattr) {
|
|
+ ret = -EINVAL;
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_QCI_SFI_ATTR_INDEX, sfi_handle))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ ret = tsnops->qci_sfi_counters_get(netdev, sfi_handle, &sficount);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ if (nla_put(rep_skb, TSN_QCI_SFI_ATTR_COUNTERS,
|
|
+ sizeof(struct tsn_qci_psfp_sfi_counters), &sficount))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ nla_nest_end(rep_skb, sfiattr);
|
|
+
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+err:
|
|
+ nlmsg_free(rep_skb);
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, -EINVAL);
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int tsn_qci_sfi_counters_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_qci_sfi_counters_get(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_qci_sgi_set(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ struct nlattr *sgia[TSN_QCI_SGI_ATTR_MAX + 1];
|
|
+ struct nlattr *admin[TSN_SGI_ATTR_CTRL_MAX + 1];
|
|
+ int ret = 0;
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_qci_psfp_sgi_conf sgi;
|
|
+ struct tsn_qci_psfp_gcl *gcl = NULL;
|
|
+ u16 sgi_handle = 0;
|
|
+ u16 listcount = 0;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ memset(&sgi, 0, sizeof(struct tsn_qci_psfp_sgi_conf));
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_QCI_SGI]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_QCI_SGI];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(sgia, TSN_QCI_SGI_ATTR_MAX,
|
|
+ na, qci_sgi_policy);
|
|
+ if (ret) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (sgia[TSN_QCI_SGI_ATTR_ENABLE] && sgia[TSN_QCI_SGI_ATTR_DISABLE]) {
|
|
+ pr_err("tsn: enable or disable?\n");
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ if (sgia[TSN_QCI_SGI_ATTR_INDEX])
|
|
+ sgi_handle = nla_get_u32(sgia[TSN_QCI_SGI_ATTR_INDEX]);
|
|
+
|
|
+ if (sgia[TSN_QCI_SGI_ATTR_DISABLE]) {
|
|
+ sgi.gate_enabled = 0;
|
|
+ goto loaddev;
|
|
+ } else {
|
|
+ /* set default to be enable*/
|
|
+ sgi.gate_enabled = 1;
|
|
+ }
|
|
+
|
|
+ if (sgia[TSN_QCI_SGI_ATTR_CONFCHANGE])
|
|
+ sgi.config_change = 1;
|
|
+
|
|
+ if (sgia[TSN_QCI_SGI_ATTR_IRXEN])
|
|
+ sgi.block_invalid_rx_enable = 1;
|
|
+
|
|
+ if (sgia[TSN_QCI_SGI_ATTR_IRX])
|
|
+ sgi.block_invalid_rx = 1;
|
|
+
|
|
+ if (sgia[TSN_QCI_SGI_ATTR_OEXEN])
|
|
+ sgi.block_octets_exceeded_enable = 1;
|
|
+
|
|
+ if (sgia[TSN_QCI_SGI_ATTR_OEX])
|
|
+ sgi.block_octets_exceeded = 1;
|
|
+
|
|
+ if (sgia[TSN_QCI_SGI_ATTR_ADMINENTRY]) {
|
|
+ struct nlattr *entry;
|
|
+ int rem;
|
|
+ int count = 0;
|
|
+
|
|
+ na = sgia[TSN_QCI_SGI_ATTR_ADMINENTRY];
|
|
+ ret = NLA_PARSE_NESTED(admin, TSN_SGI_ATTR_CTRL_MAX,
|
|
+ na, qci_sgi_ctrl_policy);
|
|
+
|
|
+ /* Other parameters in admin control */
|
|
+ if (admin[TSN_SGI_ATTR_CTRL_INITSTATE])
|
|
+ sgi.admin.gate_states = 1;
|
|
+
|
|
+ if (admin[TSN_SGI_ATTR_CTRL_CYTIME])
|
|
+ sgi.admin.cycle_time =
|
|
+ nla_get_u32(admin[TSN_SGI_ATTR_CTRL_CYTIME]);
|
|
+
|
|
+ if (admin[TSN_SGI_ATTR_CTRL_CYTIMEEX])
|
|
+ sgi.admin.cycle_time_extension =
|
|
+ nla_get_u32(admin[TSN_SGI_ATTR_CTRL_CYTIMEEX]);
|
|
+
|
|
+ if (admin[TSN_SGI_ATTR_CTRL_BTIME])
|
|
+ sgi.admin.base_time =
|
|
+ nla_get_u64(admin[TSN_SGI_ATTR_CTRL_BTIME]);
|
|
+
|
|
+ if (admin[TSN_SGI_ATTR_CTRL_INITIPV])
|
|
+ sgi.admin.init_ipv =
|
|
+ nla_get_s8(admin[TSN_SGI_ATTR_CTRL_INITIPV]);
|
|
+ else
|
|
+ sgi.admin.init_ipv = -1;
|
|
+
|
|
+ if (admin[TSN_SGI_ATTR_CTRL_LEN]) {
|
|
+ sgi.admin.control_list_length =
|
|
+ nla_get_u8(admin[TSN_SGI_ATTR_CTRL_LEN]);
|
|
+ listcount = sgi.admin.control_list_length;
|
|
+ }
|
|
+
|
|
+ if (!listcount)
|
|
+ goto loaddev;
|
|
+
|
|
+ gcl = kmalloc_array(listcount, sizeof(*gcl), GFP_KERNEL);
|
|
+
|
|
+ memset(gcl, 0, listcount * sizeof(struct tsn_qci_psfp_gcl));
|
|
+
|
|
+ /* Check the whole admin attrs,
|
|
+ * checkout the TSN_SGI_ATTR_CTRL_GCLENTRY attributes
|
|
+ */
|
|
+ nla_for_each_nested(entry, na, rem) {
|
|
+ struct nlattr *gcl_entry[TSN_SGI_ATTR_GCL_MAX + 1];
|
|
+ struct nlattr *ti, *om;
|
|
+
|
|
+ if (nla_type(entry) != TSN_SGI_ATTR_CTRL_GCLENTRY)
|
|
+ continue;
|
|
+
|
|
+ /* parse each TSN_SGI_ATTR_CTRL_GCLENTRY */
|
|
+ ret = NLA_PARSE_NESTED(gcl_entry, TSN_SGI_ATTR_GCL_MAX,
|
|
+ entry, qci_sgi_gcl_policy);
|
|
+ /* Parse gate control list */
|
|
+ if (gcl_entry[TSN_SGI_ATTR_GCL_GATESTATE])
|
|
+ (gcl + count)->gate_state = 1;
|
|
+
|
|
+ if (gcl_entry[TSN_SGI_ATTR_GCL_IPV])
|
|
+ (gcl + count)->ipv =
|
|
+ nla_get_s8(gcl_entry[TSN_SGI_ATTR_GCL_IPV]);
|
|
+
|
|
+ if (gcl_entry[TSN_SGI_ATTR_GCL_INTERVAL]) {
|
|
+ ti = gcl_entry[TSN_SGI_ATTR_GCL_INTERVAL];
|
|
+ (gcl + count)->time_interval = nla_get_u32(ti);
|
|
+ }
|
|
+
|
|
+ if (gcl_entry[TSN_SGI_ATTR_GCL_OCTMAX]) {
|
|
+ om = gcl_entry[TSN_SGI_ATTR_GCL_OCTMAX];
|
|
+ (gcl + count)->octet_max = nla_get_u32(om);
|
|
+ }
|
|
+
|
|
+ count++;
|
|
+
|
|
+ if (count >= listcount)
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ if (count < listcount) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ pr_err("tsn: count less than TSN_SGI_ATTR_CTRL_LEN\n");
|
|
+ kfree(gcl);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ } else {
|
|
+ pr_info("tsn: no admin list parameters setting\n");
|
|
+ }
|
|
+
|
|
+loaddev:
|
|
+ if (!tsnops->qci_sgi_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ kfree(gcl);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ sgi.admin.gcl = gcl;
|
|
+
|
|
+ ret = tsnops->qci_sgi_set(netdev, sgi_handle, &sgi);
|
|
+ kfree(gcl);
|
|
+ if (!ret)
|
|
+ return tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, 0);
|
|
+
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int tsn_qci_sgi_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_qci_sgi_set(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_qci_sgi_get(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *sgiattr, *adminattr, *sglattr;
|
|
+ struct nlattr *sgi[TSN_QCI_SGI_ATTR_MAX + 1];
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ struct net_device *netdev;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_qci_psfp_sgi_conf sgiadmin;
|
|
+ struct tsn_qci_psfp_gcl *gcl = NULL;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ u16 sgi_handle;
|
|
+ u8 listcount, i;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_QCI_SGI]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ pr_err("tsn: no sgi handle input\n");
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_QCI_SGI];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(sgi, TSN_QCI_SGI_ATTR_MAX,
|
|
+ na, qci_sgi_policy);
|
|
+ if (ret)
|
|
+ return -EINVAL;
|
|
+
|
|
+ if (!sgi[TSN_QCI_SGI_ATTR_INDEX]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ pr_err("tsn: no sgi handle input\n");
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ sgi_handle = nla_get_u32(sgi[TSN_QCI_SGI_ATTR_INDEX]);
|
|
+
|
|
+ /* Get config data from device */
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ memset(&sgiadmin, 0, sizeof(struct tsn_qci_psfp_sgi_conf));
|
|
+
|
|
+ if (!tsnops->qci_sgi_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qci_sgi_get(netdev, sgi_handle, &sgiadmin);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ /* Form netlink reply data */
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd,
|
|
+ &rep_skb, NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ sgiattr = nla_nest_start_noflag(rep_skb, TSN_ATTR_QCI_SGI);
|
|
+ if (!sgiattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_QCI_SGI_ATTR_INDEX, sgi_handle))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ /* Gate enable? sgiadmin.gate_enabled */
|
|
+ if (sgiadmin.gate_enabled) {
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SGI_ATTR_ENABLE))
|
|
+ return -EMSGSIZE;
|
|
+ } else {
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SGI_ATTR_DISABLE))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (sgiadmin.config_change)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SGI_ATTR_CONFCHANGE))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (sgiadmin.block_invalid_rx_enable)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SGI_ATTR_IRXEN))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (sgiadmin.block_invalid_rx)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SGI_ATTR_IRX))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (sgiadmin.block_octets_exceeded_enable)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SGI_ATTR_OEXEN))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (sgiadmin.block_octets_exceeded)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SGI_ATTR_OEX))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ /* Administration */
|
|
+ adminattr = nla_nest_start_noflag(rep_skb, TSN_QCI_SGI_ATTR_ADMINENTRY);
|
|
+ if (!adminattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (sgiadmin.admin.gate_states)
|
|
+ if (nla_put_flag(rep_skb, TSN_SGI_ATTR_CTRL_INITSTATE))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_SGI_ATTR_CTRL_CYTIME,
|
|
+ sgiadmin.admin.cycle_time) ||
|
|
+ nla_put_u32(rep_skb, TSN_SGI_ATTR_CTRL_CYTIMEEX,
|
|
+ sgiadmin.admin.cycle_time_extension) ||
|
|
+ NLA_PUT_U64(rep_skb, TSN_SGI_ATTR_CTRL_BTIME,
|
|
+ sgiadmin.admin.base_time) ||
|
|
+ nla_put_u8(rep_skb, TSN_SGI_ATTR_CTRL_INITIPV,
|
|
+ sgiadmin.admin.init_ipv))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ listcount = sgiadmin.admin.control_list_length;
|
|
+ if (!listcount)
|
|
+ goto out1;
|
|
+
|
|
+ if (!sgiadmin.admin.gcl) {
|
|
+ pr_err("error: no gate control list\n");
|
|
+ ret = -EINVAL;
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ gcl = sgiadmin.admin.gcl;
|
|
+
|
|
+ /* loop list */
|
|
+ for (i = 0; i < listcount; i++) {
|
|
+ s8 ipv;
|
|
+ u32 ti, omax;
|
|
+
|
|
+ if (!(gcl + i)) {
|
|
+ pr_err("error: list count too big\n");
|
|
+ ret = -EINVAL;
|
|
+ kfree(sgiadmin.admin.gcl);
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ /* Adminastration entry */
|
|
+ sglattr = nla_nest_start_noflag(rep_skb,
|
|
+ TSN_SGI_ATTR_CTRL_GCLENTRY);
|
|
+ if (!sglattr)
|
|
+ return -EMSGSIZE;
|
|
+ ipv = (gcl + i)->ipv;
|
|
+ ti = (gcl + i)->time_interval;
|
|
+ omax = (gcl + i)->octet_max;
|
|
+
|
|
+ if ((gcl + i)->gate_state)
|
|
+ if (nla_put_flag(rep_skb, TSN_SGI_ATTR_GCL_GATESTATE))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_s8(rep_skb, TSN_SGI_ATTR_GCL_IPV, ipv) ||
|
|
+ nla_put_u32(rep_skb, TSN_SGI_ATTR_GCL_INTERVAL, ti) ||
|
|
+ nla_put_u32(rep_skb, TSN_SGI_ATTR_GCL_OCTMAX, omax))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ /* End administration entry */
|
|
+ nla_nest_end(rep_skb, sglattr);
|
|
+ }
|
|
+
|
|
+ kfree(sgiadmin.admin.gcl);
|
|
+ if (nla_put_u8(rep_skb, TSN_SGI_ATTR_CTRL_LEN, listcount))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+out1:
|
|
+ /* End adminastration */
|
|
+ nla_nest_end(rep_skb, adminattr);
|
|
+
|
|
+ nla_nest_end(rep_skb, sgiattr);
|
|
+
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+err:
|
|
+ nlmsg_free(rep_skb);
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, ret);
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int tsn_qci_sgi_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_qci_sgi_get(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_qci_sgi_status_get(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *sgiattr, *operattr, *sglattr;
|
|
+ struct nlattr *sgi[TSN_QCI_SGI_ATTR_MAX + 1];
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ struct net_device *netdev;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_psfp_sgi_status sgistat;
|
|
+ struct tsn_qci_psfp_gcl *gcl = NULL;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ u16 sgi_handle;
|
|
+ u8 listcount;
|
|
+ int valid, i;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_QCI_SGI]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ pr_err("tsn: no sgi handle input\n");
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_QCI_SGI];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(sgi, TSN_QCI_SGI_ATTR_MAX,
|
|
+ na, qci_sgi_policy);
|
|
+ if (ret)
|
|
+ return -EINVAL;
|
|
+
|
|
+ if (!sgi[TSN_QCI_SGI_ATTR_INDEX]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ pr_err("tsn: no sgi handle input\n");
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ sgi_handle = nla_get_u32(sgi[TSN_QCI_SGI_ATTR_INDEX]);
|
|
+
|
|
+ /* Get status data from device */
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ memset(&sgistat, 0, sizeof(struct tsn_psfp_sgi_status));
|
|
+
|
|
+ if (!tsnops->qci_sgi_status_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ valid = tsnops->qci_sgi_status_get(netdev, sgi_handle, &sgistat);
|
|
+ if (valid < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, valid);
|
|
+ return valid;
|
|
+ }
|
|
+
|
|
+ /* Form netlink reply data */
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd,
|
|
+ &rep_skb, NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ /* Down one netlink attribute level */
|
|
+ sgiattr = nla_nest_start_noflag(rep_skb, TSN_ATTR_QCI_SGI);
|
|
+ if (!sgiattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_QCI_SGI_ATTR_INDEX, sgi_handle))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ /* Gate enable */
|
|
+ if (valid == 1) {
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SGI_ATTR_ENABLE))
|
|
+ return -EMSGSIZE;
|
|
+ } else {
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SGI_ATTR_DISABLE))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_QCI_SGI_ATTR_TICKG,
|
|
+ sgistat.tick_granularity) ||
|
|
+ NLA_PUT_U64(rep_skb, TSN_QCI_SGI_ATTR_CCTIME,
|
|
+ sgistat.config_change_time) ||
|
|
+ NLA_PUT_U64(rep_skb, TSN_QCI_SGI_ATTR_CUTIME,
|
|
+ sgistat.current_time) ||
|
|
+ NLA_PUT_U64(rep_skb, TSN_QCI_SGI_ATTR_CCERROR,
|
|
+ sgistat.config_change_error))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (sgistat.config_pending)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_SGI_ATTR_CPENDING))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ /* operation data */
|
|
+ operattr = nla_nest_start_noflag(rep_skb, TSN_QCI_SGI_ATTR_OPERENTRY);
|
|
+ if (!operattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (sgistat.oper.gate_states)
|
|
+ if (nla_put_flag(rep_skb, TSN_SGI_ATTR_CTRL_INITSTATE))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_SGI_ATTR_CTRL_CYTIME,
|
|
+ sgistat.oper.cycle_time) ||
|
|
+ nla_put_u32(rep_skb, TSN_SGI_ATTR_CTRL_CYTIMEEX,
|
|
+ sgistat.oper.cycle_time_extension) ||
|
|
+ NLA_PUT_U64(rep_skb, TSN_SGI_ATTR_CTRL_BTIME,
|
|
+ sgistat.oper.base_time) ||
|
|
+ nla_put_u8(rep_skb, TSN_SGI_ATTR_CTRL_INITIPV,
|
|
+ sgistat.oper.init_ipv))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ /* Loop list */
|
|
+ listcount = sgistat.oper.control_list_length;
|
|
+ if (!listcount)
|
|
+ goto out1;
|
|
+
|
|
+ if (!sgistat.oper.gcl) {
|
|
+ pr_err("error: list lenghth is not zero!\n");
|
|
+ ret = -EINVAL;
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ gcl = sgistat.oper.gcl;
|
|
+
|
|
+ /* loop list */
|
|
+ for (i = 0; i < listcount; i++) {
|
|
+ s8 ipv;
|
|
+ u32 ti, omax;
|
|
+
|
|
+ if (!(gcl + i)) {
|
|
+ pr_err("error: list count too big\n");
|
|
+ ret = -EINVAL;
|
|
+ kfree(sgistat.oper.gcl);
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ /* Operation entry */
|
|
+ sglattr = nla_nest_start_noflag(rep_skb,
|
|
+ TSN_SGI_ATTR_CTRL_GCLENTRY);
|
|
+ if (!sglattr)
|
|
+ return -EMSGSIZE;
|
|
+ ipv = (gcl + i)->ipv;
|
|
+ ti = (gcl + i)->time_interval;
|
|
+ omax = (gcl + i)->octet_max;
|
|
+
|
|
+ if ((gcl + i)->gate_state)
|
|
+ if (nla_put_flag(rep_skb, TSN_SGI_ATTR_GCL_GATESTATE))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_s8(rep_skb, TSN_SGI_ATTR_GCL_IPV, ipv) ||
|
|
+ nla_put_u32(rep_skb, TSN_SGI_ATTR_GCL_INTERVAL, ti) ||
|
|
+ nla_put_u32(rep_skb, TSN_SGI_ATTR_GCL_OCTMAX, omax))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ /* End operation entry */
|
|
+ nla_nest_end(rep_skb, sglattr);
|
|
+ }
|
|
+
|
|
+ kfree(sgistat.oper.gcl);
|
|
+ if (nla_put_u8(rep_skb, TSN_SGI_ATTR_CTRL_LEN, listcount))
|
|
+ return -EMSGSIZE;
|
|
+out1:
|
|
+ /* End operation */
|
|
+ nla_nest_end(rep_skb, operattr);
|
|
+
|
|
+ nla_nest_end(rep_skb, sgiattr);
|
|
+
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+err:
|
|
+ nlmsg_free(rep_skb);
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, ret);
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int tsn_qci_sgi_status_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_qci_sgi_status_get(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_qci_fmi_set(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *fmi[TSN_QCI_FMI_ATTR_MAX + 1];
|
|
+ u32 index;
|
|
+ int ret;
|
|
+ struct net_device *netdev;
|
|
+ struct tsn_qci_psfp_fmi fmiconf;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ bool enable = 0;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ memset(&fmiconf, 0, sizeof(struct tsn_qci_psfp_fmi));
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_QCI_FMI])
|
|
+ return -EINVAL;
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_QCI_FMI];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(fmi, TSN_QCI_FMI_ATTR_MAX, na, qci_fmi_policy);
|
|
+ if (ret) {
|
|
+ pr_info("tsn: parse value TSN_QCI_FMI_ATTR_MAX error.");
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (!fmi[TSN_QCI_FMI_ATTR_INDEX])
|
|
+ return -EINVAL;
|
|
+
|
|
+ index = nla_get_u32(fmi[TSN_QCI_FMI_ATTR_INDEX]);
|
|
+
|
|
+ if (fmi[TSN_QCI_FMI_ATTR_DISABLE])
|
|
+ goto loaddev;
|
|
+
|
|
+ enable = 1;
|
|
+
|
|
+ if (fmi[TSN_QCI_FMI_ATTR_CIR])
|
|
+ fmiconf.cir = nla_get_u32(fmi[TSN_QCI_FMI_ATTR_CIR]);
|
|
+
|
|
+ if (fmi[TSN_QCI_FMI_ATTR_CBS])
|
|
+ fmiconf.cbs = nla_get_u32(fmi[TSN_QCI_FMI_ATTR_CBS]);
|
|
+
|
|
+ if (fmi[TSN_QCI_FMI_ATTR_EIR])
|
|
+ fmiconf.eir = nla_get_u32(fmi[TSN_QCI_FMI_ATTR_EIR]);
|
|
+
|
|
+ if (fmi[TSN_QCI_FMI_ATTR_EBS])
|
|
+ fmiconf.ebs = nla_get_u32(fmi[TSN_QCI_FMI_ATTR_EBS]);
|
|
+
|
|
+ if (fmi[TSN_QCI_FMI_ATTR_CF])
|
|
+ fmiconf.cf = 1;
|
|
+
|
|
+ if (fmi[TSN_QCI_FMI_ATTR_CM])
|
|
+ fmiconf.cm = 1;
|
|
+
|
|
+ if (fmi[TSN_QCI_FMI_ATTR_DROPYL])
|
|
+ fmiconf.drop_on_yellow = 1;
|
|
+
|
|
+ if (fmi[TSN_QCI_FMI_ATTR_MAREDEN])
|
|
+ fmiconf.mark_red_enable = 1;
|
|
+
|
|
+ if (fmi[TSN_QCI_FMI_ATTR_MARED])
|
|
+ fmiconf.mark_red = 1;
|
|
+
|
|
+loaddev:
|
|
+
|
|
+ if (!tsnops->qci_fmi_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qci_fmi_set(netdev, index, enable, &fmiconf);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ ret = tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, 0);
|
|
+
|
|
+ if (ret)
|
|
+ return ret;
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_qci_fmi_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_qci_fmi_set(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_qci_fmi_get(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *fmi[TSN_QCI_FMI_ATTR_MAX + 1], *fmiattr;
|
|
+ u32 index;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ struct net_device *netdev;
|
|
+ struct tsn_qci_psfp_fmi fmiconf;
|
|
+ struct tsn_qci_psfp_fmi_counters counters;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_QCI_FMI])
|
|
+ return -EINVAL;
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_QCI_FMI];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(fmi, TSN_QCI_FMI_ATTR_MAX,
|
|
+ na, qci_fmi_policy);
|
|
+ if (ret) {
|
|
+ pr_info("tsn: parse value TSN_QCI_FMI_ATTR_MAX error.");
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (!fmi[TSN_QCI_FMI_ATTR_INDEX])
|
|
+ return -EINVAL;
|
|
+
|
|
+ index = nla_get_u32(fmi[TSN_QCI_FMI_ATTR_INDEX]);
|
|
+
|
|
+ /* Get data from device */
|
|
+ memset(&fmiconf, 0, sizeof(struct tsn_qci_psfp_fmi));
|
|
+ memset(&counters, 0, sizeof(struct tsn_qci_psfp_fmi_counters));
|
|
+
|
|
+ if (!tsnops->qci_fmi_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qci_fmi_get(netdev, index, &fmiconf, &counters);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY, netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ /* Form netlink reply data */
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd,
|
|
+ &rep_skb, NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ fmiattr = nla_nest_start_noflag(rep_skb, TSN_ATTR_QCI_FMI);
|
|
+ if (!fmiattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_QCI_FMI_ATTR_INDEX, index) ||
|
|
+ nla_put_u32(rep_skb, TSN_QCI_FMI_ATTR_CIR, fmiconf.cir) ||
|
|
+ nla_put_u32(rep_skb, TSN_QCI_FMI_ATTR_CBS, fmiconf.cbs) ||
|
|
+ nla_put_u32(rep_skb, TSN_QCI_FMI_ATTR_EIR, fmiconf.eir) ||
|
|
+ nla_put_u32(rep_skb, TSN_QCI_FMI_ATTR_EBS, fmiconf.ebs))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (fmiconf.cf)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_FMI_ATTR_CF))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (fmiconf.cm)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_FMI_ATTR_CM))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (fmiconf.drop_on_yellow)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_FMI_ATTR_DROPYL))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (fmiconf.mark_red_enable)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_FMI_ATTR_MAREDEN))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (fmiconf.mark_red)
|
|
+ if (nla_put_flag(rep_skb, TSN_QCI_FMI_ATTR_MAREDEN))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put(rep_skb, TSN_QCI_FMI_ATTR_COUNTERS,
|
|
+ sizeof(struct tsn_qci_psfp_fmi_counters), &counters))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ nla_nest_end(rep_skb, fmiattr);
|
|
+
|
|
+ tsn_send_reply(rep_skb, info);
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_qci_fmi_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_qci_fmi_get(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_qbv_set(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *na1;
|
|
+ struct nlattr *qbv_table;
|
|
+ struct nlattr *qbv[TSN_QBV_ATTR_MAX + 1];
|
|
+ struct nlattr *qbvctrl[TSN_QBV_ATTR_CTRL_MAX + 1];
|
|
+ int rem;
|
|
+ int ret = 0;
|
|
+ struct net_device *netdev;
|
|
+ struct tsn_qbv_conf qbvconfig;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_qbv_entry *gatelist = NULL;
|
|
+ int count = 0;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ memset(&qbvconfig, 0, sizeof(struct tsn_qbv_conf));
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_QBV])
|
|
+ return -EINVAL;
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_QBV];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(qbv, TSN_QBV_ATTR_MAX, na, qbv_policy);
|
|
+ if (ret)
|
|
+ return -EINVAL;
|
|
+
|
|
+ if (qbv[TSN_QBV_ATTR_ENABLE])
|
|
+ qbvconfig.gate_enabled = 1;
|
|
+ else
|
|
+ goto setdrive;
|
|
+
|
|
+ if (qbv[TSN_QBV_ATTR_CONFIGCHANGE])
|
|
+ qbvconfig.config_change = 1;
|
|
+
|
|
+ if (!qbv[TSN_QBV_ATTR_ADMINENTRY]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ na1 = qbv[TSN_QBV_ATTR_ADMINENTRY];
|
|
+ NLA_PARSE_NESTED(qbvctrl, TSN_QBV_ATTR_CTRL_MAX,
|
|
+ na1, qbv_ctrl_policy);
|
|
+
|
|
+ if (qbvctrl[TSN_QBV_ATTR_CTRL_CYCLETIME]) {
|
|
+ qbvconfig.admin.cycle_time =
|
|
+ nla_get_u32(qbvctrl[TSN_QBV_ATTR_CTRL_CYCLETIME]);
|
|
+ }
|
|
+
|
|
+ if (qbvctrl[TSN_QBV_ATTR_CTRL_CYCLETIMEEXT]) {
|
|
+ qbvconfig.admin.cycle_time_extension =
|
|
+ nla_get_u32(qbvctrl[TSN_QBV_ATTR_CTRL_CYCLETIMEEXT]);
|
|
+ }
|
|
+
|
|
+ if (qbvctrl[TSN_QBV_ATTR_CTRL_BASETIME]) {
|
|
+ qbvconfig.admin.base_time =
|
|
+ nla_get_u64(qbvctrl[TSN_QBV_ATTR_CTRL_BASETIME]);
|
|
+ }
|
|
+
|
|
+ if (qbvctrl[TSN_QBV_ATTR_CTRL_GATESTATE]) {
|
|
+ qbvconfig.admin.gate_states =
|
|
+ nla_get_u8(qbvctrl[TSN_QBV_ATTR_CTRL_GATESTATE]);
|
|
+ }
|
|
+
|
|
+ if (qbvctrl[TSN_QBV_ATTR_CTRL_LISTCOUNT]) {
|
|
+ int listcount;
|
|
+
|
|
+ listcount = nla_get_u32(qbvctrl[TSN_QBV_ATTR_CTRL_LISTCOUNT]);
|
|
+
|
|
+ qbvconfig.admin.control_list_length = listcount;
|
|
+
|
|
+ gatelist = kmalloc_array(listcount,
|
|
+ sizeof(*gatelist),
|
|
+ GFP_KERNEL);
|
|
+
|
|
+ nla_for_each_nested(qbv_table, na1, rem) {
|
|
+ struct nlattr *qbv_entry[TSN_QBV_ATTR_ENTRY_MAX + 1];
|
|
+
|
|
+ if (nla_type(qbv_table) != TSN_QBV_ATTR_CTRL_LISTENTRY)
|
|
+ continue;
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(qbv_entry,
|
|
+ TSN_QBV_ATTR_ENTRY_MAX,
|
|
+ qbv_table, qbv_entry_policy);
|
|
+ if (ret)
|
|
+ return -EINVAL;
|
|
+
|
|
+ (gatelist + count)->gate_state =
|
|
+ nla_get_u8(qbv_entry[TSN_QBV_ATTR_ENTRY_GC]);
|
|
+ (gatelist + count)->time_interval =
|
|
+ nla_get_u32(qbv_entry[TSN_QBV_ATTR_ENTRY_TM]);
|
|
+ count++;
|
|
+ if (count > listcount)
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (gatelist)
|
|
+ qbvconfig.admin.control_list = gatelist;
|
|
+
|
|
+setdrive:
|
|
+ if (!tsnops->qbv_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ goto err;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qbv_set(netdev, &qbvconfig);
|
|
+
|
|
+ /* send back */
|
|
+ if (ret < 0)
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ else
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, 0);
|
|
+
|
|
+err:
|
|
+ kfree(gatelist);
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int tsn_qbv_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME]) {
|
|
+ cmd_qbv_set(info);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_qbv_get(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *qbv, *qbvadminattr;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ int len = 0, i = 0;
|
|
+ struct net_device *netdev;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_qbv_conf qbvconf;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ memset(&qbvconf, 0, sizeof(struct tsn_qbv_conf));
|
|
+
|
|
+ if (!tsnops->qbv_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qbv_get(netdev, &qbvconf);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd,
|
|
+ &rep_skb, NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ qbv = nla_nest_start_noflag(rep_skb, TSN_ATTR_QBV);
|
|
+ if (!qbv)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ qbvadminattr = nla_nest_start_noflag(rep_skb, TSN_QBV_ATTR_ADMINENTRY);
|
|
+ if (!qbvadminattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (qbvconf.admin.control_list) {
|
|
+ len = qbvconf.admin.control_list_length;
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_CTRL_LISTCOUNT, len))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ for (i = 0; i < len; i++) {
|
|
+ struct nlattr *qbv_table;
|
|
+ u8 gs;
|
|
+ u32 tp;
|
|
+ int glisttype = TSN_QBV_ATTR_CTRL_LISTENTRY;
|
|
+
|
|
+ gs = (qbvconf.admin.control_list + i)->gate_state;
|
|
+ tp = (qbvconf.admin.control_list + i)->time_interval;
|
|
+
|
|
+ qbv_table =
|
|
+ nla_nest_start_noflag(rep_skb, glisttype);
|
|
+ if (!qbv_table)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_ENTRY_ID, i) ||
|
|
+ nla_put_u8(rep_skb, TSN_QBV_ATTR_ENTRY_GC, gs) ||
|
|
+ nla_put_u32(rep_skb, TSN_QBV_ATTR_ENTRY_TM, tp))
|
|
+ return -EMSGSIZE;
|
|
+ nla_nest_end(rep_skb, qbv_table);
|
|
+ }
|
|
+
|
|
+ if (qbvconf.admin.gate_states)
|
|
+ if (nla_put_u8(rep_skb, TSN_QBV_ATTR_CTRL_GATESTATE,
|
|
+ qbvconf.admin.gate_states))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (qbvconf.admin.cycle_time)
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_CTRL_CYCLETIME,
|
|
+ qbvconf.admin.cycle_time))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (qbvconf.admin.cycle_time_extension)
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_CTRL_CYCLETIMEEXT,
|
|
+ qbvconf.admin.cycle_time_extension))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (qbvconf.admin.base_time)
|
|
+ if (NLA_PUT_U64(rep_skb, TSN_QBV_ATTR_CTRL_BASETIME,
|
|
+ qbvconf.admin.base_time))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ kfree(qbvconf.admin.control_list);
|
|
+
|
|
+ } else {
|
|
+ pr_info("tsn: error get administrator data.");
|
|
+ }
|
|
+
|
|
+ nla_nest_end(rep_skb, qbvadminattr);
|
|
+
|
|
+ if (qbvconf.gate_enabled) {
|
|
+ if (nla_put_flag(rep_skb, TSN_QBV_ATTR_ENABLE))
|
|
+ return -EMSGSIZE;
|
|
+ } else {
|
|
+ if (nla_put_flag(rep_skb, TSN_QBV_ATTR_DISABLE))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (qbvconf.maxsdu)
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_MAXSDU, qbvconf.maxsdu))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (qbvconf.config_change)
|
|
+ if (nla_put_flag(rep_skb, TSN_QBV_ATTR_CONFIGCHANGE))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ nla_nest_end(rep_skb, qbv);
|
|
+
|
|
+ tsn_send_reply(rep_skb, info);
|
|
+
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int cmd_qbv_status_get(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *qbv, *qbvoperattr;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ int len = 0, i = 0;
|
|
+ struct net_device *netdev;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_qbv_status qbvstatus;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ memset(&qbvstatus, 0, sizeof(struct tsn_qbv_status));
|
|
+
|
|
+ if (!tsnops->qbv_get_status) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qbv_get_status(netdev, &qbvstatus);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd,
|
|
+ &rep_skb, NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ qbv = nla_nest_start_noflag(rep_skb, TSN_ATTR_QBV);
|
|
+ if (!qbv)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ qbvoperattr = nla_nest_start_noflag(rep_skb, TSN_QBV_ATTR_OPERENTRY);
|
|
+ if (!qbvoperattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (qbvstatus.oper.control_list) {
|
|
+ len = qbvstatus.oper.control_list_length;
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_CTRL_LISTCOUNT, len)) {
|
|
+ nla_nest_cancel(rep_skb, qbvoperattr);
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ for (i = 0; i < len; i++) {
|
|
+ struct nlattr *qbv_table;
|
|
+ u8 gs;
|
|
+ u32 tp;
|
|
+ int glisttype = TSN_QBV_ATTR_CTRL_LISTENTRY;
|
|
+
|
|
+ gs = (qbvstatus.oper.control_list + i)->gate_state;
|
|
+ tp = (qbvstatus.oper.control_list + i)->time_interval;
|
|
+
|
|
+ qbv_table = nla_nest_start_noflag(rep_skb, glisttype);
|
|
+ if (!qbv_table)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_ENTRY_ID, i) ||
|
|
+ nla_put_u8(rep_skb, TSN_QBV_ATTR_ENTRY_GC, gs) ||
|
|
+ nla_put_u32(rep_skb, TSN_QBV_ATTR_ENTRY_TM, tp)) {
|
|
+ nla_nest_cancel(rep_skb, qbv_table);
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ nla_nest_end(rep_skb, qbv_table);
|
|
+ }
|
|
+
|
|
+ if (qbvstatus.oper.gate_states) {
|
|
+ if (nla_put_u8(rep_skb, TSN_QBV_ATTR_CTRL_GATESTATE,
|
|
+ qbvstatus.oper.gate_states))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (qbvstatus.oper.cycle_time) {
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_CTRL_CYCLETIME,
|
|
+ qbvstatus.oper.cycle_time))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (qbvstatus.oper.cycle_time_extension) {
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_CTRL_CYCLETIMEEXT,
|
|
+ qbvstatus.oper.cycle_time_extension))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (qbvstatus.oper.base_time) {
|
|
+ if (NLA_PUT_U64(rep_skb, TSN_QBV_ATTR_CTRL_BASETIME,
|
|
+ qbvstatus.oper.base_time))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ kfree(qbvstatus.oper.control_list);
|
|
+ } else {
|
|
+ pr_info("tsn: error get operation list data.");
|
|
+ }
|
|
+
|
|
+ nla_nest_end(rep_skb, qbvoperattr);
|
|
+
|
|
+ if (qbvstatus.config_change_time) {
|
|
+ if (NLA_PUT_U64(rep_skb, TSN_QBV_ATTR_CONFIGCHANGETIME,
|
|
+ qbvstatus.config_change_time))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (qbvstatus.tick_granularity) {
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_GRANULARITY,
|
|
+ qbvstatus.tick_granularity))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (qbvstatus.current_time) {
|
|
+ if (NLA_PUT_U64(rep_skb, TSN_QBV_ATTR_CURRENTTIME,
|
|
+ qbvstatus.current_time))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (qbvstatus.config_pending) {
|
|
+ if (nla_put_flag(rep_skb, TSN_QBV_ATTR_CONFIGPENDING))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (qbvstatus.config_change_error) {
|
|
+ if (NLA_PUT_U64(rep_skb, TSN_QBV_ATTR_CONFIGCHANGEERROR,
|
|
+ qbvstatus.config_change_error))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (qbvstatus.supported_list_max) {
|
|
+ if (nla_put_u32(rep_skb, TSN_QBV_ATTR_LISTMAX,
|
|
+ qbvstatus.supported_list_max))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ nla_nest_end(rep_skb, qbv);
|
|
+
|
|
+ tsn_send_reply(rep_skb, info);
|
|
+
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static int tsn_qbv_status_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME])
|
|
+ cmd_qbv_status_get(info);
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_qbv_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME])
|
|
+ cmd_qbv_get(info);
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_cbs_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ struct nlattr *cbsa[TSN_CBS_ATTR_MAX + 1];
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ int ret;
|
|
+ u8 tc, bw;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_CBS]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_CBS];
|
|
+
|
|
+ if (!tsnops->cbs_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(cbsa, TSN_CBS_ATTR_MAX, na, cbs_policy);
|
|
+ if (ret) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (!cbsa[TSN_CBS_ATTR_TC_INDEX]) {
|
|
+ pr_err("tsn: no TSN_CBS_ATTR_TC_INDEX input\n");
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+ tc = nla_get_u8(cbsa[TSN_CBS_ATTR_TC_INDEX]);
|
|
+
|
|
+ if (!cbsa[TSN_CBS_ATTR_BW]) {
|
|
+ pr_err("tsn: no TSN_CBS_ATTR_BW input\n");
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ bw = nla_get_u8(cbsa[TSN_CBS_ATTR_BW]);
|
|
+ if (bw > 100) {
|
|
+ pr_err("tsn: TSN_CBS_ATTR_BW isn't in the range of 0~100\n");
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->cbs_set(netdev, tc, bw);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, 0);
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_cbs_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *cbsattr;
|
|
+ struct nlattr *cbsa[TSN_CBS_ATTR_MAX + 1];
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ u8 tc;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_CBS]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (!tsnops->cbs_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_CBS];
|
|
+ ret = NLA_PARSE_NESTED(cbsa, TSN_CBS_ATTR_MAX, na, cbs_policy);
|
|
+ if (ret) {
|
|
+ pr_err("tsn: parse value TSN_CBS_ATTR_MAX error.");
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ /* Get status data from device */
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ /* Form netlink reply data */
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd, &rep_skb,
|
|
+ NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ cbsattr = nla_nest_start_noflag(rep_skb, TSN_ATTR_CBS);
|
|
+ if (!cbsattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (!cbsa[TSN_CBS_ATTR_TC_INDEX]) {
|
|
+ pr_err("tsn: must to specify the TSN_CBS_ATTR_TC_INDEX\n");
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+ tc = nla_get_u8(cbsa[TSN_CBS_ATTR_TC_INDEX]);
|
|
+
|
|
+ ret = tsnops->cbs_get(netdev, tc);
|
|
+ if (ret < 0) {
|
|
+ pr_err("tsn: cbs_get return error\n");
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ if (nla_put_u8(rep_skb, TSN_CBS_ATTR_BW, ret & 0XF))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ nla_nest_end(rep_skb, cbsattr);
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+}
|
|
+
|
|
+static int cmd_qbu_set(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ struct nlattr *qbua[TSN_QBU_ATTR_MAX + 1];
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ int ret;
|
|
+ u8 preemptible = 0;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_QBU]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_QBU];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(qbua, TSN_QBU_ATTR_MAX, na, qbu_policy);
|
|
+ if (ret) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (qbua[TSN_QBU_ATTR_ADMIN_STATE])
|
|
+ preemptible = nla_get_u8(qbua[TSN_QBU_ATTR_ADMIN_STATE]);
|
|
+ else
|
|
+ pr_info("No preemptible TSN_QBU_ATTR_ADMIN_STATE config!\n");
|
|
+
|
|
+ if (!tsnops->qbu_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qbu_set(netdev, preemptible);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, 0);
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_qbu_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME])
|
|
+ return cmd_qbu_set(info);
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int cmd_qbu_get_status(struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *qbuattr;
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_preempt_status pps;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ /* Get status data from device */
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ memset(&pps, 0, sizeof(struct tsn_preempt_status));
|
|
+
|
|
+ if (!tsnops->qbu_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->qbu_get(netdev, &pps);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ /* Form netlink reply data */
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd,
|
|
+ &rep_skb, NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ qbuattr = nla_nest_start_noflag(rep_skb, TSN_ATTR_QBU);
|
|
+ if (!qbuattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_u8(rep_skb, TSN_QBU_ATTR_ADMIN_STATE, pps.admin_state) ||
|
|
+ nla_put_u32(rep_skb,
|
|
+ TSN_QBU_ATTR_HOLD_ADVANCE, pps.hold_advance) ||
|
|
+ nla_put_u32(rep_skb,
|
|
+ TSN_QBU_ATTR_RELEASE_ADVANCE, pps.release_advance))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (pps.preemption_active) {
|
|
+ if (nla_put_flag(rep_skb, TSN_QBU_ATTR_ACTIVE))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (nla_put_u8(rep_skb, TSN_QBU_ATTR_HOLD_REQUEST, pps.hold_request))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ nla_nest_end(rep_skb, qbuattr);
|
|
+
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+}
|
|
+
|
|
+static int tsn_qbu_get_status(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ if (info->attrs[TSN_ATTR_IFNAME])
|
|
+ return cmd_qbu_get_status(info);
|
|
+
|
|
+ return -1;
|
|
+}
|
|
+
|
|
+static int tsn_tsd_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ struct nlattr *ntsd[TSN_TSD_ATTR_MAX + 1];
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct tsn_tsd tsd;
|
|
+ int ret;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ memset(&tsd, 0, sizeof(struct tsn_tsd));
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_TSD]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_TSD];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(ntsd, TSN_TSD_ATTR_MAX, na, tsd_policy);
|
|
+ if (ret) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (!tsnops->tsd_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (nla_get_flag(ntsd[TSN_TSD_ATTR_DISABLE])) {
|
|
+ tsd.enable = false;
|
|
+ } else {
|
|
+ if (ntsd[TSN_TSD_ATTR_PERIOD])
|
|
+ tsd.period = nla_get_u32(ntsd[TSN_TSD_ATTR_PERIOD]);
|
|
+
|
|
+ if (!tsd.period) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (ntsd[TSN_TSD_ATTR_MAX_FRM_NUM])
|
|
+ tsd.maxFrameNum =
|
|
+ nla_get_u32(ntsd[TSN_TSD_ATTR_MAX_FRM_NUM]);
|
|
+
|
|
+ if (ntsd[TSN_TSD_ATTR_SYN_IMME])
|
|
+ tsd.syn_flag = 2;
|
|
+ else
|
|
+ tsd.syn_flag = 1;
|
|
+
|
|
+ tsd.enable = true;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->tsd_set(netdev, &tsd);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, 0);
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_tsd_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na, *tsdattr;
|
|
+ struct nlattr *tsda[TSN_TSD_ATTR_MAX + 1];
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_tsd_status tts;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_TSD]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ if (!tsnops->tsd_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = tsnops->tsd_get(netdev, &tts);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_TSD];
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(tsda, TSN_TSD_ATTR_MAX,
|
|
+ na, tsd_policy);
|
|
+ if (ret) {
|
|
+ pr_err("tsn: parse value TSN_TSD_ATTR_MAX error.");
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ /* Get status data from device */
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ /* Form netlink reply data */
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd, &rep_skb,
|
|
+ NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ tsdattr = nla_nest_start_noflag(rep_skb, TSN_ATTR_TSD);
|
|
+ if (!tsdattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_u32(rep_skb, TSN_TSD_ATTR_PERIOD, tts.period) ||
|
|
+ nla_put_u32(rep_skb, TSN_TSD_ATTR_MAX_FRM_NUM, tts.maxFrameNum) ||
|
|
+ nla_put_u32(rep_skb, TSN_TSD_ATTR_CYCLE_NUM, tts.cycleNum) ||
|
|
+ nla_put_u32(rep_skb, TSN_TSD_ATTR_LOSS_STEPS, tts.loss_steps) ||
|
|
+ nla_put_u32(rep_skb, TSN_TSD_ATTR_MAX_FRM_NUM, tts.maxFrameNum))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (!tts.enable) {
|
|
+ if (nla_put_flag(rep_skb, TSN_TSD_ATTR_DISABLE))
|
|
+ return -EMSGSIZE;
|
|
+ } else {
|
|
+ if (nla_put_flag(rep_skb, TSN_TSD_ATTR_ENABLE))
|
|
+ return -EMSGSIZE;
|
|
+ }
|
|
+
|
|
+ if (tts.flag == 2)
|
|
+ if (nla_put_flag(rep_skb, TSN_TSD_ATTR_SYN_IMME))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ nla_nest_end(rep_skb, tsdattr);
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+}
|
|
+
|
|
+static int tsn_ct_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ struct nlattr *cta[TSN_CT_ATTR_MAX + 1];
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ int ret;
|
|
+ u8 queue_stat;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_CT]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_CT];
|
|
+
|
|
+ if (!tsnops->ct_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(cta, TSN_CT_ATTR_MAX,
|
|
+ na, ct_policy);
|
|
+ if (ret) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ queue_stat = nla_get_u8(cta[TSN_CT_ATTR_QUEUE_STATE]);
|
|
+
|
|
+ ret = tsnops->ct_set(netdev, queue_stat);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, 0);
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_cbgen_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ struct nlattr *cbgena[TSN_CBGEN_ATTR_MAX + 1];
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ int ret;
|
|
+ u32 index;
|
|
+ struct tsn_seq_gen_conf sg_conf;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_CBGEN]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_CBGEN];
|
|
+
|
|
+ if (!tsnops->cbgen_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(cbgena, TSN_CBGEN_ATTR_MAX,
|
|
+ na, cbgen_policy);
|
|
+ if (ret) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ index = nla_get_u32(cbgena[TSN_CBGEN_ATTR_INDEX]);
|
|
+
|
|
+ memset(&sg_conf, 0, sizeof(struct tsn_seq_gen_conf));
|
|
+ sg_conf.iport_mask = nla_get_u8(cbgena[TSN_CBGEN_ATTR_PORT_MASK]);
|
|
+ sg_conf.split_mask = nla_get_u8(cbgena[TSN_CBGEN_ATTR_SPLIT_MASK]);
|
|
+ sg_conf.seq_len = nla_get_u8(cbgena[TSN_CBGEN_ATTR_SEQ_LEN]);
|
|
+ sg_conf.seq_num = nla_get_u32(cbgena[TSN_CBGEN_ATTR_SEQ_NUM]);
|
|
+
|
|
+ ret = tsnops->cbgen_set(netdev, index, &sg_conf);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, 0);
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_cbrec_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ struct nlattr *cbreca[TSN_CBREC_ATTR_MAX + 1];
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ int ret;
|
|
+ u32 index;
|
|
+ struct tsn_seq_rec_conf sr_conf;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_CBREC]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_CBREC];
|
|
+
|
|
+ if (!tsnops->cbrec_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(cbreca, TSN_CBREC_ATTR_MAX,
|
|
+ na, cbrec_policy);
|
|
+ if (ret) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ index = nla_get_u32(cbreca[TSN_CBREC_ATTR_INDEX]);
|
|
+
|
|
+ memset(&sr_conf, 0, sizeof(struct tsn_seq_rec_conf));
|
|
+ sr_conf.seq_len = nla_get_u8(cbreca[TSN_CBREC_ATTR_SEQ_LEN]);
|
|
+ sr_conf.his_len = nla_get_u8(cbreca[TSN_CBREC_ATTR_HIS_LEN]);
|
|
+ sr_conf.rtag_pop_en = nla_get_flag(cbreca[TSN_CBREC_ATTR_TAG_POP_EN]);
|
|
+
|
|
+ ret = tsnops->cbrec_set(netdev, index, &sr_conf);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, 0);
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int tsn_cbstatus_get(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ struct nlattr *cba[TSN_CBSTAT_ATTR_MAX + 1];
|
|
+ struct nlattr *cbattr;
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ struct sk_buff *rep_skb;
|
|
+ int ret;
|
|
+ unsigned int index;
|
|
+ struct genlmsghdr *genlhdr;
|
|
+ struct tsn_cb_status cbstat;
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ /* Get status data from device */
|
|
+ genlhdr = info->genlhdr;
|
|
+
|
|
+ memset(&cbstat, 0, sizeof(struct tsn_cb_status));
|
|
+
|
|
+ if (!tsnops->cb_get) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_CBSTAT];
|
|
+ ret = NLA_PARSE_NESTED(cba, TSN_CBSTAT_ATTR_MAX,
|
|
+ na, cbstat_policy);
|
|
+ if (ret) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ index = nla_get_u32(cba[TSN_CBSTAT_ATTR_INDEX]);
|
|
+
|
|
+ ret = tsnops->cb_get(netdev, index, &cbstat);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ /* Form netlink reply data */
|
|
+ ret = tsn_prepare_reply(info, genlhdr->cmd,
|
|
+ &rep_skb, NLMSG_ALIGN(MAX_ATTR_SIZE));
|
|
+ if (ret < 0)
|
|
+ return ret;
|
|
+
|
|
+ if (nla_put_string(rep_skb, TSN_ATTR_IFNAME, netdev->name))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ cbattr = nla_nest_start_noflag(rep_skb, TSN_ATTR_CBSTAT);
|
|
+ if (!cbattr)
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ if (nla_put_u8(rep_skb, TSN_CBSTAT_ATTR_GEN_REC, cbstat.gen_rec) ||
|
|
+ nla_put_u8(rep_skb, TSN_CBSTAT_ATTR_ERR, cbstat.err) ||
|
|
+ nla_put_u32(rep_skb, TSN_CBSTAT_ATTR_SEQ_NUM,
|
|
+ cbstat.seq_num) ||
|
|
+ nla_put_u8(rep_skb, TSN_CBSTAT_ATTR_SEQ_LEN, cbstat.seq_len) ||
|
|
+ nla_put_u8(rep_skb, TSN_CBSTAT_ATTR_SPLIT_MASK,
|
|
+ cbstat.split_mask) ||
|
|
+ nla_put_u8(rep_skb, TSN_CBSTAT_ATTR_PORT_MASK,
|
|
+ cbstat.iport_mask) ||
|
|
+ nla_put_u8(rep_skb, TSN_CBSTAT_ATTR_HIS_LEN, cbstat.his_len) ||
|
|
+ nla_put_u32(rep_skb, TSN_CBSTAT_ATTR_SEQ_HIS,
|
|
+ cbstat.seq_his))
|
|
+ return -EMSGSIZE;
|
|
+
|
|
+ nla_nest_end(rep_skb, cbattr);
|
|
+
|
|
+ return tsn_send_reply(rep_skb, info);
|
|
+}
|
|
+
|
|
+static int tsn_dscp_set(struct sk_buff *skb, struct genl_info *info)
|
|
+{
|
|
+ struct nlattr *na;
|
|
+ struct nlattr *dscpa[TSN_DSCP_ATTR_MAX + 1];
|
|
+ struct net_device *netdev;
|
|
+ const struct tsn_ops *tsnops;
|
|
+ int ret;
|
|
+ bool enable = 0;
|
|
+ struct tsn_port *port;
|
|
+ int dscp_ix;
|
|
+ struct tsn_qos_switch_dscp_conf dscp_conf;
|
|
+
|
|
+ port = tsn_init_check(info, &netdev);
|
|
+ if (!port)
|
|
+ return -ENODEV;
|
|
+
|
|
+ tsnops = port->tsnops;
|
|
+
|
|
+ if (!info->attrs[TSN_ATTR_DSCP]) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ na = info->attrs[TSN_ATTR_DSCP];
|
|
+
|
|
+ if (!tsnops->dscp_set) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EPERM);
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ ret = NLA_PARSE_NESTED(dscpa, TSN_DSCP_ATTR_MAX,
|
|
+ na, dscp_policy);
|
|
+ if (ret) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, -EINVAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ enable = 1;
|
|
+ if (dscpa[TSN_DSCP_ATTR_DISABLE])
|
|
+ enable = 0;
|
|
+ dscp_ix = nla_get_u32(dscpa[TSN_DSCP_ATTR_INDEX]);
|
|
+ dscp_conf.cos = nla_get_u32(dscpa[TSN_DSCP_ATTR_COS]);
|
|
+ dscp_conf.dpl = nla_get_u32(dscpa[TSN_DSCP_ATTR_DPL]);
|
|
+ ret = tsnops->dscp_set(netdev, enable, dscp_ix, &dscp_conf);
|
|
+ if (ret < 0) {
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, ret);
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ tsn_simple_reply(info, TSN_CMD_REPLY,
|
|
+ netdev->name, 0);
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static const struct genl_ops tsnnl_ops[] = {
|
|
+ {
|
|
+ .cmd = TSN_CMD_ECHO,
|
|
+ .doit = tsn_echo_cmd,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_CAP_GET,
|
|
+ .doit = tsn_cap_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QBV_SET,
|
|
+ .doit = tsn_qbv_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QBV_GET,
|
|
+ .doit = tsn_qbv_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QBV_GET_STATUS,
|
|
+ .doit = tsn_qbv_status_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_CB_STREAMID_SET,
|
|
+ .doit = tsn_cb_streamid_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_CB_STREAMID_GET,
|
|
+ .doit = tsn_cb_streamid_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_CB_STREAMID_GET_COUNTS,
|
|
+ .doit = tsn_cb_streamid_counters_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QCI_CAP_GET,
|
|
+ .doit = tsn_qci_cap_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QCI_SFI_SET,
|
|
+ .doit = tsn_qci_sfi_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QCI_SFI_GET,
|
|
+ .doit = tsn_qci_sfi_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QCI_SFI_GET_COUNTS,
|
|
+ .doit = tsn_qci_sfi_counters_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QCI_SGI_SET,
|
|
+ .doit = tsn_qci_sgi_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QCI_SGI_GET,
|
|
+ .doit = tsn_qci_sgi_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QCI_SGI_GET_STATUS,
|
|
+ .doit = tsn_qci_sgi_status_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QCI_FMI_SET,
|
|
+ .doit = tsn_qci_fmi_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QCI_FMI_GET,
|
|
+ .doit = tsn_qci_fmi_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_CBS_SET,
|
|
+ .doit = tsn_cbs_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_CBS_GET,
|
|
+ .doit = tsn_cbs_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QBU_SET,
|
|
+ .doit = tsn_qbu_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_QBU_GET_STATUS,
|
|
+ .doit = tsn_qbu_get_status,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_TSD_SET,
|
|
+ .doit = tsn_tsd_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_TSD_GET,
|
|
+ .doit = tsn_tsd_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_CT_SET,
|
|
+ .doit = tsn_ct_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_CBGEN_SET,
|
|
+ .doit = tsn_cbgen_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_CBREC_SET,
|
|
+ .doit = tsn_cbrec_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_CBSTAT_GET,
|
|
+ .doit = tsn_cbstatus_get,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+ {
|
|
+ .cmd = TSN_CMD_DSCP_SET,
|
|
+ .doit = tsn_dscp_set,
|
|
+ .flags = GENL_ADMIN_PERM,
|
|
+ },
|
|
+};
|
|
+
|
|
+static const struct genl_multicast_group tsn_mcgrps[] = {
|
|
+ [TSN_MCGRP_QBV] = { .name = TSN_MULTICAST_GROUP_QBV},
|
|
+ [TSN_MCGRP_QCI] = { .name = TSN_MULTICAST_GROUP_QCI},
|
|
+};
|
|
+
|
|
+static struct genl_family tsn_family = {
|
|
+ .name = TSN_GENL_NAME,
|
|
+ .version = TSN_GENL_VERSION,
|
|
+ .maxattr = TSN_CMD_ATTR_MAX,
|
|
+ .module = THIS_MODULE,
|
|
+ .netnsok = true,
|
|
+ .ops = tsnnl_ops,
|
|
+ .n_ops = ARRAY_SIZE(tsnnl_ops),
|
|
+ .mcgrps = tsn_mcgrps,
|
|
+ .n_mcgrps = ARRAY_SIZE(tsn_mcgrps),
|
|
+};
|
|
+
|
|
+int tsn_port_register(struct net_device *netdev,
|
|
+ struct tsn_ops *tsnops, u16 groupid)
|
|
+{
|
|
+ struct tsn_port *port;
|
|
+
|
|
+ if (list_empty(&port_list)) {
|
|
+ INIT_LIST_HEAD(&port_list);
|
|
+ } else {
|
|
+ list_for_each_entry(port, &port_list, list) {
|
|
+ if (port->netdev == netdev) {
|
|
+ pr_info("TSN device already registered!\n");
|
|
+ return -1;
|
|
+ }
|
|
+ }
|
|
+ }
|
|
+
|
|
+ port = kzalloc(sizeof(*port), GFP_KERNEL);
|
|
+ if (!port)
|
|
+ return -1;
|
|
+
|
|
+ port->netdev = netdev;
|
|
+ port->groupid = groupid;
|
|
+ port->tsnops = tsnops;
|
|
+ port->nd.dev = netdev;
|
|
+
|
|
+ if (groupid < GROUP_OFFSET_SWITCH)
|
|
+ port->type = TSN_ENDPOINT;
|
|
+ else
|
|
+ port->type = TSN_SWITCH;
|
|
+
|
|
+ list_add_tail(&port->list, &port_list);
|
|
+
|
|
+ if (tsnops && tsnops->device_init)
|
|
+ port->tsnops->device_init(netdev);
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+EXPORT_SYMBOL(tsn_port_register);
|
|
+
|
|
+void tsn_port_unregister(struct net_device *netdev)
|
|
+{
|
|
+ struct tsn_port *p;
|
|
+
|
|
+ list_for_each_entry(p, &port_list, list) {
|
|
+ if (!p || !p->netdev)
|
|
+ continue;
|
|
+ if (p->netdev == netdev) {
|
|
+ if (p->tsnops->device_deinit)
|
|
+ p->tsnops->device_deinit(netdev);
|
|
+ list_del(&p->list);
|
|
+ kfree(p);
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+}
|
|
+EXPORT_SYMBOL(tsn_port_unregister);
|
|
+
|
|
+static int tsn_multicast_to_user(unsigned long event,
|
|
+ struct tsn_notifier_info *tsn_info)
|
|
+{
|
|
+ struct sk_buff *skb;
|
|
+ struct genlmsghdr *nlh;
|
|
+ int res;
|
|
+ struct tsn_qbv_conf *qbvdata;
|
|
+
|
|
+ /* If new attributes are added, please revisit this allocation */
|
|
+ skb = genlmsg_new(sizeof(*tsn_info), GFP_KERNEL);
|
|
+ if (!skb) {
|
|
+ pr_err("Allocation failure.\n");
|
|
+ return -ENOMEM;
|
|
+ }
|
|
+
|
|
+ switch (event) {
|
|
+ case TSN_QBV_CONFIGCHANGETIME_ARRIVE:
|
|
+ nlh = genlmsg_put(skb, 0, 1, &tsn_family,
|
|
+ GFP_KERNEL, TSN_CMD_QBV_SET);
|
|
+ qbvdata = &tsn_info->ntdata.qbv_notify;
|
|
+ res = NLA_PUT_U64(skb, TSN_QBV_ATTR_CTRL_BASETIME,
|
|
+ qbvdata->admin.base_time);
|
|
+
|
|
+ if (res) {
|
|
+ pr_err("put data failure!\n");
|
|
+ goto done;
|
|
+ }
|
|
+
|
|
+ res = nla_put_u32(skb, TSN_QBV_ATTR_CTRL_CYCLETIME,
|
|
+ qbvdata->admin.cycle_time);
|
|
+ if (res) {
|
|
+ pr_err("put data failure!\n");
|
|
+ goto done;
|
|
+ }
|
|
+
|
|
+ if (qbvdata->gate_enabled)
|
|
+ res = nla_put_flag(skb, TSN_QBV_ATTR_ENABLE +
|
|
+ TSN_QBV_ATTR_CTRL_MAX);
|
|
+ else
|
|
+ res = nla_put_flag(skb, TSN_QBV_ATTR_DISABLE +
|
|
+ TSN_QBV_ATTR_CTRL_MAX);
|
|
+ if (res) {
|
|
+ pr_err("put data failure!\n");
|
|
+ goto done;
|
|
+ }
|
|
+
|
|
+ res = nla_put_u32(skb, TSN_QBV_ATTR_CTRL_UNSPEC,
|
|
+ tsn_info->dev->ifindex);
|
|
+ if (res) {
|
|
+ pr_err("put data failure!\n");
|
|
+ goto done;
|
|
+ }
|
|
+
|
|
+ break;
|
|
+ default:
|
|
+ pr_info("event not supportted!\n");
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ (void)genlmsg_end(skb, nlh);
|
|
+
|
|
+ res = genlmsg_multicast_allns(&tsn_family, skb, 0,
|
|
+ TSN_MCGRP_QBV, GFP_KERNEL);
|
|
+ skb = NULL;
|
|
+ if (res && res != -ESRCH) {
|
|
+ pr_err("genlmsg_multicast_allns error: %d\n", res);
|
|
+ goto done;
|
|
+ }
|
|
+
|
|
+ if (res == -ESRCH)
|
|
+ res = 0;
|
|
+
|
|
+done:
|
|
+ if (skb) {
|
|
+ nlmsg_free(skb);
|
|
+ skb = NULL;
|
|
+ }
|
|
+
|
|
+ return res;
|
|
+}
|
|
+
|
|
+/* called with RTNL or RCU */
|
|
+static int tsn_event(struct notifier_block *unused,
|
|
+ unsigned long event, void *ptr)
|
|
+{
|
|
+ struct tsn_notifier_info *tsn_info;
|
|
+ int err = NOTIFY_DONE;
|
|
+
|
|
+ switch (event) {
|
|
+ case TSN_QBV_CONFIGCHANGETIME_ARRIVE:
|
|
+ tsn_info = ptr;
|
|
+ err = tsn_multicast_to_user(event, tsn_info);
|
|
+ if (err) {
|
|
+ err = notifier_from_errno(err);
|
|
+ break;
|
|
+ }
|
|
+ break;
|
|
+ default:
|
|
+ pr_info("event not supportted!\n");
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ return err;
|
|
+}
|
|
+
|
|
+static struct notifier_block tsn_notifier = {
|
|
+ .notifier_call = tsn_event,
|
|
+};
|
|
+
|
|
+static int __init tsn_genetlink_init(void)
|
|
+{
|
|
+ int ret;
|
|
+
|
|
+ pr_info("tsn generic netlink module v%d init...\n", TSN_GENL_VERSION);
|
|
+
|
|
+ ret = genl_register_family(&tsn_family);
|
|
+
|
|
+ if (ret != 0) {
|
|
+ pr_info("failed to init tsn generic netlink example module\n");
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ register_tsn_notifier(&tsn_notifier);
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static void __exit tsn_genetlink_exit(void)
|
|
+{
|
|
+ int ret;
|
|
+
|
|
+ ret = genl_unregister_family(&tsn_family);
|
|
+ if (ret != 0)
|
|
+ pr_info("failed to unregister family:%i\n", ret);
|
|
+
|
|
+ unregister_tsn_notifier(&tsn_notifier);
|
|
+}
|
|
+
|
|
+module_init(tsn_genetlink_init);
|
|
+module_exit(tsn_genetlink_exit);
|
|
+MODULE_LICENSE("GPL");
|