xref: /illumos-gate/usr/src/lib/libc/port/rt/clock_timer.c (revision 5ad42b1b)
17c478bd9Sstevel@tonic-gate /*
27c478bd9Sstevel@tonic-gate  * CDDL HEADER START
37c478bd9Sstevel@tonic-gate  *
47c478bd9Sstevel@tonic-gate  * The contents of this file are subject to the terms of the
534709573Sraf  * Common Development and Distribution License (the "License").
634709573Sraf  * You may not use this file except in compliance with the License.
77c478bd9Sstevel@tonic-gate  *
87c478bd9Sstevel@tonic-gate  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
97c478bd9Sstevel@tonic-gate  * or http://www.opensolaris.org/os/licensing.
107c478bd9Sstevel@tonic-gate  * See the License for the specific language governing permissions
117c478bd9Sstevel@tonic-gate  * and limitations under the License.
127c478bd9Sstevel@tonic-gate  *
137c478bd9Sstevel@tonic-gate  * When distributing Covered Code, include this CDDL HEADER in each
147c478bd9Sstevel@tonic-gate  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
157c478bd9Sstevel@tonic-gate  * If applicable, add the following below this CDDL HEADER, with the
167c478bd9Sstevel@tonic-gate  * fields enclosed by brackets "[]" replaced with your own identifying
177c478bd9Sstevel@tonic-gate  * information: Portions Copyright [yyyy] [name of copyright owner]
187c478bd9Sstevel@tonic-gate  *
197c478bd9Sstevel@tonic-gate  * CDDL HEADER END
207c478bd9Sstevel@tonic-gate  */
2134709573Sraf 
227c478bd9Sstevel@tonic-gate /*
23*5ad42b1bSSurya Prakki  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
247c478bd9Sstevel@tonic-gate  * Use is subject to license terms.
257c478bd9Sstevel@tonic-gate  */
267c478bd9Sstevel@tonic-gate 
277257d1b4Sraf #include "lint.h"
287c478bd9Sstevel@tonic-gate #include <time.h>
297c478bd9Sstevel@tonic-gate #include <sys/types.h>
3034709573Sraf #include <stdlib.h>
3134709573Sraf #include <string.h>
3234709573Sraf #include <errno.h>
3334709573Sraf #include "sigev_thread.h"
3434709573Sraf 
35f841f6adSraf /*
36f841f6adSraf  * System call wrappers found elsewhere in libc (common/sys/__clock_timer.s).
37f841f6adSraf  */
38f841f6adSraf extern int __clock_getres(clockid_t, timespec_t *);
39f841f6adSraf extern int __clock_gettime(clockid_t, timespec_t *);
40f841f6adSraf extern int __clock_settime(clockid_t, const timespec_t *);
41f841f6adSraf extern int __timer_create(clockid_t, struct sigevent *, timer_t *);
42f841f6adSraf extern int __timer_delete(timer_t);
43f841f6adSraf extern int __timer_getoverrun(timer_t);
44f841f6adSraf extern int __timer_gettime(timer_t, itimerspec_t *);
45f841f6adSraf extern int __timer_settime(timer_t, int, const itimerspec_t *, itimerspec_t *);
46f841f6adSraf 
4734709573Sraf /*
4834709573Sraf  * Array of pointers to tcd's, indexed by timer id.
4934709573Sraf  * No more than 'timer_max' timers can be created by any process.
5034709573Sraf  */
5134709573Sraf int timer_max = 0;
5234709573Sraf thread_communication_data_t **timer_tcd;
5334709573Sraf static pthread_once_t timer_once = PTHREAD_ONCE_INIT;
5434709573Sraf 
5534709573Sraf static void
timer_init(void)5634709573Sraf timer_init(void)
5734709573Sraf {
5834709573Sraf 	timer_max = (int)_sysconf(_SC_TIMER_MAX);
5934709573Sraf 	timer_tcd = malloc(timer_max * sizeof (*timer_tcd));
6034709573Sraf 	(void) memset(timer_tcd, 0, timer_max * sizeof (*timer_tcd));
6134709573Sraf }
627c478bd9Sstevel@tonic-gate 
637c478bd9Sstevel@tonic-gate int
clock_getres(clockid_t clock_id,timespec_t * res)647257d1b4Sraf clock_getres(clockid_t clock_id, timespec_t *res)
657c478bd9Sstevel@tonic-gate {
667c478bd9Sstevel@tonic-gate 	return (__clock_getres(clock_id, res));
677c478bd9Sstevel@tonic-gate }
687c478bd9Sstevel@tonic-gate 
697c478bd9Sstevel@tonic-gate int
clock_gettime(clockid_t clock_id,timespec_t * tp)707257d1b4Sraf clock_gettime(clockid_t clock_id, timespec_t *tp)
717c478bd9Sstevel@tonic-gate {
727c478bd9Sstevel@tonic-gate 	return (__clock_gettime(clock_id, tp));
737c478bd9Sstevel@tonic-gate }
747c478bd9Sstevel@tonic-gate 
757c478bd9Sstevel@tonic-gate int
clock_settime(clockid_t clock_id,const timespec_t * tp)767257d1b4Sraf clock_settime(clockid_t clock_id, const timespec_t *tp)
777c478bd9Sstevel@tonic-gate {
787c478bd9Sstevel@tonic-gate 	return (__clock_settime(clock_id, tp));
797c478bd9Sstevel@tonic-gate }
807c478bd9Sstevel@tonic-gate 
817c478bd9Sstevel@tonic-gate int
timer_create(clockid_t clock_id,struct sigevent * sigevp,timer_t * timerid)827257d1b4Sraf timer_create(clockid_t clock_id, struct sigevent *sigevp, timer_t *timerid)
837c478bd9Sstevel@tonic-gate {
8434709573Sraf 	struct sigevent sigevent;
8534709573Sraf 	port_notify_t port_notify;
8634709573Sraf 	thread_communication_data_t *tcdp;
8734709573Sraf 	int sigev_thread = 0;
8834709573Sraf 	int rc;
8934709573Sraf 
9034709573Sraf 	(void) pthread_once(&timer_once, timer_init);
9134709573Sraf 
9234709573Sraf 	if (sigevp != NULL &&
9334709573Sraf 	    sigevp->sigev_notify == SIGEV_THREAD &&
9434709573Sraf 	    sigevp->sigev_notify_function != NULL) {
9534709573Sraf 		sigev_thread = 1;
9634709573Sraf 		tcdp = setup_sigev_handler(sigevp, TIMER);
9734709573Sraf 		if (tcdp == NULL)
9834709573Sraf 			return (-1);
9934709573Sraf 		/* copy the sigevent structure so we can modify it */
10034709573Sraf 		sigevent = *sigevp;
10134709573Sraf 		sigevp = &sigevent;
10234709573Sraf 		port_notify.portnfy_port = tcdp->tcd_port;
10334709573Sraf 		port_notify.portnfy_user = NULL;
10434709573Sraf 		sigevp->sigev_value.sival_ptr = &port_notify;
10534709573Sraf 	}
10634709573Sraf 
10734709573Sraf 	rc = __timer_create(clock_id, sigevp, timerid);
108f841f6adSraf 
109f841f6adSraf 	if (sigev_thread) {
110f841f6adSraf 		if (rc == 0) {
111f841f6adSraf 			if ((rc = launch_spawner(tcdp)) != 0)
112*5ad42b1bSSurya Prakki 				(void) __timer_delete(*timerid);
113f841f6adSraf 			else
114f841f6adSraf 				timer_tcd[*timerid] = tcdp;
115f841f6adSraf 		}
116f841f6adSraf 		if (rc != 0)
117f841f6adSraf 			free_sigev_handler(tcdp);
11834709573Sraf 	}
11934709573Sraf 
12034709573Sraf 	return (rc);
1217c478bd9Sstevel@tonic-gate }
1227c478bd9Sstevel@tonic-gate 
1237c478bd9Sstevel@tonic-gate int
timer_delete(timer_t timerid)1247257d1b4Sraf timer_delete(timer_t timerid)
1257c478bd9Sstevel@tonic-gate {
12634709573Sraf 	int rc;
12734709573Sraf 
12834709573Sraf 	if ((rc = del_sigev_timer(timerid)) == 0)
12934709573Sraf 		return (__timer_delete(timerid));
13034709573Sraf 	else
13134709573Sraf 		return (rc);
1327c478bd9Sstevel@tonic-gate }
1337c478bd9Sstevel@tonic-gate 
1347c478bd9Sstevel@tonic-gate int
timer_getoverrun(timer_t timerid)1357257d1b4Sraf timer_getoverrun(timer_t timerid)
1367c478bd9Sstevel@tonic-gate {
13734709573Sraf 	return (__timer_getoverrun(timerid) + sigev_timer_getoverrun(timerid));
1387c478bd9Sstevel@tonic-gate }
1397c478bd9Sstevel@tonic-gate 
1407c478bd9Sstevel@tonic-gate int
timer_gettime(timer_t timerid,itimerspec_t * value)1417257d1b4Sraf timer_gettime(timer_t timerid, itimerspec_t *value)
1427c478bd9Sstevel@tonic-gate {
1437c478bd9Sstevel@tonic-gate 	return (__timer_gettime(timerid, value));
1447c478bd9Sstevel@tonic-gate }
1457c478bd9Sstevel@tonic-gate 
1467c478bd9Sstevel@tonic-gate int
timer_settime(timer_t timerid,int flags,const itimerspec_t * value,itimerspec_t * ovalue)1477257d1b4Sraf timer_settime(timer_t timerid, int flags, const itimerspec_t *value,
14834709573Sraf 	itimerspec_t *ovalue)
1497c478bd9Sstevel@tonic-gate {
1507c478bd9Sstevel@tonic-gate 	return (__timer_settime(timerid, flags, value, ovalue));
1517c478bd9Sstevel@tonic-gate }
1527c478bd9Sstevel@tonic-gate 
153f841f6adSraf /*
154f841f6adSraf  * Cleanup after fork1() in the child process.
155f841f6adSraf  */
156f841f6adSraf void
postfork1_child_sigev_timer(void)157f841f6adSraf postfork1_child_sigev_timer(void)
1587c478bd9Sstevel@tonic-gate {
159f841f6adSraf 	thread_communication_data_t *tcdp;
160f841f6adSraf 	int timer;
1617c478bd9Sstevel@tonic-gate 
162f841f6adSraf 	for (timer = 0; timer < timer_max; timer++) {
163f841f6adSraf 		if ((tcdp = timer_tcd[timer]) != NULL) {
164f841f6adSraf 			timer_tcd[timer] = NULL;
165f841f6adSraf 			tcd_teardown(tcdp);
166f841f6adSraf 		}
167f841f6adSraf 	}
1687c478bd9Sstevel@tonic-gate }
169