Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2015 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
| 16 | |
| 17 | //#define LOG_NDEBUG 0 |
| 18 | #define LOG_TAG "DrmSessionManager_test" |
Robert Shih | a90aee3 | 2019-11-14 15:43:39 -0800 | [diff] [blame] | 19 | #include <android/binder_auto_utils.h> |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 20 | #include <utils/Log.h> |
| 21 | |
| 22 | #include <gtest/gtest.h> |
| 23 | |
Robert Shih | a90aee3 | 2019-11-14 15:43:39 -0800 | [diff] [blame] | 24 | #include <aidl/android/media/BnResourceManagerClient.h> |
| 25 | #include <aidl/android/media/BnResourceManagerService.h> |
Robert Shih | a90aee3 | 2019-11-14 15:43:39 -0800 | [diff] [blame] | 26 | |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 27 | #include <media/stagefright/foundation/ADebug.h> |
Ronghua Wu | 14bcaca | 2015-03-16 11:24:30 -0700 | [diff] [blame] | 28 | #include <media/stagefright/ProcessInfoInterface.h> |
Jeff Tinker | 7d2c6e8 | 2018-02-16 16:14:59 -0800 | [diff] [blame] | 29 | #include <mediadrm/DrmSessionManager.h> |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 30 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 31 | #include <algorithm> |
Robert Shih | a90aee3 | 2019-11-14 15:43:39 -0800 | [diff] [blame] | 32 | #include <iostream> |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 33 | #include <vector> |
| 34 | |
| 35 | #include "ResourceManagerService.h" |
| 36 | |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 37 | namespace android { |
| 38 | |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 39 | using Status = ::ndk::ScopedAStatus; |
| 40 | using ::aidl::android::media::BnResourceManagerClient; |
| 41 | using ::aidl::android::media::BnResourceManagerService; |
| 42 | using ::aidl::android::media::MediaResourceParcel; |
| 43 | using ::aidl::android::media::IResourceManagerClient; |
Chong Zhang | 181e695 | 2019-10-09 13:23:39 -0700 | [diff] [blame] | 44 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 45 | static Vector<uint8_t> toAndroidVector(const std::vector<uint8_t> &vec) { |
| 46 | Vector<uint8_t> aVec; |
| 47 | for (auto b : vec) { |
| 48 | aVec.push_back(b); |
| 49 | } |
| 50 | return aVec; |
| 51 | } |
| 52 | |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 53 | struct FakeProcessInfo : public ProcessInfoInterface { |
| 54 | FakeProcessInfo() {} |
| 55 | virtual ~FakeProcessInfo() {} |
| 56 | |
Ronghua Wu | 5c3da20 | 2015-02-22 08:45:28 -0800 | [diff] [blame] | 57 | virtual bool getPriority(int pid, int* priority) { |
| 58 | // For testing, use pid as priority. |
| 59 | // Lower the value higher the priority. |
| 60 | *priority = pid; |
| 61 | return true; |
| 62 | } |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 63 | |
Ronghua Wu | d11c43a | 2016-01-27 16:26:12 -0800 | [diff] [blame] | 64 | virtual bool isValidPid(int /* pid */) { |
| 65 | return true; |
| 66 | } |
| 67 | |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 68 | private: |
| 69 | DISALLOW_EVIL_CONSTRUCTORS(FakeProcessInfo); |
| 70 | }; |
| 71 | |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 72 | struct FakeDrm : public BnResourceManagerClient { |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 73 | FakeDrm(const std::vector<uint8_t>& sessionId, const sp<DrmSessionManager>& manager) |
| 74 | : mSessionId(toAndroidVector(sessionId)), |
| 75 | mReclaimed(false), |
| 76 | mDrmSessionManager(manager) {} |
| 77 | |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 78 | Status reclaimResource(bool* _aidl_return) { |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 79 | mReclaimed = true; |
| 80 | mDrmSessionManager->removeSession(mSessionId); |
Chong Zhang | 181e695 | 2019-10-09 13:23:39 -0700 | [diff] [blame] | 81 | *_aidl_return = true; |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 82 | return Status::ok(); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 83 | } |
| 84 | |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 85 | Status getName(::std::string* _aidl_return) { |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 86 | String8 name("FakeDrm["); |
| 87 | for (size_t i = 0; i < mSessionId.size(); ++i) { |
| 88 | name.appendFormat("%02x", mSessionId[i]); |
| 89 | } |
| 90 | name.append("]"); |
Chong Zhang | 181e695 | 2019-10-09 13:23:39 -0700 | [diff] [blame] | 91 | *_aidl_return = name; |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 92 | return Status::ok(); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 93 | } |
| 94 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 95 | bool isReclaimed() const { |
| 96 | return mReclaimed; |
| 97 | } |
| 98 | |
| 99 | const Vector<uint8_t> mSessionId; |
| 100 | |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 101 | private: |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 102 | bool mReclaimed; |
| 103 | const sp<DrmSessionManager> mDrmSessionManager; |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 104 | |
| 105 | DISALLOW_EVIL_CONSTRUCTORS(FakeDrm); |
| 106 | }; |
| 107 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 108 | struct FakeSystemCallback : |
| 109 | public ResourceManagerService::SystemCallbackInterface { |
| 110 | FakeSystemCallback() {} |
| 111 | |
| 112 | virtual void noteStartVideo(int /*uid*/) override {} |
| 113 | |
| 114 | virtual void noteStopVideo(int /*uid*/) override {} |
| 115 | |
| 116 | virtual void noteResetVideo() override {} |
| 117 | |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 118 | virtual bool requestCpusetBoost(bool /*enable*/) override { |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 119 | return true; |
| 120 | } |
| 121 | |
| 122 | protected: |
| 123 | virtual ~FakeSystemCallback() {} |
| 124 | |
| 125 | private: |
| 126 | |
| 127 | DISALLOW_EVIL_CONSTRUCTORS(FakeSystemCallback); |
| 128 | }; |
| 129 | |
Ronghua Wu | 5c3da20 | 2015-02-22 08:45:28 -0800 | [diff] [blame] | 130 | static const int kTestPid1 = 30; |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 131 | static const int kTestPid2 = 20; |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 132 | static const std::vector<uint8_t> kTestSessionId1{1, 2, 3}; |
| 133 | static const std::vector<uint8_t> kTestSessionId2{4, 5, 6, 7, 8}; |
| 134 | static const std::vector<uint8_t> kTestSessionId3{9, 0}; |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 135 | |
| 136 | class DrmSessionManagerTest : public ::testing::Test { |
| 137 | public: |
| 138 | DrmSessionManagerTest() |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 139 | : mService(::ndk::SharedRefBase::make<ResourceManagerService> |
| 140 | (new FakeProcessInfo(), new FakeSystemCallback())), |
| 141 | mDrmSessionManager(new DrmSessionManager(mService)), |
| 142 | mTestDrm1(::ndk::SharedRefBase::make<FakeDrm>( |
| 143 | kTestSessionId1, mDrmSessionManager)), |
| 144 | mTestDrm2(::ndk::SharedRefBase::make<FakeDrm>( |
| 145 | kTestSessionId2, mDrmSessionManager)), |
| 146 | mTestDrm3(::ndk::SharedRefBase::make<FakeDrm>( |
| 147 | kTestSessionId3, mDrmSessionManager)) { |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 148 | } |
| 149 | |
| 150 | protected: |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 151 | void addSession() { |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 152 | mDrmSessionManager->addSession(kTestPid1, mTestDrm1, mTestDrm1->mSessionId); |
| 153 | mDrmSessionManager->addSession(kTestPid2, mTestDrm2, mTestDrm2->mSessionId); |
| 154 | mDrmSessionManager->addSession(kTestPid2, mTestDrm3, mTestDrm3->mSessionId); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 155 | } |
| 156 | |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 157 | std::shared_ptr<ResourceManagerService> mService; |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 158 | sp<DrmSessionManager> mDrmSessionManager; |
Robert Shih | a90aee3 | 2019-11-14 15:43:39 -0800 | [diff] [blame] | 159 | std::shared_ptr<FakeDrm> mTestDrm1; |
| 160 | std::shared_ptr<FakeDrm> mTestDrm2; |
| 161 | std::shared_ptr<FakeDrm> mTestDrm3; |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 162 | }; |
| 163 | |
| 164 | TEST_F(DrmSessionManagerTest, addSession) { |
| 165 | addSession(); |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 166 | |
| 167 | EXPECT_EQ(3u, mDrmSessionManager->getSessionCount()); |
| 168 | EXPECT_TRUE(mDrmSessionManager->containsSession(mTestDrm1->mSessionId)); |
| 169 | EXPECT_TRUE(mDrmSessionManager->containsSession(mTestDrm2->mSessionId)); |
| 170 | EXPECT_TRUE(mDrmSessionManager->containsSession(mTestDrm3->mSessionId)); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 171 | } |
| 172 | |
| 173 | TEST_F(DrmSessionManagerTest, useSession) { |
| 174 | addSession(); |
| 175 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 176 | mDrmSessionManager->useSession(mTestDrm1->mSessionId); |
| 177 | mDrmSessionManager->useSession(mTestDrm3->mSessionId); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 178 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 179 | EXPECT_EQ(3u, mDrmSessionManager->getSessionCount()); |
| 180 | EXPECT_TRUE(mDrmSessionManager->containsSession(mTestDrm1->mSessionId)); |
| 181 | EXPECT_TRUE(mDrmSessionManager->containsSession(mTestDrm2->mSessionId)); |
| 182 | EXPECT_TRUE(mDrmSessionManager->containsSession(mTestDrm3->mSessionId)); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 183 | } |
| 184 | |
| 185 | TEST_F(DrmSessionManagerTest, removeSession) { |
| 186 | addSession(); |
| 187 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 188 | mDrmSessionManager->removeSession(mTestDrm2->mSessionId); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 189 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 190 | EXPECT_EQ(2u, mDrmSessionManager->getSessionCount()); |
| 191 | EXPECT_TRUE(mDrmSessionManager->containsSession(mTestDrm1->mSessionId)); |
| 192 | EXPECT_FALSE(mDrmSessionManager->containsSession(mTestDrm2->mSessionId)); |
| 193 | EXPECT_TRUE(mDrmSessionManager->containsSession(mTestDrm3->mSessionId)); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 194 | } |
| 195 | |
| 196 | TEST_F(DrmSessionManagerTest, reclaimSession) { |
| 197 | EXPECT_FALSE(mDrmSessionManager->reclaimSession(kTestPid1)); |
| 198 | addSession(); |
| 199 | |
| 200 | // calling pid priority is too low |
Ronghua Wu | 5c3da20 | 2015-02-22 08:45:28 -0800 | [diff] [blame] | 201 | EXPECT_FALSE(mDrmSessionManager->reclaimSession(50)); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 202 | |
Ronghua Wu | 5c3da20 | 2015-02-22 08:45:28 -0800 | [diff] [blame] | 203 | EXPECT_TRUE(mDrmSessionManager->reclaimSession(10)); |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 204 | EXPECT_TRUE(mTestDrm1->isReclaimed()); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 205 | |
| 206 | // add a session from a higher priority process. |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 207 | const std::vector<uint8_t> sid{1, 3, 5}; |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 208 | std::shared_ptr<FakeDrm> drm = |
| 209 | ::ndk::SharedRefBase::make<FakeDrm>(sid, mDrmSessionManager); |
| 210 | mDrmSessionManager->addSession(15, drm, drm->mSessionId); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 211 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 212 | // make sure mTestDrm2 is reclaimed next instead of mTestDrm3 |
| 213 | mDrmSessionManager->useSession(mTestDrm3->mSessionId); |
Ronghua Wu | 5c3da20 | 2015-02-22 08:45:28 -0800 | [diff] [blame] | 214 | EXPECT_TRUE(mDrmSessionManager->reclaimSession(18)); |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 215 | EXPECT_TRUE(mTestDrm2->isReclaimed()); |
| 216 | |
| 217 | EXPECT_EQ(2u, mDrmSessionManager->getSessionCount()); |
| 218 | EXPECT_FALSE(mDrmSessionManager->containsSession(mTestDrm1->mSessionId)); |
| 219 | EXPECT_FALSE(mDrmSessionManager->containsSession(mTestDrm2->mSessionId)); |
| 220 | EXPECT_TRUE(mDrmSessionManager->containsSession(mTestDrm3->mSessionId)); |
| 221 | EXPECT_TRUE(mDrmSessionManager->containsSession(drm->mSessionId)); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 222 | } |
| 223 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 224 | TEST_F(DrmSessionManagerTest, reclaimAfterUse) { |
| 225 | // nothing to reclaim yet |
| 226 | EXPECT_FALSE(mDrmSessionManager->reclaimSession(kTestPid1)); |
| 227 | EXPECT_FALSE(mDrmSessionManager->reclaimSession(kTestPid2)); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 228 | |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 229 | // add sessions from same pid |
Chong Zhang | e45ac60 | 2019-11-26 14:35:06 -0800 | [diff] [blame] | 230 | mDrmSessionManager->addSession(kTestPid2, mTestDrm1, mTestDrm1->mSessionId); |
| 231 | mDrmSessionManager->addSession(kTestPid2, mTestDrm2, mTestDrm2->mSessionId); |
| 232 | mDrmSessionManager->addSession(kTestPid2, mTestDrm3, mTestDrm3->mSessionId); |
Robert Shih | c3af31b | 2019-09-20 21:45:01 -0700 | [diff] [blame] | 233 | |
| 234 | // use some but not all sessions |
| 235 | mDrmSessionManager->useSession(mTestDrm1->mSessionId); |
| 236 | mDrmSessionManager->useSession(mTestDrm1->mSessionId); |
| 237 | mDrmSessionManager->useSession(mTestDrm2->mSessionId); |
| 238 | |
| 239 | // calling pid priority is too low |
| 240 | int lowPriorityPid = kTestPid2 + 1; |
| 241 | EXPECT_FALSE(mDrmSessionManager->reclaimSession(lowPriorityPid)); |
| 242 | |
| 243 | // unused session is reclaimed first |
| 244 | int highPriorityPid = kTestPid2 - 1; |
| 245 | EXPECT_TRUE(mDrmSessionManager->reclaimSession(highPriorityPid)); |
| 246 | EXPECT_FALSE(mTestDrm1->isReclaimed()); |
| 247 | EXPECT_FALSE(mTestDrm2->isReclaimed()); |
| 248 | EXPECT_TRUE(mTestDrm3->isReclaimed()); |
| 249 | mDrmSessionManager->removeSession(mTestDrm3->mSessionId); |
| 250 | |
| 251 | // less-used session is reclaimed next |
| 252 | EXPECT_TRUE(mDrmSessionManager->reclaimSession(highPriorityPid)); |
| 253 | EXPECT_FALSE(mTestDrm1->isReclaimed()); |
| 254 | EXPECT_TRUE(mTestDrm2->isReclaimed()); |
| 255 | EXPECT_TRUE(mTestDrm3->isReclaimed()); |
| 256 | |
| 257 | // most-used session still open |
| 258 | EXPECT_EQ(1u, mDrmSessionManager->getSessionCount()); |
| 259 | EXPECT_TRUE(mDrmSessionManager->containsSession(mTestDrm1->mSessionId)); |
| 260 | EXPECT_FALSE(mDrmSessionManager->containsSession(mTestDrm2->mSessionId)); |
| 261 | EXPECT_FALSE(mDrmSessionManager->containsSession(mTestDrm3->mSessionId)); |
Ronghua Wu | 10305cc | 2015-02-22 07:55:32 -0800 | [diff] [blame] | 262 | } |
| 263 | |
| 264 | } // namespace android |