如何在Linux下用C确定USB存储(USB闪存盘)'设备路径'

时间:2013-12-05 10:37:07

标签: c linux flash usb

如何在Linux下使用C以编程方式枚举所有USB存储?我想获取诸如'/ dev / sdb4'之类的字符串,以便我可以使用mount()来挂载它们。

我想有一些方法可以列出所有USB设备,但我只想要USB存储设备。

此外,我们能区分USB闪存盘(直接插入机器的USB端口)和USB盘(通常通过USB线连接到机器)吗?

感谢。

1 个答案:

答案 0 :(得分:3)

您可以在目录/proc/scsi/usb-storage中找到所有USB存储设备。通过列出此目录的内容,您可以找到SCSI主机设备编号,然后您可以使用该编号检查文件/sys/class/scsi_disk/N:*:*:*,其中N是SCSI主机设备的编号。 /sys/class/scsi_disk/中的文件是指向实际设备目录的链接。

以下是这样做的一种方法,尽管它不是最好的方法之一。我想你可能想依靠自动挂载器而不是手动编码。

根据您附加的特定磁盘设备(在您的示例中列出两个)进行混合和匹配的部分留给读者练习。

#include <stdio.h>
#include <dirent.h>
#include <errno.h>
#include <string.h>
#include <libgen.h>
#include <linux/limits.h>
#include <glob.h>
#include <unistd.h>

#define PROC_USB_DEV "/proc/scsi/usb-storage"
#define SYS_SCSI_HOST "/sys/class/scsi_host/host%s/device"
#define SYS_SCSI_DEV "/sys/class/scsi_disk/%s:*"
#define SYS_SCSI_TGT "%s/device/block/*"

int main(int argc, char **argv) {
  DIR *dp = NULL;
  struct dirent *dt = NULL;

  if((dp = opendir(PROC_USB_DEV)) == NULL) {
    fprintf(stderr, "Can not open %s: %s", PROC_USB_DEV, strerror(errno)); 
    return 2;
  }

  /* find usb storage device hosts which appear as scsi hosts */
  /* for the sake of example this one hasn't been done with glob(3) */
  while((dt = readdir(dp)) != NULL) {
    int scsi_dev = 0;
    /* skip '.' and '..', possibly others too */
    if((scsi_dev = atoi(basename(dt->d_name))) < 1) continue;

    char buf[PATH_MAX];
    char pat[PATH_MAX];

    snprintf(buf, PATH_MAX, SYS_SCSI_HOST, basename(dt->d_name));
    snprintf(pat, PATH_MAX, SYS_SCSI_DEV, basename(dt->d_name));

    glob_t hosts;
    size_t count;

    /* find SCSI host device paths */
    glob(pat, 0, 0, &hosts);
    if(hosts.gl_pathc > 0) {
      char **p;
      int n;
      for(p = hosts.gl_pathv, n = hosts.gl_pathc; n; p++, n--) {
        char tgtbuf[PATH_MAX + NAME_MAX];

        snprintf(tgtbuf, PATH_MAX + NAME_MAX, SYS_SCSI_TGT, *p);

        /* find SCSI disk device paths */
        glob_t devs;
        glob(tgtbuf, 0, 0, &devs);
        if(devs.gl_pathc > 0) {
          char **ptr;
          int c;
          for(ptr = devs.gl_pathv, c = devs.gl_pathc; c; ptr++, c--) {
            printf("We would now call mount(2) for /dev/%s.\n",
                   basename(*ptr));
          }
        }
        globfree(&devs);
      }
      globfree(&hosts);
    }
  }
}

作为最后一点:在一般的编码风格方面,这很可能不是最好的例子。