chiark / gitweb /
fixups to get back to proper patch order
[elogind.git] / test / udev-test.pl
1 #!/usr/bin/perl
2
3 # udev-test
4 #
5 # Provides automated testing of the udev binary.
6 # The whole test is self contained in this file, except the matching sysfs tree.
7 # Simply extend the @tests array, to add a new test variant.
8 #
9 # Every test is driven by its own temporary config file.
10 # This program prepares the environment, creates the config and calls udev.
11 #
12 # udev reads the config, looks at the provided sysfs and
13 # first creates and then removes the device node.
14 # After creation and removal the result is checked against the
15 # expected value and the result is printed.
16 #
17 # happy testing,
18 # Kay Sievers <kay.sievers@vrfy.org>, 2003
19 #
20 # Modified April 9, 2004 by Leann Ogasawara <ogasawara@osdl.org>
21 #  - expanded @tests array to add more symlinks and permissions tests
22 #  - some of the symlinks tests also test lack of node creation
23 #  - added symlink_test() function
24 #  - moved permissions and major_minor tests into their own functions
25
26 use warnings;
27 use strict;
28
29 my $PWD = $ENV{PWD};
30 my $sysfs     = "sys/";
31 my $udev_bin  = "../udev";
32 my $udev_root = "udev-root/"; # !!! directory will be removed !!!
33 my $udev_db   = ".udevdb";
34 my $perm      = "udev.permissions";
35 my $main_conf = "udev-test.conf";
36 my $conf_tmp  = "udev-test.rules";
37
38
39 my @tests = (
40         {
41                 desc            => "label test of scsi disc",
42                 subsys          => "block",
43                 devpath         => "/block/sda",
44                 exp_name        => "boot_disk" ,
45                 conf            => <<EOF
46 BUS="scsi", SYSFS{vendor}="IBM-ESXS", NAME="boot_disk%n"
47 KERNEL="ttyUSB0", NAME="visor"
48 EOF
49         },
50         {
51                 desc            => "label test of scsi partition",
52                 subsys          => "block",
53                 devpath         => "/block/sda/sda1",
54                 exp_name        => "boot_disk1" ,
55                 conf            => <<EOF
56 BUS="scsi", SYSFS{vendor}="IBM-ESXS", NAME="boot_disk%n"
57 EOF
58         },
59         {
60                 desc            => "label test of pattern match",
61                 subsys          => "block",
62                 devpath         => "/block/sda/sda1",
63                 exp_name        => "boot_disk1" ,
64                 conf            => <<EOF
65 BUS="scsi", SYSFS{vendor}="?IBM-ESXS", NAME="boot_disk%n-1"
66 BUS="scsi", SYSFS{vendor}="IBM-ESXS?", NAME="boot_disk%n-2"
67 BUS="scsi", SYSFS{vendor}="IBM-ES??", NAME="boot_disk%n"
68 BUS="scsi", SYSFS{vendor}="IBM-ESXSS", NAME="boot_disk%n-3"
69 EOF
70         },
71         {
72                 desc            => "label test of multiple sysfs files",
73                 subsys          => "block",
74                 devpath         => "/block/sda/sda1",
75                 exp_name        => "boot_disk1" ,
76                 conf            => <<EOF
77 BUS="scsi", SYSFS{vendor}="IBM-ESXS", SYSFS{model}="ST336605LW   !#", NAME="boot_diskX%n"
78 BUS="scsi", SYSFS{vendor}="IBM-ESXS", SYSFS{model}="ST336605LW    !#", NAME="boot_disk%n"
79 EOF
80         },
81         {
82                 desc            => "label test of max sysfs files",
83                 subsys          => "block",
84                 devpath         => "/block/sda/sda1",
85                 exp_name        => "boot_disk1" ,
86                 conf            => <<EOF
87 BUS="scsi", SYSFS{vendor}="IBM-ESXS", SYSFS{model}="ST336605LW    !#", SYSFS{scsi_level}="4", SYSFS{rev}="B245", SYSFS{type}="2", SYSFS{queue_depth}="32", NAME="boot_diskXX%n"
88 BUS="scsi", SYSFS{vendor}="IBM-ESXS", SYSFS{model}="ST336605LW    !#", SYSFS{scsi_level}="4", SYSFS{rev}="B245", SYSFS{type}="0", NAME="boot_disk%n"
89 EOF
90         },
91         {
92                 desc            => "catch device by *",
93                 subsys          => "tty",
94                 devpath         => "/class/tty/ttyUSB0",
95                 exp_name        => "visor/0" ,
96                 conf            => <<EOF
97 KERNEL="ttyUSB*", NAME="visor/%n"
98 EOF
99         },
100         {
101                 desc            => "catch device by * - take 2",
102                 subsys          => "tty",
103                 devpath         => "/class/tty/ttyUSB0",
104                 exp_name        => "visor/0" ,
105                 conf            => <<EOF
106 KERNEL="*USB1", NAME="bad"
107 KERNEL="*USB0", NAME="visor/%n"
108 EOF
109         },
110         {
111                 desc            => "catch device by ?",
112                 subsys          => "tty",
113                 devpath         => "/class/tty/ttyUSB0",
114                 exp_name        => "visor/0" ,
115                 conf            => <<EOF
116 KERNEL="ttyUSB??*", NAME="visor/%n-1"
117 KERNEL="ttyUSB??", NAME="visor/%n-2"
118 KERNEL="ttyUSB?", NAME="visor/%n"
119 EOF
120         },
121         {
122                 desc            => "catch device by character class",
123                 subsys          => "tty",
124                 devpath         => "/class/tty/ttyUSB0",
125                 exp_name        => "visor/0" ,
126                 conf            => <<EOF
127 KERNEL="ttyUSB[A-Z]*", NAME="visor/%n-1"
128 KERNEL="ttyUSB?[0-9]", NAME="visor/%n-2"
129 KERNEL="ttyUSB[0-9]*", NAME="visor/%n"
130 EOF
131         },
132         {
133                 desc            => "replace kernel name",
134                 subsys          => "tty",
135                 devpath         => "/class/tty/ttyUSB0",
136                 exp_name        => "visor" ,
137                 conf            => <<EOF
138 KERNEL="ttyUSB0", NAME="visor"
139 EOF
140         },
141         {
142                 desc            => "Handle comment lines in config file (and replace kernel name)",
143                 subsys          => "tty",
144                 devpath         => "/class/tty/ttyUSB0",
145                 exp_name        => "visor" ,
146                 conf            => <<EOF
147 # this is a comment
148 KERNEL="ttyUSB0", NAME="visor"
149
150 EOF
151         },
152         {
153                 desc            => "Handle comment lines in config file with whitespace (and replace kernel name)",
154                 subsys          => "tty",
155                 devpath         => "/class/tty/ttyUSB0",
156                 exp_name        => "visor" ,
157                 conf            => <<EOF
158  # this is a comment with whitespace before the comment 
159 KERNEL="ttyUSB0", NAME="visor"
160
161 EOF
162         },
163         {
164                 desc            => "Handle whitespace only lines (and replace kernel name)",
165                 subsys          => "tty",
166                 devpath         => "/class/tty/ttyUSB0",
167                 exp_name        => "whitespace" ,
168                 conf            => <<EOF
169
170  
171
172  # this is a comment with whitespace before the comment 
173 KERNEL="ttyUSB0", NAME="whitespace"
174
175  
176
177 EOF
178         },
179         {
180                 desc            => "Handle empty lines in config file (and replace kernel name)",
181                 subsys          => "tty",
182                 devpath         => "/class/tty/ttyUSB0",
183                 exp_name        => "visor" ,
184                 conf            => <<EOF
185
186 KERNEL="ttyUSB0", NAME="visor"
187
188 EOF
189         },
190         {
191                 desc            => "subdirectory handling",
192                 subsys          => "tty",
193                 devpath         => "/class/tty/ttyUSB0",
194                 exp_name        => "sub/direct/ory/visor" ,
195                 conf            => <<EOF
196 KERNEL="ttyUSB0", NAME="sub/direct/ory/visor"
197 EOF
198         },
199         {
200                 desc            => "place on bus of scsi partition",
201                 subsys          => "block",
202                 devpath         => "/block/sda/sda3",
203                 exp_name        => "first_disk3" ,
204                 conf            => <<EOF
205 BUS="scsi", PLACE="0:0:0:0", NAME="first_disk%n"
206 EOF
207         },
208         {
209                 desc            => "test NAME substitution chars",
210                 subsys          => "block",
211                 devpath         => "/block/sda/sda3",
212                 exp_name        => "Major:8:minor:3:kernelnumber:3:bus:0:0:0:0" ,
213                 conf            => <<EOF
214 BUS="scsi", PLACE="0:0:0:0", NAME="Major:%M:minor:%m:kernelnumber:%n:bus:%b"
215 EOF
216         },
217         {
218                 desc            => "test NAME substitution chars (with length limit)",
219                 subsys          => "block",
220                 devpath         => "/block/sda/sda3",
221                 exp_name        => "M8-m3-n3-b0:0-sIBM" ,
222                 conf            => <<EOF
223 BUS="scsi", PLACE="0:0:0:0", NAME="M%M-m%m-n%n-b%3b-s%3s{vendor}"
224 EOF
225         },
226         {
227                 desc            => "old style SYSFS_ attribute",
228                 subsys          => "block",
229                 devpath         => "/block/sda",
230                 exp_name        => "good" ,
231                 conf            => <<EOF
232 BUS="scsi", SYSFS_vendor="IBM-ESXS", NAME="good"
233 EOF
234         },
235         {
236                 desc            => "sustitution of sysfs value (%s{file})",
237                 subsys          => "block",
238                 devpath         => "/block/sda",
239                 exp_name        => "disk-IBM-ESXS-sda" ,
240                 conf            => <<EOF
241 BUS="scsi", SYSFS{vendor}="IBM-ESXS", NAME="disk-%s{vendor}-%k"
242 KERNEL="ttyUSB0", NAME="visor"
243 EOF
244         },
245         {
246                 desc            => "program result substitution",
247                 subsys          => "block",
248                 devpath         => "/block/sda/sda3",
249                 exp_name        => "special-device-3" ,
250                 conf            => <<EOF
251 BUS="scsi", PROGRAM="/bin/echo -n special-device", RESULT="-special-*", NAME="%c-1-%n"
252 BUS="scsi", PROGRAM="/bin/echo -n special-device", RESULT="special--*", NAME="%c-2-%n"
253 BUS="scsi", PROGRAM="/bin/echo -n special-device", RESULT="special-device-", NAME="%c-3-%n"
254 BUS="scsi", PROGRAM="/bin/echo -n special-device", RESULT="special-devic", NAME="%c-4-%n"
255 BUS="scsi", PROGRAM="/bin/echo -n special-device", RESULT="special-*", NAME="%c-%n"
256 EOF
257         },
258         {
259                 desc            => "program result substitution (no argument should be subsystem)",
260                 subsys          => "block",
261                 devpath         => "/block/sda/sda3",
262                 exp_name        => "subsys_block" ,
263                 conf            => <<EOF
264 BUS="scsi", PROGRAM="/bin/echo", RESULT="block", NAME="subsys_block"
265 EOF
266         },
267         {
268                 desc            => "program result substitution (newline removal)",
269                 subsys          => "block",
270                 devpath         => "/block/sda/sda3",
271                 exp_name        => "newline_removed" ,
272                 conf            => <<EOF
273 BUS="scsi", PROGRAM="/bin/echo test", RESULT="test", NAME="newline_removed"
274 EOF
275         },
276         {
277                 desc            => "program result substitution",
278                 subsys          => "block",
279                 devpath         => "/block/sda/sda3",
280                 exp_name        => "test-0:0:0:0" ,
281                 conf            => <<EOF
282 BUS="scsi", PROGRAM="/bin/echo -n test-%b", RESULT="test-0:0*", NAME="%c"
283 EOF
284         },
285         {
286                 desc            => "program with escaped format char (tricky: callout returns format char!)",
287                 subsys          => "block",
288                 devpath         => "/block/sda/sda3",
289                 exp_name        => "escape-3" ,
290                 conf            => <<EOF
291 BUS="scsi", PROGRAM="/bin/echo -n escape-%%n", KERNEL="sda3", NAME="%c"
292 EOF
293         },
294         {
295                 desc            => "program with lots of arguments",
296                 subsys          => "block",
297                 devpath         => "/block/sda/sda3",
298                 exp_name        => "foo9" ,
299                 conf            => <<EOF
300 BUS="scsi", PROGRAM="/bin/echo -n foo3 foo4 foo5 foo6 foo7 foo8 foo9", KERNEL="sda3", NAME="%c{7}"
301 EOF
302         },
303         {
304                 desc            => "program with subshell",
305                 subsys          => "block",
306                 devpath         => "/block/sda/sda3",
307                 exp_name        => "bar9" ,
308                 conf            => <<EOF
309 BUS="scsi", PROGRAM="/bin/sh -c 'echo foo3 foo4 foo5 foo6 foo7 foo8 foo9 | sed  s/foo9/bar9/'", KERNEL="sda3", NAME="%c{7}"
310 EOF
311         },
312         {
313                 desc            => "program arguments combined with apostrophes",
314                 subsys          => "block",
315                 devpath         => "/block/sda/sda3",
316                 exp_name        => "foo7" ,
317                 conf            => <<EOF
318 BUS="scsi", PROGRAM="/bin/echo -n 'foo3 foo4'   'foo5   foo6   foo7 foo8'", KERNEL="sda3", NAME="%c{5}"
319 EOF
320         },
321         {
322                 desc            => "characters before the %c{N} substitution",
323                 subsys          => "block",
324                 devpath         => "/block/sda/sda3",
325                 exp_name        => "my-foo9" ,
326                 conf            => <<EOF
327 BUS="scsi", PROGRAM="/bin/echo -n foo3 foo4 foo5 foo6 foo7 foo8 foo9", KERNEL="sda3", NAME="my-%c{7}"
328 EOF
329         },
330         {
331                 desc            => "substitute the second to last argument",
332                 subsys          => "block",
333                 devpath         => "/block/sda/sda3",
334                 exp_name        => "my-foo8" ,
335                 conf            => <<EOF
336 BUS="scsi", PROGRAM="/bin/echo -n foo3 foo4 foo5 foo6 foo7 foo8 foo9", KERNEL="sda3", NAME="my-%c{6}"
337 EOF
338         },
339         {
340                 desc            => "invalid program for device with no bus",
341                 subsys          => "tty",
342                 devpath         => "/class/tty/console",
343                 exp_name        => "TTY" ,
344                 conf            => <<EOF
345 BUS="scsi", PROGRAM="/bin/echo -n foo", RESULT="foo", NAME="foo"
346 KERNEL="console", NAME="TTY"
347 EOF
348         },
349         {
350                 desc            => "valid program for device with no bus",
351                 subsys          => "tty",
352                 devpath         => "/class/tty/console",
353                 exp_name        => "foo" ,
354                 conf            => <<EOF
355 PROGRAM="/bin/echo -n foo", RESULT="foo", NAME="foo"
356 KERNEL="console", NAME="TTY"
357 EOF
358         },
359         {
360                 desc            => "invalid label for device with no bus",
361                 subsys          => "tty",
362                 devpath         => "/class/tty/console",
363                 exp_name        => "TTY" ,
364                 conf            => <<EOF
365 BUS="foo", SYSFS{dev}="5:1", NAME="foo"
366 KERNEL="console", NAME="TTY"
367 EOF
368         },
369         {
370                 desc            => "valid label for device with no bus",
371                 subsys          => "tty",
372                 devpath         => "/class/tty/console",
373                 exp_name        => "foo" ,
374                 conf            => <<EOF
375 SYSFS{dev}="5:1", NAME="foo"
376 KERNEL="console", NAME="TTY"
377 EOF
378         },
379         {
380                 desc            => "program and bus type match",
381                 subsys          => "block",
382                 devpath         => "/block/sda",
383                 exp_name        => "scsi-0:0:0:0" ,
384                 conf            => <<EOF
385 BUS="usb", PROGRAM="/bin/echo -n usb-%b", NAME="%c"
386 BUS="scsi", PROGRAM="/bin/echo -n scsi-%b", NAME="%c"
387 BUS="foo", PROGRAM="/bin/echo -n foo-%b", NAME="%c"
388 EOF
389         },
390         {
391                 desc            => "create all possible partitions",
392                 subsys          => "block",
393                 devpath         => "/block/sda",
394                 exp_name        => "boot_disk15" ,
395                 conf            => <<EOF
396 BUS="scsi", SYSFS{vendor}="IBM-ESXS", NAME{all_partitions}="boot_disk"
397 EOF
398         },
399         {
400                 desc            => "sysfs parent hierarchy",
401                 subsys          => "tty",
402                 devpath         => "/class/tty/ttyUSB0",
403                 exp_name        => "visor" ,
404                 conf            => <<EOF
405 SYSFS{idProduct}="2008", NAME="visor"
406 EOF
407         },
408         {
409                 desc            => "name test with ! in the name",
410                 subsys          => "block",
411                 devpath         => "/block/rd!c0d0",
412                 exp_name        => "rd/c0d0" ,
413                 conf            => <<EOF
414 BUS="scsi", NAME="%k"
415 KERNEL="ttyUSB0", NAME="visor"
416 EOF
417         },
418         {
419                 desc            => "name test with ! in the name, but no matching rule",
420                 subsys          => "block",
421                 devpath         => "/block/rd!c0d0",
422                 exp_name        => "rd/c0d0" ,
423                 conf            => <<EOF
424 KERNEL="ttyUSB0", NAME="visor"
425 EOF
426         },
427         {
428                 desc            => "name test with ! in the name for a partition",
429                 subsys          => "block",
430                 devpath         => "/block/cciss!c0d0/cciss!c0d0p1",
431                 exp_name        => "cciss/c0d0p1" ,
432                 conf            => <<EOF
433 BUS="scsi", NAME="%k"
434 KERNEL="ttyUSB0", NAME="visor"
435 EOF
436         },
437         {
438                 desc            => "ID rule",
439                 subsys          => "block",
440                 devpath         => "/block/sda",
441                 exp_name        => "scsi-0:0:0:0",
442                 conf            => <<EOF
443 BUS="usb", ID="0:0:0:0", NAME="not-scsi"
444 BUS="scsi", ID="0:0:0:1", NAME="no-match"
445 BUS="scsi", ID=":0", NAME="short-id"
446 BUS="scsi", ID="/0:0:0:0", NAME="no-match"
447 BUS="scsi", ID="0:0:0:0", NAME="scsi-0:0:0:0"
448 EOF
449         },
450         {
451                 desc            => "ID wildcard all",
452                 subsys          => "block",
453                 devpath         => "/block/sda",
454                 exp_name        => "scsi-0:0:0:0",
455                 conf            => <<EOF
456 BUS="scsi", ID="*:1", NAME="no-match"
457 BUS="scsi", ID="*:0:1", NAME="no-match"
458 BUS="scsi", ID="*:0:0:1", NAME="no-match"
459 BUS="scsi", ID="*", NAME="scsi-0:0:0:0"
460 BUS="scsi", ID="0:0:0:0", NAME="bad"
461 EOF
462         },
463         {
464                 desc            => "ID wildcard partial",
465                 subsys          => "block",
466                 devpath         => "/block/sda",
467                 exp_name        => "scsi-0:0:0:0",
468                 conf            => <<EOF
469 BUS="scsi", ID="*:0", NAME="scsi-0:0:0:0"
470 BUS="scsi", ID="0:0:0:0", NAME="bad"
471 EOF
472         },
473         {
474                 desc            => "ID wildcard partial 2",
475                 subsys          => "block",
476                 devpath         => "/block/sda",
477                 exp_name        => "scsi-0:0:0:0",
478                 conf            => <<EOF
479 BUS="scsi", ID="*:0:0:0", NAME="scsi-0:0:0:0"
480 BUS="scsi", ID="0:0:0:0", NAME="bad"
481 EOF
482         },
483         {
484                 desc            => "ignore SYSFS attribute whitespace",
485                 subsys          => "block",
486                 devpath         => "/block/sda",
487                 exp_name        => "ignored",
488                 conf            => <<EOF
489 BUS="scsi", SYSFS{whitespace_test}="WHITE  SPACE", NAME="ignored"
490 EOF
491         },
492         {
493                 desc            => "do not ignore SYSFS attribute whitespace",
494                 subsys          => "block",
495                 devpath         => "/block/sda",
496                 exp_name        => "matched-with-space",
497                 conf            => <<EOF
498 BUS="scsi", SYSFS{whitespace_test}="WHITE  SPACE ", NAME="wrong-to-ignore"
499 BUS="scsi", SYSFS{whitespace_test}="WHITE  SPACE   ", NAME="matched-with-space"
500 EOF
501         },
502         {
503                 desc            => "permissions test",
504                 subsys          => "block",
505                 devpath         => "/block/sda",
506                 exp_name        => "node",
507                 exp_perms       => "5000::0444",
508                 conf            => <<EOF
509 BUS="scsi", KERNEL="sda", NAME="node", OWNER="5000", MODE="0444"
510 EOF
511         },
512         {
513                 desc            => "permissions ttyUSB0:root:uucp:0660",
514                 subsys          => "tty",
515                 devpath         => "/class/tty/ttyUSB0",
516                 exp_name        => "ttyUSB0",
517                 exp_perms       => "0:14:0660",
518                 conf            => <<EOF
519 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n"
520 EOF
521         },
522         {
523                 desc            => "permissions tty0::root:0444",
524                 subsys          => "tty",
525                 devpath         => "/class/tty/tty0",
526                 exp_name        => "tty0",
527                 exp_perms       => "0:0:0444",
528                 conf            => <<EOF
529 KERNEL="tty0", NAME="tty0"
530 EOF
531         },
532         {
533                 desc            => "permissions tty1:root::0555",
534                 subsys          => "tty",
535                 devpath         => "/class/tty/tty1",
536                 exp_name        => "tty1",
537                 exp_perms       => "0:0:0555",
538                 conf            => <<EOF
539 KERNEL="tty1", NAME="tty1"
540 EOF
541         },
542         {
543                 desc            => "permissions tty2:::0777",
544                 subsys          => "tty",
545                 devpath         => "/class/tty/tty2",
546                 exp_name        => "tty2",
547                 exp_perms       => "0:0:0777",
548                 conf            => <<EOF
549 KERNEL="tty2", NAME="tty2"
550 EOF
551         },
552         {
553                 desc            => "permissions tty3::: (default mode applied)",
554                 subsys          => "tty",
555                 devpath         => "/class/tty/tty3",
556                 exp_name        => "tty3",
557                 exp_perms       => "0:0:600",
558                 conf            => <<EOF
559 KERNEL="tty3", NAME="tty3"
560 EOF
561         },
562         {
563                 desc            => "permissions i2c-300:root:sys:0744",
564                 subsys          => "i2c-dev",
565                 devpath         => "/class/i2c-dev/i2c-300",
566                 exp_name        => "i2c-300",
567                 exp_perms       => "0:3:0744",
568                 conf            => <<EOF
569 KERNEL="i2c-300", NAME="i2c-300"
570 EOF
571         },
572         {
573                 desc            => "permissions i2c-fake1:root:7:0007",
574                 subsys          => "i2c-dev",
575                 devpath         => "/class/i2c-dev/i2c-fake1",
576                 exp_name        => "i2c-fake1",
577                 exp_perms       => "0:7:0007",
578                 conf            => <<EOF
579 KERNEL="i2c-fake1", NAME="i2c-fake1"
580 EOF
581         },
582         {
583                 desc            => "permissions ttyS[01]:0:5:0700",
584                 subsys          => "tty",
585                 devpath         => "/class/tty/ttyS1",
586                 exp_name        => "ttyS1",
587                 exp_perms       => "0:5:0700",
588                 conf            => <<EOF
589 KERNEL="ttyS1", NAME="ttyS1"
590 EOF
591         },
592         {
593                 desc            => "permissions ttyS[4-9]:tty:5:0060",
594                 subsys          => "tty",
595                 devpath         => "/class/tty/ttyS7",
596                 exp_name        => "ttyS7",
597                 exp_perms       => "0:5:0060",
598                 conf            => <<EOF
599 KERNEL="ttyS7", NAME="ttyS7"
600 EOF
601         },
602         {
603                 desc            => "permissions tty4:0:5:0707",
604                 subsys          => "ttyS4",
605                 devpath         => "/class/tty/tty4",
606                 exp_name        => "tty4",
607                 exp_perms       => "0:5:0707",
608                 conf            => <<EOF
609 KERNEL="tty4", NAME="tty4"
610 EOF
611         },
612         {
613                 desc            => "permissions tty4?:0:5:0007",
614                 subsys          => "tty",
615                 devpath         => "/class/tty/tty44",
616                 exp_name        => "tty44",
617                 exp_perms       => "0:5:0007",
618                 conf            => <<EOF
619 KERNEL="tty44", NAME="tty44"
620 EOF
621         },
622         {
623                 desc            => "permissions tty3[!3]:::0467",
624                 subsys          => "tty",
625                 devpath         => "/class/tty/tty35",
626                 exp_name        => "tty35",
627                 exp_perms       => "0:0:0467",
628                 conf            => <<EOF
629 KERNEL="tty35", NAME="tty35"
630 EOF
631         },
632         {
633                 desc            => "permissions tty33:bad:name:0500",
634                 subsys          => "tty",
635                 devpath         => "/class/tty/tty33",
636                 exp_name        => "tty33",
637                 exp_perms       => "0:0:0500",
638                 conf            => <<EOF
639 KERNEL="tty33", NAME="tty33"
640 EOF
641         },
642         {
643                 desc            => "permissions rtc:0:users:0600",
644                 subsys          => "misc",
645                 devpath         => "/class/misc/rtc",
646                 exp_name        => "misc/rtc",
647                 exp_perms       => "0:100:0600",
648                 conf            => <<EOF
649 KERNEL="rtc", NAME="misc/rtc"
650 EOF
651         },
652         {
653                 desc            => "permissions misc:0:users:0600",
654                 subsys          => "misc",
655                 devpath         => "/class/misc/psaux",
656                 exp_name        => "misc/psaux",
657                 exp_perms       => "0:100:0600",
658                 conf            => <<EOF
659 KERNEL="psaux", NAME="misc/psaux"
660 EOF
661         },
662         {
663                 desc            => "permissions set OWNER=5000",
664                 subsys          => "block",
665                 devpath         => "/block/sda",
666                 exp_name        => "node",
667                 exp_perms       => "5000::0600",
668                 conf            => <<EOF
669 BUS="scsi", KERNEL="sda", NAME="node", OWNER="5000"
670 EOF
671         },
672         {
673                 desc            => "permissions set GROUP=100",
674                 subsys          => "block",
675                 devpath         => "/block/sda",
676                 exp_name        => "node",
677                 exp_perms       => ":100:0600",
678                 conf            => <<EOF
679 BUS="scsi", KERNEL="sda", NAME="node", GROUP="100"
680 EOF
681         },
682         {
683                 desc            => "permissions set mode=0777",
684                 subsys          => "block",
685                 devpath         => "/block/sda",
686                 exp_name        => "node",
687                 exp_perms       => "::0777",
688                 conf            => <<EOF
689 BUS="scsi", KERNEL="sda", NAME="node", MODE="0777"
690 EOF
691         },
692         {
693                 desc            => "permissions set OWNER=5000 GROUP=100 MODE=0777",
694                 subsys          => "block",
695                 devpath         => "/block/sda",
696                 exp_name        => "node",
697                 exp_perms       => "5000:100:0777",
698                 conf            => <<EOF
699 BUS="scsi", KERNEL="sda", NAME="node", OWNER="5000", GROUP="100", MODE="0777"
700 EOF
701         },
702         {
703                 desc            => "permissions override OWNER to 5000",
704                 subsys          => "tty",
705                 devpath         => "/class/tty/ttyUSB0",
706                 exp_name        => "ttyUSB0",
707                 exp_perms       => "5000:14:0660",
708                 conf            => <<EOF
709 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", OWNER="5000"
710 EOF
711         },
712         {
713                 desc            => "permissions override GROUP to 100",
714                 subsys          => "tty",
715                 devpath         => "/class/tty/ttyUSB0",
716                 exp_name        => "ttyUSB0",
717                 exp_perms       => ":100:0660",
718                 conf            => <<EOF
719 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", GROUP="100"
720 EOF
721         },
722         {
723                 desc            => "permissions override MODE to 0060",
724                 subsys          => "tty",
725                 devpath         => "/class/tty/ttyUSB0",
726                 exp_name        => "ttyUSB0",
727                 exp_perms       => ":14:0060",
728                 conf            => <<EOF
729 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", MODE="0060"
730 EOF
731         },
732         {
733                 desc            => "permissions override OWNER, GROUP, MODE",
734                 subsys          => "tty",
735                 devpath         => "/class/tty/ttyUSB0",
736                 exp_name        => "ttyUSB0",
737                 exp_perms       => "5000:100:0777",
738                 conf            => <<EOF
739 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", OWNER="5000", GROUP="100", MODE="0777"
740 EOF
741         },
742         {
743                 desc            => "major/minor number test",
744                 subsys          => "block",
745                 devpath         => "/block/sda",
746                 exp_name        => "node",
747                 exp_majorminor  => "8:0",
748                 conf            => <<EOF
749 BUS="scsi", KERNEL="sda", NAME="node"
750 EOF
751         },
752         {
753                 desc            => "big minor number test",
754                 subsys          => "i2c-dev",
755                 devpath         => "/class/i2c-dev/i2c-300",
756                 exp_name        => "node",
757                 exp_majorminor  => "89:300",
758                 conf            => <<EOF
759 KERNEL="i2c-300", NAME="node"
760 EOF
761         },
762         {
763                 desc            => "big major number test",
764                 subsys          => "i2c-dev",
765                 devpath         => "/class/i2c-dev/i2c-fake1",
766                 exp_name        => "node",
767                 exp_majorminor  => "4095:1",
768                 conf            => <<EOF
769 KERNEL="i2c-fake1", NAME="node"
770 EOF
771         },
772         {
773                 desc            => "big major and big minor number test",
774                 subsys          => "i2c-dev",
775                 devpath         => "/class/i2c-dev/i2c-fake2",
776                 exp_name        => "node",
777                 exp_majorminor  => "4094:89999",
778                 conf            => <<EOF
779 KERNEL="i2c-fake2", NAME="node"
780 EOF
781         },
782         {
783                 desc            => "multiple symlinks with format char",
784                 subsys          => "tty",
785                 devpath         => "/class/tty/ttyUSB0",
786                 exp_name        => "symlink2-ttyUSB0",
787                 exp_target      => "ttyUSB0",
788                 conf            => <<EOF
789 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", SYMLINK="symlink1-%n symlink2-%k symlink3-%b"
790 EOF
791         },
792         {
793                 desc            => "symlink creation (same directory)",
794                 subsys          => "tty",
795                 devpath         => "/class/tty/ttyUSB0",
796                 exp_name        => "visor0",
797                 exp_target      => "ttyUSB0",
798                 conf            => <<EOF
799 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", SYMLINK="visor%n"
800 EOF
801         },
802         {
803                 desc            => "symlink creation (relative link forward)",
804                 subsys          => "block",
805                 devpath         => "/block/sda/sda2",
806                 exp_name        => "1/2/symlink" ,
807                 exp_target      => "a/b/node",
808                 conf            => <<EOF
809 BUS="scsi", SYSFS{vendor}="IBM-ESXS", NAME="1/2/a/b/node", SYMLINK="1/2/symlink"
810 EOF
811         },
812         {
813                 desc            => "symlink creation (relative link back and forward)",
814                 subsys          => "block",
815                 devpath         => "/block/sda/sda2",
816                 exp_name        => "1/2/c/d/symlink" ,
817                 exp_target      => "../../a/b/node",
818                 conf            => <<EOF
819 BUS="scsi", SYSFS{vendor}="IBM-ESXS", NAME="1/2/a/b/node", SYMLINK="1/2/c/d/symlink"
820 EOF
821         },
822         {
823                 desc            => "multiple symlinks",
824                 subsys          => "tty",
825                 devpath         => "/class/tty/ttyUSB0",
826                 exp_name        => "second-0" ,
827                 exp_target      => "visor" ,
828                 conf            => <<EOF
829 KERNEL="ttyUSB0", NAME="visor", SYMLINK="first-%n second-%n third-%n"
830 EOF
831         },
832         {
833                 desc            => "symlink only rule",
834                 subsys          => "block",
835                 devpath         => "/block/sda",
836                 exp_name        => "symlink-only2",
837                 exp_target      => "link",
838                 conf            => <<EOF
839 BUS="scsi", KERNEL="sda", SYMLINK="symlink-only1"
840 BUS="scsi", KERNEL="sda", SYMLINK="symlink-only2"
841 BUS="scsi", KERNEL="sda", NAME="link", SYMLINK="symlink0"
842 EOF
843         },
844         {
845                 desc            => "symlink name empty",
846                 subsys          => "block",
847                 devpath         => "/block/sda",
848                 exp_name        => "",
849                 exp_target      => "link",
850                 exp_error       => "yes",
851                 conf            => <<EOF
852 BUS="scsi", KERNEL="sda", NAME="link", SYMLINK=""
853 EOF
854         },
855         {
856                 desc            => "symlink name '.'",
857                 subsys          => "block",
858                 devpath         => "/block/sda",
859                 exp_name        => ".",
860                 exp_target      => "link",
861                 exp_error       => "yes",
862                 conf            => <<EOF
863 BUS="scsi", KERNEL="sda", NAME="link", SYMLINK="."
864 EOF
865         },
866         {
867                 desc            => "symlink to empty name",
868                 subsys          => "block",
869                 devpath         => "/block/sda",
870                 exp_name        => "symlink",
871                 exp_target      => "",
872                 exp_error       => "yes",
873                 conf            => <<EOF
874 BUS="scsi", KERNEL="sda", NAME="", SYMLINK="symlink"
875 EOF
876         },
877         {
878                 desc            => "symlink and name empty",
879                 subsys          => "block",
880                 devpath         => "/block/sda",
881                 exp_name        => "",
882                 exp_target      => "",
883                 exp_error       => "yes",
884                 conf            => <<EOF
885 BUS="scsi", KERNEL="sda", NAME="", SYMLINK=""
886 EOF
887         },
888         {
889                 desc            => "symlink node to itself",
890                 subsys          => "tty",
891                 devpath         => "/class/tty/tty0",
892                 exp_name        => "link",
893                 exp_target      => "link",
894                 conf            => <<EOF
895 KERNEL="tty0", NAME="link", SYMLINK="link"
896 EOF
897         },
898         {
899                 desc            => "symlink %n substitution",
900                 subsys          => "tty",
901                 devpath         => "/class/tty/ttyUSB0",
902                 exp_name        => "symlink0",
903                 exp_target      => "ttyUSB0",
904                 conf            => <<EOF
905 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", SYMLINK="symlink%n"
906 EOF
907         },
908         {
909                 desc            => "symlink %k substitution",
910                 subsys          => "tty",
911                 devpath         => "/class/tty/ttyUSB0",
912                 exp_name        => "symlink-ttyUSB0",
913                 exp_target      => "ttyUSB0",
914                 conf            => <<EOF
915 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", SYMLINK="symlink-%k"
916 EOF
917         },
918         {
919                 desc            => "symlink %M:%m substitution",
920                 subsys          => "tty",
921                 devpath         => "/class/tty/ttyUSB0",
922                 exp_name        => "major-188:0",
923                 exp_target      => "ttyUSB0",
924                 conf            => <<EOF
925 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", SYMLINK="major-%M:%m"
926 EOF
927         },
928         {
929                 desc            => "symlink %b substitution",
930                 subsys          => "block",
931                 devpath         => "/block/sda",
932                 exp_name        => "symlink-0:0:0:0",
933                 exp_target      => "node",
934                 conf            => <<EOF
935 BUS="scsi", KERNEL="sda", NAME="node", SYMLINK="symlink-%b"
936 EOF
937         },
938         {
939                 desc            => "symlink %c substitution",
940                 subsys          => "tty",
941                 devpath         => "/class/tty/ttyUSB0",
942                 exp_name        => "test",
943                 exp_target      => "ttyUSB0",
944                 conf            => <<EOF
945 KERNEL="ttyUSB[0-9]*", PROGRAM="/bin/echo test" NAME="ttyUSB%n", SYMLINK="%c"
946 EOF
947         },
948         {
949                 desc            => "symlink %c{N} substitution",
950                 subsys          => "tty",
951                 devpath         => "/class/tty/ttyUSB0",
952                 exp_name        => "test",
953                 exp_target      => "ttyUSB0",
954                 conf            => <<EOF
955 KERNEL="ttyUSB[0-9]*", PROGRAM="/bin/echo symlink test this" NAME="ttyUSB%n", SYMLINK="%c{2}"
956 EOF
957         },
958         {
959                 desc            => "symlink %c{N+} substitution",
960                 subsys          => "tty",
961                 devpath         => "/class/tty/ttyUSB0",
962                 exp_name        => "this",
963                 exp_target      => "ttyUSB0",
964                 conf            => <<EOF
965 KERNEL="ttyUSB[0-9]*", PROGRAM="/bin/echo symlink test this" NAME="ttyUSB%n", SYMLINK="%c{2+}"
966 EOF
967         },
968         {
969                 desc            => "symlink only rule with %c{N+}",
970                 subsys          => "block",
971                 devpath         => "/block/sda",
972                 exp_name        => "test",
973                 exp_target      => "link",
974                 conf            => <<EOF
975 BUS="scsi", KERNEL="sda", PROGRAM="/bin/echo link test this" SYMLINK="%c{2+}"
976 BUS="scsi", KERNEL="sda", NAME="link", SYMLINK="symlink0"
977 EOF
978         },
979         {
980                 desc            => "symlink %s{filename} substitution",
981                 subsys          => "tty",
982                 devpath         => "/class/tty/ttyUSB0",
983                 exp_name        => "188:0",
984                 exp_target      => "ttyUSB0",
985                 conf            => <<EOF
986 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", SYMLINK="%s{dev}"
987 EOF
988         },
989         {
990                 desc            => "symlink %Ns{filename} substitution",
991                 subsys          => "tty",
992                 devpath         => "/class/tty/ttyUSB0",
993                 exp_name        => "188",
994                 exp_target      => "ttyUSB0",
995                 conf            => <<EOF
996 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", SYMLINK="%3s{dev}"
997 EOF
998         },
999         {
1000                 desc            => "symlink with '%' in name",
1001                 subsys          => "tty",
1002                 devpath         => "/class/tty/ttyUSB0",
1003                 exp_name        => "percent%sign",
1004                 exp_target      => "ttyUSB0",
1005                 conf            => <<EOF
1006 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", SYMLINK="percent%%sign"
1007 EOF
1008         },
1009         {
1010                 desc            => "symlink with '%' in name",
1011                 subsys          => "tty",
1012                 devpath         => "/class/tty/ttyUSB0",
1013                 exp_name        => "%ttyUSB0_name",
1014                 exp_target      => "ttyUSB0",
1015                 conf            => <<EOF
1016 KERNEL="ttyUSB[0-9]*", NAME="ttyUSB%n", SYMLINK="%%%k_name"
1017 EOF
1018         },
1019         {
1020                 desc            => "program result substitution (numbered part of)",
1021                 subsys          => "block",
1022                 devpath         => "/block/sda/sda3",
1023                 exp_name        => "link1",
1024                 exp_target      => "node",
1025                 conf            => <<EOF
1026 BUS="scsi", PROGRAM="/bin/echo -n node link1 link2", RESULT="node *", NAME="%c{1}", SYMLINK="%c{2} %c{3}"
1027 EOF
1028         },
1029         {
1030                 desc            => "program result substitution (numbered part of+)",
1031                 subsys          => "block",
1032                 devpath         => "/block/sda/sda3",
1033                 exp_name        => "link4",
1034                 exp_target      => "node",
1035                 conf            => <<EOF
1036 BUS="scsi", PROGRAM="/bin/echo -n node link1 link2 link3 link4", RESULT="node *", NAME="%c{1}", SYMLINK="%c{2+}"
1037 EOF
1038         },
1039         {
1040                 desc            => "enumeration char test (single test)",
1041                 subsys          => "block",
1042                 devpath         => "/block/sda",
1043                 exp_name        => "cdrom",
1044                 conf            => <<EOF
1045 KERNEL="sda", NAME="cdrom%e"
1046 EOF
1047         },
1048         {
1049                 desc            => "enumeration char test sequence (1/5 keep)",
1050                 subsys          => "block",
1051                 devpath         => "/block/sda",
1052                 exp_name        => "cdrom",
1053                 option          => "keep",
1054                 conf            => <<EOF
1055 KERNEL="sda", NAME="cdrom%e"
1056 EOF
1057         },
1058         {
1059                 desc            => "enumeration char test sequence 2/5 (keep)",
1060                 subsys          => "block",
1061                 devpath         => "/block/sda/sda1",
1062                 exp_name        => "enum",
1063                 option          => "keep",
1064                 conf            => <<EOF
1065 KERNEL="sda1", NAME="enum%e"
1066 EOF
1067         },
1068         {
1069                 desc            => "enumeration char test sequence 3/5 (keep)",
1070                 subsys          => "block",
1071                 devpath         => "/block/sda/sda2",
1072                 exp_name        => "cdrom1",
1073                 option          => "keep",
1074                 conf            => <<EOF
1075 KERNEL="sda2", NAME="cdrom%e"
1076 EOF
1077         },
1078         {
1079                 desc            => "enumeration char test sequence 4/5 (keep)",
1080                 subsys          => "block",
1081                 devpath         => "/block/sda/sda3",
1082                 exp_name        => "enum1",
1083                 option          => "keep",
1084                 conf            => <<EOF
1085 KERNEL="sda3", NAME="enum%e"
1086 EOF
1087         },
1088         {
1089                 desc            => "enumeration char test sequence 5/5 (clean)",
1090                 subsys          => "block",
1091                 devpath         => "/block/sda/sda4",
1092                 exp_name        => "cdrom2",
1093                 option          => "clear",
1094                 conf            => <<EOF
1095 KERNEL="sda4", NAME="cdrom%e"
1096 EOF
1097         },
1098         {
1099                 desc            => "enumeration char test after cleanup (single test)",
1100                 subsys          => "block",
1101                 devpath         => "/block/sda",
1102                 exp_name        => "cdrom",
1103                 conf            => <<EOF
1104 KERNEL="sda", NAME="cdrom%e"
1105 EOF
1106         },
1107         {
1108                 desc            => "ignore remove event test",
1109                 subsys          => "block",
1110                 devpath         => "/block/sda",
1111                 exp_name        => "node",
1112                 exp_error       => "yes",
1113                 conf            => <<EOF
1114 BUS="scsi", KERNEL="sda", NAME{ignore_remove}="node"
1115 EOF
1116         },
1117         {
1118                 desc            => "ignore remove event test (with all partitions)",
1119                 subsys          => "block",
1120                 devpath         => "/block/sda",
1121                 exp_name        => "node14",
1122                 exp_error       => "yes",
1123                 option          => "clear",
1124                 conf            => <<EOF
1125 BUS="scsi", KERNEL="sda", NAME{ignore_remove, all_partitions}="node"
1126 EOF
1127         },
1128         {
1129                 desc            => "SUBSYSTEM match test",
1130                 subsys          => "block",
1131                 devpath         => "/block/sda",
1132                 exp_name        => "node",
1133                 conf            => <<EOF
1134 BUS="scsi", KERNEL="sda", NAME="should_not_match", SUBSYSTEM="vc"
1135 BUS="scsi", KERNEL="sda", NAME="node", SUBSYSTEM="block"
1136 BUS="scsi", KERNEL="sda", NAME="should_not_match2", SUBSYSTEM="vc"
1137 EOF
1138         },
1139         {
1140                 desc            => "DRIVER match test",
1141                 subsys          => "block",
1142                 devpath         => "/block/sda",
1143                 exp_name        => "node",
1144                 conf            => <<EOF
1145 BUS="scsi", KERNEL="sda", NAME="should_not_match", DRIVER="sd-wrong"
1146 BUS="scsi", KERNEL="sda", NAME="node", DRIVER="sd"
1147 EOF
1148         },
1149 );
1150
1151 # set env
1152 $ENV{UDEV_TEST} = "yes";
1153 $ENV{SYSFS_PATH} = $sysfs;
1154 $ENV{UDEV_CONFIG_FILE} = $main_conf;
1155 $ENV{UDEV_NO_DEVD} = "yes";
1156
1157
1158 sub udev {
1159         my ($action, $subsys, $devpath, $config) = @_;
1160
1161         $ENV{DEVPATH} = $devpath;
1162
1163         # create temporary config
1164         open CONF, ">$conf_tmp" || die "unable to create config file: $conf_tmp";
1165         print CONF $$config;
1166         close CONF;
1167
1168         $ENV{ACTION} = $action;
1169         system("$udev_bin $subsys");
1170 }
1171
1172 my $error = 0;
1173
1174 sub permissions_test {
1175         my($config, $uid, $gid, $mode) = @_;
1176
1177         my $wrong = 0;
1178         $config->{exp_perms} =~ m/^(.*):(.*):(.*)$/;
1179         if ($1 ne "") {
1180                 if ($uid != $1) { $wrong = 1; };
1181         }
1182         if ($2 ne "") {
1183                 if ($gid != $2) { $wrong = 1; };
1184         }
1185         if ($3 ne "") {
1186                 if (($mode & 07777) != oct($3)) { $wrong = 1; };
1187         }
1188         if ($wrong == 0) {
1189                 print "permissions: ok    ";
1190         } else {
1191                 printf "expected permissions are: %i:%i:%#o\n", $1, $2, oct($3);
1192                 printf "created permissions are : %i:%i:%#o\n", $uid, $gid, $mode & 07777;
1193                 $error++;
1194         }
1195 }
1196
1197 sub major_minor_test {
1198         my($config, $rdev) = @_;
1199
1200         my $major = ($rdev >> 8) & 0xfff;
1201         my $minor = ($rdev & 0xff) | (($rdev >> 12) & 0xfff00);
1202         my $wrong = 0;
1203
1204         $config->{exp_majorminor} =~ m/^(.*):(.*)$/;
1205         if ($1 ne "") {
1206                 if ($major != $1) { $wrong = 1; };
1207         }
1208         if ($2 ne "") {
1209                 if ($minor != $2) { $wrong = 1; };
1210         }
1211         if ($wrong == 0) {
1212                 print "major:minor: ok    ";
1213         } else {
1214                 printf "expected major:minor is: %i:%i\n", $1, $2;
1215                 printf "created major:minor is : %i:%i\n", $major, $minor;
1216                 print "major:minor: error    ";
1217                 $error++;
1218         }
1219 }
1220
1221 sub symlink_test {
1222         my ($config) = @_;
1223
1224         my $output = `ls -l $PWD/$udev_root$config->{exp_name}`;
1225
1226         if ($output =~ m/(.*)-> (.*)/) {
1227                 if ($2 eq $config->{exp_target}) {
1228                         print "symlink: ok    ";
1229                 } else {
1230                         print "expected symlink from: \'$config->{exp_name}\' to \'$config->{exp_target}\'\n";
1231                         print "created symlink from: \'$config->{exp_name}\' to \'$2\'\n";
1232                         if ($config->{exp_error}) {
1233                                 print "as expected    ";
1234                         } else {
1235                                 $error++;
1236                         }
1237                 }
1238         } else {
1239                 print "expected symlink from: \'$config->{exp_name}\' to \'$config->{exp_target}\'\n";
1240                 print "symlink: not created ";
1241                 if ($config->{exp_error}) {
1242                         print "as expected    ";
1243                 } else {
1244                         $error++;
1245                 }
1246         }
1247 }
1248
1249 sub run_test {
1250         my ($config, $number) = @_;
1251
1252         print "TEST $number: $config->{desc}\n";
1253
1254         if ($config->{exp_target}) {
1255                 print "device \'$config->{devpath}\' expecting symlink '$config->{exp_name}' to node \'$config->{exp_target}\'\n";
1256         } else {
1257                 print "device \'$config->{devpath}\' expecting node \'$config->{exp_name}\'\n";
1258         }
1259
1260
1261         udev("add", $config->{subsys}, $config->{devpath}, \$config->{conf});
1262
1263         if ((-e "$PWD/$udev_root$config->{exp_name}") ||
1264             (-l "$PWD/$udev_root$config->{exp_name}")) {
1265
1266                 my ($dev, $ino, $mode, $nlink, $uid, $gid, $rdev, $size,
1267                     $atime, $mtime, $ctime, $blksize, $blocks) = stat("$PWD/$udev_root$config->{exp_name}");
1268
1269                 if (defined($config->{exp_perms})) {
1270                         permissions_test($config, $uid, $gid, $mode);
1271                 }
1272                 if (defined($config->{exp_majorminor})) {
1273                         major_minor_test($config, $rdev);
1274                 }
1275                 if (defined($config->{exp_target})) {
1276                         symlink_test($config);
1277                 }
1278                 print "add: ok    ";
1279         } else {
1280                 print "add: error ";
1281                 if ($config->{exp_error}) {
1282                         print "as expected    ";
1283                 } else {
1284                         print "\n\n";
1285                         system("tree $udev_root");
1286                         print "\n";
1287                         $error++;
1288                 }
1289         }
1290
1291         if (defined($config->{option}) && $config->{option} eq "keep") {
1292                 print "\n\n";
1293                 return;
1294         }
1295
1296         udev("remove", $config->{subsys}, $config->{devpath}, \$config->{conf});
1297         if ((-e "$PWD/$udev_root$config->{exp_name}") ||
1298             (-l "$PWD/$udev_root$config->{exp_name}")) {
1299                 print "remove: error ";
1300                 if ($config->{exp_error}) {
1301                         print "as expected\n\n";
1302                 } else {
1303                         print "\n\n";
1304                         system("tree $udev_root");
1305                         print "\n";
1306                         $error++;
1307                 }
1308         } else {
1309                 print "remove: ok\n\n";
1310         }
1311
1312         if (defined($config->{option}) && $config->{option} eq "clear") {
1313                 system("rm -rf $udev_db");
1314                 system("rm -rf $udev_root");
1315                 mkdir($udev_root) || die "unable to create udev_root: $udev_root\n";
1316         }
1317
1318 }
1319
1320 # only run if we have root permissions
1321 # due to mknod restrictions
1322 if (!($<==0)) {
1323         print "Must have root permissions to run properly.\n";
1324         exit;
1325 }
1326
1327 # prepare
1328 system("rm -rf $udev_root");
1329 mkdir($udev_root) || die "unable to create udev_root: $udev_root\n";
1330
1331 # create initial config file
1332 open CONF, ">$main_conf" || die "unable to create config file: $main_conf";
1333 print CONF "udev_root=\"$udev_root\"\n";
1334 print CONF "udev_db=\"$udev_db\"\n";
1335 print CONF "udev_rules=\"$conf_tmp\"\n";
1336 print CONF "udev_permissions=\"$perm\"\n";
1337 close CONF;
1338
1339 my $test_num = 1;
1340
1341 if ($ARGV[0]) {
1342         # only run one test
1343         $test_num = $ARGV[0];
1344
1345         if (defined($tests[$test_num-1]->{desc})) {
1346                 print "udev-test will run test number $test_num only:\n\n";
1347                 run_test($tests[$test_num-1], $test_num);
1348         } else {
1349                 print "test does not exist.\n";
1350         }
1351 } else {
1352         # test all
1353         print "\nudev-test will run ".($#tests + 1)." tests:\n\n";
1354
1355         foreach my $config (@tests) {
1356                 run_test($config, $test_num);
1357                 $test_num++;
1358         }
1359 }
1360
1361 print "$error errors occured\n\n";
1362
1363 # cleanup
1364 system("rm -rf $udev_db");
1365 system("rm -rf $udev_root");
1366 unlink($conf_tmp);
1367 unlink($main_conf);
1368