1*387f9dfdSAndroid Build Coastguard Worker /* 2*387f9dfdSAndroid Build Coastguard Worker * Copyright (c) 2020 Facebook, Inc. 3*387f9dfdSAndroid Build Coastguard Worker * 4*387f9dfdSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*387f9dfdSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*387f9dfdSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*387f9dfdSAndroid Build Coastguard Worker * 8*387f9dfdSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*387f9dfdSAndroid Build Coastguard Worker * 10*387f9dfdSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*387f9dfdSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*387f9dfdSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*387f9dfdSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*387f9dfdSAndroid Build Coastguard Worker * limitations under the License. 15*387f9dfdSAndroid Build Coastguard Worker */ 16*387f9dfdSAndroid Build Coastguard Worker 17*387f9dfdSAndroid Build Coastguard Worker #include <linux/version.h> 18*387f9dfdSAndroid Build Coastguard Worker #include <unistd.h> 19*387f9dfdSAndroid Build Coastguard Worker #include <sys/types.h> 20*387f9dfdSAndroid Build Coastguard Worker #include <sys/socket.h> 21*387f9dfdSAndroid Build Coastguard Worker #include <string> 22*387f9dfdSAndroid Build Coastguard Worker 23*387f9dfdSAndroid Build Coastguard Worker #include "BPF.h" 24*387f9dfdSAndroid Build Coastguard Worker #include "catch.hpp" 25*387f9dfdSAndroid Build Coastguard Worker 26*387f9dfdSAndroid Build Coastguard Worker #if LINUX_VERSION_CODE >= KERNEL_VERSION(4, 18, 0) 27*387f9dfdSAndroid Build Coastguard Worker 28*387f9dfdSAndroid Build Coastguard Worker // Prior to 5.15, the socket must be TCP established socket to be updatable. 29*387f9dfdSAndroid Build Coastguard Worker // https://git.kernel.org/pub/scm/linux/kernel/git/bpf/bpf-next.git/commit/?id=0c48eefae712c2fd91480346a07a1a9cd0f9470b 30*387f9dfdSAndroid Build Coastguard Worker #if LINUX_VERSION_CODE < KERNEL_VERSION(5, 15, 0) 31*387f9dfdSAndroid Build Coastguard Worker bool expected_update_result = false; 32*387f9dfdSAndroid Build Coastguard Worker #else 33*387f9dfdSAndroid Build Coastguard Worker bool expected_update_result = true; 34*387f9dfdSAndroid Build Coastguard Worker #endif 35*387f9dfdSAndroid Build Coastguard Worker 36*387f9dfdSAndroid Build Coastguard Worker TEST_CASE("test sock map", "[sockmap]") { 37*387f9dfdSAndroid Build Coastguard Worker { 38*387f9dfdSAndroid Build Coastguard Worker const std::string BPF_PROGRAM = R"( 39*387f9dfdSAndroid Build Coastguard Worker BPF_SOCKMAP(sk_map1, 10); 40*387f9dfdSAndroid Build Coastguard Worker BPF_SOCKMAP(sk_map2, 10); 41*387f9dfdSAndroid Build Coastguard Worker int test(struct bpf_sock_ops *skops) 42*387f9dfdSAndroid Build Coastguard Worker { 43*387f9dfdSAndroid Build Coastguard Worker u32 key = 0, val = 0; 44*387f9dfdSAndroid Build Coastguard Worker 45*387f9dfdSAndroid Build Coastguard Worker sk_map2.update(&key, &val); 46*387f9dfdSAndroid Build Coastguard Worker sk_map2.delete(&key); 47*387f9dfdSAndroid Build Coastguard Worker sk_map2.sock_map_update(skops, &key, 0); 48*387f9dfdSAndroid Build Coastguard Worker 49*387f9dfdSAndroid Build Coastguard Worker return 0; 50*387f9dfdSAndroid Build Coastguard Worker } 51*387f9dfdSAndroid Build Coastguard Worker )"; 52*387f9dfdSAndroid Build Coastguard Worker 53*387f9dfdSAndroid Build Coastguard Worker // make sure program is loaded successfully 54*387f9dfdSAndroid Build Coastguard Worker ebpf::BPF bpf; 55*387f9dfdSAndroid Build Coastguard Worker ebpf::StatusTuple res(0); 56*387f9dfdSAndroid Build Coastguard Worker res = bpf.init(BPF_PROGRAM); 57*387f9dfdSAndroid Build Coastguard Worker REQUIRE(res.ok()); 58*387f9dfdSAndroid Build Coastguard Worker 59*387f9dfdSAndroid Build Coastguard Worker // create a udp socket so we can do some map operations. 60*387f9dfdSAndroid Build Coastguard Worker int sockfd = socket(AF_INET, SOCK_DGRAM, 0); 61*387f9dfdSAndroid Build Coastguard Worker REQUIRE(sockfd >= 0); 62*387f9dfdSAndroid Build Coastguard Worker 63*387f9dfdSAndroid Build Coastguard Worker auto sk_map = bpf.get_sockmap_table("sk_map1"); 64*387f9dfdSAndroid Build Coastguard Worker int key = 0, val = sockfd; 65*387f9dfdSAndroid Build Coastguard Worker 66*387f9dfdSAndroid Build Coastguard Worker res = sk_map.remove_value(key); 67*387f9dfdSAndroid Build Coastguard Worker REQUIRE(!res.ok()); 68*387f9dfdSAndroid Build Coastguard Worker 69*387f9dfdSAndroid Build Coastguard Worker res = sk_map.update_value(key, val); 70*387f9dfdSAndroid Build Coastguard Worker REQUIRE(res.ok() == expected_update_result); 71*387f9dfdSAndroid Build Coastguard Worker } 72*387f9dfdSAndroid Build Coastguard Worker } 73*387f9dfdSAndroid Build Coastguard Worker 74*387f9dfdSAndroid Build Coastguard Worker TEST_CASE("test sock hash", "[sockhash]") { 75*387f9dfdSAndroid Build Coastguard Worker { 76*387f9dfdSAndroid Build Coastguard Worker const std::string BPF_PROGRAM = R"( 77*387f9dfdSAndroid Build Coastguard Worker BPF_SOCKHASH(sk_hash1, u32, 10); 78*387f9dfdSAndroid Build Coastguard Worker BPF_SOCKHASH(sk_hash2, u32, 10); 79*387f9dfdSAndroid Build Coastguard Worker int test(struct bpf_sock_ops *skops) 80*387f9dfdSAndroid Build Coastguard Worker { 81*387f9dfdSAndroid Build Coastguard Worker u32 key = 0, val = 0; 82*387f9dfdSAndroid Build Coastguard Worker struct sk_msg_buff *msg; 83*387f9dfdSAndroid Build Coastguard Worker struct sk_buff *skb; 84*387f9dfdSAndroid Build Coastguard Worker 85*387f9dfdSAndroid Build Coastguard Worker sk_hash2.update(&key, &val); 86*387f9dfdSAndroid Build Coastguard Worker sk_hash2.delete(&key); 87*387f9dfdSAndroid Build Coastguard Worker sk_hash2.sock_hash_update(skops, &key, 0); 88*387f9dfdSAndroid Build Coastguard Worker sk_hash2.msg_redirect_hash(msg, &key, 0); 89*387f9dfdSAndroid Build Coastguard Worker sk_hash2.sk_redirect_hash(skb, &key, 0); 90*387f9dfdSAndroid Build Coastguard Worker 91*387f9dfdSAndroid Build Coastguard Worker return 0; 92*387f9dfdSAndroid Build Coastguard Worker } 93*387f9dfdSAndroid Build Coastguard Worker )"; 94*387f9dfdSAndroid Build Coastguard Worker 95*387f9dfdSAndroid Build Coastguard Worker // make sure program is loaded successfully 96*387f9dfdSAndroid Build Coastguard Worker ebpf::BPF bpf; 97*387f9dfdSAndroid Build Coastguard Worker ebpf::StatusTuple res(0); 98*387f9dfdSAndroid Build Coastguard Worker res = bpf.init(BPF_PROGRAM); 99*387f9dfdSAndroid Build Coastguard Worker REQUIRE(res.ok()); 100*387f9dfdSAndroid Build Coastguard Worker 101*387f9dfdSAndroid Build Coastguard Worker // create a udp socket so we can do some map operations. 102*387f9dfdSAndroid Build Coastguard Worker int sockfd = socket(AF_INET, SOCK_DGRAM, 0); 103*387f9dfdSAndroid Build Coastguard Worker REQUIRE(sockfd >= 0); 104*387f9dfdSAndroid Build Coastguard Worker 105*387f9dfdSAndroid Build Coastguard Worker auto sk_hash = bpf.get_sockhash_table("sk_hash1"); 106*387f9dfdSAndroid Build Coastguard Worker int key = 0, val = sockfd; 107*387f9dfdSAndroid Build Coastguard Worker 108*387f9dfdSAndroid Build Coastguard Worker res = sk_hash.remove_value(key); 109*387f9dfdSAndroid Build Coastguard Worker REQUIRE(!res.ok()); 110*387f9dfdSAndroid Build Coastguard Worker 111*387f9dfdSAndroid Build Coastguard Worker res = sk_hash.update_value(key, val); 112*387f9dfdSAndroid Build Coastguard Worker REQUIRE(res.ok() == expected_update_result); 113*387f9dfdSAndroid Build Coastguard Worker } 114*387f9dfdSAndroid Build Coastguard Worker } 115*387f9dfdSAndroid Build Coastguard Worker 116*387f9dfdSAndroid Build Coastguard Worker #endif 117