1 /*
2  * Copyright (C) 2018 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_TAG "android.hardware.usb.gadget@1.1-service.redfin"
18 
19 #include "UsbGadget.h"
20 #include <dirent.h>
21 #include <fcntl.h>
22 #include <stdio.h>
23 #include <sys/inotify.h>
24 #include <sys/mount.h>
25 #include <sys/stat.h>
26 #include <sys/types.h>
27 #include <unistd.h>
28 
29 namespace android {
30 namespace hardware {
31 namespace usb {
32 namespace gadget {
33 namespace V1_1 {
34 namespace implementation {
35 
UsbGadget()36 UsbGadget::UsbGadget() {
37     if (access(OS_DESC_PATH, R_OK) != 0) {
38         ALOGE("configfs setup not done yet");
39         abort();
40     }
41 }
42 
currentFunctionsAppliedCallback(bool functionsApplied,void * payload)43 void currentFunctionsAppliedCallback(bool functionsApplied, void *payload) {
44     UsbGadget *gadget = (UsbGadget *)payload;
45     gadget->mCurrentUsbFunctionsApplied = functionsApplied;
46 }
47 
getCurrentUsbFunctions(const sp<V1_0::IUsbGadgetCallback> & callback)48 Return<void> UsbGadget::getCurrentUsbFunctions(const sp<V1_0::IUsbGadgetCallback> &callback) {
49     Return<void> ret = callback->getCurrentUsbFunctionsCb(
50         mCurrentUsbFunctions,
51         mCurrentUsbFunctionsApplied ? Status::FUNCTIONS_APPLIED : Status::FUNCTIONS_NOT_APPLIED);
52     if (!ret.isOk())
53         ALOGE("Call to getCurrentUsbFunctionsCb failed %s", ret.description().c_str());
54 
55     return Void();
56 }
57 
tearDownGadget()58 V1_0::Status UsbGadget::tearDownGadget() {
59     if (resetGadget() != Status::SUCCESS)
60         return Status::ERROR;
61 
62     if (monitorFfs.isMonitorRunning()) {
63         monitorFfs.reset();
64     } else {
65         ALOGI("mMonitor not running");
66     }
67     return Status::SUCCESS;
68 }
69 
validateAndSetVidPid(uint64_t functions)70 static V1_0::Status validateAndSetVidPid(uint64_t functions) {
71     V1_0::Status ret = Status::SUCCESS;
72     std::string vendorFunctions = getVendorFunctions();
73 
74     switch (functions) {
75         case static_cast<uint64_t>(GadgetFunction::MTP):
76             if (vendorFunctions == "diag") {
77                 ret = setVidPid("0x05C6", "0x901B");
78             } else {
79                 if (!(vendorFunctions == "user" || vendorFunctions == "")) {
80                     ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
81                     ret = Status::CONFIGURATION_NOT_SUPPORTED;
82                 } else {
83                     ret = setVidPid("0x18d1", "0x4ee1");
84                 }
85             }
86             break;
87         case GadgetFunction::ADB | GadgetFunction::MTP:
88             if (vendorFunctions == "diag") {
89                 ret = setVidPid("0x05C6", "0x903A");
90             } else {
91                 if (!(vendorFunctions == "user" || vendorFunctions == "")) {
92                     ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
93                     ret = Status::CONFIGURATION_NOT_SUPPORTED;
94                 } else {
95                     ret = setVidPid("0x18d1", "0x4ee2");
96                 }
97             }
98             break;
99         case static_cast<uint64_t>(GadgetFunction::RNDIS):
100             if (vendorFunctions == "diag") {
101                 ret = setVidPid("0x05C6", "0x902C");
102             } else if (vendorFunctions == "serial_cdev,diag") {
103                 ret = setVidPid("0x05C6", "0x90B5");
104             } else if (vendorFunctions == "diag,diag_mdm,qdss,qdss_mdm,serial_cdev,dpl_gsi") {
105                 ret = setVidPid("0x05C6", "0x90E6");
106             } else {
107                 if (!(vendorFunctions == "user" || vendorFunctions == "")) {
108                     ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
109                     ret = Status::CONFIGURATION_NOT_SUPPORTED;
110                 } else {
111                     ret = setVidPid("0x18d1", "0x4ee3");
112                 }
113             }
114             break;
115         case GadgetFunction::ADB | GadgetFunction::RNDIS:
116             if (vendorFunctions == "diag") {
117                 ret = setVidPid("0x05C6", "0x902D");
118             } else if (vendorFunctions == "serial_cdev,diag") {
119                 ret = setVidPid("0x05C6", "0x90B6");
120             } else if (vendorFunctions == "diag,diag_mdm,qdss,qdss_mdm,serial_cdev,dpl_gsi") {
121                 ret = setVidPid("0x05C6", "0x90E7");
122             } else {
123                 if (!(vendorFunctions == "user" || vendorFunctions == "")) {
124                     ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
125                     ret = Status::CONFIGURATION_NOT_SUPPORTED;
126                 } else {
127                     ret = setVidPid("0x18d1", "0x4ee4");
128                 }
129             }
130             break;
131         case static_cast<uint64_t>(GadgetFunction::PTP):
132             if (!(vendorFunctions == "user" || vendorFunctions == "")) {
133                 ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
134                 ret = Status::CONFIGURATION_NOT_SUPPORTED;
135             } else {
136                 ret = setVidPid("0x18d1", "0x4ee5");
137             }
138             break;
139         case GadgetFunction::ADB | GadgetFunction::PTP:
140             if (!(vendorFunctions == "user" || vendorFunctions == "")) {
141                 ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
142                 ret = Status::CONFIGURATION_NOT_SUPPORTED;
143             } else {
144                 ret = setVidPid("0x18d1", "0x4ee6");
145             }
146             break;
147         case static_cast<uint64_t>(GadgetFunction::ADB):
148             if (vendorFunctions == "diag") {
149                 ret = setVidPid("0x05C6", "0x901D");
150             } else if (vendorFunctions == "diag,serial_cdev,rmnet_gsi") {
151                 ret = setVidPid("0x05C6", "0x9091");
152             } else if (vendorFunctions == "diag,serial_cdev") {
153                 ret = setVidPid("0x05C6", "0x901F");
154             } else if (vendorFunctions == "diag,serial_cdev,rmnet_gsi,dpl_gsi,qdss") {
155                 ret = setVidPid("0x05C6", "0x90DB");
156             } else if (vendorFunctions ==
157                        "diag,diag_mdm,qdss,qdss_mdm,serial_cdev,dpl_gsi,rmnet_gsi") {
158                 ret = setVidPid("0x05C6", "0x90E5");
159             } else {
160                 if (!(vendorFunctions == "user" || vendorFunctions == "")) {
161                     ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
162                     ret = Status::CONFIGURATION_NOT_SUPPORTED;
163                 } else {
164                     ret = setVidPid("0x18d1", "0x4ee7");
165                 }
166             }
167             break;
168         case static_cast<uint64_t>(GadgetFunction::MIDI):
169             if (!(vendorFunctions == "user" || vendorFunctions == "")) {
170                 ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
171                 ret = Status::CONFIGURATION_NOT_SUPPORTED;
172             } else {
173                 ret = setVidPid("0x18d1", "0x4ee8");
174             }
175             break;
176         case GadgetFunction::ADB | GadgetFunction::MIDI:
177             if (!(vendorFunctions == "user" || vendorFunctions == "")) {
178                 ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
179                 ret = Status::CONFIGURATION_NOT_SUPPORTED;
180             } else {
181                 ret = setVidPid("0x18d1", "0x4ee9");
182             }
183             break;
184         case static_cast<uint64_t>(GadgetFunction::ACCESSORY):
185             if (!(vendorFunctions == "user" || vendorFunctions == ""))
186                 ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
187             ret = setVidPid("0x18d1", "0x2d00");
188             break;
189         case GadgetFunction::ADB | GadgetFunction::ACCESSORY:
190             if (!(vendorFunctions == "user" || vendorFunctions == ""))
191                 ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
192             ret = setVidPid("0x18d1", "0x2d01");
193             break;
194         case static_cast<uint64_t>(GadgetFunction::AUDIO_SOURCE):
195             if (!(vendorFunctions == "user" || vendorFunctions == ""))
196                 ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
197             ret = setVidPid("0x18d1", "0x2d02");
198             break;
199         case GadgetFunction::ADB | GadgetFunction::AUDIO_SOURCE:
200             if (!(vendorFunctions == "user" || vendorFunctions == ""))
201                 ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
202             ret = setVidPid("0x18d1", "0x2d03");
203             break;
204         case GadgetFunction::ACCESSORY | GadgetFunction::AUDIO_SOURCE:
205             if (!(vendorFunctions == "user" || vendorFunctions == ""))
206                 ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
207             ret = setVidPid("0x18d1", "0x2d04");
208             break;
209         case GadgetFunction::ADB | GadgetFunction::ACCESSORY | GadgetFunction::AUDIO_SOURCE:
210             if (!(vendorFunctions == "user" || vendorFunctions == ""))
211                 ALOGE("Invalid vendorFunctions set: %s", vendorFunctions.c_str());
212             ret = setVidPid("0x18d1", "0x2d05");
213             break;
214         default:
215             ALOGE("Combination not supported");
216             ret = Status::CONFIGURATION_NOT_SUPPORTED;
217     }
218     return ret;
219 }
220 
reset()221 Return<Status> UsbGadget::reset() {
222     ALOGI("USB Gadget reset");
223 
224     if (!WriteStringToFile("none", PULLUP_PATH)) {
225         ALOGI("Gadget cannot be pulled down");
226         return Status::ERROR;
227     }
228 
229     usleep(kDisconnectWaitUs);
230 
231     if (!WriteStringToFile(kGadgetName, PULLUP_PATH)) {
232         ALOGI("Gadget cannot be pulled up");
233         return Status::ERROR;
234     }
235 
236     return Status::SUCCESS;
237 }
238 
setupFunctions(uint64_t functions,const sp<V1_0::IUsbGadgetCallback> & callback,uint64_t timeout)239 V1_0::Status UsbGadget::setupFunctions(uint64_t functions,
240                                        const sp<V1_0::IUsbGadgetCallback> &callback,
241                                        uint64_t timeout) {
242     bool ffsEnabled = false;
243     int i = 0;
244 
245     if (addGenericAndroidFunctions(&monitorFfs, functions, &ffsEnabled, &i) != Status::SUCCESS)
246         return Status::ERROR;
247 
248     std::string vendorFunctions = getVendorFunctions();
249 
250     if (vendorFunctions != "") {
251         ALOGI("enable usbradio debug functions");
252         char *function = strtok(const_cast<char *>(vendorFunctions.c_str()), ",");
253         while (function != NULL) {
254             if (string(function) == "diag" && linkFunction("diag.diag", i++))
255                 return Status::ERROR;
256             if (string(function) == "diag_mdm" && linkFunction("diag.diag_mdm", i++))
257                 return Status::ERROR;
258             if (string(function) == "qdss" && linkFunction("qdss.qdss", i++))
259                 return Status::ERROR;
260             if (string(function) == "qdss_mdm" && linkFunction("qdss.qdss_mdm", i++))
261                 return Status::ERROR;
262             if (string(function) == "serial_cdev" && linkFunction("cser.dun.0", i++))
263                 return Status::ERROR;
264             if (string(function) == "dpl_gsi" && linkFunction("gsi.dpl", i++))
265                 return Status::ERROR;
266             if (string(function) == "rmnet_gsi" && linkFunction("gsi.rmnet", i++))
267                 return Status::ERROR;
268             function = strtok(NULL, ",");
269         }
270     }
271 
272     if ((functions & GadgetFunction::ADB) != 0) {
273         ffsEnabled = true;
274         if (addAdb(&monitorFfs, &i) != Status::SUCCESS)
275             return Status::ERROR;
276     }
277 
278     // Pull up the gadget right away when there are no ffs functions.
279     if (!ffsEnabled) {
280         if (!WriteStringToFile(kGadgetName, PULLUP_PATH))
281             return Status::ERROR;
282         mCurrentUsbFunctionsApplied = true;
283         if (callback)
284             callback->setCurrentUsbFunctionsCb(functions, Status::SUCCESS);
285         return Status::SUCCESS;
286     }
287 
288     monitorFfs.registerFunctionsAppliedCallback(&currentFunctionsAppliedCallback, this);
289     // Monitors the ffs paths to pull up the gadget when descriptors are written.
290     // Also takes of the pulling up the gadget again if the userspace process
291     // dies and restarts.
292     monitorFfs.startMonitor();
293 
294     if (kDebug)
295         ALOGI("Mainthread in Cv");
296 
297     if (callback) {
298         bool pullup = monitorFfs.waitForPullUp(timeout);
299         Return<void> ret = callback->setCurrentUsbFunctionsCb(
300             functions, pullup ? Status::SUCCESS : Status::ERROR);
301         if (!ret.isOk())
302             ALOGE("setCurrentUsbFunctionsCb error %s", ret.description().c_str());
303     }
304 
305     return Status::SUCCESS;
306 }
307 
setCurrentUsbFunctions(uint64_t functions,const sp<V1_0::IUsbGadgetCallback> & callback,uint64_t timeout)308 Return<void> UsbGadget::setCurrentUsbFunctions(uint64_t functions,
309                                                const sp<V1_0::IUsbGadgetCallback> &callback,
310                                                uint64_t timeout) {
311     std::unique_lock<std::mutex> lk(mLockSetCurrentFunction);
312 
313     mCurrentUsbFunctions = functions;
314     mCurrentUsbFunctionsApplied = false;
315 
316     // Unlink the gadget and stop the monitor if running.
317     V1_0::Status status = tearDownGadget();
318     if (status != Status::SUCCESS) {
319         goto error;
320     }
321 
322     ALOGI("Returned from tearDown gadget");
323 
324     // Leave the gadget pulled down to give time for the host to sense disconnect.
325     usleep(kDisconnectWaitUs);
326 
327     if (functions == static_cast<uint64_t>(GadgetFunction::NONE)) {
328         if (callback == NULL)
329             return Void();
330         Return<void> ret = callback->setCurrentUsbFunctionsCb(functions, Status::SUCCESS);
331         if (!ret.isOk())
332             ALOGE("Error while calling setCurrentUsbFunctionsCb %s", ret.description().c_str());
333         return Void();
334     }
335 
336     status = validateAndSetVidPid(functions);
337 
338     if (status != Status::SUCCESS) {
339         goto error;
340     }
341 
342     status = setupFunctions(functions, callback, timeout);
343     if (status != Status::SUCCESS) {
344         goto error;
345     }
346 
347     ALOGI("Usb Gadget setcurrent functions called successfully");
348     return Void();
349 
350 error:
351     ALOGI("Usb Gadget setcurrent functions failed");
352     if (callback == NULL)
353         return Void();
354     Return<void> ret = callback->setCurrentUsbFunctionsCb(functions, status);
355     if (!ret.isOk())
356         ALOGE("Error while calling setCurrentUsbFunctionsCb %s", ret.description().c_str());
357     return Void();
358 }
359 }  // namespace implementation
360 }  // namespace V1_1
361 }  // namespace gadget
362 }  // namespace usb
363 }  // namespace hardware
364 }  // namespace android
365