10010e23aSHasso Tepper /*-
20010e23aSHasso Tepper * Copyright (c) 2000, 2001 Michael Smith
30010e23aSHasso Tepper * Copyright (c) 2000 BSDi
40010e23aSHasso Tepper * All rights reserved.
50010e23aSHasso Tepper *
60010e23aSHasso Tepper * Redistribution and use in source and binary forms, with or without
70010e23aSHasso Tepper * modification, are permitted provided that the following conditions
80010e23aSHasso Tepper * are met:
90010e23aSHasso Tepper * 1. Redistributions of source code must retain the above copyright
100010e23aSHasso Tepper * notice, this list of conditions and the following disclaimer.
110010e23aSHasso Tepper * 2. Redistributions in binary form must reproduce the above copyright
120010e23aSHasso Tepper * notice, this list of conditions and the following disclaimer in the
130010e23aSHasso Tepper * documentation and/or other materials provided with the distribution.
140010e23aSHasso Tepper *
150010e23aSHasso Tepper * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
160010e23aSHasso Tepper * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
170010e23aSHasso Tepper * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
180010e23aSHasso Tepper * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
190010e23aSHasso Tepper * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
200010e23aSHasso Tepper * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
210010e23aSHasso Tepper * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
220010e23aSHasso Tepper * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
230010e23aSHasso Tepper * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
240010e23aSHasso Tepper * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
250010e23aSHasso Tepper * SUCH DAMAGE.
260010e23aSHasso Tepper *
270010e23aSHasso Tepper * $FreeBSD: src/usr.sbin/devinfo/devinfo.c,v 1.7 2007/10/27 13:06:15 jhb Exp $
280010e23aSHasso Tepper */
290010e23aSHasso Tepper
300010e23aSHasso Tepper /*
310010e23aSHasso Tepper * Print information about system device configuration.
320010e23aSHasso Tepper */
330010e23aSHasso Tepper
340010e23aSHasso Tepper #include <sys/types.h>
350010e23aSHasso Tepper #include <err.h>
360010e23aSHasso Tepper #include <stdio.h>
370010e23aSHasso Tepper #include <stdlib.h>
380010e23aSHasso Tepper #include <unistd.h>
390010e23aSHasso Tepper #include "devinfo.h"
400010e23aSHasso Tepper
410010e23aSHasso Tepper int rflag;
420010e23aSHasso Tepper int vflag;
430010e23aSHasso Tepper
440010e23aSHasso Tepper static void print_resource(struct devinfo_res *);
450010e23aSHasso Tepper static int print_device_matching_resource(struct devinfo_res *, void *);
460010e23aSHasso Tepper static int print_device_rman_resources(struct devinfo_rman *, void *);
470010e23aSHasso Tepper static int print_device(struct devinfo_dev *, void *);
480010e23aSHasso Tepper static int print_rman_resource(struct devinfo_res *, void *);
490010e23aSHasso Tepper static int print_rman(struct devinfo_rman *, void *);
500010e23aSHasso Tepper
510010e23aSHasso Tepper struct indent_arg
520010e23aSHasso Tepper {
530010e23aSHasso Tepper int indent;
540010e23aSHasso Tepper void *arg;
550010e23aSHasso Tepper };
560010e23aSHasso Tepper
570010e23aSHasso Tepper /*
580010e23aSHasso Tepper * Print a resource.
590010e23aSHasso Tepper */
600010e23aSHasso Tepper void
print_resource(struct devinfo_res * res)610010e23aSHasso Tepper print_resource(struct devinfo_res *res)
620010e23aSHasso Tepper {
630010e23aSHasso Tepper struct devinfo_rman *rman;
640010e23aSHasso Tepper int hexmode;
650010e23aSHasso Tepper
660010e23aSHasso Tepper rman = devinfo_handle_to_rman(res->dr_rman);
670010e23aSHasso Tepper hexmode = (rman->dm_size > 1000) || (rman->dm_size == 0);
680010e23aSHasso Tepper printf(hexmode ? "0x%lx" : "%lu", res->dr_start);
690010e23aSHasso Tepper if (res->dr_size > 1)
700010e23aSHasso Tepper printf(hexmode ? "-0x%lx" : "-%lu",
710010e23aSHasso Tepper res->dr_start + res->dr_size - 1);
720010e23aSHasso Tepper }
730010e23aSHasso Tepper
740010e23aSHasso Tepper /*
750010e23aSHasso Tepper * Print resource information if this resource matches the
760010e23aSHasso Tepper * given device.
770010e23aSHasso Tepper *
780010e23aSHasso Tepper * If the given indent is 0, return an indicator that a matching
790010e23aSHasso Tepper * resource exists.
800010e23aSHasso Tepper */
810010e23aSHasso Tepper int
print_device_matching_resource(struct devinfo_res * res,void * arg)820010e23aSHasso Tepper print_device_matching_resource(struct devinfo_res *res, void *arg)
830010e23aSHasso Tepper {
840010e23aSHasso Tepper struct indent_arg *ia = (struct indent_arg *)arg;
850010e23aSHasso Tepper struct devinfo_dev *dev = (struct devinfo_dev *)ia->arg;
860010e23aSHasso Tepper int i;
870010e23aSHasso Tepper
880010e23aSHasso Tepper if (devinfo_handle_to_device(res->dr_device) == dev) {
890010e23aSHasso Tepper /* in 'detect' mode, found a match */
900010e23aSHasso Tepper if (ia->indent == 0)
910010e23aSHasso Tepper return(1);
920010e23aSHasso Tepper for (i = 0; i < ia->indent; i++)
930010e23aSHasso Tepper printf(" ");
940010e23aSHasso Tepper print_resource(res);
950010e23aSHasso Tepper printf("\n");
960010e23aSHasso Tepper }
970010e23aSHasso Tepper return(0);
980010e23aSHasso Tepper }
990010e23aSHasso Tepper
1000010e23aSHasso Tepper /*
1010010e23aSHasso Tepper * Print resource information for this device and resource manager.
1020010e23aSHasso Tepper */
1030010e23aSHasso Tepper int
print_device_rman_resources(struct devinfo_rman * rman,void * arg)1040010e23aSHasso Tepper print_device_rman_resources(struct devinfo_rman *rman, void *arg)
1050010e23aSHasso Tepper {
1060010e23aSHasso Tepper struct indent_arg *ia = (struct indent_arg *)arg;
1070010e23aSHasso Tepper int indent, i;
1080010e23aSHasso Tepper
1090010e23aSHasso Tepper indent = ia->indent;
1100010e23aSHasso Tepper
1110010e23aSHasso Tepper /* check whether there are any resources matching this device */
1120010e23aSHasso Tepper ia->indent = 0;
1130010e23aSHasso Tepper if (devinfo_foreach_rman_resource(rman,
1140010e23aSHasso Tepper print_device_matching_resource, ia) != 0) {
1150010e23aSHasso Tepper
1160010e23aSHasso Tepper /* there are, print header */
1170010e23aSHasso Tepper for (i = 0; i < indent; i++)
1180010e23aSHasso Tepper printf(" ");
1190010e23aSHasso Tepper printf("%s:\n", rman->dm_desc);
1200010e23aSHasso Tepper
1210010e23aSHasso Tepper /* print resources */
1220010e23aSHasso Tepper ia->indent = indent + 4;
1230010e23aSHasso Tepper devinfo_foreach_rman_resource(rman,
1240010e23aSHasso Tepper print_device_matching_resource, ia);
1250010e23aSHasso Tepper }
1260010e23aSHasso Tepper ia->indent = indent;
1270010e23aSHasso Tepper return(0);
1280010e23aSHasso Tepper }
1290010e23aSHasso Tepper
1300010e23aSHasso Tepper /*
1310010e23aSHasso Tepper * Print information about a device.
1320010e23aSHasso Tepper */
1330010e23aSHasso Tepper int
print_device(struct devinfo_dev * dev,void * arg)1340010e23aSHasso Tepper print_device(struct devinfo_dev *dev, void *arg)
1350010e23aSHasso Tepper {
1360010e23aSHasso Tepper struct indent_arg ia;
1370010e23aSHasso Tepper int i, indent;
1380010e23aSHasso Tepper
1399f05f13fSImre Vadász if (vflag > 1 ||
1409f05f13fSImre Vadász (vflag && (dev->dd_flags & DIF_ENABLED)) ||
141*feaf0601SSascha Wildner (dev->dd_name[0] != 0 && dev->dd_state >= DS_INPROGRESS)) {
1420010e23aSHasso Tepper indent = (int)(intptr_t)arg;
1430010e23aSHasso Tepper for (i = 0; i < indent; i++)
1440010e23aSHasso Tepper printf(" ");
1459f05f13fSImre Vadász if (vflag && (dev->dd_flags & DIF_ENABLED) == 0) {
1469f05f13fSImre Vadász if (dev->dd_name[0])
1479f05f13fSImre Vadász printf("%s", dev->dd_name);
1489f05f13fSImre Vadász printf("(disabled)");
1499f05f13fSImre Vadász } else {
1500010e23aSHasso Tepper printf("%s", dev->dd_name[0] ? dev->dd_name : "unknown");
1519f05f13fSImre Vadász }
1520010e23aSHasso Tepper if (vflag && *dev->dd_pnpinfo)
1530010e23aSHasso Tepper printf(" pnpinfo %s", dev->dd_pnpinfo);
1540010e23aSHasso Tepper if (vflag && *dev->dd_location)
1550010e23aSHasso Tepper printf(" at %s", dev->dd_location);
1560010e23aSHasso Tepper printf("\n");
1570010e23aSHasso Tepper if (rflag) {
1580010e23aSHasso Tepper ia.indent = indent + 4;
1590010e23aSHasso Tepper ia.arg = dev;
1600010e23aSHasso Tepper devinfo_foreach_rman(print_device_rman_resources,
1610010e23aSHasso Tepper (void *)&ia);
1620010e23aSHasso Tepper }
1630010e23aSHasso Tepper }
1640010e23aSHasso Tepper
1650010e23aSHasso Tepper return(devinfo_foreach_device_child(dev, print_device,
1660010e23aSHasso Tepper (void *)((char *)arg + 2)));
1670010e23aSHasso Tepper }
1680010e23aSHasso Tepper
1690010e23aSHasso Tepper /*
1700010e23aSHasso Tepper * Print information about a resource under a resource manager.
1710010e23aSHasso Tepper */
1720010e23aSHasso Tepper int
print_rman_resource(struct devinfo_res * res,void * arg __unused)1730010e23aSHasso Tepper print_rman_resource(struct devinfo_res *res, void *arg __unused)
1740010e23aSHasso Tepper {
1750010e23aSHasso Tepper struct devinfo_dev *dev;
1760010e23aSHasso Tepper
1770010e23aSHasso Tepper printf(" ");
1780010e23aSHasso Tepper print_resource(res);
1790010e23aSHasso Tepper dev = devinfo_handle_to_device(res->dr_device);
1800010e23aSHasso Tepper if ((dev != NULL) && (dev->dd_name[0] != 0)) {
1810010e23aSHasso Tepper printf(" (%s)", dev->dd_name);
1820010e23aSHasso Tepper } else {
1830010e23aSHasso Tepper printf(" ----");
1840010e23aSHasso Tepper }
1850010e23aSHasso Tepper printf("\n");
1860010e23aSHasso Tepper return(0);
1870010e23aSHasso Tepper }
1880010e23aSHasso Tepper
1890010e23aSHasso Tepper /*
1900010e23aSHasso Tepper * Print information about a resource manager.
1910010e23aSHasso Tepper */
1920010e23aSHasso Tepper int
print_rman(struct devinfo_rman * rman,void * arg __unused)1930010e23aSHasso Tepper print_rman(struct devinfo_rman *rman, void *arg __unused)
1940010e23aSHasso Tepper {
1950010e23aSHasso Tepper printf("%s:\n", rman->dm_desc);
1960010e23aSHasso Tepper devinfo_foreach_rman_resource(rman, print_rman_resource, 0);
1970010e23aSHasso Tepper return(0);
1980010e23aSHasso Tepper }
1990010e23aSHasso Tepper
2000010e23aSHasso Tepper int
main(int argc,char * argv[])2010010e23aSHasso Tepper main(int argc, char *argv[])
2020010e23aSHasso Tepper {
2030010e23aSHasso Tepper struct devinfo_dev *root;
2040010e23aSHasso Tepper int c, uflag;
2050010e23aSHasso Tepper
2060010e23aSHasso Tepper uflag = 0;
2070010e23aSHasso Tepper while ((c = getopt(argc, argv, "ruv")) != -1) {
2080010e23aSHasso Tepper switch(c) {
2090010e23aSHasso Tepper case 'r':
2100010e23aSHasso Tepper rflag++;
2110010e23aSHasso Tepper break;
2120010e23aSHasso Tepper case 'u':
2130010e23aSHasso Tepper uflag++;
2140010e23aSHasso Tepper break;
2150010e23aSHasso Tepper case 'v':
2160010e23aSHasso Tepper vflag++;
2170010e23aSHasso Tepper break;
2180010e23aSHasso Tepper default:
2190010e23aSHasso Tepper fprintf(stderr, "%s\n%s\n",
2200010e23aSHasso Tepper "usage: devinfo [-rv]",
2210010e23aSHasso Tepper " devinfo -u");
2220010e23aSHasso Tepper exit(1);
2230010e23aSHasso Tepper }
2240010e23aSHasso Tepper }
2250010e23aSHasso Tepper
2260010e23aSHasso Tepper if (devinfo_init())
2270010e23aSHasso Tepper err(1, "devinfo_init");
2280010e23aSHasso Tepper
2290010e23aSHasso Tepper if ((root = devinfo_handle_to_device(DEVINFO_ROOT_DEVICE)) == NULL)
2300010e23aSHasso Tepper errx(1, "can't find root device");
2310010e23aSHasso Tepper
2320010e23aSHasso Tepper /* print resource usage? */
2330010e23aSHasso Tepper if (uflag) {
2340010e23aSHasso Tepper devinfo_foreach_rman(print_rman, NULL);
2350010e23aSHasso Tepper } else {
2360010e23aSHasso Tepper /* print device hierarchy */
23760233e58SSascha Wildner devinfo_foreach_device_child(root, print_device, NULL);
2380010e23aSHasso Tepper }
2390010e23aSHasso Tepper return(0);
2400010e23aSHasso Tepper }
241