blob: bb2303026544b50af07a3c3d8ed9f3c5b75a78f1 [file]
/*
* Copyright (C) 2020 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 <binder/Binder.h>
#include <binder/IPCThreadState.h>
#include <binder/IServiceManager.h>
#include <binder/ProcessState.h>
#include <binderdebug/BinderDebug.h>
#include <binderdebug/FileReader.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include <semaphore.h>
#include <thread>
#include <algorithm>
#include <android/binderdebug/test/BnControl.h>
#include <android/binderdebug/test/IControl.h>
namespace android {
namespace binderdebug {
namespace test {
using ::testing::_;
using ::testing::DoAll;
using ::testing::Invoke;
using ::testing::Return;
class Control : public BnControl {
public:
Control() {sem_init(&s, 1, 0);};
::android::binder::Status Continue() override;
sem_t s;
};
::android::binder::Status Control::Continue() {
IPCThreadState::self()->flushCommands();
sem_post(&s);
return binder::Status::ok();
}
TEST(BinderDebugTests, BinderPid) {
BinderPidInfo pidInfo;
const auto& status = getBinderPidInfo(BinderDebugContext::BINDER, getpid(), &pidInfo);
ASSERT_EQ(status, OK);
// There should be one referenced PID for servicemanager
EXPECT_TRUE(!pidInfo.refPids.empty());
}
TEST(BinderDebugTests, BinderThreads) {
BinderPidInfo pidInfo;
const auto& status = getBinderPidInfo(BinderDebugContext::BINDER, getpid(), &pidInfo);
ASSERT_EQ(status, OK);
EXPECT_TRUE(pidInfo.threadUsage <= pidInfo.threadCount);
// The second looper thread can sometimes take longer to spawn.
EXPECT_GE(pidInfo.threadCount, 1);
}
TEST(BinderDebugTests, ServiceManagerBinderIsVisible) {
sp<IBinder> binder = defaultServiceManager()->checkService(String16("activity"));
ASSERT_NE(binder, nullptr);
pid_t pid;
status_t status = binder->getDebugPid(&pid);
ASSERT_EQ(status, OK);
BinderPidInfo pidInfo;
status = getBinderPidInfo(BinderDebugContext::BINDER, pid, &pidInfo);
ASSERT_EQ(status, OK);
EXPECT_GT(pidInfo.threadCount, 0);
EXPECT_GE(pidInfo.threadUsage, 0);
bool found = std::any_of(pidInfo.refPids.begin(), pidInfo.refPids.end(),
[&](const auto& pair) {
const auto& pids = pair.second;
return std::find(pids.begin(), pids.end(), getpid()) != pids.end();
});
EXPECT_TRUE(found);
}
class MockFileReader : public FileReader {
public:
MockFileReader() = default;
bool Open(const std::string& path) override {
auto it = file_contents_.find(path);
if (it == file_contents_.end()) {
return false;
}
current_stream_.str(it->second);
current_stream_.clear();
current_stream_.seekg(0);
return true;
}
bool IsOpen() const override { return !current_stream_.str().empty(); }
bool GetLine(std::string& line) override {
return static_cast<bool>(std::getline(current_stream_, line));
}
void Close() override {
// We don't clear the stream here to allow multiple reads in the same test.
}
void SetFileContent(const std::string& path, const std::string& content) {
file_contents_[path] = content;
}
private:
std::map<std::string, std::string> file_contents_;
std::stringstream current_stream_;
};
TEST(BinderDebugParsingTest, GetBinderPidInfo_CBinder) {
const pid_t pid = 1234;
auto fileReader = std::make_unique<MockFileReader>();
const std::string proc_content =
"proc 1234\n"
"context binder\n"
" thread 1235: l 00 need_return 0 tr 0\n" // isInUse = true, isBinderThread = false
" thread 1236: l 11 need_return 1 tr 0\n" // isInUse = false, isBinderThread = true
" thread 1237: l 10 need_return 0 tr 0\n" // isInUse = false, isBinderThread = false
" thread 1238: l 12 need_return 0 tr 0\n" // isInUse = false, isBinderThread = true
" thread 1239: l 01 need_return 0 tr 0\n" // isInUse = true, isBinderThread = true
" node 1: u0000000012345678 c0000000087654321 pri 0:120 hs 1 hw 1 ls 0 lw 0 is 2 iw 2 "
"tr 1 proc 5678 9012\n";
fileReader->SetFileContent("/dev/binderfs/binder_logs/proc/1234", proc_content);
BinderPidInfo pidInfo;
status_t status =
getBinderPidInfo(BinderDebugContext::BINDER, pid, std::move(fileReader), &pidInfo);
EXPECT_EQ(OK, status);
EXPECT_EQ(3, pidInfo.threadCount);
EXPECT_EQ(1, pidInfo.threadUsage);
ASSERT_EQ(1, pidInfo.refPids.count(0x12345678));
EXPECT_EQ(2, pidInfo.refPids[0x12345678].size());
EXPECT_EQ(5678, pidInfo.refPids[0x12345678][0]);
EXPECT_EQ(9012, pidInfo.refPids[0x12345678][1]);
}
TEST(BinderDebugParsingTest, GetBinderPidInfo_RustBinder) {
const pid_t pid = 1234;
auto fileReader = std::make_unique<MockFileReader>();
const std::string proc_content = "proc 1234\n"
"context binder\n"
" thread 1235: l 00 need_return false\n"
" thread 1236: l 11 need_return true\n"
" thread 1237: l 10 need_return false\n"
" thread 1238: l 12 need_return false\n"
" thread 1239: l 01 need_return false\n"
" node 1: u0000000012345678 c0000000087654321 pri 0:120 hs "
"true hw true cs 1 cw 1 proc 5678 9012\n";
fileReader->SetFileContent("/dev/binderfs/binder_logs/proc/1234", proc_content);
BinderPidInfo pidInfo;
status_t status =
getBinderPidInfo(BinderDebugContext::BINDER, pid, std::move(fileReader), &pidInfo);
EXPECT_EQ(OK, status);
EXPECT_EQ(3, pidInfo.threadCount);
EXPECT_EQ(1, pidInfo.threadUsage);
ASSERT_EQ(1, pidInfo.refPids.count(0x12345678));
EXPECT_EQ(2, pidInfo.refPids[0x12345678].size());
EXPECT_EQ(5678, pidInfo.refPids[0x12345678][0]);
EXPECT_EQ(9012, pidInfo.refPids[0x12345678][1]);
}
TEST(BinderDebugParsingTest, GetBinderClientPids_CBinder) {
const pid_t clientPid = 1234;
const pid_t servicePid = 5678;
const int32_t handle = 912;
auto fileReader = std::make_unique<MockFileReader>();
const std::string proc_content_1234 =
"proc 1234\n"
"context binder\n"
" ref 52493: desc 910 node 52492 s 1 w 1 d 0000000000000000\n"
" ref 52495: desc 911 dead node 52494 s 1 w 1 d 0000000000000000\n"
" ref 52497: desc 912 node 52496 s 1 w 1 d 0000000000000000\n";
fileReader->SetFileContent("/dev/binderfs/binder_logs/proc/1234", proc_content_1234);
const std::string proc_content_5678 =
"proc 5678\n"
"context binder\n"
" node 52492: u0000000000000123 c0000000000000456 pri 0:120 hs 1 hw 1 ls 1 lw 0 is 0 "
"iw 2 tr 1 proc 111 222\n"
" node 52496: u0000000000000123 c0000000000000456 pri 0:120 hs 1 hw 1 ls 0 lw 0 is 4 "
"iw 4 tr 1 proc 333 444 555 666\n";
fileReader->SetFileContent("/dev/binderfs/binder_logs/proc/5678", proc_content_5678);
std::vector<pid_t> pids;
status_t status = getBinderClientPids(BinderDebugContext::BINDER, clientPid, servicePid, handle,
std::move(fileReader), &pids);
ASSERT_EQ(OK, status);
ASSERT_EQ(4, pids.size());
EXPECT_EQ(333, pids[0]);
EXPECT_EQ(444, pids[1]);
EXPECT_EQ(555, pids[2]);
EXPECT_EQ(666, pids[3]);
}
TEST(BinderDebugParsingTest, GetBinderClientPids_RustBinder) {
const pid_t clientPid = 1234;
const pid_t servicePid = 5678;
const int32_t handle = 912;
auto fileReader = std::make_unique<MockFileReader>();
const std::string proc_content_1234 = "proc 1234\n"
"context binder\n"
" ref 52493: desc 910 node 52492 s 1 w 1\n"
" ref 52495: desc 911 dead node 52494 s 1 w 1\n"
" ref 52497: desc 912 node 52496 s 1 w 1\n";
fileReader->SetFileContent("/dev/binderfs/binder_logs/proc/1234", proc_content_1234);
const std::string proc_content_5678 = "proc 5678\n"
"context binder\n"
" node 52492: u0000000000000123 c0000000000000456 pri "
"0:120 hs true hw true cs 2 cw 2 proc 111 222\n"
" node 52496: u0000000000000123 c0000000000000456 pri "
"0:120 hs true hw true cs 4 cw 4 proc 333 444 555 666\n";
fileReader->SetFileContent("/dev/binderfs/binder_logs/proc/5678", proc_content_5678);
std::vector<pid_t> pids;
status_t status = getBinderClientPids(BinderDebugContext::BINDER, clientPid, servicePid, handle,
std::move(fileReader), &pids);
ASSERT_EQ(OK, status);
ASSERT_EQ(4, pids.size());
EXPECT_EQ(333, pids[0]);
EXPECT_EQ(444, pids[1]);
EXPECT_EQ(555, pids[2]);
EXPECT_EQ(666, pids[3]);
}
TEST(BinderDebugParsingTest, GetBinderTransactions) {
const pid_t pid = 1234;
auto fileReader = std::make_unique<MockFileReader>();
const std::string transactions_content =
"proc 3456\n"
"context binder\n"
" node 1: u0000000012345678 c0000000087654321 pri 0:120 hs true hw true cs 1 cw 1 "
"proc 5678 9012\n"
"proc 1234\n"
"context binder\n"
" pending transaction 123 from ...\n"
" more data for 1234\n"
"proc 6543\n"
"context binder\n"
" node work 1111: u0000000012345678 c0000000087654321\n";
fileReader->SetFileContent("/dev/binderfs/binder_logs/transactions", transactions_content);
std::string output;
status_t status = getBinderTransactions(pid, std::move(fileReader), output);
ASSERT_EQ(OK, status);
EXPECT_EQ("proc 1234\ncontext binder\n pending transaction 123 from ...\n more data for "
"1234\n",
output);
}
} // namespace test
} // namespace binderdebug
} // namespace android
int main(int argc, char** argv) {
using namespace android;
using namespace android::binderdebug;
using namespace android::binderdebug::test;
::testing::InitGoogleTest(&argc, argv);
// Create a child/client process to call into the main process so we can ensure
// looper thread has been registered before attempting to get the BinderPidInfo
pid_t pid = fork();
if (pid == 0) {
sp<IBinder> binder = android::defaultServiceManager()->getService(String16("binderdebug"));
sp<IControl> service;
if (binder != nullptr) {
service = android::interface_cast<IControl>(binder);
}
service->Continue();
exit(0);
}
sp<Control> iface = new Control;
android::defaultServiceManager()->addService(String16("binderdebug"), iface);
android::ProcessState::self()->setThreadPoolMaxThreadCount(8);
ProcessState::self()->startThreadPool();
sem_wait(&iface->s);
return RUN_ALL_TESTS();
}