/* * Copyright (c) 2019 Mellanox Technologies, Inc. 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 #include #include #include #include #include "mlx5dv_dr.h" int dr_devx_query_esw_vport_context(struct ibv_context *ctx, bool other_vport, uint16_t vport_number, uint64_t *icm_address_rx, uint64_t *icm_address_tx) { uint32_t out[DEVX_ST_SZ_DW(query_esw_vport_context_out)] = {}; uint32_t in[DEVX_ST_SZ_DW(query_esw_vport_context_in)] = {}; int err; DEVX_SET(query_esw_vport_context_in, in, opcode, MLX5_CMD_OP_QUERY_ESW_VPORT_CONTEXT); DEVX_SET(query_esw_vport_context_in, in, other_vport, other_vport); DEVX_SET(query_esw_vport_context_in, in, vport_number, vport_number); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Query eswitch vport context failed %d\n", err); return err; } *icm_address_rx = DEVX_GET64(query_esw_vport_context_out, out, esw_vport_context.sw_steering_vport_icm_address_rx); *icm_address_tx = DEVX_GET64(query_esw_vport_context_out, out, esw_vport_context.sw_steering_vport_icm_address_tx); return 0; } static int dr_devx_query_nic_vport_context(struct ibv_context *ctx, bool *roce_en) { uint32_t out[DEVX_ST_SZ_DW(query_nic_vport_context_out)] = {}; uint32_t in[DEVX_ST_SZ_DW(query_nic_vport_context_in)] = {}; int err; DEVX_SET(query_nic_vport_context_in, in, opcode, MLX5_CMD_OP_QUERY_NIC_VPORT_CONTEXT); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Query nic vport context failed %d\n", err); return err; } *roce_en = DEVX_GET(query_nic_vport_context_out, out, nic_vport_context.roce_en); return 0; } int dr_devx_query_gvmi(struct ibv_context *ctx, bool other_vport, uint16_t vport_number, uint16_t *gvmi) { uint32_t out[DEVX_ST_SZ_DW(query_hca_cap_out)] = {}; uint32_t in[DEVX_ST_SZ_DW(query_hca_cap_in)] = {}; int err; DEVX_SET(query_hca_cap_in, in, opcode, MLX5_CMD_OP_QUERY_HCA_CAP); DEVX_SET(query_hca_cap_in, in, other_function, other_vport); DEVX_SET(query_hca_cap_in, in, function_id, vport_number); DEVX_SET(query_hca_cap_in, in, op_mod, MLX5_SET_HCA_CAP_OP_MOD_GENERAL_DEVICE | HCA_CAP_OPMOD_GET_CUR); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Query general failed %d\n", err); return err; } *gvmi = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.vhca_id); return 0; } static int dr_devx_query_esw_func(struct ibv_context *ctx, uint16_t max_sfs, bool *host_pf_vhca_id_valid, uint16_t *host_pf_vhca_id) { uint32_t in[DEVX_ST_SZ_DW(query_esw_functions_in)] = {}; size_t outsz; void *out; int err; outsz = DEVX_ST_SZ_BYTES(query_esw_functions_out) + (max_sfs - 1) * DEVX_FLD_SZ_BYTES(query_esw_functions_out, host_sf_enable); out = calloc(1, outsz); if (!out) { errno = ENOMEM; return errno; } DEVX_SET(query_esw_functions_in, in, opcode, MLX5_CMD_OP_QUERY_ESW_FUNCTIONS); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, outsz); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Query esw func failed %d\n", err); free(out); return err; } *host_pf_vhca_id_valid = DEVX_GET(query_esw_functions_out, out, host_params_context.host_pf_vhca_id_valid); *host_pf_vhca_id = DEVX_GET(query_esw_functions_out, out, host_params_context.host_pf_vhca_id); free(out); return 0; } int dr_devx_query_esw_caps(struct ibv_context *ctx, struct dr_esw_caps *caps) { uint32_t out[DEVX_ST_SZ_DW(query_hca_cap_out)] = {}; uint32_t in[DEVX_ST_SZ_DW(query_hca_cap_in)] = {}; void *esw_caps; int err; DEVX_SET(query_hca_cap_in, in, opcode, MLX5_CMD_OP_QUERY_HCA_CAP); DEVX_SET(query_hca_cap_in, in, op_mod, MLX5_SET_HCA_CAP_OP_MOD_ESW_FLOW_TABLE | HCA_CAP_OPMOD_GET_CUR); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Query general failed %d\n", err); return err; } esw_caps = DEVX_ADDR_OF(query_hca_cap_out, out, capability.flow_table_eswitch_cap); caps->drop_icm_address_rx = DEVX_GET64(flow_table_eswitch_cap, esw_caps, sw_steering_fdb_action_drop_icm_address_rx); caps->drop_icm_address_tx = DEVX_GET64(flow_table_eswitch_cap, esw_caps, sw_steering_fdb_action_drop_icm_address_tx); caps->uplink_icm_address_rx = DEVX_GET64(flow_table_eswitch_cap, esw_caps, sw_steering_uplink_icm_address_rx); caps->uplink_icm_address_tx = DEVX_GET64(flow_table_eswitch_cap, esw_caps, sw_steering_uplink_icm_address_tx); caps->sw_owner_v2 = DEVX_GET(flow_table_eswitch_cap, esw_caps, flow_table_properties_nic_esw_fdb.sw_owner_v2); if (!caps->sw_owner_v2) caps->sw_owner = DEVX_GET(flow_table_eswitch_cap, esw_caps, flow_table_properties_nic_esw_fdb.sw_owner); return 0; } int dr_devx_query_device(struct ibv_context *ctx, struct dr_devx_caps *caps) { uint32_t out[DEVX_ST_SZ_DW(query_hca_cap_out)] = {}; uint32_t in[DEVX_ST_SZ_DW(query_hca_cap_in)] = {}; bool host_pf_vhca_id_valid = false; uint16_t host_pf_vhca_id = 0; uint32_t max_sfs = 0; bool roce, sf_supp; int err; DEVX_SET(query_hca_cap_in, in, opcode, MLX5_CMD_OP_QUERY_HCA_CAP); DEVX_SET(query_hca_cap_in, in, op_mod, MLX5_SET_HCA_CAP_OP_MOD_GENERAL_DEVICE | HCA_CAP_OPMOD_GET_CUR); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Query general failed %d\n", err); return err; } caps->prio_tag_required = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.prio_tag_required); caps->eswitch_manager = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.eswitch_manager); caps->gvmi = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.vhca_id); caps->flex_protocols = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_protocols); caps->isolate_vl_tc = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.isolate_vl_tc_new); caps->flex_parser_header_modify = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_header_modify); sf_supp = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.sf); caps->definer_format_sup = DEVX_GET64(query_hca_cap_out, out, capability.cmd_hca_cap.match_definer_format_supported); roce = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.roce); caps->sw_format_ver = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.steering_format_version); caps->support_modify_argument = DEVX_GET64(query_hca_cap_out, out, capability.cmd_hca_cap.general_obj_types) & (1LL << MLX5_OBJ_TYPE_HEADER_MODIFY_ARGUMENT); caps->roce_caps.fl_rc_qp_when_roce_disabled = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.fl_rc_qp_when_roce_disabled); caps->roce_caps.qp_ts_format = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.sq_ts_format); if (caps->support_modify_argument) { caps->log_header_modify_argument_granularity = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.log_header_modify_argument_granularity); caps->log_header_modify_argument_max_alloc = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.log_header_modify_argument_max_alloc); } if (caps->flex_protocols & MLX5_FLEX_PARSER_ICMP_V4_ENABLED) { caps->flex_parser_id_icmp_dw0 = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_icmp_dw0); caps->flex_parser_id_icmp_dw1 = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_icmp_dw1); } if (caps->flex_protocols & MLX5_FLEX_PARSER_ICMP_V6_ENABLED) { caps->flex_parser_id_icmpv6_dw0 = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_icmpv6_dw0); caps->flex_parser_id_icmpv6_dw1 = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_icmpv6_dw1); } if (caps->flex_protocols & MLX5_FLEX_PARSER_GENEVE_OPT_0_ENABLED) caps->flex_parser_id_geneve_opt_0 = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_geneve_opt_0); if (caps->flex_protocols & MLX5_FLEX_PARSER_MPLS_OVER_GRE_ENABLED) caps->flex_parser_id_mpls_over_gre = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_outer_first_mpls_over_gre); if (caps->flex_protocols & mlx5_FLEX_PARSER_MPLS_OVER_UDP_ENABLED) caps->flex_parser_id_mpls_over_udp = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_outer_first_mpls_over_udp_label); if (caps->flex_protocols & MLX5_FLEX_PARSER_GTPU_DW_0_ENABLED) caps->flex_parser_id_gtpu_dw_0 = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_gtpu_dw_0); if (caps->flex_protocols & MLX5_FLEX_PARSER_GTPU_TEID_ENABLED) caps->flex_parser_id_gtpu_teid = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_gtpu_teid); if (caps->flex_protocols & MLX5_FLEX_PARSER_GTPU_DW_2_ENABLED) caps->flex_parser_id_gtpu_dw_2 = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_gtpu_dw_2); if (caps->flex_protocols & MLX5_FLEX_PARSER_GTPU_FIRST_EXT_DW_0_ENABLED) caps->flex_parser_id_gtpu_first_ext_dw_0 = DEVX_GET(query_hca_cap_out, out, capability.cmd_hca_cap.flex_parser_id_gtpu_first_ext_dw_0); DEVX_SET(query_hca_cap_in, in, op_mod, MLX5_SET_HCA_CAP_OP_MOD_NIC_FLOW_TABLE | HCA_CAP_OPMOD_GET_CUR); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Query flow tables failed %d\n", err); return err; } caps->max_encap_size = DEVX_GET(query_hca_cap_out, out, capability.flow_table_nic_cap.max_encap_header_size); caps->nic_rx_drop_address = DEVX_GET64(query_hca_cap_out, out, capability.flow_table_nic_cap. sw_steering_nic_rx_action_drop_icm_address); caps->nic_tx_drop_address = DEVX_GET64(query_hca_cap_out, out, capability.flow_table_nic_cap. sw_steering_nic_tx_action_drop_icm_address); caps->nic_tx_allow_address = DEVX_GET64(query_hca_cap_out, out, capability.flow_table_nic_cap. sw_steering_nic_tx_action_allow_icm_address); caps->rx_sw_owner_v2 = DEVX_GET(query_hca_cap_out, out, capability.flow_table_nic_cap. flow_table_properties_nic_receive.sw_owner_v2); caps->tx_sw_owner_v2 = DEVX_GET(query_hca_cap_out, out, capability.flow_table_nic_cap. flow_table_properties_nic_transmit.sw_owner_v2); if (!caps->rx_sw_owner_v2) caps->rx_sw_owner = DEVX_GET(query_hca_cap_out, out, capability.flow_table_nic_cap. flow_table_properties_nic_receive.sw_owner); if (!caps->tx_sw_owner_v2) caps->tx_sw_owner = DEVX_GET(query_hca_cap_out, out, capability.flow_table_nic_cap. flow_table_properties_nic_transmit.sw_owner); caps->max_ft_level = DEVX_GET(query_hca_cap_out, out, capability.flow_table_nic_cap. flow_table_properties_nic_receive.max_ft_level); /* l4_csum_ok is the indication for definer support csum and ok bits. * Since we don't have definer versions we rely on new field support */ caps->definer_supp_checksum = DEVX_GET(query_hca_cap_out, out, capability.flow_table_nic_cap. ft_field_bitmask_support_2_nic_receive. outer_l4_checksum_ok); /* geneve_tlv_option_0_exist is the indication for STE support for * lookup type flex_parser_ok. */ caps->flex_parser_ok_bits_supp = DEVX_GET(query_hca_cap_out, out, capability.flow_table_nic_cap. flow_table_properties_nic_receive. ft_field_support. geneve_tlv_option_0_exist); caps->support_full_tnl_hdr = (DEVX_GET(query_hca_cap_out, out, capability.flow_table_nic_cap. ft_field_bitmask_support_2_nic_receive. tunnel_header_0_1) && DEVX_GET(query_hca_cap_out, out, capability.flow_table_nic_cap. ft_field_bitmask_support_2_nic_receive. tunnel_header_2_3)); if (sf_supp && caps->eswitch_manager) { DEVX_SET(query_hca_cap_in, in, op_mod, MLX5_SET_HCA_CAP_OP_MOD_ESW | HCA_CAP_OPMOD_GET_CUR); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Query eswitch capabilities failed %d\n", err); return err; } max_sfs = 1 << DEVX_GET(query_hca_cap_out, out, capability.e_switch_cap.log_max_esw_sf); } if (caps->eswitch_manager) { /* Check if ECPF */ if (DEVX_GET(query_hca_cap_out, out, capability.e_switch_cap.esw_manager_vport_number_valid)) { if (DEVX_GET(query_hca_cap_out, out, capability.e_switch_cap.esw_manager_vport_number) == ECPF_PORT) caps->is_ecpf = true; } else { err = dr_devx_query_esw_func(ctx, max_sfs, &host_pf_vhca_id_valid, &host_pf_vhca_id); if (!err && host_pf_vhca_id_valid && host_pf_vhca_id != caps->gvmi) caps->is_ecpf = true; } } DEVX_SET(query_hca_cap_in, in, op_mod, MLX5_SET_HCA_CAP_OP_MOD_DEVICE_MEMORY | HCA_CAP_OPMOD_GET_CUR); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Query flow device memory caps failed %d\n", err); return err; } caps->log_icm_size = DEVX_GET(query_hca_cap_out, out, capability.device_mem_cap.log_steering_sw_icm_size); caps->hdr_modify_icm_addr = DEVX_GET64(query_hca_cap_out, out, capability.device_mem_cap. header_modify_sw_icm_start_address); caps->log_modify_hdr_icm_size = DEVX_GET(query_hca_cap_out, out, capability.device_mem_cap.log_header_modify_sw_icm_size); caps->log_modify_pattern_icm_size = DEVX_GET(query_hca_cap_out, out, capability.device_mem_cap.log_header_modify_pattern_sw_icm_size); caps->hdr_modify_pattern_icm_addr = DEVX_GET64(query_hca_cap_out, out, capability.device_mem_cap.header_modify_pattern_sw_icm_start_address); caps->log_sw_encap_icm_size = DEVX_GET(query_hca_cap_out, out, capability.device_mem_cap.log_indirect_encap_sw_icm_size); if (caps->log_sw_encap_icm_size) caps->indirect_encap_icm_base = DEVX_GET64(query_hca_cap_out, out, capability.device_mem_cap.indirect_encap_icm_base); /* RoCE caps */ if (roce) { err = dr_devx_query_nic_vport_context(ctx, &caps->roce_caps.roce_en); if (err) return err; DEVX_SET(query_hca_cap_in, in, opcode, MLX5_CMD_OP_QUERY_HCA_CAP); DEVX_SET(query_hca_cap_in, in, op_mod, MLX5_SET_HCA_CAP_OP_MOD_ROCE | HCA_CAP_OPMOD_GET_CUR); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Query RoCE capabilities failed %d\n", err); return err; } caps->roce_caps.fl_rc_qp_when_roce_disabled |= DEVX_GET(query_hca_cap_out, out, capability.roce_caps.fl_rc_qp_when_roce_disabled); caps->roce_caps.fl_rc_qp_when_roce_enabled = DEVX_GET(query_hca_cap_out, out, capability.roce_caps.fl_rc_qp_when_roce_enabled); caps->roce_caps.qp_ts_format = DEVX_GET(query_hca_cap_out, out, capability.roce_caps.qp_ts_format); } return 0; } int dr_devx_sync_steering(struct ibv_context *ctx) { uint32_t out[DEVX_ST_SZ_DW(sync_steering_out)] = {}; uint32_t in[DEVX_ST_SZ_DW(sync_steering_in)] = {}; int err; DEVX_SET(sync_steering_in, in, opcode, MLX5_CMD_OP_SYNC_STEERING); err = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (err) { err = mlx5_get_cmd_status_err(err, out); dr_dbg_ctx(ctx, "Sync steering failed %d\n", err); } return err; } struct mlx5dv_devx_obj * dr_devx_create_flow_table(struct ibv_context *ctx, struct dr_devx_flow_table_attr *ft_attr) { uint32_t out[DEVX_ST_SZ_DW(create_flow_table_out)] = {}; uint32_t in[DEVX_ST_SZ_DW(create_flow_table_in)] = {}; struct mlx5dv_devx_obj *obj; void *ft_ctx; DEVX_SET(create_flow_table_in, in, opcode, MLX5_CMD_OP_CREATE_FLOW_TABLE); DEVX_SET(create_flow_table_in, in, table_type, ft_attr->type); ft_ctx = DEVX_ADDR_OF(create_flow_table_in, in, flow_table_context); DEVX_SET(flow_table_context, ft_ctx, termination_table, ft_attr->term_tbl); DEVX_SET(flow_table_context, ft_ctx, sw_owner, ft_attr->sw_owner); DEVX_SET(flow_table_context, ft_ctx, level, ft_attr->level); DEVX_SET(flow_table_context, ft_ctx, reformat_en, ft_attr->reformat_en); if (ft_attr->sw_owner) { /* icm_addr_0 used for FDB RX / NIC TX / NIC_RX * icm_addr_1 used for FDB TX */ if (ft_attr->type == FS_FT_NIC_RX) { DEVX_SET64(flow_table_context, ft_ctx, sw_owner_icm_root_0, ft_attr->icm_addr_rx); } else if (ft_attr->type == FS_FT_NIC_TX) { DEVX_SET64(flow_table_context, ft_ctx, sw_owner_icm_root_0, ft_attr->icm_addr_tx); } else if (ft_attr->type == FS_FT_FDB) { DEVX_SET64(flow_table_context, ft_ctx, sw_owner_icm_root_0, ft_attr->icm_addr_rx); DEVX_SET64(flow_table_context, ft_ctx, sw_owner_icm_root_1, ft_attr->icm_addr_tx); } else { assert(false); } } obj = mlx5dv_devx_obj_create(ctx, in, sizeof(in), out, sizeof(out)); if (!obj) errno = mlx5_get_cmd_status_err(errno, out); return obj; } int dr_devx_query_flow_table(struct mlx5dv_devx_obj *obj, uint32_t type, uint64_t *rx_icm_addr, uint64_t *tx_icm_addr) { uint32_t out[DEVX_ST_SZ_DW(query_flow_table_out)] = {}; uint32_t in[DEVX_ST_SZ_DW(query_flow_table_in)] = {}; int ret; DEVX_SET(query_flow_table_in, in, opcode, MLX5_CMD_OP_QUERY_FLOW_TABLE); DEVX_SET(query_flow_table_in, in, table_type, type); DEVX_SET(query_flow_table_in, in, table_id, obj->object_id); ret = mlx5dv_devx_obj_query(obj, in, sizeof(in), out, sizeof(out)); if (ret) { dr_dbg_ctx(obj->context, "Failed to query flow table id %u\n", obj->object_id); return mlx5_get_cmd_status_err(ret, out); } switch (type) { case FS_FT_NIC_TX: *tx_icm_addr = DEVX_GET64(query_flow_table_out, out, flow_table_context.sw_owner_icm_root_0); *rx_icm_addr = 0; break; case FS_FT_NIC_RX: *rx_icm_addr = DEVX_GET64(query_flow_table_out, out, flow_table_context.sw_owner_icm_root_0); *tx_icm_addr = 0; break; case FS_FT_FDB: *rx_icm_addr = DEVX_GET64(query_flow_table_out, out, flow_table_context.sw_owner_icm_root_0); *tx_icm_addr = DEVX_GET64(query_flow_table_out, out, flow_table_context.sw_owner_icm_root_1); break; default: errno = EINVAL; return errno; } return 0; } static struct mlx5dv_devx_obj * dr_devx_create_flow_group(struct ibv_context *ctx, struct dr_devx_flow_group_attr *fg_attr) { uint32_t out[DEVX_ST_SZ_DW(create_flow_group_out)] = {}; uint32_t inlen = DEVX_ST_SZ_BYTES(create_flow_group_in); struct mlx5dv_devx_obj *obj; uint32_t *in; in = calloc(1, inlen); if (!in) { errno = ENOMEM; return NULL; } DEVX_SET(create_flow_group_in, in, opcode, MLX5_CMD_OP_CREATE_FLOW_GROUP); DEVX_SET(create_flow_group_in, in, table_type, fg_attr->table_type); DEVX_SET(create_flow_group_in, in, table_id, fg_attr->table_id); obj = mlx5dv_devx_obj_create(ctx, in, inlen, out, sizeof(out)); if (!obj) errno = mlx5_get_cmd_status_err(errno, out); free(in); return obj; } static struct mlx5dv_devx_obj * dr_devx_set_fte(struct ibv_context *ctx, struct dr_devx_flow_fte_attr *fte_attr) { uint32_t out[DEVX_ST_SZ_DW(set_fte_out)] = {}; struct mlx5dv_devx_obj *obj; uint32_t dest_entry_size; void *in_flow_context; uint32_t list_size; uint8_t *in_dests; uint32_t inlen; uint32_t *in; uint32_t i; if (fte_attr->extended_dest) dest_entry_size = DEVX_ST_SZ_BYTES(extended_dest_format); else dest_entry_size = DEVX_ST_SZ_BYTES(dest_format); inlen = DEVX_ST_SZ_BYTES(set_fte_in) + fte_attr->dest_size * dest_entry_size; in = calloc(1, inlen); if (!in) { errno = ENOMEM; return NULL; } DEVX_SET(set_fte_in, in, opcode, MLX5_CMD_OP_SET_FLOW_TABLE_ENTRY); DEVX_SET(set_fte_in, in, table_type, fte_attr->table_type); DEVX_SET(set_fte_in, in, table_id, fte_attr->table_id); in_flow_context = DEVX_ADDR_OF(set_fte_in, in, flow_context); DEVX_SET(flow_context, in_flow_context, group_id, fte_attr->group_id); DEVX_SET(flow_context, in_flow_context, flow_tag, fte_attr->flow_tag); DEVX_SET(flow_context, in_flow_context, action, fte_attr->action); DEVX_SET(flow_context, in_flow_context, extended_destination, fte_attr->extended_dest); in_dests = DEVX_ADDR_OF(flow_context, in_flow_context, destination); if (fte_attr->action & MLX5_FLOW_CONTEXT_ACTION_FWD_DEST) { list_size = 0; for (i = 0; i < fte_attr->dest_size; i++) { uint32_t id; uint32_t type = fte_attr->dest_arr[i].type; if (type == MLX5_FLOW_DEST_TYPE_COUNTER) continue; switch (type) { case MLX5_FLOW_DEST_TYPE_VPORT: id = fte_attr->dest_arr[i].vport_num; break; case MLX5_FLOW_DEST_TYPE_TIR: id = fte_attr->dest_arr[i].tir_num; break; case MLX5_FLOW_DEST_TYPE_FT: id = fte_attr->dest_arr[i].ft_id; break; default: errno = EOPNOTSUPP; goto err_out; } DEVX_SET(dest_format, in_dests, destination_type, type); DEVX_SET(dest_format, in_dests, destination_id, id); if (fte_attr->dest_arr[i].has_reformat) { if (!fte_attr->extended_dest) { errno = EINVAL; goto err_out; } DEVX_SET(dest_format, in_dests, packet_reformat, 1); DEVX_SET(extended_dest_format, in_dests, packet_reformat_id, fte_attr->dest_arr[i].reformat_id); } in_dests += dest_entry_size; list_size++; } DEVX_SET(flow_context, in_flow_context, destination_list_size, list_size); } if (fte_attr->action & MLX5_FLOW_CONTEXT_ACTION_COUNT) { list_size = 0; for (i = 0; i < fte_attr->dest_size; i++) { if (fte_attr->dest_arr[i].type != MLX5_FLOW_DEST_TYPE_COUNTER) continue; DEVX_SET(flow_counter_list, in_dests, flow_counter_id, fte_attr->dest_arr[i].counter_id); in_dests += dest_entry_size; list_size++; } DEVX_SET(flow_context, in_flow_context, flow_counter_list_size, list_size); } obj = mlx5dv_devx_obj_create(ctx, in, inlen, out, sizeof(out)); if (!obj) errno = mlx5_get_cmd_status_err(errno, out); free(in); return obj; err_out: free(in); return NULL; } struct dr_devx_tbl * dr_devx_create_always_hit_ft(struct ibv_context *ctx, struct dr_devx_flow_table_attr *ft_attr, struct dr_devx_flow_group_attr *fg_attr, struct dr_devx_flow_fte_attr *fte_attr) { struct mlx5dv_devx_obj *fte_dvo; struct mlx5dv_devx_obj *fg_dvo; struct mlx5dv_devx_obj *ft_dvo; struct dr_devx_tbl *tbl; tbl = calloc(1, sizeof(*tbl)); if (!tbl) { errno = ENOMEM; return NULL; } ft_dvo = dr_devx_create_flow_table(ctx, ft_attr); if (!ft_dvo) goto free_tbl; fg_attr->table_id = ft_dvo->object_id; fg_attr->table_type = ft_attr->type; fg_dvo = dr_devx_create_flow_group(ctx, fg_attr); if (!fg_dvo) goto free_ft_dvo; fte_attr->table_id = ft_dvo->object_id; fte_attr->table_type = ft_attr->type; fte_attr->group_id = fg_dvo->object_id; fte_dvo = dr_devx_set_fte(ctx, fte_attr); if (!fte_dvo) goto free_fg_dvo; tbl->type = ft_attr->type; tbl->level = ft_attr->level; tbl->ft_dvo = ft_dvo; tbl->fg_dvo = fg_dvo; tbl->fte_dvo = fte_dvo; return tbl; free_fg_dvo: mlx5dv_devx_obj_destroy(fg_dvo); free_ft_dvo: mlx5dv_devx_obj_destroy(ft_dvo); free_tbl: free(tbl); return NULL; } void dr_devx_destroy_always_hit_ft(struct dr_devx_tbl *devx_tbl) { mlx5dv_devx_obj_destroy(devx_tbl->fte_dvo); mlx5dv_devx_obj_destroy(devx_tbl->fg_dvo); mlx5dv_devx_obj_destroy(devx_tbl->ft_dvo); free(devx_tbl); } struct mlx5dv_devx_obj * dr_devx_create_flow_sampler(struct ibv_context *ctx, struct dr_devx_flow_sampler_attr *sampler_attr) { uint32_t out[DEVX_ST_SZ_DW(general_obj_out_cmd_hdr)] = {}; uint32_t in[DEVX_ST_SZ_DW(create_flow_sampler_in)] = {}; struct mlx5dv_devx_obj *obj; void *attr; attr = DEVX_ADDR_OF(create_flow_sampler_in, in, hdr); DEVX_SET(general_obj_in_cmd_hdr, attr, opcode, MLX5_CMD_OP_CREATE_GENERAL_OBJECT); DEVX_SET(general_obj_in_cmd_hdr, attr, obj_type, MLX5_OBJ_TYPE_FLOW_SAMPLER); attr = DEVX_ADDR_OF(create_flow_sampler_in, in, sampler); DEVX_SET(flow_sampler, attr, table_type, sampler_attr->table_type); DEVX_SET(flow_sampler, attr, level, sampler_attr->level); DEVX_SET(flow_sampler, attr, sample_ratio, sampler_attr->sample_ratio); DEVX_SET(flow_sampler, attr, ignore_flow_level, sampler_attr->ignore_flow_level); DEVX_SET(flow_sampler, attr, default_table_id, sampler_attr->default_next_table_id); DEVX_SET(flow_sampler, attr, sample_table_id, sampler_attr->sample_table_id); obj = mlx5dv_devx_obj_create(ctx, in, sizeof(in), out, sizeof(out)); if (!obj) errno = mlx5_get_cmd_status_err(errno, out); return obj; } int dr_devx_query_flow_sampler(struct mlx5dv_devx_obj *obj, uint64_t *rx_icm_addr, uint64_t *tx_icm_addr) { uint32_t out[DEVX_ST_SZ_DW(query_flow_sampler_out)] = {}; uint32_t in[DEVX_ST_SZ_DW(general_obj_in_cmd_hdr)] = {}; void *attr; int ret; DEVX_SET(general_obj_in_cmd_hdr, in, opcode, MLX5_CMD_OP_QUERY_GENERAL_OBJECT); DEVX_SET(general_obj_in_cmd_hdr, in, obj_type, MLX5_OBJ_TYPE_FLOW_SAMPLER); DEVX_SET(general_obj_in_cmd_hdr, in, obj_id, obj->object_id); ret = mlx5dv_devx_obj_query(obj, in, sizeof(in), out, sizeof(out)); if (ret) { dr_dbg_ctx(obj->context, "Failed to query flow sampler id %u\n", obj->object_id); return mlx5_get_cmd_status_err(ret, out); } attr = DEVX_ADDR_OF(query_flow_sampler_out, out, obj); *rx_icm_addr = DEVX_GET64(flow_sampler, attr, sw_steering_icm_address_rx); *tx_icm_addr = DEVX_GET64(flow_sampler, attr, sw_steering_icm_address_tx); return 0; } struct mlx5dv_devx_obj *dr_devx_create_definer(struct ibv_context *ctx, uint16_t format_id, uint8_t *match_mask) { uint32_t out[DEVX_ST_SZ_DW(general_obj_out_cmd_hdr)] = {}; uint32_t in[DEVX_ST_SZ_DW(create_definer_in)] = {}; struct mlx5dv_devx_obj *obj; void *ptr; DEVX_SET(general_obj_in_cmd_hdr, in, opcode, MLX5_CMD_OP_CREATE_GENERAL_OBJECT); DEVX_SET(general_obj_in_cmd_hdr, in, obj_type, MLX5_OBJ_TYPE_MATCH_DEFINER); ptr = DEVX_ADDR_OF(create_definer_in, in, definer); DEVX_SET(definer, ptr, format_id, format_id); ptr = DEVX_ADDR_OF(definer, ptr, match_mask_dw_7_0); memcpy(ptr, match_mask, DEVX_FLD_SZ_BYTES(definer, match_mask_dw_7_0)); obj = mlx5dv_devx_obj_create(ctx, in, sizeof(in), out, sizeof(out)); if (!obj) errno = mlx5_get_cmd_status_err(errno, out); return obj; } struct mlx5dv_devx_obj *dr_devx_create_reformat_ctx(struct ibv_context *ctx, enum reformat_type rt, size_t reformat_size, void *reformat_data) { uint32_t out[DEVX_ST_SZ_DW(alloc_packet_reformat_context_out)] = {}; size_t insz, cmd_data_sz, cmd_total_sz; struct mlx5dv_devx_obj *obj; void *prctx; void *pdata; void *in; cmd_total_sz = DEVX_ST_SZ_BYTES(alloc_packet_reformat_context_in); cmd_data_sz = DEVX_FLD_SZ_BYTES(alloc_packet_reformat_context_in, packet_reformat_context.reformat_data); insz = align(cmd_total_sz + reformat_size - cmd_data_sz, 4); in = calloc(1, insz); if (!in) { errno = ENOMEM; return NULL; } DEVX_SET(alloc_packet_reformat_context_in, in, opcode, MLX5_CMD_OP_ALLOC_PACKET_REFORMAT_CONTEXT); prctx = DEVX_ADDR_OF(alloc_packet_reformat_context_in, in, packet_reformat_context); pdata = DEVX_ADDR_OF(packet_reformat_context_in, prctx, reformat_data); DEVX_SET(packet_reformat_context_in, prctx, reformat_type, rt); DEVX_SET(packet_reformat_context_in, prctx, reformat_data_size, reformat_size); memcpy(pdata, reformat_data, reformat_size); obj = mlx5dv_devx_obj_create(ctx, in, insz, out, sizeof(out)); if (!obj) errno = mlx5_get_cmd_status_err(errno, out); free(in); return obj; } struct mlx5dv_devx_obj *dr_devx_create_meter(struct ibv_context *ctx, struct mlx5dv_dr_flow_meter_attr *meter_attr) { uint32_t out[DEVX_ST_SZ_DW(general_obj_out_cmd_hdr)] = {}; uint32_t in[DEVX_ST_SZ_DW(create_flow_meter_in)] = {}; struct mlx5dv_devx_obj *obj; void *attr; if (meter_attr->flow_meter_parameter_sz > DEVX_FLD_SZ_BYTES(flow_meter, flow_meter_params)) { errno = EINVAL; return NULL; } attr = DEVX_ADDR_OF(create_flow_meter_in, in, hdr); DEVX_SET(general_obj_in_cmd_hdr, attr, opcode, MLX5_CMD_OP_CREATE_GENERAL_OBJECT); DEVX_SET(general_obj_in_cmd_hdr, attr, obj_type, MLX5_OBJ_TYPE_FLOW_METER); attr = DEVX_ADDR_OF(create_flow_meter_in, in, meter); DEVX_SET(flow_meter, attr, active, meter_attr->active); DEVX_SET(flow_meter, attr, return_reg_id, meter_attr->reg_c_index); DEVX_SET(flow_meter, attr, table_type, meter_attr->next_table->table_type); DEVX_SET(flow_meter, attr, destination_table_id, meter_attr->next_table->devx_obj->object_id); attr = DEVX_ADDR_OF(flow_meter, attr, flow_meter_params); memcpy(attr, meter_attr->flow_meter_parameter, meter_attr->flow_meter_parameter_sz); obj = mlx5dv_devx_obj_create(ctx, in, sizeof(in), out, sizeof(out)); if (!obj) errno = mlx5_get_cmd_status_err(errno, out); return obj; } int dr_devx_query_meter(struct mlx5dv_devx_obj *obj, uint64_t *rx_icm_addr, uint64_t *tx_icm_addr) { uint32_t in[DEVX_ST_SZ_DW(general_obj_in_cmd_hdr)] = {}; uint32_t out[DEVX_ST_SZ_DW(query_flow_meter_out)] = {}; void *attr; int ret; DEVX_SET(general_obj_in_cmd_hdr, in, opcode, MLX5_CMD_OP_QUERY_GENERAL_OBJECT); DEVX_SET(general_obj_in_cmd_hdr, in, obj_type, MLX5_OBJ_TYPE_FLOW_METER); DEVX_SET(general_obj_in_cmd_hdr, in, obj_id, obj->object_id); ret = mlx5dv_devx_obj_query(obj, in, sizeof(in), out, sizeof(out)); if (ret) { dr_dbg_ctx(obj->context, "Failed to query flow meter id %u\n", obj->object_id); return mlx5_get_cmd_status_err(ret, out); } attr = DEVX_ADDR_OF(query_flow_meter_out, out, obj); *rx_icm_addr = DEVX_GET64(flow_meter, attr, sw_steering_icm_address_rx); *tx_icm_addr = DEVX_GET64(flow_meter, attr, sw_steering_icm_address_tx); return 0; } int dr_devx_modify_meter(struct mlx5dv_devx_obj *obj, struct mlx5dv_dr_flow_meter_attr *meter_attr, __be64 modify_bits) { uint32_t out[DEVX_ST_SZ_DW(general_obj_out_cmd_hdr)] = {}; uint32_t in[DEVX_ST_SZ_DW(create_flow_meter_in)] = {}; void *attr; int ret; if (meter_attr->flow_meter_parameter_sz > DEVX_FLD_SZ_BYTES(flow_meter, flow_meter_params)) { errno = EINVAL; return errno; } attr = DEVX_ADDR_OF(create_flow_meter_in, in, hdr); DEVX_SET(general_obj_in_cmd_hdr, attr, opcode, MLX5_CMD_OP_MODIFY_GENERAL_OBJECT); DEVX_SET(general_obj_in_cmd_hdr, attr, obj_type, MLX5_OBJ_TYPE_FLOW_METER); DEVX_SET(general_obj_in_cmd_hdr, in, obj_id, obj->object_id); attr = DEVX_ADDR_OF(create_flow_meter_in, in, meter); memcpy(DEVX_ADDR_OF(flow_meter, attr, modify_field_select), &modify_bits, sizeof(modify_bits)); DEVX_SET(flow_meter, attr, active, meter_attr->active); attr = DEVX_ADDR_OF(flow_meter, attr, flow_meter_params); memcpy(attr, meter_attr->flow_meter_parameter, meter_attr->flow_meter_parameter_sz); ret = mlx5dv_devx_obj_modify(obj, in, sizeof(in), out, sizeof(out)); return ret ? mlx5_get_cmd_status_err(ret, out) : 0; } struct mlx5dv_devx_obj *dr_devx_create_qp(struct ibv_context *ctx, struct dr_devx_qp_create_attr *attr) { uint32_t in[DEVX_ST_SZ_DW(create_qp_in)] = {}; uint32_t out[DEVX_ST_SZ_DW(create_qp_out)] = {}; void *qpc = DEVX_ADDR_OF(create_qp_in, in, qpc); struct mlx5dv_devx_obj *obj; DEVX_SET(create_qp_in, in, opcode, MLX5_CMD_OP_CREATE_QP); DEVX_SET(qpc, qpc, st, attr->service_type); DEVX_SET(qpc, qpc, pm_state, attr->pm_state); DEVX_SET(qpc, qpc, pd, attr->pdn); DEVX_SET(qpc, qpc, uar_page, attr->page_id); DEVX_SET(qpc, qpc, cqn_snd, attr->cqn); DEVX_SET(qpc, qpc, cqn_rcv, attr->cqn); DEVX_SET(qpc, qpc, log_sq_size, ilog32(attr->sq_wqe_cnt - 1)); DEVX_SET(qpc, qpc, log_rq_stride, attr->rq_wqe_shift - 4); DEVX_SET(qpc, qpc, log_rq_size, ilog32(attr->rq_wqe_cnt - 1)); DEVX_SET(qpc, qpc, dbr_umem_id, attr->db_umem_id); DEVX_SET(qpc, qpc, isolate_vl_tc, attr->isolate_vl_tc); DEVX_SET(qpc, qpc, ts_format, attr->qp_ts_format); DEVX_SET(create_qp_in, in, wq_umem_id, attr->buff_umem_id); obj = mlx5dv_devx_obj_create(ctx, in, sizeof(in), out, sizeof(out)); if (!obj) errno = mlx5_get_cmd_status_err(errno, out); return obj; } int dr_devx_modify_qp_rst2init(struct ibv_context *ctx, struct mlx5dv_devx_obj *qp_obj, uint16_t port) { uint32_t in[DEVX_ST_SZ_DW(rst2init_qp_in)] = {}; uint32_t out[DEVX_ST_SZ_DW(rst2init_qp_out)] = {}; void *qpc = DEVX_ADDR_OF(rst2init_qp_in, in, qpc); int ret; DEVX_SET(rst2init_qp_in, in, opcode, MLX5_CMD_OP_RST2INIT_QP); DEVX_SET(rst2init_qp_in, in, qpn, qp_obj->object_id); DEVX_SET(qpc, qpc, primary_address_path.vhca_port_num, port); DEVX_SET(qpc, qpc, pm_state, MLX5_QPC_PM_STATE_MIGRATED); DEVX_SET(qpc, qpc, rre, 1); DEVX_SET(qpc, qpc, rwe, 1); ret = mlx5dv_devx_obj_modify(qp_obj, in, sizeof(in), out, sizeof(out)); return ret ? mlx5_get_cmd_status_err(ret, out) : 0; } #define DR_DEVX_ICM_UDP_PORT 49999 int dr_devx_modify_qp_init2rtr(struct ibv_context *ctx, struct mlx5dv_devx_obj *qp_obj, struct dr_devx_qp_rtr_attr *attr) { uint32_t in[DEVX_ST_SZ_DW(init2rtr_qp_in)] = {}; uint32_t out[DEVX_ST_SZ_DW(init2rtr_qp_out)] = {}; void *qpc = DEVX_ADDR_OF(init2rtr_qp_in, in, qpc); int ret; DEVX_SET(init2rtr_qp_in, in, opcode, MLX5_CMD_OP_INIT2RTR_QP); DEVX_SET(init2rtr_qp_in, in, qpn, qp_obj->object_id); DEVX_SET(qpc, qpc, mtu, attr->mtu); DEVX_SET(qpc, qpc, log_msg_max, DR_CHUNK_SIZE_MAX - 1); DEVX_SET(qpc, qpc, remote_qpn, attr->qp_num); if (attr->fl) { DEVX_SET(qpc, qpc, primary_address_path.fl, attr->fl); } else { memcpy(DEVX_ADDR_OF(qpc, qpc, primary_address_path.rmac_47_32), attr->dgid_attr.mac, sizeof(attr->dgid_attr.mac)); memcpy(DEVX_ADDR_OF(qpc, qpc, primary_address_path.rgid_rip), attr->dgid_attr.gid.raw, sizeof(attr->dgid_attr.gid.raw)); DEVX_SET(qpc, qpc, primary_address_path.src_addr_index, attr->sgid_index); if (attr->dgid_attr.roce_ver == MLX5_ROCE_VERSION_2) DEVX_SET(qpc, qpc, primary_address_path.udp_sport, DR_DEVX_ICM_UDP_PORT); } DEVX_SET(qpc, qpc, primary_address_path.vhca_port_num, attr->port_num); DEVX_SET(qpc, qpc, min_rnr_nak, 1); ret = mlx5dv_devx_obj_modify(qp_obj, in, sizeof(in), out, sizeof(out)); return ret ? mlx5_get_cmd_status_err(ret, out) : 0; } int dr_devx_modify_qp_rtr2rts(struct ibv_context *ctx, struct mlx5dv_devx_obj *qp_obj, struct dr_devx_qp_rts_attr *attr) { uint32_t in[DEVX_ST_SZ_DW(rtr2rts_qp_in)] = {}; uint32_t out[DEVX_ST_SZ_DW(rtr2rts_qp_out)] = {}; void *qpc = DEVX_ADDR_OF(rtr2rts_qp_in, in, qpc); int ret; DEVX_SET(rtr2rts_qp_in, in, opcode, MLX5_CMD_OP_RTR2RTS_QP); DEVX_SET(rtr2rts_qp_in, in, qpn, qp_obj->object_id); DEVX_SET(qpc, qpc, log_ack_req_freq, 0); DEVX_SET(qpc, qpc, retry_count, attr->retry_cnt); DEVX_SET(qpc, qpc, rnr_retry, attr->rnr_retry); DEVX_SET(qpc, qpc, primary_address_path.ack_timeout, 0x8); /* ~1ms */ ret = mlx5dv_devx_obj_modify(qp_obj, in, sizeof(in), out, sizeof(out)); return ret ? mlx5_get_cmd_status_err(ret, out) : 0; } int dr_devx_query_gid(struct ibv_context *ctx, uint8_t vhca_port_num, uint16_t index, struct dr_gid_attr *attr) { uint32_t out[DEVX_ST_SZ_DW(query_roce_address_out)] = {}; uint32_t in[DEVX_ST_SZ_DW(query_roce_address_in)] = {}; int ret; DEVX_SET(query_roce_address_in, in, opcode, MLX5_CMD_OP_QUERY_ROCE_ADDRESS); DEVX_SET(query_roce_address_in, in, roce_address_index, index); DEVX_SET(query_roce_address_in, in, vhca_port_num, vhca_port_num); ret = mlx5dv_devx_general_cmd(ctx, in, sizeof(in), out, sizeof(out)); if (ret) return mlx5_get_cmd_status_err(ret, out); memcpy(&attr->gid, DEVX_ADDR_OF(query_roce_address_out, out, roce_address.source_l3_address), sizeof(attr->gid)); memcpy(attr->mac, DEVX_ADDR_OF(query_roce_address_out, out, roce_address.source_mac_47_32), sizeof(attr->mac)); if (DEVX_GET(query_roce_address_out, out, roce_address.roce_version) == MLX5_ROCE_VERSION_2) attr->roce_ver = MLX5_ROCE_VERSION_2; else attr->roce_ver = MLX5_ROCE_VERSION_1; return 0; } struct mlx5dv_devx_obj *dr_devx_create_modify_header_arg(struct ibv_context *ctx, uint16_t log_obj_range, uint32_t pd) { uint32_t out[DEVX_ST_SZ_DW(general_obj_out_cmd_hdr)] = {}; uint32_t in[DEVX_ST_SZ_DW(create_modify_header_arg_in)] = {}; void *attr; attr = DEVX_ADDR_OF(create_modify_header_arg_in, in, hdr); DEVX_SET(general_obj_in_cmd_hdr, attr, opcode, MLX5_CMD_OP_CREATE_GENERAL_OBJECT); DEVX_SET(general_obj_in_cmd_hdr, attr, obj_type, MLX5_OBJ_TYPE_HEADER_MODIFY_ARGUMENT); DEVX_SET(general_obj_in_cmd_hdr, attr, log_obj_range, log_obj_range); attr = DEVX_ADDR_OF(create_modify_header_arg_in, in, arg); DEVX_SET(modify_header_arg, attr, access_pd, pd); return mlx5dv_devx_obj_create(ctx, in, sizeof(in), out, sizeof(out)); }