Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/drivers/net/ethernet/intel/iavf/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 25 kB image not shown  

Quelle  iavf_fdir.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0
/* Copyright (c) 2020, Intel Corporation. */

/* flow director ethtool support for iavf */

#include <linux/bitfield.h>
#include "iavf.h"

#define GTPU_PORT 2152
#define NAT_T_ESP_PORT 4500
#define PFCP_PORT 8805

static const struct in6_addr ipv6_addr_full_mask = {
 .in6_u = {
  .u6_addr8 = {
   0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
   0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
  }
 }
};

static const struct in6_addr ipv6_addr_zero_mask = {
 .in6_u = {
  .u6_addr8 = {
   0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
  }
 }
};

/**
 * iavf_validate_fdir_fltr_masks - validate Flow Director filter fields masks
 * @adapter: pointer to the VF adapter structure
 * @fltr: Flow Director filter data structure
 *
 * Returns 0 if all masks of packet fields are either full or empty. Returns
 * error on at least one partial mask.
 */

int iavf_validate_fdir_fltr_masks(struct iavf_adapter *adapter,
      struct iavf_fdir_fltr *fltr)
{
 if (fltr->eth_mask.etype && fltr->eth_mask.etype != htons(U16_MAX))
  goto partial_mask;

 if (fltr->ip_ver == 4) {
  if (fltr->ip_mask.v4_addrs.src_ip &&
      fltr->ip_mask.v4_addrs.src_ip != htonl(U32_MAX))
   goto partial_mask;

  if (fltr->ip_mask.v4_addrs.dst_ip &&
      fltr->ip_mask.v4_addrs.dst_ip != htonl(U32_MAX))
   goto partial_mask;

  if (fltr->ip_mask.tos && fltr->ip_mask.tos != U8_MAX)
   goto partial_mask;
 } else if (fltr->ip_ver == 6) {
  if (memcmp(&fltr->ip_mask.v6_addrs.src_ip, &ipv6_addr_zero_mask,
      sizeof(struct in6_addr)) &&
      memcmp(&fltr->ip_mask.v6_addrs.src_ip, &ipv6_addr_full_mask,
      sizeof(struct in6_addr)))
   goto partial_mask;

  if (memcmp(&fltr->ip_mask.v6_addrs.dst_ip, &ipv6_addr_zero_mask,
      sizeof(struct in6_addr)) &&
      memcmp(&fltr->ip_mask.v6_addrs.dst_ip, &ipv6_addr_full_mask,
      sizeof(struct in6_addr)))
   goto partial_mask;

  if (fltr->ip_mask.tclass && fltr->ip_mask.tclass != U8_MAX)
   goto partial_mask;
 }

 if (fltr->ip_mask.proto && fltr->ip_mask.proto != U8_MAX)
  goto partial_mask;

 if (fltr->ip_mask.src_port && fltr->ip_mask.src_port != htons(U16_MAX))
  goto partial_mask;

 if (fltr->ip_mask.dst_port && fltr->ip_mask.dst_port != htons(U16_MAX))
  goto partial_mask;

 if (fltr->ip_mask.spi && fltr->ip_mask.spi != htonl(U32_MAX))
  goto partial_mask;

 if (fltr->ip_mask.l4_header &&
     fltr->ip_mask.l4_header != htonl(U32_MAX))
  goto partial_mask;

 return 0;

partial_mask:
 dev_err(&adapter->pdev->dev, "Failed to add Flow Director filter, partial masks are not supported\n");
 return -EOPNOTSUPP;
}

/**
 * iavf_pkt_udp_no_pay_len - the length of UDP packet without payload
 * @fltr: Flow Director filter data structure
 */

static u16 iavf_pkt_udp_no_pay_len(struct iavf_fdir_fltr *fltr)
{
 return sizeof(struct ethhdr) +
        (fltr->ip_ver == 4 ? sizeof(struct iphdr) : sizeof(struct ipv6hdr)) +
        sizeof(struct udphdr);
}

/**
 * iavf_fill_fdir_gtpu_hdr - fill the GTP-U protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the GTP-U protocol header is set successfully
 */

static int
iavf_fill_fdir_gtpu_hdr(struct iavf_fdir_fltr *fltr,
   struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *uhdr = &proto_hdrs->proto_hdr[proto_hdrs->count - 1];
 struct virtchnl_proto_hdr *ghdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 struct virtchnl_proto_hdr *ehdr = NULL; /* Extension Header if it exists */
 u16 adj_offs, hdr_offs;
 int i;

 VIRTCHNL_SET_PROTO_HDR_TYPE(ghdr, GTPU_IP);

 adj_offs = iavf_pkt_udp_no_pay_len(fltr);

 for (i = 0; i < fltr->flex_cnt; i++) {
#define IAVF_GTPU_HDR_TEID_OFFS0 4
#define IAVF_GTPU_HDR_TEID_OFFS1 6
#define IAVF_GTPU_HDR_N_PDU_AND_NEXT_EXTHDR_OFFS 10
#define IAVF_GTPU_HDR_NEXT_EXTHDR_TYPE_MASK  0x00FF /* skip N_PDU */
/* PDU Session Container Extension Header (PSC) */
#define IAVF_GTPU_PSC_EXTHDR_TYPE   0x85
#define IAVF_GTPU_HDR_PSC_PDU_TYPE_AND_QFI_OFFS  13
#define IAVF_GTPU_HDR_PSC_PDU_QFI_MASK   0x3F /* skip Type */
#define IAVF_GTPU_EH_QFI_IDX    1

  if (fltr->flex_words[i].offset < adj_offs)
   return -EINVAL;

  hdr_offs = fltr->flex_words[i].offset - adj_offs;

  switch (hdr_offs) {
  case IAVF_GTPU_HDR_TEID_OFFS0:
  case IAVF_GTPU_HDR_TEID_OFFS1: {
   __be16 *pay_word = (__be16 *)ghdr->buffer;

   pay_word[hdr_offs >> 1] = htons(fltr->flex_words[i].word);
   VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(ghdr, GTPU_IP, TEID);
   }
   break;
  case IAVF_GTPU_HDR_N_PDU_AND_NEXT_EXTHDR_OFFS:
   if ((fltr->flex_words[i].word &
        IAVF_GTPU_HDR_NEXT_EXTHDR_TYPE_MASK) !=
      IAVF_GTPU_PSC_EXTHDR_TYPE)
    return -EOPNOTSUPP;
   if (!ehdr)
    ehdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
   VIRTCHNL_SET_PROTO_HDR_TYPE(ehdr, GTPU_EH);
   break;
  case IAVF_GTPU_HDR_PSC_PDU_TYPE_AND_QFI_OFFS:
   if (!ehdr)
    return -EINVAL;
   ehdr->buffer[IAVF_GTPU_EH_QFI_IDX] =
     fltr->flex_words[i].word &
      IAVF_GTPU_HDR_PSC_PDU_QFI_MASK;
   VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(ehdr, GTPU_EH, QFI);
   break;
  default:
   return -EINVAL;
  }
 }

 uhdr->field_selector = 0; /* The PF ignores the UDP header fields */

 return 0;
}

/**
 * iavf_fill_fdir_pfcp_hdr - fill the PFCP protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the PFCP protocol header is set successfully
 */

static int
iavf_fill_fdir_pfcp_hdr(struct iavf_fdir_fltr *fltr,
   struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *uhdr = &proto_hdrs->proto_hdr[proto_hdrs->count - 1];
 struct virtchnl_proto_hdr *hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 u16 adj_offs, hdr_offs;
 int i;

 VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, PFCP);

 adj_offs = iavf_pkt_udp_no_pay_len(fltr);

 for (i = 0; i < fltr->flex_cnt; i++) {
#define IAVF_PFCP_HDR_SFIELD_AND_MSG_TYPE_OFFS 0
  if (fltr->flex_words[i].offset < adj_offs)
   return -EINVAL;

  hdr_offs = fltr->flex_words[i].offset - adj_offs;

  switch (hdr_offs) {
  case IAVF_PFCP_HDR_SFIELD_AND_MSG_TYPE_OFFS:
   hdr->buffer[0] = (fltr->flex_words[i].word >> 8) & 0xff;
   VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, PFCP, S_FIELD);
   break;
  default:
   return -EINVAL;
  }
 }

 uhdr->field_selector = 0; /* The PF ignores the UDP header fields */

 return 0;
}

/**
 * iavf_fill_fdir_nat_t_esp_hdr - fill the NAT-T-ESP protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the NAT-T-ESP protocol header is set successfully
 */

static int
iavf_fill_fdir_nat_t_esp_hdr(struct iavf_fdir_fltr *fltr,
        struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *uhdr = &proto_hdrs->proto_hdr[proto_hdrs->count - 1];
 struct virtchnl_proto_hdr *hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 u16 adj_offs, hdr_offs;
 u32 spi = 0;
 int i;

 VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, ESP);

 adj_offs = iavf_pkt_udp_no_pay_len(fltr);

 for (i = 0; i < fltr->flex_cnt; i++) {
#define IAVF_NAT_T_ESP_SPI_OFFS0 0
#define IAVF_NAT_T_ESP_SPI_OFFS1 2
  if (fltr->flex_words[i].offset < adj_offs)
   return -EINVAL;

  hdr_offs = fltr->flex_words[i].offset - adj_offs;

  switch (hdr_offs) {
  case IAVF_NAT_T_ESP_SPI_OFFS0:
   spi |= fltr->flex_words[i].word << 16;
   break;
  case IAVF_NAT_T_ESP_SPI_OFFS1:
   spi |= fltr->flex_words[i].word;
   break;
  default:
   return -EINVAL;
  }
 }

 if (!spi)
  return -EOPNOTSUPP; /* Not support IKE Header Format with SPI 0 */

 *(__be32 *)hdr->buffer = htonl(spi);
 VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, ESP, SPI);

 uhdr->field_selector = 0; /* The PF ignores the UDP header fields */

 return 0;
}

/**
 * iavf_fill_fdir_udp_flex_pay_hdr - fill the UDP payload header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the UDP payload defined protocol header is set successfully
 */

static int
iavf_fill_fdir_udp_flex_pay_hdr(struct iavf_fdir_fltr *fltr,
    struct virtchnl_proto_hdrs *proto_hdrs)
{
 int err;

 switch (ntohs(fltr->ip_data.dst_port)) {
 case GTPU_PORT:
  err = iavf_fill_fdir_gtpu_hdr(fltr, proto_hdrs);
  break;
 case NAT_T_ESP_PORT:
  err = iavf_fill_fdir_nat_t_esp_hdr(fltr, proto_hdrs);
  break;
 case PFCP_PORT:
  err = iavf_fill_fdir_pfcp_hdr(fltr, proto_hdrs);
  break;
 default:
  err = -EOPNOTSUPP;
  break;
 }

 return err;
}

/**
 * iavf_fill_fdir_ip4_hdr - fill the IPv4 protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the IPv4 protocol header is set successfully
 */

static int
iavf_fill_fdir_ip4_hdr(struct iavf_fdir_fltr *fltr,
         struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 struct iphdr *iph = (struct iphdr *)hdr->buffer;

 VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, IPV4);

 if (fltr->ip_mask.tos == U8_MAX) {
  iph->tos = fltr->ip_data.tos;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, IPV4, DSCP);
 }

 if (fltr->ip_mask.proto == U8_MAX) {
  iph->protocol = fltr->ip_data.proto;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, IPV4, PROT);
 }

 if (fltr->ip_mask.v4_addrs.src_ip == htonl(U32_MAX)) {
  iph->saddr = fltr->ip_data.v4_addrs.src_ip;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, IPV4, SRC);
 }

 if (fltr->ip_mask.v4_addrs.dst_ip == htonl(U32_MAX)) {
  iph->daddr = fltr->ip_data.v4_addrs.dst_ip;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, IPV4, DST);
 }

 return 0;
}

/**
 * iavf_fill_fdir_ip6_hdr - fill the IPv6 protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the IPv6 protocol header is set successfully
 */

static int
iavf_fill_fdir_ip6_hdr(struct iavf_fdir_fltr *fltr,
         struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 struct ipv6hdr *iph = (struct ipv6hdr *)hdr->buffer;

 VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, IPV6);

 if (fltr->ip_mask.tclass == U8_MAX) {
  iph->priority = (fltr->ip_data.tclass >> 4) & 0xF;
  iph->flow_lbl[0] = FIELD_PREP(0xF0, fltr->ip_data.tclass);
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, IPV6, TC);
 }

 if (fltr->ip_mask.proto == U8_MAX) {
  iph->nexthdr = fltr->ip_data.proto;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, IPV6, PROT);
 }

 if (!memcmp(&fltr->ip_mask.v6_addrs.src_ip, &ipv6_addr_full_mask,
      sizeof(struct in6_addr))) {
  memcpy(&iph->saddr, &fltr->ip_data.v6_addrs.src_ip,
         sizeof(struct in6_addr));
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, IPV6, SRC);
 }

 if (!memcmp(&fltr->ip_mask.v6_addrs.dst_ip, &ipv6_addr_full_mask,
      sizeof(struct in6_addr))) {
  memcpy(&iph->daddr, &fltr->ip_data.v6_addrs.dst_ip,
         sizeof(struct in6_addr));
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, IPV6, DST);
 }

 return 0;
}

/**
 * iavf_fill_fdir_tcp_hdr - fill the TCP protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the TCP protocol header is set successfully
 */

static int
iavf_fill_fdir_tcp_hdr(struct iavf_fdir_fltr *fltr,
         struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 struct tcphdr *tcph = (struct tcphdr *)hdr->buffer;

 VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, TCP);

 if (fltr->ip_mask.src_port == htons(U16_MAX)) {
  tcph->source = fltr->ip_data.src_port;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, TCP, SRC_PORT);
 }

 if (fltr->ip_mask.dst_port == htons(U16_MAX)) {
  tcph->dest = fltr->ip_data.dst_port;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, TCP, DST_PORT);
 }

 return 0;
}

/**
 * iavf_fill_fdir_udp_hdr - fill the UDP protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the UDP protocol header is set successfully
 */

static int
iavf_fill_fdir_udp_hdr(struct iavf_fdir_fltr *fltr,
         struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 struct udphdr *udph = (struct udphdr *)hdr->buffer;

 VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, UDP);

 if (fltr->ip_mask.src_port == htons(U16_MAX)) {
  udph->source = fltr->ip_data.src_port;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, UDP, SRC_PORT);
 }

 if (fltr->ip_mask.dst_port == htons(U16_MAX)) {
  udph->dest = fltr->ip_data.dst_port;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, UDP, DST_PORT);
 }

 if (!fltr->flex_cnt)
  return 0;

 return iavf_fill_fdir_udp_flex_pay_hdr(fltr, proto_hdrs);
}

/**
 * iavf_fill_fdir_sctp_hdr - fill the SCTP protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the SCTP protocol header is set successfully
 */

static int
iavf_fill_fdir_sctp_hdr(struct iavf_fdir_fltr *fltr,
   struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 struct sctphdr *sctph = (struct sctphdr *)hdr->buffer;

 VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, SCTP);

 if (fltr->ip_mask.src_port == htons(U16_MAX)) {
  sctph->source = fltr->ip_data.src_port;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, SCTP, SRC_PORT);
 }

 if (fltr->ip_mask.dst_port == htons(U16_MAX)) {
  sctph->dest = fltr->ip_data.dst_port;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, SCTP, DST_PORT);
 }

 return 0;
}

/**
 * iavf_fill_fdir_ah_hdr - fill the AH protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the AH protocol header is set successfully
 */

static int
iavf_fill_fdir_ah_hdr(struct iavf_fdir_fltr *fltr,
        struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 struct ip_auth_hdr *ah = (struct ip_auth_hdr *)hdr->buffer;

 VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, AH);

 if (fltr->ip_mask.spi == htonl(U32_MAX)) {
  ah->spi = fltr->ip_data.spi;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, AH, SPI);
 }

 return 0;
}

/**
 * iavf_fill_fdir_esp_hdr - fill the ESP protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the ESP protocol header is set successfully
 */

static int
iavf_fill_fdir_esp_hdr(struct iavf_fdir_fltr *fltr,
         struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 struct ip_esp_hdr *esph = (struct ip_esp_hdr *)hdr->buffer;

 VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, ESP);

 if (fltr->ip_mask.spi == htonl(U32_MAX)) {
  esph->spi = fltr->ip_data.spi;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, ESP, SPI);
 }

 return 0;
}

/**
 * iavf_fill_fdir_l4_hdr - fill the L4 protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the L4 protocol header is set successfully
 */

static int
iavf_fill_fdir_l4_hdr(struct iavf_fdir_fltr *fltr,
        struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *hdr;
 __be32 *l4_4_data;

 if (!fltr->ip_mask.proto) /* IPv4/IPv6 header only */
  return 0;

 hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 l4_4_data = (__be32 *)hdr->buffer;

 /* L2TPv3 over IP with 'Session ID' */
 if (fltr->ip_data.proto == 115 && fltr->ip_mask.l4_header == htonl(U32_MAX)) {
  VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, L2TPV3);
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, L2TPV3, SESS_ID);

  *l4_4_data = fltr->ip_data.l4_header;
 } else {
  return -EOPNOTSUPP;
 }

 return 0;
}

/**
 * iavf_fill_fdir_eth_hdr - fill the Ethernet protocol header
 * @fltr: Flow Director filter data structure
 * @proto_hdrs: Flow Director protocol headers data structure
 *
 * Returns 0 if the Ethernet protocol header is set successfully
 */

static int
iavf_fill_fdir_eth_hdr(struct iavf_fdir_fltr *fltr,
         struct virtchnl_proto_hdrs *proto_hdrs)
{
 struct virtchnl_proto_hdr *hdr = &proto_hdrs->proto_hdr[proto_hdrs->count++];
 struct ethhdr *ehdr = (struct ethhdr *)hdr->buffer;

 VIRTCHNL_SET_PROTO_HDR_TYPE(hdr, ETH);

 if (fltr->eth_mask.etype == htons(U16_MAX)) {
  if (fltr->eth_data.etype == htons(ETH_P_IP) ||
      fltr->eth_data.etype == htons(ETH_P_IPV6))
   return -EOPNOTSUPP;

  ehdr->h_proto = fltr->eth_data.etype;
  VIRTCHNL_ADD_PROTO_HDR_FIELD_BIT(hdr, ETH, ETHERTYPE);
 }

 return 0;
}

/**
 * iavf_fill_fdir_add_msg - fill the Flow Director filter into virtchnl message
 * @adapter: pointer to the VF adapter structure
 * @fltr: Flow Director filter data structure
 *
 * Returns 0 if the add Flow Director virtchnl message is filled successfully
 */

int iavf_fill_fdir_add_msg(struct iavf_adapter *adapter, struct iavf_fdir_fltr *fltr)
{
 struct virtchnl_fdir_add *vc_msg = &fltr->vc_add_msg;
 struct virtchnl_proto_hdrs *proto_hdrs;
 int err;

 proto_hdrs = &vc_msg->rule_cfg.proto_hdrs;

 err = iavf_fill_fdir_eth_hdr(fltr, proto_hdrs); /* L2 always exists */
 if (err)
  return err;

 switch (fltr->flow_type) {
 case IAVF_FDIR_FLOW_IPV4_TCP:
  err = iavf_fill_fdir_ip4_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_tcp_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV4_UDP:
  err = iavf_fill_fdir_ip4_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_udp_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV4_SCTP:
  err = iavf_fill_fdir_ip4_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_sctp_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV4_AH:
  err = iavf_fill_fdir_ip4_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_ah_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV4_ESP:
  err = iavf_fill_fdir_ip4_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_esp_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV4_OTHER:
  err = iavf_fill_fdir_ip4_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_l4_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV6_TCP:
  err = iavf_fill_fdir_ip6_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_tcp_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV6_UDP:
  err = iavf_fill_fdir_ip6_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_udp_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV6_SCTP:
  err = iavf_fill_fdir_ip6_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_sctp_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV6_AH:
  err = iavf_fill_fdir_ip6_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_ah_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV6_ESP:
  err = iavf_fill_fdir_ip6_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_esp_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_IPV6_OTHER:
  err = iavf_fill_fdir_ip6_hdr(fltr, proto_hdrs) |
        iavf_fill_fdir_l4_hdr(fltr, proto_hdrs);
  break;
 case IAVF_FDIR_FLOW_NON_IP_L2:
  break;
 default:
  err = -EINVAL;
  break;
 }

 if (err)
  return err;

 vc_msg->vsi_id = adapter->vsi.id;
 vc_msg->rule_cfg.action_set.count = 1;
 vc_msg->rule_cfg.action_set.actions[0].type = fltr->action;
 vc_msg->rule_cfg.action_set.actions[0].act_conf.queue.index = fltr->q_index;

 return 0;
}

/**
 * iavf_fdir_flow_proto_name - get the flow protocol name
 * @flow_type: Flow Director filter flow type
 **/

static const char *iavf_fdir_flow_proto_name(enum iavf_fdir_flow_type flow_type)
{
 switch (flow_type) {
 case IAVF_FDIR_FLOW_IPV4_TCP:
 case IAVF_FDIR_FLOW_IPV6_TCP:
  return "TCP";
 case IAVF_FDIR_FLOW_IPV4_UDP:
 case IAVF_FDIR_FLOW_IPV6_UDP:
  return "UDP";
 case IAVF_FDIR_FLOW_IPV4_SCTP:
 case IAVF_FDIR_FLOW_IPV6_SCTP:
  return "SCTP";
 case IAVF_FDIR_FLOW_IPV4_AH:
 case IAVF_FDIR_FLOW_IPV6_AH:
  return "AH";
 case IAVF_FDIR_FLOW_IPV4_ESP:
 case IAVF_FDIR_FLOW_IPV6_ESP:
  return "ESP";
 case IAVF_FDIR_FLOW_IPV4_OTHER:
 case IAVF_FDIR_FLOW_IPV6_OTHER:
  return "Other";
 case IAVF_FDIR_FLOW_NON_IP_L2:
  return "Ethernet";
 default:
  return NULL;
 }
}

/**
 * iavf_print_fdir_fltr
 * @adapter: adapter structure
 * @fltr: Flow Director filter to print
 *
 * Print the Flow Director filter
 **/

void iavf_print_fdir_fltr(struct iavf_adapter *adapter, struct iavf_fdir_fltr *fltr)
{
 const char *proto = iavf_fdir_flow_proto_name(fltr->flow_type);

 if (!proto)
  return;

 switch (fltr->flow_type) {
 case IAVF_FDIR_FLOW_IPV4_TCP:
 case IAVF_FDIR_FLOW_IPV4_UDP:
 case IAVF_FDIR_FLOW_IPV4_SCTP:
  dev_info(&adapter->pdev->dev, "Rule ID: %u dst_ip: %pI4 src_ip %pI4 %s: dst_port %hu src_port %hu\n",
    fltr->loc,
    &fltr->ip_data.v4_addrs.dst_ip,
    &fltr->ip_data.v4_addrs.src_ip,
    proto,
    ntohs(fltr->ip_data.dst_port),
    ntohs(fltr->ip_data.src_port));
  break;
 case IAVF_FDIR_FLOW_IPV4_AH:
 case IAVF_FDIR_FLOW_IPV4_ESP:
  dev_info(&adapter->pdev->dev, "Rule ID: %u dst_ip: %pI4 src_ip %pI4 %s: SPI %u\n",
    fltr->loc,
    &fltr->ip_data.v4_addrs.dst_ip,
    &fltr->ip_data.v4_addrs.src_ip,
    proto,
    ntohl(fltr->ip_data.spi));
  break;
 case IAVF_FDIR_FLOW_IPV4_OTHER:
  dev_info(&adapter->pdev->dev, "Rule ID: %u dst_ip: %pI4 src_ip %pI4 proto: %u L4_bytes: 0x%x\n",
    fltr->loc,
    &fltr->ip_data.v4_addrs.dst_ip,
    &fltr->ip_data.v4_addrs.src_ip,
    fltr->ip_data.proto,
    ntohl(fltr->ip_data.l4_header));
  break;
 case IAVF_FDIR_FLOW_IPV6_TCP:
 case IAVF_FDIR_FLOW_IPV6_UDP:
 case IAVF_FDIR_FLOW_IPV6_SCTP:
  dev_info(&adapter->pdev->dev, "Rule ID: %u dst_ip: %pI6 src_ip %pI6 %s: dst_port %hu src_port %hu\n",
    fltr->loc,
    &fltr->ip_data.v6_addrs.dst_ip,
    &fltr->ip_data.v6_addrs.src_ip,
    proto,
    ntohs(fltr->ip_data.dst_port),
    ntohs(fltr->ip_data.src_port));
  break;
 case IAVF_FDIR_FLOW_IPV6_AH:
 case IAVF_FDIR_FLOW_IPV6_ESP:
  dev_info(&adapter->pdev->dev, "Rule ID: %u dst_ip: %pI6 src_ip %pI6 %s: SPI %u\n",
    fltr->loc,
    &fltr->ip_data.v6_addrs.dst_ip,
    &fltr->ip_data.v6_addrs.src_ip,
    proto,
    ntohl(fltr->ip_data.spi));
  break;
 case IAVF_FDIR_FLOW_IPV6_OTHER:
  dev_info(&adapter->pdev->dev, "Rule ID: %u dst_ip: %pI6 src_ip %pI6 proto: %u L4_bytes: 0x%x\n",
    fltr->loc,
    &fltr->ip_data.v6_addrs.dst_ip,
    &fltr->ip_data.v6_addrs.src_ip,
    fltr->ip_data.proto,
    ntohl(fltr->ip_data.l4_header));
  break;
 case IAVF_FDIR_FLOW_NON_IP_L2:
  dev_info(&adapter->pdev->dev, "Rule ID: %u eth_type: 0x%x\n",
    fltr->loc,
    ntohs(fltr->eth_data.etype));
  break;
 default:
  break;
 }
}

/**
 * iavf_fdir_is_dup_fltr - test if filter is already in list
 * @adapter: pointer to the VF adapter structure
 * @fltr: Flow Director filter data structure
 *
 * Returns true if the filter is found in the list
 */

bool iavf_fdir_is_dup_fltr(struct iavf_adapter *adapter, struct iavf_fdir_fltr *fltr)
{
 struct iavf_fdir_fltr *tmp;
 bool ret = false;

 spin_lock_bh(&adapter->fdir_fltr_lock);
 list_for_each_entry(tmp, &adapter->fdir_list_head, list) {
  if (iavf_is_raw_fdir(fltr))
   continue;

  if (tmp->flow_type != fltr->flow_type)
   continue;

  if (!memcmp(&tmp->eth_data, &fltr->eth_data,
       sizeof(fltr->eth_data)) &&
      !memcmp(&tmp->ip_data, &fltr->ip_data,
       sizeof(fltr->ip_data)) &&
      !memcmp(&tmp->ext_data, &fltr->ext_data,
       sizeof(fltr->ext_data))) {
   ret = true;
   break;
  }
 }
 spin_unlock_bh(&adapter->fdir_fltr_lock);

 return ret;
}

/**
 * iavf_find_fdir_fltr - find FDIR filter
 * @adapter: pointer to the VF adapter structure
 * @is_raw: filter type, is raw (tc u32) or not (ethtool)
 * @data: data to ID the filter, type dependent
 *
 * Returns: pointer to Flow Director filter if found or NULL. Lock must be held.
 */

struct iavf_fdir_fltr *iavf_find_fdir_fltr(struct iavf_adapter *adapter,
        bool is_raw, u32 data)
{
 struct iavf_fdir_fltr *rule;

 list_for_each_entry(rule, &adapter->fdir_list_head, list) {
  if ((is_raw && rule->cls_u32_handle == data) ||
      (!is_raw && rule->loc == data))
   return rule;
 }

 return NULL;
}

/**
 * iavf_fdir_add_fltr - add a new node to the flow director filter list
 * @adapter: pointer to the VF adapter structure
 * @fltr: filter node to add to structure
 *
 * Return: 0 on success or negative errno on failure.
 */

int iavf_fdir_add_fltr(struct iavf_adapter *adapter,
         struct iavf_fdir_fltr *fltr)
{
 struct iavf_fdir_fltr *rule, *parent = NULL;

 spin_lock_bh(&adapter->fdir_fltr_lock);
 if (iavf_fdir_max_reached(adapter)) {
  spin_unlock_bh(&adapter->fdir_fltr_lock);
  dev_err(&adapter->pdev->dev,
   "Unable to add Flow Director filter (limit (%u) reached)\n",
   IAVF_MAX_FDIR_FILTERS);
  return -ENOSPC;
 }

 list_for_each_entry(rule, &adapter->fdir_list_head, list) {
  if (iavf_is_raw_fdir(fltr))
   break;

  if (rule->loc >= fltr->loc)
   break;
  parent = rule;
 }

 if (parent)
  list_add(&fltr->list, &parent->list);
 else
  list_add(&fltr->list, &adapter->fdir_list_head);

 iavf_inc_fdir_active_fltr(adapter, fltr);

 if (adapter->link_up)
  fltr->state = IAVF_FDIR_FLTR_ADD_REQUEST;
 else
  fltr->state = IAVF_FDIR_FLTR_INACTIVE;
 spin_unlock_bh(&adapter->fdir_fltr_lock);

 if (adapter->link_up)
  iavf_schedule_aq_request(adapter, IAVF_FLAG_AQ_ADD_FDIR_FILTER);

 return 0;
}

/**
 * iavf_fdir_del_fltr - delete a flow director filter from the list
 * @adapter: pointer to the VF adapter structure
 * @is_raw: filter type, is raw (tc u32) or not (ethtool)
 * @data: data to ID the filter, type dependent
 *
 * Return: 0 on success or negative errno on failure.
 */

int iavf_fdir_del_fltr(struct iavf_adapter *adapter, bool is_raw, u32 data)
{
 struct iavf_fdir_fltr *fltr = NULL;
 int err = 0;

 spin_lock_bh(&adapter->fdir_fltr_lock);
 fltr = iavf_find_fdir_fltr(adapter, is_raw, data);

 if (fltr) {
  if (fltr->state == IAVF_FDIR_FLTR_ACTIVE) {
   fltr->state = IAVF_FDIR_FLTR_DEL_REQUEST;
  } else if (fltr->state == IAVF_FDIR_FLTR_INACTIVE) {
   list_del(&fltr->list);
   iavf_dec_fdir_active_fltr(adapter, fltr);
   kfree(fltr);
   fltr = NULL;
  } else {
   err = -EBUSY;
  }
 } else if (adapter->fdir_active_fltr) {
  err = -EINVAL;
 }

 if (fltr && fltr->state == IAVF_FDIR_FLTR_DEL_REQUEST)
  iavf_schedule_aq_request(adapter, IAVF_FLAG_AQ_DEL_FDIR_FILTER);

 spin_unlock_bh(&adapter->fdir_fltr_lock);
 return err;
}

Messung V0.5
C=97 H=92 G=94

¤ Dauer der Verarbeitung: 0.7 Sekunden  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.