1 // SPDX-License-Identifier: GPL-2.0
2 /*
3 * Copyright (c) 2015 Fujitsu Ltd.
4 * Author: Guangwen Feng <fenggw-fnst@cn.fujitsu.com>
5 */
6
7 /*
8 * DESCRIPTION
9 * Test for feature F_SETLEASE of fcntl(2).
10 * "F_SETLEASE is used to establish a lease which provides a mechanism:
11 * When a process (the lease breaker) performs an open(2) or truncate(2)
12 * that conflicts with the lease, the system call will be blocked by
13 * kernel, meanwhile the kernel notifies the lease holder by sending
14 * it a signal (SIGIO by default), after the lease holder successes
15 * to downgrade or remove the lease, the kernel permits the system
16 * call of the lease breaker to proceed."
17 */
18
19 #include <errno.h>
20
21 #include "tst_test.h"
22 #include "tst_timer.h"
23 #include "tst_safe_macros.h"
24
25 /*
26 * MIN_TIME_LIMIT is defined to 5 senconds as a minimal acceptable
27 * amount of time for the lease breaker waiting for unblock via
28 * lease holder voluntarily downgrade or remove the lease, if the
29 * lease breaker is unblocked within MIN_TIME_LIMIT we may consider
30 * that the feature of the lease mechanism works well.
31 *
32 * The lease-break-time is set to 45 seconds for timeout in kernel.
33 */
34 #define MIN_TIME_LIMIT 5
35
36 #define OP_OPEN_RDONLY 0
37 #define OP_OPEN_WRONLY 1
38 #define OP_OPEN_RDWR 2
39 #define OP_TRUNCATE 3
40
41 #define FILE_MODE (S_IRWXU | S_IRWXG | S_IRWXO | S_ISUID | S_ISGID)
42 #define PATH_LS_BRK_T "/proc/sys/fs/lease-break-time"
43
44 static void do_test(unsigned int);
45 static void do_child(unsigned int);
46
47 static int fd;
48 static int ls_brk_t;
49 static long type;
50 static sigset_t newset, oldset;
51
52 /* Time limit for lease holder to receive SIGIO. */
53 static struct timespec timeout = {.tv_sec = 5};
54
55 static struct test_case_t {
56 int lease_type;
57 int op_type;
58 char *desc;
59 } test_cases[] = {
60 {F_WRLCK, OP_OPEN_RDONLY,
61 "open(O_RDONLY) conflicts with fcntl(F_SETLEASE, F_WRLCK)"},
62 {F_WRLCK, OP_OPEN_WRONLY,
63 "open(O_WRONLY) conflicts with fcntl(F_SETLEASE, F_WRLCK)"},
64 {F_WRLCK, OP_OPEN_RDWR,
65 "open(O_RDWR) conflicts with fcntl(F_SETLEASE, F_WRLCK)"},
66 {F_WRLCK, OP_TRUNCATE,
67 "truncate() conflicts with fcntl(F_SETLEASE, F_WRLCK)"},
68 {F_RDLCK, OP_OPEN_WRONLY,
69 "open(O_WRONLY) conflicts with fcntl(F_SETLEASE, F_RDLCK)"},
70 {F_RDLCK, OP_OPEN_RDWR,
71 "open(O_RDWR) conflicts with fcntl(F_SETLEASE, F_RDLCK)"},
72 {F_RDLCK, OP_TRUNCATE,
73 "truncate() conflicts with fcntl(F_SETLEASE, F_RDLCK)"},
74 };
75
setup(void)76 static void setup(void)
77 {
78 tst_timer_check(CLOCK_MONOTONIC);
79
80 /* Backup and set the lease-break-time. */
81 SAFE_FILE_SCANF(PATH_LS_BRK_T, "%d", &ls_brk_t);
82 SAFE_FILE_PRINTF(PATH_LS_BRK_T, "%d", 45);
83
84 SAFE_TOUCH("file", FILE_MODE, NULL);
85
86 sigemptyset(&newset);
87 sigaddset(&newset, SIGIO);
88
89 if (sigprocmask(SIG_SETMASK, &newset, &oldset) < 0)
90 tst_brk(TBROK | TERRNO, "sigprocmask() failed");
91 }
92
do_test(unsigned int i)93 static void do_test(unsigned int i)
94 {
95 pid_t cpid;
96
97 cpid = SAFE_FORK();
98 if (cpid == 0) {
99 do_child(i);
100 return;
101 }
102
103 fd = SAFE_OPEN("file", O_RDONLY);
104
105 TEST(fcntl(fd, F_SETLEASE, test_cases[i].lease_type));
106 if (TST_RET == -1) {
107 if (type == TST_OVERLAYFS_MAGIC && TST_ERR == EAGAIN) {
108 tst_res(TINFO | TTERRNO,
109 "fcntl(F_SETLEASE, F_WRLCK) failed on overlayfs as expected");
110 } else {
111 tst_res(TFAIL | TTERRNO, "fcntl() failed to set lease");
112 }
113 TST_CHECKPOINT_WAKE(0);
114 goto exit;
115 }
116
117 TST_CHECKPOINT_WAKE(0);
118 /* Wait for SIGIO caused by lease breaker. */
119 TEST(sigtimedwait(&newset, NULL, &timeout));
120 if (TST_RET == -1) {
121 if (TST_ERR == EAGAIN) {
122 tst_res(TFAIL | TTERRNO,
123 "failed to receive SIGIO within %lis",
124 timeout.tv_sec);
125 goto exit;
126 }
127 tst_brk(TBROK | TTERRNO, "sigtimedwait() failed");
128 }
129
130 /* Try to downgrade or remove the lease. */
131 switch (test_cases[i].lease_type) {
132 case F_WRLCK:
133 TEST(fcntl(fd, F_SETLEASE, F_RDLCK));
134 if (TST_RET == 0) {
135 if (test_cases[i].op_type == OP_OPEN_RDONLY)
136 break;
137
138 tst_res(TFAIL,
139 "fcntl() downgraded lease when not read-only");
140 }
141
142 if (test_cases[i].op_type == OP_OPEN_RDONLY) {
143 tst_res(TFAIL | TTERRNO,
144 "fcntl() failed to downgrade lease");
145 }
146
147 /* Falls through */
148 case F_RDLCK:
149 TEST(fcntl(fd, F_SETLEASE, F_UNLCK));
150 if (TST_RET == -1) {
151 tst_res(TFAIL | TTERRNO,
152 "fcntl() failed to remove the lease");
153 }
154 break;
155 default:
156 break;
157 }
158
159 exit:
160 tst_reap_children();
161 SAFE_CLOSE(fd);
162 }
163
do_child(unsigned int i)164 static void do_child(unsigned int i)
165 {
166 long long elapsed_ms;
167
168 TST_CHECKPOINT_WAIT(0);
169
170 tst_timer_start(CLOCK_MONOTONIC);
171
172 switch (test_cases[i].op_type) {
173 case OP_OPEN_RDONLY:
174 SAFE_OPEN("file", O_RDONLY);
175 break;
176 case OP_OPEN_WRONLY:
177 SAFE_OPEN("file", O_WRONLY);
178 break;
179 case OP_OPEN_RDWR:
180 SAFE_OPEN("file", O_RDWR);
181 break;
182 case OP_TRUNCATE:
183 SAFE_TRUNCATE("file", 0);
184 break;
185 default:
186 break;
187 }
188
189 tst_timer_stop();
190
191 elapsed_ms = tst_timer_elapsed_ms();
192
193 if (elapsed_ms < MIN_TIME_LIMIT * 1000) {
194 tst_res(TPASS, "%s, unblocked within %ds",
195 test_cases[i].desc, MIN_TIME_LIMIT);
196 } else {
197 tst_res(TFAIL,
198 "%s, blocked too long %llims, expected within %ds",
199 test_cases[i].desc, elapsed_ms, MIN_TIME_LIMIT);
200 }
201 }
202
cleanup(void)203 static void cleanup(void)
204 {
205 if (sigprocmask(SIG_SETMASK, &oldset, NULL) < 0)
206 tst_res(TWARN | TERRNO, "sigprocmask restore oldset failed");
207
208 if (fd > 0)
209 SAFE_CLOSE(fd);
210
211 /* Restore the lease-break-time. */
212 FILE_PRINTF(PATH_LS_BRK_T, "%d", ls_brk_t);
213 }
214
215 static struct tst_test test = {
216 .forks_child = 1,
217 .needs_root = 1,
218 .needs_checkpoints = 1,
219 .tcnt = ARRAY_SIZE(test_cases),
220 .setup = setup,
221 .test = do_test,
222 .cleanup = cleanup,
223 .skip_filesystems = (const char *const []) {
224 "tmpfs",
225 "ramfs",
226 "nfs",
227 NULL
228 },
229 };
230