18c2ecf20Sopenharmony_ci/* Copyright (c) 2013-2015 PLUMgrid, http://plumgrid.com 28c2ecf20Sopenharmony_ci * Copyright (c) 2015 BMW Car IT GmbH 38c2ecf20Sopenharmony_ci * 48c2ecf20Sopenharmony_ci * This program is free software; you can redistribute it and/or 58c2ecf20Sopenharmony_ci * modify it under the terms of version 2 of the GNU General Public 68c2ecf20Sopenharmony_ci * License as published by the Free Software Foundation. 78c2ecf20Sopenharmony_ci */ 88c2ecf20Sopenharmony_ci#include <linux/version.h> 98c2ecf20Sopenharmony_ci#include <linux/ptrace.h> 108c2ecf20Sopenharmony_ci#include <uapi/linux/bpf.h> 118c2ecf20Sopenharmony_ci#include <bpf/bpf_helpers.h> 128c2ecf20Sopenharmony_ci 138c2ecf20Sopenharmony_ci#define MAX_ENTRIES 20 148c2ecf20Sopenharmony_ci#define MAX_CPU 4 158c2ecf20Sopenharmony_ci 168c2ecf20Sopenharmony_ci/* We need to stick to static allocated memory (an array instead of 178c2ecf20Sopenharmony_ci * hash table) because managing dynamic memory from the 188c2ecf20Sopenharmony_ci * trace_preempt_[on|off] tracepoints hooks is not supported. 198c2ecf20Sopenharmony_ci */ 208c2ecf20Sopenharmony_ci 218c2ecf20Sopenharmony_cistruct { 228c2ecf20Sopenharmony_ci __uint(type, BPF_MAP_TYPE_ARRAY); 238c2ecf20Sopenharmony_ci __type(key, int); 248c2ecf20Sopenharmony_ci __type(value, u64); 258c2ecf20Sopenharmony_ci __uint(max_entries, MAX_CPU); 268c2ecf20Sopenharmony_ci} my_map SEC(".maps"); 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ciSEC("kprobe/trace_preempt_off") 298c2ecf20Sopenharmony_ciint bpf_prog1(struct pt_regs *ctx) 308c2ecf20Sopenharmony_ci{ 318c2ecf20Sopenharmony_ci int cpu = bpf_get_smp_processor_id(); 328c2ecf20Sopenharmony_ci u64 *ts = bpf_map_lookup_elem(&my_map, &cpu); 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_ci if (ts) 358c2ecf20Sopenharmony_ci *ts = bpf_ktime_get_ns(); 368c2ecf20Sopenharmony_ci 378c2ecf20Sopenharmony_ci return 0; 388c2ecf20Sopenharmony_ci} 398c2ecf20Sopenharmony_ci 408c2ecf20Sopenharmony_cistatic unsigned int log2(unsigned int v) 418c2ecf20Sopenharmony_ci{ 428c2ecf20Sopenharmony_ci unsigned int r; 438c2ecf20Sopenharmony_ci unsigned int shift; 448c2ecf20Sopenharmony_ci 458c2ecf20Sopenharmony_ci r = (v > 0xFFFF) << 4; v >>= r; 468c2ecf20Sopenharmony_ci shift = (v > 0xFF) << 3; v >>= shift; r |= shift; 478c2ecf20Sopenharmony_ci shift = (v > 0xF) << 2; v >>= shift; r |= shift; 488c2ecf20Sopenharmony_ci shift = (v > 0x3) << 1; v >>= shift; r |= shift; 498c2ecf20Sopenharmony_ci r |= (v >> 1); 508c2ecf20Sopenharmony_ci 518c2ecf20Sopenharmony_ci return r; 528c2ecf20Sopenharmony_ci} 538c2ecf20Sopenharmony_ci 548c2ecf20Sopenharmony_cistatic unsigned int log2l(unsigned long v) 558c2ecf20Sopenharmony_ci{ 568c2ecf20Sopenharmony_ci unsigned int hi = v >> 32; 578c2ecf20Sopenharmony_ci 588c2ecf20Sopenharmony_ci if (hi) 598c2ecf20Sopenharmony_ci return log2(hi) + 32; 608c2ecf20Sopenharmony_ci else 618c2ecf20Sopenharmony_ci return log2(v); 628c2ecf20Sopenharmony_ci} 638c2ecf20Sopenharmony_ci 648c2ecf20Sopenharmony_cistruct { 658c2ecf20Sopenharmony_ci __uint(type, BPF_MAP_TYPE_ARRAY); 668c2ecf20Sopenharmony_ci __type(key, int); 678c2ecf20Sopenharmony_ci __type(value, long); 688c2ecf20Sopenharmony_ci __uint(max_entries, MAX_CPU * MAX_ENTRIES); 698c2ecf20Sopenharmony_ci} my_lat SEC(".maps"); 708c2ecf20Sopenharmony_ci 718c2ecf20Sopenharmony_ciSEC("kprobe/trace_preempt_on") 728c2ecf20Sopenharmony_ciint bpf_prog2(struct pt_regs *ctx) 738c2ecf20Sopenharmony_ci{ 748c2ecf20Sopenharmony_ci u64 *ts, cur_ts, delta; 758c2ecf20Sopenharmony_ci int key, cpu; 768c2ecf20Sopenharmony_ci long *val; 778c2ecf20Sopenharmony_ci 788c2ecf20Sopenharmony_ci cpu = bpf_get_smp_processor_id(); 798c2ecf20Sopenharmony_ci ts = bpf_map_lookup_elem(&my_map, &cpu); 808c2ecf20Sopenharmony_ci if (!ts) 818c2ecf20Sopenharmony_ci return 0; 828c2ecf20Sopenharmony_ci 838c2ecf20Sopenharmony_ci cur_ts = bpf_ktime_get_ns(); 848c2ecf20Sopenharmony_ci delta = log2l(cur_ts - *ts); 858c2ecf20Sopenharmony_ci 868c2ecf20Sopenharmony_ci if (delta > MAX_ENTRIES - 1) 878c2ecf20Sopenharmony_ci delta = MAX_ENTRIES - 1; 888c2ecf20Sopenharmony_ci 898c2ecf20Sopenharmony_ci key = cpu * MAX_ENTRIES + delta; 908c2ecf20Sopenharmony_ci val = bpf_map_lookup_elem(&my_lat, &key); 918c2ecf20Sopenharmony_ci if (val) 928c2ecf20Sopenharmony_ci __sync_fetch_and_add((long *)val, 1); 938c2ecf20Sopenharmony_ci 948c2ecf20Sopenharmony_ci return 0; 958c2ecf20Sopenharmony_ci 968c2ecf20Sopenharmony_ci} 978c2ecf20Sopenharmony_ci 988c2ecf20Sopenharmony_cichar _license[] SEC("license") = "GPL"; 998c2ecf20Sopenharmony_ciu32 _version SEC("version") = LINUX_VERSION_CODE; 100