lspci: new applet. +573 bytes

Signed-off-by: Souf Oued <souf_oued@yahoo.fr>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
diff --git a/include/applets.h b/include/applets.h
index d1a84ee..7d5d440 100644
--- a/include/applets.h
+++ b/include/applets.h
@@ -246,6 +246,7 @@
 IF_LSATTR(APPLET(lsattr, _BB_DIR_BIN, _BB_SUID_DROP))
 IF_LSMOD(APPLET(lsmod, _BB_DIR_SBIN, _BB_SUID_DROP))
 IF_MODPROBE_SMALL(APPLET_ODDNAME(lsmod, modprobe, _BB_DIR_SBIN, _BB_SUID_DROP, modprobe))
+IF_LSPCI(APPLET(lspci, _BB_DIR_SBIN, _BB_SUID_DROP))
 IF_UNLZMA(APPLET_ODDNAME(lzmacat, unlzma, _BB_DIR_USR_BIN, _BB_SUID_DROP, lzmacat))
 IF_LZOP(APPLET(lzop, _BB_DIR_BIN, _BB_SUID_DROP))
 IF_LZOP(APPLET_ODDNAME(lzopcat, lzop, _BB_DIR_USR_BIN, _BB_SUID_DROP, lzopcat))
diff --git a/include/usage.h b/include/usage.h
index 1505ac3..6b27384 100644
--- a/include/usage.h
+++ b/include/usage.h
@@ -2502,6 +2502,14 @@
 #define lsmod_full_usage "\n\n" \
        "List the currently loaded kernel modules"
 
+#define lspci_trivial_usage \
+       "[-mk]"
+#define lspci_full_usage "\n\n" \
+       "List all PCI devices" \
+     "\n" \
+     "\n	-m	Parseable output" \
+     "\n	-k	Show driver" \
+
 #if ENABLE_FEATURE_MAKEDEVS_LEAF
 #define makedevs_trivial_usage \
        "NAME TYPE MAJOR MINOR FIRST LAST [s]"
diff --git a/util-linux/Config.in b/util-linux/Config.in
index 9827145..94337df 100644
--- a/util-linux/Config.in
+++ b/util-linux/Config.in
@@ -348,6 +348,15 @@
 	  file or block device, and to query the status of a loop device. This
 	  version does not currently support enabling data encryption.
 
+config LSPCI
+	bool "lspci"
+	default n
+	help
+	  lspci is a utility for displaying information about PCI buses in the
+	  system and devices connected to them.
+
+	  This version uses sysfs (/sys/bus/pci/devices) only.
+
 config MDEV
 	bool "mdev"
 	default n
diff --git a/util-linux/Kbuild b/util-linux/Kbuild
index 89886cb..3004fd0 100644
--- a/util-linux/Kbuild
+++ b/util-linux/Kbuild
@@ -21,6 +21,7 @@
 lib-$(CONFIG_IPCRM)             += ipcrm.o
 lib-$(CONFIG_IPCS)              += ipcs.o
 lib-$(CONFIG_LOSETUP)           += losetup.o
+lib-$(CONFIG_LSPCI)             += lspci.o
 lib-$(CONFIG_MDEV)              += mdev.o
 lib-$(CONFIG_MKFS_EXT2)         += mkfs_ext2.o
 lib-$(CONFIG_MKFS_MINIX)        += mkfs_minix.o
diff --git a/util-linux/lspci.c b/util-linux/lspci.c
new file mode 100644
index 0000000..752d4f5
--- /dev/null
+++ b/util-linux/lspci.c
@@ -0,0 +1,103 @@
+/* vi: set sw=4 ts=4: */
+/*
+* lspci implementation for busybox
+*
+* Copyright (C) 2009  Malek Degachi <malek-degachi@laposte.net>
+*
+* Licensed under the GPL v2 or later, see the file LICENSE in this tarball.
+*/
+#include <libbb.h>
+
+enum {
+	OPT_m = (1 << 0),
+	OPT_k = (1 << 1),
+};
+
+/*
+ * PCI_SLOT_NAME PCI_CLASS: PCI_VID:PCI_DID [PCI_SUBSYS_VID:PCI_SUBSYS_DID] [DRIVER]
+ */
+static int FAST_FUNC fileAction(
+		const char *fileName,
+		struct stat *statbuf UNUSED_PARAM,
+		void *userData UNUSED_PARAM,
+		int depth UNUSED_PARAM)
+{
+	parser_t *parser;
+	char *tokens[3];
+	char *pci_slot_name = NULL, *driver = NULL;
+	int pci_class = 0, pci_vid = 0, pci_did = 0;
+	int pci_subsys_vid = 0, pci_subsys_did = 0;
+
+	char *uevent_filename = concat_path_file(fileName, "/uevent");
+	parser = config_open2(uevent_filename, fopen_for_read);
+	free(uevent_filename);
+
+	while (config_read(parser, tokens, 3, 2, "\0:=", PARSE_NORMAL)) {
+		if (strcmp(tokens[0], "DRIVER") == 0) {
+			driver = xstrdup(tokens[1]);
+			continue;
+		}
+
+		if (strcmp(tokens[0], "PCI_CLASS") == 0) {
+			pci_class = xstrtou(tokens[1], 16)>>8;
+			continue;
+		}
+
+		if (strcmp(tokens[0], "PCI_ID") == 0) {
+			pci_vid = xstrtou(tokens[1], 16);
+			pci_did = xstrtou(tokens[2], 16);
+			continue;
+		}
+
+		if (strcmp(tokens[0], "PCI_SUBSYS_ID") == 0) {
+			pci_subsys_vid = xstrtou(tokens[1], 16);
+			pci_subsys_did = xstrtou(tokens[2], 16);
+			continue;
+		}
+
+		if (strcmp(tokens[0], "PCI_SLOT_NAME") == 0) {
+			pci_slot_name = xstrdup(tokens[2]);
+			continue;
+		}
+	}
+	config_close(parser);
+
+
+	if (option_mask32 & OPT_m) {
+		printf("%s \"Class %04x\" \"%04x\" \"%04x\" \"%04x\" \"%04x\"",
+		       pci_slot_name, pci_class, pci_vid, pci_did,
+		       pci_subsys_vid, pci_subsys_did);
+	} else {
+		printf("%s Class %04x: %04x:%04x",
+		       pci_slot_name, pci_class, pci_vid, pci_did);
+	}
+
+	if ((option_mask32 & OPT_k) && driver) {
+		if (option_mask32 & OPT_m) {
+			printf(" \"%s\"", driver);
+		} else {
+			printf(" %s", driver);
+		}
+	}
+	bb_putchar('\n');
+
+	free(driver);
+	free(pci_slot_name);
+
+	return TRUE;
+}
+
+int lspci_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
+int lspci_main(int argc UNUSED_PARAM, char **argv)
+{
+	getopt32(argv, "m" /*non-compat:*/ "k" /*ignored:*/ "nv");
+
+	recursive_action("/sys/bus/pci/devices",
+			ACTION_RECURSE,
+			fileAction,
+			NULL, /* dirAction */
+			NULL, /* userData */
+			0 /* depth */);
+
+	return EXIT_SUCCESS;
+}