Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

test: verify fdinfo for pidfd of reaped process

Test that the fdinfo field of a pidfd referring to a dead process
correctly shows Pid: -1 and NSpid: -1.

Cc: Christian Kellner <christian@kellner.me>
Cc: linux-kselftest@vger.kernel.org
Reviewed-by: Christian Kellner <christian@kellner.me>
Signed-off-by: Christian Brauner <christian.brauner@ubuntu.com>
Link: https://lore.kernel.org/r/20191017101832.5985-2-christian.brauner@ubuntu.com

+46 -15
+46 -15
tools/testing/selftests/pidfd/pidfd_fdinfo_test.c
··· 113 113 return ret; 114 114 } 115 115 116 + static inline void child_close(struct child *child) 117 + { 118 + close(child->fd); 119 + } 120 + 116 121 static inline int child_join(struct child *child, struct error *err) 117 122 { 118 123 int r; 119 124 120 - (void)close(child->fd); 121 125 r = wait_for_pid(child->pid); 122 126 if (r < 0) 123 127 error_set(err, PIDFD_ERROR, "waitpid failed (ret %d, errno %d)", ··· 132 128 return r; 133 129 } 134 130 131 + static inline int child_join_close(struct child *child, struct error *err) 132 + { 133 + child_close(child); 134 + return child_join(child, err); 135 + } 136 + 135 137 static inline void trim_newline(char *str) 136 138 { 137 139 char *pos = strrchr(str, '\n'); ··· 146 136 *pos = '\0'; 147 137 } 148 138 149 - static int verify_fdinfo_nspid(int pidfd, struct error *err, 150 - const char *expect, ...) 139 + static int verify_fdinfo(int pidfd, struct error *err, const char *prefix, 140 + size_t prefix_len, const char *expect, ...) 151 141 { 152 142 char buffer[512] = {0, }; 153 143 char path[512] = {0, }; ··· 170 160 pidfd); 171 161 172 162 while (getline(&line, &n, f) != -1) { 173 - if (strncmp(line, "NSpid:", 6)) 163 + char *val; 164 + 165 + if (strncmp(line, prefix, prefix_len)) 174 166 continue; 175 167 176 168 found = 1; 177 169 178 - r = strcmp(line + 6, buffer); 170 + val = line + prefix_len; 171 + r = strcmp(val, buffer); 179 172 if (r != 0) { 180 173 trim_newline(line); 181 174 trim_newline(buffer); 182 - error_set(err, PIDFD_FAIL, "NSpid: '%s' != '%s'", 183 - line + 6, buffer); 175 + error_set(err, PIDFD_FAIL, "%s '%s' != '%s'", 176 + prefix, val, buffer); 184 177 } 185 178 break; 186 179 } ··· 192 179 fclose(f); 193 180 194 181 if (found == 0) 195 - return error_set(err, PIDFD_FAIL, "NSpid not found for fd %d", 196 - pidfd); 182 + return error_set(err, PIDFD_FAIL, "%s not found for fd %d", 183 + prefix, pidfd); 197 184 198 185 return PIDFD_PASS; 199 186 } ··· 226 213 } 227 214 228 215 pidfd = *(int *)args; 229 - r = verify_fdinfo_nspid(pidfd, &err, "\t0\n"); 216 + r = verify_fdinfo(pidfd, &err, "NSpid:", 6, "\t0\n"); 230 217 231 218 if (r != PIDFD_PASS) 232 219 ksft_print_msg("NSpid fdinfo check failed: %s\n", err.msg); ··· 255 242 /* The children will have pid 1 in the new pid namespace, 256 243 * so the line must be 'NSPid:\t<pid>\t1'. 257 244 */ 258 - verify_fdinfo_nspid(a.fd, &err, "\t%d\t%d\n", a.pid, 1); 259 - verify_fdinfo_nspid(b.fd, &err, "\t%d\t%d\n", b.pid, 1); 245 + verify_fdinfo(a.fd, &err, "NSpid:", 6, "\t%d\t%d\n", a.pid, 1); 246 + verify_fdinfo(b.fd, &err, "NSpid:", 6, "\t%d\t%d\n", b.pid, 1); 260 247 261 248 /* wait for the process, check the exit status and set 262 249 * 'err' accordingly, if it is not already set. 263 250 */ 264 - child_join(&a, &err); 265 - child_join(&b, &err); 251 + child_join_close(&a, &err); 252 + child_join_close(&b, &err); 266 253 254 + error_report(&err, test_name); 255 + } 256 + 257 + static void test_pidfd_dead_fdinfo(void) 258 + { 259 + struct child a; 260 + struct error err = {0, }; 261 + const char *test_name = "pidfd check fdinfo for dead process"; 262 + 263 + /* Create a new child in a new pid and mount namespace */ 264 + a = clone_newns(child_fdinfo_nspid_test, NULL, &err); 265 + error_check(&err, test_name); 266 + child_join(&a, &err); 267 + 268 + verify_fdinfo(a.fd, &err, "Pid:", 4, "\t-1\n"); 269 + verify_fdinfo(a.fd, &err, "NSpid:", 6, "\t-1\n"); 270 + child_close(&a); 267 271 error_report(&err, test_name); 268 272 } 269 273 270 274 int main(int argc, char **argv) 271 275 { 272 276 ksft_print_header(); 273 - ksft_set_plan(1); 277 + ksft_set_plan(2); 274 278 275 279 test_pidfd_fdinfo_nspid(); 280 + test_pidfd_dead_fdinfo(); 276 281 277 282 return ksft_exit_pass(); 278 283 }