blob: 6a69f5ab137ec880f7cd26ba79ffd8e3fd907c10 [file] [log] [blame]
Jon Medhurst15ce78d2014-04-10 09:02:02 +01001/**
Jon Medhurstb1d07442015-05-08 12:04:18 +01002 * Copyright (C) ARM Limited 2013-2015. All rights reserved.
Jon Medhurst15ce78d2014-04-10 09:02:02 +01003 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 as
6 * published by the Free Software Foundation.
7 */
8
9#include "UEvent.h"
10
Jon Medhurst15ce78d2014-04-10 09:02:02 +010011#include <string.h>
12#include <sys/socket.h>
13#include <unistd.h>
14
Jon Medhurste31266f2014-08-04 15:47:44 +010015#include <linux/netlink.h>
16
Jon Medhurst15ce78d2014-04-10 09:02:02 +010017#include "Logging.h"
Jon Medhurst96b56152014-10-30 18:01:15 +000018#include "OlySocket.h"
Jon Medhurst15ce78d2014-04-10 09:02:02 +010019
20static const char EMPTY[] = "";
21static const char ACTION[] = "ACTION=";
22static const char DEVPATH[] = "DEVPATH=";
23static const char SUBSYSTEM[] = "SUBSYSTEM=";
24
25UEvent::UEvent() : mFd(-1) {
26}
27
28UEvent::~UEvent() {
29 if (mFd >= 0) {
30 close(mFd);
31 }
32}
33
34bool UEvent::init() {
Jon Medhurst96b56152014-10-30 18:01:15 +000035 mFd = socket_cloexec(PF_NETLINK, SOCK_RAW, NETLINK_KOBJECT_UEVENT);
Jon Medhurst15ce78d2014-04-10 09:02:02 +010036 if (mFd < 0) {
Jon Medhurstb1d07442015-05-08 12:04:18 +010037 logg->logMessage("socket failed");
Jon Medhurst15ce78d2014-04-10 09:02:02 +010038 return false;
39 }
40
41 struct sockaddr_nl sockaddr;
42 memset(&sockaddr, 0, sizeof(sockaddr));
43 sockaddr.nl_family = AF_NETLINK;
44 sockaddr.nl_groups = 1; // bitmask: (1 << 0) == kernel events, (1 << 1) == udev events
45 sockaddr.nl_pid = 0;
46 if (bind(mFd, (struct sockaddr *)&sockaddr, sizeof(sockaddr)) != 0) {
Jon Medhurstb1d07442015-05-08 12:04:18 +010047 logg->logMessage("bind failed");
Jon Medhurst15ce78d2014-04-10 09:02:02 +010048 return false;
49 }
50
51 return true;
52}
53
54bool UEvent::read(UEventResult *const result) {
55 ssize_t bytes = recv(mFd, result->mBuf, sizeof(result->mBuf), 0);
56 if (bytes <= 0) {
Jon Medhurstb1d07442015-05-08 12:04:18 +010057 logg->logMessage("recv failed");
Jon Medhurst15ce78d2014-04-10 09:02:02 +010058 return false;
59 }
60
61 result->mAction = EMPTY;
62 result->mDevPath = EMPTY;
63 result->mSubsystem = EMPTY;
64
65 for (int pos = 0; pos < bytes; pos += strlen(result->mBuf + pos) + 1) {
66 char *const str = result->mBuf + pos;
Jon Medhurstb1d07442015-05-08 12:04:18 +010067 logg->logMessage("uevent + %i: %s", pos, str);
Jon Medhurst15ce78d2014-04-10 09:02:02 +010068 if (strncmp(str, ACTION, sizeof(ACTION) - 1) == 0) {
69 result->mAction = str + sizeof(ACTION) - 1;
70 } else if (strncmp(str, DEVPATH, sizeof(DEVPATH) - 1) == 0) {
71 result->mDevPath = str + sizeof(DEVPATH) - 1;
72 } else if (strncmp(str, SUBSYSTEM, sizeof(SUBSYSTEM) - 1) == 0) {
73 result->mSubsystem = str + sizeof(SUBSYSTEM) - 1;
74 }
75 }
76
77 return true;
78}