xref: /titanic_51/usr/src/lib/libslp/clib/SLPOpen.c (revision 7c478bd95313f5f23a4c958a745db2134aa03244)
1*7c478bd9Sstevel@tonic-gate /*
2*7c478bd9Sstevel@tonic-gate  * CDDL HEADER START
3*7c478bd9Sstevel@tonic-gate  *
4*7c478bd9Sstevel@tonic-gate  * The contents of this file are subject to the terms of the
5*7c478bd9Sstevel@tonic-gate  * Common Development and Distribution License, Version 1.0 only
6*7c478bd9Sstevel@tonic-gate  * (the "License").  You may not use this file except in compliance
7*7c478bd9Sstevel@tonic-gate  * with the License.
8*7c478bd9Sstevel@tonic-gate  *
9*7c478bd9Sstevel@tonic-gate  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10*7c478bd9Sstevel@tonic-gate  * or http://www.opensolaris.org/os/licensing.
11*7c478bd9Sstevel@tonic-gate  * See the License for the specific language governing permissions
12*7c478bd9Sstevel@tonic-gate  * and limitations under the License.
13*7c478bd9Sstevel@tonic-gate  *
14*7c478bd9Sstevel@tonic-gate  * When distributing Covered Code, include this CDDL HEADER in each
15*7c478bd9Sstevel@tonic-gate  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16*7c478bd9Sstevel@tonic-gate  * If applicable, add the following below this CDDL HEADER, with the
17*7c478bd9Sstevel@tonic-gate  * fields enclosed by brackets "[]" replaced with your own identifying
18*7c478bd9Sstevel@tonic-gate  * information: Portions Copyright [yyyy] [name of copyright owner]
19*7c478bd9Sstevel@tonic-gate  *
20*7c478bd9Sstevel@tonic-gate  * CDDL HEADER END
21*7c478bd9Sstevel@tonic-gate  */
22*7c478bd9Sstevel@tonic-gate /*
23*7c478bd9Sstevel@tonic-gate  * Copyright (c) 1999 by Sun Microsystems, Inc.
24*7c478bd9Sstevel@tonic-gate  * All rights reserved.
25*7c478bd9Sstevel@tonic-gate  */
26*7c478bd9Sstevel@tonic-gate 
27*7c478bd9Sstevel@tonic-gate #pragma ident	"%Z%%M%	%I%	%E% SMI"
28*7c478bd9Sstevel@tonic-gate 
29*7c478bd9Sstevel@tonic-gate #include <stdio.h>
30*7c478bd9Sstevel@tonic-gate #include <stdlib.h>
31*7c478bd9Sstevel@tonic-gate #include <thread.h>
32*7c478bd9Sstevel@tonic-gate #include <synch.h>
33*7c478bd9Sstevel@tonic-gate #include <syslog.h>
34*7c478bd9Sstevel@tonic-gate #include <arpa/inet.h>
35*7c478bd9Sstevel@tonic-gate #include <slp-internal.h>
36*7c478bd9Sstevel@tonic-gate #include <slp_net_utils.h>
37*7c478bd9Sstevel@tonic-gate 
38*7c478bd9Sstevel@tonic-gate SLPError SLPOpen(const char *pcLang, SLPBoolean isAsync, SLPHandle *phSLP) {
39*7c478bd9Sstevel@tonic-gate 	slp_handle_impl_t *hp;
40*7c478bd9Sstevel@tonic-gate 
41*7c478bd9Sstevel@tonic-gate 	if (!pcLang || !phSLP) {
42*7c478bd9Sstevel@tonic-gate 		return (SLP_PARAMETER_BAD);
43*7c478bd9Sstevel@tonic-gate 	}
44*7c478bd9Sstevel@tonic-gate 
45*7c478bd9Sstevel@tonic-gate 	/* allocate the handle */
46*7c478bd9Sstevel@tonic-gate 	if (!(hp = malloc(sizeof (*hp)))) {
47*7c478bd9Sstevel@tonic-gate 		slp_err(LOG_CRIT, 0, "SLPOpen", "out of memory");
48*7c478bd9Sstevel@tonic-gate 		return (SLP_MEMORY_ALLOC_FAILED);
49*7c478bd9Sstevel@tonic-gate 	}
50*7c478bd9Sstevel@tonic-gate 
51*7c478bd9Sstevel@tonic-gate 	/* initialize outcall synchronization */
52*7c478bd9Sstevel@tonic-gate 	hp->pending_outcall = SLP_FALSE;
53*7c478bd9Sstevel@tonic-gate 	(void) mutex_init(&(hp->outcall_lock), NULL, NULL);
54*7c478bd9Sstevel@tonic-gate 	(void) cond_init(&(hp->outcall_cv), NULL, NULL);
55*7c478bd9Sstevel@tonic-gate 	hp->close_on_end = SLP_FALSE;
56*7c478bd9Sstevel@tonic-gate 	hp->consumer_tid = 0;
57*7c478bd9Sstevel@tonic-gate 
58*7c478bd9Sstevel@tonic-gate 	/* locale property overrides argument */
59*7c478bd9Sstevel@tonic-gate 	if (!(hp->locale = SLPGetProperty(SLP_CONFIG_LOCALE))) {
60*7c478bd9Sstevel@tonic-gate 		hp->locale = pcLang;
61*7c478bd9Sstevel@tonic-gate 	}
62*7c478bd9Sstevel@tonic-gate 	/* Make sure the language string is under our ownership */
63*7c478bd9Sstevel@tonic-gate 	if (!(hp->locale = strdup(hp->locale))) {
64*7c478bd9Sstevel@tonic-gate 		free(hp);
65*7c478bd9Sstevel@tonic-gate 		slp_err(LOG_CRIT, 0, "SLPOpen", "out of memory");
66*7c478bd9Sstevel@tonic-gate 		return (SLP_MEMORY_ALLOC_FAILED);
67*7c478bd9Sstevel@tonic-gate 	}
68*7c478bd9Sstevel@tonic-gate 
69*7c478bd9Sstevel@tonic-gate 	hp->cancel = 0;
70*7c478bd9Sstevel@tonic-gate 
71*7c478bd9Sstevel@tonic-gate 	/* Asynchronous operation? */
72*7c478bd9Sstevel@tonic-gate 	if (isAsync)
73*7c478bd9Sstevel@tonic-gate 		hp->async = SLP_TRUE;
74*7c478bd9Sstevel@tonic-gate 	else
75*7c478bd9Sstevel@tonic-gate 		hp->async = SLP_FALSE;
76*7c478bd9Sstevel@tonic-gate 
77*7c478bd9Sstevel@tonic-gate 	/* TCP vars -- these are NULL until actually needed */
78*7c478bd9Sstevel@tonic-gate 	hp->tcp_lock = NULL;
79*7c478bd9Sstevel@tonic-gate 	hp->tcp_wait = NULL;
80*7c478bd9Sstevel@tonic-gate 	hp->tcp_ref_cnt = 0;
81*7c478bd9Sstevel@tonic-gate 
82*7c478bd9Sstevel@tonic-gate 	/* Consumer / Producer pipe */
83*7c478bd9Sstevel@tonic-gate 	hp->q = NULL;
84*7c478bd9Sstevel@tonic-gate 
85*7c478bd9Sstevel@tonic-gate 	/* Interface info, loaded on demand */
86*7c478bd9Sstevel@tonic-gate 	hp->ifinfo = NULL;
87*7c478bd9Sstevel@tonic-gate 
88*7c478bd9Sstevel@tonic-gate 	/* force multicast, false by default */
89*7c478bd9Sstevel@tonic-gate 	hp->force_multicast = SLP_FALSE;
90*7c478bd9Sstevel@tonic-gate 
91*7c478bd9Sstevel@tonic-gate 	/* internal call, false by default */
92*7c478bd9Sstevel@tonic-gate 	hp->internal_call = SLP_FALSE;
93*7c478bd9Sstevel@tonic-gate 
94*7c478bd9Sstevel@tonic-gate 	*phSLP = hp;
95*7c478bd9Sstevel@tonic-gate 	return (SLP_OK);
96*7c478bd9Sstevel@tonic-gate }
97*7c478bd9Sstevel@tonic-gate 
98*7c478bd9Sstevel@tonic-gate void slp_cleanup_handle(slp_handle_impl_t *hp) {
99*7c478bd9Sstevel@tonic-gate 	/* free the handle */
100*7c478bd9Sstevel@tonic-gate 	if (hp->tcp_lock) free(hp->tcp_lock);
101*7c478bd9Sstevel@tonic-gate 	if (hp->tcp_wait) free(hp->tcp_wait);
102*7c478bd9Sstevel@tonic-gate 	if (hp->ifinfo) {
103*7c478bd9Sstevel@tonic-gate 		slp_free_ifinfo(hp->ifinfo);
104*7c478bd9Sstevel@tonic-gate 		free(hp->ifinfo);
105*7c478bd9Sstevel@tonic-gate 	}
106*7c478bd9Sstevel@tonic-gate 	free((void *) hp->locale);
107*7c478bd9Sstevel@tonic-gate 	free(hp);
108*7c478bd9Sstevel@tonic-gate }
109*7c478bd9Sstevel@tonic-gate 
110*7c478bd9Sstevel@tonic-gate void SLPClose(SLPHandle hSLP) {
111*7c478bd9Sstevel@tonic-gate 	slp_handle_impl_t *hp = (slp_handle_impl_t *)hSLP;
112*7c478bd9Sstevel@tonic-gate 
113*7c478bd9Sstevel@tonic-gate 	if (!hSLP) {
114*7c478bd9Sstevel@tonic-gate 		return;
115*7c478bd9Sstevel@tonic-gate 	}
116*7c478bd9Sstevel@tonic-gate 
117*7c478bd9Sstevel@tonic-gate 	/*
118*7c478bd9Sstevel@tonic-gate 	 * If an outcall is pending on this handle:
119*7c478bd9Sstevel@tonic-gate 	 *   If we are being called from a callback resulting
120*7c478bd9Sstevel@tonic-gate 	 *   from the outcall associated with this handle or
121*7c478bd9Sstevel@tonic-gate 	 *   if close_on_end has already been set:
122*7c478bd9Sstevel@tonic-gate 	 *	just set close on end and return -- the cleanup
123*7c478bd9Sstevel@tonic-gate 	 *	will be done when the outcall is finished.
124*7c478bd9Sstevel@tonic-gate 	 *   else
125*7c478bd9Sstevel@tonic-gate 	 *	wait on the outcall cv for the outcall to complete
126*7c478bd9Sstevel@tonic-gate 	 * Proceed with cleanup
127*7c478bd9Sstevel@tonic-gate 	 */
128*7c478bd9Sstevel@tonic-gate 	(void) mutex_lock(&(hp->outcall_lock));
129*7c478bd9Sstevel@tonic-gate 	if (hp->pending_outcall) {
130*7c478bd9Sstevel@tonic-gate 	    /* end the consumer thread */
131*7c478bd9Sstevel@tonic-gate 	    /* this will also kill the producer thread and close net */
132*7c478bd9Sstevel@tonic-gate 	    hp->cancel = 1;
133*7c478bd9Sstevel@tonic-gate 	    if (hp->q) {
134*7c478bd9Sstevel@tonic-gate 		if (slp_enqueue_at_head(hp->q, NULL) != SLP_OK) {
135*7c478bd9Sstevel@tonic-gate 		    goto cleanup;
136*7c478bd9Sstevel@tonic-gate 		}
137*7c478bd9Sstevel@tonic-gate 	    }
138*7c478bd9Sstevel@tonic-gate 
139*7c478bd9Sstevel@tonic-gate 	    if (thr_self() == hp->consumer_tid || hp->close_on_end) {
140*7c478bd9Sstevel@tonic-gate 		/* SLPClose called from callback */
141*7c478bd9Sstevel@tonic-gate 		hp->close_on_end = SLP_TRUE;
142*7c478bd9Sstevel@tonic-gate 		(void) mutex_unlock(&(hp->outcall_lock));
143*7c478bd9Sstevel@tonic-gate 		return;
144*7c478bd9Sstevel@tonic-gate 	    }
145*7c478bd9Sstevel@tonic-gate 	    /* else not called from callback; wait for outcall to end */
146*7c478bd9Sstevel@tonic-gate 	    while (hp->pending_outcall) {
147*7c478bd9Sstevel@tonic-gate 		(void) cond_wait(&(hp->outcall_cv), &(hp->outcall_lock));
148*7c478bd9Sstevel@tonic-gate 	    }
149*7c478bd9Sstevel@tonic-gate 	}
150*7c478bd9Sstevel@tonic-gate 	(void) mutex_unlock(&(hp->outcall_lock));
151*7c478bd9Sstevel@tonic-gate 
152*7c478bd9Sstevel@tonic-gate cleanup:
153*7c478bd9Sstevel@tonic-gate 	slp_cleanup_handle(hp);
154*7c478bd9Sstevel@tonic-gate }
155