1*38e8c45fSAndroid Build Coastguard Worker /*
2*38e8c45fSAndroid Build Coastguard Worker * Copyright (C) 2005 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 "PermissionController"
18*38e8c45fSAndroid Build Coastguard Worker
19*38e8c45fSAndroid Build Coastguard Worker #include <binder/IPermissionController.h>
20*38e8c45fSAndroid Build Coastguard Worker
21*38e8c45fSAndroid Build Coastguard Worker #include <utils/Log.h>
22*38e8c45fSAndroid Build Coastguard Worker #include <binder/Parcel.h>
23*38e8c45fSAndroid Build Coastguard Worker #include <utils/String8.h>
24*38e8c45fSAndroid Build Coastguard Worker
25*38e8c45fSAndroid Build Coastguard Worker namespace android {
26*38e8c45fSAndroid Build Coastguard Worker
27*38e8c45fSAndroid Build Coastguard Worker // ----------------------------------------------------------------------
28*38e8c45fSAndroid Build Coastguard Worker
29*38e8c45fSAndroid Build Coastguard Worker class BpPermissionController : public BpInterface<IPermissionController>
30*38e8c45fSAndroid Build Coastguard Worker {
31*38e8c45fSAndroid Build Coastguard Worker public:
BpPermissionController(const sp<IBinder> & impl)32*38e8c45fSAndroid Build Coastguard Worker explicit BpPermissionController(const sp<IBinder>& impl)
33*38e8c45fSAndroid Build Coastguard Worker : BpInterface<IPermissionController>(impl)
34*38e8c45fSAndroid Build Coastguard Worker {
35*38e8c45fSAndroid Build Coastguard Worker }
36*38e8c45fSAndroid Build Coastguard Worker
checkPermission(const String16 & permission,int32_t pid,int32_t uid)37*38e8c45fSAndroid Build Coastguard Worker virtual bool checkPermission(const String16& permission, int32_t pid, int32_t uid)
38*38e8c45fSAndroid Build Coastguard Worker {
39*38e8c45fSAndroid Build Coastguard Worker Parcel data, reply;
40*38e8c45fSAndroid Build Coastguard Worker data.writeInterfaceToken(IPermissionController::getInterfaceDescriptor());
41*38e8c45fSAndroid Build Coastguard Worker data.writeString16(permission);
42*38e8c45fSAndroid Build Coastguard Worker data.writeInt32(pid);
43*38e8c45fSAndroid Build Coastguard Worker data.writeInt32(uid);
44*38e8c45fSAndroid Build Coastguard Worker remote()->transact(CHECK_PERMISSION_TRANSACTION, data, &reply);
45*38e8c45fSAndroid Build Coastguard Worker // fail on exception
46*38e8c45fSAndroid Build Coastguard Worker if (reply.readExceptionCode() != 0) return 0;
47*38e8c45fSAndroid Build Coastguard Worker return reply.readInt32() != 0;
48*38e8c45fSAndroid Build Coastguard Worker }
49*38e8c45fSAndroid Build Coastguard Worker
noteOp(const String16 & op,int32_t uid,const String16 & packageName)50*38e8c45fSAndroid Build Coastguard Worker virtual int32_t noteOp(const String16& op, int32_t uid, const String16& packageName)
51*38e8c45fSAndroid Build Coastguard Worker {
52*38e8c45fSAndroid Build Coastguard Worker Parcel data, reply;
53*38e8c45fSAndroid Build Coastguard Worker data.writeInterfaceToken(IPermissionController::getInterfaceDescriptor());
54*38e8c45fSAndroid Build Coastguard Worker data.writeString16(op);
55*38e8c45fSAndroid Build Coastguard Worker data.writeInt32(uid);
56*38e8c45fSAndroid Build Coastguard Worker data.writeString16(packageName);
57*38e8c45fSAndroid Build Coastguard Worker remote()->transact(NOTE_OP_TRANSACTION, data, &reply);
58*38e8c45fSAndroid Build Coastguard Worker // fail on exception
59*38e8c45fSAndroid Build Coastguard Worker if (reply.readExceptionCode() != 0) return 2; // MODE_ERRORED
60*38e8c45fSAndroid Build Coastguard Worker return reply.readInt32();
61*38e8c45fSAndroid Build Coastguard Worker }
62*38e8c45fSAndroid Build Coastguard Worker
getPackagesForUid(const uid_t uid,Vector<String16> & packages)63*38e8c45fSAndroid Build Coastguard Worker virtual void getPackagesForUid(const uid_t uid, Vector<String16>& packages)
64*38e8c45fSAndroid Build Coastguard Worker {
65*38e8c45fSAndroid Build Coastguard Worker Parcel data, reply;
66*38e8c45fSAndroid Build Coastguard Worker data.writeInterfaceToken(IPermissionController::getInterfaceDescriptor());
67*38e8c45fSAndroid Build Coastguard Worker data.writeInt32(uid);
68*38e8c45fSAndroid Build Coastguard Worker remote()->transact(GET_PACKAGES_FOR_UID_TRANSACTION, data, &reply);
69*38e8c45fSAndroid Build Coastguard Worker // fail on exception
70*38e8c45fSAndroid Build Coastguard Worker if (reply.readExceptionCode() != 0) {
71*38e8c45fSAndroid Build Coastguard Worker return;
72*38e8c45fSAndroid Build Coastguard Worker }
73*38e8c45fSAndroid Build Coastguard Worker const int32_t size = reply.readInt32();
74*38e8c45fSAndroid Build Coastguard Worker if (size <= 0) {
75*38e8c45fSAndroid Build Coastguard Worker return;
76*38e8c45fSAndroid Build Coastguard Worker }
77*38e8c45fSAndroid Build Coastguard Worker for (int i = 0; i < size; i++) {
78*38e8c45fSAndroid Build Coastguard Worker packages.push(reply.readString16());
79*38e8c45fSAndroid Build Coastguard Worker }
80*38e8c45fSAndroid Build Coastguard Worker }
81*38e8c45fSAndroid Build Coastguard Worker
isRuntimePermission(const String16 & permission)82*38e8c45fSAndroid Build Coastguard Worker virtual bool isRuntimePermission(const String16& permission)
83*38e8c45fSAndroid Build Coastguard Worker {
84*38e8c45fSAndroid Build Coastguard Worker Parcel data, reply;
85*38e8c45fSAndroid Build Coastguard Worker data.writeInterfaceToken(IPermissionController::getInterfaceDescriptor());
86*38e8c45fSAndroid Build Coastguard Worker data.writeString16(permission);
87*38e8c45fSAndroid Build Coastguard Worker remote()->transact(IS_RUNTIME_PERMISSION_TRANSACTION, data, &reply);
88*38e8c45fSAndroid Build Coastguard Worker // fail on exception
89*38e8c45fSAndroid Build Coastguard Worker if (reply.readExceptionCode() != 0) return false;
90*38e8c45fSAndroid Build Coastguard Worker return reply.readInt32() != 0;
91*38e8c45fSAndroid Build Coastguard Worker }
92*38e8c45fSAndroid Build Coastguard Worker
getPackageUid(const String16 & package,int flags)93*38e8c45fSAndroid Build Coastguard Worker virtual int getPackageUid(const String16& package, int flags)
94*38e8c45fSAndroid Build Coastguard Worker {
95*38e8c45fSAndroid Build Coastguard Worker Parcel data, reply;
96*38e8c45fSAndroid Build Coastguard Worker data.writeInterfaceToken(IPermissionController::getInterfaceDescriptor());
97*38e8c45fSAndroid Build Coastguard Worker data.writeString16(package);
98*38e8c45fSAndroid Build Coastguard Worker data.writeInt32(flags);
99*38e8c45fSAndroid Build Coastguard Worker remote()->transact(GET_PACKAGE_UID_TRANSACTION, data, &reply);
100*38e8c45fSAndroid Build Coastguard Worker // fail on exception
101*38e8c45fSAndroid Build Coastguard Worker if (reply.readExceptionCode() != 0) return false;
102*38e8c45fSAndroid Build Coastguard Worker return reply.readInt32();
103*38e8c45fSAndroid Build Coastguard Worker }
104*38e8c45fSAndroid Build Coastguard Worker };
105*38e8c45fSAndroid Build Coastguard Worker
106*38e8c45fSAndroid Build Coastguard Worker IMPLEMENT_META_INTERFACE(PermissionController, "android.os.IPermissionController")
107*38e8c45fSAndroid Build Coastguard Worker
108*38e8c45fSAndroid Build Coastguard Worker // ----------------------------------------------------------------------
109*38e8c45fSAndroid Build Coastguard Worker
110*38e8c45fSAndroid Build Coastguard Worker // NOLINTNEXTLINE(google-default-arguments)
onTransact(uint32_t code,const Parcel & data,Parcel * reply,uint32_t flags)111*38e8c45fSAndroid Build Coastguard Worker status_t BnPermissionController::onTransact(
112*38e8c45fSAndroid Build Coastguard Worker uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
113*38e8c45fSAndroid Build Coastguard Worker {
114*38e8c45fSAndroid Build Coastguard Worker switch(code) {
115*38e8c45fSAndroid Build Coastguard Worker case CHECK_PERMISSION_TRANSACTION: {
116*38e8c45fSAndroid Build Coastguard Worker CHECK_INTERFACE(IPermissionController, data, reply);
117*38e8c45fSAndroid Build Coastguard Worker String16 permission = data.readString16();
118*38e8c45fSAndroid Build Coastguard Worker int32_t pid = data.readInt32();
119*38e8c45fSAndroid Build Coastguard Worker int32_t uid = data.readInt32();
120*38e8c45fSAndroid Build Coastguard Worker bool res = checkPermission(permission, pid, uid);
121*38e8c45fSAndroid Build Coastguard Worker reply->writeNoException();
122*38e8c45fSAndroid Build Coastguard Worker reply->writeInt32(res ? 1 : 0);
123*38e8c45fSAndroid Build Coastguard Worker return NO_ERROR;
124*38e8c45fSAndroid Build Coastguard Worker } break;
125*38e8c45fSAndroid Build Coastguard Worker
126*38e8c45fSAndroid Build Coastguard Worker case NOTE_OP_TRANSACTION: {
127*38e8c45fSAndroid Build Coastguard Worker CHECK_INTERFACE(IPermissionController, data, reply);
128*38e8c45fSAndroid Build Coastguard Worker String16 op = data.readString16();
129*38e8c45fSAndroid Build Coastguard Worker int32_t uid = data.readInt32();
130*38e8c45fSAndroid Build Coastguard Worker String16 packageName = data.readString16();
131*38e8c45fSAndroid Build Coastguard Worker int32_t res = noteOp(op, uid, packageName);
132*38e8c45fSAndroid Build Coastguard Worker reply->writeNoException();
133*38e8c45fSAndroid Build Coastguard Worker reply->writeInt32(res);
134*38e8c45fSAndroid Build Coastguard Worker return NO_ERROR;
135*38e8c45fSAndroid Build Coastguard Worker } break;
136*38e8c45fSAndroid Build Coastguard Worker
137*38e8c45fSAndroid Build Coastguard Worker case GET_PACKAGES_FOR_UID_TRANSACTION: {
138*38e8c45fSAndroid Build Coastguard Worker CHECK_INTERFACE(IPermissionController, data, reply);
139*38e8c45fSAndroid Build Coastguard Worker int32_t uid = data.readInt32();
140*38e8c45fSAndroid Build Coastguard Worker Vector<String16> packages;
141*38e8c45fSAndroid Build Coastguard Worker getPackagesForUid(uid, packages);
142*38e8c45fSAndroid Build Coastguard Worker reply->writeNoException();
143*38e8c45fSAndroid Build Coastguard Worker size_t size = packages.size();
144*38e8c45fSAndroid Build Coastguard Worker reply->writeInt32(size);
145*38e8c45fSAndroid Build Coastguard Worker for (size_t i = 0; i < size; i++) {
146*38e8c45fSAndroid Build Coastguard Worker reply->writeString16(packages[i]);
147*38e8c45fSAndroid Build Coastguard Worker }
148*38e8c45fSAndroid Build Coastguard Worker return NO_ERROR;
149*38e8c45fSAndroid Build Coastguard Worker } break;
150*38e8c45fSAndroid Build Coastguard Worker
151*38e8c45fSAndroid Build Coastguard Worker case IS_RUNTIME_PERMISSION_TRANSACTION: {
152*38e8c45fSAndroid Build Coastguard Worker CHECK_INTERFACE(IPermissionController, data, reply);
153*38e8c45fSAndroid Build Coastguard Worker String16 permission = data.readString16();
154*38e8c45fSAndroid Build Coastguard Worker const bool res = isRuntimePermission(permission);
155*38e8c45fSAndroid Build Coastguard Worker reply->writeNoException();
156*38e8c45fSAndroid Build Coastguard Worker reply->writeInt32(res ? 1 : 0);
157*38e8c45fSAndroid Build Coastguard Worker return NO_ERROR;
158*38e8c45fSAndroid Build Coastguard Worker } break;
159*38e8c45fSAndroid Build Coastguard Worker
160*38e8c45fSAndroid Build Coastguard Worker case GET_PACKAGE_UID_TRANSACTION: {
161*38e8c45fSAndroid Build Coastguard Worker CHECK_INTERFACE(IPermissionController, data, reply);
162*38e8c45fSAndroid Build Coastguard Worker String16 package = data.readString16();
163*38e8c45fSAndroid Build Coastguard Worker int flags = data.readInt32();
164*38e8c45fSAndroid Build Coastguard Worker const int uid = getPackageUid(package, flags);
165*38e8c45fSAndroid Build Coastguard Worker reply->writeNoException();
166*38e8c45fSAndroid Build Coastguard Worker reply->writeInt32(uid);
167*38e8c45fSAndroid Build Coastguard Worker return NO_ERROR;
168*38e8c45fSAndroid Build Coastguard Worker } break;
169*38e8c45fSAndroid Build Coastguard Worker
170*38e8c45fSAndroid Build Coastguard Worker default:
171*38e8c45fSAndroid Build Coastguard Worker return BBinder::onTransact(code, data, reply, flags);
172*38e8c45fSAndroid Build Coastguard Worker }
173*38e8c45fSAndroid Build Coastguard Worker }
174*38e8c45fSAndroid Build Coastguard Worker
175*38e8c45fSAndroid Build Coastguard Worker } // namespace android
176