blob: 9653a03ad5c6212e474f8f6e83cb9f50501dbeb1 [file] [log] [blame] [edit]
/*
* systool.c
*
* Sysfs utility to list buses, classes, and devices
*
* Copyright (C) IBM Corp. 2003-2005
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation version 2 of the License.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 675 Mass Ave, Cambridge, MA 02139, USA.
*
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <dirent.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <ctype.h>
#include "libsysfs.h"
#include "names.h"
#define safestrcpy(to, from) strncpy(to, from, sizeof(to)-1)
#define safestrcat(to, from) strncat(to, from, sizeof(to) - strlen(to)-1)
#define safestrcpymax(to, from, max) \
do { \
to[max-1] = '\0'; \
strncpy(to, from, max-1); \
} while (0)
#define safestrcatmax(to, from, max) \
do { \
to[max-1] = '\0'; \
strncat(to, from, max - strlen(to)-1); \
} while (0)
/* Command Options */
static int show_options = 0; /* bitmask of show options */
static char *attribute_to_show = NULL; /* show value for this attribute */
static char *device_to_show = NULL; /* show only this bus device */
static char sysfs_mnt_path[SYSFS_PATH_MAX]; /* sysfs mount point */
struct pci_access *pacc = NULL;
char *show_bus = NULL;
static void show_device(struct sysfs_device *device, int level);
static void show_class_device(struct sysfs_class_device *dev, int level);
#define SHOW_ATTRIBUTES 0x01 /* show attributes command option */
#define SHOW_ATTRIBUTE_VALUE 0x02 /* show an attribute value option */
#define SHOW_DEVICES 0x04 /* show only devices option */
#define SHOW_DRIVERS 0x08 /* show only drivers option */
#define SHOW_ALL_ATTRIB_VALUES 0x10 /* show all attributes with values */
#define SHOW_CHILDREN 0x20 /* show device children */
#define SHOW_PARENT 0x40 /* show device parent */
#define SHOW_PATH 0x80 /* show device/driver path */
#define SHOW_ALL 0xff
static char cmd_options[] = "aA:b:c:dDhm:pPv";
/*
* binary_files - defines existing sysfs binary files. These files will be
* printed in hex.
*/
static char *binary_files[] = {
"config",
"data"
};
static int binfiles = 2;
static unsigned int get_pciconfig_word(int offset, unsigned char *buf)
{
unsigned short val = (unsigned char)buf[offset] |
((unsigned char)buf[offset+1] << 8);
return val;
}
/**
* usage: prints utility usage.
*/
static void usage(void)
{
fprintf(stdout, "Usage: systool [<options> [device]]\n");
fprintf(stdout, "\t-a\t\t\tShow attributes\n");
fprintf(stdout, "\t-b <bus_name>\t\tShow a specific bus\n");
fprintf(stdout, "\t-c <class_name>\t\tShow a specific class\n");
fprintf(stdout, "\t-d\t\t\tShow only devices\n");
fprintf(stdout, "\t-h\t\t\tShow usage\n");
fprintf(stdout, "\t-m <module_name>\tShow a specific module\n");
fprintf(stdout, "\t-p\t\t\tShow path to device/driver\n");
fprintf(stdout, "\t-v\t\t\tShow all attributes with values\n");
fprintf(stdout, "\t-A <attribute_name>\tShow attribute value\n");
fprintf(stdout, "\t-D\t\t\tShow only drivers\n");
fprintf(stdout, "\t-P\t\t\tShow device's parent\n");
}
/**
* indent: called before printing a line, it adds indent to the line up to
* level passed in.
* @level: number of spaces to indent.
*/
static void indent(int level)
{
int i;
for (i = 0; i < level; i++)
fprintf(stdout, " ");
}
/**
* remove_end_newline: removes newline on the end of an attribute value
* @value: string to remove newline from
*/
static void remove_end_newline(char *value)
{
char *p = value + (strlen(value) - 1);
if (p && *p == '\n')
*p = '\0';
}
/**
* isbinaryvalue: checks to see if attribute is binary or not.
* @attr: attribute to check.
* returns 1 if binary, 0 if not.
*/
static int isbinaryvalue(struct sysfs_attribute *attr)
{
int i;
if (!attr || !attr->value)
return 0;
for (i = 0; i < binfiles; i++)
if ((strcmp(attr->name, binary_files[i])) == 0)
return 1;
return 0;
}
/**
* show_attribute_value: prints out single attribute value.
* @attr: attricute to print.
*/
static void show_attribute_value(struct sysfs_attribute *attr, int level)
{
if (!attr)
return;
if (attr->method & SYSFS_METHOD_SHOW) {
if (isbinaryvalue(attr)) {
int i;
for (i = 0; i < attr->len; i++) {
if (!(i % 16) && (i != 0)) {
fprintf(stdout, "\n");
indent(level+22);
} else if (!(i % 8) && (i != 0))
fprintf(stdout, " ");
fprintf(stdout, " %02x",
(unsigned char)attr->value[i]);
}
fprintf(stdout, "\n");
} else if (attr->value && strlen(attr->value) > 0) {
remove_end_newline(attr->value);
fprintf(stdout, "\"%s\"\n", attr->value);
} else
fprintf(stdout, "\n");
} else {
fprintf(stdout, "<store method only>\n");
}
}
/**
* show_attribute: prints out a single attribute
* @attr: attribute to print.
*/
static void show_attribute(struct sysfs_attribute *attr, int level)
{
if (!attr)
return;
if (show_options & SHOW_ALL_ATTRIB_VALUES) {
indent(level);
fprintf(stdout, "%-20s= ", attr->name);
show_attribute_value(attr, level);
} else if ((show_options & SHOW_ATTRIBUTES) || ((show_options
& SHOW_ATTRIBUTE_VALUE) && (strcmp(attr->name, attribute_to_show)
== 0))) {
indent(level);
fprintf (stdout, "%-20s", attr->name);
if (show_options & SHOW_ATTRIBUTE_VALUE && attr->value
!= NULL && (strcmp(attr->name, attribute_to_show)) == 0) {
fprintf(stdout, "= ");
show_attribute_value(attr, level);
} else
fprintf(stdout, "\n");
}
}
/**
* show_attributes: prints out a list of attributes.
* @attributes: print this dlist of attributes/files.
*/
static void show_attributes(struct dlist *attributes, int level)
{
if (attributes) {
struct sysfs_attribute *cur;
dlist_for_each_data(attributes, cur,
struct sysfs_attribute) {
show_attribute(cur, level);
}
}
}
/**
* show_device_parent: prints device's parent (if present)
* @device: sysfs_device whose parent information is needed
*/
static void show_device_parent(struct sysfs_device *device, int level)
{
struct sysfs_device *parent;
parent = sysfs_get_device_parent(device);
if (parent) {
fprintf(stdout, "\n");
indent(level);
fprintf(stdout, "Device \"%s\"'s parent\n", device->name);
show_device(parent, (level+2));
}
}
/**
* show_device: prints out device information.
* @device: device to print.
*/
static void show_device(struct sysfs_device *device, int level)
{
struct dlist *attributes;
unsigned int vendor_id, device_id;
char buf[128], value[256], path[SYSFS_PATH_MAX];
if (device) {
indent(level);
if (show_bus && (!(strcmp(show_bus, "pci")))) {
fprintf(stdout, "%s ", device->bus_id);
memset(path, 0, SYSFS_PATH_MAX);
memset(value, 0, SYSFS_PATH_MAX);
safestrcpy(path, device->path);
safestrcat(path, "/config");
struct sysfs_attribute *attr;
attr = sysfs_open_attribute(path);
if (attr) {
if (!sysfs_read_attribute(attr)) {
vendor_id = get_pciconfig_word
(PCI_VENDOR_ID, (unsigned char *)attr->value);
device_id = get_pciconfig_word
(PCI_DEVICE_ID, (unsigned char *)attr->value);
fprintf(stdout, "%s\n",
pci_lookup_name(pacc, (unsigned char *)buf, 128,
PCI_LOOKUP_VENDOR |
PCI_LOOKUP_DEVICE,
vendor_id, device_id, 0, 0));
}
sysfs_close_attribute(attr);
} else
fprintf(stdout, "\n");
} else
fprintf(stdout, "Device = \"%s\"\n", device->bus_id);
if (show_options & (SHOW_PATH | SHOW_ALL_ATTRIB_VALUES)) {
indent(level);
fprintf(stdout, "Device path = \"%s\"\n",
device->path);
}
if (show_options & (SHOW_ATTRIBUTES | SHOW_ATTRIBUTE_VALUE |
SHOW_ALL_ATTRIB_VALUES)) {
attributes = sysfs_get_device_attributes(device);
if (attributes)
show_attributes(attributes, (level+2));
}
if ((device_to_show) && (show_options & SHOW_PARENT)) {
show_options &= ~SHOW_PARENT;
show_device_parent(device, (level+2));
}
if (show_options ^ SHOW_DEVICES)
if (!(show_options & SHOW_DRIVERS))
fprintf(stdout, "\n");
}
}
/**
* show_driver_attributes: prints out driver attributes .
* @driver: print this driver's attributes.
*/
static void show_driver_attributes(struct sysfs_driver *driver, int level)
{
if (driver) {
struct dlist *attributes;
attributes = sysfs_get_driver_attributes(driver);
if (attributes) {
struct sysfs_attribute *cur;
dlist_for_each_data(attributes, cur,
struct sysfs_attribute) {
show_attribute(cur, (level));
}
fprintf(stdout, "\n");
}
}
}
/**
* show_driver: prints out driver information.
* @driver: driver to print.
*/
static void show_driver(struct sysfs_driver *driver, int level)
{
struct dlist *devlist;
if (driver) {
indent(level);
fprintf(stdout, "Driver = \"%s\"\n", driver->name);
if (show_options & (SHOW_PATH | SHOW_ALL_ATTRIB_VALUES)) {
indent(level);
fprintf(stdout, "Driver path = \"%s\"\n",
driver->path);
}
if (show_options & (SHOW_ATTRIBUTES | SHOW_ATTRIBUTE_VALUE
| SHOW_ALL_ATTRIB_VALUES))
show_driver_attributes(driver, (level+2));
devlist = sysfs_get_driver_devices(driver);
if (devlist) {
struct sysfs_device *cur;
indent(level+2);
fprintf(stdout, "Devices using \"%s\" are:\n",
driver->name);
dlist_for_each_data(devlist, cur,
struct sysfs_device) {
if (show_options & SHOW_DRIVERS) {
show_device(cur, (level+4));
fprintf(stdout, "\n");
} else {
indent(level+4);
fprintf(stdout, "\"%s\"\n", cur->name);
}
}
}
fprintf(stdout, "\n");
}
}
/**
* show_sysfs_bus: prints out everything on a bus.
* @busname: bus to print.
* returns 0 with success or 1 with error.
*/
static int show_sysfs_bus(char *busname)
{
struct sysfs_bus *bus;
struct sysfs_device *curdev;
struct sysfs_driver *curdrv;
struct dlist *devlist;
struct dlist *drvlist;
if (!busname) {
errno = EINVAL;
return 1;
}
bus = sysfs_open_bus(busname);
if (bus == NULL) {
fprintf(stderr, "Error opening bus %s\n", busname);
return 1;
}
fprintf(stdout, "Bus = \"%s\"\n", busname);
if (show_options ^ (SHOW_DEVICES | SHOW_DRIVERS))
fprintf(stdout, "\n");
if (show_options & SHOW_DEVICES) {
devlist = sysfs_get_bus_devices(bus);
if (devlist) {
dlist_for_each_data(devlist, curdev,
struct sysfs_device) {
if (!device_to_show || (strcmp(device_to_show,
curdev->bus_id) == 0))
show_device(curdev, 2);
}
}
}
if (show_options & SHOW_DRIVERS) {
drvlist = sysfs_get_bus_drivers(bus);
if (drvlist) {
dlist_for_each_data(drvlist, curdrv,
struct sysfs_driver) {
show_driver(curdrv, 2);
}
}
}
sysfs_close_bus(bus);
return 0;
}
/**
* show_classdev_parent: prints the class device's parent if present
* @dev: class device whose parent is needed
*/
static void show_classdev_parent(struct sysfs_class_device *dev, int level)
{
struct sysfs_class_device *parent;
parent = sysfs_get_classdev_parent(dev);
if (parent) {
fprintf(stdout, "\n");
indent(level);
fprintf(stdout, "Class device \"%s\"'s parent is\n",
dev->name);
show_class_device(parent, level+2);
}
}
/**
* show_class_device: prints out class device.
* @dev: class device to print.
*/
static void show_class_device(struct sysfs_class_device *dev, int level)
{
struct dlist *attributes;
struct sysfs_device *device;
if (dev) {
indent(level);
fprintf(stdout, "Class Device = \"%s\"\n", dev->name);
if (show_options & (SHOW_PATH | SHOW_ALL_ATTRIB_VALUES)) {
indent(level);
fprintf(stdout, "Class Device path = \"%s\"\n",
dev->path);
}
if (show_options & (SHOW_ATTRIBUTES | SHOW_ATTRIBUTE_VALUE
| SHOW_ALL_ATTRIB_VALUES)) {
attributes = sysfs_get_classdev_attributes(dev);
if (attributes)
show_attributes(attributes, (level+2));
fprintf(stdout, "\n");
}
if (show_options & (SHOW_DEVICES | SHOW_ALL_ATTRIB_VALUES)) {
device = sysfs_get_classdev_device(dev);
if (device) {
show_device(device, (level+2));
}
}
if ((device_to_show) && (show_options & SHOW_PARENT)) {
show_options &= ~SHOW_PARENT;
show_classdev_parent(dev, level+2);
}
if (show_options & ~(SHOW_ATTRIBUTES | SHOW_ATTRIBUTE_VALUE
| SHOW_ALL_ATTRIB_VALUES))
fprintf(stdout, "\n");
}
}
/**
* show_sysfs_class: prints out sysfs class and all its devices.
* @classname: class to print.
* returns 0 with success and 1 with error.
*/
static int show_sysfs_class(char *classname)
{
struct sysfs_class *cls;
struct sysfs_class_device *cur;
struct dlist *clsdevlist;
if (!classname) {
errno = EINVAL;
return 1;
}
cls = sysfs_open_class(classname);
if (cls == NULL) {
fprintf(stderr, "Error opening class %s\n", classname);
return 1;
}
fprintf(stdout, "Class = \"%s\"\n\n", classname);
clsdevlist = sysfs_get_class_devices(cls);
if (clsdevlist) {
dlist_for_each_data(clsdevlist, cur,
struct sysfs_class_device) {
if (device_to_show == NULL || (strcmp(device_to_show,
cur->name) == 0))
show_class_device(cur, 2);
}
}
sysfs_close_class(cls);
return 0;
}
static int show_sysfs_module(char *module)
{
struct sysfs_module *mod = NULL;
if (!module) {
errno = EINVAL;
return 1;
}
mod = sysfs_open_module(module);
if (mod == NULL) {
fprintf(stderr, "Error opening module %s\n", module);
return 1;
}
fprintf(stdout, "Module = \"%s\"\n\n", module);
if (show_options & (SHOW_ATTRIBUTES | SHOW_ATTRIBUTE_VALUE
| SHOW_ALL_ATTRIB_VALUES)) {
struct dlist *attributes = NULL;
struct sysfs_attribute *cur;
attributes = sysfs_get_module_attributes(mod);
if (attributes) {
if (show_options & (SHOW_ATTRIBUTES
| SHOW_ALL_ATTRIB_VALUES)) {
indent(2);
fprintf(stdout, "Attributes:\n");
}
dlist_for_each_data(attributes, cur,
struct sysfs_attribute) {
show_attribute(cur, (4));
}
}
attributes = sysfs_get_module_parms(mod);
if (attributes) {
if (show_options & (SHOW_ATTRIBUTES
| SHOW_ALL_ATTRIB_VALUES)) {
fprintf(stdout, "\n");
indent(2);
fprintf(stdout, "Parameters:\n");
}
dlist_for_each_data(attributes, cur,
struct sysfs_attribute) {
show_attribute(cur, (4));
}
}
attributes = sysfs_get_module_sections(mod);
if (attributes) {
if (show_options & (SHOW_ATTRIBUTES
| SHOW_ALL_ATTRIB_VALUES)) {
fprintf(stdout, "\n");
indent(2);
fprintf(stdout, "Sections:\n");
}
dlist_for_each_data(attributes, cur,
struct sysfs_attribute) {
show_attribute(cur, (4));
}
fprintf(stdout, "\n");
}
}
sysfs_close_module(mod);
return 0;
}
/**
* show_default_info: prints current buses, classes, and root devices
* supported by sysfs.
* returns 0 with success or 1 with error.
*/
static int show_default_info(void)
{
char subsys[SYSFS_NAME_LEN];
struct dlist *list;
char *cur;
int retval = 0;
safestrcpy(subsys, sysfs_mnt_path);
safestrcat(subsys, "/");
safestrcat(subsys, SYSFS_BUS_NAME);
list = sysfs_open_directory_list(subsys);
if (list) {
fprintf(stdout, "Supported sysfs buses:\n");
dlist_for_each_data(list, cur, char)
fprintf(stdout, "\t%s\n", cur);
sysfs_close_list(list);
}
safestrcpy(subsys, sysfs_mnt_path);
safestrcat(subsys, "/");
safestrcat(subsys, SYSFS_CLASS_NAME);
list = sysfs_open_directory_list(subsys);
if (list) {
fprintf(stdout, "Supported sysfs classes:\n");
dlist_for_each_data(list, cur, char)
fprintf(stdout, "\t%s\n", cur);
sysfs_close_list(list);
}
safestrcpy(subsys, sysfs_mnt_path);
safestrcat(subsys, "/");
safestrcat(subsys, SYSFS_DEVICES_NAME);
list = sysfs_open_directory_list(subsys);
if (list) {
fprintf(stdout, "Supported sysfs devices:\n");
dlist_for_each_data(list, cur, char)
fprintf(stdout, "\t%s\n", cur);
sysfs_close_list(list);
}
safestrcpy(subsys, sysfs_mnt_path);
safestrcat(subsys, "/");
safestrcat(subsys, SYSFS_MODULE_NAME);
list = sysfs_open_directory_list(subsys);
if (list) {
fprintf(stdout, "Supported sysfs modules:\n");
dlist_for_each_data(list, cur, char)
fprintf(stdout, "\t%s\n", cur);
sysfs_close_list(list);
}
return retval;
}
/**
* check_sysfs_mounted: Checks to see if sysfs is mounted.
* returns 0 if not and 1 if true.
*/
static int check_sysfs_is_mounted(void)
{
if (sysfs_get_mnt_path(sysfs_mnt_path, SYSFS_PATH_MAX) != 0)
return 0;
return 1;
}
/* MAIN */
int main(int argc, char *argv[])
{
char *show_class = NULL;
char *show_module = NULL;
char *show_root = NULL;
int retval = 0;
int opt;
char *pci_id_file = "/usr/local/share/pci.ids";
while((opt = getopt(argc, argv, cmd_options)) != EOF) {
switch(opt) {
case 'a':
show_options |= SHOW_ATTRIBUTES;
break;
case 'A':
if ((strlen(optarg) + 1) > SYSFS_NAME_LEN) {
fprintf(stderr,
"Attribute name %s is too long\n",
optarg);
exit(1);
}
attribute_to_show = optarg;
show_options |= SHOW_ATTRIBUTE_VALUE;
break;
case 'b':
show_bus = optarg;
break;
case 'c':
show_class = optarg;
break;
case 'd':
show_options |= SHOW_DEVICES;
break;
case 'D':
show_options |= SHOW_DRIVERS;
break;
case 'h':
usage();
exit(0);
break;
case 'm':
show_module = optarg;
case 'p':
show_options |= SHOW_PATH;
break;
case 'P':
show_options |= SHOW_PARENT;
break;
case 'v':
show_options |= SHOW_ALL_ATTRIB_VALUES;
break;
default:
usage();
exit(1);
}
}
argc -= optind;
argv += optind;
switch(argc) {
case 0:
break;
case 1:
/* get bus to view */
if ((strlen(*argv)) < SYSFS_NAME_LEN) {
device_to_show = *argv;
show_options |= SHOW_DEVICES;
} else {
fprintf(stderr,
"Invalid argument - device name too long\n");
exit(1);
}
break;
default:
usage();
exit(1);
}
if (check_sysfs_is_mounted() == 0) {
fprintf(stderr, "Unable to find sysfs mount point!\n");
exit(1);
}
if ((!show_bus && !show_class && !show_module && !show_root) &&
(show_options & (SHOW_ATTRIBUTES |
SHOW_ATTRIBUTE_VALUE | SHOW_DEVICES |
SHOW_DRIVERS | SHOW_ALL_ATTRIB_VALUES))) {
fprintf(stderr,
"Please specify a bus, class, module, or root device\n");
usage();
exit(1);
}
/* default is to print devices */
if (!(show_options & (SHOW_DEVICES | SHOW_DRIVERS)))
show_options |= SHOW_DEVICES;
if (show_bus) {
if ((!(strcmp(show_bus, "pci")))) {
pacc = (struct pci_access *)
calloc(1, sizeof(struct pci_access));
pacc->pci_id_file_name = (unsigned char *)pci_id_file;
pacc->numeric_ids = 0;
}
retval = show_sysfs_bus(show_bus);
}
if (show_class)
retval = show_sysfs_class(show_class);
if (show_module)
retval = show_sysfs_module(show_module);
if (!show_bus && !show_class && !show_module && !show_root)
retval = show_default_info();
if (show_bus) {
if ((!(strcmp(show_bus, "pci")))) {
pci_free_name_list(pacc);
free (pacc);
pacc = NULL;
}
}
if (!(show_options ^ SHOW_DEVICES))
fprintf(stdout, "\n");
exit(retval);
}