xref: /titanic_53/usr/src/cmd/sendmail/libsm/sem.c (revision 49218d4f8e4d84d1c08aeb267bcf6e451f2056dc)
17c478bd9Sstevel@tonic-gate /*
27c478bd9Sstevel@tonic-gate  * Copyright (c) 2000-2001, 2005 Sendmail, Inc. and its suppliers.
37c478bd9Sstevel@tonic-gate  *      All rights reserved.
47c478bd9Sstevel@tonic-gate  *
57c478bd9Sstevel@tonic-gate  * By using this file, you agree to the terms and conditions set
67c478bd9Sstevel@tonic-gate  * forth in the LICENSE file which can be found at the top level of
77c478bd9Sstevel@tonic-gate  * the sendmail distribution.
87c478bd9Sstevel@tonic-gate  */
97c478bd9Sstevel@tonic-gate 
107c478bd9Sstevel@tonic-gate #pragma ident	"%Z%%M%	%I%	%E% SMI"
117c478bd9Sstevel@tonic-gate 
127c478bd9Sstevel@tonic-gate #include <sm/gen.h>
13*49218d4fSjbeck SM_RCSID("@(#)$Id: sem.c,v 1.13 2005/08/12 20:39:59 ca Exp $")
147c478bd9Sstevel@tonic-gate 
157c478bd9Sstevel@tonic-gate #if SM_CONF_SEM
167c478bd9Sstevel@tonic-gate # include <stdlib.h>
177c478bd9Sstevel@tonic-gate # include <unistd.h>
187c478bd9Sstevel@tonic-gate # include <sm/sem.h>
197c478bd9Sstevel@tonic-gate # include <sm/heap.h>
20*49218d4fSjbeck # include <errno.h>
217c478bd9Sstevel@tonic-gate 
227c478bd9Sstevel@tonic-gate /*
237c478bd9Sstevel@tonic-gate **  SM_SEM_START -- initialize semaphores
247c478bd9Sstevel@tonic-gate **
257c478bd9Sstevel@tonic-gate **	Parameters:
267c478bd9Sstevel@tonic-gate **		key -- key for semaphores.
277c478bd9Sstevel@tonic-gate **		nsem -- number of semaphores.
287c478bd9Sstevel@tonic-gate **		semflg -- flag for semget(), if 0, use a default.
297c478bd9Sstevel@tonic-gate **		owner -- create semaphores.
307c478bd9Sstevel@tonic-gate **
317c478bd9Sstevel@tonic-gate **	Returns:
327c478bd9Sstevel@tonic-gate **		id for semaphores.
337c478bd9Sstevel@tonic-gate **		< 0 on failure.
347c478bd9Sstevel@tonic-gate */
357c478bd9Sstevel@tonic-gate 
367c478bd9Sstevel@tonic-gate int
377c478bd9Sstevel@tonic-gate sm_sem_start(key, nsem, semflg, owner)
387c478bd9Sstevel@tonic-gate 	key_t key;
397c478bd9Sstevel@tonic-gate 	int nsem;
407c478bd9Sstevel@tonic-gate 	int semflg;
417c478bd9Sstevel@tonic-gate 	bool owner;
427c478bd9Sstevel@tonic-gate {
43*49218d4fSjbeck 	int semid, i, err;
447c478bd9Sstevel@tonic-gate 	unsigned short *semvals;
457c478bd9Sstevel@tonic-gate 
467c478bd9Sstevel@tonic-gate 	semvals = NULL;
477c478bd9Sstevel@tonic-gate 	if (semflg == 0)
487c478bd9Sstevel@tonic-gate 		semflg = (SEM_A|SEM_R)|((SEM_A|SEM_R) >> 3);
497c478bd9Sstevel@tonic-gate 	if (owner)
507c478bd9Sstevel@tonic-gate 		semflg |= IPC_CREAT|IPC_EXCL;
517c478bd9Sstevel@tonic-gate 	semid = semget(key, nsem, semflg);
527c478bd9Sstevel@tonic-gate 	if (semid < 0)
537c478bd9Sstevel@tonic-gate 		goto error;
547c478bd9Sstevel@tonic-gate 
557c478bd9Sstevel@tonic-gate 	if (owner)
567c478bd9Sstevel@tonic-gate 	{
577c478bd9Sstevel@tonic-gate 		union semun semarg;
587c478bd9Sstevel@tonic-gate 
597c478bd9Sstevel@tonic-gate 		semvals = (unsigned short *) sm_malloc(nsem * sizeof semvals);
607c478bd9Sstevel@tonic-gate 		if (semvals == NULL)
617c478bd9Sstevel@tonic-gate 			goto error;
627c478bd9Sstevel@tonic-gate 		semarg.array = semvals;
637c478bd9Sstevel@tonic-gate 
647c478bd9Sstevel@tonic-gate 		/* initialize semaphore values to be available */
657c478bd9Sstevel@tonic-gate 		for (i = 0; i < nsem; i++)
667c478bd9Sstevel@tonic-gate 			semvals[i] = 1;
677c478bd9Sstevel@tonic-gate 		if (semctl(semid, 0, SETALL, semarg) < 0)
687c478bd9Sstevel@tonic-gate 			goto error;
697c478bd9Sstevel@tonic-gate 	}
707c478bd9Sstevel@tonic-gate 	return semid;
717c478bd9Sstevel@tonic-gate 
727c478bd9Sstevel@tonic-gate error:
73*49218d4fSjbeck 	err = errno;
747c478bd9Sstevel@tonic-gate 	if (semvals != NULL)
757c478bd9Sstevel@tonic-gate 		sm_free(semvals);
767c478bd9Sstevel@tonic-gate 	if (semid >= 0)
777c478bd9Sstevel@tonic-gate 		sm_sem_stop(semid);
78*49218d4fSjbeck 	return (err > 0) ? (0 - err) : -1;
797c478bd9Sstevel@tonic-gate }
807c478bd9Sstevel@tonic-gate 
817c478bd9Sstevel@tonic-gate /*
827c478bd9Sstevel@tonic-gate **  SM_SEM_STOP -- stop using semaphores.
837c478bd9Sstevel@tonic-gate **
847c478bd9Sstevel@tonic-gate **	Parameters:
857c478bd9Sstevel@tonic-gate **		semid -- id for semaphores.
867c478bd9Sstevel@tonic-gate **
877c478bd9Sstevel@tonic-gate **	Returns:
887c478bd9Sstevel@tonic-gate **		0 on success.
897c478bd9Sstevel@tonic-gate **		< 0 on failure.
907c478bd9Sstevel@tonic-gate */
917c478bd9Sstevel@tonic-gate 
927c478bd9Sstevel@tonic-gate int
937c478bd9Sstevel@tonic-gate sm_sem_stop(semid)
947c478bd9Sstevel@tonic-gate 	int semid;
957c478bd9Sstevel@tonic-gate {
967c478bd9Sstevel@tonic-gate 	return semctl(semid, 0, IPC_RMID, NULL);
977c478bd9Sstevel@tonic-gate }
987c478bd9Sstevel@tonic-gate 
997c478bd9Sstevel@tonic-gate /*
1007c478bd9Sstevel@tonic-gate **  SM_SEM_ACQ -- acquire semaphore.
1017c478bd9Sstevel@tonic-gate **
1027c478bd9Sstevel@tonic-gate **	Parameters:
1037c478bd9Sstevel@tonic-gate **		semid -- id for semaphores.
1047c478bd9Sstevel@tonic-gate **		semnum -- number of semaphore.
1057c478bd9Sstevel@tonic-gate **		timeout -- how long to wait for operation to succeed.
1067c478bd9Sstevel@tonic-gate **
1077c478bd9Sstevel@tonic-gate **	Returns:
1087c478bd9Sstevel@tonic-gate **		0 on success.
1097c478bd9Sstevel@tonic-gate **		< 0 on failure.
1107c478bd9Sstevel@tonic-gate */
1117c478bd9Sstevel@tonic-gate 
1127c478bd9Sstevel@tonic-gate int
1137c478bd9Sstevel@tonic-gate sm_sem_acq(semid, semnum, timeout)
1147c478bd9Sstevel@tonic-gate 	int semid;
1157c478bd9Sstevel@tonic-gate 	int semnum;
1167c478bd9Sstevel@tonic-gate 	int timeout;
1177c478bd9Sstevel@tonic-gate {
1187c478bd9Sstevel@tonic-gate 	int r;
1197c478bd9Sstevel@tonic-gate 	struct sembuf semops[1];
1207c478bd9Sstevel@tonic-gate 
1217c478bd9Sstevel@tonic-gate 	semops[0].sem_num = semnum;
1227c478bd9Sstevel@tonic-gate 	semops[0].sem_op = -1;
1237c478bd9Sstevel@tonic-gate 	semops[0].sem_flg = SEM_UNDO |
1247c478bd9Sstevel@tonic-gate 			    (timeout != SM_TIME_FOREVER ? 0 : IPC_NOWAIT);
1257c478bd9Sstevel@tonic-gate 	if (timeout == SM_TIME_IMMEDIATE || timeout == SM_TIME_FOREVER)
1267c478bd9Sstevel@tonic-gate 		return semop(semid, semops, 1);
1277c478bd9Sstevel@tonic-gate 	do
1287c478bd9Sstevel@tonic-gate 	{
1297c478bd9Sstevel@tonic-gate 		r = semop(semid, semops, 1);
1307c478bd9Sstevel@tonic-gate 		if (r == 0)
1317c478bd9Sstevel@tonic-gate 			return r;
1327c478bd9Sstevel@tonic-gate 		sleep(1);
1337c478bd9Sstevel@tonic-gate 		--timeout;
1347c478bd9Sstevel@tonic-gate 	} while (timeout > 0);
1357c478bd9Sstevel@tonic-gate 	return r;
1367c478bd9Sstevel@tonic-gate }
1377c478bd9Sstevel@tonic-gate 
1387c478bd9Sstevel@tonic-gate /*
1397c478bd9Sstevel@tonic-gate **  SM_SEM_REL -- release semaphore.
1407c478bd9Sstevel@tonic-gate **
1417c478bd9Sstevel@tonic-gate **	Parameters:
1427c478bd9Sstevel@tonic-gate **		semid -- id for semaphores.
1437c478bd9Sstevel@tonic-gate **		semnum -- number of semaphore.
1447c478bd9Sstevel@tonic-gate **		timeout -- how long to wait for operation to succeed.
1457c478bd9Sstevel@tonic-gate **
1467c478bd9Sstevel@tonic-gate **	Returns:
1477c478bd9Sstevel@tonic-gate **		0 on success.
1487c478bd9Sstevel@tonic-gate **		< 0 on failure.
1497c478bd9Sstevel@tonic-gate */
1507c478bd9Sstevel@tonic-gate 
1517c478bd9Sstevel@tonic-gate int
1527c478bd9Sstevel@tonic-gate sm_sem_rel(semid, semnum, timeout)
1537c478bd9Sstevel@tonic-gate 	int semid;
1547c478bd9Sstevel@tonic-gate 	int semnum;
1557c478bd9Sstevel@tonic-gate 	int timeout;
1567c478bd9Sstevel@tonic-gate {
1577c478bd9Sstevel@tonic-gate 	int r;
1587c478bd9Sstevel@tonic-gate 	struct sembuf semops[1];
1597c478bd9Sstevel@tonic-gate 
1607c478bd9Sstevel@tonic-gate #if PARANOID
1617c478bd9Sstevel@tonic-gate 	/* XXX should we check whether the value is already 0 ? */
1627c478bd9Sstevel@tonic-gate 	SM_REQUIRE(sm_get_sem(semid, semnum) > 0);
1637c478bd9Sstevel@tonic-gate #endif /* PARANOID */
1647c478bd9Sstevel@tonic-gate 
1657c478bd9Sstevel@tonic-gate 	semops[0].sem_num = semnum;
1667c478bd9Sstevel@tonic-gate 	semops[0].sem_op = 1;
1677c478bd9Sstevel@tonic-gate 	semops[0].sem_flg = SEM_UNDO |
1687c478bd9Sstevel@tonic-gate 			    (timeout != SM_TIME_FOREVER ? 0 : IPC_NOWAIT);
1697c478bd9Sstevel@tonic-gate 	if (timeout == SM_TIME_IMMEDIATE || timeout == SM_TIME_FOREVER)
1707c478bd9Sstevel@tonic-gate 		return semop(semid, semops, 1);
1717c478bd9Sstevel@tonic-gate 	do
1727c478bd9Sstevel@tonic-gate 	{
1737c478bd9Sstevel@tonic-gate 		r = semop(semid, semops, 1);
1747c478bd9Sstevel@tonic-gate 		if (r == 0)
1757c478bd9Sstevel@tonic-gate 			return r;
1767c478bd9Sstevel@tonic-gate 		sleep(1);
1777c478bd9Sstevel@tonic-gate 		--timeout;
1787c478bd9Sstevel@tonic-gate 	} while (timeout > 0);
1797c478bd9Sstevel@tonic-gate 	return r;
1807c478bd9Sstevel@tonic-gate }
1817c478bd9Sstevel@tonic-gate 
1827c478bd9Sstevel@tonic-gate /*
1837c478bd9Sstevel@tonic-gate **  SM_SEM_GET -- get semaphore value.
1847c478bd9Sstevel@tonic-gate **
1857c478bd9Sstevel@tonic-gate **	Parameters:
1867c478bd9Sstevel@tonic-gate **		semid -- id for semaphores.
1877c478bd9Sstevel@tonic-gate **		semnum -- number of semaphore.
1887c478bd9Sstevel@tonic-gate **
1897c478bd9Sstevel@tonic-gate **	Returns:
1907c478bd9Sstevel@tonic-gate **		value of semaphore on success.
1917c478bd9Sstevel@tonic-gate **		< 0 on failure.
1927c478bd9Sstevel@tonic-gate */
1937c478bd9Sstevel@tonic-gate 
1947c478bd9Sstevel@tonic-gate int
1957c478bd9Sstevel@tonic-gate sm_sem_get(semid, semnum)
1967c478bd9Sstevel@tonic-gate 	int semid;
1977c478bd9Sstevel@tonic-gate 	int semnum;
1987c478bd9Sstevel@tonic-gate {
1997c478bd9Sstevel@tonic-gate 	int semval;
2007c478bd9Sstevel@tonic-gate 
2017c478bd9Sstevel@tonic-gate 	if ((semval = semctl(semid, semnum, GETVAL, NULL)) < 0)
2027c478bd9Sstevel@tonic-gate 		return -1;
2037c478bd9Sstevel@tonic-gate 	return semval;
2047c478bd9Sstevel@tonic-gate }
2057c478bd9Sstevel@tonic-gate #endif /* SM_CONF_SEM */
206