231 lines
6.0 KiB
C
231 lines
6.0 KiB
C
/* *************************************************************************
|
|
* Copyright (c) International Business Machines Corp., 2009
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
|
* the GNU General Public License for more details.
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
* Author: Veerendra C <vechandr@in.ibm.com>
|
|
*
|
|
* Test Assertion:
|
|
* This testcase verifies the semaphore isoloation in 2 diff containers.
|
|
* It tries to create/access a semaphore created with the same KEY.
|
|
*
|
|
* Description:
|
|
* Create 2 'containers' with the below flag value
|
|
* Flag = clone, clone(CLONE_NEWIPC), or unshare(CLONE_NEWIPC)
|
|
* In Cont1, create semaphore with key 124326L
|
|
* In Cont2, try to access the semaphore created in Cont1.
|
|
* PASS :
|
|
* If flag = None and the semaphore is accessible in Cont2.
|
|
* If flag = unshare/clone and the semaphore is not accessible in Cont2.
|
|
* If semaphore is not accessible in Cont2, creates new semaphore with
|
|
* the same key to double check isloation in IPCNS.
|
|
*
|
|
* FAIL :
|
|
* If flag = none and the semaphore is not accessible.
|
|
* If flag = unshare/clone and semaphore is accessible in Cont2.
|
|
* If the new semaphore creation Fails.
|
|
***************************************************************************/
|
|
|
|
#define _GNU_SOURCE 1
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <sys/ipc.h>
|
|
#include <sys/sem.h>
|
|
#include <libclone.h>
|
|
#include "test.h"
|
|
#include "ipcns_helper.h"
|
|
|
|
#define MY_KEY 124326L
|
|
#define UNSHARESTR "unshare"
|
|
#define CLONESTR "clone"
|
|
#define NONESTR "none"
|
|
|
|
char *TCID = "semtest_2ns";
|
|
int TST_TOTAL = 1;
|
|
int p1[2];
|
|
int p2[2];
|
|
static struct sembuf semop_lock[2] = {
|
|
/* sem_num, sem_op, flag */
|
|
{0, 0, 0}, /* wait for sem#0 to become 0 */
|
|
{0, 1, SEM_UNDO} /* then increment sem#0 by 1 */
|
|
};
|
|
|
|
static struct sembuf semop_unlock[1] = {
|
|
/* sem_num, sem_op, flag */
|
|
{0, -1, (IPC_NOWAIT | SEM_UNDO)} /* decrement sem#0 by 1 (sets it to 0) */
|
|
};
|
|
|
|
/*
|
|
* sem_lock() - Locks the semaphore for crit-sec updation, and unlocks it later
|
|
*/
|
|
void sem_lock(int id)
|
|
{
|
|
/* Checking the semlock and simulating as if the crit-sec is updated */
|
|
if (semop(id, &semop_lock[0], 2) < 0) {
|
|
perror("sem lock error");
|
|
tst_brkm(TBROK, NULL, "semop failed");
|
|
}
|
|
tst_resm(TINFO, "Sem1: File locked, Critical section is updated...");
|
|
sleep(2);
|
|
if (semop(id, &semop_unlock[0], 1) < 0) {
|
|
perror("sem unlock error");
|
|
tst_brkm(TBROK, NULL, "semop failed");
|
|
}
|
|
}
|
|
|
|
/*
|
|
* check_sem1 - does not read -- it writes to check_sem2() when it's done.
|
|
*/
|
|
int check_sem1(void *vtest)
|
|
{
|
|
int id1;
|
|
|
|
(void) vtest;
|
|
|
|
close(p1[0]);
|
|
/* 1. Create (or fetch if existing) the binary semaphore */
|
|
id1 = semget(MY_KEY, 1, IPC_CREAT | IPC_EXCL | 0666);
|
|
if (id1 == -1) {
|
|
perror("Semaphore create");
|
|
if (errno != EEXIST) {
|
|
perror("semget failure");
|
|
tst_brkm(TBROK, NULL, "semget failure");
|
|
}
|
|
id1 = semget(MY_KEY, 1, 0);
|
|
if (id1 == -1) {
|
|
perror("Semaphore create");
|
|
tst_brkm(TBROK, NULL, "semget failure");
|
|
}
|
|
}
|
|
|
|
write(p1[1], "go", 3);
|
|
tst_resm(TINFO, "Cont1: Able to create semaphore");
|
|
tst_exit();
|
|
}
|
|
|
|
/*
|
|
* check_sem2() reads from check_sem1() and writes to main() when it's done.
|
|
*/
|
|
|
|
int check_sem2(void *vtest)
|
|
{
|
|
char buf[3];
|
|
int id2;
|
|
|
|
(void) vtest;
|
|
|
|
close(p1[1]);
|
|
close(p2[0]);
|
|
read(p1[0], buf, 3);
|
|
|
|
id2 = semget(MY_KEY, 1, 0);
|
|
if (id2 != -1) {
|
|
sem_lock(id2);
|
|
write(p2[1], "exists", 7);
|
|
} else {
|
|
/* Trying to create a new semaphore, if semaphore is not existing */
|
|
id2 = semget(MY_KEY, 1, IPC_CREAT | IPC_EXCL | 0666);
|
|
if (id2 == -1) {
|
|
perror("Semaphore create");
|
|
if (errno != EEXIST) {
|
|
perror("semget failure");
|
|
tst_resm(TBROK, "semget failure");
|
|
}
|
|
} else
|
|
tst_resm(TINFO,
|
|
"Cont2: Able to create semaphore with sameKey");
|
|
/* Passing the pipe Not-found mesg */
|
|
write(p2[1], "notfnd", 7);
|
|
}
|
|
|
|
tst_exit();
|
|
}
|
|
|
|
static void setup(void)
|
|
{
|
|
tst_require_root();
|
|
check_newipc();
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
int ret, id, use_clone = T_NONE;
|
|
char *tsttype = NONESTR;
|
|
char buf[7];
|
|
|
|
setup();
|
|
|
|
if (argc != 2) {
|
|
tst_resm(TINFO, "Usage: %s <clone| unshare| none>", argv[0]);
|
|
tst_resm(TINFO, " where clone, unshare, or fork specifies"
|
|
" unshare method.");
|
|
tst_exit();
|
|
}
|
|
|
|
/* Using PIPE's to sync between container and Parent */
|
|
if (pipe(p1) == -1) {
|
|
perror("pipe1");
|
|
tst_exit();
|
|
}
|
|
if (pipe(p2) == -1) {
|
|
perror("pipe2");
|
|
tst_exit();
|
|
}
|
|
|
|
if (strcmp(argv[1], "clone") == 0) {
|
|
use_clone = T_CLONE;
|
|
tsttype = CLONESTR;
|
|
} else if (strcmp(argv[1], "unshare") == 0) {
|
|
use_clone = T_UNSHARE;
|
|
tsttype = UNSHARESTR;
|
|
}
|
|
|
|
tst_resm(TINFO, "Semaphore Namespaces Test : %s", tsttype);
|
|
|
|
/* Create 2 containers */
|
|
ret = do_clone_unshare_test(use_clone, CLONE_NEWIPC, check_sem1, NULL);
|
|
if (ret < 0) {
|
|
tst_brkm(TFAIL, NULL, "clone/unshare failed");
|
|
}
|
|
|
|
ret = do_clone_unshare_test(use_clone, CLONE_NEWIPC, check_sem2, NULL);
|
|
if (ret < 0) {
|
|
tst_brkm(TFAIL, NULL, "clone/unshare failed");
|
|
}
|
|
close(p2[1]);
|
|
read(p2[0], buf, 7);
|
|
|
|
if (strcmp(buf, "exists") == 0)
|
|
if (use_clone == T_NONE)
|
|
tst_resm(TPASS,
|
|
"Plain cloned process able to access the semaphore "
|
|
"created");
|
|
else
|
|
tst_resm(TFAIL,
|
|
"%s : In namespace2 found the semaphore "
|
|
"created in Namespace1", tsttype);
|
|
else if (use_clone == T_NONE)
|
|
tst_resm(TFAIL, "Plain cloned process didn't find semaphore");
|
|
else
|
|
tst_resm(TPASS,
|
|
"%s : In namespace2 unable to access the semaphore "
|
|
"created in Namespace1", tsttype);
|
|
|
|
/* Delete the semaphore */
|
|
id = semget(MY_KEY, 1, 0);
|
|
semctl(id, IPC_RMID, 0);
|
|
tst_exit();
|
|
}
|