xref: /titanic_44/usr/src/cmd/picl/plugins/sun4u/snowbird/envmond/piclenvmond.c (revision 7c478bd95313f5f23a4c958a745db2134aa03244)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License, Version 1.0 only
6  * (the "License").  You may not use this file except in compliance
7  * with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or http://www.opensolaris.org/os/licensing.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  */
22 /*
23  * Copyright 2004 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 #pragma ident	"%Z%%M%	%I%	%E% SMI"
28 
29 #include <unistd.h>
30 #include <picl.h>
31 #include <picltree.h>
32 #include <picldefs.h>
33 #include <pthread.h>
34 #include <syslog.h>
35 #include <string.h>
36 #include <libnvpair.h>
37 #include <libintl.h>
38 #include "piclenvmond.h"
39 
40 /* external funcs and varaibles */
41 extern void env_handle_event(const char *, const void *, size_t);
42 extern picl_errno_t env_init();
43 extern void env_platmod_fini();
44 extern int sensor_fd;
45 extern pthread_t env_temp_thr_tid;
46 
47 /* local defines */
48 #define	TIMEOUT	(10)
49 
50 #pragma	init(piclenvmond_register)
51 
52 /*
53  * Plugin registration entry points
54  */
55 static void piclenvmond_register(void);
56 static void piclenvmond_init(void);
57 static void piclenvmond_fini(void);
58 static void piclenvmond_evhandler(const char *, const void *, size_t, void *);
59 
60 int env_debug = 0x0;
61 
62 static picld_plugin_reg_t envmond_reg_info = {
63 	PICLD_PLUGIN_VERSION_1,
64 	PICLD_PLUGIN_CRITICAL,
65 	"SUNW_piclenvmond",
66 	piclenvmond_init,
67 	piclenvmond_fini
68 };
69 
70 typedef struct {
71 	picl_nodehdl_t nodehdl;
72 	char node_name[PICL_PROPNAMELEN_MAX];
73 } env_callback_args_t;
74 
75 /*
76  * picld entry points
77  */
78 static void
piclenvmond_register(void)79 piclenvmond_register(void)
80 {
81 	(void) picld_plugin_register(&envmond_reg_info);
82 }
83 
84 /*
85  * picld entry point
86  *  - do all the initialization
87  *  - register for interested picl events
88  */
89 static void
piclenvmond_init(void)90 piclenvmond_init(void)
91 {
92 	picl_errno_t rc = PICL_SUCCESS;
93 
94 	if ((rc = env_init()) != PICL_SUCCESS) {
95 		syslog(LOG_ERR, gettext("SUNW_envmond:envmond init failed, "
96 			"error = %d"), rc);
97 		return;
98 	}
99 
100 	/* register handler for state change events */
101 	(void) ptree_register_handler(PICLEVENT_STATE_CHANGE,
102 		piclenvmond_evhandler, NULL);
103 	/* register handler for condition change events */
104 	(void) ptree_register_handler(PICLEVENT_CONDITION_CHANGE,
105 		piclenvmond_evhandler, NULL);
106 
107 }
108 
109 static void
piclenvmond_fini(void)110 piclenvmond_fini(void)
111 {
112 	void		*exitval;
113 
114 	/* unregister event handler */
115 	(void) ptree_unregister_handler(PICLEVENT_STATE_CHANGE,
116 		piclenvmond_evhandler, NULL);
117 	(void) ptree_unregister_handler(PICLEVENT_CONDITION_CHANGE,
118 		piclenvmond_evhandler, NULL);
119 
120 	/* cancel all the threads */
121 	(void) pthread_cancel(env_temp_thr_tid);
122 	(void) pthread_join(env_temp_thr_tid, &exitval);
123 
124 	/* do any platform specific cleanups required */
125 	env_platmod_fini();
126 	(void) close(sensor_fd);
127 }
128 
129 /*ARGSUSED*/
130 static void
piclenvmond_evhandler(const char * ename,const void * earg,size_t size,void * cookie)131 piclenvmond_evhandler(const char *ename, const void *earg, size_t size,
132 	void *cookie)
133 {
134 	env_handle_event(ename, earg, size);
135 }
136 
137 /*
138  * Utility functions
139  */
140 
141 /*
142  * create_property -- Create a PICL property
143  */
144 picl_errno_t
env_create_property(int ptype,int pmode,size_t psize,char * pname,int (* readfn)(ptree_rarg_t *,void *),int (* writefn)(ptree_warg_t *,const void *),picl_nodehdl_t nodeh,picl_prophdl_t * propp,void * vbuf)145 env_create_property(int ptype, int pmode, size_t psize, char *pname,
146 	int (*readfn)(ptree_rarg_t *, void *),
147 	int (*writefn)(ptree_warg_t *, const void *),
148 	picl_nodehdl_t nodeh, picl_prophdl_t *propp, void *vbuf)
149 {
150 	picl_errno_t		rc;		/* return code */
151 	ptree_propinfo_t	propinfo;	/* propinfo structure */
152 	picl_prophdl_t		proph;
153 
154 	rc = ptree_get_prop_by_name(nodeh, pname, &proph);
155 	if (rc == PICL_SUCCESS) {	/* prop. already exists */
156 		return (rc);
157 	}
158 
159 	rc = ptree_init_propinfo(&propinfo, PTREE_PROPINFO_VERSION,
160 		ptype, pmode, psize, pname, readfn, writefn);
161 	if (rc != PICL_SUCCESS) {
162 		syslog(LOG_ERR, PTREE_INIT_PROPINFO_FAILED_MSG, rc);
163 		return (rc);
164 	}
165 
166 	rc = ptree_create_and_add_prop(nodeh, &propinfo, vbuf, propp);
167 	if (rc != PICL_SUCCESS) {
168 		syslog(LOG_ERR, PTREE_CREATE_AND_ADD_PROP_FAILED_MSG, rc);
169 		return (rc);
170 	}
171 	return (PICL_SUCCESS);
172 }
173 
174 /*
175  * The picl event completion handler.
176  */
177 /* ARGSUSED */
178 static void
event_completion_handler(char * ename,void * earg,size_t size)179 event_completion_handler(char *ename, void *earg, size_t size)
180 {
181 	free(earg);
182 	free(ename);
183 }
184 
185 /*
186  * utility routine to post PICL events
187  */
188 /*ARGSUSED*/
189 static int
post_picl_event(const char * ename,char * envl,size_t elen,picl_nodehdl_t nodeh,int cond_wait)190 post_picl_event(const char *ename, char	*envl, size_t elen,
191 	picl_nodehdl_t nodeh, int cond_wait)
192 {
193 	nvlist_t	*nvlp;
194 	size_t		nvl_size;
195 	char		*pack_buf = NULL;
196 	char		*evname;
197 
198 	if (nodeh == 0) {
199 		return (PICL_FAILURE);
200 	}
201 	if ((evname = strdup(ename)) == NULL)
202 		return (PICL_FAILURE);
203 	if (envl) {
204 		if (nvlist_unpack(envl, elen, &nvlp, 0) < 0) {
205 			nvlist_free(nvlp);
206 			free(evname);
207 			return (PICL_FAILURE);
208 		}
209 	} else {
210 		if (nvlist_alloc(&nvlp, NV_UNIQUE_NAME_TYPE, NULL)) {
211 			free(evname);
212 			return (PICL_FAILURE);
213 		}
214 	}
215 
216 	if (nvlist_add_uint64(nvlp, PICLEVENTARG_NODEHANDLE, nodeh) == -1) {
217 		nvlist_free(nvlp);
218 		free(evname);
219 		return (PICL_FAILURE);
220 	}
221 
222 	if (nvlist_pack(nvlp, &pack_buf, &nvl_size, NV_ENCODE_NATIVE, NULL)) {
223 		nvlist_free(nvlp);
224 		free(evname);
225 		return (PICL_FAILURE);
226 	}
227 	nvlist_free(nvlp);
228 
229 	if (env_debug & EVENTS) {
230 		char enodename[PICL_PROPNAMELEN_MAX];
231 		if (ptree_get_propval_by_name(nodeh, PICL_PROP_NAME,
232 			enodename, sizeof (enodename)) == PICL_SUCCESS)
233 			syslog(LOG_INFO, "envmond:Posting %s on %s\n",
234 				ename, enodename);
235 	}
236 
237 	if (ptree_post_event(evname, pack_buf, nvl_size,
238 		event_completion_handler) != 0) {
239 		syslog(LOG_ERR, gettext("SUNW_envmond: Error posting %s PICL"
240 			" event."), ename);
241 		free(pack_buf);
242 		free(evname);
243 		return (PICL_FAILURE);
244 	}
245 	return (PICL_SUCCESS);
246 }
247 
248 /*
249  * post dr_req events
250  */
251 picl_errno_t
post_dr_req_event(picl_nodehdl_t fruh,char * dr_req_type,uint8_t wait)252 post_dr_req_event(picl_nodehdl_t fruh, char *dr_req_type, uint8_t wait)
253 {
254 	nvlist_t	*nvlp;	/* nvlist of event specific args */
255 	size_t		nvl_size;
256 	char		*pack_buf = NULL;
257 	char		dr_ap_id[PICL_PROPNAMELEN_MAX];
258 	int rc = PICL_SUCCESS;
259 
260 	if (env_debug & DEBUG)
261 		syslog(LOG_DEBUG, "Post %s on %llx", dr_req_type, fruh);
262 	if (fruh == 0) {
263 		return (PICL_INVALIDARG);
264 	}
265 	if ((rc = ptree_get_propval_by_name(fruh, PICL_PROP_NAME,
266 		dr_ap_id, sizeof (dr_ap_id))) != PICL_SUCCESS) {
267 		return (rc);
268 	}
269 
270 	if (nvlist_alloc(&nvlp, NV_UNIQUE_NAME_TYPE, NULL)) {
271 		return (PICL_FAILURE);
272 	}
273 
274 	if (nvlist_add_string(nvlp, PICLEVENTARG_AP_ID, dr_ap_id) == -1) {
275 		nvlist_free(nvlp);
276 		return (PICL_FAILURE);
277 	}
278 	if (nvlist_add_string(nvlp, PICLEVENTARG_DR_REQ_TYPE, dr_req_type)
279 		== -1) {
280 		nvlist_free(nvlp);
281 		return (PICL_FAILURE);
282 	}
283 	if (nvlist_pack(nvlp, &pack_buf, &nvl_size, NV_ENCODE_NATIVE, NULL)) {
284 		nvlist_free(nvlp);
285 		return (PICL_FAILURE);
286 	}
287 	nvlist_free(nvlp);
288 
289 	if (env_debug & DEBUG)
290 		syslog(LOG_DEBUG, "Posting %s on %s", dr_req_type, dr_ap_id);
291 	rc = post_picl_event(PICLEVENT_DR_REQ, pack_buf, nvl_size, fruh,
292 		wait);
293 
294 	free(pack_buf);
295 	return (rc);
296 }
297 
298 /*
299  * routine to post dr_ap_state change events
300  */
301 picl_errno_t
post_dr_ap_state_change_event(picl_nodehdl_t nodehdl,char * dr_hint,uint8_t wait)302 post_dr_ap_state_change_event(picl_nodehdl_t nodehdl, char *dr_hint,
303 	uint8_t wait)
304 {
305 	nvlist_t	*nvlp;	/* nvlist of event specific args */
306 	size_t		nvl_size;
307 	char		*pack_buf = NULL;
308 	char		dr_ap_id[PICL_PROPNAMELEN_MAX];
309 	int rc = PICL_SUCCESS;
310 
311 	if (nodehdl == 0) {
312 		return (PICL_FAILURE);
313 	}
314 	if ((rc = ptree_get_propval_by_name(nodehdl, PICL_PROP_NAME,
315 		dr_ap_id, sizeof (dr_ap_id))) != PICL_SUCCESS) {
316 		return (rc);
317 	}
318 	if (nvlist_alloc(&nvlp, NV_UNIQUE_NAME_TYPE, NULL)) {
319 		return (PICL_FAILURE);
320 	}
321 
322 	if (nvlist_add_string(nvlp, PICLEVENTARG_AP_ID, dr_ap_id) == -1) {
323 		nvlist_free(nvlp);
324 		return (PICL_FAILURE);
325 	}
326 	if (nvlist_add_string(nvlp, PICLEVENTARG_HINT, dr_hint) == -1) {
327 		nvlist_free(nvlp);
328 		return (PICL_FAILURE);
329 	}
330 	if (nvlist_pack(nvlp, &pack_buf, &nvl_size, NV_ENCODE_NATIVE, NULL)) {
331 		nvlist_free(nvlp);
332 		return (PICL_FAILURE);
333 	}
334 	nvlist_free(nvlp);
335 	rc = post_picl_event(PICLEVENT_DR_AP_STATE_CHANGE, pack_buf,
336 		nvl_size, nodehdl, wait);
337 	free(pack_buf);
338 	return (rc);
339 }
340 
341 picl_errno_t
post_cpu_state_change_event(picl_nodehdl_t fruh,char * event_type,uint8_t wait)342 post_cpu_state_change_event(picl_nodehdl_t fruh, char *event_type, uint8_t wait)
343 {
344 	nvlist_t	*nvlp;	/* nvlist of event specific args */
345 	size_t		nvl_size;
346 	char		*pack_buf = NULL;
347 	int rc = PICL_SUCCESS;
348 
349 	if (fruh == 0) {
350 		return (PICL_FAILURE);
351 	}
352 
353 	if (nvlist_alloc(&nvlp, NV_UNIQUE_NAME_TYPE, NULL))
354 		return (PICL_FAILURE);
355 
356 	if (nvlist_add_int64(nvlp, PICLEVENTARG_NODEHANDLE, fruh)) {
357 		nvlist_free(nvlp);
358 		return (PICL_FAILURE);
359 	}
360 
361 	if (nvlist_add_string(nvlp, PICLEVENTARG_CPU_EV_TYPE,
362 		event_type) == -1) {
363 		nvlist_free(nvlp);
364 		return (PICL_FAILURE);
365 	}
366 
367 	if (nvlist_pack(nvlp, &pack_buf, &nvl_size, NV_ENCODE_NATIVE, NULL)) {
368 		nvlist_free(nvlp);
369 		return (PICL_FAILURE);
370 	}
371 	nvlist_free(nvlp);
372 	rc = post_picl_event(PICLEVENT_CPU_STATE_CHANGE, pack_buf,
373 		nvl_size, fruh, wait);
374 	free(pack_buf);
375 	return (rc);
376 }
377 
378 int
post_sensor_event(picl_nodehdl_t hdl,char * sensor_evalue,uint8_t wait)379 post_sensor_event(picl_nodehdl_t hdl, char *sensor_evalue, uint8_t wait)
380 {
381 	nvlist_t	*nvlp;	/* nvlist of event specific args */
382 	size_t		nvl_size;
383 	char		*pack_buf = NULL;
384 	char		dr_ap_id[PICL_PROPNAMELEN_MAX];
385 	int rc = PICL_SUCCESS;
386 
387 	if (env_debug & DEBUG)
388 		syslog(LOG_DEBUG, "Post %s on %llx", sensor_evalue, hdl);
389 	if (hdl == 0)
390 		return (PICL_FAILURE);
391 
392 	if (nvlist_alloc(&nvlp, NV_UNIQUE_NAME_TYPE, NULL))
393 		return (PICL_FAILURE);
394 
395 	if (nvlist_add_string(nvlp, PICLEVENTARG_CONDITION,
396 		sensor_evalue) == -1) {
397 		nvlist_free(nvlp);
398 		return (PICL_FAILURE);
399 	}
400 	if (nvlist_pack(nvlp, &pack_buf, &nvl_size, NV_ENCODE_NATIVE, NULL)) {
401 		nvlist_free(nvlp);
402 		return (PICL_FAILURE);
403 	}
404 	nvlist_free(nvlp);
405 
406 	if (env_debug & DEBUG) {
407 		if (ptree_get_propval_by_name(hdl, PICL_PROP_NAME, dr_ap_id,
408 			sizeof (dr_ap_id)) == PICL_SUCCESS)
409 			syslog(LOG_DEBUG, "Posting %s on %s", sensor_evalue,
410 				dr_ap_id);
411 	}
412 	rc = post_picl_event(PICLEVENT_CONDITION_CHANGE, pack_buf, nvl_size,
413 		hdl, wait);
414 	free(pack_buf);
415 	return (rc);
416 }
417 
418 /*
419  * return B_TRUE if admin lock is enabled
420  * return B_FALSE if admin lock is disabled
421  */
422 boolean_t
env_admin_lock_enabled(picl_nodehdl_t fruh)423 env_admin_lock_enabled(picl_nodehdl_t fruh)
424 {
425 	char		adminlock[PICL_PROPNAMELEN_MAX];
426 
427 	if (ptree_get_propval_by_name(fruh, PICL_PROP_ADMIN_LOCK,
428 		adminlock, sizeof (adminlock))
429 		!= PICL_SUCCESS) {
430 		return (B_FALSE);
431 	}
432 	if (strcmp(adminlock, PICL_ADMINLOCK_ENABLED) == 0) {
433 		return (B_TRUE);
434 	}
435 	return (B_FALSE);
436 }
437