18c2ecf20Sopenharmony_ci/* eBPF example program: 28c2ecf20Sopenharmony_ci * - creates arraymap in kernel with key 4 bytes and value 8 bytes 38c2ecf20Sopenharmony_ci * 48c2ecf20Sopenharmony_ci * - loads eBPF program: 58c2ecf20Sopenharmony_ci * r0 = skb->data[ETH_HLEN + offsetof(struct iphdr, protocol)]; 68c2ecf20Sopenharmony_ci * *(u32*)(fp - 4) = r0; 78c2ecf20Sopenharmony_ci * // assuming packet is IPv4, lookup ip->proto in a map 88c2ecf20Sopenharmony_ci * value = bpf_map_lookup_elem(map_fd, fp - 4); 98c2ecf20Sopenharmony_ci * if (value) 108c2ecf20Sopenharmony_ci * (*(u64*)value) += 1; 118c2ecf20Sopenharmony_ci * 128c2ecf20Sopenharmony_ci * - attaches this program to loopback interface "lo" raw socket 138c2ecf20Sopenharmony_ci * 148c2ecf20Sopenharmony_ci * - every second user space reads map[tcp], map[udp], map[icmp] to see 158c2ecf20Sopenharmony_ci * how many packets of given protocol were seen on "lo" 168c2ecf20Sopenharmony_ci */ 178c2ecf20Sopenharmony_ci#include <stdio.h> 188c2ecf20Sopenharmony_ci#include <unistd.h> 198c2ecf20Sopenharmony_ci#include <assert.h> 208c2ecf20Sopenharmony_ci#include <linux/bpf.h> 218c2ecf20Sopenharmony_ci#include <string.h> 228c2ecf20Sopenharmony_ci#include <stdlib.h> 238c2ecf20Sopenharmony_ci#include <errno.h> 248c2ecf20Sopenharmony_ci#include <sys/socket.h> 258c2ecf20Sopenharmony_ci#include <arpa/inet.h> 268c2ecf20Sopenharmony_ci#include <linux/if_ether.h> 278c2ecf20Sopenharmony_ci#include <linux/ip.h> 288c2ecf20Sopenharmony_ci#include <stddef.h> 298c2ecf20Sopenharmony_ci#include <bpf/bpf.h> 308c2ecf20Sopenharmony_ci#include "bpf_insn.h" 318c2ecf20Sopenharmony_ci#include "sock_example.h" 328c2ecf20Sopenharmony_ci 338c2ecf20Sopenharmony_cichar bpf_log_buf[BPF_LOG_BUF_SIZE]; 348c2ecf20Sopenharmony_ci 358c2ecf20Sopenharmony_cistatic int test_sock(void) 368c2ecf20Sopenharmony_ci{ 378c2ecf20Sopenharmony_ci int sock = -1, map_fd, prog_fd, i, key; 388c2ecf20Sopenharmony_ci long long value = 0, tcp_cnt, udp_cnt, icmp_cnt; 398c2ecf20Sopenharmony_ci 408c2ecf20Sopenharmony_ci map_fd = bpf_create_map(BPF_MAP_TYPE_ARRAY, sizeof(key), sizeof(value), 418c2ecf20Sopenharmony_ci 256, 0); 428c2ecf20Sopenharmony_ci if (map_fd < 0) { 438c2ecf20Sopenharmony_ci printf("failed to create map '%s'\n", strerror(errno)); 448c2ecf20Sopenharmony_ci goto cleanup; 458c2ecf20Sopenharmony_ci } 468c2ecf20Sopenharmony_ci 478c2ecf20Sopenharmony_ci struct bpf_insn prog[] = { 488c2ecf20Sopenharmony_ci BPF_MOV64_REG(BPF_REG_6, BPF_REG_1), 498c2ecf20Sopenharmony_ci BPF_LD_ABS(BPF_B, ETH_HLEN + offsetof(struct iphdr, protocol) /* R0 = ip->proto */), 508c2ecf20Sopenharmony_ci BPF_STX_MEM(BPF_W, BPF_REG_10, BPF_REG_0, -4), /* *(u32 *)(fp - 4) = r0 */ 518c2ecf20Sopenharmony_ci BPF_MOV64_REG(BPF_REG_2, BPF_REG_10), 528c2ecf20Sopenharmony_ci BPF_ALU64_IMM(BPF_ADD, BPF_REG_2, -4), /* r2 = fp - 4 */ 538c2ecf20Sopenharmony_ci BPF_LD_MAP_FD(BPF_REG_1, map_fd), 548c2ecf20Sopenharmony_ci BPF_RAW_INSN(BPF_JMP | BPF_CALL, 0, 0, 0, BPF_FUNC_map_lookup_elem), 558c2ecf20Sopenharmony_ci BPF_JMP_IMM(BPF_JEQ, BPF_REG_0, 0, 2), 568c2ecf20Sopenharmony_ci BPF_MOV64_IMM(BPF_REG_1, 1), /* r1 = 1 */ 578c2ecf20Sopenharmony_ci BPF_RAW_INSN(BPF_STX | BPF_XADD | BPF_DW, BPF_REG_0, BPF_REG_1, 0, 0), /* xadd r0 += r1 */ 588c2ecf20Sopenharmony_ci BPF_MOV64_IMM(BPF_REG_0, 0), /* r0 = 0 */ 598c2ecf20Sopenharmony_ci BPF_EXIT_INSN(), 608c2ecf20Sopenharmony_ci }; 618c2ecf20Sopenharmony_ci size_t insns_cnt = sizeof(prog) / sizeof(struct bpf_insn); 628c2ecf20Sopenharmony_ci 638c2ecf20Sopenharmony_ci prog_fd = bpf_load_program(BPF_PROG_TYPE_SOCKET_FILTER, prog, insns_cnt, 648c2ecf20Sopenharmony_ci "GPL", 0, bpf_log_buf, BPF_LOG_BUF_SIZE); 658c2ecf20Sopenharmony_ci if (prog_fd < 0) { 668c2ecf20Sopenharmony_ci printf("failed to load prog '%s'\n", strerror(errno)); 678c2ecf20Sopenharmony_ci goto cleanup; 688c2ecf20Sopenharmony_ci } 698c2ecf20Sopenharmony_ci 708c2ecf20Sopenharmony_ci sock = open_raw_sock("lo"); 718c2ecf20Sopenharmony_ci 728c2ecf20Sopenharmony_ci if (setsockopt(sock, SOL_SOCKET, SO_ATTACH_BPF, &prog_fd, 738c2ecf20Sopenharmony_ci sizeof(prog_fd)) < 0) { 748c2ecf20Sopenharmony_ci printf("setsockopt %s\n", strerror(errno)); 758c2ecf20Sopenharmony_ci goto cleanup; 768c2ecf20Sopenharmony_ci } 778c2ecf20Sopenharmony_ci 788c2ecf20Sopenharmony_ci for (i = 0; i < 10; i++) { 798c2ecf20Sopenharmony_ci key = IPPROTO_TCP; 808c2ecf20Sopenharmony_ci assert(bpf_map_lookup_elem(map_fd, &key, &tcp_cnt) == 0); 818c2ecf20Sopenharmony_ci 828c2ecf20Sopenharmony_ci key = IPPROTO_UDP; 838c2ecf20Sopenharmony_ci assert(bpf_map_lookup_elem(map_fd, &key, &udp_cnt) == 0); 848c2ecf20Sopenharmony_ci 858c2ecf20Sopenharmony_ci key = IPPROTO_ICMP; 868c2ecf20Sopenharmony_ci assert(bpf_map_lookup_elem(map_fd, &key, &icmp_cnt) == 0); 878c2ecf20Sopenharmony_ci 888c2ecf20Sopenharmony_ci printf("TCP %lld UDP %lld ICMP %lld packets\n", 898c2ecf20Sopenharmony_ci tcp_cnt, udp_cnt, icmp_cnt); 908c2ecf20Sopenharmony_ci sleep(1); 918c2ecf20Sopenharmony_ci } 928c2ecf20Sopenharmony_ci 938c2ecf20Sopenharmony_cicleanup: 948c2ecf20Sopenharmony_ci /* maps, programs, raw sockets will auto cleanup on process exit */ 958c2ecf20Sopenharmony_ci return 0; 968c2ecf20Sopenharmony_ci} 978c2ecf20Sopenharmony_ci 988c2ecf20Sopenharmony_ciint main(void) 998c2ecf20Sopenharmony_ci{ 1008c2ecf20Sopenharmony_ci FILE *f; 1018c2ecf20Sopenharmony_ci 1028c2ecf20Sopenharmony_ci f = popen("ping -4 -c5 localhost", "r"); 1038c2ecf20Sopenharmony_ci (void)f; 1048c2ecf20Sopenharmony_ci 1058c2ecf20Sopenharmony_ci return test_sock(); 1068c2ecf20Sopenharmony_ci} 107