/* SPDX-License-Identifier: GPL-2.0-or-later */ /* * Copyright (C) 2020, Umang Jain * * hotplug-cameras.cpp - Test cameraAdded/cameraRemoved signals in CameraManager */ #include #include #include #include #include #include #include #include #include #include #include #include "test.h" using namespace libcamera; using namespace std::chrono_literals; class HotplugTest : public Test { protected: void cameraAddedHandler([[maybe_unused]] std::shared_ptr cam) { cameraAdded_ = true; } void cameraRemovedHandler([[maybe_unused]] std::shared_ptr cam) { cameraRemoved_ = true; } int init() { if (!File::exists("/sys/module/uvcvideo")) { std::cout << "uvcvideo driver is not loaded, skipping" << std::endl; return TestSkip; } if (geteuid() != 0) { std::cout << "This test requires root permissions, skipping" << std::endl; return TestSkip; } cm_ = new CameraManager(); if (cm_->start()) { std::cout << "Failed to start camera manager" << std::endl; return TestFail; } cameraAdded_ = false; cameraRemoved_ = false; cm_->cameraAdded.connect(this, &HotplugTest::cameraAddedHandler); cm_->cameraRemoved.connect(this, &HotplugTest::cameraRemovedHandler); return 0; } int run() { DIR *dir; struct dirent *dirent; std::string uvcDeviceDir; dir = opendir(uvcDriverDir_.c_str()); /* Find a UVC device directory, which we can bind/unbind. */ while ((dirent = readdir(dir)) != nullptr) { if (!File::exists(uvcDriverDir_ + dirent->d_name + "/video4linux")) continue; uvcDeviceDir = dirent->d_name; break; } closedir(dir); /* If no UVC device found, skip the test. */ if (uvcDeviceDir.empty()) return TestSkip; /* Unbind a camera and process events. */ std::ofstream(uvcDriverDir_ + "unbind", std::ios::binary) << uvcDeviceDir; Timer timer; timer.start(1000ms); while (timer.isRunning() && !cameraRemoved_) Thread::current()->eventDispatcher()->processEvents(); if (!cameraRemoved_) { std::cout << "Camera unplug not detected" << std::endl; return TestFail; } /* Bind the camera again and process events. */ std::ofstream(uvcDriverDir_ + "bind", std::ios::binary) << uvcDeviceDir; timer.start(1000ms); while (timer.isRunning() && !cameraAdded_) Thread::current()->eventDispatcher()->processEvents(); if (!cameraAdded_) { std::cout << "Camera plug not detected" << std::endl; return TestFail; } return TestPass; } void cleanup() { cm_->stop(); delete cm_; } private: CameraManager *cm_; static const std::string uvcDriverDir_; bool cameraRemoved_; bool cameraAdded_; }; const std::string HotplugTest::uvcDriverDir_ = "/sys/bus/usb/drivers/uvcvideo/"; TEST_REGISTER(HotplugTest) -86fa7300fa91&id=d15f979ead99fd0b2acc7cbe5bd1375d7c6df706'>plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131