#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <sys/sem.h>
#include <sys/shm.h>
#include <errno.h>
#include <sys/ipc_impl.h>
#include <zone.h>
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>
#include <locale.h>
#define NULL_MSG ((struct msqid_ds *)NULL)
#define NULL_SEM ((struct semid_ds *)NULL)
#define NULL_SHM ((struct shmid_ds *)NULL)
#define USAGE "usage: ipcrm [-z zone] [ [-q msqid] [-m shmid] " \
"[-s semid]\n\t [-Q msgkey] [-M shmkey] [-S semkey] ... ]\n"
#define IPC_KEYMATCH(perm, zoneid, key) \
((perm).ipcx_key == (key) && (perm).ipcx_zoneid == (zoneid))
static char opts[] = "z:q:m:s:Q:M:S:";
extern char *optarg;
extern int optind;
static zoneid_t zoneid;
static int zflg;
static int *idlist, nids;
static void
oops(char *thing, char *arg)
{
char *e;
switch (errno) {
case ENOENT:
case EINVAL:
e = "not found";
break;
case EPERM:
e = "permission denied";
break;
default:
e = "unknown error";
}
(void) fprintf(stderr, gettext("ipcrm: %s(%s): %s\n"), thing, arg, e);
}
static key_t
getkey(char *kp)
{
key_t k;
char *tp;
if ((k = (key_t)strtoul(kp, &tp, 0)) == IPC_PRIVATE || *tp != '\0') {
(void) fprintf(stderr, gettext("ipcrm: illegal key: %s\n"),
kp);
return (0);
}
return (k);
}
static uint_t
getids(int (*idsfunc)(int *, uint_t, uint_t *))
{
uint_t n;
for (;;) {
if (idsfunc(idlist, nids, &n) != 0)
goto err;
if (n <= nids)
break;
idlist = realloc(idlist, (nids = n) * sizeof (int));
if (idlist == NULL)
goto err;
}
return (n);
err:
perror("ipcrm");
exit(1);
}
static int
msggetid(char *arg)
{
int id = atol(arg);
struct msqid_ds64 qds;
if (!zflg)
return (id);
if (msgctl64(id, IPC_STAT64, &qds) < 0) {
oops("msgctl", arg);
return (-1);
}
if (qds.msgx_perm.ipcx_zoneid != zoneid) {
errno = EINVAL;
oops("msgctl", arg);
return (-1);
}
return (id);
}
static int
msggetkey(char *kp)
{
key_t k;
int id, i;
uint_t n;
struct msqid_ds64 qds;
if ((k = getkey(kp)) == 0)
return (-1);
if (!zflg) {
if ((id = msgget(k, 0)) == -1)
oops("msgget", kp);
return (id);
}
n = getids(msgids);
for (i = 0; i < n; i++) {
id = idlist[i];
if (msgctl64(id, IPC_STAT64, &qds) < 0)
continue;
if (IPC_KEYMATCH(qds.msgx_perm, zoneid, k))
return (id);
}
(void) fprintf(stderr, gettext("ipcrm: unknown key: %s\n"), kp);
return (-1);
}
static int
semgetid(char *arg)
{
int id = atol(arg);
struct semid_ds64 sds;
union semun {
int val;
struct semid_ds64 *buf;
ushort_t *array;
} semarg;
if (!zflg)
return (id);
semarg.buf = &sds;
if (semctl64(id, 0, IPC_STAT64, semarg) < 0) {
oops("semctl", arg);
return (-1);
}
if (sds.semx_perm.ipcx_zoneid != zoneid) {
errno = EINVAL;
oops("semctl", arg);
return (-1);
}
return (id);
}
static int
semgetkey(char *kp)
{
key_t k;
int id, i;
uint_t n;
struct semid_ds64 sds;
union semun {
int val;
struct semid_ds64 *buf;
ushort_t *array;
} semarg;
if ((k = getkey(kp)) == 0)
return (-1);
if (!zflg) {
if ((id = semget(k, 0, 0)) == -1)
oops("semget", kp);
return (id);
}
n = getids(semids);
semarg.buf = &sds;
for (i = 0; i < n; i++) {
int id;
id = idlist[i];
if (semctl64(id, 0, IPC_STAT64, semarg) < 0)
continue;
if (IPC_KEYMATCH(sds.semx_perm, zoneid, k))
return (id);
}
(void) fprintf(stderr, gettext("ipcrm: unknown key: %s\n"), kp);
return (-1);
}
static int
shmgetid(char *arg)
{
int id = atol(arg);
struct shmid_ds64 mds;
if (!zflg)
return (id);
if (shmctl64(id, IPC_STAT64, &mds) < 0) {
oops("shmctl", arg);
return (-1);
}
if (mds.shmx_perm.ipcx_zoneid != zoneid) {
errno = EINVAL;
oops("shmctl", arg);
return (-1);
}
return (id);
}
static int
shmgetkey(char *kp)
{
key_t k;
int id, i;
uint_t n;
struct shmid_ds64 mds;
if ((k = getkey(kp)) == 0)
return (-1);
if (!zflg) {
if ((id = shmget(k, 0, 0)) == -1)
oops("shmget", kp);
return (id);
}
n = getids(shmids);
for (i = 0; i < n; i++) {
int id;
id = idlist[i];
if (shmctl64(id, IPC_STAT64, &mds) < 0)
continue;
if (IPC_KEYMATCH(mds.shmx_perm, zoneid, k))
return (id);
}
(void) fprintf(stderr, gettext("ipcrm: unknown key: %s\n"), kp);
return (-1);
}
static zoneid_t
getzone(char *arg)
{
zoneid_t zoneid;
if (zone_get_id(arg, &zoneid) != 0) {
(void) fprintf(stderr, gettext("ipcrm: unknown zone: %s\n"),
arg);
exit(1);
}
return (zoneid);
}
int
main(int argc, char **argv)
{
int o;
int err;
int ipc_id;
(void) setlocale(LC_ALL, "");
(void) textdomain(TEXT_DOMAIN);
(void) signal(SIGSYS, SIG_IGN);
zoneid = getzoneid();
err = 0;
while ((o = getopt(argc, argv, opts)) != EOF) {
switch (o) {
case 'z':
zflg++;
zoneid = getzone(optarg);
break;
case 'q':
case 'm':
case 's':
case 'Q':
case 'M':
case 'S':
break;
case '?':
default:
err++;
break;
}
}
if (err || (optind < argc)) {
(void) fprintf(stderr, gettext(USAGE));
return (err);
}
if (zflg > 1) {
(void) fprintf(stderr,
gettext("multiple -z options not allowed\n"));
(void) fprintf(stderr, gettext(USAGE));
return (1);
}
optind = 1;
while ((o = getopt(argc, argv, opts)) != EOF) {
switch (o) {
case 'z':
break;
case 'q':
if ((ipc_id = msggetid(optarg)) < 0) {
err++;
} else if (msgctl(ipc_id, IPC_RMID, NULL_MSG) == -1) {
oops("msgctl", optarg);
err++;
}
break;
case 'm':
if ((ipc_id = shmgetid(optarg)) < 0) {
err++;
} else if (shmctl(ipc_id, IPC_RMID, NULL_SHM) == -1) {
oops("shmctl", optarg);
err++;
}
break;
case 's':
if ((ipc_id = semgetid(optarg)) < 0) {
err++;
} else if (semctl(ipc_id, 0, IPC_RMID, NULL_SEM) ==
-1) {
oops("semctl", optarg);
err++;
}
break;
case 'Q':
if ((ipc_id = msggetkey(optarg)) == -1) {
err++;
break;
}
if (msgctl(ipc_id, IPC_RMID, NULL_MSG) == -1) {
oops("msgctl", optarg);
err++;
}
break;
case 'M':
if ((ipc_id = shmgetkey(optarg)) == -1) {
err++;
break;
}
if (shmctl(ipc_id, IPC_RMID, NULL_SHM) == -1) {
oops("shmctl", optarg);
err++;
}
break;
case 'S':
if ((ipc_id = semgetkey(optarg)) == -1) {
err++;
break;
}
if (semctl(ipc_id, 0, IPC_RMID, NULL_SEM) == -1) {
oops("semctl", optarg);
err++;
}
break;
}
}
return (err);
}