/*
|
* Copyright (C) 2013 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 "NetworkStats"
|
|
#include <errno.h>
|
#include <inttypes.h>
|
#include <sys/stat.h>
|
#include <sys/types.h>
|
|
#include <jni.h>
|
|
#include <nativehelper/JNIHelp.h>
|
#include <nativehelper/ScopedUtfChars.h>
|
#include <nativehelper/ScopedLocalRef.h>
|
#include <nativehelper/ScopedPrimitiveArray.h>
|
|
#include <utils/Log.h>
|
#include <utils/misc.h>
|
|
#include "android-base/unique_fd.h"
|
#include "bpf/BpfUtils.h"
|
#include "netdbpf/BpfNetworkStats.h"
|
|
using android::bpf::parseBpfNetworkStatsDetail;
|
using android::bpf::stats_line;
|
|
namespace android {
|
|
static jclass gStringClass;
|
|
static struct {
|
jfieldID size;
|
jfieldID capacity;
|
jfieldID iface;
|
jfieldID uid;
|
jfieldID set;
|
jfieldID tag;
|
jfieldID metered;
|
jfieldID roaming;
|
jfieldID defaultNetwork;
|
jfieldID rxBytes;
|
jfieldID rxPackets;
|
jfieldID txBytes;
|
jfieldID txPackets;
|
jfieldID operations;
|
} gNetworkStatsClassInfo;
|
|
static jobjectArray get_string_array(JNIEnv* env, jobject obj, jfieldID field, int size, bool grow)
|
{
|
if (!grow) {
|
jobjectArray array = (jobjectArray)env->GetObjectField(obj, field);
|
if (array != NULL) {
|
return array;
|
}
|
}
|
return env->NewObjectArray(size, gStringClass, NULL);
|
}
|
|
static jintArray get_int_array(JNIEnv* env, jobject obj, jfieldID field, int size, bool grow)
|
{
|
if (!grow) {
|
jintArray array = (jintArray)env->GetObjectField(obj, field);
|
if (array != NULL) {
|
return array;
|
}
|
}
|
return env->NewIntArray(size);
|
}
|
|
static jlongArray get_long_array(JNIEnv* env, jobject obj, jfieldID field, int size, bool grow)
|
{
|
if (!grow) {
|
jlongArray array = (jlongArray)env->GetObjectField(obj, field);
|
if (array != NULL) {
|
return array;
|
}
|
}
|
return env->NewLongArray(size);
|
}
|
|
static int legacyReadNetworkStatsDetail(std::vector<stats_line>* lines,
|
const std::vector<std::string>& limitIfaces,
|
int limitTag, int limitUid, const char* path) {
|
FILE* fp = fopen(path, "re");
|
if (fp == NULL) {
|
return -1;
|
}
|
|
int lastIdx = 1;
|
int idx;
|
char buffer[384];
|
while (fgets(buffer, sizeof(buffer), fp) != NULL) {
|
stats_line s;
|
int64_t rawTag;
|
char* pos = buffer;
|
char* endPos;
|
// First field is the index.
|
idx = (int)strtol(pos, &endPos, 10);
|
//ALOGI("Index #%d: %s", idx, buffer);
|
if (pos == endPos) {
|
// Skip lines that don't start with in index. In particular,
|
// this will skip the initial header line.
|
continue;
|
}
|
if (idx != lastIdx + 1) {
|
ALOGE("inconsistent idx=%d after lastIdx=%d: %s", idx, lastIdx, buffer);
|
fclose(fp);
|
return -1;
|
}
|
lastIdx = idx;
|
pos = endPos;
|
// Skip whitespace.
|
while (*pos == ' ') {
|
pos++;
|
}
|
// Next field is iface.
|
int ifaceIdx = 0;
|
while (*pos != ' ' && *pos != 0 && ifaceIdx < (int)(sizeof(s.iface)-1)) {
|
s.iface[ifaceIdx] = *pos;
|
ifaceIdx++;
|
pos++;
|
}
|
if (*pos != ' ') {
|
ALOGE("bad iface: %s", buffer);
|
fclose(fp);
|
return -1;
|
}
|
s.iface[ifaceIdx] = 0;
|
if (limitIfaces.size() > 0) {
|
// Is this an iface the caller is interested in?
|
int i = 0;
|
while (i < (int)limitIfaces.size()) {
|
if (limitIfaces[i] == s.iface) {
|
break;
|
}
|
i++;
|
}
|
if (i >= (int)limitIfaces.size()) {
|
// Nothing matched; skip this line.
|
//ALOGI("skipping due to iface: %s", buffer);
|
continue;
|
}
|
}
|
|
// Ignore whitespace
|
while (*pos == ' ') pos++;
|
|
// Find end of tag field
|
endPos = pos;
|
while (*endPos != ' ') endPos++;
|
|
// Three digit field is always 0x0, otherwise parse
|
if (endPos - pos == 3) {
|
rawTag = 0;
|
} else {
|
if (sscanf(pos, "%" PRIx64, &rawTag) != 1) {
|
ALOGE("bad tag: %s", pos);
|
fclose(fp);
|
return -1;
|
}
|
}
|
s.tag = rawTag >> 32;
|
if (limitTag != -1 && s.tag != static_cast<uint32_t>(limitTag)) {
|
//ALOGI("skipping due to tag: %s", buffer);
|
continue;
|
}
|
pos = endPos;
|
|
// Ignore whitespace
|
while (*pos == ' ') pos++;
|
|
// Parse remaining fields.
|
if (sscanf(pos, "%u %u %" PRIu64 " %" PRIu64 " %" PRIu64 " %" PRIu64,
|
&s.uid, &s.set, &s.rxBytes, &s.rxPackets,
|
&s.txBytes, &s.txPackets) == 6) {
|
if (limitUid != -1 && static_cast<uint32_t>(limitUid) != s.uid) {
|
//ALOGI("skipping due to uid: %s", buffer);
|
continue;
|
}
|
lines->push_back(s);
|
} else {
|
//ALOGI("skipping due to bad remaining fields: %s", pos);
|
}
|
}
|
|
if (fclose(fp) != 0) {
|
ALOGE("Failed to close netstats file");
|
return -1;
|
}
|
return 0;
|
}
|
|
static int statsLinesToNetworkStats(JNIEnv* env, jclass clazz, jobject stats,
|
std::vector<stats_line>& lines) {
|
int size = lines.size();
|
|
bool grow = size > env->GetIntField(stats, gNetworkStatsClassInfo.capacity);
|
|
ScopedLocalRef<jobjectArray> iface(env, get_string_array(env, stats,
|
gNetworkStatsClassInfo.iface, size, grow));
|
if (iface.get() == NULL) return -1;
|
ScopedIntArrayRW uid(env, get_int_array(env, stats,
|
gNetworkStatsClassInfo.uid, size, grow));
|
if (uid.get() == NULL) return -1;
|
ScopedIntArrayRW set(env, get_int_array(env, stats,
|
gNetworkStatsClassInfo.set, size, grow));
|
if (set.get() == NULL) return -1;
|
ScopedIntArrayRW tag(env, get_int_array(env, stats,
|
gNetworkStatsClassInfo.tag, size, grow));
|
if (tag.get() == NULL) return -1;
|
ScopedIntArrayRW metered(env, get_int_array(env, stats,
|
gNetworkStatsClassInfo.metered, size, grow));
|
if (metered.get() == NULL) return -1;
|
ScopedIntArrayRW roaming(env, get_int_array(env, stats,
|
gNetworkStatsClassInfo.roaming, size, grow));
|
if (roaming.get() == NULL) return -1;
|
ScopedIntArrayRW defaultNetwork(env, get_int_array(env, stats,
|
gNetworkStatsClassInfo.defaultNetwork, size, grow));
|
if (defaultNetwork.get() == NULL) return -1;
|
ScopedLongArrayRW rxBytes(env, get_long_array(env, stats,
|
gNetworkStatsClassInfo.rxBytes, size, grow));
|
if (rxBytes.get() == NULL) return -1;
|
ScopedLongArrayRW rxPackets(env, get_long_array(env, stats,
|
gNetworkStatsClassInfo.rxPackets, size, grow));
|
if (rxPackets.get() == NULL) return -1;
|
ScopedLongArrayRW txBytes(env, get_long_array(env, stats,
|
gNetworkStatsClassInfo.txBytes, size, grow));
|
if (txBytes.get() == NULL) return -1;
|
ScopedLongArrayRW txPackets(env, get_long_array(env, stats,
|
gNetworkStatsClassInfo.txPackets, size, grow));
|
if (txPackets.get() == NULL) return -1;
|
ScopedLongArrayRW operations(env, get_long_array(env, stats,
|
gNetworkStatsClassInfo.operations, size, grow));
|
if (operations.get() == NULL) return -1;
|
|
for (int i = 0; i < size; i++) {
|
ScopedLocalRef<jstring> ifaceString(env, env->NewStringUTF(lines[i].iface));
|
env->SetObjectArrayElement(iface.get(), i, ifaceString.get());
|
|
uid[i] = lines[i].uid;
|
set[i] = lines[i].set;
|
tag[i] = lines[i].tag;
|
// Metered, roaming and defaultNetwork are populated in Java-land.
|
rxBytes[i] = lines[i].rxBytes;
|
rxPackets[i] = lines[i].rxPackets;
|
txBytes[i] = lines[i].txBytes;
|
txPackets[i] = lines[i].txPackets;
|
}
|
|
env->SetIntField(stats, gNetworkStatsClassInfo.size, size);
|
if (grow) {
|
env->SetIntField(stats, gNetworkStatsClassInfo.capacity, size);
|
env->SetObjectField(stats, gNetworkStatsClassInfo.iface, iface.get());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.uid, uid.getJavaArray());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.set, set.getJavaArray());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.tag, tag.getJavaArray());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.metered, metered.getJavaArray());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.roaming, roaming.getJavaArray());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.defaultNetwork,
|
defaultNetwork.getJavaArray());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.rxBytes, rxBytes.getJavaArray());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.rxPackets, rxPackets.getJavaArray());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.txBytes, txBytes.getJavaArray());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.txPackets, txPackets.getJavaArray());
|
env->SetObjectField(stats, gNetworkStatsClassInfo.operations, operations.getJavaArray());
|
}
|
return 0;
|
}
|
|
static int readNetworkStatsDetail(JNIEnv* env, jclass clazz, jobject stats, jstring path,
|
jint limitUid, jobjectArray limitIfacesObj, jint limitTag,
|
jboolean useBpfStats) {
|
|
std::vector<std::string> limitIfaces;
|
if (limitIfacesObj != NULL && env->GetArrayLength(limitIfacesObj) > 0) {
|
int num = env->GetArrayLength(limitIfacesObj);
|
for (int i = 0; i < num; i++) {
|
jstring string = (jstring)env->GetObjectArrayElement(limitIfacesObj, i);
|
ScopedUtfChars string8(env, string);
|
if (string8.c_str() != NULL) {
|
limitIfaces.push_back(std::string(string8.c_str()));
|
}
|
}
|
}
|
std::vector<stats_line> lines;
|
|
|
if (useBpfStats) {
|
if (parseBpfNetworkStatsDetail(&lines, limitIfaces, limitTag, limitUid) < 0)
|
return -1;
|
} else {
|
ScopedUtfChars path8(env, path);
|
if (path8.c_str() == NULL) {
|
ALOGE("the qtaguid legacy path is invalid: %s", path8.c_str());
|
return -1;
|
}
|
if (legacyReadNetworkStatsDetail(&lines, limitIfaces, limitTag,
|
limitUid, path8.c_str()) < 0)
|
return -1;
|
}
|
|
return statsLinesToNetworkStats(env, clazz, stats, lines);
|
}
|
|
static int readNetworkStatsDev(JNIEnv* env, jclass clazz, jobject stats) {
|
std::vector<stats_line> lines;
|
|
if (parseBpfNetworkStatsDev(&lines) < 0)
|
return -1;
|
|
return statsLinesToNetworkStats(env, clazz, stats, lines);
|
}
|
|
static const JNINativeMethod gMethods[] = {
|
{ "nativeReadNetworkStatsDetail",
|
"(Landroid/net/NetworkStats;Ljava/lang/String;I[Ljava/lang/String;IZ)I",
|
(void*) readNetworkStatsDetail },
|
{ "nativeReadNetworkStatsDev", "(Landroid/net/NetworkStats;)I",
|
(void*) readNetworkStatsDev },
|
};
|
|
int register_android_server_net_NetworkStatsFactory(JNIEnv* env) {
|
int err = jniRegisterNativeMethods(env, "com/android/server/net/NetworkStatsFactory", gMethods,
|
NELEM(gMethods));
|
gStringClass = env->FindClass("java/lang/String");
|
gStringClass = static_cast<jclass>(env->NewGlobalRef(gStringClass));
|
|
jclass clazz = env->FindClass("android/net/NetworkStats");
|
gNetworkStatsClassInfo.size = env->GetFieldID(clazz, "size", "I");
|
gNetworkStatsClassInfo.capacity = env->GetFieldID(clazz, "capacity", "I");
|
gNetworkStatsClassInfo.iface = env->GetFieldID(clazz, "iface", "[Ljava/lang/String;");
|
gNetworkStatsClassInfo.uid = env->GetFieldID(clazz, "uid", "[I");
|
gNetworkStatsClassInfo.set = env->GetFieldID(clazz, "set", "[I");
|
gNetworkStatsClassInfo.tag = env->GetFieldID(clazz, "tag", "[I");
|
gNetworkStatsClassInfo.metered = env->GetFieldID(clazz, "metered", "[I");
|
gNetworkStatsClassInfo.roaming = env->GetFieldID(clazz, "roaming", "[I");
|
gNetworkStatsClassInfo.defaultNetwork = env->GetFieldID(clazz, "defaultNetwork", "[I");
|
gNetworkStatsClassInfo.rxBytes = env->GetFieldID(clazz, "rxBytes", "[J");
|
gNetworkStatsClassInfo.rxPackets = env->GetFieldID(clazz, "rxPackets", "[J");
|
gNetworkStatsClassInfo.txBytes = env->GetFieldID(clazz, "txBytes", "[J");
|
gNetworkStatsClassInfo.txPackets = env->GetFieldID(clazz, "txPackets", "[J");
|
gNetworkStatsClassInfo.operations = env->GetFieldID(clazz, "operations", "[J");
|
|
return err;
|
}
|
|
}
|