1. 28 6月, 2021 1 次提交
  2. 25 6月, 2021 20 次提交
  3. 24 6月, 2021 5 次提交
    • M
      bpf: Support all gso types in bpf_skb_change_proto() · 0bc919d3
      Maciej Żenczykowski 提交于
      Since we no longer modify gso_size, it is now theoretically
      safe to not set SKB_GSO_DODGY and reset gso_segs to zero.
      
      This also means the skb_is_gso_tcp() check should no longer
      be necessary.
      
      Unfortunately we cannot remove the skb_{decrease,increase}_gso_size()
      helpers, as they are still used elsewhere:
      
        bpf_skb_net_grow() without BPF_F_ADJ_ROOM_FIXED_GSO
        bpf_skb_net_shrink() without BPF_F_ADJ_ROOM_FIXED_GSO
        net/core/lwt_bpf.c's handle_gso_type()
      Signed-off-by: NMaciej Żenczykowski <maze@google.com>
      Signed-off-by: NDaniel Borkmann <daniel@iogearbox.net>
      Cc: Dongseok Yi <dseok.yi@samsung.com>
      Cc: Willem de Bruijn <willemb@google.com>
      Link: https://lore.kernel.org/bpf/20210617000953.2787453-3-zenczykowski@gmail.com
      0bc919d3
    • M
      bpf: Do not change gso_size during bpf_skb_change_proto() · 364745fb
      Maciej Żenczykowski 提交于
      This is technically a backwards incompatible change in behaviour, but I'm
      going to argue that it is very unlikely to break things, and likely to fix
      *far* more then it breaks.
      
      In no particular order, various reasons follow:
      
      (a) I've long had a bug assigned to myself to debug a super rare kernel crash
      on Android Pixel phones which can (per stacktrace) be traced back to BPF clat
      IPv6 to IPv4 protocol conversion causing some sort of ugly failure much later
      on during transmit deep in the GSO engine, AFAICT precisely because of this
      change to gso_size, though I've never been able to manually reproduce it. I
      believe it may be related to the particular network offload support of attached
      USB ethernet dongle being used for tethering off of an IPv6-only cellular
      connection. The reason might be we end up with more segments than max permitted,
      or with a GSO packet with only one segment... (either way we break some
      assumption and hit a BUG_ON)
      
      (b) There is no check that the gso_size is > 20 when reducing it by 20, so we
      might end up with a negative (or underflowing) gso_size or a gso_size of 0.
      This can't possibly be good. Indeed this is probably somehow exploitable (or
      at least can result in a kernel crash) by delivering crafted packets and perhaps
      triggering an infinite loop or a divide by zero... As a reminder: gso_size (MSS)
      is related to MTU, but not directly derived from it: gso_size/MSS may be
      significantly smaller then one would get by deriving from local MTU. And on
      some NICs (which do loose MTU checking on receive, it may even potentially be
      larger, for example my work pc with 1500 MTU can receive 1520 byte frames [and
      sometimes does due to bugs in a vendor plat46 implementation]). Indeed even just
      going from 21 to 1 is potentially problematic because it increases the number
      of segments by a factor of 21 (think DoS, or some other crash due to too many
      segments).
      
      (c) It's always safe to not increase the gso_size, because it doesn't result in
      the max packet size increasing.  So the skb_increase_gso_size() call was always
      unnecessary for correctness (and outright undesirable, see later). As such the
      only part which is potentially dangerous (ie. could cause backwards compatibility
      issues) is the removal of the skb_decrease_gso_size() call.
      
      (d) If the packets are ultimately destined to the local device, then there is
      absolutely no benefit to playing around with gso_size. It only matters if the
      packets will egress the device. ie. we're either forwarding, or transmitting
      from the device.
      
      (e) This logic only triggers for packets which are GSO. It does not trigger for
      skbs which are not GSO. It will not convert a non-GSO MTU sized packet into a
      GSO packet (and you don't even know what the MTU is, so you can't even fix it).
      As such your transmit path must *already* be able to handle an MTU 20 bytes
      larger then your receive path (for IPv4 to IPv6 translation) - and indeed 28
      bytes larger due to IPv4 fragments. Thus removing the skb_decrease_gso_size()
      call doesn't actually increase the size of the packets your transmit side must
      be able to handle. ie. to handle non-GSO max-MTU packets, the IPv4/IPv6 device/
      route MTUs must already be set correctly. Since for example with an IPv4 egress
      MTU of 1500, IPv4 to IPv6 translation will already build 1520 byte IPv6 frames,
      so you need a 1520 byte device MTU. This means if your IPv6 device's egress
      MTU is 1280, your IPv4 route must be 1260 (and actually 1252, because of the
      need to handle fragments). This is to handle normal non-GSO packets. Thus the
      reduction is simply not needed for GSO packets, because when they're correctly
      built, they will already be the right size.
      
      (f) TSO/GSO should be able to exactly undo GRO: the number of packets (TCP
      segments) should not be modified, so that TCP's MSS counting works correctly
      (this matters for congestion control). If protocol conversion changes the
      gso_size, then the number of TCP segments may increase or decrease. Packet loss
      after protocol conversion can result in partial loss of MSS segments that the
      sender sent. How's the sending TCP stack going to react to receiving ACKs/SACKs
      in the middle of the segments it sent?
      
      (g) skb_{decrease,increase}_gso_size() are already no-ops for GSO_BY_FRAGS
      case (besides triggering WARN_ON_ONCE). This means you already cannot guarantee
      that gso_size (and thus resulting packet MTU) is changed. ie. you must assume
      it won't be changed.
      
      (h) changing gso_size is outright buggy for UDP GSO packets, where framing
      matters (I believe that's also the case for SCTP, but it's already excluded
      by [g]).  So the only remaining case is TCP, which also doesn't want it
      (see [f]).
      
      (i) see also the reasoning on the previous attempt at fixing this
      (commit fa7b83bf) which shows that the current
      behaviour causes TCP packet loss:
      
        In the forwarding path GRO -> BPF 6 to 4 -> GSO for TCP traffic, the
        coalesced packet payload can be > MSS, but < MSS + 20.
      
        bpf_skb_proto_6_to_4() will upgrade the MSS and it can be > the payload
        length. After then tcp_gso_segment checks for the payload length if it
        is <= MSS. The condition is causing the packet to be dropped.
      
        tcp_gso_segment():
          [...]
          mss = skb_shinfo(skb)->gso_size;
          if (unlikely(skb->len <= mss)) goto out;
          [...]
      
      Thus changing the gso_size is simply a very bad idea. Increasing is unnecessary
      and buggy, and decreasing can go negative.
      
      Fixes: 6578171a ("bpf: add bpf_skb_change_proto helper")
      Signed-off-by: NMaciej Żenczykowski <maze@google.com>
      Signed-off-by: NDaniel Borkmann <daniel@iogearbox.net>
      Cc: Dongseok Yi <dseok.yi@samsung.com>
      Cc: Willem de Bruijn <willemb@google.com>
      Link: https://lore.kernel.org/bpf/CANP3RGfjLikQ6dg=YpBU0OeHvyv7JOki7CyOUS9modaXAi-9vQ@mail.gmail.com
      Link: https://lore.kernel.org/bpf/20210617000953.2787453-2-zenczykowski@gmail.com
      364745fb
    • M
      Revert "bpf: Check for BPF_F_ADJ_ROOM_FIXED_GSO when bpf_skb_change_proto" · ba47396e
      Maciej Żenczykowski 提交于
      This reverts commit fa7b83bf.
      
      See the followup commit for the reasoning why I believe the appropriate
      approach is to simply make this change without a flag, but it can basically
      be summarized as using this helper without the flag is bug-prone or outright
      buggy, and thus the default should be this new behaviour.
      
      As this commit has only made it into net-next/master, but not into
      any real release, such a backwards incompatible change is still ok.
      Signed-off-by: NMaciej Żenczykowski <maze@google.com>
      Signed-off-by: NDaniel Borkmann <daniel@iogearbox.net>
      Cc: Dongseok Yi <dseok.yi@samsung.com>
      Cc: Willem de Bruijn <willemb@google.com>
      Link: https://lore.kernel.org/bpf/20210617000953.2787453-1-zenczykowski@gmail.com
      ba47396e
    • S
      media, bpf: Do not copy more entries than user space requested · 647d446d
      Sean Young 提交于
      The syscall bpf(BPF_PROG_QUERY, &attr) should use the prog_cnt field to
      see how many entries user space provided and return ENOSPC if there are
      more programs than that. Before this patch, this is not checked and
      ENOSPC is never returned.
      
      Note that one lirc device is limited to 64 bpf programs, and user space
      I'm aware of -- ir-keytable -- always gives enough space for 64 entries
      already. However, we should not copy program ids than are requested.
      Signed-off-by: NSean Young <sean@mess.org>
      Signed-off-by: NDaniel Borkmann <daniel@iogearbox.net>
      Link: https://lore.kernel.org/bpf/20210623213754.632-1-sean@mess.org
      647d446d
    • J
      bpf, x86: Remove unused cnt increase from EMIT macro · ced50fc4
      Jiri Olsa 提交于
      Removing unused cnt increase from EMIT macro together with cnt declarations.
      This was introduced in commit [1] to ensure proper code generation. But that
      code was removed in commit [2] and this extra code was left in.
      
        [1] b52f00e6 ("x86: bpf_jit: implement bpf_tail_call() helper")
        [2] ebf7d1f5 ("bpf, x64: rework pro/epilogue and tailcall handling in JIT")
      Signed-off-by: NJiri Olsa <jolsa@redhat.com>
      Signed-off-by: NDaniel Borkmann <daniel@iogearbox.net>
      Link: https://lore.kernel.org/bpf/20210623112504.709856-1-jolsa@kernel.org
      ced50fc4
  4. 23 6月, 2021 1 次提交
  5. 22 6月, 2021 2 次提交
    • K
      libbpf: Switch to void * casting in netlink helpers · ee62a5c6
      Kumar Kartikeya Dwivedi 提交于
      Netlink helpers I added in 8bbb77b7 ("libbpf: Add various netlink
      helpers") used char * casts everywhere, and there were a few more that
      existed from before.
      
      Convert all of them to void * cast, as it is treated equivalently by
      clang/gcc for the purposes of pointer arithmetic and to follow the
      convention elsewhere in the kernel/libbpf.
      Signed-off-by: NKumar Kartikeya Dwivedi <memxor@gmail.com>
      Signed-off-by: NDaniel Borkmann <daniel@iogearbox.net>
      Link: https://lore.kernel.org/bpf/20210619041454.417577-2-memxor@gmail.com
      ee62a5c6
    • K
      libbpf: Add request buffer type for netlink messages · 0ae64fb6
      Kumar Kartikeya Dwivedi 提交于
      Coverity complains about OOB writes to nlmsghdr. There is no OOB as we
      write to the trailing buffer, but static analyzers and compilers may
      rightfully be confused as the nlmsghdr pointer has subobject provenance
      (and hence subobject bounds).
      
      Fix this by using an explicit request structure containing the nlmsghdr,
      struct tcmsg/ifinfomsg, and attribute buffer.
      
      Also switch nh_tail (renamed to req_tail) to cast req * to char * so
      that it can be understood as arithmetic on pointer to the representation
      array (hence having same bound as request structure), which should
      further appease analyzers.
      
      As a bonus, callers don't have to pass sizeof(req) all the time now, as
      size is implicitly obtained using the pointer. While at it, also reduce
      the size of attribute buffer to 128 bytes (132 for ifinfomsg using
      functions due to the padding).
      
      Summary of problem:
      
        Even though C standard allows interconvertibility of pointer to first
        member and pointer to struct, for the purposes of alias analysis it
        would still consider the first as having pointer value "pointer to T"
        where T is type of first member hence having subobject bounds,
        allowing analyzers within reason to complain when object is accessed
        beyond the size of pointed to object.
      
        The only exception to this rule may be when a char * is formed to a
        member subobject. It is not possible for the compiler to be able to
        tell the intent of the programmer that it is a pointer to member
        object or the underlying representation array of the containing
        object, so such diagnosis is suppressed.
      
      Fixes: 715c5ce4 ("libbpf: Add low level TC-BPF management API")
      Signed-off-by: NKumar Kartikeya Dwivedi <memxor@gmail.com>
      Signed-off-by: NDaniel Borkmann <daniel@iogearbox.net>
      Link: https://lore.kernel.org/bpf/20210619041454.417577-1-memxor@gmail.com
      0ae64fb6
  6. 21 6月, 2021 1 次提交
  7. 19 6月, 2021 3 次提交
  8. 18 6月, 2021 7 次提交
新手
引导
客服 返回
顶部