xref: /illumos-gate/usr/src/cmd/picl/plugins/sun4v/mdesc/mdescplugin.c (revision 2e837a72011f54762249b6612c2a64f171efcd43)
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 (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /*
23  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 /*
28  * The MDESC picl plugin serves 2 different functionalities.
29  * --The first is to look up certain CPU properties in the MDESC an to add
30  * these properties in the already created CPU PICL nodes in the /platform
31  * section of the tree.
32  * --The second functionality is to create a /disk_discovery section of the
33  * PICL tree which will have a disk node created for each disk node in the
34  * machine description.
35  */
36 
37 #include "mdescplugin.h"
38 #include <libnvpair.h>
39 
40 #pragma init(mdescplugin_register)	/* place in .init section */
41 
42 picl_nodehdl_t	root_node;
43 md_t		*mdp;
44 mde_cookie_t	rootnode;
45 
46 void mdescplugin_init(void);
47 void mdescplugin_fini(void);
48 static void signal_devtree(void);
49 
50 extern int add_cpu_prop(picl_nodehdl_t node, void *args);
51 extern int disk_discovery(void);
52 extern md_t *mdesc_devinit(void);
53 extern void mdesc_devfini(md_t *mdp);
54 extern int update_devices(char *dev, int op);
55 
56 picld_plugin_reg_t mdescplugin_reg = {
57 	PICLD_PLUGIN_VERSION_1,
58 	PICLD_PLUGIN_CRITICAL,
59 	"mdesc_plugin",
60 	mdescplugin_init,
61 	mdescplugin_fini
62 };
63 
64 #define	DISK_FOUND 0x00
65 #define	DISK_NOT_FOUND 0x01
66 
67 typedef struct disk_lookup {
68 	char *path;
69 	picl_nodehdl_t disk;
70 	int result;
71 } disk_lookup_t;
72 
73 int
74 find_disk(picl_nodehdl_t node, void *args)
75 {
76 	disk_lookup_t *lookup  = (disk_lookup_t *)args;
77 	int status;
78 	char path[PICL_PROPNAMELEN_MAX];
79 
80 	status = ptree_get_propval_by_name(node, "Path", (void *)&path,
81 	    PICL_PROPNAMELEN_MAX);
82 	if (status != PICL_SUCCESS) {
83 		return (PICL_WALK_CONTINUE);
84 	}
85 
86 	if (strcmp(path, lookup->path) == 0) {
87 		lookup->disk = node;
88 		lookup->result = DISK_FOUND;
89 		return (PICL_WALK_TERMINATE);
90 	}
91 
92 	return (PICL_WALK_CONTINUE);
93 }
94 
95 /*
96  * DR event handler
97  * respond to the picl events:
98  *      PICLEVENT_DR_AP_STATE_CHANGE
99  */
100 static void
101 dr_handler(const char *ename, const void *earg, size_t size, void *cookie)
102 {
103 	nvlist_t	*nvlp = NULL;
104 	char		*dtype;
105 	char		*ap_id;
106 	char		*hint;
107 
108 
109 	if (strcmp(ename, PICLEVENT_DR_AP_STATE_CHANGE) != 0) {
110 		return;
111 	}
112 
113 	if (nvlist_unpack((char *)earg, size, &nvlp, NULL)) {
114 		return;
115 	}
116 
117 	if (nvlist_lookup_string(nvlp, PICLEVENTARG_DATA_TYPE, &dtype)) {
118 		nvlist_free(nvlp);
119 		return;
120 	}
121 
122 	if (strcmp(dtype, PICLEVENTARG_PICLEVENT_DATA) != 0) {
123 		nvlist_free(nvlp);
124 		return;
125 	}
126 
127 	if (nvlist_lookup_string(nvlp, PICLEVENTARG_AP_ID, &ap_id)) {
128 		nvlist_free(nvlp);
129 		return;
130 	}
131 
132 	if (nvlist_lookup_string(nvlp, PICLEVENTARG_HINT, &hint)) {
133 		nvlist_free(nvlp);
134 		return;
135 	}
136 
137 	mdp = mdesc_devinit();
138 	if (mdp == NULL) {
139 		nvlist_free(nvlp);
140 		return;
141 	}
142 
143 	rootnode = md_root_node(mdp);
144 
145 	if (strcmp(hint, DR_HINT_INSERT) == 0)
146 		(void) update_devices(ap_id, DEV_ADD);
147 	else if (strcmp(hint, DR_HINT_REMOVE) == 0)
148 		(void) update_devices(ap_id, DEV_REMOVE);
149 
150 	mdesc_devfini(mdp);
151 	nvlist_free(nvlp);
152 
153 	/*
154 	 * Signal the devtree plugin to add more cpu properties.
155 	 */
156 	signal_devtree();
157 }
158 
159 /*
160  * Discovery event handler
161  * respond to the picl events:
162  *      PICLEVENT_SYSEVENT_DEVICE_ADDED
163  *      PICLEVENT_SYSEVENT_DEVICE_REMOVED
164  */
165 static void
166 dsc_handler(const char *ename, const void *earg, size_t size, void *cookie)
167 {
168 	nvlist_t	*nvlp = NULL;
169 	char		*path;
170 	disk_lookup_t	lookup;
171 	int		status;
172 
173 	/*
174 	 * retrieve the device's physical path from the event arg
175 	 * and determine which disk (if any) we are working with
176 	 */
177 	if (nvlist_unpack((char *)earg, size, &nvlp, NULL))
178 		return;
179 	if (nvlist_lookup_string(nvlp, "devfs-path", &path))
180 		return;
181 
182 	lookup.path = strdup(path);
183 	lookup.disk = NULL;
184 	lookup.result = DISK_NOT_FOUND;
185 
186 	status = ptree_walk_tree_by_class(root_node, "disk", (void *)&lookup,
187 	    find_disk);
188 	if (status != PICL_SUCCESS) {
189 		return;
190 	}
191 
192 	if (lookup.result == DISK_FOUND) {
193 		if (strcmp(ename, PICLEVENT_SYSEVENT_DEVICE_ADDED) == 0)
194 			ptree_update_propval_by_name(lookup.disk, "State",
195 			    (void *)strdup(CONFIGURED), PICL_PROPNAMELEN_MAX);
196 		else if (strcmp(ename, PICLEVENT_SYSEVENT_DEVICE_REMOVED) == 0)
197 			ptree_update_propval_by_name(lookup.disk, "State",
198 			    (void *)strdup(UNCONFIGURED), PICL_PROPNAMELEN_MAX);
199 	}
200 
201 	nvlist_free(nvlp);
202 }
203 
204 /*ARGSUSED*/
205 static void
206 mdesc_ev_completion_handler(char *ename, void *earg, size_t size)
207 {
208 	free(earg);
209 }
210 
211 static void
212 signal_devtree(void)
213 {
214 	nvlist_t *nvl;
215 	char *packed_nvl;
216 	size_t nvl_size;
217 	int status;
218 
219 	if (nvlist_alloc(&nvl, NV_UNIQUE_NAME_TYPE, NULL) != 0)
220 		return;
221 
222 	/*
223 	 * Right now (Aug. 2007) snowbird is the only other platform
224 	 * which uses this event.  Since that's a sun4u platform and
225 	 * this is sun4v we do not have to worry about possible confusion
226 	 * or interference between the two by grabbing this event for
227 	 * our own use here.  This event is consumed by the devtree
228 	 * plug-in.  The event signals the plug-in to re-run its
229 	 * cpu initialization function, which will cause it to add
230 	 * additional information to the cpu devtree nodes (particularly,
231 	 * the administrative state of the cpus.)
232 	 */
233 	if (nvlist_add_string(nvl, PICLEVENTARG_EVENT_NAME,
234 	    PICLEVENT_CPU_STATE_CHANGE) != 0) {
235 		free(nvl);
236 		return;
237 	}
238 
239 	/*
240 	 * The devtree plug-in needs to see a devfs path argument for
241 	 * any event it considers.  We supply one here which is essentially
242 	 * a dummy since it is not processed by the devtree plug-in for
243 	 * this event.
244 	 */
245 	if (nvlist_add_string(nvl, PICLEVENTARG_DEVFS_PATH, "/cpu") != 0) {
246 		free(nvl);
247 		return;
248 	}
249 	packed_nvl = NULL;
250 	if (nvlist_pack(nvl, &packed_nvl, &nvl_size, NV_ENCODE_NATIVE,
251 	    0) != 0) {
252 		free(nvl);
253 		return;
254 	}
255 	if ((status = ptree_post_event(PICLEVENT_CPU_STATE_CHANGE,
256 	    packed_nvl, nvl_size, mdesc_ev_completion_handler)) !=
257 	    PICL_SUCCESS) {
258 		free(nvl);
259 		syslog(LOG_WARNING,
260 		    "signal_devtree: can't post cpu event: %d\n", status);
261 	}
262 }
263 
264 void
265 mdescplugin_init(void)
266 {
267 	int		status;
268 
269 	status = ptree_get_root(&root_node);
270 	if (status != PICL_SUCCESS) {
271 		return;
272 	}
273 
274 	mdp = mdesc_devinit();
275 	if (mdp == NULL)
276 		return;
277 
278 	/*
279 	 * update the cpu configuration in case the snapshot cache used by the
280 	 * devtree plugin is out of date.
281 	 */
282 	(void) update_devices(OBP_CPU, DEV_ADD);
283 	(void) update_devices(OBP_CPU, DEV_REMOVE);
284 
285 	rootnode = md_root_node(mdp);
286 
287 	/*
288 	 * This is the start of the CPU property augmentation code.
289 	 * add_cpu_prop and the rest of the CPU code lives in cpu_prop_update.c
290 	 */
291 	status = ptree_walk_tree_by_class(root_node, "cpu", NULL, add_cpu_prop);
292 	if (status != PICL_SUCCESS) {
293 		return;
294 	}
295 
296 	signal_devtree();
297 
298 	(void) disk_discovery();
299 
300 	/*
301 	 * register dsc_handler for both "sysevent-device-added" and
302 	 * and for "sysevent-device-removed" PICL events
303 	 */
304 	(void) ptree_register_handler(PICLEVENT_SYSEVENT_DEVICE_ADDED,
305 	    dsc_handler, NULL);
306 	(void) ptree_register_handler(PICLEVENT_SYSEVENT_DEVICE_REMOVED,
307 	    dsc_handler, NULL);
308 	(void) ptree_register_handler(PICLEVENT_DR_AP_STATE_CHANGE,
309 	    dr_handler, NULL);
310 
311 	mdesc_devfini(mdp);
312 }
313 
314 void
315 mdescplugin_fini(void)
316 {
317 	/* unregister the event handler */
318 	(void) ptree_unregister_handler(PICLEVENT_SYSEVENT_DEVICE_ADDED,
319 	    dsc_handler, NULL);
320 	(void) ptree_unregister_handler(PICLEVENT_SYSEVENT_DEVICE_REMOVED,
321 	    dsc_handler, NULL);
322 	(void) ptree_unregister_handler(PICLEVENT_DR_AP_STATE_CHANGE,
323 	    dr_handler, NULL);
324 }
325 
326 void
327 mdescplugin_register(void)
328 {
329 	picld_plugin_register(&mdescplugin_reg);
330 }
331