Martin KaFai Lau wrote:
> This patch attaches a classifier prog to the ingress filter.
> It exercises the following helpers with different socket pointer
> types in different logical branches:
> 1. bpf_sk_release()
> 2. bpf_sk_assign()
> 3. bpf_skc_to_tcp_request_sock(), bpf_skc_to_tcp_sock()
> 4. bpf_tcp_gen_syncookie, bpf_tcp_check_syncookie
> 
> Signed-off-by: Martin KaFai Lau <ka...@fb.com>
> ---
>  tools/testing/selftests/bpf/bpf_tcp_helpers.h |   5 +
>  .../bpf/prog_tests/btf_skc_cls_ingress.c      | 234 ++++++++++++++++++
>  .../bpf/progs/test_btf_skc_cls_ingress.c      | 174 +++++++++++++
>  3 files changed, 413 insertions(+)
>  create mode 100644 
> tools/testing/selftests/bpf/prog_tests/btf_skc_cls_ingress.c
>  create mode 100644 
> tools/testing/selftests/bpf/progs/test_btf_skc_cls_ingress.c
> 


Hi Martin,

One piece I'm missing is how does this handle null pointer dereferences
from network side when reading BTF objects? I've not got through all the
code yet so maybe I'm just not there yet.

For example,

> diff --git a/tools/testing/selftests/bpf/bpf_tcp_helpers.h 
> b/tools/testing/selftests/bpf/bpf_tcp_helpers.h
> index a0e8b3758bd7..2915664c335d 100644
> --- a/tools/testing/selftests/bpf/bpf_tcp_helpers.h
> +++ b/tools/testing/selftests/bpf/bpf_tcp_helpers.h
> @@ -16,6 +16,7 @@ BPF_PROG(name, args)
>  
>  struct sock_common {
>       unsigned char   skc_state;
> +     __u16           skc_num;
>  } __attribute__((preserve_access_index));
>  
>  enum sk_pacing {
> @@ -45,6 +46,10 @@ struct inet_connection_sock {
>       __u64                     icsk_ca_priv[104 / sizeof(__u64)];
>  } __attribute__((preserve_access_index));
>  
> +struct request_sock {
> +     struct sock_common              __req_common;
> +} __attribute__((preserve_access_index));
> +
>  struct tcp_sock {
>       struct inet_connection_sock     inet_conn;

add some pointer from tcp_sock which is likely not set so should be NULL,

        struct tcp_fastopen_request *fastopen_req;

[...]

> +     if (bpf_skc->state == BPF_TCP_NEW_SYN_RECV) {
> +             struct request_sock *req_sk;
> +
> +             req_sk = (struct request_sock 
> *)bpf_skc_to_tcp_request_sock(bpf_skc);
> +             if (!req_sk) {
> +                     LOG();
> +                     goto release;
> +             }
> +
> +             if (bpf_sk_assign(skb, req_sk, 0)) {
> +                     LOG();
> +                     goto release;
> +             }
> +
> +             req_sk_sport = req_sk->__req_common.skc_num;
> +
> +             bpf_sk_release(req_sk);
> +             return TC_ACT_OK;
> +     } else if (bpf_skc->state == BPF_TCP_LISTEN) {
> +             struct tcp_sock *tp;
> +
> +             tp = bpf_skc_to_tcp_sock(bpf_skc);
> +             if (!tp) {
> +                     LOG();
> +                     goto release;
> +             }
> +
> +             if (bpf_sk_assign(skb, tp, 0)) {
> +                     LOG();
> +                     goto release;
> +             }
> +


Then use it here without a null check in the BPF program,

                fastopen = tp->fastopen_req;
                if (fastopen->size > 0x1234)
                      (do something)

> +             listen_tp_sport = 
> tp->inet_conn.icsk_inet.sk.__sk_common.skc_num;
> +
> +             test_syncookie_helper(ip6h, th, tp, skb);
> +             bpf_sk_release(tp);
> +             return TC_ACT_OK;
> +     }

My quick check shows this didn't actually fault and the xlated
looks like it did the read and dereference. Must be missing
something? We shouldn't have fault_handler set for cls_ingress.

Perhaps a comment in the cover letter about this would be
helpful? Or if I'm just being dense this morning let me know
as well. ;)

> +
> +     if (bpf_sk_assign(skb, bpf_skc, 0))
> +             LOG();
> +
> +release:
> +     bpf_sk_release(bpf_skc);
> +     return TC_ACT_OK;
> +}

Reply via email to