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