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