mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2024-12-26 23:25:23 +07:00
e0456717e4
Pull networking updates from David Miller: 1) Add TX fast path in mac80211, from Johannes Berg. 2) Add TSO/GRO support to ibmveth, from Thomas Falcon 3) Move away from cached routes in ipv6, just like ipv4, from Martin KaFai Lau. 4) Lots of new rhashtable tests, from Thomas Graf. 5) Run ingress qdisc lockless, from Alexei Starovoitov. 6) Allow servers to fetch TCP packet headers for SYN packets of new connections, for fingerprinting. From Eric Dumazet. 7) Add mode parameter to pktgen, for testing receive. From Alexei Starovoitov. 8) Cache access optimizations via simplifications of build_skb(), from Alexander Duyck. 9) Move page frag allocator under mm/, also from Alexander. 10) Add xmit_more support to hv_netvsc, from KY Srinivasan. 11) Add a counter guard in case we try to perform endless reclassify loops in the packet scheduler. 12) Extern flow dissector to be programmable and use it in new "Flower" classifier. From Jiri Pirko. 13) AF_PACKET fanout rollover fixes, performance improvements, and new statistics. From Willem de Bruijn. 14) Add netdev driver for GENEVE tunnels, from John W Linville. 15) Add ingress netfilter hooks and filtering, from Pablo Neira Ayuso. 16) Fix handling of epoll edge triggers in TCP, from Eric Dumazet. 17) Add an ECN retry fallback for the initial TCP handshake, from Daniel Borkmann. 18) Add tail call support to BPF, from Alexei Starovoitov. 19) Add several pktgen helper scripts, from Jesper Dangaard Brouer. 20) Add zerocopy support to AF_UNIX, from Hannes Frederic Sowa. 21) Favor even port numbers for allocation to connect() requests, and odd port numbers for bind(0), in an effort to help avoid ip_local_port_range exhaustion. From Eric Dumazet. 22) Add Cavium ThunderX driver, from Sunil Goutham. 23) Allow bpf programs to access skb_iif and dev->ifindex SKB metadata, from Alexei Starovoitov. 24) Add support for T6 chips in cxgb4vf driver, from Hariprasad Shenai. 25) Double TCP Small Queues default to 256K to accomodate situations like the XEN driver and wireless aggregation. From Wei Liu. 26) Add more entropy inputs to flow dissector, from Tom Herbert. 27) Add CDG congestion control algorithm to TCP, from Kenneth Klette Jonassen. 28) Convert ipset over to RCU locking, from Jozsef Kadlecsik. 29) Track and act upon link status of ipv4 route nexthops, from Andy Gospodarek. * git://git.kernel.org/pub/scm/linux/kernel/git/davem/net-next: (1670 commits) bridge: vlan: flush the dynamically learned entries on port vlan delete bridge: multicast: add a comment to br_port_state_selection about blocking state net: inet_diag: export IPV6_V6ONLY sockopt stmmac: troubleshoot unexpected bits in des0 & des1 net: ipv4 sysctl option to ignore routes when nexthop link is down net: track link-status of ipv4 nexthops net: switchdev: ignore unsupported bridge flags net: Cavium: Fix MAC address setting in shutdown state drivers: net: xgene: fix for ACPI support without ACPI ip: report the original address of ICMP messages net/mlx5e: Prefetch skb data on RX net/mlx5e: Pop cq outside mlx5e_get_cqe net/mlx5e: Remove mlx5e_cq.sqrq back-pointer net/mlx5e: Remove extra spaces net/mlx5e: Avoid TX CQE generation if more xmit packets expected net/mlx5e: Avoid redundant dev_kfree_skb() upon NOP completion net/mlx5e: Remove re-assignment of wq type in mlx5e_enable_rq() net/mlx5e: Use skb_shinfo(skb)->gso_segs rather than counting them net/mlx5e: Static mapping of netdev priv resources to/from netdev TX queues net/mlx4_en: Use HW counters for rx/tx bytes/packets in PF device ...
444 lines
12 KiB
C
444 lines
12 KiB
C
/*
|
|
* Copyright (c) 2013-2015, Mellanox Technologies. All rights reserved.
|
|
*
|
|
* This software is available to you under a choice of one of two
|
|
* licenses. You may choose to be licensed under the terms of the GNU
|
|
* General Public License (GPL) Version 2, available from the file
|
|
* COPYING in the main directory of this source tree, or the
|
|
* OpenIB.org BSD license below:
|
|
*
|
|
* Redistribution and use in source and binary forms, with or
|
|
* without modification, are permitted provided that the following
|
|
* conditions are met:
|
|
*
|
|
* - Redistributions of source code must retain the above
|
|
* copyright notice, this list of conditions and the following
|
|
* disclaimer.
|
|
*
|
|
* - Redistributions in binary form must reproduce the above
|
|
* copyright notice, this list of conditions and the following
|
|
* disclaimer in the documentation and/or other materials
|
|
* provided with the distribution.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
* SOFTWARE.
|
|
*/
|
|
|
|
#include <linux/mlx5/cmd.h>
|
|
#include <rdma/ib_mad.h>
|
|
#include <rdma/ib_smi.h>
|
|
#include "mlx5_ib.h"
|
|
|
|
enum {
|
|
MLX5_IB_VENDOR_CLASS1 = 0x9,
|
|
MLX5_IB_VENDOR_CLASS2 = 0xa
|
|
};
|
|
|
|
int mlx5_MAD_IFC(struct mlx5_ib_dev *dev, int ignore_mkey, int ignore_bkey,
|
|
u8 port, const struct ib_wc *in_wc, const struct ib_grh *in_grh,
|
|
const void *in_mad, void *response_mad)
|
|
{
|
|
u8 op_modifier = 0;
|
|
|
|
/* Key check traps can't be generated unless we have in_wc to
|
|
* tell us where to send the trap.
|
|
*/
|
|
if (ignore_mkey || !in_wc)
|
|
op_modifier |= 0x1;
|
|
if (ignore_bkey || !in_wc)
|
|
op_modifier |= 0x2;
|
|
|
|
return mlx5_core_mad_ifc(dev->mdev, in_mad, response_mad, op_modifier, port);
|
|
}
|
|
|
|
int mlx5_ib_process_mad(struct ib_device *ibdev, int mad_flags, u8 port_num,
|
|
const struct ib_wc *in_wc, const struct ib_grh *in_grh,
|
|
const struct ib_mad_hdr *in, size_t in_mad_size,
|
|
struct ib_mad_hdr *out, size_t *out_mad_size,
|
|
u16 *out_mad_pkey_index)
|
|
{
|
|
u16 slid;
|
|
int err;
|
|
const struct ib_mad *in_mad = (const struct ib_mad *)in;
|
|
struct ib_mad *out_mad = (struct ib_mad *)out;
|
|
|
|
BUG_ON(in_mad_size != sizeof(*in_mad) ||
|
|
*out_mad_size != sizeof(*out_mad));
|
|
|
|
slid = in_wc ? in_wc->slid : be16_to_cpu(IB_LID_PERMISSIVE);
|
|
|
|
if (in_mad->mad_hdr.method == IB_MGMT_METHOD_TRAP && slid == 0)
|
|
return IB_MAD_RESULT_SUCCESS | IB_MAD_RESULT_CONSUMED;
|
|
|
|
if (in_mad->mad_hdr.mgmt_class == IB_MGMT_CLASS_SUBN_LID_ROUTED ||
|
|
in_mad->mad_hdr.mgmt_class == IB_MGMT_CLASS_SUBN_DIRECTED_ROUTE) {
|
|
if (in_mad->mad_hdr.method != IB_MGMT_METHOD_GET &&
|
|
in_mad->mad_hdr.method != IB_MGMT_METHOD_SET &&
|
|
in_mad->mad_hdr.method != IB_MGMT_METHOD_TRAP_REPRESS)
|
|
return IB_MAD_RESULT_SUCCESS;
|
|
|
|
/* Don't process SMInfo queries -- the SMA can't handle them.
|
|
*/
|
|
if (in_mad->mad_hdr.attr_id == IB_SMP_ATTR_SM_INFO)
|
|
return IB_MAD_RESULT_SUCCESS;
|
|
} else if (in_mad->mad_hdr.mgmt_class == IB_MGMT_CLASS_PERF_MGMT ||
|
|
in_mad->mad_hdr.mgmt_class == MLX5_IB_VENDOR_CLASS1 ||
|
|
in_mad->mad_hdr.mgmt_class == MLX5_IB_VENDOR_CLASS2 ||
|
|
in_mad->mad_hdr.mgmt_class == IB_MGMT_CLASS_CONG_MGMT) {
|
|
if (in_mad->mad_hdr.method != IB_MGMT_METHOD_GET &&
|
|
in_mad->mad_hdr.method != IB_MGMT_METHOD_SET)
|
|
return IB_MAD_RESULT_SUCCESS;
|
|
} else {
|
|
return IB_MAD_RESULT_SUCCESS;
|
|
}
|
|
|
|
err = mlx5_MAD_IFC(to_mdev(ibdev),
|
|
mad_flags & IB_MAD_IGNORE_MKEY,
|
|
mad_flags & IB_MAD_IGNORE_BKEY,
|
|
port_num, in_wc, in_grh, in_mad, out_mad);
|
|
if (err)
|
|
return IB_MAD_RESULT_FAILURE;
|
|
|
|
/* set return bit in status of directed route responses */
|
|
if (in_mad->mad_hdr.mgmt_class == IB_MGMT_CLASS_SUBN_DIRECTED_ROUTE)
|
|
out_mad->mad_hdr.status |= cpu_to_be16(1 << 15);
|
|
|
|
if (in_mad->mad_hdr.method == IB_MGMT_METHOD_TRAP_REPRESS)
|
|
/* no response for trap repress */
|
|
return IB_MAD_RESULT_SUCCESS | IB_MAD_RESULT_CONSUMED;
|
|
|
|
return IB_MAD_RESULT_SUCCESS | IB_MAD_RESULT_REPLY;
|
|
}
|
|
|
|
int mlx5_query_ext_port_caps(struct mlx5_ib_dev *dev, u8 port)
|
|
{
|
|
struct ib_smp *in_mad = NULL;
|
|
struct ib_smp *out_mad = NULL;
|
|
int err = -ENOMEM;
|
|
u16 packet_error;
|
|
|
|
in_mad = kzalloc(sizeof(*in_mad), GFP_KERNEL);
|
|
out_mad = kmalloc(sizeof(*out_mad), GFP_KERNEL);
|
|
if (!in_mad || !out_mad)
|
|
goto out;
|
|
|
|
init_query_mad(in_mad);
|
|
in_mad->attr_id = MLX5_ATTR_EXTENDED_PORT_INFO;
|
|
in_mad->attr_mod = cpu_to_be32(port);
|
|
|
|
err = mlx5_MAD_IFC(dev, 1, 1, 1, NULL, NULL, in_mad, out_mad);
|
|
|
|
packet_error = be16_to_cpu(out_mad->status);
|
|
|
|
dev->mdev->port_caps[port - 1].ext_port_cap = (!err && !packet_error) ?
|
|
MLX_EXT_PORT_CAP_FLAG_EXTENDED_PORT_INFO : 0;
|
|
|
|
out:
|
|
kfree(in_mad);
|
|
kfree(out_mad);
|
|
return err;
|
|
}
|
|
|
|
int mlx5_query_mad_ifc_smp_attr_node_info(struct ib_device *ibdev,
|
|
struct ib_smp *out_mad)
|
|
{
|
|
struct ib_smp *in_mad = NULL;
|
|
int err = -ENOMEM;
|
|
|
|
in_mad = kzalloc(sizeof(*in_mad), GFP_KERNEL);
|
|
if (!in_mad)
|
|
return -ENOMEM;
|
|
|
|
init_query_mad(in_mad);
|
|
in_mad->attr_id = IB_SMP_ATTR_NODE_INFO;
|
|
|
|
err = mlx5_MAD_IFC(to_mdev(ibdev), 1, 1, 1, NULL, NULL, in_mad,
|
|
out_mad);
|
|
|
|
kfree(in_mad);
|
|
return err;
|
|
}
|
|
|
|
int mlx5_query_mad_ifc_system_image_guid(struct ib_device *ibdev,
|
|
__be64 *sys_image_guid)
|
|
{
|
|
struct ib_smp *out_mad = NULL;
|
|
int err = -ENOMEM;
|
|
|
|
out_mad = kmalloc(sizeof(*out_mad), GFP_KERNEL);
|
|
if (!out_mad)
|
|
return -ENOMEM;
|
|
|
|
err = mlx5_query_mad_ifc_smp_attr_node_info(ibdev, out_mad);
|
|
if (err)
|
|
goto out;
|
|
|
|
memcpy(sys_image_guid, out_mad->data + 4, 8);
|
|
|
|
out:
|
|
kfree(out_mad);
|
|
|
|
return err;
|
|
}
|
|
|
|
int mlx5_query_mad_ifc_max_pkeys(struct ib_device *ibdev,
|
|
u16 *max_pkeys)
|
|
{
|
|
struct ib_smp *out_mad = NULL;
|
|
int err = -ENOMEM;
|
|
|
|
out_mad = kmalloc(sizeof(*out_mad), GFP_KERNEL);
|
|
if (!out_mad)
|
|
return -ENOMEM;
|
|
|
|
err = mlx5_query_mad_ifc_smp_attr_node_info(ibdev, out_mad);
|
|
if (err)
|
|
goto out;
|
|
|
|
*max_pkeys = be16_to_cpup((__be16 *)(out_mad->data + 28));
|
|
|
|
out:
|
|
kfree(out_mad);
|
|
|
|
return err;
|
|
}
|
|
|
|
int mlx5_query_mad_ifc_vendor_id(struct ib_device *ibdev,
|
|
u32 *vendor_id)
|
|
{
|
|
struct ib_smp *out_mad = NULL;
|
|
int err = -ENOMEM;
|
|
|
|
out_mad = kmalloc(sizeof(*out_mad), GFP_KERNEL);
|
|
if (!out_mad)
|
|
return -ENOMEM;
|
|
|
|
err = mlx5_query_mad_ifc_smp_attr_node_info(ibdev, out_mad);
|
|
if (err)
|
|
goto out;
|
|
|
|
*vendor_id = be32_to_cpup((__be32 *)(out_mad->data + 36)) & 0xffff;
|
|
|
|
out:
|
|
kfree(out_mad);
|
|
|
|
return err;
|
|
}
|
|
|
|
int mlx5_query_mad_ifc_node_desc(struct mlx5_ib_dev *dev, char *node_desc)
|
|
{
|
|
struct ib_smp *in_mad = NULL;
|
|
struct ib_smp *out_mad = NULL;
|
|
int err = -ENOMEM;
|
|
|
|
in_mad = kzalloc(sizeof(*in_mad), GFP_KERNEL);
|
|
out_mad = kmalloc(sizeof(*out_mad), GFP_KERNEL);
|
|
if (!in_mad || !out_mad)
|
|
goto out;
|
|
|
|
init_query_mad(in_mad);
|
|
in_mad->attr_id = IB_SMP_ATTR_NODE_DESC;
|
|
|
|
err = mlx5_MAD_IFC(dev, 1, 1, 1, NULL, NULL, in_mad, out_mad);
|
|
if (err)
|
|
goto out;
|
|
|
|
memcpy(node_desc, out_mad->data, 64);
|
|
out:
|
|
kfree(in_mad);
|
|
kfree(out_mad);
|
|
return err;
|
|
}
|
|
|
|
int mlx5_query_mad_ifc_node_guid(struct mlx5_ib_dev *dev, __be64 *node_guid)
|
|
{
|
|
struct ib_smp *in_mad = NULL;
|
|
struct ib_smp *out_mad = NULL;
|
|
int err = -ENOMEM;
|
|
|
|
in_mad = kzalloc(sizeof(*in_mad), GFP_KERNEL);
|
|
out_mad = kmalloc(sizeof(*out_mad), GFP_KERNEL);
|
|
if (!in_mad || !out_mad)
|
|
goto out;
|
|
|
|
init_query_mad(in_mad);
|
|
in_mad->attr_id = IB_SMP_ATTR_NODE_INFO;
|
|
|
|
err = mlx5_MAD_IFC(dev, 1, 1, 1, NULL, NULL, in_mad, out_mad);
|
|
if (err)
|
|
goto out;
|
|
|
|
memcpy(node_guid, out_mad->data + 12, 8);
|
|
out:
|
|
kfree(in_mad);
|
|
kfree(out_mad);
|
|
return err;
|
|
}
|
|
|
|
int mlx5_query_mad_ifc_pkey(struct ib_device *ibdev, u8 port, u16 index,
|
|
u16 *pkey)
|
|
{
|
|
struct ib_smp *in_mad = NULL;
|
|
struct ib_smp *out_mad = NULL;
|
|
int err = -ENOMEM;
|
|
|
|
in_mad = kzalloc(sizeof(*in_mad), GFP_KERNEL);
|
|
out_mad = kmalloc(sizeof(*out_mad), GFP_KERNEL);
|
|
if (!in_mad || !out_mad)
|
|
goto out;
|
|
|
|
init_query_mad(in_mad);
|
|
in_mad->attr_id = IB_SMP_ATTR_PKEY_TABLE;
|
|
in_mad->attr_mod = cpu_to_be32(index / 32);
|
|
|
|
err = mlx5_MAD_IFC(to_mdev(ibdev), 1, 1, port, NULL, NULL, in_mad,
|
|
out_mad);
|
|
if (err)
|
|
goto out;
|
|
|
|
*pkey = be16_to_cpu(((__be16 *)out_mad->data)[index % 32]);
|
|
|
|
out:
|
|
kfree(in_mad);
|
|
kfree(out_mad);
|
|
return err;
|
|
}
|
|
|
|
int mlx5_query_mad_ifc_gids(struct ib_device *ibdev, u8 port, int index,
|
|
union ib_gid *gid)
|
|
{
|
|
struct ib_smp *in_mad = NULL;
|
|
struct ib_smp *out_mad = NULL;
|
|
int err = -ENOMEM;
|
|
|
|
in_mad = kzalloc(sizeof(*in_mad), GFP_KERNEL);
|
|
out_mad = kmalloc(sizeof(*out_mad), GFP_KERNEL);
|
|
if (!in_mad || !out_mad)
|
|
goto out;
|
|
|
|
init_query_mad(in_mad);
|
|
in_mad->attr_id = IB_SMP_ATTR_PORT_INFO;
|
|
in_mad->attr_mod = cpu_to_be32(port);
|
|
|
|
err = mlx5_MAD_IFC(to_mdev(ibdev), 1, 1, port, NULL, NULL, in_mad,
|
|
out_mad);
|
|
if (err)
|
|
goto out;
|
|
|
|
memcpy(gid->raw, out_mad->data + 8, 8);
|
|
|
|
init_query_mad(in_mad);
|
|
in_mad->attr_id = IB_SMP_ATTR_GUID_INFO;
|
|
in_mad->attr_mod = cpu_to_be32(index / 8);
|
|
|
|
err = mlx5_MAD_IFC(to_mdev(ibdev), 1, 1, port, NULL, NULL, in_mad,
|
|
out_mad);
|
|
if (err)
|
|
goto out;
|
|
|
|
memcpy(gid->raw + 8, out_mad->data + (index % 8) * 8, 8);
|
|
|
|
out:
|
|
kfree(in_mad);
|
|
kfree(out_mad);
|
|
return err;
|
|
}
|
|
|
|
int mlx5_query_mad_ifc_port(struct ib_device *ibdev, u8 port,
|
|
struct ib_port_attr *props)
|
|
{
|
|
struct mlx5_ib_dev *dev = to_mdev(ibdev);
|
|
struct mlx5_core_dev *mdev = dev->mdev;
|
|
struct ib_smp *in_mad = NULL;
|
|
struct ib_smp *out_mad = NULL;
|
|
int ext_active_speed;
|
|
int err = -ENOMEM;
|
|
|
|
if (port < 1 || port > MLX5_CAP_GEN(mdev, num_ports)) {
|
|
mlx5_ib_warn(dev, "invalid port number %d\n", port);
|
|
return -EINVAL;
|
|
}
|
|
|
|
in_mad = kzalloc(sizeof(*in_mad), GFP_KERNEL);
|
|
out_mad = kmalloc(sizeof(*out_mad), GFP_KERNEL);
|
|
if (!in_mad || !out_mad)
|
|
goto out;
|
|
|
|
memset(props, 0, sizeof(*props));
|
|
|
|
init_query_mad(in_mad);
|
|
in_mad->attr_id = IB_SMP_ATTR_PORT_INFO;
|
|
in_mad->attr_mod = cpu_to_be32(port);
|
|
|
|
err = mlx5_MAD_IFC(dev, 1, 1, port, NULL, NULL, in_mad, out_mad);
|
|
if (err) {
|
|
mlx5_ib_warn(dev, "err %d\n", err);
|
|
goto out;
|
|
}
|
|
|
|
props->lid = be16_to_cpup((__be16 *)(out_mad->data + 16));
|
|
props->lmc = out_mad->data[34] & 0x7;
|
|
props->sm_lid = be16_to_cpup((__be16 *)(out_mad->data + 18));
|
|
props->sm_sl = out_mad->data[36] & 0xf;
|
|
props->state = out_mad->data[32] & 0xf;
|
|
props->phys_state = out_mad->data[33] >> 4;
|
|
props->port_cap_flags = be32_to_cpup((__be32 *)(out_mad->data + 20));
|
|
props->gid_tbl_len = out_mad->data[50];
|
|
props->max_msg_sz = 1 << MLX5_CAP_GEN(mdev, log_max_msg);
|
|
props->pkey_tbl_len = mdev->port_caps[port - 1].pkey_table_len;
|
|
props->bad_pkey_cntr = be16_to_cpup((__be16 *)(out_mad->data + 46));
|
|
props->qkey_viol_cntr = be16_to_cpup((__be16 *)(out_mad->data + 48));
|
|
props->active_width = out_mad->data[31] & 0xf;
|
|
props->active_speed = out_mad->data[35] >> 4;
|
|
props->max_mtu = out_mad->data[41] & 0xf;
|
|
props->active_mtu = out_mad->data[36] >> 4;
|
|
props->subnet_timeout = out_mad->data[51] & 0x1f;
|
|
props->max_vl_num = out_mad->data[37] >> 4;
|
|
props->init_type_reply = out_mad->data[41] >> 4;
|
|
|
|
/* Check if extended speeds (EDR/FDR/...) are supported */
|
|
if (props->port_cap_flags & IB_PORT_EXTENDED_SPEEDS_SUP) {
|
|
ext_active_speed = out_mad->data[62] >> 4;
|
|
|
|
switch (ext_active_speed) {
|
|
case 1:
|
|
props->active_speed = 16; /* FDR */
|
|
break;
|
|
case 2:
|
|
props->active_speed = 32; /* EDR */
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* If reported active speed is QDR, check if is FDR-10 */
|
|
if (props->active_speed == 4) {
|
|
if (mdev->port_caps[port - 1].ext_port_cap &
|
|
MLX_EXT_PORT_CAP_FLAG_EXTENDED_PORT_INFO) {
|
|
init_query_mad(in_mad);
|
|
in_mad->attr_id = MLX5_ATTR_EXTENDED_PORT_INFO;
|
|
in_mad->attr_mod = cpu_to_be32(port);
|
|
|
|
err = mlx5_MAD_IFC(dev, 1, 1, port,
|
|
NULL, NULL, in_mad, out_mad);
|
|
if (err)
|
|
goto out;
|
|
|
|
/* Checking LinkSpeedActive for FDR-10 */
|
|
if (out_mad->data[15] & 0x1)
|
|
props->active_speed = 8;
|
|
}
|
|
}
|
|
|
|
out:
|
|
kfree(in_mad);
|
|
kfree(out_mad);
|
|
|
|
return err;
|
|
}
|