-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathDrmHotplugThread.cpp
65 lines (54 loc) · 1.59 KB
/
DrmHotplugThread.cpp
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
// SPDX-License-Identifier: Apache-2.0
// Copyright (C) 2019 Stephan Gerhold
#define LOG_TAG "drmfb-hotplug"
#include <cutils/uevent.h>
#include <android-base/unique_fd.h>
#include <android-base/logging.h>
#include "DrmHotplugThread.h"
#include "DrmDevice.h"
namespace android {
namespace hardware {
namespace graphics {
namespace composer {
namespace V2_1 {
namespace drmfb {
namespace {
constexpr int RECEIVE_BUFFER = 1 * 1024 * 1024; // 1 MiB
constexpr int MESSAGE_BUFFER = 1 * 1024; // 1 KiB
}
DrmHotplugThread::DrmHotplugThread(DrmDevice& device)
: GraphicsThread("drm-hotplug"), mDevice(device) {}
bool DrmHotplugThread::receiveEvent(int fd) {
char msg[MESSAGE_BUFFER];
auto n = uevent_kernel_multicast_recv(fd, msg, sizeof(msg));
if (n < 0)
return false;
bool drm = false;
bool hotplug = false;
for (auto buf = msg, end = msg + n; buf < end; buf += strlen(buf) + 1) {
if (!strcmp(buf, "DEVTYPE=drm_minor"))
drm = true;
else if (!strcmp(buf, "HOTPLUG=1"))
hotplug = true;
}
return drm && hotplug;
}
void DrmHotplugThread::work(std::unique_lock<std::mutex>& lock) {
base::unique_fd fd{uevent_open_socket(RECEIVE_BUFFER, true)};
if (fd < 0) {
PLOG(ERROR) << "Failed to open uevent socket";
return;
}
loop(lock, [this, &fd] {
if (receiveEvent(fd)) {
LOG(DEBUG) << "Received hotplug uevent";
mDevice.update();
}
});
}
} // namespace drmfb
} // namespace V2_1
} // namespace composer
} // namespace graphics
} // namespace hardware
} // namespace android