// Copyright (C) 2016 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.
#include <unistd.h>
#include <map>
#include <sstream>
#include <string>
#include <android-base/file.h>
#include <android-base/logging.h>
#include <android-base/strings.h>
void Usage(char* argv[]) {
printf("Usage: %s <status field> <value> [<status field> <value>]*\n", argv[0]);
printf("E.g.: $ %s Uid \"1000 1000 1000 1000\"\n", argv[0]);
}
int main(int argc, char* argv[]) {
if (argc < 3) {
Usage(argv);
LOG(FATAL) << "no status field requested";
}
if (argc % 2 == 0) {
// Since |argc| counts argv[0], if |argc| is odd, then the number of
// command-line arguments is even.
Usage(argv);
LOG(FATAL) << "need even number of command-line arguments";
}
std::string status;
bool res = android::base::ReadFileToString("/proc/self/status", &status, true);
if (!res) {
PLOG(FATAL) << "could not read /proc/self/status";
}
std::map<std::string, std::string> fields;
std::vector<std::string> lines = android::base::Split(status, "\n");
for (const auto& line : lines) {
std::vector<std::string> tokens = android::base::Split(line, ":");
if (tokens.size() >= 2) {
std::string field = tokens[0];
std::string value = android::base::Trim(tokens[1]);
if (field.length() > 0) {
fields[field] = value;
}
}
}
bool test_fails = false;
size_t uargc = static_cast<size_t>(argc); // |argc| >= 3.
for (size_t i = 1; i < static_cast<size_t>(argc); i = i + 2) {
std::string expected_value = argv[i + 1];
auto f = fields.find(argv[i]);
if (f != fields.end()) {
if (f->second != expected_value) {
LOG(ERROR) << "field '" << argv[i] << "' expected '" << expected_value
<< "', actual '" << f->second << "'";
test_fails = true;
}
} else {
LOG(WARNING) << "could not find field '" << argv[i] << "'";
}
}
return test_fails ? 1 : 0;
}