1 /*
2 * Copyright (C) 2020 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 "TranscodingSessionController"
19
20 #define VALIDATE_STATE 1
21
22 #include <android/permission_manager.h>
23 #include <inttypes.h>
24 #include <media/TranscodingSessionController.h>
25 #include <media/TranscodingUidPolicy.h>
26 #include <utils/AndroidThreads.h>
27 #include <utils/Log.h>
28
29 #include <thread>
30 #include <utility>
31
32 namespace android {
33
34 static_assert((SessionIdType)-1 < 0, "SessionIdType should be signed");
35
36 constexpr static uid_t OFFLINE_UID = -1;
37 constexpr static size_t kSessionHistoryMax = 100;
38
39 //static
sessionToString(const SessionKeyType & sessionKey)40 String8 TranscodingSessionController::sessionToString(const SessionKeyType& sessionKey) {
41 return String8::format("{client:%lld, session:%d}", (long long)sessionKey.first,
42 sessionKey.second);
43 }
44
45 //static
sessionStateToString(const Session::State sessionState)46 const char* TranscodingSessionController::sessionStateToString(const Session::State sessionState) {
47 switch (sessionState) {
48 case Session::State::NOT_STARTED:
49 return "NOT_STARTED";
50 case Session::State::RUNNING:
51 return "RUNNING";
52 case Session::State::PAUSED:
53 return "PAUSED";
54 case Session::State::FINISHED:
55 return "FINISHED";
56 case Session::State::CANCELED:
57 return "CANCELED";
58 case Session::State::ERROR:
59 return "ERROR";
60 default:
61 break;
62 }
63 return "(unknown)";
64 }
65
66 ///////////////////////////////////////////////////////////////////////////////
67 struct TranscodingSessionController::Watchdog {
68 Watchdog(TranscodingSessionController* owner, int64_t timeoutUs);
69 ~Watchdog();
70
71 // Starts monitoring the session.
72 void start(const SessionKeyType& key);
73 // Stops monitoring the session.
74 void stop();
75 // Signals that the session is still alive. Must be sent at least every mTimeoutUs.
76 // (Timeout will happen if no ping in mTimeoutUs since the last ping.)
77 void keepAlive();
78
79 private:
80 void threadLoop();
81 void updateTimer_l();
82
83 TranscodingSessionController* mOwner;
84 const int64_t mTimeoutUs;
85 mutable std::mutex mLock;
86 std::condition_variable mCondition GUARDED_BY(mLock);
87 // Whether watchdog is monitoring a session for timeout.
88 bool mActive GUARDED_BY(mLock);
89 // Whether watchdog is aborted and the monitoring thread should exit.
90 bool mAbort GUARDED_BY(mLock);
91 // When watchdog is active, the next timeout time point.
92 std::chrono::steady_clock::time_point mNextTimeoutTime GUARDED_BY(mLock);
93 // When watchdog is active, the session being watched.
94 SessionKeyType mSessionToWatch GUARDED_BY(mLock);
95 std::thread mThread;
96 };
97
Watchdog(TranscodingSessionController * owner,int64_t timeoutUs)98 TranscodingSessionController::Watchdog::Watchdog(TranscodingSessionController* owner,
99 int64_t timeoutUs)
100 : mOwner(owner),
101 mTimeoutUs(timeoutUs),
102 mActive(false),
103 mAbort(false),
104 mThread(&Watchdog::threadLoop, this) {
105 ALOGV("Watchdog CTOR: %p", this);
106 }
107
~Watchdog()108 TranscodingSessionController::Watchdog::~Watchdog() {
109 ALOGV("Watchdog DTOR: %p", this);
110
111 {
112 // Exit the looper thread.
113 std::scoped_lock lock{mLock};
114
115 mAbort = true;
116 mCondition.notify_one();
117 }
118
119 mThread.join();
120 ALOGV("Watchdog DTOR: %p, done.", this);
121 }
122
start(const SessionKeyType & key)123 void TranscodingSessionController::Watchdog::start(const SessionKeyType& key) {
124 std::scoped_lock lock{mLock};
125
126 if (!mActive) {
127 ALOGI("Watchdog start: %s", sessionToString(key).c_str());
128
129 mActive = true;
130 mSessionToWatch = key;
131 updateTimer_l();
132 mCondition.notify_one();
133 }
134 }
135
stop()136 void TranscodingSessionController::Watchdog::stop() {
137 std::scoped_lock lock{mLock};
138
139 if (mActive) {
140 ALOGI("Watchdog stop: %s", sessionToString(mSessionToWatch).c_str());
141
142 mActive = false;
143 mCondition.notify_one();
144 }
145 }
146
keepAlive()147 void TranscodingSessionController::Watchdog::keepAlive() {
148 std::scoped_lock lock{mLock};
149
150 if (mActive) {
151 ALOGI("Watchdog keepAlive: %s", sessionToString(mSessionToWatch).c_str());
152
153 updateTimer_l();
154 mCondition.notify_one();
155 }
156 }
157
158 // updateTimer_l() is only called with lock held.
updateTimer_l()159 void TranscodingSessionController::Watchdog::updateTimer_l() NO_THREAD_SAFETY_ANALYSIS {
160 std::chrono::microseconds timeout(mTimeoutUs);
161 mNextTimeoutTime = std::chrono::steady_clock::now() + timeout;
162 }
163
164 // Unfortunately std::unique_lock is incompatible with -Wthread-safety.
threadLoop()165 void TranscodingSessionController::Watchdog::threadLoop() NO_THREAD_SAFETY_ANALYSIS {
166 androidSetThreadPriority(0 /*tid (0 = current) */, ANDROID_PRIORITY_BACKGROUND);
167 std::unique_lock<std::mutex> lock{mLock};
168
169 while (!mAbort) {
170 if (!mActive) {
171 mCondition.wait(lock);
172 continue;
173 }
174 // Watchdog active, wait till next timeout time.
175 if (mCondition.wait_until(lock, mNextTimeoutTime) == std::cv_status::timeout) {
176 // If timeout happens, report timeout and deactivate watchdog.
177 mActive = false;
178 // Make a copy of session key, as once we unlock, it could be unprotected.
179 SessionKeyType sessionKey = mSessionToWatch;
180
181 ALOGE("Watchdog timeout: %s", sessionToString(sessionKey).c_str());
182
183 lock.unlock();
184 mOwner->onError(sessionKey.first, sessionKey.second,
185 TranscodingErrorCode::kWatchdogTimeout);
186 lock.lock();
187 }
188 }
189 }
190 ///////////////////////////////////////////////////////////////////////////////
191 struct TranscodingSessionController::Pacer {
Pacerandroid::TranscodingSessionController::Pacer192 Pacer(const ControllerConfig& config)
193 : mBurstThresholdMs(config.pacerBurstThresholdMs),
194 mBurstCountQuota(config.pacerBurstCountQuota),
195 mBurstTimeQuotaSec(config.pacerBurstTimeQuotaSeconds) {}
196
197 ~Pacer() = default;
198
199 bool onSessionStarted(uid_t uid, uid_t callingUid);
200 void onSessionCompleted(uid_t uid, std::chrono::microseconds runningTime);
201 void onSessionCancelled(uid_t uid);
202
203 private:
204 // Threshold of time between finish/start below which a back-to-back start is counted.
205 int32_t mBurstThresholdMs;
206 // Maximum allowed back-to-back start count.
207 int32_t mBurstCountQuota;
208 // Maximum allowed back-to-back running time.
209 int32_t mBurstTimeQuotaSec;
210
211 struct UidHistoryEntry {
212 bool sessionActive = false;
213 int32_t burstCount = 0;
214 std::chrono::steady_clock::duration burstDuration{0};
215 std::chrono::steady_clock::time_point lastCompletedTime;
216 };
217 std::map<uid_t, UidHistoryEntry> mUidHistoryMap;
218 std::unordered_set<uid_t> mMtpUids;
219 std::unordered_set<uid_t> mNonMtpUids;
220
221 bool isSubjectToQuota(uid_t uid, uid_t callingUid);
222 };
223
isSubjectToQuota(uid_t uid,uid_t callingUid)224 bool TranscodingSessionController::Pacer::isSubjectToQuota(uid_t uid, uid_t callingUid) {
225 // Submitting with self uid is not limited (which can only happen if it's used as an
226 // app-facing API). MediaProvider usage always submit on behalf of other uids.
227 if (uid == callingUid) {
228 return false;
229 }
230
231 if (mMtpUids.find(uid) != mMtpUids.end()) {
232 return false;
233 }
234
235 if (mNonMtpUids.find(uid) != mNonMtpUids.end()) {
236 return true;
237 }
238
239 // We don't have MTP permission info about this uid yet, check permission and save the result.
240 int32_t result;
241 if (__builtin_available(android __TRANSCODING_MIN_API__, *)) {
242 if (APermissionManager_checkPermission("android.permission.ACCESS_MTP", -1 /*pid*/, uid,
243 &result) == PERMISSION_MANAGER_STATUS_OK &&
244 result == PERMISSION_MANAGER_PERMISSION_GRANTED) {
245 mMtpUids.insert(uid);
246 return false;
247 }
248 }
249
250 mNonMtpUids.insert(uid);
251 return true;
252 }
253
onSessionStarted(uid_t uid,uid_t callingUid)254 bool TranscodingSessionController::Pacer::onSessionStarted(uid_t uid, uid_t callingUid) {
255 if (!isSubjectToQuota(uid, callingUid)) {
256 ALOGI("Pacer::onSessionStarted: uid %d (caling uid: %d): not subject to quota", uid,
257 callingUid);
258 return true;
259 }
260
261 // If uid doesn't exist, only insert the entry and mark session active. Skip quota checking.
262 if (mUidHistoryMap.find(uid) == mUidHistoryMap.end()) {
263 mUidHistoryMap.emplace(uid, UidHistoryEntry{});
264 mUidHistoryMap[uid].sessionActive = true;
265 ALOGV("Pacer::onSessionStarted: uid %d: new", uid);
266 return true;
267 }
268
269 // TODO: if Thermal throttling or resoure lost happened to occurr between this start
270 // and the previous completion, we should deduct the paused time from the elapsed time.
271 // (Individual session's pause time, on the other hand, doesn't need to be deducted
272 // because it doesn't affect the gap between last completion and the start.
273 auto timeSinceLastComplete =
274 std::chrono::steady_clock::now() - mUidHistoryMap[uid].lastCompletedTime;
275 if (mUidHistoryMap[uid].burstCount >= mBurstCountQuota &&
276 mUidHistoryMap[uid].burstDuration >= std::chrono::seconds(mBurstTimeQuotaSec)) {
277 ALOGW("Pacer::onSessionStarted: uid %d: over quota, burst count %d, time %lldms", uid,
278 mUidHistoryMap[uid].burstCount,
279 (long long)mUidHistoryMap[uid].burstDuration.count() / 1000000);
280 return false;
281 }
282
283 // If not over quota, allow the session, and reset as long as this is not too close
284 // to previous completion.
285 if (timeSinceLastComplete > std::chrono::milliseconds(mBurstThresholdMs)) {
286 ALOGV("Pacer::onSessionStarted: uid %d: reset quota", uid);
287 mUidHistoryMap[uid].burstCount = 0;
288 mUidHistoryMap[uid].burstDuration = std::chrono::milliseconds(0);
289 } else {
290 ALOGV("Pacer::onSessionStarted: uid %d: burst count %d, time %lldms", uid,
291 mUidHistoryMap[uid].burstCount,
292 (long long)mUidHistoryMap[uid].burstDuration.count() / 1000000);
293 }
294
295 mUidHistoryMap[uid].sessionActive = true;
296 return true;
297 }
298
onSessionCompleted(uid_t uid,std::chrono::microseconds runningTime)299 void TranscodingSessionController::Pacer::onSessionCompleted(
300 uid_t uid, std::chrono::microseconds runningTime) {
301 // Skip quota update if this uid missed the start. (Could happen if the uid is added via
302 // addClientUid() after the session start.)
303 if (mUidHistoryMap.find(uid) == mUidHistoryMap.end() || !mUidHistoryMap[uid].sessionActive) {
304 ALOGV("Pacer::onSessionCompleted: uid %d: not started", uid);
305 return;
306 }
307 ALOGV("Pacer::onSessionCompleted: uid %d: runningTime %lld", uid, runningTime.count() / 1000);
308 mUidHistoryMap[uid].sessionActive = false;
309 mUidHistoryMap[uid].burstCount++;
310 mUidHistoryMap[uid].burstDuration += runningTime;
311 mUidHistoryMap[uid].lastCompletedTime = std::chrono::steady_clock::now();
312 }
313
onSessionCancelled(uid_t uid)314 void TranscodingSessionController::Pacer::onSessionCancelled(uid_t uid) {
315 if (mUidHistoryMap.find(uid) == mUidHistoryMap.end()) {
316 ALOGV("Pacer::onSessionCancelled: uid %d: not present", uid);
317 return;
318 }
319 // This is only called if a uid is removed from a session (due to it being killed
320 // or the original submitting client was gone but session was kept for offline use).
321 // Since the uid is going to miss the onSessionCompleted(), we can't track this
322 // session, and have to check back at next onSessionStarted().
323 mUidHistoryMap[uid].sessionActive = false;
324 }
325
326 ///////////////////////////////////////////////////////////////////////////////
327
TranscodingSessionController(const TranscoderFactoryType & transcoderFactory,const std::shared_ptr<UidPolicyInterface> & uidPolicy,const std::shared_ptr<ResourcePolicyInterface> & resourcePolicy,const std::shared_ptr<ThermalPolicyInterface> & thermalPolicy,const ControllerConfig * config)328 TranscodingSessionController::TranscodingSessionController(
329 const TranscoderFactoryType& transcoderFactory,
330 const std::shared_ptr<UidPolicyInterface>& uidPolicy,
331 const std::shared_ptr<ResourcePolicyInterface>& resourcePolicy,
332 const std::shared_ptr<ThermalPolicyInterface>& thermalPolicy,
333 const ControllerConfig* config)
334 : mTranscoderFactory(transcoderFactory),
335 mUidPolicy(uidPolicy),
336 mResourcePolicy(resourcePolicy),
337 mThermalPolicy(thermalPolicy),
338 mCurrentSession(nullptr),
339 mResourceLost(false) {
340 // Only push empty offline queue initially. Realtime queues are added when requests come in.
341 mUidSortedList.push_back(OFFLINE_UID);
342 mOfflineUidIterator = mUidSortedList.begin();
343 mSessionQueues.emplace(OFFLINE_UID, SessionQueueType());
344 mUidPackageNames[OFFLINE_UID] = "(offline)";
345 mThermalThrottling = thermalPolicy->getThrottlingStatus();
346 if (config != nullptr) {
347 mConfig = *config;
348 }
349 mPacer.reset(new Pacer(mConfig));
350 ALOGD("@@@ watchdog %lld, burst count %d, burst time %d, burst threshold %d",
351 (long long)mConfig.watchdogTimeoutUs, mConfig.pacerBurstCountQuota,
352 mConfig.pacerBurstTimeQuotaSeconds, mConfig.pacerBurstThresholdMs);
353 }
354
~TranscodingSessionController()355 TranscodingSessionController::~TranscodingSessionController() {}
356
dumpSession_l(const Session & session,String8 & result,bool closedSession)357 void TranscodingSessionController::dumpSession_l(const Session& session, String8& result,
358 bool closedSession) {
359 const size_t SIZE = 256;
360 char buffer[SIZE];
361 const TranscodingRequestParcel& request = session.request;
362 snprintf(buffer, SIZE, " Session: %s, %s, %d%%\n", sessionToString(session.key).c_str(),
363 sessionStateToString(session.getState()), session.lastProgress);
364 result.append(buffer);
365 snprintf(buffer, SIZE, " pkg: %s\n", request.clientPackageName.c_str());
366 result.append(buffer);
367 snprintf(buffer, SIZE, " src: %s\n", request.sourceFilePath.c_str());
368 result.append(buffer);
369 snprintf(buffer, SIZE, " dst: %s\n", request.destinationFilePath.c_str());
370 result.append(buffer);
371
372 if (closedSession) {
373 snprintf(buffer, SIZE,
374 " waiting: %.1fs, running: %.1fs, paused: %.1fs, paused count: %d\n",
375 session.waitingTime.count() / 1000000.0f, session.runningTime.count() / 1000000.0f,
376 session.pausedTime.count() / 1000000.0f, session.pauseCount);
377 result.append(buffer);
378 }
379 }
380
dumpAllSessions(int fd,const Vector<String16> & args __unused)381 void TranscodingSessionController::dumpAllSessions(int fd, const Vector<String16>& args __unused) {
382 String8 result;
383
384 const size_t SIZE = 256;
385 char buffer[SIZE];
386 std::scoped_lock lock{mLock};
387
388 snprintf(buffer, SIZE, "\n========== Dumping live sessions queues =========\n");
389 result.append(buffer);
390 snprintf(buffer, SIZE, " Total num of Sessions: %zu\n", mSessionMap.size());
391 result.append(buffer);
392
393 std::vector<int32_t> uids(mUidSortedList.begin(), mUidSortedList.end());
394
395 for (int32_t i = 0; i < uids.size(); i++) {
396 const uid_t uid = uids[i];
397
398 if (mSessionQueues[uid].empty()) {
399 continue;
400 }
401 snprintf(buffer, SIZE, " uid: %d, pkg: %s\n", uid,
402 mUidPackageNames.count(uid) > 0 ? mUidPackageNames[uid].c_str() : "(unknown)");
403 result.append(buffer);
404 snprintf(buffer, SIZE, " Num of sessions: %zu\n", mSessionQueues[uid].size());
405 result.append(buffer);
406 for (auto& sessionKey : mSessionQueues[uid]) {
407 auto sessionIt = mSessionMap.find(sessionKey);
408 if (sessionIt == mSessionMap.end()) {
409 snprintf(buffer, SIZE, "Failed to look up Session %s \n",
410 sessionToString(sessionKey).c_str());
411 result.append(buffer);
412 continue;
413 }
414 dumpSession_l(sessionIt->second, result);
415 }
416 }
417
418 snprintf(buffer, SIZE, "\n========== Dumping past sessions =========\n");
419 result.append(buffer);
420 for (auto& session : mSessionHistory) {
421 dumpSession_l(session, result, true /*closedSession*/);
422 }
423
424 write(fd, result.c_str(), result.size());
425 }
426
427 /*
428 * Returns nullptr if there is no session, or we're paused globally (due to resource lost,
429 * thermal throttling, etc.). Otherwise, return the session that should be run next.
430 */
getTopSession_l()431 TranscodingSessionController::Session* TranscodingSessionController::getTopSession_l() {
432 if (mSessionMap.empty()) {
433 return nullptr;
434 }
435
436 // Return nullptr if we're paused globally due to resource lost or thermal throttling.
437 if (((mResourcePolicy != nullptr && mResourceLost) ||
438 (mThermalPolicy != nullptr && mThermalThrottling))) {
439 return nullptr;
440 }
441
442 uid_t topUid = *mUidSortedList.begin();
443 // If the current session is running, and it's in the topUid's queue, let it continue
444 // to run even if it's not the earliest in that uid's queue.
445 // For example, uid(B) is added to a session while it's pending in uid(A)'s queue, then
446 // B is brought to front which caused the session to run, then user switches back to A.
447 if (mCurrentSession != nullptr && mCurrentSession->getState() == Session::RUNNING &&
448 mCurrentSession->allClientUids.count(topUid) > 0) {
449 return mCurrentSession;
450 }
451 SessionKeyType topSessionKey = *mSessionQueues[topUid].begin();
452 return &mSessionMap[topSessionKey];
453 }
454
setSessionState_l(Session * session,Session::State state)455 void TranscodingSessionController::setSessionState_l(Session* session, Session::State state) {
456 bool wasRunning = (session->getState() == Session::RUNNING);
457 session->setState(state);
458 bool isRunning = (session->getState() == Session::RUNNING);
459
460 if (wasRunning == isRunning) {
461 return;
462 }
463
464 // Currently we only have 1 running session, and we always put the previous
465 // session in non-running state before we run the new session, so it's okay
466 // to start/stop the watchdog here. If this assumption changes, we need to
467 // track the number of running sessions and start/stop watchdog based on that.
468 if (isRunning) {
469 mWatchdog->start(session->key);
470 } else {
471 mWatchdog->stop();
472 }
473 }
474
setState(Session::State newState)475 void TranscodingSessionController::Session::setState(Session::State newState) {
476 if (state == newState) {
477 return;
478 }
479 auto nowTime = std::chrono::steady_clock::now();
480 if (state != INVALID) {
481 std::chrono::microseconds elapsedTime =
482 std::chrono::duration_cast<std::chrono::microseconds>(nowTime - stateEnterTime);
483 switch (state) {
484 case PAUSED:
485 pausedTime = pausedTime + elapsedTime;
486 break;
487 case RUNNING:
488 runningTime = runningTime + elapsedTime;
489 break;
490 case NOT_STARTED:
491 waitingTime = waitingTime + elapsedTime;
492 break;
493 default:
494 break;
495 }
496 }
497 if (newState == PAUSED) {
498 pauseCount++;
499 }
500 stateEnterTime = nowTime;
501 state = newState;
502 }
503
updateCurrentSession_l()504 void TranscodingSessionController::updateCurrentSession_l() {
505 Session* curSession = mCurrentSession;
506 Session* topSession = nullptr;
507
508 // Delayed init of transcoder and watchdog.
509 if (mTranscoder == nullptr) {
510 mTranscoder = mTranscoderFactory(shared_from_this());
511 mWatchdog = std::make_shared<Watchdog>(this, mConfig.watchdogTimeoutUs);
512 }
513
514 // If we found a different top session, or the top session's running state is not
515 // correct. Take some actions to ensure it's correct.
516 while ((topSession = getTopSession_l()) != curSession ||
517 (topSession != nullptr && !topSession->isRunning())) {
518 ALOGV("updateCurrentSession_l: topSession is %s, curSession is %s",
519 topSession == nullptr ? "null" : sessionToString(topSession->key).c_str(),
520 curSession == nullptr ? "null" : sessionToString(curSession->key).c_str());
521
522 // If current session is running, pause it first. Note this is needed for either
523 // cases: 1) Top session is changing to another session, or 2) Top session is
524 // changing to null (which means we should be globally paused).
525 if (curSession != nullptr && curSession->getState() == Session::RUNNING) {
526 mTranscoder->pause(curSession->key.first, curSession->key.second);
527 setSessionState_l(curSession, Session::PAUSED);
528 }
529
530 if (topSession == nullptr) {
531 // Nothing more to run (either no session or globally paused).
532 break;
533 }
534
535 // Otherwise, ensure topSession is running.
536 if (topSession->getState() == Session::NOT_STARTED) {
537 // Check if at least one client has quota to start the session.
538 bool keepForClient = false;
539 for (uid_t uid : topSession->allClientUids) {
540 if (mPacer->onSessionStarted(uid, topSession->callingUid)) {
541 keepForClient = true;
542 // DO NOT break here, because book-keeping still needs to happen
543 // for the other uids.
544 }
545 }
546 if (!keepForClient) {
547 // Unfortunately all uids requesting this session are out of quota.
548 // Drop this session and try the next one.
549 {
550 auto clientCallback = mSessionMap[topSession->key].callback.lock();
551 if (clientCallback != nullptr) {
552 clientCallback->onTranscodingFailed(
553 topSession->key.second, TranscodingErrorCode::kDroppedByService);
554 }
555 }
556 removeSession_l(topSession->key, Session::DROPPED_BY_PACER);
557 continue;
558 }
559 mTranscoder->start(topSession->key.first, topSession->key.second, topSession->request,
560 topSession->callingUid, topSession->callback.lock());
561 setSessionState_l(topSession, Session::RUNNING);
562 } else if (topSession->getState() == Session::PAUSED) {
563 mTranscoder->resume(topSession->key.first, topSession->key.second, topSession->request,
564 topSession->callingUid, topSession->callback.lock());
565 setSessionState_l(topSession, Session::RUNNING);
566 }
567 break;
568 }
569 mCurrentSession = topSession;
570 }
571
addUidToSession_l(uid_t clientUid,const SessionKeyType & sessionKey)572 void TranscodingSessionController::addUidToSession_l(uid_t clientUid,
573 const SessionKeyType& sessionKey) {
574 // If it's an offline session, the queue was already added in constructor.
575 // If it's a real-time sessions, check if a queue is already present for the uid,
576 // and add a new queue if needed.
577 if (clientUid != OFFLINE_UID) {
578 if (mSessionQueues.count(clientUid) == 0) {
579 mUidPolicy->registerMonitorUid(clientUid);
580 if (mUidPolicy->isUidOnTop(clientUid)) {
581 mUidSortedList.push_front(clientUid);
582 } else {
583 // Shouldn't be submitting real-time requests from non-top app,
584 // put it in front of the offline queue.
585 mUidSortedList.insert(mOfflineUidIterator, clientUid);
586 }
587 } else if (clientUid != *mUidSortedList.begin()) {
588 if (mUidPolicy->isUidOnTop(clientUid)) {
589 mUidSortedList.remove(clientUid);
590 mUidSortedList.push_front(clientUid);
591 }
592 }
593 }
594 // Append this session to the uid's queue.
595 mSessionQueues[clientUid].push_back(sessionKey);
596 }
597
removeSession_l(const SessionKeyType & sessionKey,Session::State finalState,const std::shared_ptr<std::function<bool (uid_t uid)>> & keepUid)598 void TranscodingSessionController::removeSession_l(
599 const SessionKeyType& sessionKey, Session::State finalState,
600 const std::shared_ptr<std::function<bool(uid_t uid)>>& keepUid) {
601 ALOGV("%s: session %s", __FUNCTION__, sessionToString(sessionKey).c_str());
602
603 if (mSessionMap.count(sessionKey) == 0) {
604 ALOGE("session %s doesn't exist", sessionToString(sessionKey).c_str());
605 return;
606 }
607
608 // Remove session from uid's queue.
609 bool uidQueueRemoved = false;
610 std::unordered_set<uid_t> remainingUids;
611 for (uid_t uid : mSessionMap[sessionKey].allClientUids) {
612 if (keepUid != nullptr) {
613 if ((*keepUid)(uid)) {
614 remainingUids.insert(uid);
615 continue;
616 }
617 // If we have uids to keep, the session is not going to any final
618 // state we can't use onSessionCompleted as the running time will
619 // not be valid. Only notify pacer to stop tracking this session.
620 mPacer->onSessionCancelled(uid);
621 }
622 SessionQueueType& sessionQueue = mSessionQueues[uid];
623 auto it = std::find(sessionQueue.begin(), sessionQueue.end(), sessionKey);
624 if (it == sessionQueue.end()) {
625 ALOGW("couldn't find session %s in queue for uid %d",
626 sessionToString(sessionKey).c_str(), uid);
627 continue;
628 }
629 sessionQueue.erase(it);
630
631 // If this is the last session in a real-time queue, remove this uid's queue.
632 if (uid != OFFLINE_UID && sessionQueue.empty()) {
633 mUidSortedList.remove(uid);
634 mSessionQueues.erase(uid);
635 mUidPolicy->unregisterMonitorUid(uid);
636
637 uidQueueRemoved = true;
638 }
639 }
640
641 if (uidQueueRemoved) {
642 std::unordered_set<uid_t> topUids = mUidPolicy->getTopUids();
643 moveUidsToTop_l(topUids, false /*preserveTopUid*/);
644 }
645
646 if (keepUid != nullptr) {
647 mSessionMap[sessionKey].allClientUids = remainingUids;
648 return;
649 }
650
651 // Clear current session.
652 if (mCurrentSession == &mSessionMap[sessionKey]) {
653 mCurrentSession = nullptr;
654 }
655
656 setSessionState_l(&mSessionMap[sessionKey], finalState);
657
658 // We can use onSessionCompleted() even for CANCELLED, because runningTime is
659 // now updated by setSessionState_l().
660 for (uid_t uid : mSessionMap[sessionKey].allClientUids) {
661 mPacer->onSessionCompleted(uid, mSessionMap[sessionKey].runningTime);
662 }
663
664 mSessionHistory.push_back(mSessionMap[sessionKey]);
665 if (mSessionHistory.size() > kSessionHistoryMax) {
666 mSessionHistory.erase(mSessionHistory.begin());
667 }
668
669 // Remove session from session map.
670 mSessionMap.erase(sessionKey);
671 }
672
673 /**
674 * Moves the set of uids to the front of mUidSortedList (which is used to pick
675 * the next session to run).
676 *
677 * This is called when 1) we received a onTopUidsChanged() callback from UidPolicy,
678 * or 2) we removed the session queue for a uid because it becomes empty.
679 *
680 * In case of 1), if there are multiple uids in the set, and the current front
681 * uid in mUidSortedList is still in the set, we try to keep that uid at front
682 * so that current session run is not interrupted. (This is not a concern for case 2)
683 * because the queue for a uid was just removed entirely.)
684 */
moveUidsToTop_l(const std::unordered_set<uid_t> & uids,bool preserveTopUid)685 void TranscodingSessionController::moveUidsToTop_l(const std::unordered_set<uid_t>& uids,
686 bool preserveTopUid) {
687 // If uid set is empty, nothing to do. Do not change the queue status.
688 if (uids.empty()) {
689 return;
690 }
691
692 // Save the current top uid.
693 uid_t curTopUid = *mUidSortedList.begin();
694 bool pushCurTopToFront = false;
695 int32_t numUidsMoved = 0;
696
697 // Go through the sorted uid list once, and move the ones in top set to front.
698 for (auto it = mUidSortedList.begin(); it != mUidSortedList.end();) {
699 uid_t uid = *it;
700
701 if (uid != OFFLINE_UID && uids.count(uid) > 0) {
702 it = mUidSortedList.erase(it);
703
704 // If this is the top we're preserving, don't push it here, push
705 // it after the for-loop.
706 if (uid == curTopUid && preserveTopUid) {
707 pushCurTopToFront = true;
708 } else {
709 mUidSortedList.push_front(uid);
710 }
711
712 // If we found all uids in the set, break out.
713 if (++numUidsMoved == uids.size()) {
714 break;
715 }
716 } else {
717 ++it;
718 }
719 }
720
721 if (pushCurTopToFront) {
722 mUidSortedList.push_front(curTopUid);
723 }
724 }
725
submit(ClientIdType clientId,SessionIdType sessionId,uid_t callingUid,uid_t clientUid,const TranscodingRequestParcel & request,const std::weak_ptr<ITranscodingClientCallback> & callback)726 bool TranscodingSessionController::submit(
727 ClientIdType clientId, SessionIdType sessionId, uid_t callingUid, uid_t clientUid,
728 const TranscodingRequestParcel& request,
729 const std::weak_ptr<ITranscodingClientCallback>& callback) {
730 SessionKeyType sessionKey = std::make_pair(clientId, sessionId);
731
732 ALOGV("%s: session %s, uid %d, prioirty %d", __FUNCTION__, sessionToString(sessionKey).c_str(),
733 clientUid, (int32_t)request.priority);
734
735 std::scoped_lock lock{mLock};
736
737 if (mSessionMap.count(sessionKey) > 0) {
738 ALOGE("session %s already exists", sessionToString(sessionKey).c_str());
739 return false;
740 }
741
742 // Add the uid package name to the store of package names we already know.
743 if (mUidPackageNames.count(clientUid) == 0) {
744 mUidPackageNames.emplace(clientUid, request.clientPackageName);
745 }
746
747 // TODO(chz): only support offline vs real-time for now. All kUnspecified sessions
748 // go to offline queue.
749 if (request.priority == TranscodingSessionPriority::kUnspecified) {
750 clientUid = OFFLINE_UID;
751 }
752
753 // Add session to session map.
754 mSessionMap[sessionKey].key = sessionKey;
755 mSessionMap[sessionKey].callingUid = callingUid;
756 mSessionMap[sessionKey].allClientUids.insert(clientUid);
757 mSessionMap[sessionKey].request = request;
758 mSessionMap[sessionKey].callback = callback;
759 setSessionState_l(&mSessionMap[sessionKey], Session::NOT_STARTED);
760
761 addUidToSession_l(clientUid, sessionKey);
762
763 updateCurrentSession_l();
764
765 validateState_l();
766 return true;
767 }
768
cancel(ClientIdType clientId,SessionIdType sessionId)769 bool TranscodingSessionController::cancel(ClientIdType clientId, SessionIdType sessionId) {
770 SessionKeyType sessionKey = std::make_pair(clientId, sessionId);
771
772 ALOGV("%s: session %s", __FUNCTION__, sessionToString(sessionKey).c_str());
773
774 std::list<SessionKeyType> sessionsToRemove, sessionsForOffline;
775
776 std::scoped_lock lock{mLock};
777
778 if (sessionId < 0) {
779 for (auto it = mSessionMap.begin(); it != mSessionMap.end(); ++it) {
780 if (it->first.first == clientId) {
781 // If there is offline request, only keep the offline client;
782 // otherwise remove the session.
783 if (it->second.allClientUids.count(OFFLINE_UID) > 0) {
784 sessionsForOffline.push_back(it->first);
785 } else {
786 sessionsToRemove.push_back(it->first);
787 }
788 }
789 }
790 } else {
791 if (mSessionMap.count(sessionKey) == 0) {
792 ALOGE("session %s doesn't exist", sessionToString(sessionKey).c_str());
793 return false;
794 }
795 sessionsToRemove.push_back(sessionKey);
796 }
797
798 for (auto it = sessionsToRemove.begin(); it != sessionsToRemove.end(); ++it) {
799 // If the session has ever been started, stop it now.
800 // Note that stop() is needed even if the session is currently paused. This instructs
801 // the transcoder to discard any states for the session, otherwise the states may
802 // never be discarded.
803 if (mSessionMap[*it].getState() != Session::NOT_STARTED) {
804 mTranscoder->stop(it->first, it->second);
805 }
806
807 // Remove the session.
808 removeSession_l(*it, Session::CANCELED);
809 }
810
811 auto keepUid = std::make_shared<std::function<bool(uid_t)>>(
812 [](uid_t uid) { return uid == OFFLINE_UID; });
813 for (auto it = sessionsForOffline.begin(); it != sessionsForOffline.end(); ++it) {
814 removeSession_l(*it, Session::CANCELED, keepUid);
815 }
816
817 // Start next session.
818 updateCurrentSession_l();
819
820 validateState_l();
821 return true;
822 }
823
addClientUid(ClientIdType clientId,SessionIdType sessionId,uid_t clientUid)824 bool TranscodingSessionController::addClientUid(ClientIdType clientId, SessionIdType sessionId,
825 uid_t clientUid) {
826 SessionKeyType sessionKey = std::make_pair(clientId, sessionId);
827
828 std::scoped_lock lock{mLock};
829
830 if (mSessionMap.count(sessionKey) == 0) {
831 ALOGE("session %s doesn't exist", sessionToString(sessionKey).c_str());
832 return false;
833 }
834
835 if (mSessionMap[sessionKey].allClientUids.count(clientUid) > 0) {
836 ALOGE("session %s already has uid %d", sessionToString(sessionKey).c_str(), clientUid);
837 return false;
838 }
839
840 mSessionMap[sessionKey].allClientUids.insert(clientUid);
841 addUidToSession_l(clientUid, sessionKey);
842
843 updateCurrentSession_l();
844
845 validateState_l();
846 return true;
847 }
848
getClientUids(ClientIdType clientId,SessionIdType sessionId,std::vector<int32_t> * out_clientUids)849 bool TranscodingSessionController::getClientUids(ClientIdType clientId, SessionIdType sessionId,
850 std::vector<int32_t>* out_clientUids) {
851 SessionKeyType sessionKey = std::make_pair(clientId, sessionId);
852
853 std::scoped_lock lock{mLock};
854
855 if (mSessionMap.count(sessionKey) == 0) {
856 ALOGE("session %s doesn't exist", sessionToString(sessionKey).c_str());
857 return false;
858 }
859
860 out_clientUids->clear();
861 for (uid_t uid : mSessionMap[sessionKey].allClientUids) {
862 if (uid != OFFLINE_UID) {
863 out_clientUids->push_back(uid);
864 }
865 }
866 return true;
867 }
868
getSession(ClientIdType clientId,SessionIdType sessionId,TranscodingRequestParcel * request)869 bool TranscodingSessionController::getSession(ClientIdType clientId, SessionIdType sessionId,
870 TranscodingRequestParcel* request) {
871 SessionKeyType sessionKey = std::make_pair(clientId, sessionId);
872
873 std::scoped_lock lock{mLock};
874
875 if (mSessionMap.count(sessionKey) == 0) {
876 ALOGE("session %s doesn't exist", sessionToString(sessionKey).c_str());
877 return false;
878 }
879
880 *(TranscodingRequest*)request = mSessionMap[sessionKey].request;
881 return true;
882 }
883
notifyClient(ClientIdType clientId,SessionIdType sessionId,const char * reason,std::function<void (const SessionKeyType &)> func)884 void TranscodingSessionController::notifyClient(ClientIdType clientId, SessionIdType sessionId,
885 const char* reason,
886 std::function<void(const SessionKeyType&)> func) {
887 SessionKeyType sessionKey = std::make_pair(clientId, sessionId);
888
889 std::scoped_lock lock{mLock};
890
891 if (mSessionMap.count(sessionKey) == 0) {
892 ALOGW("%s: ignoring %s for session %s that doesn't exist", __FUNCTION__, reason,
893 sessionToString(sessionKey).c_str());
894 return;
895 }
896
897 // Only ignore if session was never started. In particular, propagate the status
898 // to client if the session is paused. Transcoder could have posted finish when
899 // we're pausing it, and the finish arrived after we changed current session.
900 if (mSessionMap[sessionKey].getState() == Session::NOT_STARTED) {
901 ALOGW("%s: ignoring %s for session %s that was never started", __FUNCTION__, reason,
902 sessionToString(sessionKey).c_str());
903 return;
904 }
905
906 ALOGV("%s: session %s %s", __FUNCTION__, sessionToString(sessionKey).c_str(), reason);
907 func(sessionKey);
908 }
909
onStarted(ClientIdType clientId,SessionIdType sessionId)910 void TranscodingSessionController::onStarted(ClientIdType clientId, SessionIdType sessionId) {
911 notifyClient(clientId, sessionId, "started", [=](const SessionKeyType& sessionKey) {
912 auto callback = mSessionMap[sessionKey].callback.lock();
913 if (callback != nullptr) {
914 callback->onTranscodingStarted(sessionId);
915 }
916 });
917 }
918
onPaused(ClientIdType clientId,SessionIdType sessionId)919 void TranscodingSessionController::onPaused(ClientIdType clientId, SessionIdType sessionId) {
920 notifyClient(clientId, sessionId, "paused", [=](const SessionKeyType& sessionKey) {
921 auto callback = mSessionMap[sessionKey].callback.lock();
922 if (callback != nullptr) {
923 callback->onTranscodingPaused(sessionId);
924 }
925 });
926 }
927
onResumed(ClientIdType clientId,SessionIdType sessionId)928 void TranscodingSessionController::onResumed(ClientIdType clientId, SessionIdType sessionId) {
929 notifyClient(clientId, sessionId, "resumed", [=](const SessionKeyType& sessionKey) {
930 auto callback = mSessionMap[sessionKey].callback.lock();
931 if (callback != nullptr) {
932 callback->onTranscodingResumed(sessionId);
933 }
934 });
935 }
936
onFinish(ClientIdType clientId,SessionIdType sessionId)937 void TranscodingSessionController::onFinish(ClientIdType clientId, SessionIdType sessionId) {
938 notifyClient(clientId, sessionId, "finish", [=](const SessionKeyType& sessionKey) {
939 {
940 auto clientCallback = mSessionMap[sessionKey].callback.lock();
941 if (clientCallback != nullptr) {
942 clientCallback->onTranscodingFinished(
943 sessionId, TranscodingResultParcel({sessionId, -1 /*actualBitrateBps*/,
944 std::nullopt /*sessionStats*/}));
945 }
946 }
947
948 // Remove the session.
949 removeSession_l(sessionKey, Session::FINISHED);
950
951 // Start next session.
952 updateCurrentSession_l();
953
954 validateState_l();
955 });
956 }
957
onError(ClientIdType clientId,SessionIdType sessionId,TranscodingErrorCode err)958 void TranscodingSessionController::onError(ClientIdType clientId, SessionIdType sessionId,
959 TranscodingErrorCode err) {
960 notifyClient(clientId, sessionId, "error", [=](const SessionKeyType& sessionKey) {
961 if (err == TranscodingErrorCode::kWatchdogTimeout) {
962 // Abandon the transcoder, as its handler thread might be stuck in some call to
963 // MediaTranscoder altogether, and may not be able to handle any new tasks.
964 mTranscoder->stop(clientId, sessionId, true /*abandon*/);
965 // Clear the last ref count before we create new transcoder.
966 mTranscoder = nullptr;
967 mTranscoder = mTranscoderFactory(shared_from_this());
968 }
969
970 {
971 auto clientCallback = mSessionMap[sessionKey].callback.lock();
972 if (clientCallback != nullptr) {
973 clientCallback->onTranscodingFailed(sessionId, err);
974 }
975 }
976
977 // Remove the session.
978 removeSession_l(sessionKey, Session::ERROR);
979
980 // Start next session.
981 updateCurrentSession_l();
982
983 validateState_l();
984 });
985 }
986
onProgressUpdate(ClientIdType clientId,SessionIdType sessionId,int32_t progress)987 void TranscodingSessionController::onProgressUpdate(ClientIdType clientId, SessionIdType sessionId,
988 int32_t progress) {
989 notifyClient(clientId, sessionId, "progress", [=](const SessionKeyType& sessionKey) {
990 auto callback = mSessionMap[sessionKey].callback.lock();
991 if (callback != nullptr) {
992 callback->onProgressUpdate(sessionId, progress);
993 }
994 mSessionMap[sessionKey].lastProgress = progress;
995 });
996 }
997
onHeartBeat(ClientIdType clientId,SessionIdType sessionId)998 void TranscodingSessionController::onHeartBeat(ClientIdType clientId, SessionIdType sessionId) {
999 notifyClient(clientId, sessionId, "heart-beat",
1000 [=](const SessionKeyType& /*sessionKey*/) { mWatchdog->keepAlive(); });
1001 }
1002
onResourceLost(ClientIdType clientId,SessionIdType sessionId)1003 void TranscodingSessionController::onResourceLost(ClientIdType clientId, SessionIdType sessionId) {
1004 ALOGI("%s", __FUNCTION__);
1005
1006 notifyClient(clientId, sessionId, "resource_lost", [=](const SessionKeyType& sessionKey) {
1007 if (mResourceLost) {
1008 return;
1009 }
1010
1011 Session* resourceLostSession = &mSessionMap[sessionKey];
1012 if (resourceLostSession->getState() != Session::RUNNING) {
1013 ALOGW("session %s lost resource but is no longer running",
1014 sessionToString(sessionKey).c_str());
1015 return;
1016 }
1017 // If we receive a resource loss event, the transcoder already paused the transcoding,
1018 // so we don't need to call onPaused() to pause it. However, we still need to notify
1019 // the client and update the session state here.
1020 setSessionState_l(resourceLostSession, Session::PAUSED);
1021 // Notify the client as a paused event.
1022 auto clientCallback = resourceLostSession->callback.lock();
1023 if (clientCallback != nullptr) {
1024 clientCallback->onTranscodingPaused(sessionKey.second);
1025 }
1026 if (mResourcePolicy != nullptr) {
1027 mResourcePolicy->setPidResourceLost(resourceLostSession->request.clientPid);
1028 }
1029 mResourceLost = true;
1030
1031 validateState_l();
1032 });
1033 }
1034
onTopUidsChanged(const std::unordered_set<uid_t> & uids)1035 void TranscodingSessionController::onTopUidsChanged(const std::unordered_set<uid_t>& uids) {
1036 if (uids.empty()) {
1037 ALOGW("%s: ignoring empty uids", __FUNCTION__);
1038 return;
1039 }
1040
1041 std::string uidStr;
1042 for (auto it = uids.begin(); it != uids.end(); it++) {
1043 if (!uidStr.empty()) {
1044 uidStr += ", ";
1045 }
1046 uidStr += std::to_string(*it);
1047 }
1048
1049 ALOGD("%s: topUids: size %zu, uids: %s", __FUNCTION__, uids.size(), uidStr.c_str());
1050
1051 std::scoped_lock lock{mLock};
1052
1053 moveUidsToTop_l(uids, true /*preserveTopUid*/);
1054
1055 updateCurrentSession_l();
1056
1057 validateState_l();
1058 }
1059
onUidGone(uid_t goneUid)1060 void TranscodingSessionController::onUidGone(uid_t goneUid) {
1061 ALOGD("%s: gone uid %u", __FUNCTION__, goneUid);
1062
1063 std::list<SessionKeyType> sessionsToRemove, sessionsForOtherUids;
1064
1065 std::scoped_lock lock{mLock};
1066
1067 for (auto it = mSessionMap.begin(); it != mSessionMap.end(); ++it) {
1068 if (it->second.allClientUids.count(goneUid) > 0) {
1069 // If goneUid is the only uid, remove the session; otherwise, only
1070 // remove the uid from the session.
1071 if (it->second.allClientUids.size() > 1) {
1072 sessionsForOtherUids.push_back(it->first);
1073 } else {
1074 sessionsToRemove.push_back(it->first);
1075 }
1076 }
1077 }
1078
1079 for (auto it = sessionsToRemove.begin(); it != sessionsToRemove.end(); ++it) {
1080 // If the session has ever been started, stop it now.
1081 // Note that stop() is needed even if the session is currently paused. This instructs
1082 // the transcoder to discard any states for the session, otherwise the states may
1083 // never be discarded.
1084 if (mSessionMap[*it].getState() != Session::NOT_STARTED) {
1085 mTranscoder->stop(it->first, it->second);
1086 }
1087
1088 {
1089 auto clientCallback = mSessionMap[*it].callback.lock();
1090 if (clientCallback != nullptr) {
1091 clientCallback->onTranscodingFailed(it->second,
1092 TranscodingErrorCode::kUidGoneCancelled);
1093 }
1094 }
1095
1096 // Remove the session.
1097 removeSession_l(*it, Session::CANCELED);
1098 }
1099
1100 auto keepUid = std::make_shared<std::function<bool(uid_t)>>(
1101 [goneUid](uid_t uid) { return uid != goneUid; });
1102 for (auto it = sessionsForOtherUids.begin(); it != sessionsForOtherUids.end(); ++it) {
1103 removeSession_l(*it, Session::CANCELED, keepUid);
1104 }
1105
1106 // Start next session.
1107 updateCurrentSession_l();
1108
1109 validateState_l();
1110 }
1111
onResourceAvailable()1112 void TranscodingSessionController::onResourceAvailable() {
1113 std::scoped_lock lock{mLock};
1114
1115 if (!mResourceLost) {
1116 return;
1117 }
1118
1119 ALOGI("%s", __FUNCTION__);
1120
1121 mResourceLost = false;
1122 updateCurrentSession_l();
1123
1124 validateState_l();
1125 }
1126
onThrottlingStarted()1127 void TranscodingSessionController::onThrottlingStarted() {
1128 std::scoped_lock lock{mLock};
1129
1130 if (mThermalThrottling) {
1131 return;
1132 }
1133
1134 ALOGI("%s", __FUNCTION__);
1135
1136 mThermalThrottling = true;
1137 updateCurrentSession_l();
1138
1139 validateState_l();
1140 }
1141
onThrottlingStopped()1142 void TranscodingSessionController::onThrottlingStopped() {
1143 std::scoped_lock lock{mLock};
1144
1145 if (!mThermalThrottling) {
1146 return;
1147 }
1148
1149 ALOGI("%s", __FUNCTION__);
1150
1151 mThermalThrottling = false;
1152 updateCurrentSession_l();
1153
1154 validateState_l();
1155 }
1156
validateState_l()1157 void TranscodingSessionController::validateState_l() {
1158 #ifdef VALIDATE_STATE
1159 LOG_ALWAYS_FATAL_IF(mSessionQueues.count(OFFLINE_UID) != 1,
1160 "mSessionQueues offline queue number is not 1");
1161 LOG_ALWAYS_FATAL_IF(*mOfflineUidIterator != OFFLINE_UID,
1162 "mOfflineUidIterator not pointing to offline uid");
1163 LOG_ALWAYS_FATAL_IF(mUidSortedList.size() != mSessionQueues.size(),
1164 "mUidSortedList and mSessionQueues size mismatch, %zu vs %zu",
1165 mUidSortedList.size(), mSessionQueues.size());
1166
1167 int32_t totalSessions = 0;
1168 for (auto uid : mUidSortedList) {
1169 LOG_ALWAYS_FATAL_IF(mSessionQueues.count(uid) != 1,
1170 "mSessionQueues count for uid %d is not 1", uid);
1171 for (auto& sessionKey : mSessionQueues[uid]) {
1172 LOG_ALWAYS_FATAL_IF(mSessionMap.count(sessionKey) != 1,
1173 "mSessions count for session %s is not 1",
1174 sessionToString(sessionKey).c_str());
1175 }
1176
1177 totalSessions += mSessionQueues[uid].size();
1178 }
1179 int32_t totalSessionsAlternative = 0;
1180 for (auto const& s : mSessionMap) {
1181 totalSessionsAlternative += s.second.allClientUids.size();
1182 }
1183 LOG_ALWAYS_FATAL_IF(totalSessions != totalSessionsAlternative,
1184 "session count (including dup) from mSessionQueues doesn't match that from "
1185 "mSessionMap, %d vs %d",
1186 totalSessions, totalSessionsAlternative);
1187 #endif // VALIDATE_STATE
1188 }
1189
1190 } // namespace android
1191