]> git.ipfire.org Git - thirdparty/systemd.git/blame - test/udev-test.pl
systemctl: don't use get_process_comm() on non-local PIDs (#7518)
[thirdparty/systemd.git] / test / udev-test.pl
CommitLineData
a367f04e
GKH
1#!/usr/bin/perl
2
438d4c3c 3# udev test
a367f04e
GKH
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#
65005a7f 12# udev parses the rules, looks at the provided sysfs and
a367f04e
GKH
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#
1298001e 17# Copyright (C) 2004-2012 Kay Sievers <kay@vrfy.org>
c4edd0ad 18# Copyright (C) 2004 Leann Ogasawara <ogasawara@osdl.org>
a367f04e
GKH
19
20use warnings;
21use strict;
22
9b80f05f 23my $udev_bin = "./test-udev";
912541b0 24my $valgrind = 0;
333e07b7 25my $gdb = 0;
587751eb 26my $strace = 0;
b3f24900 27my $udev_bin_valgrind = "valgrind --tool=memcheck --leak-check=yes --track-origins=yes --quiet $udev_bin";
333e07b7 28my $udev_bin_gdb = "gdb --args $udev_bin";
587751eb 29my $udev_bin_strace = "strace -efile $udev_bin";
6ada823a 30my $udev_run = "test/run";
2ce8d27b 31my $udev_tmpfs = "test/tmpfs";
21d9e3f3
EV
32my $udev_sys = "${udev_tmpfs}/sys";
33my $udev_dev = "${udev_tmpfs}/dev";
6ada823a
KS
34my $udev_rules_dir = "$udev_run/udev/rules.d";
35my $udev_rules = "$udev_rules_dir/udev-test.rules";
0eb3cc88 36my $EXIT_TEST_SKIP = 77;
a367f04e 37
bf8f7583
MP
38my $rules_10k_tags = "";
39for (my $i = 1; $i <= 10000; ++$i) {
d3fc8bf4 40 $rules_10k_tags .= 'KERNEL=="sda", TAG+="test' . $i . "\"\n";
bf8f7583
MP
41}
42
a367f04e 43my @tests = (
912541b0
KS
44 {
45 desc => "no rules",
46 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
47 exp_name => "sda" ,
48 exp_rem_error => "yes",
49 rules => <<EOF
5754e74c 50#
bcf44d55 51EOF
912541b0
KS
52 },
53 {
54 desc => "label test of scsi disc",
55 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
56 exp_name => "boot_disk" ,
57 rules => <<EOF
0f52fdee 58SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA", SYMLINK+="boot_disk%n"
5754e74c 59KERNEL=="ttyACM0", SYMLINK+="modem"
c4edd0ad 60EOF
912541b0
KS
61 },
62 {
63 desc => "label test of scsi disc",
64 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
65 exp_name => "boot_disk" ,
66 rules => <<EOF
5754e74c
KS
67SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA", SYMLINK+="boot_disk%n"
68KERNEL=="ttyACM0", SYMLINK+="modem"
c4edd0ad 69EOF
912541b0
KS
70 },
71 {
72 desc => "label test of scsi disc",
73 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
74 exp_name => "boot_disk" ,
75 rules => <<EOF
5754e74c
KS
76SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA", SYMLINK+="boot_disk%n"
77KERNEL=="ttyACM0", SYMLINK+="modem"
a367f04e 78EOF
912541b0
KS
79 },
80 {
81 desc => "label test of scsi partition",
82 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
83 exp_name => "boot_disk1" ,
84 rules => <<EOF
5754e74c 85SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA", SYMLINK+="boot_disk%n"
83be97ba 86EOF
912541b0
KS
87 },
88 {
89 desc => "label test of pattern match",
90 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
91 exp_name => "boot_disk1" ,
92 rules => <<EOF
5754e74c
KS
93SUBSYSTEMS=="scsi", ATTRS{vendor}=="?ATA", SYMLINK+="boot_disk%n-1"
94SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA?", SYMLINK+="boot_disk%n-2"
95SUBSYSTEMS=="scsi", ATTRS{vendor}=="A??", SYMLINK+="boot_disk%n"
96SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATAS", SYMLINK+="boot_disk%n-3"
358c8c20 97EOF
912541b0
KS
98 },
99 {
100 desc => "label test of multiple sysfs files",
101 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
102 exp_name => "boot_disk1" ,
103 rules => <<EOF
5754e74c
KS
104SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA", ATTRS{model}=="ST910021AS X ", SYMLINK+="boot_diskX%n"
105SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA", ATTRS{model}=="ST910021AS", SYMLINK+="boot_disk%n"
358c8c20 106EOF
912541b0
KS
107 },
108 {
109 desc => "label test of max sysfs files (skip invalid rule)",
110 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
111 exp_name => "boot_disk1" ,
112 rules => <<EOF
5754e74c
KS
113SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA", ATTRS{model}=="ST910021AS", ATTRS{scsi_level}=="6", ATTRS{rev}=="4.06", ATTRS{type}=="0", ATTRS{queue_depth}=="32", SYMLINK+="boot_diskXX%n"
114SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA", ATTRS{model}=="ST910021AS", ATTRS{scsi_level}=="6", ATTRS{rev}=="4.06", ATTRS{type}=="0", SYMLINK+="boot_disk%n"
0db6d4cc 115EOF
912541b0
KS
116 },
117 {
118 desc => "catch device by *",
119 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
120 exp_name => "modem/0" ,
121 rules => <<EOF
5754e74c 122KERNEL=="ttyACM*", SYMLINK+="modem/%n"
2e317184 123EOF
912541b0
KS
124 },
125 {
126 desc => "catch device by * - take 2",
127 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
128 exp_name => "modem/0" ,
129 rules => <<EOF
5754e74c
KS
130KERNEL=="*ACM1", SYMLINK+="bad"
131KERNEL=="*ACM0", SYMLINK+="modem/%n"
9f1da361 132EOF
912541b0
KS
133 },
134 {
135 desc => "catch device by ?",
136 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
137 exp_name => "modem/0" ,
138 rules => <<EOF
5754e74c
KS
139KERNEL=="ttyACM??*", SYMLINK+="modem/%n-1"
140KERNEL=="ttyACM??", SYMLINK+="modem/%n-2"
141KERNEL=="ttyACM?", SYMLINK+="modem/%n"
9f1da361 142EOF
912541b0
KS
143 },
144 {
145 desc => "catch device by character class",
146 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
147 exp_name => "modem/0" ,
148 rules => <<EOF
5754e74c
KS
149KERNEL=="ttyACM[A-Z]*", SYMLINK+="modem/%n-1"
150KERNEL=="ttyACM?[0-9]", SYMLINK+="modem/%n-2"
151KERNEL=="ttyACM[0-9]*", SYMLINK+="modem/%n"
a367f04e 152EOF
912541b0
KS
153 },
154 {
155 desc => "replace kernel name",
156 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
157 exp_name => "modem" ,
158 rules => <<EOF
5754e74c 159KERNEL=="ttyACM0", SYMLINK+="modem"
281ff00a 160EOF
912541b0
KS
161 },
162 {
163 desc => "Handle comment lines in config file (and replace kernel name)",
164 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
165 exp_name => "modem" ,
166 rules => <<EOF
281ff00a 167# this is a comment
5754e74c 168KERNEL=="ttyACM0", SYMLINK+="modem"
281ff00a
GKH
169
170EOF
912541b0
KS
171 },
172 {
173 desc => "Handle comment lines in config file with whitespace (and replace kernel name)",
174 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
175 exp_name => "modem" ,
176 rules => <<EOF
d914e445 177 # this is a comment with whitespace before the comment
5754e74c 178KERNEL=="ttyACM0", SYMLINK+="modem"
281ff00a 179
3db7fa27 180EOF
912541b0
KS
181 },
182 {
183 desc => "Handle whitespace only lines (and replace kernel name)",
184 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
185 exp_name => "whitespace" ,
186 rules => <<EOF
3db7fa27 187
3db7fa27 188
d914e445
KS
189
190 # this is a comment with whitespace before the comment
5754e74c 191KERNEL=="ttyACM0", SYMLINK+="whitespace"
3db7fa27 192
d914e445 193
3db7fa27 194
281ff00a 195EOF
912541b0
KS
196 },
197 {
198 desc => "Handle empty lines in config file (and replace kernel name)",
199 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
200 exp_name => "modem" ,
201 rules => <<EOF
281ff00a 202
5754e74c 203KERNEL=="ttyACM0", SYMLINK+="modem"
281ff00a 204
9f8dfa19 205EOF
912541b0
KS
206 },
207 {
208 desc => "Handle backslashed multi lines in config file (and replace kernel name)",
209 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
210 exp_name => "modem" ,
211 rules => <<EOF
c7fcba1b 212KERNEL=="ttyACM0", \\
5754e74c 213SYMLINK+="modem"
9f8dfa19 214
77313cd0 215EOF
912541b0
KS
216 },
217 {
218 desc => "preserve backslashes, if they are not for a newline",
219 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
220 exp_name => "aaa",
221 rules => <<EOF
d914e445 222KERNEL=="ttyACM0", PROGRAM=="/bin/echo -e \\101", RESULT=="A", SYMLINK+="aaa"
9f8dfa19 223EOF
912541b0
KS
224 },
225 {
226 desc => "Handle stupid backslashed multi lines in config file (and replace kernel name)",
227 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
228 exp_name => "modem" ,
229 rules => <<EOF
9f8dfa19
KS
230
231#
232\\
233
d960ad15 234\\
9f8dfa19
KS
235
236#\\
237
c7fcba1b 238KERNEL=="ttyACM0", \\
912541b0 239 SYMLINK+="modem"
9f8dfa19 240
5499d319 241EOF
912541b0
KS
242 },
243 {
244 desc => "subdirectory handling",
245 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
246 exp_name => "sub/direct/ory/modem" ,
247 rules => <<EOF
5754e74c 248KERNEL=="ttyACM0", SYMLINK+="sub/direct/ory/modem"
a367f04e 249EOF
912541b0
KS
250 },
251 {
252 desc => "parent device name match of scsi partition",
253 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
254 exp_name => "first_disk5" ,
255 rules => <<EOF
5754e74c 256SUBSYSTEMS=="scsi", KERNELS=="0:0:0:0", SYMLINK+="first_disk%n"
c4edd0ad 257EOF
912541b0
KS
258 },
259 {
260 desc => "test substitution chars",
261 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
262 exp_name => "Major:8:minor:5:kernelnumber:5:id:0:0:0:0" ,
263 rules => <<EOF
5754e74c 264SUBSYSTEMS=="scsi", KERNELS=="0:0:0:0", SYMLINK+="Major:%M:minor:%m:kernelnumber:%n:id:%b"
319c6700 265EOF
912541b0
KS
266 },
267 {
268 desc => "import of shell-value returned from program",
269 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
270 exp_name => "node12345678",
271 rules => <<EOF
d914e445 272SUBSYSTEMS=="scsi", IMPORT{program}="/bin/echo -e \' TEST_KEY=12345678\\n TEST_key2=98765\'", SYMLINK+="node\$env{TEST_KEY}"
5754e74c 273KERNEL=="ttyACM0", SYMLINK+="modem"
a27cd06c 274EOF
912541b0
KS
275 },
276 {
277 desc => "sustitution of sysfs value (%s{file})",
278 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
279 exp_name => "disk-ATA-sda" ,
280 rules => <<EOF
5754e74c
KS
281SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA", SYMLINK+="disk-%s{vendor}-%k"
282KERNEL=="ttyACM0", SYMLINK+="modem"
a367f04e 283EOF
912541b0
KS
284 },
285 {
286 desc => "program result substitution",
287 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
288 exp_name => "special-device-5" ,
289 not_exp_name => "not" ,
290 rules => <<EOF
d914e445
KS
291SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n special-device", RESULT=="-special-*", SYMLINK+="not"
292SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n special-device", RESULT=="special-*", SYMLINK+="%c-%n"
bbbe503e 293EOF
912541b0
KS
294 },
295 {
296 desc => "program result substitution (newline removal)",
297 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
298 exp_name => "newline_removed" ,
299 rules => <<EOF
d914e445 300SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo test", RESULT=="test", SYMLINK+="newline_removed"
f3b04a2e 301EOF
912541b0
KS
302 },
303 {
304 desc => "program result substitution",
305 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
306 exp_name => "test-0:0:0:0" ,
307 rules => <<EOF
d914e445 308SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n test-%b", RESULT=="test-0:0*", SYMLINK+="%c"
dde05ccb 309EOF
912541b0
KS
310 },
311 {
312 desc => "program with lots of arguments",
313 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
314 exp_name => "foo9" ,
315 rules => <<EOF
d914e445 316SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n foo3 foo4 foo5 foo6 foo7 foo8 foo9", KERNEL=="sda5", SYMLINK+="%c{7}"
35b38379 317EOF
912541b0
KS
318 },
319 {
320 desc => "program with subshell",
321 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
322 exp_name => "bar9" ,
323 rules => <<EOF
d914e445 324SUBSYSTEMS=="scsi", PROGRAM=="/bin/sh -c 'echo foo3 foo4 foo5 foo6 foo7 foo8 foo9 | sed s/foo9/bar9/'", KERNEL=="sda5", SYMLINK+="%c{7}"
35b38379 325EOF
912541b0
KS
326 },
327 {
328 desc => "program arguments combined with apostrophes",
329 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
330 exp_name => "foo7" ,
331 rules => <<EOF
d914e445 332SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n 'foo3 foo4' 'foo5 foo6 foo7 foo8'", KERNEL=="sda5", SYMLINK+="%c{5}"
7e760b79
FB
333EOF
334 },
335 {
336 desc => "program arguments combined with escaped double quotes, part 1",
337 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
338 exp_name => "foo2" ,
339 rules => <<EOF
340SUBSYSTEMS=="scsi", PROGRAM=="/bin/sh -c 'printf %%s \\\"foo1 foo2\\\" | grep \\\"foo1 foo2\\\"'", KERNEL=="sda5", SYMLINK+="%c{2}"
341EOF
342 },
343 {
344 desc => "program arguments combined with escaped double quotes, part 2",
345 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
346 exp_name => "foo2" ,
347 rules => <<EOF
348SUBSYSTEMS=="scsi", PROGRAM=="/bin/sh -c \\\"printf %%s 'foo1 foo2' | grep 'foo1 foo2'\\\"", KERNEL=="sda5", SYMLINK+="%c{2}"
349EOF
350 },
351 {
352 desc => "program arguments combined with escaped double quotes, part 3",
353 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
354 exp_name => "foo2" ,
355 rules => <<EOF
356SUBSYSTEMS=="scsi", PROGRAM=="/bin/sh -c 'printf \\\"%%s %%s\\\" \\\"foo1 foo2\\\" \\\"foo3\\\"| grep \\\"foo1 foo2\\\"'", KERNEL=="sda5", SYMLINK+="%c{2}"
56c963dc 357EOF
912541b0
KS
358 },
359 {
360 desc => "characters before the %c{N} substitution",
361 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
362 exp_name => "my-foo9" ,
363 rules => <<EOF
d914e445 364SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n foo3 foo4 foo5 foo6 foo7 foo8 foo9", KERNEL=="sda5", SYMLINK+="my-%c{7}"
56c963dc 365EOF
912541b0
KS
366 },
367 {
368 desc => "substitute the second to last argument",
369 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
370 exp_name => "my-foo8" ,
371 rules => <<EOF
d914e445 372SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n foo3 foo4 foo5 foo6 foo7 foo8 foo9", KERNEL=="sda5", SYMLINK+="my-%c{6}"
bf5d2964 373EOF
912541b0
KS
374 },
375 {
376 desc => "test substitution by variable name",
377 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
378 exp_name => "Major:8-minor:5-kernelnumber:5-id:0:0:0:0",
379 rules => <<EOF
5754e74c 380SUBSYSTEMS=="scsi", KERNELS=="0:0:0:0", SYMLINK+="Major:\$major-minor:\$minor-kernelnumber:\$number-id:\$id"
bf5d2964 381EOF
912541b0
KS
382 },
383 {
384 desc => "test substitution by variable name 2",
385 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
386 exp_name => "Major:8-minor:5-kernelnumber:5-id:0:0:0:0",
387 rules => <<EOF
5754e74c 388SUBSYSTEMS=="scsi", KERNELS=="0:0:0:0", DEVPATH=="*/sda/*", SYMLINK+="Major:\$major-minor:%m-kernelnumber:\$number-id:\$id"
bf5d2964 389EOF
912541b0
KS
390 },
391 {
392 desc => "test substitution by variable name 3",
393 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
394 exp_name => "850:0:0:05" ,
395 rules => <<EOF
5754e74c 396SUBSYSTEMS=="scsi", KERNELS=="0:0:0:0", DEVPATH=="*/sda/*", SYMLINK+="%M%m%b%n"
bf5d2964 397EOF
912541b0
KS
398 },
399 {
400 desc => "test substitution by variable name 4",
401 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
402 exp_name => "855" ,
403 rules => <<EOF
5754e74c 404SUBSYSTEMS=="scsi", KERNELS=="0:0:0:0", DEVPATH=="*/sda/*", SYMLINK+="\$major\$minor\$number"
bf5d2964 405EOF
912541b0
KS
406 },
407 {
408 desc => "test substitution by variable name 5",
409 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
410 exp_name => "8550:0:0:0" ,
411 rules => <<EOF
5754e74c 412SUBSYSTEMS=="scsi", KERNELS=="0:0:0:0", DEVPATH=="*/sda/*", SYMLINK+="\$major%m%n\$id"
8ff8bbba 413EOF
912541b0
KS
414 },
415 {
416 desc => "non matching SUBSYSTEMS for device with no parent",
417 devpath => "/devices/virtual/tty/console",
418 exp_name => "TTY",
419 rules => <<EOF
d914e445 420SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n foo", RESULT=="foo", SYMLINK+="foo"
5754e74c 421KERNEL=="console", SYMLINK+="TTY"
1d936fbc 422EOF
912541b0
KS
423 },
424 {
425 desc => "non matching SUBSYSTEMS",
426 devpath => "/devices/virtual/tty/console",
427 exp_name => "TTY" ,
428 rules => <<EOF
5754e74c
KS
429SUBSYSTEMS=="foo", ATTRS{dev}=="5:1", SYMLINK+="foo"
430KERNEL=="console", SYMLINK+="TTY"
64682333 431EOF
912541b0
KS
432 },
433 {
434 desc => "ATTRS match",
435 devpath => "/devices/virtual/tty/console",
436 exp_name => "foo" ,
437 rules => <<EOF
5754e74c
KS
438KERNEL=="console", SYMLINK+="TTY"
439ATTRS{dev}=="5:1", SYMLINK+="foo"
a402404f 440EOF
912541b0
KS
441 },
442 {
443 desc => "ATTR (empty file)",
444 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
445 exp_name => "empty" ,
446 rules => <<EOF
5754e74c
KS
447KERNEL=="sda", ATTR{test_empty_file}=="?*", SYMLINK+="something"
448KERNEL=="sda", ATTR{test_empty_file}!="", SYMLINK+="not-empty"
449KERNEL=="sda", ATTR{test_empty_file}=="", SYMLINK+="empty"
450KERNEL=="sda", ATTR{test_empty_file}!="?*", SYMLINK+="not-something"
a402404f 451EOF
912541b0
KS
452 },
453 {
454 desc => "ATTR (non-existent file)",
455 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
456 exp_name => "non-existent" ,
457 rules => <<EOF
5754e74c
KS
458KERNEL=="sda", ATTR{nofile}=="?*", SYMLINK+="something"
459KERNEL=="sda", ATTR{nofile}!="", SYMLINK+="not-empty"
460KERNEL=="sda", ATTR{nofile}=="", SYMLINK+="empty"
461KERNEL=="sda", ATTR{nofile}!="?*", SYMLINK+="not-something"
462KERNEL=="sda", TEST!="nofile", SYMLINK+="non-existent"
463KERNEL=="sda", SYMLINK+="wrong"
772558f4 464EOF
912541b0
KS
465 },
466 {
467 desc => "program and bus type match",
468 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
469 exp_name => "scsi-0:0:0:0" ,
470 rules => <<EOF
d914e445
KS
471SUBSYSTEMS=="usb", PROGRAM=="/bin/echo -n usb-%b", SYMLINK+="%c"
472SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n scsi-%b", SYMLINK+="%c"
473SUBSYSTEMS=="foo", PROGRAM=="/bin/echo -n foo-%b", SYMLINK+="%c"
50e5de03 474EOF
912541b0
KS
475 },
476 {
477 desc => "sysfs parent hierarchy",
478 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
479 exp_name => "modem" ,
480 rules => <<EOF
5754e74c 481ATTRS{idProduct}=="007b", SYMLINK+="modem"
f0142622 482EOF
912541b0
KS
483 },
484 {
485 desc => "name test with ! in the name",
486 devpath => "/devices/virtual/block/fake!blockdev0",
487 exp_name => "is/a/fake/blockdev0" ,
488 rules => <<EOF
5754e74c
KS
489SUBSYSTEMS=="scsi", SYMLINK+="is/not/a/%k"
490SUBSYSTEM=="block", SYMLINK+="is/a/%k"
491KERNEL=="ttyACM0", SYMLINK+="modem"
b9fc973b 492EOF
912541b0
KS
493 },
494 {
495 desc => "name test with ! in the name, but no matching rule",
496 devpath => "/devices/virtual/block/fake!blockdev0",
497 exp_name => "fake/blockdev0" ,
498 exp_rem_error => "yes",
499 rules => <<EOF
5754e74c 500KERNEL=="ttyACM0", SYMLINK+="modem"
93656247 501EOF
912541b0
KS
502 },
503 {
504 desc => "KERNELS rule",
505 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
506 exp_name => "scsi-0:0:0:0",
507 rules => <<EOF
5754e74c
KS
508SUBSYSTEMS=="usb", KERNELS=="0:0:0:0", SYMLINK+="not-scsi"
509SUBSYSTEMS=="scsi", KERNELS=="0:0:0:1", SYMLINK+="no-match"
510SUBSYSTEMS=="scsi", KERNELS==":0", SYMLINK+="short-id"
511SUBSYSTEMS=="scsi", KERNELS=="/0:0:0:0", SYMLINK+="no-match"
512SUBSYSTEMS=="scsi", KERNELS=="0:0:0:0", SYMLINK+="scsi-0:0:0:0"
93656247 513EOF
912541b0
KS
514 },
515 {
516 desc => "KERNELS wildcard all",
517 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
518 exp_name => "scsi-0:0:0:0",
519 rules => <<EOF
5754e74c
KS
520SUBSYSTEMS=="scsi", KERNELS=="*:1", SYMLINK+="no-match"
521SUBSYSTEMS=="scsi", KERNELS=="*:0:1", SYMLINK+="no-match"
522SUBSYSTEMS=="scsi", KERNELS=="*:0:0:1", SYMLINK+="no-match"
523SUBSYSTEMS=="scsi", KERNEL=="0:0:0:0", SYMLINK+="before"
524SUBSYSTEMS=="scsi", KERNELS=="*", SYMLINK+="scsi-0:0:0:0"
93656247 525EOF
912541b0
KS
526 },
527 {
528 desc => "KERNELS wildcard partial",
529 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
530 exp_name => "scsi-0:0:0:0",
531 rules => <<EOF
5754e74c
KS
532SUBSYSTEMS=="scsi", KERNELS=="0:0:0:0", SYMLINK+="before"
533SUBSYSTEMS=="scsi", KERNELS=="*:0", SYMLINK+="scsi-0:0:0:0"
93656247 534EOF
912541b0
KS
535 },
536 {
537 desc => "KERNELS wildcard partial 2",
538 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
539 exp_name => "scsi-0:0:0:0",
540 rules => <<EOF
5754e74c
KS
541SUBSYSTEMS=="scsi", KERNELS=="0:0:0:0", SYMLINK+="before"
542SUBSYSTEMS=="scsi", KERNELS=="*:0:0:0", SYMLINK+="scsi-0:0:0:0"
eef54479 543EOF
912541b0
KS
544 },
545 {
546 desc => "substitute attr with link target value (first match)",
547 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
548 exp_name => "driver-is-sd",
549 rules => <<EOF
5754e74c 550SUBSYSTEMS=="scsi", SYMLINK+="driver-is-\$attr{driver}"
eef54479 551EOF
912541b0
KS
552 },
553 {
554 desc => "substitute attr with link target value (currently selected device)",
555 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
556 exp_name => "driver-is-ahci",
557 rules => <<EOF
5754e74c 558SUBSYSTEMS=="pci", SYMLINK+="driver-is-\$attr{driver}"
d5f91372 559EOF
912541b0
KS
560 },
561 {
562 desc => "ignore ATTRS attribute whitespace",
563 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
564 exp_name => "ignored",
565 rules => <<EOF
5754e74c 566SUBSYSTEMS=="scsi", ATTRS{whitespace_test}=="WHITE SPACE", SYMLINK+="ignored"
d5f91372 567EOF
912541b0
KS
568 },
569 {
570 desc => "do not ignore ATTRS attribute whitespace",
571 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
572 exp_name => "matched-with-space",
573 rules => <<EOF
5754e74c
KS
574SUBSYSTEMS=="scsi", ATTRS{whitespace_test}=="WHITE SPACE ", SYMLINK+="wrong-to-ignore"
575SUBSYSTEMS=="scsi", ATTRS{whitespace_test}=="WHITE SPACE ", SYMLINK+="matched-with-space"
0a5417a0 576EOF
912541b0
KS
577 },
578 {
579 desc => "permissions USER=bad GROUP=name",
580 devpath => "/devices/virtual/tty/tty33",
581 exp_name => "tty33",
582 exp_perms => "0:0:0600",
583 rules => <<EOF
6ada823a 584KERNEL=="tty33", OWNER="bad", GROUP="name"
b8669191 585EOF
912541b0
KS
586 },
587 {
9158d03e 588 desc => "permissions OWNER=1",
912541b0
KS
589 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
590 exp_name => "node",
9158d03e 591 exp_perms => "1::0600",
912541b0 592 rules => <<EOF
9158d03e 593SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="node", OWNER="1"
b8669191 594EOF
912541b0
KS
595 },
596 {
9158d03e 597 desc => "permissions GROUP=1",
912541b0
KS
598 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
599 exp_name => "node",
9158d03e 600 exp_perms => ":1:0660",
912541b0 601 rules => <<EOF
9158d03e 602SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="node", GROUP="1"
9b434de1 603EOF
912541b0
KS
604 },
605 {
606 desc => "textual user id",
607 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
608 exp_name => "node",
609 exp_perms => "nobody::0600",
610 rules => <<EOF
5754e74c 611SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="node", OWNER="nobody"
9b434de1 612EOF
912541b0
KS
613 },
614 {
615 desc => "textual group id",
616 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
617 exp_name => "node",
618 exp_perms => ":daemon:0660",
619 rules => <<EOF
5754e74c 620SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="node", GROUP="daemon"
c8278763 621EOF
912541b0
KS
622 },
623 {
624 desc => "textual user/group id",
625 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
626 exp_name => "node",
627 exp_perms => "root:mail:0660",
628 rules => <<EOF
5754e74c 629SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="node", OWNER="root", GROUP="mail"
b8669191 630EOF
912541b0
KS
631 },
632 {
633 desc => "permissions MODE=0777",
634 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
635 exp_name => "node",
636 exp_perms => "::0777",
637 rules => <<EOF
5754e74c 638SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="node", MODE="0777"
b8669191 639EOF
912541b0
KS
640 },
641 {
9158d03e 642 desc => "permissions OWNER=1 GROUP=1 MODE=0777",
912541b0
KS
643 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
644 exp_name => "node",
9158d03e 645 exp_perms => "1:1:0777",
912541b0 646 rules => <<EOF
9158d03e 647SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="node", OWNER="1", GROUP="1", MODE="0777"
b8669191 648EOF
912541b0
KS
649 },
650 {
9158d03e 651 desc => "permissions OWNER to 1",
912541b0
KS
652 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
653 exp_name => "ttyACM0",
9158d03e 654 exp_perms => "1::",
912541b0 655 rules => <<EOF
9158d03e 656KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n", OWNER="1"
b8669191 657EOF
912541b0
KS
658 },
659 {
9158d03e 660 desc => "permissions GROUP to 1",
912541b0
KS
661 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
662 exp_name => "ttyACM0",
9158d03e 663 exp_perms => ":1:0660",
912541b0 664 rules => <<EOF
9158d03e 665KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n", GROUP="1"
b8669191 666EOF
912541b0
KS
667 },
668 {
669 desc => "permissions MODE to 0060",
670 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
671 exp_name => "ttyACM0",
672 exp_perms => "::0060",
673 rules => <<EOF
5754e74c 674KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n", MODE="0060"
b8669191 675EOF
912541b0
KS
676 },
677 {
678 desc => "permissions OWNER, GROUP, MODE",
679 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
680 exp_name => "ttyACM0",
9158d03e 681 exp_perms => "1:1:0777",
912541b0 682 rules => <<EOF
9158d03e 683KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n", OWNER="1", GROUP="1", MODE="0777"
e9390146 684EOF
912541b0
KS
685 },
686 {
687 desc => "permissions only rule",
688 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
689 exp_name => "ttyACM0",
9158d03e 690 exp_perms => "1:1:0777",
912541b0 691 rules => <<EOF
9158d03e
TG
692KERNEL=="ttyACM[0-9]*", OWNER="1", GROUP="1", MODE="0777"
693KERNEL=="ttyUSX[0-9]*", OWNER="2", GROUP="2", MODE="0444"
5754e74c 694KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n"
eb870090 695EOF
912541b0
KS
696 },
697 {
698 desc => "multiple permissions only rule",
699 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
700 exp_name => "ttyACM0",
9158d03e 701 exp_perms => "1:1:0777",
912541b0 702 rules => <<EOF
9158d03e
TG
703SUBSYSTEM=="tty", OWNER="1"
704SUBSYSTEM=="tty", GROUP="1"
28ce66de 705SUBSYSTEM=="tty", MODE="0777"
9158d03e 706KERNEL=="ttyUSX[0-9]*", OWNER="2", GROUP="2", MODE="0444"
5754e74c 707KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n"
eb870090 708EOF
912541b0
KS
709 },
710 {
711 desc => "permissions only rule with override at SYMLINK+ rule",
712 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
713 exp_name => "ttyACM0",
9158d03e 714 exp_perms => "1:2:0777",
912541b0 715 rules => <<EOF
9158d03e
TG
716SUBSYSTEM=="tty", OWNER="1"
717SUBSYSTEM=="tty", GROUP="1"
28ce66de 718SUBSYSTEM=="tty", MODE="0777"
9158d03e
TG
719KERNEL=="ttyUSX[0-9]*", OWNER="2", GROUP="2", MODE="0444"
720KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n", GROUP="2"
fa19f181 721EOF
912541b0
KS
722 },
723 {
724 desc => "major/minor number test",
725 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
726 exp_name => "node",
727 exp_majorminor => "8:0",
728 rules => <<EOF
5754e74c 729SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="node"
7d12d4e1 730EOF
912541b0
KS
731 },
732 {
733 desc => "big major number test",
734 devpath => "/devices/virtual/misc/misc-fake1",
735 exp_name => "node",
21d9e3f3 736 exp_majorminor => "4095:1",
912541b0 737 rules => <<EOF
5754e74c 738KERNEL=="misc-fake1", SYMLINK+="node"
7d12d4e1 739EOF
912541b0
KS
740 },
741 {
742 desc => "big major and big minor number test",
743 devpath => "/devices/virtual/misc/misc-fake89999",
744 exp_name => "node",
21d9e3f3 745 exp_majorminor => "4095:89999",
912541b0 746 rules => <<EOF
5754e74c 747KERNEL=="misc-fake89999", SYMLINK+="node"
2b0f835c 748EOF
912541b0
KS
749 },
750 {
751 desc => "multiple symlinks with format char",
752 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
753 exp_name => "symlink2-ttyACM0",
754 rules => <<EOF
5754e74c 755KERNEL=="ttyACM[0-9]*", SYMLINK="symlink1-%n symlink2-%k symlink3-%b"
7b2bdb4b 756EOF
912541b0
KS
757 },
758 {
759 desc => "multiple symlinks with a lot of s p a c e s",
760 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
761 exp_name => "one",
762 not_exp_name => " ",
763 rules => <<EOF
5754e74c 764KERNEL=="ttyACM[0-9]*", SYMLINK=" one two "
9cd7b128
DS
765EOF
766 },
767 {
768 desc => "symlink with spaces in substituted variable",
769 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
770 exp_name => "name-one_two_three-end",
771 not_exp_name => " ",
772 rules => <<EOF
773ENV{WITH_WS}="one two three"
774SYMLINK="name-\$env{WITH_WS}-end"
775EOF
776 },
777 {
778 desc => "symlink with leading space in substituted variable",
779 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
780 exp_name => "name-one_two_three-end",
781 not_exp_name => " ",
782 rules => <<EOF
783ENV{WITH_WS}=" one two three"
784SYMLINK="name-\$env{WITH_WS}-end"
785EOF
786 },
787 {
788 desc => "symlink with trailing space in substituted variable",
789 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
790 exp_name => "name-one_two_three-end",
791 not_exp_name => " ",
792 rules => <<EOF
793ENV{WITH_WS}="one two three "
794SYMLINK="name-\$env{WITH_WS}-end"
795EOF
796 },
797 {
798 desc => "symlink with lots of space in substituted variable",
799 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
800 exp_name => "name-one_two_three-end",
801 not_exp_name => " ",
802 rules => <<EOF
803ENV{WITH_WS}=" one two three "
804SYMLINK="name-\$env{WITH_WS}-end"
805EOF
806 },
807 {
808 desc => "symlink with multiple spaces in substituted variable",
809 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
810 exp_name => "name-one_two_three-end",
811 not_exp_name => " ",
812 rules => <<EOF
813ENV{WITH_WS}=" one two three "
814SYMLINK="name-\$env{WITH_WS}-end"
815EOF
816 },
817 {
818 desc => "symlink with space and var with space, part 1",
819 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
820 exp_name => "first",
821 not_exp_name => " ",
822 rules => <<EOF
823ENV{WITH_WS}=" one two three "
824SYMLINK=" first name-\$env{WITH_WS}-end another_symlink a b c "
825EOF
826 },
827 {
828 desc => "symlink with space and var with space, part 2",
829 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
830 exp_name => "name-one_two_three-end",
831 not_exp_name => " ",
832 rules => <<EOF
833ENV{WITH_WS}=" one two three "
834SYMLINK=" first name-\$env{WITH_WS}-end another_symlink a b c "
835EOF
836 },
837 {
838 desc => "symlink with space and var with space, part 3",
839 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
840 exp_name => "another_symlink",
841 not_exp_name => " ",
842 rules => <<EOF
843ENV{WITH_WS}=" one two three "
844SYMLINK=" first name-\$env{WITH_WS}-end another_symlink a b c "
b8669191 845EOF
912541b0
KS
846 },
847 {
848 desc => "symlink creation (same directory)",
849 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
850 exp_name => "modem0",
851 rules => <<EOF
5754e74c 852KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n", SYMLINK="modem%n"
b8669191 853EOF
912541b0
KS
854 },
855 {
856 desc => "multiple symlinks",
857 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
858 exp_name => "second-0" ,
859 rules => <<EOF
5754e74c 860KERNEL=="ttyACM0", SYMLINK="first-%n second-%n third-%n"
b8669191 861EOF
912541b0
KS
862 },
863 {
864 desc => "symlink name '.'",
865 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
866 exp_name => ".",
867 exp_add_error => "yes",
868 exp_rem_error => "yes",
869 rules => <<EOF
5754e74c 870SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="."
b8669191 871EOF
912541b0
KS
872 },
873 {
874 desc => "symlink node to itself",
875 devpath => "/devices/virtual/tty/tty0",
876 exp_name => "link",
877 exp_add_error => "yes",
878 exp_rem_error => "yes",
879 option => "clean",
880 rules => <<EOF
5754e74c 881KERNEL=="tty0", SYMLINK+="tty0"
b8669191 882EOF
912541b0
KS
883 },
884 {
885 desc => "symlink %n substitution",
886 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
887 exp_name => "symlink0",
888 rules => <<EOF
5754e74c 889KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n", SYMLINK+="symlink%n"
b8669191 890EOF
912541b0
KS
891 },
892 {
893 desc => "symlink %k substitution",
894 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
895 exp_name => "symlink-ttyACM0",
896 rules => <<EOF
5754e74c 897KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n", SYMLINK+="symlink-%k"
b8669191 898EOF
912541b0
KS
899 },
900 {
901 desc => "symlink %M:%m substitution",
902 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
903 exp_name => "major-166:0",
904 rules => <<EOF
5754e74c 905KERNEL=="ttyACM[0-9]*", SYMLINK+="ttyACM%n", SYMLINK+="major-%M:%m"
b8669191 906EOF
912541b0
KS
907 },
908 {
909 desc => "symlink %b substitution",
910 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
911 exp_name => "symlink-0:0:0:0",
912 rules => <<EOF
220893b3 913SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="symlink-%b"
b8669191 914EOF
912541b0
KS
915 },
916 {
917 desc => "symlink %c substitution",
918 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
919 exp_name => "test",
920 rules => <<EOF
d914e445 921KERNEL=="ttyACM[0-9]*", PROGRAM=="/bin/echo test", SYMLINK+="%c"
b8669191 922EOF
912541b0
KS
923 },
924 {
925 desc => "symlink %c{N} substitution",
926 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
927 exp_name => "test",
928 rules => <<EOF
d914e445 929KERNEL=="ttyACM[0-9]*", PROGRAM=="/bin/echo symlink test this", SYMLINK+="%c{2}"
b8669191 930EOF
912541b0
KS
931 },
932 {
933 desc => "symlink %c{N+} substitution",
934 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
935 exp_name => "this",
936 rules => <<EOF
d914e445 937KERNEL=="ttyACM[0-9]*", PROGRAM=="/bin/echo symlink test this", SYMLINK+="%c{2+}"
b8669191 938EOF
912541b0
KS
939 },
940 {
941 desc => "symlink only rule with %c{N+}",
942 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
943 exp_name => "test",
944 rules => <<EOF
d914e445 945SUBSYSTEMS=="scsi", KERNEL=="sda", PROGRAM=="/bin/echo link test this" SYMLINK+="%c{2+}"
b8669191 946EOF
912541b0
KS
947 },
948 {
949 desc => "symlink %s{filename} substitution",
950 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
951 exp_name => "166:0",
952 rules => <<EOF
5754e74c 953KERNEL=="ttyACM[0-9]*", SYMLINK+="%s{dev}"
b8669191 954EOF
912541b0
KS
955 },
956 {
957 desc => "program result substitution (numbered part of)",
958 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
959 exp_name => "link1",
960 rules => <<EOF
d914e445 961SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n node link1 link2", RESULT=="node *", SYMLINK+="%c{2} %c{3}"
b8669191 962EOF
912541b0
KS
963 },
964 {
965 desc => "program result substitution (numbered part of+)",
966 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda5",
967 exp_name => "link4",
968 rules => <<EOF
d914e445 969SUBSYSTEMS=="scsi", PROGRAM=="/bin/echo -n node link1 link2 link3 link4", RESULT=="node *", SYMLINK+="%c{2+}"
7efa217d 970EOF
912541b0
KS
971 },
972 {
973 desc => "SUBSYSTEM match test",
974 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
975 exp_name => "node",
976 rules => <<EOF
5754e74c
KS
977SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="should_not_match", SUBSYSTEM=="vc"
978SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="node", SUBSYSTEM=="block"
979SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="should_not_match2", SUBSYSTEM=="vc"
2092fbcd 980EOF
912541b0
KS
981 },
982 {
983 desc => "DRIVERS match test",
984 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
985 exp_name => "node",
986 rules => <<EOF
5754e74c
KS
987SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="should_not_match", DRIVERS=="sd-wrong"
988SUBSYSTEMS=="scsi", KERNEL=="sda", SYMLINK+="node", DRIVERS=="sd"
c1ab0461 989EOF
912541b0
KS
990 },
991 {
992 desc => "devnode substitution test",
993 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
994 exp_name => "node",
995 rules => <<EOF
220893b3 996SUBSYSTEMS=="scsi", KERNEL=="sda", PROGRAM=="/usr/bin/test -b %N" SYMLINK+="node"
69aa6dfb 997EOF
912541b0
KS
998 },
999 {
1000 desc => "parent node name substitution test",
1001 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1002 exp_name => "sda-part-1",
1003 rules => <<EOF
5754e74c 1004SUBSYSTEMS=="scsi", KERNEL=="sda1", SYMLINK+="%P-part-1"
69aa6dfb 1005EOF
912541b0
KS
1006 },
1007 {
1008 desc => "udev_root substitution",
1009 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
4cb72937 1010 exp_name => "start-/dev-end",
912541b0 1011 rules => <<EOF
5754e74c 1012SUBSYSTEMS=="scsi", KERNEL=="sda1", SYMLINK+="start-%r-end"
3b6ed8bb 1013EOF
912541b0
KS
1014 },
1015 {
1016 desc => "last_rule option",
1017 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1018 exp_name => "last",
1019 rules => <<EOF
c4edd0ad 1020SUBSYSTEMS=="scsi", KERNEL=="sda1", SYMLINK+="last", OPTIONS="last_rule"
5754e74c 1021SUBSYSTEMS=="scsi", KERNEL=="sda1", SYMLINK+="very-last"
28ce66de 1022EOF
912541b0
KS
1023 },
1024 {
1025 desc => "negation KERNEL!=",
1026 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1027 exp_name => "match",
1028 rules => <<EOF
5754e74c
KS
1029SUBSYSTEMS=="scsi", KERNEL!="sda1", SYMLINK+="matches-but-is-negated"
1030SUBSYSTEMS=="scsi", KERNEL=="sda1", SYMLINK+="before"
1031SUBSYSTEMS=="scsi", KERNEL!="xsda1", SYMLINK+="match"
28ce66de 1032EOF
912541b0
KS
1033 },
1034 {
1035 desc => "negation SUBSYSTEM!=",
1036 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1037 exp_name => "not-anything",
1038 rules => <<EOF
5754e74c
KS
1039SUBSYSTEMS=="scsi", SUBSYSTEM=="block", KERNEL!="sda1", SYMLINK+="matches-but-is-negated"
1040SUBSYSTEMS=="scsi", KERNEL=="sda1", SYMLINK+="before"
1041SUBSYSTEMS=="scsi", SUBSYSTEM!="anything", SYMLINK+="not-anything"
28ce66de 1042EOF
912541b0
KS
1043 },
1044 {
1045 desc => "negation PROGRAM!= exit code",
1046 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1047 exp_name => "nonzero-program",
1048 rules => <<EOF
5754e74c 1049SUBSYSTEMS=="scsi", KERNEL=="sda1", SYMLINK+="before"
d914e445 1050KERNEL=="sda1", PROGRAM!="/bin/false", SYMLINK+="nonzero-program"
3e5958de 1051EOF
912541b0
KS
1052 },
1053 {
1054 desc => "ENV{} test",
1055 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1056 exp_name => "true",
1057 rules => <<EOF
a1af6b04 1058ENV{ENV_KEY_TEST}="test"
5754e74c
KS
1059SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ENV_KEY_TEST}=="go", SYMLINK+="wrong"
1060SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ENV_KEY_TEST}=="test", SYMLINK+="true"
1061SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ENV_KEY_TEST}=="bad", SYMLINK+="bad"
3e5958de 1062EOF
912541b0
KS
1063 },
1064 {
1065 desc => "ENV{} test",
1066 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1067 exp_name => "true",
1068 rules => <<EOF
a1af6b04 1069ENV{ENV_KEY_TEST}="test"
5754e74c
KS
1070SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ENV_KEY_TEST}=="go", SYMLINK+="wrong"
1071SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ENV_KEY_TEST}=="yes", ENV{ACTION}=="add", ENV{DEVPATH}=="*/block/sda/sdax1", SYMLINK+="no"
1072SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ENV_KEY_TEST}=="test", ENV{ACTION}=="add", ENV{DEVPATH}=="*/block/sda/sda1", SYMLINK+="true"
1073SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ENV_KEY_TEST}=="bad", SYMLINK+="bad"
5618b561 1074EOF
912541b0
KS
1075 },
1076 {
1077 desc => "ENV{} test (assign)",
1078 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1079 exp_name => "true",
1080 rules => <<EOF
c4edd0ad 1081SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ASSIGN}="true"
5754e74c
KS
1082SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ASSIGN}=="yes", SYMLINK+="no"
1083SUBSYSTEMS=="scsi", KERNEL=="sda1", SYMLINK+="before"
1084SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ASSIGN}=="true", SYMLINK+="true"
ac528431 1085EOF
912541b0
KS
1086 },
1087 {
1088 desc => "ENV{} test (assign 2 times)",
1089 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1090 exp_name => "true",
1091 rules => <<EOF
ac528431
KS
1092SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ASSIGN}="true"
1093SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ASSIGN}="absolutely-\$env{ASSIGN}"
5754e74c
KS
1094SUBSYSTEMS=="scsi", KERNEL=="sda1", SYMLINK+="before"
1095SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ASSIGN}=="yes", SYMLINK+="no"
1096SUBSYSTEMS=="scsi", KERNEL=="sda1", ENV{ASSIGN}=="absolutely-true", SYMLINK+="true"
5618b561 1097EOF
912541b0
KS
1098 },
1099 {
1100 desc => "ENV{} test (assign2)",
1101 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1102 exp_name => "part",
1103 rules => <<EOF
d59c84ef
KS
1104SUBSYSTEM=="block", KERNEL=="*[0-9]", ENV{PARTITION}="true", ENV{MAINDEVICE}="false"
1105SUBSYSTEM=="block", KERNEL=="*[!0-9]", ENV{PARTITION}="false", ENV{MAINDEVICE}="true"
5754e74c
KS
1106ENV{MAINDEVICE}=="true", SYMLINK+="disk"
1107SUBSYSTEM=="block", SYMLINK+="before"
1108ENV{PARTITION}=="true", SYMLINK+="part"
18614ab2 1109EOF
912541b0
KS
1110 },
1111 {
1112 desc => "untrusted string sanitize",
1113 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1114 exp_name => "sane",
1115 rules => <<EOF
d914e445 1116SUBSYSTEMS=="scsi", KERNEL=="sda1", PROGRAM=="/bin/echo -e name; (/usr/bin/badprogram)", RESULT=="name_ _/usr/bin/badprogram_", SYMLINK+="sane"
764ce7f2 1117EOF
912541b0
KS
1118 },
1119 {
1120 desc => "untrusted string sanitize (don't replace utf8)",
1121 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1122 exp_name => "uber",
1123 rules => <<EOF
d914e445 1124SUBSYSTEMS=="scsi", KERNEL=="sda1", PROGRAM=="/bin/echo -e \\xc3\\xbcber" RESULT=="\xc3\xbcber", SYMLINK+="uber"
764ce7f2 1125EOF
912541b0
KS
1126 },
1127 {
1128 desc => "untrusted string sanitize (replace invalid utf8)",
1129 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1130 exp_name => "replaced",
1131 rules => <<EOF
d914e445 1132SUBSYSTEMS=="scsi", KERNEL=="sda1", PROGRAM=="/bin/echo -e \\xef\\xe8garbage", RESULT=="__garbage", SYMLINK+="replaced"
98bbc835 1133EOF
912541b0
KS
1134 },
1135 {
1136 desc => "read sysfs value from parent device",
1137 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
1138 exp_name => "serial-354172020305000",
1139 rules => <<EOF
5754e74c 1140KERNEL=="ttyACM*", ATTRS{serial}=="?*", SYMLINK+="serial-%s{serial}"
db949b02 1141EOF
912541b0
KS
1142 },
1143 {
1144 desc => "match against empty key string",
1145 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1146 exp_name => "ok",
1147 rules => <<EOF
5754e74c
KS
1148KERNEL=="sda", ATTRS{nothing}!="", SYMLINK+="not-1-ok"
1149KERNEL=="sda", ATTRS{nothing}=="", SYMLINK+="not-2-ok"
1150KERNEL=="sda", ATTRS{vendor}!="", SYMLINK+="ok"
1151KERNEL=="sda", ATTRS{vendor}=="", SYMLINK+="not-3-ok"
821d0ec8 1152EOF
912541b0
KS
1153 },
1154 {
1155 desc => "check ACTION value",
1156 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1157 exp_name => "ok",
1158 rules => <<EOF
5754e74c
KS
1159ACTION=="unknown", KERNEL=="sda", SYMLINK+="unknown-not-ok"
1160ACTION=="add", KERNEL=="sda", SYMLINK+="ok"
c974742b 1161EOF
912541b0
KS
1162 },
1163 {
1164 desc => "final assignment",
1165 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1166 exp_name => "ok",
1167 exp_perms => "root:tty:0640",
1168 rules => <<EOF
d960ad15 1169KERNEL=="sda", GROUP:="tty"
5754e74c 1170KERNEL=="sda", GROUP="not-ok", MODE="0640", SYMLINK+="ok"
c974742b 1171EOF
912541b0
KS
1172 },
1173 {
1174 desc => "final assignment 2",
1175 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1176 exp_name => "ok",
1177 exp_perms => "root:tty:0640",
1178 rules => <<EOF
d960ad15 1179KERNEL=="sda", GROUP:="tty"
c974742b 1180SUBSYSTEM=="block", MODE:="640"
5754e74c 1181KERNEL=="sda", GROUP="not-ok", MODE="0666", SYMLINK+="ok"
bd0ed2ff 1182EOF
912541b0
KS
1183 },
1184 {
1185 desc => "env substitution",
1186 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1187 exp_name => "node-add-me",
1188 rules => <<EOF
5754e74c 1189KERNEL=="sda", MODE="0666", SYMLINK+="node-\$env{ACTION}-me"
995aec87 1190EOF
912541b0
KS
1191 },
1192 {
1193 desc => "reset list to current value",
1194 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
1195 exp_name => "three",
1196 not_exp_name => "two",
1197 rules => <<EOF
c7fcba1b
KS
1198KERNEL=="ttyACM[0-9]*", SYMLINK+="one"
1199KERNEL=="ttyACM[0-9]*", SYMLINK+="two"
1200KERNEL=="ttyACM[0-9]*", SYMLINK="three"
647f7c49 1201EOF
912541b0
KS
1202 },
1203 {
1204 desc => "test empty SYMLINK+ (empty override)",
1205 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
1206 exp_name => "right",
1207 not_exp_name => "wrong",
1208 rules => <<EOF
5754e74c
KS
1209KERNEL=="ttyACM[0-9]*", SYMLINK+="wrong"
1210KERNEL=="ttyACM[0-9]*", SYMLINK=""
1211KERNEL=="ttyACM[0-9]*", SYMLINK+="right"
0cd4ac47 1212EOF
912541b0
KS
1213 },
1214 {
1215 desc => "test multi matches",
1216 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
1217 exp_name => "right",
1218 rules => <<EOF
5754e74c
KS
1219KERNEL=="ttyACM*", SYMLINK+="before"
1220KERNEL=="ttyACM*|nothing", SYMLINK+="right"
0cd4ac47 1221EOF
912541b0
KS
1222 },
1223 {
1224 desc => "test multi matches 2",
1225 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
1226 exp_name => "right",
1227 rules => <<EOF
5754e74c
KS
1228KERNEL=="dontknow*|*nothing", SYMLINK+="nomatch"
1229KERNEL=="ttyACM*", SYMLINK+="before"
1230KERNEL=="dontknow*|ttyACM*|nothing*", SYMLINK+="right"
91a75e4a 1231EOF
912541b0
KS
1232 },
1233 {
1234 desc => "test multi matches 3",
1235 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
1236 exp_name => "right",
1237 rules => <<EOF
5754e74c
KS
1238KERNEL=="dontknow|nothing", SYMLINK+="nomatch"
1239KERNEL=="dontknow|ttyACM0a|nothing|attyACM0", SYMLINK+="wrong1"
1240KERNEL=="X|attyACM0|dontknow|ttyACM0a|nothing|attyACM0", SYMLINK+="wrong2"
1241KERNEL=="dontknow|ttyACM0|nothing", SYMLINK+="right"
91a75e4a 1242EOF
912541b0
KS
1243 },
1244 {
1245 desc => "test multi matches 4",
1246 devpath => "/devices/pci0000:00/0000:00:1d.7/usb5/5-2/5-2:1.0/tty/ttyACM0",
1247 exp_name => "right",
1248 rules => <<EOF
5754e74c
KS
1249KERNEL=="dontknow|nothing", SYMLINK+="nomatch"
1250KERNEL=="dontknow|ttyACM0a|nothing|attyACM0", SYMLINK+="wrong1"
1251KERNEL=="X|attyACM0|dontknow|ttyACM0a|nothing|attyACM0", SYMLINK+="wrong2"
1252KERNEL=="all|dontknow|ttyACM0", SYMLINK+="right"
1253KERNEL=="ttyACM0a|nothing", SYMLINK+="wrong3"
0bfb84e1 1254EOF
912541b0
KS
1255 },
1256 {
1257 desc => "IMPORT parent test sequence 1/2 (keep)",
1258 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1259 exp_name => "parent",
1260 option => "keep",
1261 rules => <<EOF
d914e445 1262KERNEL=="sda", IMPORT{program}="/bin/echo -e \'PARENT_KEY=parent_right\\nWRONG_PARENT_KEY=parent_wrong'"
5754e74c 1263KERNEL=="sda", SYMLINK+="parent"
0bfb84e1 1264EOF
912541b0
KS
1265 },
1266 {
1267 desc => "IMPORT parent test sequence 2/2 (keep)",
1268 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1269 exp_name => "parentenv-parent_right",
1270 option => "clean",
1271 rules => <<EOF
5754e74c 1272KERNEL=="sda1", IMPORT{parent}="PARENT*", SYMLINK+="parentenv-\$env{PARENT_KEY}\$env{WRONG_PARENT_KEY}"
594dd610 1273EOF
912541b0
KS
1274 },
1275 {
1276 desc => "GOTO test",
1277 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1278 exp_name => "right",
1279 rules => <<EOF
594dd610 1280KERNEL=="sda1", GOTO="TEST"
5754e74c 1281KERNEL=="sda1", SYMLINK+="wrong"
6880b25d 1282KERNEL=="sda1", GOTO="BAD"
5754e74c
KS
1283KERNEL=="sda1", SYMLINK+="", LABEL="NO"
1284KERNEL=="sda1", SYMLINK+="right", LABEL="TEST", GOTO="end"
1285KERNEL=="sda1", SYMLINK+="wrong2", LABEL="BAD"
9dae0e89 1286LABEL="end"
0c377989 1287EOF
912541b0
KS
1288 },
1289 {
1290 desc => "GOTO label does not exist",
1291 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1292 exp_name => "right",
1293 rules => <<EOF
0c377989 1294KERNEL=="sda1", GOTO="does-not-exist"
5754e74c 1295KERNEL=="sda1", SYMLINK+="right",
0c377989 1296LABEL="exists"
d59c84ef 1297EOF
912541b0
KS
1298 },
1299 {
1300 desc => "SYMLINK+ compare test",
1301 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1302 exp_name => "right",
1303 not_exp_name => "wrong",
1304 rules => <<EOF
5754e74c
KS
1305KERNEL=="sda1", SYMLINK+="link"
1306KERNEL=="sda1", SYMLINK=="link*", SYMLINK+="right"
1307KERNEL=="sda1", SYMLINK=="nolink*", SYMLINK+="wrong"
d59c84ef 1308EOF
912541b0
KS
1309 },
1310 {
1311 desc => "invalid key operation",
1312 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1313 exp_name => "yes",
1314 rules => <<EOF
5754e74c
KS
1315KERNEL="sda1", SYMLINK+="no"
1316KERNEL=="sda1", SYMLINK+="yes"
864b9b5e 1317EOF
912541b0
KS
1318 },
1319 {
1320 desc => "operator chars in attribute",
1321 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1322 exp_name => "yes",
1323 rules => <<EOF
5754e74c 1324KERNEL=="sda", ATTR{test:colon+plus}=="?*", SYMLINK+="yes"
d4ae9925 1325EOF
912541b0
KS
1326 },
1327 {
1328 desc => "overlong comment line",
1329 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda/sda1",
1330 exp_name => "yes",
1331 rules => <<EOF
d4ae9925 1332# 012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789
d960ad15 1333 # 012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789
5754e74c
KS
1334KERNEL=="sda1", SYMLINK+=="no"
1335KERNEL=="sda1", SYMLINK+="yes"
4ad47b2d 1336EOF
912541b0
KS
1337 },
1338 {
1339 desc => "magic subsys/kernel lookup",
1340 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1341 exp_name => "00:16:41:e2:8d:ff",
1342 rules => <<EOF
5754e74c 1343KERNEL=="sda", SYMLINK+="\$attr{[net/eth0]address}"
03f65fe6 1344EOF
912541b0
KS
1345 },
1346 {
1347 desc => "TEST absolute path",
1348 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1349 exp_name => "there",
1350 rules => <<EOF
02cd084d
TG
1351TEST=="/etc/machine-id", SYMLINK+="there"
1352TEST!="/etc/machine-id", SYMLINK+="notthere"
03f65fe6 1353EOF
912541b0
KS
1354 },
1355 {
1356 desc => "TEST subsys/kernel lookup",
1357 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1358 exp_name => "yes",
1359 rules => <<EOF
5754e74c 1360KERNEL=="sda", TEST=="[net/eth0]", SYMLINK+="yes"
03f65fe6 1361EOF
912541b0
KS
1362 },
1363 {
1364 desc => "TEST relative path",
1365 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1366 exp_name => "relative",
1367 rules => <<EOF
5754e74c 1368KERNEL=="sda", TEST=="size", SYMLINK+="relative"
0ea5e96e 1369EOF
912541b0
KS
1370 },
1371 {
1372 desc => "TEST wildcard substitution (find queue/nr_requests)",
1373 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1374 exp_name => "found-subdir",
1375 rules => <<EOF
5754e74c 1376KERNEL=="sda", TEST=="*/nr_requests", SYMLINK+="found-subdir"
cf100ca7 1377EOF
912541b0
KS
1378 },
1379 {
1380 desc => "TEST MODE=0000",
1381 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1382 exp_name => "sda",
1383 exp_perms => "0:0:0000",
1384 exp_rem_error => "yes",
1385 rules => <<EOF
cf100ca7 1386KERNEL=="sda", MODE="0000"
a367f04e 1387EOF
912541b0
KS
1388 },
1389 {
1390 desc => "TEST PROGRAM feeds OWNER, GROUP, MODE",
1391 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1392 exp_name => "sda",
9158d03e 1393 exp_perms => "1:1:0400",
912541b0
KS
1394 exp_rem_error => "yes",
1395 rules => <<EOF
6880b25d 1396KERNEL=="sda", MODE="666"
9158d03e 1397KERNEL=="sda", PROGRAM=="/bin/echo 1 1 0400", OWNER="%c{1}", GROUP="%c{2}", MODE="%c{3}"
ff94cec3 1398EOF
912541b0
KS
1399 },
1400 {
1401 desc => "TEST PROGRAM feeds MODE with overflow",
1402 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1403 exp_name => "sda",
1404 exp_perms => "0:0:0440",
1405 exp_rem_error => "yes",
1406 rules => <<EOF
6880b25d 1407KERNEL=="sda", MODE="440"
d914e445 1408KERNEL=="sda", PROGRAM=="/bin/echo 0 0 0400letsdoabuffferoverflow0123456789012345789012345678901234567890", OWNER="%c{1}", GROUP="%c{2}", MODE="%c{3}"
dc4c7e46 1409EOF
912541b0
KS
1410 },
1411 {
1412 desc => "magic [subsys/sysname] attribute substitution",
1413 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1414 exp_name => "sda-8741C4G-end",
1415 exp_perms => "0:0:0600",
1416 rules => <<EOF
d914e445 1417KERNEL=="sda", PROGRAM="/bin/true create-envp"
427e20b2 1418KERNEL=="sda", ENV{TESTENV}="change-envp"
0f52fdee 1419KERNEL=="sda", SYMLINK+="%k-%s{[dmi/id]product_name}-end"
d7867b31 1420EOF
912541b0
KS
1421 },
1422 {
1423 desc => "builtin path_id",
1424 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1425 exp_name => "disk/by-path/pci-0000:00:1f.2-scsi-0:0:0:0",
1426 rules => <<EOF
d7867b31
KS
1427KERNEL=="sda", IMPORT{builtin}="path_id"
1428KERNEL=="sda", ENV{ID_PATH}=="?*", SYMLINK+="disk/by-path/\$env{ID_PATH}"
bf8f7583
MP
1429EOF
1430 },
1431 {
1432 desc => "add and match tag",
1433 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1434 exp_name => "found",
1435 not_exp_name => "bad" ,
1436 rules => <<EOF
1437SUBSYSTEMS=="scsi", ATTRS{vendor}=="ATA", TAG+="green"
1438TAGS=="green", SYMLINK+="found"
1439TAGS=="blue", SYMLINK+="bad"
1440EOF
1441 },
1442 {
1443 desc => "don't crash with lots of tags",
1444 devpath => "/devices/pci0000:00/0000:00:1f.2/host0/target0:0:0/0:0:0:0/block/sda",
1445 exp_name => "found",
1446 rules => $rules_10k_tags . <<EOF
1447TAGS=="test1", TAGS=="test500", TAGS=="test1234", TAGS=="test9999", TAGS=="test10000", SYMLINK+="found"
ff94cec3 1448EOF
912541b0 1449 },
a367f04e
GKH
1450);
1451
a367f04e 1452sub udev {
912541b0 1453 my ($action, $devpath, $rules) = @_;
a367f04e 1454
912541b0 1455 # create temporary rules
6ada823a 1456 system("mkdir", "-p", "$udev_rules_dir");
912541b0
KS
1457 open CONF, ">$udev_rules" || die "unable to create rules file: $udev_rules";
1458 print CONF $$rules;
1459 close CONF;
a367f04e 1460
912541b0 1461 if ($valgrind > 0) {
efc9f703 1462 return system("$udev_bin_valgrind $action $devpath");
333e07b7 1463 } elsif ($gdb > 0) {
efc9f703 1464 return system("$udev_bin_gdb $action $devpath");
587751eb 1465 } elsif ($strace > 0) {
efc9f703 1466 return system("$udev_bin_strace $action $devpath");
912541b0 1467 } else {
efc9f703 1468 return system("$udev_bin", "$action", "$devpath");
912541b0 1469 }
a367f04e
GKH
1470}
1471
e5fbfe0a 1472my $error = 0;
72ffa78d 1473
b8669191 1474sub permissions_test {
912541b0 1475 my($rules, $uid, $gid, $mode) = @_;
b8669191 1476
912541b0
KS
1477 my $wrong = 0;
1478 my $userid;
1479 my $groupid;
9b434de1 1480
912541b0
KS
1481 $rules->{exp_perms} =~ m/^(.*):(.*):(.*)$/;
1482 if ($1 ne "") {
1483 if (defined(getpwnam($1))) {
1484 $userid = int(getpwnam($1));
1485 } else {
1486 $userid = $1;
1487 }
1488 if ($uid != $userid) { $wrong = 1; }
1489 }
1490 if ($2 ne "") {
1491 if (defined(getgrnam($2))) {
1492 $groupid = int(getgrnam($2));
1493 } else {
1494 $groupid = $2;
1495 }
1496 if ($gid != $groupid) { $wrong = 1; }
1497 }
1498 if ($3 ne "") {
1499 if (($mode & 07777) != oct($3)) { $wrong = 1; };
1500 }
1501 if ($wrong == 0) {
1502 print "permissions: ok\n";
1503 } else {
1504 printf " expected permissions are: %s:%s:%#o\n", $1, $2, oct($3);
1505 printf " created permissions are : %i:%i:%#o\n", $uid, $gid, $mode & 07777;
1506 print "permissions: error\n";
1507 $error++;
1508 sleep(1);
1509 }
b8669191
GKH
1510}
1511
1512sub major_minor_test {
912541b0 1513 my($rules, $rdev) = @_;
b8669191 1514
912541b0
KS
1515 my $major = ($rdev >> 8) & 0xfff;
1516 my $minor = ($rdev & 0xff) | (($rdev >> 12) & 0xfff00);
1517 my $wrong = 0;
b8669191 1518
912541b0
KS
1519 $rules->{exp_majorminor} =~ m/^(.*):(.*)$/;
1520 if ($1 ne "") {
1521 if ($major != $1) { $wrong = 1; };
1522 }
1523 if ($2 ne "") {
1524 if ($minor != $2) { $wrong = 1; };
1525 }
1526 if ($wrong == 0) {
1527 print "major:minor: ok\n";
1528 } else {
1529 printf " expected major:minor is: %i:%i\n", $1, $2;
1530 printf " created major:minor is : %i:%i\n", $major, $minor;
1531 print "major:minor: error\n";
1532 $error++;
1533 sleep(1);
1534 }
b8669191
GKH
1535}
1536
6ada823a 1537sub udev_setup {
21d9e3f3
EV
1538 system("umount", $udev_tmpfs);
1539 rmdir($udev_tmpfs);
1540 mkdir($udev_tmpfs) || die "unable to create udev_tmpfs: $udev_tmpfs\n";
1541 system("mount", "-o", "rw,mode=755,nosuid,noexec,nodev", "-t", "tmpfs", "tmpfs", $udev_tmpfs) && die "unable to mount tmpfs";
1542
6ada823a
KS
1543 mkdir($udev_dev) || die "unable to create udev_dev: $udev_dev\n";
1544 # setting group and mode of udev_dev ensures the tests work
912541b0 1545 # even if the parent directory has setgid bit enabled.
6ada823a
KS
1546 chown (0, 0, $udev_dev) || die "unable to chown $udev_dev\n";
1547 chmod (0755, $udev_dev) || die "unable to chmod $udev_dev\n";
1548
21d9e3f3 1549 system("cp", "-r", "test/sys/", $udev_sys) && die "unable to copy test/sys";
2ce8d27b 1550
6ada823a 1551 system("rm", "-rf", "$udev_run");
ff94cec3
EK
1552}
1553
2e317184 1554sub run_test {
912541b0 1555 my ($rules, $number) = @_;
efc9f703 1556 my $rc;
fa19f181 1557
912541b0
KS
1558 print "TEST $number: $rules->{desc}\n";
1559 print "device \'$rules->{devpath}\' expecting node/link \'$rules->{exp_name}\'\n";
b8669191 1560
efc9f703
MP
1561 $rc = udev("add", $rules->{devpath}, \$rules->{rules});
1562 if ($rc != 0) {
1563 print "$udev_bin add failed with code $rc\n";
1564 $error++;
1565 }
912541b0 1566 if (defined($rules->{not_exp_name})) {
6ada823a
KS
1567 if ((-e "$udev_dev/$rules->{not_exp_name}") ||
1568 (-l "$udev_dev/$rules->{not_exp_name}")) {
912541b0
KS
1569 print "nonexistent: error \'$rules->{not_exp_name}\' not expected to be there\n";
1570 $error++;
1571 sleep(1);
1572 }
1573 }
647f7c49 1574
6ada823a
KS
1575 if ((-e "$udev_dev/$rules->{exp_name}") ||
1576 (-l "$udev_dev/$rules->{exp_name}")) {
fa19f181 1577
912541b0 1578 my ($dev, $ino, $mode, $nlink, $uid, $gid, $rdev, $size,
6ada823a 1579 $atime, $mtime, $ctime, $blksize, $blocks) = stat("$udev_dev/$rules->{exp_name}");
fa19f181 1580
912541b0
KS
1581 if (defined($rules->{exp_perms})) {
1582 permissions_test($rules, $uid, $gid, $mode);
1583 }
1584 if (defined($rules->{exp_majorminor})) {
1585 major_minor_test($rules, $rdev);
1586 }
1587 print "add: ok\n";
1588 } else {
1589 print "add: error";
1590 if ($rules->{exp_add_error}) {
1591 print " as expected\n";
1592 } else {
1593 print "\n";
6ada823a 1594 system("tree", "$udev_dev");
912541b0
KS
1595 print "\n";
1596 $error++;
1597 sleep(1);
1598 }
1599 }
a367f04e 1600
912541b0
KS
1601 if (defined($rules->{option}) && $rules->{option} eq "keep") {
1602 print "\n\n";
1603 return;
1604 }
0345b862 1605
efc9f703
MP
1606 $rc = udev("remove", $rules->{devpath}, \$rules->{rules});
1607 if ($rc != 0) {
1608 print "$udev_bin remove failed with code $rc\n";
1609 $error++;
1610 }
6ada823a
KS
1611 if ((-e "$udev_dev/$rules->{exp_name}") ||
1612 (-l "$udev_dev/$rules->{exp_name}")) {
912541b0
KS
1613 print "remove: error";
1614 if ($rules->{exp_rem_error}) {
1615 print " as expected\n";
1616 } else {
1617 print "\n";
6ada823a 1618 system("tree", "$udev_dev");
912541b0
KS
1619 print "\n";
1620 $error++;
1621 sleep(1);
1622 }
1623 } else {
1624 print "remove: ok\n";
1625 }
0345b862 1626
912541b0 1627 print "\n";
9b434de1 1628
912541b0 1629 if (defined($rules->{option}) && $rules->{option} eq "clean") {
6ada823a 1630 udev_setup();
912541b0 1631 }
0345b862 1632
a367f04e
GKH
1633}
1634
800ab95b
GKH
1635# only run if we have root permissions
1636# due to mknod restrictions
1637if (!($<==0)) {
912541b0 1638 print "Must have root permissions to run properly.\n";
4d55fc5b 1639 exit($EXIT_TEST_SKIP);
800ab95b
GKH
1640}
1641
e4d214ef
DC
1642# skip the test when running in a chroot
1643system("systemd-detect-virt", "-r", "-q");
1644if ($? >> 8 == 0) {
ce5fcc69
ZJS
1645 print "Running in a chroot, skipping the test.\n";
1646 exit($EXIT_TEST_SKIP);
e4d214ef
DC
1647}
1648
0eb3cc88
JS
1649# skip the test when running in a container
1650system("systemd-detect-virt", "-c", "-q");
1651if ($? >> 8 == 0) {
ce5fcc69
ZJS
1652 print "Running in a container, skipping the test.\n";
1653 exit($EXIT_TEST_SKIP);
0eb3cc88
JS
1654}
1655
6ada823a 1656udev_setup();
2e317184
GKH
1657
1658my $test_num = 1;
e08109cb 1659my @list;
2e317184 1660
e08109cb 1661foreach my $arg (@ARGV) {
912541b0
KS
1662 if ($arg =~ m/--valgrind/) {
1663 $valgrind = 1;
1664 printf("using valgrind\n");
333e07b7
TG
1665 } elsif ($arg =~ m/--gdb/) {
1666 $gdb = 1;
1667 printf("using gdb\n");
587751eb
ZJS
1668 } elsif ($arg =~ m/--strace/) {
1669 $strace = 1;
1670 printf("using strace\n");
912541b0
KS
1671 } else {
1672 push(@list, $arg);
1673 }
e08109cb
KS
1674}
1675
1676if ($list[0]) {
912541b0
KS
1677 foreach my $arg (@list) {
1678 if (defined($tests[$arg-1]->{desc})) {
1679 print "udev-test will run test number $arg:\n\n";
1680 run_test($tests[$arg-1], $arg);
1681 } else {
1682 print "test does not exist.\n";
1683 }
1684 }
2e317184 1685} else {
912541b0
KS
1686 # test all
1687 print "\nudev-test will run ".($#tests + 1)." tests:\n\n";
2e317184 1688
912541b0
KS
1689 foreach my $rules (@tests) {
1690 run_test($rules, $test_num);
1691 $test_num++;
1692 }
2e317184
GKH
1693}
1694
ab06eef8 1695print "$error errors occurred\n\n";
a367f04e
GKH
1696
1697# cleanup
6ada823a 1698system("rm", "-rf", "$udev_run");
2ce8d27b 1699system("umount", "$udev_tmpfs");
21d9e3f3 1700rmdir($udev_tmpfs);
a367f04e 1701
034b37c8 1702if ($error > 0) {
d3fc8bf4 1703 exit(1);
034b37c8
AJ
1704}
1705exit(0);