chiark / gitweb /
libudev: enumerate - split new() and scan()
[elogind.git] / udev / lib / test-libudev.c
1 /*
2  * test-libudev
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 <stdarg.h>
22 #include <unistd.h>
23 #include <errno.h>
24 #include <string.h>
25 #include <getopt.h>
26 #include <syslog.h>
27 #include <fcntl.h>
28 #include <sys/select.h>
29
30 #include "libudev.h"
31
32 static void log_fn(struct udev *udev,
33                    int priority, const char *file, int line, const char *fn,
34                    const char *format, va_list args)
35 {
36         printf("test-libudev: %s %s:%d ", fn, file, line);
37         vprintf(format, args);
38 }
39
40 static void print_device(struct udev_device *device)
41 {
42         const char *str;
43         dev_t devnum;
44         int count;
45         struct udev_list_entry *list_entry;
46
47         printf("*** device: %p ***\n", device);
48         str = udev_device_get_action(device);
49         if (str != NULL)
50                 printf("action:    '%s'\n", str);
51
52         str = udev_device_get_syspath(device);
53         printf("syspath:   '%s'\n", str);
54
55         str = udev_device_get_devpath(device);
56         printf("devpath:   '%s'\n", str);
57
58         str = udev_device_get_subsystem(device);
59         if (str != NULL)
60                 printf("subsystem: '%s'\n", str);
61
62         str = udev_device_get_driver(device);
63         if (str != NULL)
64                 printf("driver:    '%s'\n", str);
65
66         str = udev_device_get_devnode(device);
67         if (str != NULL)
68                 printf("devname:   '%s'\n", str);
69
70         devnum = udev_device_get_devnum(device);
71         if (major(devnum) > 0)
72                 printf("devnum:    %u:%u\n", major(devnum), minor(devnum));
73
74         count = 0;
75         list_entry = udev_device_get_devlinks_list_entry(device);
76         while (list_entry != NULL) {
77                 printf("link:      '%s'\n", udev_list_entry_get_name(list_entry));
78                 count++;
79                 list_entry = udev_list_entry_get_next(list_entry);
80         }
81         if (count > 0)
82                 printf("found %i links\n", count);
83
84         count = 0;
85         list_entry = udev_device_get_properties_list_entry(device);
86         while (list_entry != NULL) {
87                 printf("property:  '%s=%s'\n",
88                        udev_list_entry_get_name(list_entry),
89                        udev_list_entry_get_value(list_entry));
90                 count++;
91                 list_entry = udev_list_entry_get_next(list_entry);
92         }
93         if (count > 0)
94                 printf("found %i properties\n", count);
95
96         str = udev_device_get_attr_value(device, "dev");
97         if (str != NULL)
98                 printf("attr{dev}: '%s'\n", str);
99
100         printf("\n");
101 }
102
103 static int test_device(struct udev *udev, const char *syspath)
104 {
105         struct udev_device *device;
106
107         printf("looking at device: %s\n", syspath);
108         device = udev_device_new_from_syspath(udev, syspath);
109         if (device == NULL) {
110                 printf("no device\n");
111                 return -1;
112         }
113         print_device(device);
114         udev_device_unref(device);
115         return 0;
116 }
117
118 static int test_device_parents(struct udev *udev, const char *syspath)
119 {
120         struct udev_device *device;
121         struct udev_device *device_parent;
122
123         printf("looking at device: %s\n", syspath);
124         device = udev_device_new_from_syspath(udev, syspath);
125         if (device == NULL)
126                 return -1;
127
128         printf("looking at parents\n");
129         device_parent = device;
130         do {
131                 print_device(device_parent);
132                 device_parent = udev_device_get_parent(device_parent);
133         } while (device_parent != NULL);
134
135         printf("looking at parents again\n");
136         device_parent = device;
137         do {
138                 print_device(device_parent);
139                 device_parent = udev_device_get_parent(device_parent);
140         } while (device_parent != NULL);
141         udev_device_unref(device);
142
143         return 0;
144 }
145
146 static int test_device_devnum(struct udev *udev)
147 {
148         dev_t devnum = makedev(1, 3);
149         struct udev_device *device;
150
151         printf("looking up device: %u:%u\n", major(devnum), minor(devnum));
152         device = udev_device_new_from_devnum(udev, 'c', devnum);
153         if (device == NULL)
154                 return -1;
155         print_device(device);
156         udev_device_unref(device);
157         return 0;
158 }
159
160 static int test_enumerate_print_list(struct udev_enumerate *enumerate)
161 {
162         struct udev_list_entry *list_entry;
163         int count = 0;
164
165         udev_list_entry_foreach(list_entry, udev_enumerate_get_list_entry(enumerate)) {
166                 struct udev_device *device;
167
168                 device = udev_device_new_from_syspath(udev_enumerate_get_udev(enumerate),
169                                                       udev_list_entry_get_name(list_entry));
170                 if (device != NULL) {
171                         printf("device:    '%s' (%s) '%s'\n",
172                                udev_device_get_syspath(device),
173                                udev_device_get_subsystem(device),
174                                udev_device_get_sysname(device));
175                         udev_device_unref(device);
176                         count++;
177                 }
178         }
179         printf("found %i devices\n\n", count);
180         return count;
181 }
182
183 static int test_monitor(struct udev *udev, const char *socket_path)
184 {
185         struct udev_monitor *udev_monitor;
186         fd_set readfds;
187         int fd;
188
189         udev_monitor = udev_monitor_new_from_socket(udev, socket_path);
190         if (udev_monitor == NULL) {
191                 printf("no socket\n");
192                 return -1;
193         }
194         if (udev_monitor_enable_receiving(udev_monitor) < 0) {
195                 printf("bind failed\n");
196                 return -1;
197         }
198
199         fd = udev_monitor_get_fd(udev_monitor);
200         FD_ZERO(&readfds);
201
202         while (1) {
203                 struct udev_device *device;
204                 int fdcount;
205
206                 FD_SET(STDIN_FILENO, &readfds);
207                 FD_SET(fd, &readfds);
208
209                 printf("waiting for events on %s, press ENTER to exit\n", socket_path);
210                 fdcount = select(fd+1, &readfds, NULL, NULL, NULL);
211                 printf("select fd count: %i\n", fdcount);
212
213                 if (FD_ISSET(fd, &readfds)) {
214                         device = udev_monitor_receive_device(udev_monitor);
215                         if (device == NULL) {
216                                 printf("no device from socket\n");
217                                 continue;
218                         }
219                         print_device(device);
220                         udev_device_unref(device);
221                 }
222
223                 if (FD_ISSET(STDIN_FILENO, &readfds)) {
224                         printf("exiting loop\n");
225                         break;
226                 }
227         }
228
229         udev_monitor_unref(udev_monitor);
230         return 0;
231 }
232
233 int main(int argc, char *argv[], char *envp[])
234 {
235         struct udev *udev = NULL;
236         static const struct option options[] = {
237                 { "syspath", 1, NULL, 'p' },
238                 { "subsystem", 1, NULL, 's' },
239                 { "socket", 1, NULL, 'S' },
240                 { "debug", 0, NULL, 'd' },
241                 { "help", 0, NULL, 'h' },
242                 { "version", 0, NULL, 'V' },
243                 {}
244         };
245         struct udev_enumerate *udev_enumerate;
246         const char *syspath = "/devices/virtual/mem/null";
247         const char *subsystem = NULL;
248         const char *socket = "@/org/kernel/udev/monitor";
249         char path[1024];
250         const char *str;
251
252         udev = udev_new();
253         printf("context: %p\n", udev);
254         if (udev == NULL) {
255                 printf("no context\n");
256                 return 1;
257         }
258         udev_set_log_fn(udev, log_fn);
259         printf("set log: %p\n", log_fn);
260
261         while (1) {
262                 int option;
263
264                 option = getopt_long(argc, argv, "+dhV", options, NULL);
265                 if (option == -1)
266                         break;
267
268                 switch (option) {
269                 case 'p':
270                         syspath = optarg;
271                         break;
272                 case 's':
273                         subsystem = optarg;
274                         break;
275                 case 'S':
276                         socket = optarg;
277                         break;
278                 case 'd':
279                         if (udev_get_log_priority(udev) < LOG_INFO)
280                                 udev_set_log_priority(udev, LOG_INFO);
281                         break;
282                 case 'h':
283                         printf("--debug --syspath= --subsystem= --socket= --help\n");
284                         goto out;
285                 case 'V':
286                         printf("%s\n", VERSION);
287                         goto out;
288                 default:
289                         goto out;
290                 }
291         }
292
293         str = udev_get_sys_path(udev);
294         printf("sys_path: '%s'\n", str);
295         str = udev_get_dev_path(udev);
296         printf("dev_path: '%s'\n", str);
297
298         /* add sys path if needed */
299         if (strncmp(syspath, udev_get_sys_path(udev), strlen(udev_get_sys_path(udev))) != 0) {
300                 snprintf(path, sizeof(path), "%s%s", udev_get_sys_path(udev), syspath);
301                 syspath = path;
302         }
303
304         test_device(udev, syspath);
305         test_device_devnum(udev);
306         test_device_parents(udev, syspath);
307
308         printf("enumerate '%s'\n", subsystem == NULL ? "<all>" : subsystem);
309         udev_enumerate = udev_enumerate_new(udev);
310         if (udev_enumerate == NULL)
311                 return -1;
312         udev_enumerate_scan_devices(udev_enumerate, subsystem, NULL);
313         test_enumerate_print_list(udev_enumerate);
314         udev_enumerate_unref(udev_enumerate);
315
316         printf("enumerate 'block'\n");
317         udev_enumerate = udev_enumerate_new(udev);
318         if (udev_enumerate == NULL)
319                 return -1;
320         udev_enumerate_scan_devices(udev_enumerate, "block", NULL);
321         test_enumerate_print_list(udev_enumerate);
322         udev_enumerate_unref(udev_enumerate);
323
324         printf("enumerate '!block'\n");
325         udev_enumerate = udev_enumerate_new(udev);
326         if (udev_enumerate == NULL)
327                 return -1;
328         udev_enumerate_scan_devices(udev_enumerate, "!block", NULL);
329         test_enumerate_print_list(udev_enumerate);
330         udev_enumerate_unref(udev_enumerate);
331
332         printf("enumerate 'pci, mem, vc'\n");
333         udev_enumerate = udev_enumerate_new(udev);
334         if (udev_enumerate == NULL)
335                 return -1;
336         udev_enumerate_scan_devices(udev_enumerate, "pci", "mem", "vc", NULL);
337         test_enumerate_print_list(udev_enumerate);
338         udev_enumerate_unref(udev_enumerate);
339
340         printf("enumerate 'subsystem'\n");
341         udev_enumerate = udev_enumerate_new(udev);
342         if (udev_enumerate == NULL)
343                 return -1;
344         udev_enumerate_scan_subsystems(udev_enumerate);
345         test_enumerate_print_list(udev_enumerate);
346         udev_enumerate_unref(udev_enumerate);
347
348         test_monitor(udev, socket);
349 out:
350         udev_unref(udev);
351         return 0;
352 }