Merge "[NETD_BPF#8] Move BpfUtils.cpp to BpfUtils.h" am: edfca8b548
am: eb3273cd75
am: 9f3e43abb6
Original change: https://android-review.googlesource.com/c/platform/system/bpf/+/1869327 Change-Id: I46ecac9bbda4a674d82421bf04c8e8ae752a1c45
This commit is contained in:
commit
a7d95946ef
3 changed files with 71 additions and 118 deletions
|
@ -46,7 +46,6 @@ cc_library {
|
|||
target: {
|
||||
android: {
|
||||
srcs: [
|
||||
"BpfUtils.cpp",
|
||||
"Loader.cpp",
|
||||
],
|
||||
sanitize: {
|
||||
|
|
|
@ -1,113 +0,0 @@
|
|||
/*
|
||||
* Copyright (C) 2017 The Android Open Source Project
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#define LOG_TAG "BpfUtils"
|
||||
|
||||
#include "bpf/BpfUtils.h"
|
||||
|
||||
#include <elf.h>
|
||||
#include <inttypes.h>
|
||||
#include <linux/bpf.h>
|
||||
#include <linux/if_ether.h>
|
||||
#include <linux/in.h>
|
||||
#include <linux/pfkeyv2.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <sys/mman.h>
|
||||
#include <sys/resource.h>
|
||||
#include <sys/socket.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/utsname.h>
|
||||
#include <sstream>
|
||||
#include <string>
|
||||
|
||||
#include <android-base/unique_fd.h>
|
||||
#include <log/log.h>
|
||||
|
||||
using android::base::unique_fd;
|
||||
|
||||
// The buffer size for the buffer that records program loading logs, needs to be large enough for
|
||||
// the largest kernel program.
|
||||
|
||||
namespace android {
|
||||
namespace bpf {
|
||||
|
||||
uint64_t getSocketCookie(int sockFd) {
|
||||
uint64_t sock_cookie;
|
||||
socklen_t cookie_len = sizeof(sock_cookie);
|
||||
int res = getsockopt(sockFd, SOL_SOCKET, SO_COOKIE, &sock_cookie, &cookie_len);
|
||||
if (res < 0) {
|
||||
res = -errno;
|
||||
ALOGE("Failed to get socket cookie: %s\n", strerror(errno));
|
||||
errno = -res;
|
||||
// 0 is an invalid cookie. See sock_gen_cookie.
|
||||
return NONEXISTENT_COOKIE;
|
||||
}
|
||||
return sock_cookie;
|
||||
}
|
||||
|
||||
int synchronizeKernelRCU() {
|
||||
// This is a temporary hack for network stats map swap on devices running
|
||||
// 4.9 kernels. The kernel code of socket release on pf_key socket will
|
||||
// explicitly call synchronize_rcu() which is exactly what we need.
|
||||
int pfSocket = socket(AF_KEY, SOCK_RAW | SOCK_CLOEXEC, PF_KEY_V2);
|
||||
|
||||
if (pfSocket < 0) {
|
||||
int ret = -errno;
|
||||
ALOGE("create PF_KEY socket failed: %s", strerror(errno));
|
||||
return ret;
|
||||
}
|
||||
|
||||
// When closing socket, synchronize_rcu() gets called in sock_release().
|
||||
if (close(pfSocket)) {
|
||||
int ret = -errno;
|
||||
ALOGE("failed to close the PF_KEY socket: %s", strerror(errno));
|
||||
return ret;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int setrlimitForTest() {
|
||||
// Set the memory rlimit for the test process if the default MEMLOCK rlimit is not enough.
|
||||
struct rlimit limit = {
|
||||
.rlim_cur = 1073741824, // 1 GiB
|
||||
.rlim_max = 1073741824, // 1 GiB
|
||||
};
|
||||
int res = setrlimit(RLIMIT_MEMLOCK, &limit);
|
||||
if (res) {
|
||||
ALOGE("Failed to set the default MEMLOCK rlimit: %s", strerror(errno));
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
unsigned kernelVersion() {
|
||||
struct utsname buf;
|
||||
int ret = uname(&buf);
|
||||
if (ret) return 0;
|
||||
|
||||
unsigned kver_major;
|
||||
unsigned kver_minor;
|
||||
unsigned kver_sub;
|
||||
char dummy;
|
||||
ret = sscanf(buf.release, "%u.%u.%u%c", &kver_major, &kver_minor, &kver_sub, &dummy);
|
||||
// Check the device kernel version
|
||||
if (ret < 3) return 0;
|
||||
|
||||
return KVER(kver_major, kver_minor, kver_sub);
|
||||
}
|
||||
|
||||
} // namespace bpf
|
||||
} // namespace android
|
|
@ -17,15 +17,24 @@
|
|||
#pragma once
|
||||
|
||||
#include <linux/if_ether.h>
|
||||
#include <linux/pfkeyv2.h>
|
||||
#include <net/if.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <sys/resource.h>
|
||||
#include <sys/socket.h>
|
||||
#include <sys/utsname.h>
|
||||
|
||||
#include <string>
|
||||
|
||||
#include <android-base/unique_fd.h>
|
||||
#include <log/log.h>
|
||||
|
||||
#include "BpfSyscallWrappers.h"
|
||||
|
||||
// The buffer size for the buffer that records program loading logs, needs to be large enough for
|
||||
// the largest kernel program.
|
||||
|
||||
namespace android {
|
||||
namespace bpf {
|
||||
|
||||
|
@ -33,13 +42,71 @@ constexpr const int OVERFLOW_COUNTERSET = 2;
|
|||
|
||||
constexpr const uint64_t NONEXISTENT_COOKIE = 0;
|
||||
|
||||
uint64_t getSocketCookie(int sockFd);
|
||||
int synchronizeKernelRCU();
|
||||
int setrlimitForTest();
|
||||
static inline uint64_t getSocketCookie(int sockFd) {
|
||||
uint64_t sock_cookie;
|
||||
socklen_t cookie_len = sizeof(sock_cookie);
|
||||
int res = getsockopt(sockFd, SOL_SOCKET, SO_COOKIE, &sock_cookie, &cookie_len);
|
||||
if (res < 0) {
|
||||
res = -errno;
|
||||
ALOGE("Failed to get socket cookie: %s\n", strerror(errno));
|
||||
errno = -res;
|
||||
// 0 is an invalid cookie. See sock_gen_cookie.
|
||||
return NONEXISTENT_COOKIE;
|
||||
}
|
||||
return sock_cookie;
|
||||
}
|
||||
|
||||
static inline int synchronizeKernelRCU() {
|
||||
// This is a temporary hack for network stats map swap on devices running
|
||||
// 4.9 kernels. The kernel code of socket release on pf_key socket will
|
||||
// explicitly call synchronize_rcu() which is exactly what we need.
|
||||
int pfSocket = socket(AF_KEY, SOCK_RAW | SOCK_CLOEXEC, PF_KEY_V2);
|
||||
|
||||
if (pfSocket < 0) {
|
||||
int ret = -errno;
|
||||
ALOGE("create PF_KEY socket failed: %s", strerror(errno));
|
||||
return ret;
|
||||
}
|
||||
|
||||
// When closing socket, synchronize_rcu() gets called in sock_release().
|
||||
if (close(pfSocket)) {
|
||||
int ret = -errno;
|
||||
ALOGE("failed to close the PF_KEY socket: %s", strerror(errno));
|
||||
return ret;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline int setrlimitForTest() {
|
||||
// Set the memory rlimit for the test process if the default MEMLOCK rlimit is not enough.
|
||||
struct rlimit limit = {
|
||||
.rlim_cur = 1073741824, // 1 GiB
|
||||
.rlim_max = 1073741824, // 1 GiB
|
||||
};
|
||||
int res = setrlimit(RLIMIT_MEMLOCK, &limit);
|
||||
if (res) {
|
||||
ALOGE("Failed to set the default MEMLOCK rlimit: %s", strerror(errno));
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
#define KVER(a, b, c) (((a) << 24) + ((b) << 16) + (c))
|
||||
|
||||
unsigned kernelVersion();
|
||||
static inline unsigned kernelVersion() {
|
||||
struct utsname buf;
|
||||
int ret = uname(&buf);
|
||||
if (ret) return 0;
|
||||
|
||||
unsigned kver_major;
|
||||
unsigned kver_minor;
|
||||
unsigned kver_sub;
|
||||
char unused;
|
||||
ret = sscanf(buf.release, "%u.%u.%u%c", &kver_major, &kver_minor, &kver_sub, &unused);
|
||||
// Check the device kernel version
|
||||
if (ret < 3) return 0;
|
||||
|
||||
return KVER(kver_major, kver_minor, kver_sub);
|
||||
}
|
||||
|
||||
static inline bool isAtLeastKernelVersion(unsigned major, unsigned minor, unsigned sub) {
|
||||
return kernelVersion() >= KVER(major, minor, sub);
|
||||
|
|
Loading…
Reference in a new issue