1f08c3bdfSopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
2f08c3bdfSopenharmony_ci/*
3f08c3bdfSopenharmony_ci * Copyright (C) 2020 Cyril Hrubis <chrubis@suse.cz>
4f08c3bdfSopenharmony_ci */
5f08c3bdfSopenharmony_ci
6f08c3bdfSopenharmony_ci/*\
7f08c3bdfSopenharmony_ci * [Description]
8f08c3bdfSopenharmony_ci *
9f08c3bdfSopenharmony_ci * Test for a SHM_SET.
10f08c3bdfSopenharmony_ci *
11f08c3bdfSopenharmony_ci * The test clears the group and others bits from the shm_perm.mode and checks
12f08c3bdfSopenharmony_ci * the result as well as if the ctime was updated correctly.
13f08c3bdfSopenharmony_ci */
14f08c3bdfSopenharmony_ci
15f08c3bdfSopenharmony_ci#define _GNU_SOURCE
16f08c3bdfSopenharmony_ci#include <stdio.h>
17f08c3bdfSopenharmony_ci#include "tst_test.h"
18f08c3bdfSopenharmony_ci#include "tst_safe_sysv_ipc.h"
19f08c3bdfSopenharmony_ci#include "libnewipc.h"
20f08c3bdfSopenharmony_ci
21f08c3bdfSopenharmony_ci#define SHM_SIZE 2048
22f08c3bdfSopenharmony_ci
23f08c3bdfSopenharmony_cistatic int shm_id = -1;
24f08c3bdfSopenharmony_ci
25f08c3bdfSopenharmony_cistatic int test_ipc_set(struct shmid_ds *ds)
26f08c3bdfSopenharmony_ci{
27f08c3bdfSopenharmony_ci	TEST(shmctl(shm_id, IPC_SET, ds));
28f08c3bdfSopenharmony_ci
29f08c3bdfSopenharmony_ci	if (TST_RET != 0) {
30f08c3bdfSopenharmony_ci		tst_res(TFAIL, "shmctl(%i, IPC_SET, ...)", shm_id);
31f08c3bdfSopenharmony_ci		return 1;
32f08c3bdfSopenharmony_ci	}
33f08c3bdfSopenharmony_ci
34f08c3bdfSopenharmony_ci	tst_res(TPASS, "shmctl(%i, IPC_SET, {shm_perm.mode=%04o})",
35f08c3bdfSopenharmony_ci		shm_id, ds->shm_perm.mode);
36f08c3bdfSopenharmony_ci	return 0;
37f08c3bdfSopenharmony_ci}
38f08c3bdfSopenharmony_ci
39f08c3bdfSopenharmony_cistatic void check_mode(struct shmid_ds *ds, short exp_mode)
40f08c3bdfSopenharmony_ci{
41f08c3bdfSopenharmony_ci	if (ds->shm_perm.mode == exp_mode) {
42f08c3bdfSopenharmony_ci		tst_res(TPASS, "shm_perm.mode=%04o", exp_mode);
43f08c3bdfSopenharmony_ci		return;
44f08c3bdfSopenharmony_ci	}
45f08c3bdfSopenharmony_ci
46f08c3bdfSopenharmony_ci	tst_res(TFAIL, "shm_perm.mode=%04o, expected %i",
47f08c3bdfSopenharmony_ci		ds->shm_perm.mode, exp_mode);
48f08c3bdfSopenharmony_ci}
49f08c3bdfSopenharmony_ci
50f08c3bdfSopenharmony_cistatic void verify_shmset(void)
51f08c3bdfSopenharmony_ci{
52f08c3bdfSopenharmony_ci	struct shmid_ds ds;
53f08c3bdfSopenharmony_ci	unsigned short old_mode;
54f08c3bdfSopenharmony_ci	time_t old_ctime;
55f08c3bdfSopenharmony_ci
56f08c3bdfSopenharmony_ci	SAFE_SHMCTL(shm_id, IPC_STAT, &ds);
57f08c3bdfSopenharmony_ci
58f08c3bdfSopenharmony_ci	old_mode = ds.shm_perm.mode;
59f08c3bdfSopenharmony_ci	old_ctime = ds.shm_ctime;
60f08c3bdfSopenharmony_ci
61f08c3bdfSopenharmony_ci	check_mode(&ds, 0666);
62f08c3bdfSopenharmony_ci
63f08c3bdfSopenharmony_ci	sleep(1);
64f08c3bdfSopenharmony_ci
65f08c3bdfSopenharmony_ci	ds.shm_perm.mode &= ~0066;
66f08c3bdfSopenharmony_ci
67f08c3bdfSopenharmony_ci	if (test_ipc_set(&ds))
68f08c3bdfSopenharmony_ci		return;
69f08c3bdfSopenharmony_ci
70f08c3bdfSopenharmony_ci	memset(&ds, 0, sizeof(ds));
71f08c3bdfSopenharmony_ci	SAFE_SHMCTL(shm_id, IPC_STAT, &ds);
72f08c3bdfSopenharmony_ci	check_mode(&ds, old_mode & ~0066);
73f08c3bdfSopenharmony_ci
74f08c3bdfSopenharmony_ci	if (ds.shm_ctime <= old_ctime || ds.shm_ctime > old_ctime + 10) {
75f08c3bdfSopenharmony_ci		tst_res(TFAIL, "shm_ctime not updated old %li new %li",
76f08c3bdfSopenharmony_ci			(long)old_ctime, (long)ds.shm_ctime);
77f08c3bdfSopenharmony_ci	} else {
78f08c3bdfSopenharmony_ci		tst_res(TPASS, "shm_ctime updated correctly diff=%li",
79f08c3bdfSopenharmony_ci			(long)(ds.shm_ctime - old_ctime));
80f08c3bdfSopenharmony_ci	}
81f08c3bdfSopenharmony_ci
82f08c3bdfSopenharmony_ci	ds.shm_perm.mode = old_mode;
83f08c3bdfSopenharmony_ci	if (test_ipc_set(&ds))
84f08c3bdfSopenharmony_ci		return;
85f08c3bdfSopenharmony_ci
86f08c3bdfSopenharmony_ci	memset(&ds, 0, sizeof(ds));
87f08c3bdfSopenharmony_ci	SAFE_SHMCTL(shm_id, IPC_STAT, &ds);
88f08c3bdfSopenharmony_ci	check_mode(&ds, old_mode & MODE_MASK);
89f08c3bdfSopenharmony_ci}
90f08c3bdfSopenharmony_ci
91f08c3bdfSopenharmony_cistatic void setup(void)
92f08c3bdfSopenharmony_ci{
93f08c3bdfSopenharmony_ci	shm_id = SAFE_SHMGET(IPC_PRIVATE, SHM_SIZE, IPC_CREAT | 0666);
94f08c3bdfSopenharmony_ci}
95f08c3bdfSopenharmony_ci
96f08c3bdfSopenharmony_cistatic void cleanup(void)
97f08c3bdfSopenharmony_ci{
98f08c3bdfSopenharmony_ci	if (shm_id >= 0)
99f08c3bdfSopenharmony_ci		SAFE_SHMCTL(shm_id, IPC_RMID, NULL);
100f08c3bdfSopenharmony_ci}
101f08c3bdfSopenharmony_ci
102f08c3bdfSopenharmony_cistatic struct tst_test test = {
103f08c3bdfSopenharmony_ci	.setup = setup,
104f08c3bdfSopenharmony_ci	.cleanup = cleanup,
105f08c3bdfSopenharmony_ci	.test_all = verify_shmset,
106f08c3bdfSopenharmony_ci};
107