1 /*
2 // Copyright (c) 2014 Intel Corporation
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 #include <poll.h>
17 #include <sys/socket.h>
18 #include <sys/un.h>
19 #include <sys/queue.h>
20 #include <linux/netlink.h>
21 #include <sys/types.h>
22 #include <unistd.h>
23 #include <DrmConfig.h>
24 #include <common/utils/HwcTrace.h>
25 #include <UeventObserver.h>
26
27 namespace android {
28 namespace intel {
29
UeventObserver()30 UeventObserver::UeventObserver()
31 : mUeventFd(-1),
32 mExitRDFd(-1),
33 mExitWDFd(-1),
34 mListeners()
35 {
36 }
37
~UeventObserver()38 UeventObserver::~UeventObserver()
39 {
40 deinitialize();
41 }
42
initialize()43 bool UeventObserver::initialize()
44 {
45 mListeners.clear();
46
47 if (mUeventFd != -1) {
48 return true;
49 }
50
51 mThread = new UeventObserverThread(this);
52 if (!mThread.get()) {
53 ELOGTRACE("failed to create uevent observer thread");
54 return false;
55 }
56
57 // init uevent socket
58 struct sockaddr_nl addr;
59 // set the socket receive buffer to 64K
60 // NOTE: this is only called for once
61 int sz = 64 * 1024;
62
63 memset(&addr, 0, sizeof(addr));
64 addr.nl_family = AF_NETLINK;
65 addr.nl_pid = pthread_self() | getpid();
66 addr.nl_groups = 0xffffffff;
67
68 mUeventFd = socket(PF_NETLINK, SOCK_DGRAM, NETLINK_KOBJECT_UEVENT);
69 if (mUeventFd < 0) {
70 DEINIT_AND_RETURN_FALSE("failed to create uevent socket");
71 }
72
73 if (setsockopt(mUeventFd, SOL_SOCKET, SO_RCVBUFFORCE, &sz, sizeof(sz))) {
74 WLOGTRACE("setsockopt() failed");
75 //return false;
76 }
77
78 if (bind(mUeventFd, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
79 DEINIT_AND_RETURN_FALSE("failed to bind scoket");
80 return false;
81 }
82
83 memset(mUeventMessage, 0, UEVENT_MSG_LEN);
84
85 int exitFds[2];
86 if (pipe(exitFds) < 0) {
87 ELOGTRACE("failed to make pipe");
88 deinitialize();
89 return false;
90 }
91 mExitRDFd = exitFds[0];
92 mExitWDFd = exitFds[1];
93
94 return true;
95 }
96
deinitialize()97 void UeventObserver::deinitialize()
98 {
99 if (mUeventFd != -1) {
100 if (mExitWDFd != -1) {
101 close(mExitWDFd);
102 mExitWDFd = -1;
103 }
104 close(mUeventFd);
105 mUeventFd = -1;
106 }
107
108 if (mThread.get()) {
109 mThread->requestExitAndWait();
110 mThread = NULL;
111 }
112
113 while (!mListeners.isEmpty()) {
114 UeventListener *listener = mListeners.valueAt(0);
115 mListeners.removeItemsAt(0);
116 delete listener;
117 }
118 }
119
start()120 void UeventObserver::start()
121 {
122 if (mThread.get()) {
123 mThread->run("UeventObserver", PRIORITY_URGENT_DISPLAY);
124 }
125 }
126
127
registerListener(const char * event,UeventListenerFunc func,void * data)128 void UeventObserver::registerListener(const char *event, UeventListenerFunc func, void *data)
129 {
130 if (!event || !func) {
131 ELOGTRACE("invalid event string or listener to register");
132 return;
133 }
134
135 String8 key(event);
136 if (mListeners.indexOfKey(key) >= 0) {
137 ELOGTRACE("listener for uevent %s exists", event);
138 return;
139 }
140
141 UeventListener *listener = new UeventListener;
142 if (!listener) {
143 ELOGTRACE("failed to create Uevent Listener");
144 return;
145 }
146 listener->func = func;
147 listener->data = data;
148
149 mListeners.add(key, listener);
150 }
151
threadLoop()152 bool UeventObserver::threadLoop()
153 {
154 if (mUeventFd == -1) {
155 ELOGTRACE("invalid uEvent file descriptor");
156 return false;
157 }
158
159 struct pollfd fds[2];
160 int nr;
161
162 fds[0].fd = mUeventFd;
163 fds[0].events = POLLIN;
164 fds[0].revents = 0;
165 fds[1].fd = mExitRDFd;
166 fds[1].events = POLLIN;
167 fds[1].revents = 0;
168 nr = poll(fds, 2, -1);
169
170 if (nr > 0 && fds[0].revents == POLLIN) {
171 int count = recv(mUeventFd, mUeventMessage, UEVENT_MSG_LEN - 2, 0);
172 if (count > 0) {
173 onUevent();
174 }
175 } else if (fds[1].revents) {
176 close(mExitRDFd);
177 mExitRDFd = -1;
178 ILOGTRACE("exiting wait");
179 return false;
180 }
181 // always looping
182 return true;
183 }
184
onUevent()185 void UeventObserver::onUevent()
186 {
187 char *msg = mUeventMessage;
188 const char *envelope = DrmConfig::getUeventEnvelope();
189 if (strncmp(msg, envelope, strlen(envelope)) != 0)
190 return;
191
192 msg += strlen(msg) + 1;
193
194 UeventListener *listener;
195 String8 key;
196 while (*msg) {
197 key = String8(msg);
198 if (mListeners.indexOfKey(key) >= 0) {
199 DLOGTRACE("received Uevent: %s", msg);
200 listener = mListeners.valueFor(key);
201 if (listener) {
202 listener->func(listener->data);
203 } else {
204 ELOGTRACE("no listener for uevent %s", msg);
205 }
206 }
207 msg += strlen(msg) + 1;
208 }
209 }
210
211 } // namespace intel
212 } // namespace android
213
214