+char **get_mounted_devices(void)
+{
+ FILE *mtab;
+ struct mntent *me;
+ char **mounted = NULL;
+ int n_mounted = 0;
+
+ mtab = setmntent("/etc/mtab", "r");
+ if(!mtab)
+ return NULL;
+
+ while((me = getmntent(mtab)))
+ {
+ mounted = (char **)realloc(mounted, (n_mounted+2)*sizeof(char *));
+ mounted[n_mounted] = strdup(me->mnt_fsname);
+ ++n_mounted;
+ }
+
+ endmntent(mtab);
+ mounted[n_mounted] = NULL;
+
+ return mounted;
+}
+
+int is_mounted(char **mounted, char *devname)
+{
+ int i;
+ for(i=0; mounted[i]; ++i)
+ if(!strcmp(devname, mounted[i]))
+ return 1;
+ return 0;
+}
+
+void free_mounted_devices(char **mounted)
+{
+ int i;
+ if(!mounted)
+ return;
+ for(i=0; mounted[i]; ++i)
+ free(mounted[i]);
+ free(mounted);
+}
+
+int is_removable(char *devpath)
+{
+ char fnbuf[256];
+ int len;
+ char *ptr;
+ int fd;
+
+ len = snprintf(fnbuf, sizeof(fnbuf), "/sys%s", devpath);
+ if(len+10>=(int)sizeof(fnbuf))
+ return 0;
+
+ for(ptr=fnbuf+len; (ptr>fnbuf && *ptr!='/'); --ptr) ;
+ strcpy(ptr, "/removable");
+ fd = open(fnbuf, O_RDONLY);
+ if(fd!=-1)
+ {
+ char c;
+ read(fd, &c, 1);
+ close(fd);
+ if(c=='1')
+ {
+ if(verbosity>=2)
+ printf(" Removable\n");
+ return 1;
+ }
+ if(verbosity>=2)
+ printf(" Not removable\n");
+ }
+
+ return 0;
+}
+
+int check_buses(char *devpath, char **buses)
+{
+ char fnbuf[256];
+ char *ptr;
+ int len;
+
+ len = snprintf(fnbuf, sizeof(fnbuf), "/sys%s", devpath);
+ if(len+10>=(int)sizeof(fnbuf))
+ return 0;
+
+ for(ptr=fnbuf+len; ptr>fnbuf+12; --ptr)
+ if(*ptr=='/')
+ {
+ char linkbuf[256];
+ strcpy(ptr, "/subsystem");
+ len = readlink(fnbuf, linkbuf, sizeof(linkbuf)-1);
+ *ptr = 0;
+
+ if(len!=-1)
+ {
+ int i;
+ linkbuf[len] = 0;
+ for(; (len>0 && linkbuf[len-1]!='/'); --len) ;
+ if(verbosity>=2)
+ printf(" Subsystem of %s is %s\n", fnbuf, linkbuf+len);
+ for(i=0; buses[i]; ++i)
+ if(strcmp(linkbuf+len, buses[i])==0)
+ return 1;
+ }
+ }
+
+ return 0;
+}
+
+int can_mount(Property *props)
+{
+ static char *removable_buses[] = { "usb", "firewire", 0 };
+ char *devpath;
+ int i;
+
+ if(!match_property_value(props, "DEVTYPE", "partition"))
+ return 0;
+
+ devpath = get_property_value(props, "DEVPATH");
+ if(is_removable(devpath))
+ return 1;
+
+ for(i=0; removable_buses[i]; ++i)
+ if(match_property_value(props, "ID_BUS", removable_buses[i]))
+ return 1;
+
+ return check_buses(devpath, removable_buses);
+}
+
+char **get_device_nodes(char *dirname)