1. 15 Feb, 2021 19 commits
    • Vladimir Oltean's avatar
      net: dsa: felix: setup MMIO filtering rules for PTP when using tag_8021q · c8c0ba4f
      Vladimir Oltean authored
      Since the tag_8021q tagger is software-defined, it has no means by
      itself for retrieving hardware timestamps of PTP event messages.
      
      Because we do want to support PTP on ocelot even with tag_8021q, we need
      to use the CPU port module for that. The RX timestamp is present in the
      Extraction Frame Header. And because we can't use NPI mode which redirects
      the CPU queues to an "external CPU" (meaning the ARM CPU running Linux),
      then we need to poll the CPU port module through the MMIO registers to
      retrieve TX and RX timestamps.
      
      Sadly, on NXP LS1028A, the Felix switch was integrated into the SoC
      without wiring the extraction IRQ line to the ARM GIC. So, if we want to
      be notified of any PTP packets received on the CPU port module, we have
      a problem.
      
      There is a possible workaround, which is to use the Ethernet CPU port as
      a notification channel that packets are available on the CPU port module
      as well. When a PTP packet is received by the DSA tagger (without timestamp,
      of course), we go to the CPU extraction queues, poll for it there, then
      we drop the original Ethernet packet and masquerade the packet retrieved
      over MMIO (plus the timestamp) as the original when we inject it up the
      stack.
      
      Create a quirk in struct felix is selected by the Felix driver (but not
      by Seville, since that doesn't support PTP at all). We want to do this
      such that the workaround is minimally invasive for future switches that
      don't require this workaround.
      
      The only traffic for which we need timestamps is PTP traffic, so add a
      redirection rule to the CPU port module for this. Currently we only have
      the need for PTP over L2, so redirection rules for UDP ports 319 and 320
      are TBD for now.
      
      Note that for the workaround of matching of PTP-over-Ethernet-port with
      PTP-over-MMIO queues to work properly, both channels need to be
      absolutely lossless. There are two parts to achieving that:
      - We keep flow control enabled on the tag_8021q CPU port
      - We put the DSA master interface in promiscuous mode, so it will never
        drop a PTP frame (for the profiles we are interested in, these are
        sent to the multicast MAC addresses of 01-80-c2-00-00-0e and
        01-1b-19-00-00-00).
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      c8c0ba4f
    • Vladimir Oltean's avatar
      net: mscc: ocelot: refactor ocelot_xtr_irq_handler into ocelot_xtr_poll · 924ee317
      Vladimir Oltean authored
      Since the felix DSA driver will need to poll the CPU port module for
      extracted frames as well, let's create some common functions that read
      an Extraction Frame Header, and then an skb, from a CPU extraction
      group.
      
      We abuse the struct ocelot_ops :: port_to_netdev function a little bit,
      in order to retrieve the DSA port net_device or the ocelot switchdev
      net_device based on the source port information from the Extraction
      Frame Header, but it's all in the benefit of code simplification -
      netdev_alloc_skb needs it. Originally, the port_to_netdev method was
      intended for parsing act->dev from tc flower offload code.
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      924ee317
    • Vladimir Oltean's avatar
      net: dsa: tag_ocelot: create separate tagger for Seville · 7c4bb540
      Vladimir Oltean authored
      The ocelot tagger is a hot mess currently, it relies on memory
      initialized by the attached driver for basic frame transmission.
      This is against all that DSA tagging protocols stand for, which is that
      the transmission and reception of a DSA-tagged frame, the data path,
      should be independent from the switch control path, because the tag
      protocol is in principle hot-pluggable and reusable across switches
      (even if in practice it wasn't until very recently). But if another
      driver like dsa_loop wants to make use of tag_ocelot, it couldn't.
      
      This was done to have common code between Felix and Ocelot, which have
      one bit difference in the frame header format. Quoting from commit
      67c24049 ("net: dsa: felix: create a template for the DSA tags on
      xmit"):
      
          Other alternatives have been analyzed, such as:
          - Create a separate tag_seville.c: too much code duplication for just 1
            bit field difference.
          - Create a separate DSA_TAG_PROTO_SEVILLE under tag_ocelot.c, just like
            tag_brcm.c, which would have a separate .xmit function. Again, too
            much code duplication for just 1 bit field difference.
          - Allocate the template from the init function of the tag_ocelot.c
            module, instead of from the driver: couldn't figure out a method of
            accessing the correct port template corresponding to the correct
            tagger in the .xmit function.
      
      The really interesting part is that Seville should have had its own
      tagging protocol defined - it is not compatible on the wire with Ocelot,
      even for that single bit. In principle, a packet generated by
      DSA_TAG_PROTO_OCELOT when booted on NXP LS1028A would look in a certain
      way, but when booted on NXP T1040 it would look differently. The reverse
      is also true: a packet generated by a Seville switch would be
      interpreted incorrectly by Wireshark if it was told it was generated by
      an Ocelot switch.
      
      Actually things are a bit more nuanced. If we concentrate only on the
      DSA tag, what I said above is true, but Ocelot/Seville also support an
      optional DSA tag prefix, which can be short or long, and it is possible
      to distinguish the two taggers based on an integer constant put in that
      prefix. Nonetheless, creating a separate tagger is still justified,
      since the tag prefix is optional, and without it, there is again no way
      to distinguish.
      
      Claiming backwards binary compatibility is a bit more tough, since I've
      already changed the format of tag_ocelot once, in commit 5124197c
      ("net: dsa: tag_ocelot: use a short prefix on both ingress and egress").
      Therefore I am not very concerned with treating this as a bugfix and
      backporting it to stable kernels (which would be another mess due to the
      fact that there would be lots of conflicts with the other DSA_TAG_PROTO*
      definitions). It's just simpler to say that the string values of the
      taggers have ABI value starting with kernel 5.12, which will be when the
      changing of tag protocol via /sys/class/net/<dsa-master>/dsa/tagging
      goes live.
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      7c4bb540
    • Vladimir Oltean's avatar
      net: dsa: tag_ocelot: single out PTP-related transmit tag processing · 62bf5fde
      Vladimir Oltean authored
      There is one place where we cannot avoid accessing driver data, and that
      is 2-step PTP TX timestamping, since the switch wants us to provide a
      timestamp request ID through the injection header, which naturally must
      come from a sequence number kept by the driver (it is generated by the
      .port_txtstamp method prior to the tagger's xmit).
      
      However, since other drivers like dsa_loop do not claim PTP support
      anyway, the DSA_SKB_CB(skb)->clone will always be NULL anyway, so if we
      move all PTP-related dereferences of struct ocelot and struct ocelot_port
      into a separate function, we can effectively ensure that this is dead
      code when the ocelot tagger is attached to non-ocelot switches, and the
      stateful portion of the tagger is more self-contained.
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      62bf5fde
    • Vladimir Oltean's avatar
      net: mscc: ocelot: use common tag parsing code with DSA · 40d3f295
      Vladimir Oltean authored
      The Injection Frame Header and Extraction Frame Header that the switch
      prepends to frames over the NPI port is also prepended to frames
      delivered over the CPU port module's queues.
      
      Let's unify the handling of the frame headers by making the ocelot
      driver call some helpers exported by the DSA tagger. Among other things,
      this allows us to get rid of the strange cpu_to_be32 when transmitting
      the Injection Frame Header on ocelot, since the packing API uses
      network byte order natively (when "quirks" is 0).
      
      The comments above ocelot_gen_ifh talk about setting pop_cnt to 3, and
      the cpu extraction queue mask to something, but the code doesn't do it,
      so we don't do it either.
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      40d3f295
    • Vladimir Oltean's avatar
      net: dsa: tag_ocelot: avoid accessing ds->priv in ocelot_rcv · 8a678bb2
      Vladimir Oltean authored
      Taggers should be written to do something valid irrespective of the
      switch driver that they are attached to. This is even more true now,
      because since the introduction of the .change_tag_protocol method, a
      certain tagger is not necessarily strictly associated with a driver any
      longer, and I would like to be able to test all taggers with dsa_loop in
      the future.
      
      In the case of ocelot, it needs to move the classified VLAN from the DSA
      tag into the skb if the port is VLAN-aware. We can allow it to do that
      by looking at the dp->vlan_filtering property, no need to invoke
      structures which are specific to ocelot.
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      8a678bb2
    • Vladimir Oltean's avatar
      net: mscc: ocelot: refactor ocelot_port_inject_frame out of ocelot_port_xmit · 137ffbc4
      Vladimir Oltean authored
      The felix DSA driver will inject some frames through register MMIO, same
      as ocelot switchdev currently does. So we need to be able to reuse the
      common code.
      
      Also create some shim definitions, since the DSA tagger can be compiled
      without support for the switch driver.
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      137ffbc4
    • Vladimir Oltean's avatar
      net: mscc: ocelot: use DIV_ROUND_UP helper in ocelot_port_inject_frame · 5f016f42
      Vladimir Oltean authored
      This looks a bit nicer than the open-coded "(x + 3) % 4" idiom.
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      5f016f42
    • Vladimir Oltean's avatar
      net: mscc: ocelot: better error handling in ocelot_xtr_irq_handler · a94306ce
      Vladimir Oltean authored
      The ocelot_rx_frame_word() function can return a negative error code,
      however this isn't being checked for consistently. Errors being ignored
      have not been seen in practice though.
      
      Also, some constructs can be simplified by using "goto" instead of
      repeated "break" statements.
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      a94306ce
    • Vladimir Oltean's avatar
      net: mscc: ocelot: only drain extraction queue on error · d7795f8f
      Vladimir Oltean authored
      It appears that the intention of this snippet of code is to not exit
      ocelot_xtr_irq_handler() while in the middle of extracting a frame.
      The problem in extracting it word by word is that future extraction
      attempts are really easy to get desynchronized, since the IRQ handler
      assumes that the first 16 bytes are the IFH, which give further
      information about the frame, such as frame length.
      
      But during normal operation, "err" will not be 0, but 4, set from here:
      
      		for (i = 0; i < OCELOT_TAG_LEN / 4; i++) {
      			err = ocelot_rx_frame_word(ocelot, grp, true, &ifh[i]);
      			if (err != 4)
      				break;
      		}
      
      		if (err != 4)
      			break;
      
      In that case, draining the extraction queue is a no-op. So explicitly
      make this code execute only on negative err.
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      d7795f8f
    • Vladimir Oltean's avatar
      net: mscc: ocelot: stop returning IRQ_NONE in ocelot_xtr_irq_handler · f833ca29
      Vladimir Oltean authored
      Since the xtr (extraction) IRQ of the ocelot switch is not shared, then
      if it fired, it means that some data must be present in the queues of
      the CPU port module. So simplify the code.
      Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
      Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      f833ca29
    • David S. Miller's avatar
      Merge branch 'bnxt_en-next' · 14026192
      David S. Miller authored
      Michael Chan says:
      
      ====================
      bnxt_en: Error recovery optimizations.
      
      This series implements some optimizations to error recovery.  One
      patch adds an echo/reply mechanism with firmware to enhance error
      detection.  The other patches speed up the recovery process by
      polling config space earlier and to selectively initialize
      context memory during re-initialization.
      ====================
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      14026192
    • Michael Chan's avatar
      bnxt_en: Improve logging of error recovery settings information. · f4d95c3c
      Michael Chan authored
      We currently only log the error recovery settings if it is enabled.
      In some cases, firmware disables error recovery after it was
      initially enabled.  Without logging anything, the user will not be
      aware of this change in setting.
      
      Log it when error recovery is disabled.  Also, change the reset count
      value from hexadecimal to decimal.
      Reviewed-by: default avatarEdwin Peer <edwin.peer@broadcom.com>
      Reviewed-by: default avatarPavan Chebbi <pavan.chebbi@broadcom.com>
      Signed-off-by: default avatarMichael Chan <michael.chan@broadcom.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      f4d95c3c
    • Michael Chan's avatar
      bnxt_en: Reply to firmware's echo request async message. · df97b34d
      Michael Chan authored
      This is a new async message that the firmware can send to check if it
      can communicate with the driver.  This is an added error detection
      scheme that firmware can use if it suspects errors in the PCIe
      interface.  When the driver receives this async message, it will reply
      back echoing some data in the async message.  If the firmware is not
      getting the reply with the proper data after some retries, error
      recovery will kick in.
      Reviewed-by: default avatarAndy Gospodarek <gospo@broadcom.com>
      Reviewed-by: default avatarEdwin Peer <edwin.peer@broadcom.com>
      Reviewed-by: default avatarVasundhara Volam <vasundhara-v.volam@broadcom.com>
      Signed-off-by: default avatarMichael Chan <michael.chan@broadcom.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      df97b34d
    • Michael Chan's avatar
      bnxt_en: Initialize "context kind" field for context memory blocks. · 41435c39
      Michael Chan authored
      If firmware provides the offset to the "context kind" field of the
      relevant context memory blocks, we'll initialize just that field for
      each block instead of initializing all of context memory.
      
      Populate the bnxt_mem_init structure with the proper offset returned
      by firmware.  If it is older firmware and the information is not
      available, we set the offset to an invalid value and fall back to
      the old behavior of initializing every byte.  Otherwise, we initialize
      only the "context kind" byte at the offset.
      Reviewed-by: default avatarEdwin Peer <edwin.peer@broadcom.com>
      Signed-off-by: default avatarMichael Chan <michael.chan@broadcom.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      41435c39
    • Michael Chan's avatar
      bnxt_en: Add context memory initialization infrastructure. · e9696ff3
      Michael Chan authored
      Currently, the driver calls memset() to set all relevant context memory
      used by the chip to the initial value.  This can take many milliseconds
      with the potentially large number of context pages allocated for the
      chip.
      
      To make this faster, we only need to initialize the "context kind" field
      of each block of context memory.  This patch sets up the infrastructure
      to do that with the bnxt_mem_init structure.  In the next patch, we'll
      add the logic to obtain the offset of the "context kind" from the
      firmware.  This patch is not changing the current behavior of calling
      memset() to initialize all relevant context memory.
      Reviewed-by: default avatarPavan Chebbi <pavan.chebbi@broadcom.com>
      Reviewed-by: default avatarEdwin Peer <edwin.peer@broadcom.com>
      Signed-off-by: default avatarMichael Chan <michael.chan@broadcom.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      e9696ff3
    • Michael Chan's avatar
      bnxt_en: Implement faster recovery for firmware fatal error. · dab62e7c
      Michael Chan authored
      During some fatal firmware error conditions, the PCI config space
      register 0x2e which normally contains the subsystem ID will become
      0xffff.  This register will revert back to the normal value after
      the chip has completed core reset.  If we detect this condition,
      we can poll this config register immediately for the value to revert.
      Because we use config read cycles to poll this register, there is no
      possibility of Master Abort if we happen to read it during core reset.
      This speeds up recovery significantly as we don't have to wait for the
      conservative min_time before polling MMIO to see if the firmware has
      come out of reset.  As soon as this register changes value we can
      proceed to re-initialize the device.
      Reviewed-by: default avatarEdwin Peer <edwin.peer@broadcom.com>
      Reviewed-by: default avatarVasundhara Volam <vasundhara-v.volam@broadcom.com>
      Reviewed-by: default avatarAndy Gospodarek <gospo@broadcom.com>
      Signed-off-by: default avatarMichael Chan <michael.chan@broadcom.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      dab62e7c
    • Edwin Peer's avatar
      bnxt_en: selectively allocate context memories · be6d755f
      Edwin Peer authored
      Newer devices may have local context memory instead of relying on the
      host for backing store. In these cases, HWRM_FUNC_BACKING_STORE_QCAPS
      will return a zero entry size to indicate contexts for which the host
      should not allocate backing store.
      
      Selectively allocate context memory based on device capabilities and
      only enable backing store for the appropriate contexts.
      Signed-off-by: default avatarEdwin Peer <edwin.peer@broadcom.com>
      Signed-off-by: default avatarMichael Chan <michael.chan@broadcom.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      be6d755f
    • Michael Chan's avatar
      bnxt_en: Update firmware interface spec to 1.10.2.16. · 31f67c2e
      Michael Chan authored
      The main changes are the echo request/response from firmware for error
      detection and the NO_FCS feature to transmit frames without FCS.
      Signed-off-by: default avatarMichael Chan <michael.chan@broadcom.com>
      Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
      31f67c2e
  2. 13 Feb, 2021 21 commits