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