123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230 |
- #include "libbb.h"
- #include <sys/ipc.h>
- #include <sys/shm.h>
- #include <sys/msg.h>
- #include <sys/sem.h>
- #if defined(__GNU_LIBRARY__) && !defined(_SEM_SEMUN_UNDEFINED)
- #else
- union semun {
- int val;
- struct semid_ds *buf;
- unsigned short *array;
- struct seminfo *__buf;
- };
- #endif
- #define IPCRM_LEGACY 1
- #if IPCRM_LEGACY
- typedef enum type_id {
- SHM,
- SEM,
- MSG
- } type_id;
- static int remove_ids(type_id type, char **argv)
- {
- unsigned long id;
- int nb_errors = 0;
- union semun arg;
- arg.val = 0;
- while (argv[0]) {
- id = bb_strtoul(argv[0], NULL, 10);
- if (errno || id > INT_MAX) {
- bb_error_msg("invalid id: %s", argv[0]);
- nb_errors++;
- } else {
- int ret = 0;
- if (type == SEM)
- ret = semctl(id, 0, IPC_RMID, arg);
- else if (type == MSG)
- ret = msgctl(id, IPC_RMID, NULL);
- else if (type == SHM)
- ret = shmctl(id, IPC_RMID, NULL);
- if (ret) {
- bb_perror_msg("can't remove id %s", argv[0]);
- nb_errors++;
- }
- }
- argv++;
- }
- return nb_errors;
- }
- #endif
- int ipcrm_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
- int ipcrm_main(int argc, char **argv)
- {
- int c;
- int error = 0;
-
- if (argc == 1)
- return 0;
- #if IPCRM_LEGACY
-
- {
- type_id what = 0;
- char w;
- w = argv[1][0];
- if ( ((w == 'm' && argv[1][1] == 's' && argv[1][2] == 'g')
- || (argv[1][0] == 's'
- && ((w = argv[1][1]) == 'h' || w == 'e')
- && argv[1][2] == 'm')
- ) && argv[1][3] == '\0'
- ) {
- if (argc < 3)
- bb_show_usage();
- if (w == 'h')
- what = SHM;
- else if (w == 'm')
- what = MSG;
- else if (w == 'e')
- what = SEM;
- if (remove_ids(what, &argv[2]))
- fflush_stdout_and_exit(EXIT_FAILURE);
- puts("resource(s) deleted");
- return 0;
- }
- }
- #endif
-
- while ((c = getopt(argc, argv, "q:m:s:Q:M:S:")) != -1) {
- int result;
- int id;
- int iskey;
-
- union semun arg;
- if (c == '?')
- bb_show_usage();
- id = 0;
- arg.val = 0;
- iskey = !(c & 0x20);
- if (iskey) {
-
- key_t key = xstrtoul(optarg, 0);
- if (key == IPC_PRIVATE) {
- error++;
- bb_error_msg("illegal key (%s)", optarg);
- continue;
- }
- c |= 0x20;
-
- id = ((c == 'q') ? msgget(key, 0) :
- (c == 'm') ? shmget(key, 0, 0) : semget(key, 0, 0));
- if (id < 0) {
- const char *errmsg;
- error++;
- switch (errno) {
- case EACCES:
- errmsg = "permission denied for";
- break;
- case EIDRM:
- errmsg = "already removed";
- break;
- case ENOENT:
- errmsg = "invalid";
- break;
- default:
- errmsg = "unknown error in";
- break;
- }
- bb_error_msg("%s %s (%s)", errmsg, "key", optarg);
- continue;
- }
- } else {
-
- id = xatoul(optarg);
- }
- result = ((c == 'q') ? msgctl(id, IPC_RMID, NULL) :
- (c == 'm') ? shmctl(id, IPC_RMID, NULL) :
- semctl(id, 0, IPC_RMID, arg));
- if (result) {
- const char *errmsg;
- const char *const what = iskey ? "key" : "id";
- error++;
- switch (errno) {
- case EACCES:
- case EPERM:
- errmsg = "permission denied for";
- break;
- case EINVAL:
- errmsg = "invalid";
- break;
- case EIDRM:
- errmsg = "already removed";
- break;
- default:
- errmsg = "unknown error in";
- break;
- }
- bb_error_msg("%s %s (%s)", errmsg, what, optarg);
- continue;
- }
- }
-
- if (optind != argc) {
- bb_show_usage();
- }
-
- return error;
- }
|