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
find_disk(picl_nodehdl_t node,void * args)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
dr_handler(const char * ename,const void * earg,size_t size,void * cookie)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
dsc_handler(const char * ename,const void * earg,size_t size,void * cookie)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
mdesc_ev_completion_handler(char * ename,void * earg,size_t size)206 mdesc_ev_completion_handler(char *ename, void *earg, size_t size)
207 {
208 free(earg);
209 }
210
211 static void
signal_devtree(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
mdescplugin_init(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
mdescplugin_fini(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
mdescplugin_register(void)327 mdescplugin_register(void)
328 {
329 picld_plugin_register(&mdescplugin_reg);
330 }
331