dmake: do not set MAKEFLAGS=k
[unleashed/tickless.git] / usr / src / cmd / picl / plugins / sun4u / snowbird / frutree / piclkstat.c
blob79ac86ecf16346e34de18f21d4148e83dec6ab4b
1 /*
2 * CDDL HEADER START
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.
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.
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]
20 * CDDL HEADER END
23 * Copyright 2004 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
28 * Implementation to get PORT nodes state and condition information
30 #include <stdio.h>
31 #include <stdlib.h>
32 #include <ctype.h>
33 #include <strings.h>
34 #include <fcntl.h>
35 #include <unistd.h>
36 #include <stropts.h>
37 #include <locale.h>
38 #include <syslog.h>
39 #include <sys/types.h>
40 #include <sys/termios.h>
41 #include <sys/stat.h>
42 #include <fcntl.h>
43 #include <kstat.h>
44 #include <signal.h>
45 #include <assert.h>
46 #include <config_admin.h>
48 #include <picl.h>
49 #include "piclfrutree.h"
51 #define LINK_UP "link_up"
52 #define DUPLEX "duplex"
53 #define IF_SPEED "ifspeed"
54 #define IERRORS "ierrors"
55 #define IPACKETS "ipackets"
56 #define OERRORS "oerrors"
57 #define OPACKETS "opackets"
58 #define NOCANPUT "nocanput"
59 #define RUNT_ERRORS "runt_errors"
60 #define COLLISIONS "collisions"
62 typedef int (*funcp)(kstat_ctl_t *, char *, int);
64 static kstat_named_t *kstat_name_lookup(kstat_ctl_t *, char *, int, char *);
65 static int kstat_network_port_state(kstat_ctl_t *kc, char *, int);
66 static int kstat_network_port_cond(kstat_ctl_t *kc, char *, int);
67 static int serial_port_state(kstat_ctl_t *, char *, int);
68 static int serial_port_cond(kstat_ctl_t *kc, char *, int);
69 static int parallel_port_state(kstat_ctl_t *, char *, int);
70 static int parallel_port_cond(kstat_ctl_t *kc, char *, int);
71 static void sig_alarm_handler(int);
73 static funcp port_state[] = {
74 kstat_network_port_state,
75 serial_port_state,
76 parallel_port_state
79 static funcp port_cond[] = {
80 kstat_network_port_cond,
81 serial_port_cond,
82 parallel_port_cond
86 * kstat_port_state: returns ethernet, or serial, or parallel port status
87 * 1 = up, 0 = down, anything else = unknown
89 int
90 kstat_port_state(frutree_port_type_t port_type, char *driver_name,
91 int driver_instance)
93 int rc = -1;
94 kstat_ctl_t *kc = NULL;
96 switch (port_type) {
97 case NETWORK_PORT:
98 case SERIAL_PORT:
99 case PARALLEL_PORT:
100 if ((kc = kstat_open()) == NULL) {
101 return (-1);
103 rc = port_state[port_type](kc, driver_name, driver_instance);
104 kstat_close(kc);
105 return (rc);
106 default:
107 return (-1);
112 * kstat_port_cond: returns ethernet, or serial, or parallel port condition
115 kstat_port_cond(frutree_port_type_t port_type, char *driver_name,
116 int driver_instance)
118 int rc = -1;
119 kstat_ctl_t *kc = NULL;
120 switch (port_type) {
121 case NETWORK_PORT:
122 case SERIAL_PORT:
123 case PARALLEL_PORT:
124 if ((kc = kstat_open()) == NULL) {
125 return (-1);
127 rc = port_cond[port_type](kc, driver_name, driver_instance);
128 kstat_close(kc);
129 return (rc);
130 default:
131 return (-1);
135 static kstat_named_t *
136 kstat_name_lookup(kstat_ctl_t *kc, char *ks_module, int ks_instance, char *name)
138 kstat_t *ksp;
140 assert(kc);
141 assert(ks_module);
142 assert(name);
144 for (ksp = kc->kc_chain; ksp; ksp = ksp->ks_next) {
145 if (strcmp(ksp->ks_module, ks_module) == 0 &&
146 ksp->ks_instance == ks_instance &&
147 ksp->ks_type == KSTAT_TYPE_NAMED &&
148 kstat_read(kc, ksp, NULL) != -1 &&
149 kstat_data_lookup(ksp, name)) {
151 ksp = kstat_lookup(kc, ks_module, ks_instance,
152 ksp->ks_name);
153 if (!ksp)
154 return (NULL);
155 if (kstat_read(kc, ksp, NULL) == -1)
156 return (NULL);
157 return ((kstat_named_t *)kstat_data_lookup(ksp, name));
160 return (NULL);
164 * kstat_network_port_state: returns kstat info of a network port
165 * 1 = up, 0 = down, anything else = unknown
167 static int
168 kstat_network_port_state(kstat_ctl_t *kc, char *ks_module, int ks_instance)
170 kstat_named_t *port_datap = NULL;
172 if ((port_datap = kstat_name_lookup(kc, ks_module, ks_instance,
173 LINK_UP)) == NULL) {
174 return (-1);
176 if (port_datap == NULL) {
177 return (-1);
179 if (port_datap->data_type == KSTAT_DATA_UINT32) {
180 if (port_datap->value.ui32 == 1) {
181 return (1);
182 } else if (port_datap->value.ui32 == 0) {
183 return (0);
184 } else {
185 return (-1);
187 } else {
188 if (port_datap->value.ui64 == 1) {
189 return (1);
190 } else if (port_datap->value.ui64 == 0) {
191 return (0);
192 } else {
193 return (-1);
199 * kstat_network_port_cond: returns kstat info of a network port
200 * 0 = OK, 1 = FAILING, 2 = FAILED, 3 = TESTING, -1 = unknown
202 static int
203 kstat_network_port_cond(kstat_ctl_t *kc, char *ks_module, int ks_instance)
205 kstat_named_t *port_datap = NULL;
206 uint64_t collisions, runt, link_up, link_duplex;
207 uint64_t ifspeed, ierrors, ipackets, oerrors, opackets;
209 if ((port_datap = kstat_name_lookup(kc, ks_module, ks_instance,
210 LINK_UP)) == NULL) {
211 return (-1);
214 if (port_datap->data_type == KSTAT_DATA_UINT32) {
215 link_up = port_datap->value.ui32;
216 } else {
217 link_up = port_datap->value.ui64;
219 if (link_up == 0) {
220 return (2);
223 if ((port_datap = kstat_name_lookup(kc, ks_module, ks_instance,
224 DUPLEX)) == NULL) {
225 return (-1);
228 if (port_datap->data_type == KSTAT_DATA_UINT32) {
229 link_duplex = port_datap->value.ui32;
230 } else {
231 link_duplex = port_datap->value.ui64;
233 if (link_duplex == 0) {
234 return (2);
237 if ((port_datap = kstat_name_lookup(kc, ks_module, ks_instance,
238 IF_SPEED)) == NULL) {
239 return (-1);
241 if (port_datap->data_type == KSTAT_DATA_UINT32) {
242 ifspeed = port_datap->value.ui32;
243 } else {
244 ifspeed = port_datap->value.ui64;
246 if (ifspeed == 0) {
247 return (2);
250 /* check for FAILING conditions */
251 if ((port_datap = kstat_name_lookup(kc, ks_module, ks_instance,
252 IERRORS)) == NULL) {
253 return (-1);
255 if (port_datap->data_type == KSTAT_DATA_UINT32) {
256 ierrors = port_datap->value.ui32;
257 } else {
258 ierrors = port_datap->value.ui64;
261 if ((port_datap = kstat_name_lookup(kc, ks_module, ks_instance,
262 IPACKETS)) == NULL) {
263 return (-1);
266 if (port_datap->data_type == KSTAT_DATA_UINT32) {
267 ipackets = port_datap->value.ui32;
268 } else {
269 ipackets = port_datap->value.ui64;
271 if (ierrors > ipackets/10) {
272 return (1);
275 if ((port_datap = kstat_name_lookup(kc, ks_module, ks_instance,
276 OERRORS)) == NULL) {
277 return (-1);
279 if (port_datap->data_type == KSTAT_DATA_UINT32) {
280 oerrors = port_datap->value.ui32;
281 } else {
282 oerrors = port_datap->value.ui64;
285 if ((port_datap = kstat_name_lookup(kc, ks_module, ks_instance,
286 OPACKETS)) == NULL) {
287 return (-1);
289 if (port_datap->data_type == KSTAT_DATA_UINT32) {
290 opackets = port_datap->value.ui32;
291 } else {
292 opackets = port_datap->value.ui64;
294 if (oerrors > opackets/10) {
295 return (1);
298 if ((port_datap = kstat_name_lookup(kc, ks_module, ks_instance,
299 RUNT_ERRORS)) == NULL) {
300 return (-1);
302 if (port_datap->data_type == KSTAT_DATA_UINT32) {
303 runt = port_datap->value.ui32;
304 } else {
305 runt = port_datap->value.ui64;
307 if (runt > ipackets/10) {
308 return (1);
311 if ((port_datap = kstat_name_lookup(kc, ks_module, ks_instance,
312 COLLISIONS)) == NULL) {
313 return (-1);
315 if (port_datap->data_type == KSTAT_DATA_UINT32) {
316 collisions = port_datap->value.ui32;
317 } else {
318 collisions = port_datap->value.ui64;
320 if (collisions > (opackets+ipackets)/30) {
321 return (1);
323 return (0);
327 * serial_port_state: returns status a serial port
328 * 1 = up, 0 = down, anything else = unknown
331 /* ARGSUSED */
332 static int
333 serial_port_state(kstat_ctl_t *kc, char *driver, int instance)
335 int fd;
336 char device[20];
337 struct termios flags;
338 struct sigaction old_sa, new_sa;
340 (void) memset(&old_sa, 0, sizeof (old_sa));
341 (void) memset(&new_sa, 0, sizeof (new_sa));
342 new_sa.sa_handler = sig_alarm_handler;
343 (void) sigaction(SIGALRM, &new_sa, &old_sa);
344 (void) alarm(1);
346 (void) snprintf(device, sizeof (device), "/dev/tty%c", instance+'a');
347 fd = open(device, O_RDONLY|O_NDELAY|O_NONBLOCK|O_NOCTTY);
349 /* Restore sig action flags */
350 (void) sigaction(SIGALRM, &old_sa, NULL);
351 /* Disable alarm */
352 (void) alarm(0);
354 if (fd == -1) {
355 return (-1);
358 if (isatty(fd) == 0) {
359 (void) close(fd);
360 return (-1);
362 (void) memset(&flags, 0, sizeof (flags));
363 if (ioctl(fd, TCGETS, &flags) != 0) {
364 (void) close(fd);
365 return (-1);
367 (void) close(fd);
368 return ((flags.c_cflag & TIOCM_LE) ? 1 : 0);
371 /* ARGSUSED */
372 static void
373 sig_alarm_handler(int signo)
378 * serial_port_cond: returns status of a serial port
379 * 0 = OK, 1 = FAILING, 2 = FAILED, 3 = TESTING, anything else = UNKNOWN
381 static int
382 serial_port_cond(kstat_ctl_t *kc, char *driver, int instance)
384 switch (serial_port_state(kc, driver, instance)) {
385 case 1:
386 return (0);
387 default:
388 return (-1);
393 * parallel_port_state: returns kstat info of a serial port
394 * 1 = up, 0 = down, anything else = unknown
396 static int
397 parallel_port_state(kstat_ctl_t *kc, char *ks_module, int ks_instance)
399 kstat_t *ksp = NULL;
400 kstat_named_t *port_datap = NULL;
401 char *data_lookup;
402 char ks_name[20];
404 (void) snprintf(ks_name, sizeof (ks_name), "%s%d", ks_module,
405 ks_instance);
406 if ((ksp = kstat_lookup(kc, ks_module, ks_instance, ks_name)) == NULL) {
407 return (-1);
409 if (kstat_read(kc, ksp, NULL) == -1) {
410 return (-1);
412 data_lookup = "";
413 port_datap = (kstat_named_t *)kstat_data_lookup(ksp, data_lookup);
414 if (port_datap == NULL) {
415 return (-1);
417 return (-1);
421 * parallel_port_cond: returns kstat info of a serial port
422 * 1 = up, 0 = down, anything else = unknown
424 static int
425 parallel_port_cond(kstat_ctl_t *kc, char *ks_module, int ks_instance)
427 return (parallel_port_state(kc, ks_module, ks_instance));