chiark / gitweb /
28d877749d7a6bf894206bc1331905d128eacbc0
[elogind.git] / udev / lib / libudev-queue.c
1 /*
2  * libudev - interface to udev device information
3  *
4  * Copyright (C) 2008 Kay Sievers <kay.sievers@vrfy.org>
5  *
6  * This program is free software: you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation, either version 2 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
18  */
19
20 #include <stdio.h>
21 #include <stdlib.h>
22 #include <stddef.h>
23 #include <unistd.h>
24 #include <errno.h>
25 #include <string.h>
26 #include <dirent.h>
27 #include <fcntl.h>
28 #include <sys/stat.h>
29
30 #include "libudev.h"
31 #include "libudev-private.h"
32
33 struct udev_queue {
34         struct udev *udev;
35         int refcount;
36         unsigned long long int last_seen_udev_seqnum;
37         struct udev_list_node queue_list;
38         struct udev_list_node failed_list;
39 };
40
41 struct udev_queue *udev_queue_new(struct udev *udev)
42 {
43         struct udev_queue *udev_queue;
44
45         if (udev == NULL)
46                 return NULL;
47
48         udev_queue = calloc(1, sizeof(struct udev_queue));
49         if (udev_queue == NULL)
50                 return NULL;
51         udev_queue->refcount = 1;
52         udev_queue->udev = udev;
53         udev_list_init(&udev_queue->queue_list);
54         udev_list_init(&udev_queue->failed_list);
55         return udev_queue;
56 }
57
58 struct udev_queue *udev_queue_ref(struct udev_queue *udev_queue)
59 {
60         if (udev_queue == NULL)
61                 return NULL;
62         udev_queue->refcount++;
63         return udev_queue;
64 }
65
66 void udev_queue_unref(struct udev_queue *udev_queue)
67 {
68         if (udev_queue == NULL)
69                 return;
70         udev_queue->refcount--;
71         if (udev_queue->refcount > 0)
72                 return;
73         udev_list_cleanup_entries(udev_queue->udev, &udev_queue->queue_list);
74         udev_list_cleanup_entries(udev_queue->udev, &udev_queue->failed_list);
75         free(udev_queue);
76 }
77
78 struct udev *udev_queue_get_udev(struct udev_queue *udev_queue)
79 {
80         if (udev_queue == NULL)
81                 return NULL;
82         return udev_queue->udev;
83 }
84
85 unsigned long long int udev_queue_get_kernel_seqnum(struct udev_queue *udev_queue)
86 {
87         char filename[UTIL_PATH_SIZE];
88         unsigned long long int seqnum;
89         int fd;
90         char buf[32];
91         ssize_t len;
92
93         if (udev_queue == NULL)
94                 return -EINVAL;
95         util_strlcpy(filename, udev_get_sys_path(udev_queue->udev), sizeof(filename));
96         util_strlcat(filename, "/kernel/uevent_seqnum", sizeof(filename));
97         fd = open(filename, O_RDONLY);
98         if (fd < 0)
99                 return 0;
100         len = read(fd, buf, sizeof(buf));
101         close(fd);
102         if (len <= 2)
103                 return 0;
104         buf[len-1] = '\0';
105         seqnum = strtoull(buf, NULL, 10);
106         dbg(udev_queue->udev, "seqnum=%llu\n", seqnum);
107         return seqnum;
108 }
109
110 unsigned long long int udev_queue_get_udev_seqnum(struct udev_queue *udev_queue)
111 {
112         char filename[UTIL_PATH_SIZE];
113         unsigned long long int seqnum;
114         int fd;
115         char buf[32];
116         ssize_t len;
117
118         if (udev_queue == NULL)
119                 return -EINVAL;
120         util_strlcpy(filename, udev_get_dev_path(udev_queue->udev), sizeof(filename));
121         util_strlcat(filename, "/.udev/uevent_seqnum", sizeof(filename));
122         fd = open(filename, O_RDONLY);
123         if (fd < 0)
124                 return 0;
125         len = read(fd, buf, sizeof(buf));
126         close(fd);
127         if (len <= 2)
128                 return 0;
129         buf[len-1] = '\0';
130         seqnum = strtoull(buf, NULL, 10);
131         dbg(udev_queue->udev, "seqnum=%llu\n", seqnum);
132         udev_queue->last_seen_udev_seqnum = seqnum;
133         return seqnum;
134 }
135
136 int udev_queue_get_queue_is_empty(struct udev_queue *udev_queue)
137 {
138         char queuename[UTIL_PATH_SIZE];
139         struct stat statbuf;
140         unsigned long long int seqnum_kernel;
141
142         if (udev_queue == NULL)
143                 return -EINVAL;
144         util_strlcpy(queuename, udev_get_dev_path(udev_queue->udev), sizeof(queuename));
145         util_strlcat(queuename, "/.udev/queue", sizeof(queuename));
146         if (stat(queuename, &statbuf) == 0) {
147                 dbg(udev_queue->udev, "queue is not empty\n");
148                 return 0;
149         }
150         seqnum_kernel = udev_queue_get_kernel_seqnum(udev_queue);
151         if (seqnum_kernel <= udev_queue->last_seen_udev_seqnum) {
152                 dbg(udev_queue->udev, "queue is empty\n");
153                 return 1;
154         }
155         udev_queue_get_udev_seqnum(udev_queue);
156         if (seqnum_kernel <= udev_queue->last_seen_udev_seqnum) {
157                 dbg(udev_queue->udev, "queue is empty\n");
158                 return 1;
159         }
160         dbg(udev_queue->udev, "queue is empty, but kernel events still pending [%llu]<->[%llu]\n",
161              seqnum_kernel, udev_queue->last_seen_udev_seqnum);
162         return 0;
163 }
164
165 int udev_queue_get_seqnum_is_finished(struct udev_queue *udev_queue, unsigned long long int seqnum)
166 {
167         char filename[UTIL_PATH_SIZE];
168         struct stat statbuf;
169
170         if (udev_queue == NULL)
171                 return -EINVAL;
172         /* if we have not seen this seqnum, check if it is/was already queued */
173         if (seqnum < udev_queue->last_seen_udev_seqnum) {
174                 udev_queue_get_udev_seqnum(udev_queue);
175                 if (seqnum < udev_queue->last_seen_udev_seqnum)
176                         return 0;
177         }
178         snprintf(filename, sizeof(filename), "%s/.udev/queue/%llu",
179                  udev_get_dev_path(udev_queue->udev), seqnum);
180         if (stat(filename, &statbuf) == 0)
181                 return 0;
182         dbg(udev_queue->udev, "seqnum: %llu finished\n", seqnum);
183         return 1;
184 }
185
186 struct udev_list_entry *udev_queue_get_queued_list_entry(struct udev_queue *udev_queue)
187 {
188         char path[UTIL_PATH_SIZE];
189         DIR *dir;
190         struct dirent *dent;
191
192         if (udev_queue == NULL)
193                 return NULL;
194         udev_list_cleanup_entries(udev_queue->udev, &udev_queue->queue_list);
195         util_strlcpy(path, udev_get_dev_path(udev_queue->udev), sizeof(path));
196         util_strlcat(path, "/.udev/queue", sizeof(path));
197         dir = opendir(path);
198         if (dir == NULL)
199                 return NULL;
200         for (dent = readdir(dir); dent != NULL; dent = readdir(dir)) {
201                 char filename[UTIL_PATH_SIZE];
202                 char syspath[UTIL_PATH_SIZE];
203                 size_t syslen;
204                 ssize_t len;
205
206                 if (dent->d_name[0] == '.')
207                         continue;
208                 util_strlcpy(filename, path, sizeof(filename));
209                 util_strlcat(filename, "/", sizeof(filename));
210                 util_strlcat(filename, dent->d_name, sizeof(filename));
211
212                 syslen = util_strlcpy(syspath, udev_get_sys_path(udev_queue->udev), sizeof(syspath));
213                 len = readlink(filename, &syspath[syslen], sizeof(syspath)-syslen);
214                 if (len < 0 || len >= (ssize_t)(sizeof(syspath)-syslen))
215                         continue;
216                 syspath[syslen + len] = '\0';
217                 dbg(udev_queue->udev, "found '%s' [%s]\n", syspath, dent->d_name);
218                 udev_list_entry_add(udev_queue->udev, &udev_queue->queue_list, syspath, dent->d_name, 0, 0);
219         }
220         closedir(dir);
221         return udev_list_get_entry(&udev_queue->queue_list);
222 }
223
224 struct udev_list_entry *udev_queue_get_failed_list_entry(struct udev_queue *udev_queue)
225 {
226         char path[UTIL_PATH_SIZE];
227         DIR *dir;
228         struct dirent *dent;
229
230         if (udev_queue == NULL)
231                 return NULL;
232         udev_list_cleanup_entries(udev_queue->udev, &udev_queue->failed_list);
233         util_strlcpy(path, udev_get_dev_path(udev_queue->udev), sizeof(path));
234         util_strlcat(path, "/.udev/failed", sizeof(path));
235         dir = opendir(path);
236         if (dir == NULL)
237                 return NULL;
238         for (dent = readdir(dir); dent != NULL; dent = readdir(dir)) {
239                 char filename[UTIL_PATH_SIZE];
240                 char syspath[UTIL_PATH_SIZE];
241                 struct stat statbuf;
242                 size_t syslen;
243                 ssize_t len;
244
245                 if (dent->d_name[0] == '.')
246                         continue;
247                 util_strlcpy(filename, path, sizeof(filename));
248                 util_strlcat(filename, "/", sizeof(filename));
249                 util_strlcat(filename, dent->d_name, sizeof(filename));
250
251                 syslen = util_strlcpy(syspath, udev_get_sys_path(udev_queue->udev), sizeof(syspath));
252                 len = readlink(filename, &syspath[syslen], sizeof(syspath)-syslen);
253                 if (len < 0 || len >= (ssize_t)(sizeof(syspath)-syslen))
254                         continue;
255                 syspath[syslen + len] = '\0';
256                 dbg(udev_queue->udev, "found '%s' [%s]\n", syspath, dent->d_name);
257                 util_strlcpy(filename, syspath, sizeof(filename));
258                 util_strlcat(filename, "/uevent", sizeof(filename));
259                 if (stat(filename, &statbuf) != 0)
260                         continue;
261                 udev_list_entry_add(udev_queue->udev, &udev_queue->failed_list, syspath, NULL, 0, 0);
262         }
263         closedir(dir);
264         return udev_list_get_entry(&udev_queue->failed_list);
265 }
266
267 int udev_queue_export_udev_seqnum(struct udev_queue *udev_queue, unsigned long long int seqnum)
268 {
269         return -1;
270 }
271
272 extern int udev_queue_export_device_queued(struct udev_queue *udev_queue, struct udev_device *udev_device)
273 {
274         return -1;
275 }
276
277 extern int udev_queue_export_device_finished(struct udev_queue *udev_queue, struct udev_device *udev_device)
278 {
279         return -1;
280 }
281
282 extern int udev_queue_export_device_failed(struct udev_queue *udev_queue, struct udev_device *udev_device)
283 {
284         return -1;
285 }