xref: /aosp_15_r20/hardware/interfaces/camera/provider/default/ExternalCameraProvider.cpp (revision 4d7e907c777eeecc4c5bd7cf640a754fac206ff7)
1*4d7e907cSAndroid Build Coastguard Worker /*
2*4d7e907cSAndroid Build Coastguard Worker  * Copyright (C) 2022 The Android Open Source Project
3*4d7e907cSAndroid Build Coastguard Worker  *
4*4d7e907cSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*4d7e907cSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*4d7e907cSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*4d7e907cSAndroid Build Coastguard Worker  *
8*4d7e907cSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*4d7e907cSAndroid Build Coastguard Worker  *
10*4d7e907cSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*4d7e907cSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*4d7e907cSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*4d7e907cSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*4d7e907cSAndroid Build Coastguard Worker  * limitations under the License.
15*4d7e907cSAndroid Build Coastguard Worker  */
16*4d7e907cSAndroid Build Coastguard Worker 
17*4d7e907cSAndroid Build Coastguard Worker #define LOG_TAG "ExtCamPrvdr"
18*4d7e907cSAndroid Build Coastguard Worker // #define LOG_NDEBUG 0
19*4d7e907cSAndroid Build Coastguard Worker 
20*4d7e907cSAndroid Build Coastguard Worker #include "ExternalCameraProvider.h"
21*4d7e907cSAndroid Build Coastguard Worker 
22*4d7e907cSAndroid Build Coastguard Worker #include <ExternalCameraDevice.h>
23*4d7e907cSAndroid Build Coastguard Worker #include <aidl/android/hardware/camera/common/Status.h>
24*4d7e907cSAndroid Build Coastguard Worker #include <convert.h>
25*4d7e907cSAndroid Build Coastguard Worker #include <cutils/properties.h>
26*4d7e907cSAndroid Build Coastguard Worker #include <linux/videodev2.h>
27*4d7e907cSAndroid Build Coastguard Worker #include <log/log.h>
28*4d7e907cSAndroid Build Coastguard Worker #include <sys/inotify.h>
29*4d7e907cSAndroid Build Coastguard Worker #include <regex>
30*4d7e907cSAndroid Build Coastguard Worker 
31*4d7e907cSAndroid Build Coastguard Worker namespace android {
32*4d7e907cSAndroid Build Coastguard Worker namespace hardware {
33*4d7e907cSAndroid Build Coastguard Worker namespace camera {
34*4d7e907cSAndroid Build Coastguard Worker namespace provider {
35*4d7e907cSAndroid Build Coastguard Worker namespace implementation {
36*4d7e907cSAndroid Build Coastguard Worker 
37*4d7e907cSAndroid Build Coastguard Worker using ::aidl::android::hardware::camera::common::Status;
38*4d7e907cSAndroid Build Coastguard Worker using ::android::hardware::camera::device::implementation::ExternalCameraDevice;
39*4d7e907cSAndroid Build Coastguard Worker using ::android::hardware::camera::device::implementation::fromStatus;
40*4d7e907cSAndroid Build Coastguard Worker using ::android::hardware::camera::external::common::ExternalCameraConfig;
41*4d7e907cSAndroid Build Coastguard Worker 
42*4d7e907cSAndroid Build Coastguard Worker namespace {
43*4d7e907cSAndroid Build Coastguard Worker // "device@<version>/external/<id>"
44*4d7e907cSAndroid Build Coastguard Worker const std::regex kDeviceNameRE("device@([0-9]+\\.[0-9]+)/external/(.+)");
45*4d7e907cSAndroid Build Coastguard Worker const int kMaxDevicePathLen = 256;
46*4d7e907cSAndroid Build Coastguard Worker constexpr char kDevicePath[] = "/dev/";
47*4d7e907cSAndroid Build Coastguard Worker constexpr char kPrefix[] = "video";
48*4d7e907cSAndroid Build Coastguard Worker constexpr int kPrefixLen = sizeof(kPrefix) - 1;
49*4d7e907cSAndroid Build Coastguard Worker constexpr int kDevicePrefixLen = sizeof(kDevicePath) + kPrefixLen - 1;
50*4d7e907cSAndroid Build Coastguard Worker 
matchDeviceName(int cameraIdOffset,const std::string & deviceName,std::string * deviceVersion,std::string * cameraDevicePath)51*4d7e907cSAndroid Build Coastguard Worker bool matchDeviceName(int cameraIdOffset, const std::string& deviceName, std::string* deviceVersion,
52*4d7e907cSAndroid Build Coastguard Worker                      std::string* cameraDevicePath) {
53*4d7e907cSAndroid Build Coastguard Worker     std::smatch sm;
54*4d7e907cSAndroid Build Coastguard Worker     if (std::regex_match(deviceName, sm, kDeviceNameRE)) {
55*4d7e907cSAndroid Build Coastguard Worker         if (deviceVersion != nullptr) {
56*4d7e907cSAndroid Build Coastguard Worker             *deviceVersion = sm[1];
57*4d7e907cSAndroid Build Coastguard Worker         }
58*4d7e907cSAndroid Build Coastguard Worker         if (cameraDevicePath != nullptr) {
59*4d7e907cSAndroid Build Coastguard Worker             *cameraDevicePath = "/dev/video" + std::to_string(std::stoi(sm[2]) - cameraIdOffset);
60*4d7e907cSAndroid Build Coastguard Worker         }
61*4d7e907cSAndroid Build Coastguard Worker         return true;
62*4d7e907cSAndroid Build Coastguard Worker     }
63*4d7e907cSAndroid Build Coastguard Worker     return false;
64*4d7e907cSAndroid Build Coastguard Worker }
65*4d7e907cSAndroid Build Coastguard Worker }  // namespace
66*4d7e907cSAndroid Build Coastguard Worker 
ExternalCameraProvider()67*4d7e907cSAndroid Build Coastguard Worker ExternalCameraProvider::ExternalCameraProvider() : mCfg(ExternalCameraConfig::loadFromCfg()) {
68*4d7e907cSAndroid Build Coastguard Worker     mHotPlugThread = std::make_shared<HotplugThread>(this);
69*4d7e907cSAndroid Build Coastguard Worker     mHotPlugThread->run();
70*4d7e907cSAndroid Build Coastguard Worker }
71*4d7e907cSAndroid Build Coastguard Worker 
~ExternalCameraProvider()72*4d7e907cSAndroid Build Coastguard Worker ExternalCameraProvider::~ExternalCameraProvider() {
73*4d7e907cSAndroid Build Coastguard Worker     mHotPlugThread->requestExitAndWait();
74*4d7e907cSAndroid Build Coastguard Worker }
75*4d7e907cSAndroid Build Coastguard Worker 
setCallback(const std::shared_ptr<ICameraProviderCallback> & in_callback)76*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus ExternalCameraProvider::setCallback(
77*4d7e907cSAndroid Build Coastguard Worker         const std::shared_ptr<ICameraProviderCallback>& in_callback) {
78*4d7e907cSAndroid Build Coastguard Worker     if (in_callback == nullptr) {
79*4d7e907cSAndroid Build Coastguard Worker         return fromStatus(Status::ILLEGAL_ARGUMENT);
80*4d7e907cSAndroid Build Coastguard Worker     }
81*4d7e907cSAndroid Build Coastguard Worker 
82*4d7e907cSAndroid Build Coastguard Worker     {
83*4d7e907cSAndroid Build Coastguard Worker         Mutex::Autolock _l(mLock);
84*4d7e907cSAndroid Build Coastguard Worker         mCallback = in_callback;
85*4d7e907cSAndroid Build Coastguard Worker     }
86*4d7e907cSAndroid Build Coastguard Worker 
87*4d7e907cSAndroid Build Coastguard Worker     for (const auto& pair : mCameraStatusMap) {
88*4d7e907cSAndroid Build Coastguard Worker         mCallback->cameraDeviceStatusChange(pair.first, pair.second);
89*4d7e907cSAndroid Build Coastguard Worker     }
90*4d7e907cSAndroid Build Coastguard Worker     return fromStatus(Status::OK);
91*4d7e907cSAndroid Build Coastguard Worker }
92*4d7e907cSAndroid Build Coastguard Worker 
getVendorTags(std::vector<VendorTagSection> * _aidl_return)93*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus ExternalCameraProvider::getVendorTags(
94*4d7e907cSAndroid Build Coastguard Worker         std::vector<VendorTagSection>* _aidl_return) {
95*4d7e907cSAndroid Build Coastguard Worker     if (_aidl_return == nullptr) {
96*4d7e907cSAndroid Build Coastguard Worker         return fromStatus(Status::ILLEGAL_ARGUMENT);
97*4d7e907cSAndroid Build Coastguard Worker     }
98*4d7e907cSAndroid Build Coastguard Worker     // No vendor tag support for USB camera
99*4d7e907cSAndroid Build Coastguard Worker     *_aidl_return = {};
100*4d7e907cSAndroid Build Coastguard Worker     return fromStatus(Status::OK);
101*4d7e907cSAndroid Build Coastguard Worker }
102*4d7e907cSAndroid Build Coastguard Worker 
getCameraIdList(std::vector<std::string> * _aidl_return)103*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus ExternalCameraProvider::getCameraIdList(std::vector<std::string>* _aidl_return) {
104*4d7e907cSAndroid Build Coastguard Worker     if (_aidl_return == nullptr) {
105*4d7e907cSAndroid Build Coastguard Worker         return fromStatus(Status::ILLEGAL_ARGUMENT);
106*4d7e907cSAndroid Build Coastguard Worker     }
107*4d7e907cSAndroid Build Coastguard Worker     // External camera HAL always report 0 camera, and extra cameras
108*4d7e907cSAndroid Build Coastguard Worker     // are just reported via cameraDeviceStatusChange callbacks
109*4d7e907cSAndroid Build Coastguard Worker     *_aidl_return = {};
110*4d7e907cSAndroid Build Coastguard Worker     return fromStatus(Status::OK);
111*4d7e907cSAndroid Build Coastguard Worker }
112*4d7e907cSAndroid Build Coastguard Worker 
getCameraDeviceInterface(const std::string & in_cameraDeviceName,std::shared_ptr<ICameraDevice> * _aidl_return)113*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus ExternalCameraProvider::getCameraDeviceInterface(
114*4d7e907cSAndroid Build Coastguard Worker         const std::string& in_cameraDeviceName, std::shared_ptr<ICameraDevice>* _aidl_return) {
115*4d7e907cSAndroid Build Coastguard Worker     if (_aidl_return == nullptr) {
116*4d7e907cSAndroid Build Coastguard Worker         return fromStatus(Status::ILLEGAL_ARGUMENT);
117*4d7e907cSAndroid Build Coastguard Worker     }
118*4d7e907cSAndroid Build Coastguard Worker     std::string cameraDevicePath, deviceVersion;
119*4d7e907cSAndroid Build Coastguard Worker     bool match = matchDeviceName(mCfg.cameraIdOffset, in_cameraDeviceName, &deviceVersion,
120*4d7e907cSAndroid Build Coastguard Worker                                  &cameraDevicePath);
121*4d7e907cSAndroid Build Coastguard Worker 
122*4d7e907cSAndroid Build Coastguard Worker     if (!match) {
123*4d7e907cSAndroid Build Coastguard Worker         *_aidl_return = nullptr;
124*4d7e907cSAndroid Build Coastguard Worker         return fromStatus(Status::ILLEGAL_ARGUMENT);
125*4d7e907cSAndroid Build Coastguard Worker     }
126*4d7e907cSAndroid Build Coastguard Worker 
127*4d7e907cSAndroid Build Coastguard Worker     if (mCameraStatusMap.count(in_cameraDeviceName) == 0 ||
128*4d7e907cSAndroid Build Coastguard Worker         mCameraStatusMap[in_cameraDeviceName] != CameraDeviceStatus::PRESENT) {
129*4d7e907cSAndroid Build Coastguard Worker         *_aidl_return = nullptr;
130*4d7e907cSAndroid Build Coastguard Worker         return fromStatus(Status::ILLEGAL_ARGUMENT);
131*4d7e907cSAndroid Build Coastguard Worker     }
132*4d7e907cSAndroid Build Coastguard Worker 
133*4d7e907cSAndroid Build Coastguard Worker     ALOGV("Constructing external camera device");
134*4d7e907cSAndroid Build Coastguard Worker     std::shared_ptr<ExternalCameraDevice> deviceImpl =
135*4d7e907cSAndroid Build Coastguard Worker             ndk::SharedRefBase::make<ExternalCameraDevice>(cameraDevicePath, mCfg);
136*4d7e907cSAndroid Build Coastguard Worker     if (deviceImpl == nullptr || deviceImpl->isInitFailed()) {
137*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: camera device %s init failed!", __FUNCTION__, cameraDevicePath.c_str());
138*4d7e907cSAndroid Build Coastguard Worker         *_aidl_return = nullptr;
139*4d7e907cSAndroid Build Coastguard Worker         return fromStatus(Status::INTERNAL_ERROR);
140*4d7e907cSAndroid Build Coastguard Worker     }
141*4d7e907cSAndroid Build Coastguard Worker 
142*4d7e907cSAndroid Build Coastguard Worker     IF_ALOGV() {
143*4d7e907cSAndroid Build Coastguard Worker         int interfaceVersion;
144*4d7e907cSAndroid Build Coastguard Worker         deviceImpl->getInterfaceVersion(&interfaceVersion);
145*4d7e907cSAndroid Build Coastguard Worker         ALOGV("%s: device interface version: %d", __FUNCTION__, interfaceVersion);
146*4d7e907cSAndroid Build Coastguard Worker     }
147*4d7e907cSAndroid Build Coastguard Worker 
148*4d7e907cSAndroid Build Coastguard Worker     *_aidl_return = deviceImpl;
149*4d7e907cSAndroid Build Coastguard Worker     return fromStatus(Status::OK);
150*4d7e907cSAndroid Build Coastguard Worker }
151*4d7e907cSAndroid Build Coastguard Worker 
notifyDeviceStateChange(int64_t)152*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus ExternalCameraProvider::notifyDeviceStateChange(int64_t) {
153*4d7e907cSAndroid Build Coastguard Worker     return fromStatus(Status::OK);
154*4d7e907cSAndroid Build Coastguard Worker }
155*4d7e907cSAndroid Build Coastguard Worker 
getConcurrentCameraIds(std::vector<ConcurrentCameraIdCombination> * _aidl_return)156*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus ExternalCameraProvider::getConcurrentCameraIds(
157*4d7e907cSAndroid Build Coastguard Worker         std::vector<ConcurrentCameraIdCombination>* _aidl_return) {
158*4d7e907cSAndroid Build Coastguard Worker     if (_aidl_return == nullptr) {
159*4d7e907cSAndroid Build Coastguard Worker         return fromStatus(Status::ILLEGAL_ARGUMENT);
160*4d7e907cSAndroid Build Coastguard Worker     }
161*4d7e907cSAndroid Build Coastguard Worker     *_aidl_return = {};
162*4d7e907cSAndroid Build Coastguard Worker     return fromStatus(Status::OK);
163*4d7e907cSAndroid Build Coastguard Worker }
164*4d7e907cSAndroid Build Coastguard Worker 
isConcurrentStreamCombinationSupported(const std::vector<CameraIdAndStreamCombination> &,bool * _aidl_return)165*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus ExternalCameraProvider::isConcurrentStreamCombinationSupported(
166*4d7e907cSAndroid Build Coastguard Worker         const std::vector<CameraIdAndStreamCombination>&, bool* _aidl_return) {
167*4d7e907cSAndroid Build Coastguard Worker     if (_aidl_return == nullptr) {
168*4d7e907cSAndroid Build Coastguard Worker         return fromStatus(Status::ILLEGAL_ARGUMENT);
169*4d7e907cSAndroid Build Coastguard Worker     }
170*4d7e907cSAndroid Build Coastguard Worker     // No concurrent stream combinations are supported
171*4d7e907cSAndroid Build Coastguard Worker     *_aidl_return = false;
172*4d7e907cSAndroid Build Coastguard Worker     return fromStatus(Status::OK);
173*4d7e907cSAndroid Build Coastguard Worker }
174*4d7e907cSAndroid Build Coastguard Worker 
addExternalCamera(const char * devName)175*4d7e907cSAndroid Build Coastguard Worker void ExternalCameraProvider::addExternalCamera(const char* devName) {
176*4d7e907cSAndroid Build Coastguard Worker     ALOGV("%s: ExtCam: adding %s to External Camera HAL!", __FUNCTION__, devName);
177*4d7e907cSAndroid Build Coastguard Worker     Mutex::Autolock _l(mLock);
178*4d7e907cSAndroid Build Coastguard Worker     std::string deviceName;
179*4d7e907cSAndroid Build Coastguard Worker     std::string cameraId =
180*4d7e907cSAndroid Build Coastguard Worker             std::to_string(mCfg.cameraIdOffset + std::atoi(devName + kDevicePrefixLen));
181*4d7e907cSAndroid Build Coastguard Worker     deviceName =
182*4d7e907cSAndroid Build Coastguard Worker             std::string("device@") + ExternalCameraDevice::kDeviceVersion + "/external/" + cameraId;
183*4d7e907cSAndroid Build Coastguard Worker     mCameraStatusMap[deviceName] = CameraDeviceStatus::PRESENT;
184*4d7e907cSAndroid Build Coastguard Worker     if (mCallback != nullptr) {
185*4d7e907cSAndroid Build Coastguard Worker         mCallback->cameraDeviceStatusChange(deviceName, CameraDeviceStatus::PRESENT);
186*4d7e907cSAndroid Build Coastguard Worker     }
187*4d7e907cSAndroid Build Coastguard Worker }
188*4d7e907cSAndroid Build Coastguard Worker 
deviceAdded(const char * devName)189*4d7e907cSAndroid Build Coastguard Worker void ExternalCameraProvider::deviceAdded(const char* devName) {
190*4d7e907cSAndroid Build Coastguard Worker     {
191*4d7e907cSAndroid Build Coastguard Worker         base::unique_fd fd(::open(devName, O_RDWR));
192*4d7e907cSAndroid Build Coastguard Worker         if (fd.get() < 0) {
193*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s open v4l2 device %s failed:%s", __FUNCTION__, devName, strerror(errno));
194*4d7e907cSAndroid Build Coastguard Worker             return;
195*4d7e907cSAndroid Build Coastguard Worker         }
196*4d7e907cSAndroid Build Coastguard Worker 
197*4d7e907cSAndroid Build Coastguard Worker         struct v4l2_capability capability;
198*4d7e907cSAndroid Build Coastguard Worker         int ret = ioctl(fd.get(), VIDIOC_QUERYCAP, &capability);
199*4d7e907cSAndroid Build Coastguard Worker         if (ret < 0) {
200*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s v4l2 QUERYCAP %s failed", __FUNCTION__, devName);
201*4d7e907cSAndroid Build Coastguard Worker             return;
202*4d7e907cSAndroid Build Coastguard Worker         }
203*4d7e907cSAndroid Build Coastguard Worker 
204*4d7e907cSAndroid Build Coastguard Worker         if (!(capability.device_caps & V4L2_CAP_VIDEO_CAPTURE)) {
205*4d7e907cSAndroid Build Coastguard Worker             ALOGW("%s device %s does not support VIDEO_CAPTURE", __FUNCTION__, devName);
206*4d7e907cSAndroid Build Coastguard Worker             return;
207*4d7e907cSAndroid Build Coastguard Worker         }
208*4d7e907cSAndroid Build Coastguard Worker     }
209*4d7e907cSAndroid Build Coastguard Worker 
210*4d7e907cSAndroid Build Coastguard Worker     // See if we can initialize ExternalCameraDevice correctly
211*4d7e907cSAndroid Build Coastguard Worker     std::shared_ptr<ExternalCameraDevice> deviceImpl =
212*4d7e907cSAndroid Build Coastguard Worker             ndk::SharedRefBase::make<ExternalCameraDevice>(devName, mCfg);
213*4d7e907cSAndroid Build Coastguard Worker     if (deviceImpl == nullptr || deviceImpl->isInitFailed()) {
214*4d7e907cSAndroid Build Coastguard Worker         ALOGW("%s: Attempt to init camera device %s failed!", __FUNCTION__, devName);
215*4d7e907cSAndroid Build Coastguard Worker         return;
216*4d7e907cSAndroid Build Coastguard Worker     }
217*4d7e907cSAndroid Build Coastguard Worker     deviceImpl.reset();
218*4d7e907cSAndroid Build Coastguard Worker     addExternalCamera(devName);
219*4d7e907cSAndroid Build Coastguard Worker }
220*4d7e907cSAndroid Build Coastguard Worker 
deviceRemoved(const char * devName)221*4d7e907cSAndroid Build Coastguard Worker void ExternalCameraProvider::deviceRemoved(const char* devName) {
222*4d7e907cSAndroid Build Coastguard Worker     Mutex::Autolock _l(mLock);
223*4d7e907cSAndroid Build Coastguard Worker     std::string deviceName;
224*4d7e907cSAndroid Build Coastguard Worker     std::string cameraId =
225*4d7e907cSAndroid Build Coastguard Worker             std::to_string(mCfg.cameraIdOffset + std::atoi(devName + kDevicePrefixLen));
226*4d7e907cSAndroid Build Coastguard Worker 
227*4d7e907cSAndroid Build Coastguard Worker     deviceName =
228*4d7e907cSAndroid Build Coastguard Worker             std::string("device@") + ExternalCameraDevice::kDeviceVersion + "/external/" + cameraId;
229*4d7e907cSAndroid Build Coastguard Worker 
230*4d7e907cSAndroid Build Coastguard Worker     if (mCameraStatusMap.erase(deviceName) == 0) {
231*4d7e907cSAndroid Build Coastguard Worker         // Unknown device, do not fire callback
232*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: cannot find camera device to remove %s", __FUNCTION__, devName);
233*4d7e907cSAndroid Build Coastguard Worker         return;
234*4d7e907cSAndroid Build Coastguard Worker     }
235*4d7e907cSAndroid Build Coastguard Worker 
236*4d7e907cSAndroid Build Coastguard Worker     if (mCallback != nullptr) {
237*4d7e907cSAndroid Build Coastguard Worker         mCallback->cameraDeviceStatusChange(deviceName, CameraDeviceStatus::NOT_PRESENT);
238*4d7e907cSAndroid Build Coastguard Worker     }
239*4d7e907cSAndroid Build Coastguard Worker }
240*4d7e907cSAndroid Build Coastguard Worker 
updateAttachedCameras()241*4d7e907cSAndroid Build Coastguard Worker void ExternalCameraProvider::updateAttachedCameras() {
242*4d7e907cSAndroid Build Coastguard Worker     ALOGV("%s start scanning for existing V4L2 devices", __FUNCTION__);
243*4d7e907cSAndroid Build Coastguard Worker 
244*4d7e907cSAndroid Build Coastguard Worker     // Find existing /dev/video* devices
245*4d7e907cSAndroid Build Coastguard Worker     DIR* devdir = opendir(kDevicePath);
246*4d7e907cSAndroid Build Coastguard Worker     if (devdir == nullptr) {
247*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: cannot open %s! Exiting threadloop", __FUNCTION__, kDevicePath);
248*4d7e907cSAndroid Build Coastguard Worker         return;
249*4d7e907cSAndroid Build Coastguard Worker     }
250*4d7e907cSAndroid Build Coastguard Worker 
251*4d7e907cSAndroid Build Coastguard Worker     struct dirent* de;
252*4d7e907cSAndroid Build Coastguard Worker     while ((de = readdir(devdir)) != nullptr) {
253*4d7e907cSAndroid Build Coastguard Worker         // Find external v4l devices that's existing before we start watching and add them
254*4d7e907cSAndroid Build Coastguard Worker         if (!strncmp(kPrefix, de->d_name, kPrefixLen)) {
255*4d7e907cSAndroid Build Coastguard Worker             std::string deviceId(de->d_name + kPrefixLen);
256*4d7e907cSAndroid Build Coastguard Worker             if (mCfg.mInternalDevices.count(deviceId) == 0) {
257*4d7e907cSAndroid Build Coastguard Worker                 ALOGV("Non-internal v4l device %s found", de->d_name);
258*4d7e907cSAndroid Build Coastguard Worker                 char v4l2DevicePath[kMaxDevicePathLen];
259*4d7e907cSAndroid Build Coastguard Worker                 snprintf(v4l2DevicePath, kMaxDevicePathLen, "%s%s", kDevicePath, de->d_name);
260*4d7e907cSAndroid Build Coastguard Worker                 deviceAdded(v4l2DevicePath);
261*4d7e907cSAndroid Build Coastguard Worker             }
262*4d7e907cSAndroid Build Coastguard Worker         }
263*4d7e907cSAndroid Build Coastguard Worker     }
264*4d7e907cSAndroid Build Coastguard Worker     closedir(devdir);
265*4d7e907cSAndroid Build Coastguard Worker }
266*4d7e907cSAndroid Build Coastguard Worker 
267*4d7e907cSAndroid Build Coastguard Worker // Start ExternalCameraProvider::HotplugThread functions
268*4d7e907cSAndroid Build Coastguard Worker 
HotplugThread(ExternalCameraProvider * parent)269*4d7e907cSAndroid Build Coastguard Worker ExternalCameraProvider::HotplugThread::HotplugThread(ExternalCameraProvider* parent)
270*4d7e907cSAndroid Build Coastguard Worker     : mParent(parent), mInternalDevices(parent->mCfg.mInternalDevices) {}
271*4d7e907cSAndroid Build Coastguard Worker 
~HotplugThread()272*4d7e907cSAndroid Build Coastguard Worker ExternalCameraProvider::HotplugThread::~HotplugThread() {
273*4d7e907cSAndroid Build Coastguard Worker     // Clean up inotify descriptor if needed.
274*4d7e907cSAndroid Build Coastguard Worker     if (mINotifyFD >= 0) {
275*4d7e907cSAndroid Build Coastguard Worker         close(mINotifyFD);
276*4d7e907cSAndroid Build Coastguard Worker     }
277*4d7e907cSAndroid Build Coastguard Worker }
278*4d7e907cSAndroid Build Coastguard Worker 
initialize()279*4d7e907cSAndroid Build Coastguard Worker bool ExternalCameraProvider::HotplugThread::initialize() {
280*4d7e907cSAndroid Build Coastguard Worker     // Update existing cameras
281*4d7e907cSAndroid Build Coastguard Worker     mParent->updateAttachedCameras();
282*4d7e907cSAndroid Build Coastguard Worker 
283*4d7e907cSAndroid Build Coastguard Worker     // Set up non-blocking fd. The threadLoop will be responsible for polling read at the
284*4d7e907cSAndroid Build Coastguard Worker     // desired frequency
285*4d7e907cSAndroid Build Coastguard Worker     mINotifyFD = inotify_init();
286*4d7e907cSAndroid Build Coastguard Worker     if (mINotifyFD < 0) {
287*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: inotify init failed! Exiting threadloop", __FUNCTION__);
288*4d7e907cSAndroid Build Coastguard Worker         return false;
289*4d7e907cSAndroid Build Coastguard Worker     }
290*4d7e907cSAndroid Build Coastguard Worker 
291*4d7e907cSAndroid Build Coastguard Worker     // Start watching /dev/ directory for created and deleted files
292*4d7e907cSAndroid Build Coastguard Worker     mWd = inotify_add_watch(mINotifyFD, kDevicePath, IN_CREATE | IN_DELETE);
293*4d7e907cSAndroid Build Coastguard Worker     if (mWd < 0) {
294*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: inotify add watch failed! Exiting threadloop", __FUNCTION__);
295*4d7e907cSAndroid Build Coastguard Worker         return false;
296*4d7e907cSAndroid Build Coastguard Worker     }
297*4d7e907cSAndroid Build Coastguard Worker 
298*4d7e907cSAndroid Build Coastguard Worker     mPollFd = {.fd = mINotifyFD, .events = POLLIN};
299*4d7e907cSAndroid Build Coastguard Worker 
300*4d7e907cSAndroid Build Coastguard Worker     mIsInitialized = true;
301*4d7e907cSAndroid Build Coastguard Worker     return true;
302*4d7e907cSAndroid Build Coastguard Worker }
303*4d7e907cSAndroid Build Coastguard Worker 
threadLoop()304*4d7e907cSAndroid Build Coastguard Worker bool ExternalCameraProvider::HotplugThread::threadLoop() {
305*4d7e907cSAndroid Build Coastguard Worker     // Initialize inotify descriptors if needed.
306*4d7e907cSAndroid Build Coastguard Worker     if (!mIsInitialized && !initialize()) {
307*4d7e907cSAndroid Build Coastguard Worker         return true;
308*4d7e907cSAndroid Build Coastguard Worker     }
309*4d7e907cSAndroid Build Coastguard Worker 
310*4d7e907cSAndroid Build Coastguard Worker     // poll /dev/* and handle timeouts and error
311*4d7e907cSAndroid Build Coastguard Worker     int pollRet = poll(&mPollFd, /* fd_count= */ 1, /* timeout= */ 250);
312*4d7e907cSAndroid Build Coastguard Worker     if (pollRet == 0) {
313*4d7e907cSAndroid Build Coastguard Worker         // no read event in 100ms
314*4d7e907cSAndroid Build Coastguard Worker         mPollFd.revents = 0;
315*4d7e907cSAndroid Build Coastguard Worker         return true;
316*4d7e907cSAndroid Build Coastguard Worker     } else if (pollRet < 0) {
317*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: error while polling for /dev/*: %d", __FUNCTION__, errno);
318*4d7e907cSAndroid Build Coastguard Worker         mPollFd.revents = 0;
319*4d7e907cSAndroid Build Coastguard Worker         return true;
320*4d7e907cSAndroid Build Coastguard Worker     } else if (mPollFd.revents & POLLERR) {
321*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: polling /dev/ returned POLLERR", __FUNCTION__);
322*4d7e907cSAndroid Build Coastguard Worker         mPollFd.revents = 0;
323*4d7e907cSAndroid Build Coastguard Worker         return true;
324*4d7e907cSAndroid Build Coastguard Worker     } else if (mPollFd.revents & POLLHUP) {
325*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: polling /dev/ returned POLLHUP", __FUNCTION__);
326*4d7e907cSAndroid Build Coastguard Worker         mPollFd.revents = 0;
327*4d7e907cSAndroid Build Coastguard Worker         return true;
328*4d7e907cSAndroid Build Coastguard Worker     } else if (mPollFd.revents & POLLNVAL) {
329*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: polling /dev/ returned POLLNVAL", __FUNCTION__);
330*4d7e907cSAndroid Build Coastguard Worker         mPollFd.revents = 0;
331*4d7e907cSAndroid Build Coastguard Worker         return true;
332*4d7e907cSAndroid Build Coastguard Worker     }
333*4d7e907cSAndroid Build Coastguard Worker     // mPollFd.revents must contain POLLIN, so safe to reset it before reading
334*4d7e907cSAndroid Build Coastguard Worker     mPollFd.revents = 0;
335*4d7e907cSAndroid Build Coastguard Worker 
336*4d7e907cSAndroid Build Coastguard Worker     uint64_t offset = 0;
337*4d7e907cSAndroid Build Coastguard Worker     ssize_t ret = read(mINotifyFD, mEventBuf, sizeof(mEventBuf));
338*4d7e907cSAndroid Build Coastguard Worker     if (ret < sizeof(struct inotify_event)) {
339*4d7e907cSAndroid Build Coastguard Worker         // invalid event. skip
340*4d7e907cSAndroid Build Coastguard Worker         return true;
341*4d7e907cSAndroid Build Coastguard Worker     }
342*4d7e907cSAndroid Build Coastguard Worker 
343*4d7e907cSAndroid Build Coastguard Worker     while (offset < ret) {
344*4d7e907cSAndroid Build Coastguard Worker         struct inotify_event* event = (struct inotify_event*)&mEventBuf[offset];
345*4d7e907cSAndroid Build Coastguard Worker         offset += sizeof(struct inotify_event) + event->len;
346*4d7e907cSAndroid Build Coastguard Worker 
347*4d7e907cSAndroid Build Coastguard Worker         if (event->wd != mWd) {
348*4d7e907cSAndroid Build Coastguard Worker             // event for an unrelated descriptor. ignore.
349*4d7e907cSAndroid Build Coastguard Worker             continue;
350*4d7e907cSAndroid Build Coastguard Worker         }
351*4d7e907cSAndroid Build Coastguard Worker 
352*4d7e907cSAndroid Build Coastguard Worker         ALOGV("%s inotify_event %s", __FUNCTION__, event->name);
353*4d7e907cSAndroid Build Coastguard Worker         if (strncmp(kPrefix, event->name, kPrefixLen) != 0) {
354*4d7e907cSAndroid Build Coastguard Worker             // event not for /dev/video*. ignore.
355*4d7e907cSAndroid Build Coastguard Worker             continue;
356*4d7e907cSAndroid Build Coastguard Worker         }
357*4d7e907cSAndroid Build Coastguard Worker 
358*4d7e907cSAndroid Build Coastguard Worker         std::string deviceId = event->name + kPrefixLen;
359*4d7e907cSAndroid Build Coastguard Worker         if (mInternalDevices.count(deviceId) != 0) {
360*4d7e907cSAndroid Build Coastguard Worker             // update to an internal device. ignore.
361*4d7e907cSAndroid Build Coastguard Worker             continue;
362*4d7e907cSAndroid Build Coastguard Worker         }
363*4d7e907cSAndroid Build Coastguard Worker 
364*4d7e907cSAndroid Build Coastguard Worker         char v4l2DevicePath[kMaxDevicePathLen];
365*4d7e907cSAndroid Build Coastguard Worker         snprintf(v4l2DevicePath, kMaxDevicePathLen, "%s%s", kDevicePath, event->name);
366*4d7e907cSAndroid Build Coastguard Worker 
367*4d7e907cSAndroid Build Coastguard Worker         if (event->mask & IN_CREATE) {
368*4d7e907cSAndroid Build Coastguard Worker             mParent->deviceAdded(v4l2DevicePath);
369*4d7e907cSAndroid Build Coastguard Worker         } else if (event->mask & IN_DELETE) {
370*4d7e907cSAndroid Build Coastguard Worker             mParent->deviceRemoved(v4l2DevicePath);
371*4d7e907cSAndroid Build Coastguard Worker         }
372*4d7e907cSAndroid Build Coastguard Worker     }
373*4d7e907cSAndroid Build Coastguard Worker     return true;
374*4d7e907cSAndroid Build Coastguard Worker }
375*4d7e907cSAndroid Build Coastguard Worker 
376*4d7e907cSAndroid Build Coastguard Worker // End ExternalCameraProvider::HotplugThread functions
377*4d7e907cSAndroid Build Coastguard Worker 
378*4d7e907cSAndroid Build Coastguard Worker }  // namespace implementation
379*4d7e907cSAndroid Build Coastguard Worker }  // namespace provider
380*4d7e907cSAndroid Build Coastguard Worker }  // namespace camera
381*4d7e907cSAndroid Build Coastguard Worker }  // namespace hardware
382*4d7e907cSAndroid Build Coastguard Worker }  // namespace android
383