blob: e149946fe1ffcad8042ff3139f4b689f4b603cb8 [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/IInterface.h>
#include <gtest/gtest.h>
using android::BBinder;
using android::IBinder;
using android::OK;
using android::sp;
using android::TransactionCodeData;
const void* kObjectId1 = reinterpret_cast<const void*>(1);
const void* kObjectId2 = reinterpret_cast<const void*>(2);
void* kObject1 = reinterpret_cast<void*>(101);
void* kObject2 = reinterpret_cast<void*>(102);
void* kObject3 = reinterpret_cast<void*>(103);
TEST(Binder, AttachObject) {
auto binder = sp<BBinder>::make();
EXPECT_EQ(nullptr, binder->attachObject(kObjectId1, kObject1, nullptr, nullptr));
EXPECT_EQ(nullptr, binder->attachObject(kObjectId2, kObject2, nullptr, nullptr));
EXPECT_EQ(kObject1, binder->attachObject(kObjectId1, kObject3, nullptr, nullptr));
}
TEST(Binder, DetachObject) {
auto binder = sp<BBinder>::make();
EXPECT_EQ(nullptr, binder->attachObject(kObjectId1, kObject1, nullptr, nullptr));
EXPECT_EQ(kObject1, binder->detachObject(kObjectId1));
EXPECT_EQ(nullptr, binder->attachObject(kObjectId1, kObject2, nullptr, nullptr));
}
TEST(Binder, AttachExtension) {
auto binder = sp<BBinder>::make();
auto ext = sp<BBinder>::make();
binder->setExtension(ext);
EXPECT_EQ(ext, binder->getExtension());
}
struct MyCookie {
bool* deleted;
};
class UniqueBinder : public BBinder {
public:
UniqueBinder(const void* c) : cookie(reinterpret_cast<const MyCookie*>(c)) {
*cookie->deleted = false;
}
~UniqueBinder() { *cookie->deleted = true; }
const MyCookie* cookie;
};
static sp<IBinder> make(const void* arg) {
return sp<UniqueBinder>::make(arg);
}
TEST(Binder, LookupOrCreateWeak) {
auto binder = sp<BBinder>::make();
bool deleted;
MyCookie cookie = {&deleted};
sp<IBinder> createdBinder = binder->lookupOrCreateWeak(kObjectId1, make, &cookie);
EXPECT_NE(binder, createdBinder);
sp<IBinder> lookedUpBinder = binder->lookupOrCreateWeak(kObjectId1, make, &cookie);
EXPECT_EQ(createdBinder, lookedUpBinder);
EXPECT_FALSE(deleted);
}
TEST(Binder, LookupOrCreateWeakDropSp) {
auto binder = sp<BBinder>::make();
bool deleted1 = false;
bool deleted2 = false;
MyCookie cookie1 = {&deleted1};
MyCookie cookie2 = {&deleted2};
sp<IBinder> createdBinder = binder->lookupOrCreateWeak(kObjectId1, make, &cookie1);
EXPECT_NE(binder, createdBinder);
createdBinder.clear();
EXPECT_TRUE(deleted1);
sp<IBinder> lookedUpBinder = binder->lookupOrCreateWeak(kObjectId1, make, &cookie2);
EXPECT_EQ(&cookie2, sp<UniqueBinder>::cast(lookedUpBinder)->cookie);
EXPECT_FALSE(deleted2);
}
TEST(Binder, GetFunctionName) {
auto binder = sp<BBinder>::make();
const char* const kFunctionNames[] = {"foo", "bar"};
alignas(16) TransactionCodeData data;
data.names = kFunctionNames;
data.count = 2;
binder->setTransactionCodeMap(&data);
EXPECT_EQ("foo", binder->getFunctionName(IBinder::FIRST_CALL_TRANSACTION));
EXPECT_EQ("bar", binder->getFunctionName(IBinder::FIRST_CALL_TRANSACTION + 1));
}
TEST(Binder, GetFunctionNameNotFound) {
auto binder = sp<BBinder>::make();
std::string expected = "#" + std::to_string(IBinder::FIRST_CALL_TRANSACTION);
EXPECT_EQ(expected, binder->getFunctionName(IBinder::FIRST_CALL_TRANSACTION));
}
TEST(Binder, GetFunctionNameInvalidCode) {
auto binder = sp<BBinder>::make();
const char* const kFunctionNames[] = {"foo", "bar"};
alignas(16) TransactionCodeData data;
data.names = kFunctionNames;
data.count = 2;
binder->setTransactionCodeMap(&data);
std::string expectedBefore = "#" + std::to_string(IBinder::FIRST_CALL_TRANSACTION - 1);
EXPECT_EQ(expectedBefore, binder->getFunctionName(IBinder::FIRST_CALL_TRANSACTION - 1));
std::string expectedAfter = "#" + std::to_string(IBinder::FIRST_CALL_TRANSACTION + 2);
EXPECT_EQ(expectedAfter, binder->getFunctionName(IBinder::FIRST_CALL_TRANSACTION + 2));
std::string expectedZero = "#0";
EXPECT_EQ(expectedZero, binder->getFunctionName(0));
std::string expectedNegative = "#" + std::to_string(static_cast<size_t>(-1));
EXPECT_EQ(expectedNegative, binder->getFunctionName(-1));
}
TEST(Binder, GetFunctionNameNullMap) {
auto binder = sp<BBinder>::make();
EXPECT_DEATH(binder->setTransactionCodeMap(nullptr), "TransactionCodeData pointer is null!");
}
TEST(Binder, SetTransactionCodeMapTwice) {
auto binder = sp<BBinder>::make();
const char* const kFunctionNames1[] = {"foo", "bar"};
alignas(16) TransactionCodeData data1;
data1.names = kFunctionNames1;
data1.count = 2;
binder->setTransactionCodeMap(&data1);
const char* const kFunctionNames2[] = {"baz", "qux"};
alignas(16) TransactionCodeData data2;
data2.names = kFunctionNames2;
data2.count = 2;
EXPECT_DEATH(binder->setTransactionCodeMap(&data2), "TransactionCodeData already set!");
}
TEST(Binder, GetFunctionNameMalformed) {
auto binder = sp<BBinder>::make();
alignas(16) TransactionCodeData malformedData;
malformedData.names = nullptr;
malformedData.count = 2;
binder->setTransactionCodeMap(&malformedData);
std::string expected = "#" + std::to_string(IBinder::FIRST_CALL_TRANSACTION);
EXPECT_EQ(expected, binder->getFunctionName(IBinder::FIRST_CALL_TRANSACTION));
}
class BinderTest : public ::testing::Test {
public:
void setStability(const sp<BBinder>& binder, int16_t level) {
binder->getPrivateAccessor().setStability(level);
}
int16_t getStability(const sp<BBinder>& binder) {
return binder->getPrivateAccessor().getStability();
}
};
TEST_F(BinderTest, PackedDataFieldsInitialState) {
auto binder = sp<BBinder>::make();
EXPECT_EQ(0, getStability(binder)); // Corresponds to STABILITY_UNDECLARED
EXPECT_FALSE(binder->wasParceled());
}
TEST_F(BinderTest, PackedDataFieldsStability) {
auto binder = sp<BBinder>::make();
setStability(binder, 3); // Corresponds to STABILITY_VENDOR
EXPECT_EQ(3, getStability(binder));
setStability(binder, 12); // Corresponds to STABILITY_SYSTEM
EXPECT_EQ(12, getStability(binder));
setStability(binder, 0);
EXPECT_EQ(0, getStability(binder));
setStability(binder, -1);
EXPECT_EQ(0, getStability(binder));
}
TEST_F(BinderTest, PackedDataFieldsParceled) {
auto binder = sp<BBinder>::make();
binder->setParceled();
EXPECT_TRUE(binder->wasParceled());
}
TEST_F(BinderTest, PackedDataFieldsAllAtOnce) {
auto binder = sp<BBinder>::make();
// Set all packed fields
setStability(binder, 12); // Corresponds to STABILITY_SYSTEM
binder->setParceled();
const char* const kFunctionNames[] = {"foo"};
alignas(16) TransactionCodeData data;
data.names = kFunctionNames;
data.count = 1;
binder->setTransactionCodeMap(&data);
// Verify all fields
EXPECT_EQ(12, getStability(binder));
EXPECT_TRUE(binder->wasParceled());
EXPECT_EQ("foo", binder->getFunctionName(IBinder::FIRST_CALL_TRANSACTION));
// Change stability and verify others are unaffected
setStability(binder, 3); // Corresponds to STABILITY_VENDOR
EXPECT_EQ(3, getStability(binder));
EXPECT_TRUE(binder->wasParceled());
EXPECT_EQ("foo", binder->getFunctionName(IBinder::FIRST_CALL_TRANSACTION));
}