mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-04 09:18:36 +03:00
957904933b
When /dev entries or sysfs entries are changing due to concurrent lvm commands, it can cause warning/error messages about missing paths.
1659 lines
39 KiB
C
1659 lines
39 KiB
C
/*
|
|
* Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
|
|
* Copyright (C) 2004-2007 Red Hat, Inc. All rights reserved.
|
|
*
|
|
* This file is part of LVM2.
|
|
*
|
|
* This copyrighted material is made available to anyone wishing to use,
|
|
* modify, copy, or redistribute it subject to the terms and conditions
|
|
* of the GNU Lesser General Public License v.2.1.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include "base/memory/zalloc.h"
|
|
#include "lib/misc/lib.h"
|
|
#include "lib/device/dev-type.h"
|
|
#include "lib/datastruct/btree.h"
|
|
#include "lib/config/config.h"
|
|
#include "lib/commands/toolcontext.h"
|
|
#include "device_mapper/misc/dm-ioctl.h"
|
|
#include "lib/misc/lvm-string.h"
|
|
|
|
#ifdef UDEV_SYNC_SUPPORT
|
|
#include <libudev.h>
|
|
#endif
|
|
#include <unistd.h>
|
|
#include <dirent.h>
|
|
|
|
struct dev_iter {
|
|
struct btree_iter *current;
|
|
struct dev_filter *filter;
|
|
};
|
|
|
|
struct dir_list {
|
|
struct dm_list list;
|
|
char dir[0];
|
|
};
|
|
|
|
static struct {
|
|
struct dm_pool *mem;
|
|
struct dm_hash_table *names;
|
|
struct dm_hash_table *vgid_index;
|
|
struct dm_hash_table *lvid_index;
|
|
struct btree *sysfs_only_devices; /* see comments in _get_device_for_sysfs_dev_name_using_devno */
|
|
struct btree *devices;
|
|
struct dm_regex *preferred_names_matcher;
|
|
const char *dev_dir;
|
|
|
|
int has_scanned;
|
|
struct dm_list dirs;
|
|
struct dm_list files;
|
|
|
|
} _cache;
|
|
|
|
#define _zalloc(x) dm_pool_zalloc(_cache.mem, (x))
|
|
#define _free(x) dm_pool_free(_cache.mem, (x))
|
|
#define _strdup(x) dm_pool_strdup(_cache.mem, (x))
|
|
|
|
static int _insert(const char *path, const struct stat *info,
|
|
int rec, int check_with_udev_db);
|
|
|
|
/* Setup non-zero members of passed zeroed 'struct device' */
|
|
static void _dev_init(struct device *dev)
|
|
{
|
|
dev->fd = -1;
|
|
dev->read_ahead = -1;
|
|
|
|
dev->ext.enabled = 0;
|
|
dev->ext.src = DEV_EXT_NONE;
|
|
|
|
dm_list_init(&dev->aliases);
|
|
}
|
|
|
|
void dev_destroy_file(struct device *dev)
|
|
{
|
|
if (!(dev->flags & DEV_ALLOCED))
|
|
return;
|
|
|
|
free((void *) dm_list_item(dev->aliases.n, struct dm_str_list)->str);
|
|
free(dev->aliases.n);
|
|
free(dev);
|
|
}
|
|
|
|
struct device *dev_create_file(const char *filename, struct device *dev,
|
|
struct dm_str_list *alias, int use_malloc)
|
|
{
|
|
int allocate = !dev;
|
|
|
|
if (allocate) {
|
|
if (use_malloc) {
|
|
if (!(dev = zalloc(sizeof(*dev)))) {
|
|
log_error("struct device allocation failed");
|
|
return NULL;
|
|
}
|
|
if (!(alias = zalloc(sizeof(*alias)))) {
|
|
log_error("struct dm_str_list allocation failed");
|
|
free(dev);
|
|
return NULL;
|
|
}
|
|
if (!(alias->str = strdup(filename))) {
|
|
log_error("filename strdup failed");
|
|
free(dev);
|
|
free(alias);
|
|
return NULL;
|
|
}
|
|
} else {
|
|
if (!(dev = _zalloc(sizeof(*dev)))) {
|
|
log_error("struct device allocation failed");
|
|
return NULL;
|
|
}
|
|
if (!(alias = _zalloc(sizeof(*alias)))) {
|
|
log_error("struct dm_str_list allocation failed");
|
|
_free(dev);
|
|
return NULL;
|
|
}
|
|
if (!(alias->str = _strdup(filename))) {
|
|
log_error("filename strdup failed");
|
|
_free(dev);
|
|
return NULL;
|
|
}
|
|
}
|
|
} else if (!(alias->str = strdup(filename))) {
|
|
log_error("filename strdup failed");
|
|
return NULL;
|
|
}
|
|
|
|
_dev_init(dev);
|
|
dev->flags = DEV_REGULAR | ((use_malloc) ? DEV_ALLOCED : 0);
|
|
dm_list_add(&dev->aliases, &alias->list);
|
|
|
|
return dev;
|
|
}
|
|
|
|
static struct device *_dev_create(dev_t d)
|
|
{
|
|
struct device *dev;
|
|
|
|
if (!(dev = _zalloc(sizeof(*dev)))) {
|
|
log_error("struct device allocation failed");
|
|
return NULL;
|
|
}
|
|
|
|
_dev_init(dev);
|
|
dev->dev = d;
|
|
|
|
return dev;
|
|
}
|
|
|
|
void dev_set_preferred_name(struct dm_str_list *sl, struct device *dev)
|
|
{
|
|
/*
|
|
* Don't interfere with ordering specified in config file.
|
|
*/
|
|
if (_cache.preferred_names_matcher)
|
|
return;
|
|
|
|
log_debug_devs("%s: New preferred name", sl->str);
|
|
dm_list_del(&sl->list);
|
|
dm_list_add_h(&dev->aliases, &sl->list);
|
|
}
|
|
|
|
/*
|
|
* Check whether path0 or path1 contains the subpath. The path that
|
|
* *does not* contain the subpath wins (return 0 or 1). If both paths
|
|
* contain the subpath, return -1. If none of them contains the subpath,
|
|
* return -2.
|
|
*/
|
|
static int _builtin_preference(const char *path0, const char *path1,
|
|
size_t skip_prefix_count, const char *subpath)
|
|
{
|
|
size_t subpath_len;
|
|
int r0, r1;
|
|
|
|
subpath_len = strlen(subpath);
|
|
|
|
r0 = !strncmp(path0 + skip_prefix_count, subpath, subpath_len);
|
|
r1 = !strncmp(path1 + skip_prefix_count, subpath, subpath_len);
|
|
|
|
if (!r0 && r1)
|
|
/* path0 does not have the subpath - it wins */
|
|
return 0;
|
|
else if (r0 && !r1)
|
|
/* path1 does not have the subpath - it wins */
|
|
return 1;
|
|
else if (r0 && r1)
|
|
/* both of them have the subpath */
|
|
return -1;
|
|
|
|
/* no path has the subpath */
|
|
return -2;
|
|
}
|
|
|
|
static int _apply_builtin_path_preference_rules(const char *path0, const char *path1)
|
|
{
|
|
size_t devdir_len;
|
|
int r;
|
|
|
|
devdir_len = strlen(_cache.dev_dir);
|
|
|
|
if (!strncmp(path0, _cache.dev_dir, devdir_len) &&
|
|
!strncmp(path1, _cache.dev_dir, devdir_len)) {
|
|
/*
|
|
* We're trying to achieve the ordering:
|
|
* /dev/block/ < /dev/dm-* < /dev/disk/ < /dev/mapper/ < anything else
|
|
*/
|
|
|
|
/* Prefer any other path over /dev/block/ path. */
|
|
if ((r = _builtin_preference(path0, path1, devdir_len, "block/")) >= -1)
|
|
return r;
|
|
|
|
/* Prefer any other path over /dev/dm-* path. */
|
|
if ((r = _builtin_preference(path0, path1, devdir_len, "dm-")) >= -1)
|
|
return r;
|
|
|
|
/* Prefer any other path over /dev/disk/ path. */
|
|
if ((r = _builtin_preference(path0, path1, devdir_len, "disk/")) >= -1)
|
|
return r;
|
|
|
|
/* Prefer any other path over /dev/mapper/ path. */
|
|
if ((r = _builtin_preference(path0, path1, 0, dm_dir())) >= -1)
|
|
return r;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/* Return 1 if we prefer path1 else return 0 */
|
|
static int _compare_paths(const char *path0, const char *path1)
|
|
{
|
|
int slash0 = 0, slash1 = 0;
|
|
int m0, m1;
|
|
const char *p;
|
|
char p0[PATH_MAX], p1[PATH_MAX];
|
|
char *s0, *s1;
|
|
struct stat stat0, stat1;
|
|
int r;
|
|
|
|
/*
|
|
* FIXME Better to compare patterns one-at-a-time against all names.
|
|
*/
|
|
if (_cache.preferred_names_matcher) {
|
|
m0 = dm_regex_match(_cache.preferred_names_matcher, path0);
|
|
m1 = dm_regex_match(_cache.preferred_names_matcher, path1);
|
|
|
|
if (m0 != m1) {
|
|
if (m0 < 0)
|
|
return 1;
|
|
if (m1 < 0)
|
|
return 0;
|
|
if (m0 < m1)
|
|
return 1;
|
|
if (m1 < m0)
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* Apply built-in preference rules first. */
|
|
if ((r = _apply_builtin_path_preference_rules(path0, path1)) >= 0)
|
|
return r;
|
|
|
|
/* Return the path with fewer slashes */
|
|
for (p = path0; p++; p = (const char *) strchr(p, '/'))
|
|
slash0++;
|
|
|
|
for (p = path1; p++; p = (const char *) strchr(p, '/'))
|
|
slash1++;
|
|
|
|
if (slash0 < slash1)
|
|
return 0;
|
|
if (slash1 < slash0)
|
|
return 1;
|
|
|
|
(void) dm_strncpy(p0, path0, sizeof(p0));
|
|
(void) dm_strncpy(p1, path1, sizeof(p1));
|
|
s0 = p0 + 1;
|
|
s1 = p1 + 1;
|
|
|
|
/*
|
|
* If we reach here, both paths are the same length.
|
|
* Now skip past identical path components.
|
|
*/
|
|
while (*s0 && *s0 == *s1)
|
|
s0++, s1++;
|
|
|
|
/* We prefer symlinks - they exist for a reason!
|
|
* So we prefer a shorter path before the first symlink in the name.
|
|
* FIXME Configuration option to invert this? */
|
|
while (s0) {
|
|
s0 = strchr(s0, '/');
|
|
s1 = strchr(s1, '/');
|
|
if (s0) {
|
|
*s0 = '\0';
|
|
*s1 = '\0';
|
|
}
|
|
if (lstat(p0, &stat0)) {
|
|
log_sys_very_verbose("lstat", p0);
|
|
return 1;
|
|
}
|
|
if (lstat(p1, &stat1)) {
|
|
log_sys_very_verbose("lstat", p1);
|
|
return 0;
|
|
}
|
|
if (S_ISLNK(stat0.st_mode) && !S_ISLNK(stat1.st_mode))
|
|
return 0;
|
|
if (!S_ISLNK(stat0.st_mode) && S_ISLNK(stat1.st_mode))
|
|
return 1;
|
|
if (s0) {
|
|
*s0++ = '/';
|
|
*s1++ = '/';
|
|
}
|
|
}
|
|
|
|
/* ASCII comparison */
|
|
if (strcmp(path0, path1) < 0)
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _add_alias(struct device *dev, const char *path)
|
|
{
|
|
struct dm_str_list *sl = _zalloc(sizeof(*sl));
|
|
struct dm_str_list *strl;
|
|
const char *oldpath;
|
|
int prefer_old = 1;
|
|
|
|
if (!sl)
|
|
return_0;
|
|
|
|
/* Is name already there? */
|
|
dm_list_iterate_items(strl, &dev->aliases) {
|
|
if (!strcmp(strl->str, path))
|
|
return 1;
|
|
}
|
|
|
|
sl->str = path;
|
|
|
|
if (!dm_list_empty(&dev->aliases)) {
|
|
oldpath = dm_list_item(dev->aliases.n, struct dm_str_list)->str;
|
|
prefer_old = _compare_paths(path, oldpath);
|
|
}
|
|
|
|
if (prefer_old)
|
|
dm_list_add(&dev->aliases, &sl->list);
|
|
else
|
|
dm_list_add_h(&dev->aliases, &sl->list);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _get_sysfs_value(const char *path, char *buf, size_t buf_size, int error_if_no_value)
|
|
{
|
|
FILE *fp;
|
|
size_t len;
|
|
int r = 0;
|
|
|
|
if (!(fp = fopen(path, "r"))) {
|
|
if (error_if_no_value)
|
|
log_sys_error("fopen", path);
|
|
return 0;
|
|
}
|
|
|
|
if (!fgets(buf, buf_size, fp)) {
|
|
if (error_if_no_value)
|
|
log_sys_error("fgets", path);
|
|
goto out;
|
|
}
|
|
|
|
if ((len = strlen(buf)) && buf[len - 1] == '\n')
|
|
buf[--len] = '\0';
|
|
|
|
if (!len && error_if_no_value)
|
|
log_error("_get_sysfs_value: %s: no value", path);
|
|
else
|
|
r = 1;
|
|
out:
|
|
if (fclose(fp))
|
|
log_sys_error("fclose", path);
|
|
|
|
return r;
|
|
}
|
|
|
|
static int _get_dm_uuid_from_sysfs(char *buf, size_t buf_size, int major, int minor)
|
|
{
|
|
char path[PATH_MAX];
|
|
|
|
if (dm_snprintf(path, sizeof(path), "%sdev/block/%d:%d/dm/uuid", dm_sysfs_dir(), major, minor) < 0) {
|
|
log_error("%d:%d: dm_snprintf failed for path to sysfs dm directory.", major, minor);
|
|
return 0;
|
|
}
|
|
|
|
return _get_sysfs_value(path, buf, buf_size, 0);
|
|
}
|
|
|
|
static struct dm_list *_get_or_add_list_by_index_key(struct dm_hash_table *idx, const char *key)
|
|
{
|
|
struct dm_list *list;
|
|
|
|
if ((list = dm_hash_lookup(idx, key)))
|
|
return list;
|
|
|
|
if (!(list = _zalloc(sizeof(*list)))) {
|
|
log_error("%s: failed to allocate device list for device cache index.", key);
|
|
return NULL;
|
|
}
|
|
|
|
dm_list_init(list);
|
|
|
|
if (!dm_hash_insert(idx, key, list)) {
|
|
log_error("%s: failed to insert device list to device cache index.", key);
|
|
return NULL;
|
|
}
|
|
|
|
return list;
|
|
}
|
|
|
|
static struct device *_insert_sysfs_dev(dev_t devno, const char *devname)
|
|
{
|
|
static struct device _fake_dev = { .flags = DEV_USED_FOR_LV };
|
|
struct stat stat0;
|
|
char path[PATH_MAX];
|
|
char *path_copy;
|
|
struct device *dev;
|
|
|
|
if (dm_snprintf(path, sizeof(path), "%s%s", _cache.dev_dir, devname) < 0) {
|
|
log_error("_insert_sysfs_dev: %s: dm_snprintf failed", devname);
|
|
return NULL;
|
|
}
|
|
|
|
if (lstat(path, &stat0) < 0) {
|
|
/* When device node does not exist return fake entry.
|
|
* This may happen when i.e. lvm2 device dir != /dev */
|
|
log_debug("%s: Not available device node", path);
|
|
return &_fake_dev;
|
|
}
|
|
|
|
if (!(dev = _dev_create(devno)))
|
|
return_NULL;
|
|
|
|
if (!(path_copy = dm_pool_strdup(_cache.mem, path))) {
|
|
log_error("_insert_sysfs_dev: %s: dm_pool_strdup failed", devname);
|
|
return NULL;
|
|
}
|
|
|
|
if (!_add_alias(dev, path_copy)) {
|
|
log_error("Couldn't add alias to dev cache.");
|
|
_free(dev);
|
|
return NULL;
|
|
}
|
|
|
|
if (!btree_insert(_cache.sysfs_only_devices, (uint32_t) devno, dev)) {
|
|
log_error("Couldn't add device to binary tree of sysfs-only devices in dev cache.");
|
|
_free(dev);
|
|
return NULL;
|
|
}
|
|
|
|
return dev;
|
|
}
|
|
|
|
static struct device *_get_device_for_sysfs_dev_name_using_devno(const char *devname)
|
|
{
|
|
char path[PATH_MAX];
|
|
char buf[PATH_MAX];
|
|
int major, minor;
|
|
dev_t devno;
|
|
struct device *dev;
|
|
|
|
if (dm_snprintf(path, sizeof(path), "%sblock/%s/dev", dm_sysfs_dir(), devname) < 0) {
|
|
log_error("_get_device_for_sysfs_dev_name_using_devno: %s: dm_snprintf failed", devname);
|
|
return NULL;
|
|
}
|
|
|
|
if (!_get_sysfs_value(path, buf, sizeof(buf), 1))
|
|
return_NULL;
|
|
|
|
if (sscanf(buf, "%d:%d", &major, &minor) != 2) {
|
|
log_error("_get_device_for_sysfs_dev_name_using_devno: %s: failed to get major and minor number", devname);
|
|
return NULL;
|
|
}
|
|
|
|
devno = MKDEV(major, minor);
|
|
if (!(dev = (struct device *) btree_lookup(_cache.devices, (uint32_t) devno))) {
|
|
/*
|
|
* If we get here, it means the device is referenced in sysfs, but it's not yet in /dev.
|
|
* This may happen in some rare cases right after LVs get created - we sync with udev
|
|
* (or alternatively we create /dev content ourselves) while VG lock is held. However,
|
|
* dev scan is done without VG lock so devices may already be in sysfs, but /dev may
|
|
* not be updated yet if we call LVM command right after LV creation. This is not a
|
|
* problem with devtmpfs as there's at least kernel name for device in /dev as soon
|
|
* as the sysfs item exists, but we still support environments without devtmpfs or
|
|
* where different directory for dev nodes is used (e.g. our test suite). So track
|
|
* such devices in _cache.sysfs_only_devices hash for the vgid/lvid check to work still.
|
|
*/
|
|
if (!(dev = (struct device *) btree_lookup(_cache.sysfs_only_devices, (uint32_t) devno)) &&
|
|
!(dev = _insert_sysfs_dev(devno, devname)))
|
|
return_NULL;
|
|
}
|
|
|
|
return dev;
|
|
}
|
|
|
|
#define NOT_LVM_UUID "-"
|
|
|
|
static int _get_vgid_and_lvid_for_dev(struct device *dev)
|
|
{
|
|
static size_t lvm_prefix_len = sizeof(UUID_PREFIX) - 1;
|
|
static size_t lvm_uuid_len = sizeof(UUID_PREFIX) - 1 + 2 * ID_LEN;
|
|
char uuid[DM_UUID_LEN];
|
|
size_t uuid_len;
|
|
|
|
if (!_get_dm_uuid_from_sysfs(uuid, sizeof(uuid), (int) MAJOR(dev->dev), (int) MINOR(dev->dev)))
|
|
return_0;
|
|
|
|
uuid_len = strlen(uuid);
|
|
|
|
/*
|
|
* UUID for LV is either "LVM-<vg_uuid><lv_uuid>" or "LVM-<vg_uuid><lv_uuid>-<suffix>",
|
|
* where vg_uuid and lv_uuid has length of ID_LEN and suffix len is not restricted
|
|
* (only restricted by whole DM UUID max len).
|
|
*/
|
|
if (((uuid_len == lvm_uuid_len) ||
|
|
((uuid_len > lvm_uuid_len) && (uuid[lvm_uuid_len] == '-'))) &&
|
|
!strncmp(uuid, UUID_PREFIX, lvm_prefix_len)) {
|
|
/* Separate VGID and LVID part from DM UUID. */
|
|
if (!(dev->vgid = dm_pool_strndup(_cache.mem, uuid + lvm_prefix_len, ID_LEN)) ||
|
|
!(dev->lvid = dm_pool_strndup(_cache.mem, uuid + lvm_prefix_len + ID_LEN, ID_LEN)))
|
|
return_0;
|
|
} else
|
|
dev->vgid = dev->lvid = NOT_LVM_UUID;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _index_dev_by_vgid_and_lvid(struct device *dev)
|
|
{
|
|
const char *devname = dev_name(dev);
|
|
char devpath[PATH_MAX];
|
|
char path[PATH_MAX];
|
|
DIR *d;
|
|
struct dirent *dirent;
|
|
struct device *holder_dev;
|
|
struct dm_list *vgid_list, *lvid_list;
|
|
struct device_list *dl_vgid, *dl_lvid;
|
|
int r = 0;
|
|
|
|
if (dev->flags & DEV_USED_FOR_LV)
|
|
/* already indexed */
|
|
return 1;
|
|
|
|
/* Get holders for device. */
|
|
if (dm_snprintf(path, sizeof(path), "%sdev/block/%d:%d/holders/", dm_sysfs_dir(), (int) MAJOR(dev->dev), (int) MINOR(dev->dev)) < 0) {
|
|
log_error("%s: dm_snprintf failed for path to holders directory.", devname);
|
|
return 0;
|
|
}
|
|
|
|
if (!(d = opendir(path))) {
|
|
if (errno == ENOENT) {
|
|
log_debug("%s: path does not exist, skipping", path);
|
|
return 1;
|
|
}
|
|
log_sys_error("opendir", path);
|
|
return 0;
|
|
}
|
|
|
|
/* Iterate over device's holders and look for LVs. */
|
|
while ((dirent = readdir(d))) {
|
|
if (!strcmp(".", dirent->d_name) ||
|
|
!strcmp("..", dirent->d_name))
|
|
continue;
|
|
|
|
if (dm_snprintf(devpath, sizeof(devpath), "%s%s", _cache.dev_dir, dirent->d_name) == -1) {
|
|
log_error("%s: dm_snprintf failed for holder %s device path.", devname, dirent->d_name);
|
|
goto out;
|
|
}
|
|
|
|
if (!(holder_dev = (struct device *) dm_hash_lookup(_cache.names, devpath))) {
|
|
/*
|
|
* Cope with situation where canonical /<dev_dir>/<dirent->d_name>
|
|
* does not exist, but some other node name or symlink exists in
|
|
* non-standard environments - someone renaming the nodes or using
|
|
* mknod with different dev names than actual kernel names.
|
|
* This looks up struct device by major:minor pair which we get
|
|
* by looking at /sys/block/<dirent->d_name>/dev sysfs attribute.
|
|
*/
|
|
if (!(holder_dev = _get_device_for_sysfs_dev_name_using_devno(dirent->d_name))) {
|
|
log_error("%s: failed to find associated device structure for holder %s.", devname, devpath);
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
/* We're only interested in a holder which is a DM device. */
|
|
if (!dm_is_dm_major(MAJOR(holder_dev->dev)))
|
|
continue;
|
|
|
|
/*
|
|
* And if it's a DM device, we're only interested in a holder which is an LVM device.
|
|
* Get the VG UUID and LV UUID if we don't have that already.
|
|
*/
|
|
if (!holder_dev->vgid && !_get_vgid_and_lvid_for_dev(holder_dev))
|
|
goto_out;
|
|
|
|
if (*holder_dev->vgid == *NOT_LVM_UUID)
|
|
continue;
|
|
|
|
/*
|
|
* Do not add internal LV devices to index.
|
|
* If a device is internal, the holder has the same VG UUID as the device.
|
|
*/
|
|
if (dm_is_dm_major(MAJOR(dev->dev))) {
|
|
if (!dev->vgid && !_get_vgid_and_lvid_for_dev(dev))
|
|
goto_out;
|
|
|
|
if (*dev->vgid != *NOT_LVM_UUID && !strcmp(holder_dev->vgid, dev->vgid))
|
|
continue;
|
|
}
|
|
|
|
if (!(vgid_list = _get_or_add_list_by_index_key(_cache.vgid_index, holder_dev->vgid)) ||
|
|
!(lvid_list = _get_or_add_list_by_index_key(_cache.lvid_index, holder_dev->lvid)))
|
|
goto_out;
|
|
|
|
/* Create dev list items for the holder device. */
|
|
if (!(dl_vgid = _zalloc(sizeof(*dl_vgid))) ||
|
|
!(dl_lvid = _zalloc(sizeof(*dl_lvid)))) {
|
|
log_error("%s: failed to allocate dev list item.", devname);
|
|
goto out;
|
|
}
|
|
|
|
dl_vgid->dev = dl_lvid->dev = dev;
|
|
|
|
/* Add dev list item to VGID device list if it's not there already. */
|
|
if (!(dev->flags & DEV_USED_FOR_LV))
|
|
dm_list_add(vgid_list, &dl_vgid->list);
|
|
|
|
/* Add dev list item to LVID device list. */
|
|
dm_list_add(lvid_list, &dl_lvid->list);
|
|
|
|
/* Mark device as used == also indexed in dev cache by VGID and LVID. */
|
|
dev->flags |= DEV_USED_FOR_LV;
|
|
}
|
|
|
|
r = 1;
|
|
out:
|
|
if (closedir(d))
|
|
log_sys_error("closedir", path);
|
|
|
|
return r;
|
|
}
|
|
|
|
struct dm_list *dev_cache_get_dev_list_for_vgid(const char *vgid)
|
|
{
|
|
return dm_hash_lookup(_cache.vgid_index, vgid);
|
|
}
|
|
|
|
struct dm_list *dev_cache_get_dev_list_for_lvid(const char *lvid)
|
|
{
|
|
return dm_hash_lookup(_cache.lvid_index, lvid);
|
|
}
|
|
|
|
/*
|
|
* Scanning code calls this when it fails to open a device using
|
|
* this path. The path is dropped from dev-cache. In the next
|
|
* dev_cache_scan it may be added again, but it could be for a
|
|
* different device.
|
|
*/
|
|
|
|
void dev_cache_failed_path(struct device *dev, const char *path)
|
|
{
|
|
struct dm_str_list *strl;
|
|
|
|
if (dm_hash_lookup(_cache.names, path))
|
|
dm_hash_remove(_cache.names, path);
|
|
|
|
dm_list_iterate_items(strl, &dev->aliases) {
|
|
if (!strcmp(strl->str, path)) {
|
|
dm_list_del(&strl->list);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Either creates a new dev, or adds an alias to
|
|
* an existing dev.
|
|
*/
|
|
static int _insert_dev(const char *path, dev_t d)
|
|
{
|
|
struct device *dev;
|
|
struct device *dev_by_devt;
|
|
struct device *dev_by_path;
|
|
char *path_copy;
|
|
|
|
dev_by_devt = (struct device *) btree_lookup(_cache.devices, (uint32_t) d);
|
|
dev_by_path = (struct device *) dm_hash_lookup(_cache.names, path);
|
|
dev = dev_by_devt;
|
|
|
|
/*
|
|
* Existing device, existing path points to the same device.
|
|
*/
|
|
if (dev_by_devt && dev_by_path && (dev_by_devt == dev_by_path)) {
|
|
log_debug_devs("Found dev %d:%d %s - exists. %.8s",
|
|
(int)MAJOR(d), (int)MINOR(d), path, dev->pvid);
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* No device or path found, add devt to cache.devices, add name to cache.names.
|
|
*/
|
|
if (!dev_by_devt && !dev_by_path) {
|
|
log_debug_devs("Found dev %d:%d %s - new.",
|
|
(int)MAJOR(d), (int)MINOR(d), path);
|
|
|
|
if (!(dev = (struct device *) btree_lookup(_cache.sysfs_only_devices, (uint32_t) d))) {
|
|
/* create new device */
|
|
if (!(dev = _dev_create(d)))
|
|
return_0;
|
|
}
|
|
|
|
if (!(btree_insert(_cache.devices, (uint32_t) d, dev))) {
|
|
log_error("Couldn't insert device into binary tree.");
|
|
_free(dev);
|
|
return 0;
|
|
}
|
|
|
|
if (!(path_copy = dm_pool_strdup(_cache.mem, path))) {
|
|
log_error("Failed to duplicate path string.");
|
|
return 0;
|
|
}
|
|
|
|
if (!_add_alias(dev, path_copy)) {
|
|
log_error("Couldn't add alias to dev cache.");
|
|
return 0;
|
|
}
|
|
|
|
if (!dm_hash_insert(_cache.names, path_copy, dev)) {
|
|
log_error("Couldn't add name to hash in dev cache.");
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Existing device, path is new, add path as a new alias for the device.
|
|
*/
|
|
if (dev_by_devt && !dev_by_path) {
|
|
log_debug_devs("Found dev %d:%d %s - new alias.",
|
|
(int)MAJOR(d), (int)MINOR(d), path);
|
|
|
|
if (!(path_copy = dm_pool_strdup(_cache.mem, path))) {
|
|
log_error("Failed to duplicate path string.");
|
|
return 0;
|
|
}
|
|
|
|
if (!_add_alias(dev, path_copy)) {
|
|
log_error("Couldn't add alias to dev cache.");
|
|
return 0;
|
|
}
|
|
|
|
if (!dm_hash_insert(_cache.names, path_copy, dev)) {
|
|
log_error("Couldn't add name to hash in dev cache.");
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* No existing device, but path exists and previously pointed
|
|
* to a different device.
|
|
*/
|
|
if (!dev_by_devt && dev_by_path) {
|
|
log_debug_devs("Found dev %d:%d %s - new device, path was previously %d:%d.",
|
|
(int)MAJOR(d), (int)MINOR(d), path,
|
|
(int)MAJOR(dev_by_path->dev), (int)MINOR(dev_by_path->dev));
|
|
|
|
if (!(dev = (struct device *) btree_lookup(_cache.sysfs_only_devices, (uint32_t) d))) {
|
|
/* create new device */
|
|
if (!(dev = _dev_create(d)))
|
|
return_0;
|
|
}
|
|
|
|
if (!(btree_insert(_cache.devices, (uint32_t) d, dev))) {
|
|
log_error("Couldn't insert device into binary tree.");
|
|
_free(dev);
|
|
return 0;
|
|
}
|
|
|
|
if (!(path_copy = dm_pool_strdup(_cache.mem, path))) {
|
|
log_error("Failed to duplicate path string.");
|
|
return 0;
|
|
}
|
|
|
|
if (!_add_alias(dev, path_copy)) {
|
|
log_error("Couldn't add alias to dev cache.");
|
|
return 0;
|
|
}
|
|
|
|
dm_hash_remove(_cache.names, path);
|
|
|
|
if (!dm_hash_insert(_cache.names, path_copy, dev)) {
|
|
log_error("Couldn't add name to hash in dev cache.");
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
/*
|
|
* Existing device, and path exists and previously pointed to
|
|
* a different device.
|
|
*/
|
|
if (dev_by_devt && dev_by_path) {
|
|
log_debug_devs("Found dev %d:%d %s - existing device, path was previously %d:%d.",
|
|
(int)MAJOR(d), (int)MINOR(d), path,
|
|
(int)MAJOR(dev_by_path->dev), (int)MINOR(dev_by_path->dev));
|
|
|
|
if (!(path_copy = dm_pool_strdup(_cache.mem, path))) {
|
|
log_error("Failed to duplicate path string.");
|
|
return 0;
|
|
}
|
|
|
|
if (!_add_alias(dev, path_copy)) {
|
|
log_error("Couldn't add alias to dev cache.");
|
|
return 0;
|
|
}
|
|
|
|
dm_hash_remove(_cache.names, path);
|
|
|
|
if (!dm_hash_insert(_cache.names, path_copy, dev)) {
|
|
log_error("Couldn't add name to hash in dev cache.");
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
log_error("Found dev %d:%d %s - failed to use.", (int)MAJOR(d), (int)MINOR(d), path);
|
|
return 0;
|
|
}
|
|
|
|
static char *_join(const char *dir, const char *name)
|
|
{
|
|
size_t len = strlen(dir) + strlen(name) + 2;
|
|
char *r = malloc(len);
|
|
if (r)
|
|
snprintf(r, len, "%s/%s", dir, name);
|
|
|
|
return r;
|
|
}
|
|
|
|
/*
|
|
* Get rid of extra slashes in the path string.
|
|
*/
|
|
static void _collapse_slashes(char *str)
|
|
{
|
|
char *ptr;
|
|
int was_slash = 0;
|
|
|
|
for (ptr = str; *ptr; ptr++) {
|
|
if (*ptr == '/') {
|
|
if (was_slash)
|
|
continue;
|
|
|
|
was_slash = 1;
|
|
} else
|
|
was_slash = 0;
|
|
*str++ = *ptr;
|
|
}
|
|
|
|
*str = *ptr;
|
|
}
|
|
|
|
static int _insert_dir(const char *dir)
|
|
{
|
|
int n, dirent_count, r = 1;
|
|
struct dirent **dirent;
|
|
char *path;
|
|
|
|
dirent_count = scandir(dir, &dirent, NULL, alphasort);
|
|
if (dirent_count > 0) {
|
|
for (n = 0; n < dirent_count; n++) {
|
|
if (dirent[n]->d_name[0] == '.') {
|
|
free(dirent[n]);
|
|
continue;
|
|
}
|
|
|
|
if (!(path = _join(dir, dirent[n]->d_name)))
|
|
return_0;
|
|
|
|
_collapse_slashes(path);
|
|
r &= _insert(path, NULL, 1, 0);
|
|
free(path);
|
|
|
|
free(dirent[n]);
|
|
}
|
|
free(dirent);
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
static int _dev_cache_iterate_devs_for_index(void)
|
|
{
|
|
struct btree_iter *iter = btree_first(_cache.devices);
|
|
struct device *dev;
|
|
int r = 1;
|
|
|
|
while (iter) {
|
|
dev = btree_get_data(iter);
|
|
|
|
if (!_index_dev_by_vgid_and_lvid(dev))
|
|
r = 0;
|
|
|
|
iter = btree_next(iter);
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
static int _dev_cache_iterate_sysfs_for_index(const char *path)
|
|
{
|
|
char devname[PATH_MAX];
|
|
DIR *d;
|
|
struct dirent *dirent;
|
|
int major, minor;
|
|
dev_t devno;
|
|
struct device *dev;
|
|
int partial_failure = 0;
|
|
int r = 0;
|
|
|
|
if (!(d = opendir(path))) {
|
|
log_sys_error("opendir", path);
|
|
return 0;
|
|
}
|
|
|
|
while ((dirent = readdir(d))) {
|
|
if (!strcmp(".", dirent->d_name) ||
|
|
!strcmp("..", dirent->d_name))
|
|
continue;
|
|
|
|
if (sscanf(dirent->d_name, "%d:%d", &major, &minor) != 2) {
|
|
log_error("_dev_cache_iterate_sysfs_for_index: %s: failed "
|
|
"to get major and minor number", dirent->d_name);
|
|
partial_failure = 1;
|
|
continue;
|
|
}
|
|
|
|
devno = MKDEV(major, minor);
|
|
if (!(dev = (struct device *) btree_lookup(_cache.devices, (uint32_t) devno)) &&
|
|
!(dev = (struct device *) btree_lookup(_cache.sysfs_only_devices, (uint32_t) devno))) {
|
|
if (!dm_device_get_name(major, minor, 1, devname, sizeof(devname)) ||
|
|
!(dev = _insert_sysfs_dev(devno, devname))) {
|
|
partial_failure = 1;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (!_index_dev_by_vgid_and_lvid(dev))
|
|
partial_failure = 1;
|
|
}
|
|
|
|
r = !partial_failure;
|
|
|
|
if (closedir(d))
|
|
log_sys_error("closedir", path);
|
|
|
|
return r;
|
|
}
|
|
|
|
int dev_cache_index_devs(void)
|
|
{
|
|
static int sysfs_has_dev_block = -1;
|
|
char path[PATH_MAX];
|
|
|
|
if (dm_snprintf(path, sizeof(path), "%sdev/block", dm_sysfs_dir()) < 0) {
|
|
log_error("dev_cache_index_devs: dm_snprintf failed.");
|
|
return 0;
|
|
}
|
|
|
|
/* Skip indexing if /sys/dev/block is not available.*/
|
|
if (sysfs_has_dev_block == -1) {
|
|
struct stat info;
|
|
if (stat(path, &info) == 0)
|
|
sysfs_has_dev_block = 1;
|
|
else {
|
|
if (errno == ENOENT) {
|
|
sysfs_has_dev_block = 0;
|
|
return 1;
|
|
}
|
|
|
|
log_sys_error("stat", path);
|
|
return 0;
|
|
}
|
|
} else if (!sysfs_has_dev_block)
|
|
return 1;
|
|
|
|
if (obtain_device_list_from_udev() &&
|
|
udev_get_library_context())
|
|
return _dev_cache_iterate_devs_for_index(); /* with udev */
|
|
|
|
return _dev_cache_iterate_sysfs_for_index(path);
|
|
}
|
|
|
|
#ifdef UDEV_SYNC_SUPPORT
|
|
|
|
static int _device_in_udev_db(const dev_t d)
|
|
{
|
|
struct udev *udev;
|
|
struct udev_device *udev_device;
|
|
|
|
if (!(udev = udev_get_library_context()))
|
|
return_0;
|
|
|
|
if ((udev_device = udev_device_new_from_devnum(udev, 'b', d))) {
|
|
udev_device_unref(udev_device);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int _insert_udev_dir(struct udev *udev, const char *dir)
|
|
{
|
|
struct udev_enumerate *udev_enum = NULL;
|
|
struct udev_list_entry *device_entry, *symlink_entry;
|
|
const char *entry_name, *node_name, *symlink_name;
|
|
struct udev_device *device;
|
|
int r = 1;
|
|
|
|
if (!(udev_enum = udev_enumerate_new(udev))) {
|
|
log_error("Failed to udev_enumerate_new.");
|
|
return 0;
|
|
}
|
|
|
|
if (udev_enumerate_add_match_subsystem(udev_enum, "block")) {
|
|
log_error("Failed to udev_enumerate_add_match_subsystem.");
|
|
goto out;
|
|
}
|
|
|
|
if (udev_enumerate_scan_devices(udev_enum)) {
|
|
log_error("Failed to udev_enumerate_scan_devices.");
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Report any missing information as "log_very_verbose" only, do not
|
|
* report it as a "warning" or "error" - the record could be removed
|
|
* by the time we ask for more info (node name, symlink name...).
|
|
* Whatever removes *any* block device in the system (even unrelated
|
|
* to our operation), we would have a warning/error on output then.
|
|
* That could be misleading. If there's really any problem with missing
|
|
* information from udev db, we can still have a look at the verbose log.
|
|
*/
|
|
udev_list_entry_foreach(device_entry, udev_enumerate_get_list_entry(udev_enum)) {
|
|
entry_name = udev_list_entry_get_name(device_entry);
|
|
|
|
if (!(device = udev_device_new_from_syspath(udev, entry_name))) {
|
|
log_very_verbose("udev failed to return a device for entry %s.",
|
|
entry_name);
|
|
continue;
|
|
}
|
|
|
|
if (!(node_name = udev_device_get_devnode(device)))
|
|
log_very_verbose("udev failed to return a device node for entry %s.",
|
|
entry_name);
|
|
else
|
|
r &= _insert(node_name, NULL, 0, 0);
|
|
|
|
udev_list_entry_foreach(symlink_entry, udev_device_get_devlinks_list_entry(device)) {
|
|
if (!(symlink_name = udev_list_entry_get_name(symlink_entry)))
|
|
log_very_verbose("udev failed to return a symlink name for entry %s.",
|
|
entry_name);
|
|
else
|
|
r &= _insert(symlink_name, NULL, 0, 0);
|
|
}
|
|
|
|
udev_device_unref(device);
|
|
}
|
|
|
|
out:
|
|
udev_enumerate_unref(udev_enum);
|
|
|
|
return r;
|
|
}
|
|
|
|
static void _insert_dirs(struct dm_list *dirs)
|
|
{
|
|
struct dir_list *dl;
|
|
struct udev *udev = NULL;
|
|
int with_udev;
|
|
|
|
with_udev = obtain_device_list_from_udev() &&
|
|
(udev = udev_get_library_context());
|
|
|
|
dm_list_iterate_items(dl, &_cache.dirs) {
|
|
if (with_udev) {
|
|
if (!_insert_udev_dir(udev, dl->dir))
|
|
log_debug_devs("%s: Failed to insert devices from "
|
|
"udev-managed directory to device "
|
|
"cache fully", dl->dir);
|
|
}
|
|
else if (!_insert_dir(dl->dir))
|
|
log_debug_devs("%s: Failed to insert devices to "
|
|
"device cache fully", dl->dir);
|
|
}
|
|
}
|
|
|
|
#else /* UDEV_SYNC_SUPPORT */
|
|
|
|
static int _device_in_udev_db(const dev_t d)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static void _insert_dirs(struct dm_list *dirs)
|
|
{
|
|
struct dir_list *dl;
|
|
|
|
dm_list_iterate_items(dl, &_cache.dirs)
|
|
_insert_dir(dl->dir);
|
|
}
|
|
|
|
#endif /* UDEV_SYNC_SUPPORT */
|
|
|
|
static int _insert(const char *path, const struct stat *info,
|
|
int rec, int check_with_udev_db)
|
|
{
|
|
struct stat tinfo;
|
|
|
|
if (!info) {
|
|
if (stat(path, &tinfo) < 0) {
|
|
log_sys_very_verbose("stat", path);
|
|
return 0;
|
|
}
|
|
info = &tinfo;
|
|
}
|
|
|
|
if (check_with_udev_db && !_device_in_udev_db(info->st_rdev)) {
|
|
log_very_verbose("%s: Not in udev db", path);
|
|
return 0;
|
|
}
|
|
|
|
if (S_ISDIR(info->st_mode)) { /* add a directory */
|
|
/* check it's not a symbolic link */
|
|
if (lstat(path, &tinfo) < 0) {
|
|
log_sys_very_verbose("lstat", path);
|
|
return 0;
|
|
}
|
|
|
|
if (S_ISLNK(tinfo.st_mode)) {
|
|
log_debug_devs("%s: Symbolic link to directory", path);
|
|
return 1;
|
|
}
|
|
|
|
if (rec && !_insert_dir(path))
|
|
return_0;
|
|
} else { /* add a device */
|
|
if (!S_ISBLK(info->st_mode))
|
|
return 1;
|
|
|
|
if (!_insert_dev(path, info->st_rdev))
|
|
return_0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
void dev_cache_scan(void)
|
|
{
|
|
log_debug_devs("Creating list of system devices.");
|
|
|
|
_cache.has_scanned = 1;
|
|
|
|
_insert_dirs(&_cache.dirs);
|
|
|
|
(void) dev_cache_index_devs();
|
|
}
|
|
|
|
int dev_cache_has_scanned(void)
|
|
{
|
|
return _cache.has_scanned;
|
|
}
|
|
|
|
static int _init_preferred_names(struct cmd_context *cmd)
|
|
{
|
|
const struct dm_config_node *cn;
|
|
const struct dm_config_value *v;
|
|
struct dm_pool *scratch = NULL;
|
|
const char **regex;
|
|
unsigned count = 0;
|
|
int i, r = 0;
|
|
|
|
_cache.preferred_names_matcher = NULL;
|
|
|
|
if (!(cn = find_config_tree_array(cmd, devices_preferred_names_CFG, NULL)) ||
|
|
cn->v->type == DM_CFG_EMPTY_ARRAY) {
|
|
log_very_verbose("devices/preferred_names %s: "
|
|
"using built-in preferences",
|
|
cn && cn->v->type == DM_CFG_EMPTY_ARRAY ? "is empty"
|
|
: "not found in config");
|
|
return 1;
|
|
}
|
|
|
|
for (v = cn->v; v; v = v->next) {
|
|
if (v->type != DM_CFG_STRING) {
|
|
log_error("preferred_names patterns must be enclosed in quotes");
|
|
return 0;
|
|
}
|
|
|
|
count++;
|
|
}
|
|
|
|
if (!(scratch = dm_pool_create("preferred device name matcher", 1024)))
|
|
return_0;
|
|
|
|
if (!(regex = dm_pool_alloc(scratch, sizeof(*regex) * count))) {
|
|
log_error("Failed to allocate preferred device name "
|
|
"pattern list.");
|
|
goto out;
|
|
}
|
|
|
|
for (v = cn->v, i = count - 1; v; v = v->next, i--) {
|
|
if (!(regex[i] = dm_pool_strdup(scratch, v->v.str))) {
|
|
log_error("Failed to allocate a preferred device name "
|
|
"pattern.");
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
if (!(_cache.preferred_names_matcher =
|
|
dm_regex_create(_cache.mem, regex, count))) {
|
|
log_error("Preferred device name pattern matcher creation failed.");
|
|
goto out;
|
|
}
|
|
|
|
r = 1;
|
|
|
|
out:
|
|
dm_pool_destroy(scratch);
|
|
|
|
return r;
|
|
}
|
|
|
|
int dev_cache_init(struct cmd_context *cmd)
|
|
{
|
|
_cache.names = NULL;
|
|
|
|
if (!(_cache.mem = dm_pool_create("dev_cache", 10 * 1024)))
|
|
return_0;
|
|
|
|
if (!(_cache.names = dm_hash_create(128)) ||
|
|
!(_cache.vgid_index = dm_hash_create(32)) ||
|
|
!(_cache.lvid_index = dm_hash_create(32))) {
|
|
dm_pool_destroy(_cache.mem);
|
|
_cache.mem = 0;
|
|
return_0;
|
|
}
|
|
|
|
if (!(_cache.devices = btree_create(_cache.mem))) {
|
|
log_error("Couldn't create binary tree for dev-cache.");
|
|
goto bad;
|
|
}
|
|
|
|
if (!(_cache.sysfs_only_devices = btree_create(_cache.mem))) {
|
|
log_error("Couldn't create binary tree for sysfs-only devices in dev cache.");
|
|
goto bad;
|
|
}
|
|
|
|
if (!(_cache.dev_dir = _strdup(cmd->dev_dir))) {
|
|
log_error("strdup dev_dir failed.");
|
|
goto bad;
|
|
}
|
|
|
|
dm_list_init(&_cache.dirs);
|
|
|
|
if (!_init_preferred_names(cmd))
|
|
goto_bad;
|
|
|
|
return 1;
|
|
|
|
bad:
|
|
dev_cache_exit();
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Returns number of devices still open.
|
|
*/
|
|
static int _check_for_open_devices(int close_immediate)
|
|
{
|
|
struct device *dev;
|
|
struct dm_hash_node *n;
|
|
int num_open = 0;
|
|
|
|
dm_hash_iterate(n, _cache.names) {
|
|
dev = (struct device *) dm_hash_get_data(_cache.names, n);
|
|
if (dev->fd >= 0) {
|
|
log_error("Device '%s' has been left open (%d remaining references).",
|
|
dev_name(dev), dev->open_count);
|
|
num_open++;
|
|
if (close_immediate && !dev_close_immediate(dev))
|
|
stack;
|
|
}
|
|
}
|
|
|
|
return num_open;
|
|
}
|
|
|
|
/*
|
|
* Returns number of devices left open.
|
|
*/
|
|
int dev_cache_check_for_open_devices(void)
|
|
{
|
|
return _check_for_open_devices(0);
|
|
}
|
|
|
|
int dev_cache_exit(void)
|
|
{
|
|
int num_open = 0;
|
|
|
|
if (_cache.names)
|
|
if ((num_open = _check_for_open_devices(1)) > 0)
|
|
log_error(INTERNAL_ERROR "%d device(s) were left open and have been closed.", num_open);
|
|
|
|
if (_cache.mem)
|
|
dm_pool_destroy(_cache.mem);
|
|
|
|
if (_cache.names)
|
|
dm_hash_destroy(_cache.names);
|
|
|
|
if (_cache.vgid_index)
|
|
dm_hash_destroy(_cache.vgid_index);
|
|
|
|
if (_cache.lvid_index)
|
|
dm_hash_destroy(_cache.lvid_index);
|
|
|
|
memset(&_cache, 0, sizeof(_cache));
|
|
|
|
return (!num_open);
|
|
}
|
|
|
|
int dev_cache_add_dir(const char *path)
|
|
{
|
|
struct dir_list *dl;
|
|
struct stat st;
|
|
|
|
if (stat(path, &st)) {
|
|
log_warn("Ignoring %s: %s.", path, strerror(errno));
|
|
/* But don't fail */
|
|
return 1;
|
|
}
|
|
|
|
if (!S_ISDIR(st.st_mode)) {
|
|
log_warn("Ignoring %s: Not a directory.", path);
|
|
return 1;
|
|
}
|
|
|
|
if (!(dl = _zalloc(sizeof(*dl) + strlen(path) + 1))) {
|
|
log_error("dir_list allocation failed");
|
|
return 0;
|
|
}
|
|
|
|
strcpy(dl->dir, path);
|
|
dm_list_add(&_cache.dirs, &dl->list);
|
|
return 1;
|
|
}
|
|
|
|
/* Check cached device name is still valid before returning it */
|
|
/* This should be a rare occurrence */
|
|
/* set quiet if the cache is expected to be out-of-date */
|
|
/* FIXME Make rest of code pass/cache struct device instead of dev_name */
|
|
const char *dev_name_confirmed(struct device *dev, int quiet)
|
|
{
|
|
struct stat buf;
|
|
const char *name;
|
|
int r;
|
|
|
|
if ((dev->flags & DEV_REGULAR))
|
|
return dev_name(dev);
|
|
|
|
while ((r = stat(name = dm_list_item(dev->aliases.n,
|
|
struct dm_str_list)->str, &buf)) ||
|
|
(buf.st_rdev != dev->dev)) {
|
|
if (r < 0) {
|
|
if (quiet)
|
|
log_sys_debug("stat", name);
|
|
else
|
|
log_sys_error("stat", name);
|
|
}
|
|
if (quiet)
|
|
log_debug_devs("Path %s no longer valid for device(%d,%d)",
|
|
name, (int) MAJOR(dev->dev),
|
|
(int) MINOR(dev->dev));
|
|
else
|
|
log_warn("Path %s no longer valid for device(%d,%d)",
|
|
name, (int) MAJOR(dev->dev),
|
|
(int) MINOR(dev->dev));
|
|
|
|
/* Remove the incorrect hash entry */
|
|
dm_hash_remove(_cache.names, name);
|
|
|
|
/* Leave list alone if there isn't an alternative name */
|
|
/* so dev_name will always find something to return. */
|
|
/* Otherwise add the name to the correct device. */
|
|
if (dm_list_size(&dev->aliases) > 1) {
|
|
dm_list_del(dev->aliases.n);
|
|
if (!r)
|
|
_insert(name, &buf, 0, obtain_device_list_from_udev());
|
|
continue;
|
|
}
|
|
|
|
/* Scanning issues this inappropriately sometimes. */
|
|
log_debug_devs("Aborting - please provide new pathname for what "
|
|
"used to be %s", name);
|
|
return NULL;
|
|
}
|
|
|
|
return dev_name(dev);
|
|
}
|
|
|
|
/* Provide a custom reason when a device is ignored */
|
|
const char *dev_cache_filtered_reason(const char *name)
|
|
{
|
|
const char *reason = "not found";
|
|
struct device *d = (struct device *) dm_hash_lookup(_cache.names, name);
|
|
|
|
if (d)
|
|
/* FIXME Record which filter caused the exclusion */
|
|
reason = "excluded by a filter";
|
|
|
|
return reason;
|
|
}
|
|
|
|
struct device *dev_cache_get(struct cmd_context *cmd, const char *name, struct dev_filter *f)
|
|
{
|
|
struct stat buf;
|
|
struct device *d = (struct device *) dm_hash_lookup(_cache.names, name);
|
|
int info_available = 0;
|
|
int ret = 1;
|
|
|
|
if (d && (d->flags & DEV_REGULAR))
|
|
return d;
|
|
|
|
/* If the entry's wrong, remove it */
|
|
if (stat(name, &buf) < 0) {
|
|
if (d)
|
|
dm_hash_remove(_cache.names, name);
|
|
log_sys_very_verbose("stat", name);
|
|
d = NULL;
|
|
} else
|
|
info_available = 1;
|
|
|
|
if (d && (buf.st_rdev != d->dev)) {
|
|
dm_hash_remove(_cache.names, name);
|
|
d = NULL;
|
|
}
|
|
|
|
if (!d) {
|
|
_insert(name, info_available ? &buf : NULL, 0, obtain_device_list_from_udev());
|
|
d = (struct device *) dm_hash_lookup(_cache.names, name);
|
|
if (!d) {
|
|
log_debug_devs("Device name not found in dev_cache repeat dev_cache_scan for %s", name);
|
|
dev_cache_scan();
|
|
d = (struct device *) dm_hash_lookup(_cache.names, name);
|
|
}
|
|
}
|
|
|
|
if (!d)
|
|
return NULL;
|
|
|
|
if (d && (d->flags & DEV_REGULAR))
|
|
return d;
|
|
|
|
if (f && !(d->flags & DEV_REGULAR)) {
|
|
ret = f->passes_filter(cmd, f, d, NULL);
|
|
|
|
if (ret == -EAGAIN) {
|
|
log_debug_devs("get device by name defer filter %s", dev_name(d));
|
|
d->flags |= DEV_FILTER_AFTER_SCAN;
|
|
ret = 1;
|
|
}
|
|
}
|
|
|
|
if (f && !(d->flags & DEV_REGULAR) && !ret)
|
|
return NULL;
|
|
|
|
return d;
|
|
}
|
|
|
|
static struct device *_dev_cache_seek_devt(dev_t dev)
|
|
{
|
|
struct device *d = NULL;
|
|
struct dm_hash_node *n = dm_hash_get_first(_cache.names);
|
|
while (n) {
|
|
d = dm_hash_get_data(_cache.names, n);
|
|
if (d->dev == dev)
|
|
return d;
|
|
n = dm_hash_get_next(_cache.names, n);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* TODO This is very inefficient. We probably want a hash table indexed by
|
|
* major:minor for keys to speed up these lookups.
|
|
*/
|
|
struct device *dev_cache_get_by_devt(struct cmd_context *cmd, dev_t dev, struct dev_filter *f, int *filtered)
|
|
{
|
|
char path[PATH_MAX];
|
|
const char *sysfs_dir;
|
|
struct stat info;
|
|
struct device *d = _dev_cache_seek_devt(dev);
|
|
int ret;
|
|
|
|
if (filtered)
|
|
*filtered = 0;
|
|
|
|
if (d && (d->flags & DEV_REGULAR))
|
|
return d;
|
|
|
|
if (!d) {
|
|
sysfs_dir = dm_sysfs_dir();
|
|
if (sysfs_dir && *sysfs_dir) {
|
|
/* First check if dev is sysfs to avoid useless scan */
|
|
if (dm_snprintf(path, sizeof(path), "%sdev/block/%d:%d",
|
|
sysfs_dir, (int)MAJOR(dev), (int)MINOR(dev)) < 0) {
|
|
log_error("dm_snprintf partition failed.");
|
|
return NULL;
|
|
}
|
|
|
|
if (lstat(path, &info)) {
|
|
log_debug("No sysfs entry for %d:%d errno %d at %s.",
|
|
(int)MAJOR(dev), (int)MINOR(dev), errno, path);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
log_debug_devs("Device num not found in dev_cache repeat dev_cache_scan for %d:%d",
|
|
(int)MAJOR(dev), (int)MINOR(dev));
|
|
dev_cache_scan();
|
|
d = _dev_cache_seek_devt(dev);
|
|
}
|
|
|
|
if (!d)
|
|
return NULL;
|
|
|
|
if (d->flags & DEV_REGULAR)
|
|
return d;
|
|
|
|
if (!f)
|
|
return d;
|
|
|
|
ret = f->passes_filter(cmd, f, d, NULL);
|
|
|
|
if (ret == -EAGAIN) {
|
|
log_debug_devs("get device by number defer filter %s", dev_name(d));
|
|
d->flags |= DEV_FILTER_AFTER_SCAN;
|
|
ret = 1;
|
|
}
|
|
|
|
if (ret)
|
|
return d;
|
|
|
|
if (filtered)
|
|
*filtered = 1;
|
|
return NULL;
|
|
}
|
|
|
|
struct dev_iter *dev_iter_create(struct dev_filter *f, int unused)
|
|
{
|
|
struct dev_iter *di = malloc(sizeof(*di));
|
|
|
|
if (!di) {
|
|
log_error("dev_iter allocation failed");
|
|
return NULL;
|
|
}
|
|
|
|
di->current = btree_first(_cache.devices);
|
|
di->filter = f;
|
|
if (di->filter)
|
|
di->filter->use_count++;
|
|
|
|
return di;
|
|
}
|
|
|
|
void dev_iter_destroy(struct dev_iter *iter)
|
|
{
|
|
if (iter->filter)
|
|
iter->filter->use_count--;
|
|
free(iter);
|
|
}
|
|
|
|
static struct device *_iter_next(struct dev_iter *iter)
|
|
{
|
|
struct device *d = btree_get_data(iter->current);
|
|
iter->current = btree_next(iter->current);
|
|
return d;
|
|
}
|
|
|
|
struct device *dev_iter_get(struct cmd_context *cmd, struct dev_iter *iter)
|
|
{
|
|
struct dev_filter *f;
|
|
int ret;
|
|
|
|
while (iter->current) {
|
|
struct device *d = _iter_next(iter);
|
|
ret = 1;
|
|
|
|
f = iter->filter;
|
|
|
|
if (f && !(d->flags & DEV_REGULAR)) {
|
|
ret = f->passes_filter(cmd, f, d, NULL);
|
|
|
|
if (ret == -EAGAIN) {
|
|
log_debug_devs("get device by iter defer filter %s", dev_name(d));
|
|
d->flags |= DEV_FILTER_AFTER_SCAN;
|
|
ret = 1;
|
|
}
|
|
}
|
|
|
|
if (!f || (d->flags & DEV_REGULAR) || ret)
|
|
return d;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
int dev_fd(struct device *dev)
|
|
{
|
|
return dev->fd;
|
|
}
|
|
|
|
const char *dev_name(const struct device *dev)
|
|
{
|
|
return (dev && dev->aliases.n) ? dm_list_item(dev->aliases.n, struct dm_str_list)->str :
|
|
unknown_device_name();
|
|
}
|
|
|
|
bool dev_cache_has_md_with_end_superblock(struct dev_types *dt)
|
|
{
|
|
struct btree_iter *iter = btree_first(_cache.devices);
|
|
struct device *dev;
|
|
|
|
while (iter) {
|
|
dev = btree_get_data(iter);
|
|
|
|
if (dev_is_md_with_end_superblock(dt, dev))
|
|
return true;
|
|
|
|
iter = btree_next(iter);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|