2 * Copyright © 2018 Alexey Dobriyan <adobriyan@gmail.com>
4 * Permission to use, copy, modify, and distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17 // 1) read of every file in /proc
18 // 2) readlink of every symlink in /proc
19 // 3) recursively (1) + (2) for every directory in /proc
20 // 4) write to /proc/*/clear_refs and /proc/*/task/*/clear_refs
21 // 5) write to /proc/sysrq-trigger
25 #include <sys/types.h>
38 static void f_reg(DIR *d
, const char *filename
)
44 /* read from /proc/kmsg can block */
45 fd
= openat(dirfd(d
), filename
, O_RDONLY
|O_NONBLOCK
);
48 rv
= read(fd
, buf
, sizeof(buf
));
49 assert((0 <= rv
&& rv
<= sizeof(buf
)) || rv
== -1);
53 static void f_reg_write(DIR *d
, const char *filename
, const char *buf
, size_t len
)
58 fd
= openat(dirfd(d
), filename
, O_WRONLY
);
61 rv
= write(fd
, buf
, len
);
62 assert((0 <= rv
&& rv
<= len
) || rv
== -1);
66 static void f_lnk(DIR *d
, const char *filename
)
71 rv
= readlinkat(dirfd(d
), filename
, buf
, sizeof(buf
));
72 assert((0 <= rv
&& rv
<= sizeof(buf
)) || rv
== -1);
75 static void f(DIR *d
, unsigned int level
)
80 assert(de
->d_type
== DT_DIR
);
81 assert(streq(de
->d_name
, "."));
84 assert(de
->d_type
== DT_DIR
);
85 assert(streq(de
->d_name
, ".."));
87 while ((de
= xreaddir(d
))) {
88 assert(!streq(de
->d_name
, "."));
89 assert(!streq(de
->d_name
, ".."));
96 if (level
== 0 && streq(de
->d_name
, "sysrq-trigger")) {
97 f_reg_write(d
, de
->d_name
, "h", 1);
98 } else if (level
== 1 && streq(de
->d_name
, "clear_refs")) {
99 f_reg_write(d
, de
->d_name
, "1", 1);
100 } else if (level
== 3 && streq(de
->d_name
, "clear_refs")) {
101 f_reg_write(d
, de
->d_name
, "1", 1);
103 f_reg(d
, de
->d_name
);
107 fd
= openat(dirfd(d
), de
->d_name
, O_DIRECTORY
|O_RDONLY
);
117 f_lnk(d
, de
->d_name
);
130 d
= opendir("/proc");
134 /* Ensure /proc is proc. */
135 if (fstatfs(dirfd(d
), &sfs
) == -1) {
138 if (sfs
.f_type
!= 0x9fa0) {
139 fprintf(stderr
, "error: unexpected f_type %lx\n", (long)sfs
.f_type
);