1 /* 2 * Copyright (C) 2010 The Android Open Source Project 3 * Copyright (C) 2012-2014, The Linux Foundation. All rights reserved. 4 * 5 * Not a Contribution, Apache license notifications and license are 6 * retained for attribution purposes only. 7 8 * Licensed under the Apache License, Version 2.0 (the "License"); 9 * you may not use this file except in compliance with the License. 10 * You may obtain a copy of the License at 11 * 12 * http://www.apache.org/licenses/LICENSE-2.0 13 * 14 * Unless required by applicable law or agreed to in writing, software 15 * distributed under the License is distributed on an "AS IS" BASIS, 16 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 17 * See the License for the specific language governing permissions and 18 * limitations under the License. 19 */ 20 21 #include <fcntl.h> 22 #include <stdint.h> 23 #include <sys/types.h> 24 #include <binder/Parcel.h> 25 #include <binder/IBinder.h> 26 #include <binder/IInterface.h> 27 #include <binder/IPCThreadState.h> 28 #include <utils/Errors.h> 29 #include <private/android_filesystem_config.h> 30 #include <IQService.h> 31 32 #define QSERVICE_DEBUG 0 33 34 using namespace android; 35 using namespace qClient; 36 37 // --------------------------------------------------------------------------- 38 39 namespace qService { 40 41 class BpQService : public BpInterface<IQService> 42 { 43 public: 44 BpQService(const sp<IBinder>& impl) 45 : BpInterface<IQService>(impl) {} 46 47 virtual void connect(const sp<IQClient>& client) { 48 ALOGD_IF(QSERVICE_DEBUG, "%s: connect HWC client", __FUNCTION__); 49 Parcel data, reply; 50 data.writeInterfaceToken(IQService::getInterfaceDescriptor()); 51 data.writeStrongBinder(IInterface::asBinder(client)); 52 remote()->transact(CONNECT_HWC_CLIENT, data, &reply); 53 } 54 55 virtual void connect(const sp<IQHDMIClient>& client) { 56 ALOGD_IF(QSERVICE_DEBUG, "%s: connect HDMI client", __FUNCTION__); 57 Parcel data, reply; 58 data.writeInterfaceToken(IQService::getInterfaceDescriptor()); 59 data.writeStrongBinder(IInterface::asBinder(client)); 60 remote()->transact(CONNECT_HDMI_CLIENT, data, &reply); 61 } 62 63 64 virtual android::status_t dispatch(uint32_t command, const Parcel* inParcel, 65 Parcel* outParcel) { 66 ALOGD_IF(QSERVICE_DEBUG, "%s: dispatch in:%p", __FUNCTION__, inParcel); 67 status_t err = (status_t) android::FAILED_TRANSACTION; 68 Parcel data; 69 Parcel *reply = outParcel; 70 data.writeInterfaceToken(IQService::getInterfaceDescriptor()); 71 if (inParcel && inParcel->dataSize() > 0) 72 data.appendFrom(inParcel, 0, inParcel->dataSize()); 73 err = remote()->transact(command, data, reply); 74 return err; 75 } 76 }; 77 78 IMPLEMENT_META_INTERFACE(QService, "android.display.IQService"); 79 80 // ---------------------------------------------------------------------- 81 82 static void getProcName(int pid, char *buf, int size); 83 84 status_t BnQService::onTransact( 85 uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags) 86 { 87 ALOGD_IF(QSERVICE_DEBUG, "%s: code: %d", __FUNCTION__, code); 88 // IPC should be from certain processes only 89 IPCThreadState* ipc = IPCThreadState::self(); 90 const int callerPid = ipc->getCallingPid(); 91 const int callerUid = ipc->getCallingUid(); 92 const int MAX_BUF_SIZE = 1024; 93 char callingProcName[MAX_BUF_SIZE] = {0}; 94 95 getProcName(callerPid, callingProcName, MAX_BUF_SIZE); 96 97 const bool permission = (callerUid == AID_MEDIA || 98 callerUid == AID_GRAPHICS || 99 callerUid == AID_ROOT || 100 callerUid == AID_SYSTEM); 101 102 if (code == CONNECT_HWC_CLIENT) { 103 CHECK_INTERFACE(IQService, data, reply); 104 if(callerUid != AID_GRAPHICS) { 105 ALOGE("display.qservice CONNECT_HWC_CLIENT access denied: \ 106 pid=%d uid=%d process=%s", 107 callerPid, callerUid, callingProcName); 108 return PERMISSION_DENIED; 109 } 110 sp<IQClient> client = 111 interface_cast<IQClient>(data.readStrongBinder()); 112 connect(client); 113 return NO_ERROR; 114 } else if(code == CONNECT_HDMI_CLIENT) { 115 CHECK_INTERFACE(IQService, data, reply); 116 if(callerUid != AID_SYSTEM && callerUid != AID_ROOT) { 117 ALOGE("display.qservice CONNECT_HDMI_CLIENT access denied: \ 118 pid=%d uid=%d process=%s", 119 callerPid, callerUid, callingProcName); 120 return PERMISSION_DENIED; 121 } 122 sp<IQHDMIClient> client = 123 interface_cast<IQHDMIClient>(data.readStrongBinder()); 124 connect(client); 125 return NO_ERROR; 126 } else if (code > COMMAND_LIST_START && code < COMMAND_LIST_END) { 127 if(!permission) { 128 ALOGE("display.qservice access denied: command=%d\ 129 pid=%d uid=%d process=%s", code, callerPid, 130 callerUid, callingProcName); 131 return PERMISSION_DENIED; 132 } 133 CHECK_INTERFACE(IQService, data, reply); 134 dispatch(code, &data, reply); 135 return NO_ERROR; 136 } else { 137 return BBinder::onTransact(code, data, reply, flags); 138 } 139 } 140 141 //Helper 142 static void getProcName(int pid, char *buf, int size) { 143 int fd = -1; 144 snprintf(buf, size, "/proc/%d/cmdline", pid); 145 fd = open(buf, O_RDONLY); 146 if (fd < 0) { 147 strlcpy(buf, "Unknown", size); 148 } else { 149 ssize_t len = read(fd, buf, size - 1); 150 if (len >= 0) 151 buf[len] = 0; 152 153 close(fd); 154 } 155 } 156 157 }; // namespace qService 158