xref: /aosp_15_r20/frameworks/native/libs/binder/PermissionCache.cpp (revision 38e8c45f13ce32b0dcecb25141ffecaf386fa17f)
1*38e8c45fSAndroid Build Coastguard Worker /*
2*38e8c45fSAndroid Build Coastguard Worker  * Copyright (C) 2009 The Android Open Source Project
3*38e8c45fSAndroid Build Coastguard Worker  *
4*38e8c45fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*38e8c45fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*38e8c45fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*38e8c45fSAndroid Build Coastguard Worker  *
8*38e8c45fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*38e8c45fSAndroid Build Coastguard Worker  *
10*38e8c45fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*38e8c45fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*38e8c45fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*38e8c45fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*38e8c45fSAndroid Build Coastguard Worker  * limitations under the License.
15*38e8c45fSAndroid Build Coastguard Worker  */
16*38e8c45fSAndroid Build Coastguard Worker 
17*38e8c45fSAndroid Build Coastguard Worker #define LOG_TAG "PermissionCache"
18*38e8c45fSAndroid Build Coastguard Worker 
19*38e8c45fSAndroid Build Coastguard Worker #include <stdint.h>
20*38e8c45fSAndroid Build Coastguard Worker #include <utils/Log.h>
21*38e8c45fSAndroid Build Coastguard Worker #include <binder/IPCThreadState.h>
22*38e8c45fSAndroid Build Coastguard Worker #include <binder/IServiceManager.h>
23*38e8c45fSAndroid Build Coastguard Worker #include <binder/PermissionCache.h>
24*38e8c45fSAndroid Build Coastguard Worker #include <utils/String8.h>
25*38e8c45fSAndroid Build Coastguard Worker 
26*38e8c45fSAndroid Build Coastguard Worker namespace android {
27*38e8c45fSAndroid Build Coastguard Worker 
28*38e8c45fSAndroid Build Coastguard Worker // ----------------------------------------------------------------------------
29*38e8c45fSAndroid Build Coastguard Worker 
ANDROID_SINGLETON_STATIC_INSTANCE(PermissionCache)30*38e8c45fSAndroid Build Coastguard Worker ANDROID_SINGLETON_STATIC_INSTANCE(PermissionCache)
31*38e8c45fSAndroid Build Coastguard Worker 
32*38e8c45fSAndroid Build Coastguard Worker // ----------------------------------------------------------------------------
33*38e8c45fSAndroid Build Coastguard Worker 
34*38e8c45fSAndroid Build Coastguard Worker PermissionCache::PermissionCache() {
35*38e8c45fSAndroid Build Coastguard Worker }
36*38e8c45fSAndroid Build Coastguard Worker 
check(bool * granted,const String16 & permission,uid_t uid) const37*38e8c45fSAndroid Build Coastguard Worker status_t PermissionCache::check(bool* granted,
38*38e8c45fSAndroid Build Coastguard Worker         const String16& permission, uid_t uid) const {
39*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock _l(mLock);
40*38e8c45fSAndroid Build Coastguard Worker     Entry e;
41*38e8c45fSAndroid Build Coastguard Worker     e.name = permission;
42*38e8c45fSAndroid Build Coastguard Worker     e.uid  = uid;
43*38e8c45fSAndroid Build Coastguard Worker     ssize_t index = mCache.indexOf(e);
44*38e8c45fSAndroid Build Coastguard Worker     if (index >= 0) {
45*38e8c45fSAndroid Build Coastguard Worker         *granted = mCache.itemAt(index).granted;
46*38e8c45fSAndroid Build Coastguard Worker         return NO_ERROR;
47*38e8c45fSAndroid Build Coastguard Worker     }
48*38e8c45fSAndroid Build Coastguard Worker     return NAME_NOT_FOUND;
49*38e8c45fSAndroid Build Coastguard Worker }
50*38e8c45fSAndroid Build Coastguard Worker 
cache(const String16 & permission,uid_t uid,bool granted)51*38e8c45fSAndroid Build Coastguard Worker void PermissionCache::cache(const String16& permission,
52*38e8c45fSAndroid Build Coastguard Worker         uid_t uid, bool granted) {
53*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock _l(mLock);
54*38e8c45fSAndroid Build Coastguard Worker     Entry e;
55*38e8c45fSAndroid Build Coastguard Worker     ssize_t index = mPermissionNamesPool.indexOf(permission);
56*38e8c45fSAndroid Build Coastguard Worker     if (index > 0) {
57*38e8c45fSAndroid Build Coastguard Worker         e.name = mPermissionNamesPool.itemAt(index);
58*38e8c45fSAndroid Build Coastguard Worker     } else {
59*38e8c45fSAndroid Build Coastguard Worker         mPermissionNamesPool.add(permission);
60*38e8c45fSAndroid Build Coastguard Worker         e.name = permission;
61*38e8c45fSAndroid Build Coastguard Worker     }
62*38e8c45fSAndroid Build Coastguard Worker     // note, we don't need to store the pid, which is not actually used in
63*38e8c45fSAndroid Build Coastguard Worker     // permission checks
64*38e8c45fSAndroid Build Coastguard Worker     e.uid  = uid;
65*38e8c45fSAndroid Build Coastguard Worker     e.granted = granted;
66*38e8c45fSAndroid Build Coastguard Worker     index = mCache.indexOf(e);
67*38e8c45fSAndroid Build Coastguard Worker     if (index < 0) {
68*38e8c45fSAndroid Build Coastguard Worker         mCache.add(e);
69*38e8c45fSAndroid Build Coastguard Worker     }
70*38e8c45fSAndroid Build Coastguard Worker }
71*38e8c45fSAndroid Build Coastguard Worker 
purge()72*38e8c45fSAndroid Build Coastguard Worker void PermissionCache::purge() {
73*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock _l(mLock);
74*38e8c45fSAndroid Build Coastguard Worker     mCache.clear();
75*38e8c45fSAndroid Build Coastguard Worker }
76*38e8c45fSAndroid Build Coastguard Worker 
checkCallingPermission(const String16 & permission)77*38e8c45fSAndroid Build Coastguard Worker bool PermissionCache::checkCallingPermission(const String16& permission) {
78*38e8c45fSAndroid Build Coastguard Worker     return PermissionCache::checkCallingPermission(permission, nullptr, nullptr);
79*38e8c45fSAndroid Build Coastguard Worker }
80*38e8c45fSAndroid Build Coastguard Worker 
checkCallingPermission(const String16 & permission,int32_t * outPid,int32_t * outUid)81*38e8c45fSAndroid Build Coastguard Worker bool PermissionCache::checkCallingPermission(
82*38e8c45fSAndroid Build Coastguard Worker         const String16& permission, int32_t* outPid, int32_t* outUid) {
83*38e8c45fSAndroid Build Coastguard Worker     IPCThreadState* ipcState = IPCThreadState::self();
84*38e8c45fSAndroid Build Coastguard Worker     pid_t pid = ipcState->getCallingPid();
85*38e8c45fSAndroid Build Coastguard Worker     uid_t uid = ipcState->getCallingUid();
86*38e8c45fSAndroid Build Coastguard Worker     if (outPid) *outPid = pid;
87*38e8c45fSAndroid Build Coastguard Worker     if (outUid) *outUid = uid;
88*38e8c45fSAndroid Build Coastguard Worker     return PermissionCache::checkPermission(permission, pid, uid);
89*38e8c45fSAndroid Build Coastguard Worker }
90*38e8c45fSAndroid Build Coastguard Worker 
checkPermission(const String16 & permission,pid_t pid,uid_t uid)91*38e8c45fSAndroid Build Coastguard Worker bool PermissionCache::checkPermission(
92*38e8c45fSAndroid Build Coastguard Worker         const String16& permission, pid_t pid, uid_t uid) {
93*38e8c45fSAndroid Build Coastguard Worker     if ((uid == 0) || (pid == getpid())) {
94*38e8c45fSAndroid Build Coastguard Worker         // root and ourselves is always okay
95*38e8c45fSAndroid Build Coastguard Worker         return true;
96*38e8c45fSAndroid Build Coastguard Worker     }
97*38e8c45fSAndroid Build Coastguard Worker 
98*38e8c45fSAndroid Build Coastguard Worker     PermissionCache& pc(PermissionCache::getInstance());
99*38e8c45fSAndroid Build Coastguard Worker     bool granted = false;
100*38e8c45fSAndroid Build Coastguard Worker     if (pc.check(&granted, permission, uid) != NO_ERROR) {
101*38e8c45fSAndroid Build Coastguard Worker         nsecs_t t = -systemTime();
102*38e8c45fSAndroid Build Coastguard Worker         granted = android::checkPermission(permission, pid, uid);
103*38e8c45fSAndroid Build Coastguard Worker         t += systemTime();
104*38e8c45fSAndroid Build Coastguard Worker         ALOGD("checking %s for uid=%d => %s (%d us)", String8(permission).c_str(), uid,
105*38e8c45fSAndroid Build Coastguard Worker               granted ? "granted" : "denied", (int)ns2us(t));
106*38e8c45fSAndroid Build Coastguard Worker         pc.cache(permission, uid, granted);
107*38e8c45fSAndroid Build Coastguard Worker     }
108*38e8c45fSAndroid Build Coastguard Worker     return granted;
109*38e8c45fSAndroid Build Coastguard Worker }
110*38e8c45fSAndroid Build Coastguard Worker 
purgeCache()111*38e8c45fSAndroid Build Coastguard Worker void PermissionCache::purgeCache() {
112*38e8c45fSAndroid Build Coastguard Worker     PermissionCache& pc(PermissionCache::getInstance());
113*38e8c45fSAndroid Build Coastguard Worker     pc.purge();
114*38e8c45fSAndroid Build Coastguard Worker }
115*38e8c45fSAndroid Build Coastguard Worker 
116*38e8c45fSAndroid Build Coastguard Worker // ---------------------------------------------------------------------------
117*38e8c45fSAndroid Build Coastguard Worker } // namespace android
118