mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-03 05:18:29 +03:00
ef96186add
The devices file /etc/lvm/devices/system.devices is a list of devices that lvm can use. This is the default system devices file, which is specified in lvm.conf devices/devicesfile. The command option --devicesfile <filename> allows lvm to be used with a different set of devices. This allows different applications to use lvm on different sets of devices, e.g. system devices do not need to be exposed to an application using lvm on its own devices, and application devices do not need to be exposed to the system. In most cases (with limited exceptions), lvm will not read or use a device not listed in the devices file. When the devices file is used, the regex filter is not used, and the filter settings in lvm.conf are ignored. filter-deviceid is used when the devices file is enabled, and rejects any device that does not match an entry in the devices file. Set use_devicesfile=0 in lvm.conf or set --devicesfile "" on the command line to disable the use of a devices file. When disabled, lvm will see and use any device on the system that passes the regex filter (and other standard filters.) A device ID, e.g. wwid or serial number from sysfs, is a unique ID that identifies a device. The device ID is generally independent of the device content, and lvm can get the device ID without reading the device. The device ID is used in the devices file as the primary method of identifying device entries, and is also included in VG metadata for PVs. Each device_id has a device_id_type which indicates where the device_id comes from, e.g. "sys_wwid" means the device_id comes from the sysfs wwid file. Others are sys_serial, mpath_uuid, loop_file, md_uuid, devname. (devname is the device path, which is a fallback when no other proper device_id_type is available.) filter-deviceid permits lvm to use only devices on the system that have a device_id matching a devices file entry. Using the device_id, lvm can determine the set of devices to use without reading any devices, so the devices file will constrain lvm in two ways: 1. it limits the devices that lvm will read. 2. it limits the devices that lvm will use. In some uncommon cases, e.g. when devices have no unique ID and device_id has to fall back to using the devname, lvm may need to read all devices on the system to determine which ones correspond to the devices file entries. In this case, the devices file does not limit the devices that lvm reads, but it does limit the devices that lvm uses. pvcreate/vgcreate/vgextend are not constrained by the devices file, and will look outside it to find the new PV. They assign the new PV a device_id and add it to the devices file. It is also possible to explicitly add new PVs to the devices file before using them in pvcreate/etc, in which case these commands would not need to look outside the devices file for the new device. vgimportdevices VG looks at all devices on the system to find an existing VG and add its devices to the devices file. The command is not limited by an existing devices file. The command will also add device_ids to the VG metadata if the VG does not yet include device_ids. vgimportdevices -a imports devices for all accessible VGs. Since vgimportdevices does not limit itself to devices in an existing devices file, the lvm.conf regex filter applies. Adding --foreign will import devices for foreign VGs, but device_ids are not added to foreign VGs. Incomplete VGs are not imported. The lvmdevices command manages the devices file. The primary purpose is to edit the devices file, but it will read PV headers to find/check PVIDs. (It does not read, process or modify VG metadata.) lvmdevices . Displays devices file entries. lvmdevices --check . Checks devices file entries. lvmdevices --update . Updates devices file entries. lvmdevices --adddev <devname> . Adds devices_file entry (reads pv header). lvmdevices --deldev <devname> . Removes devices file entry. lvmdevices --addpvid <pvid> . Reads pv header of all devices to find <pvid>, and if found adds devices file entry. lvmdevices --delpvid <pvid> . Removes devices file entry. The vgimportclone command has a new option --importdevices that does the equivalent of vgimportdevices with the cloned devices that are being imported. The devices are "uncloned" (new vgname and pvids) while at the same time adding the devices to the devices file. This allows cloned PVs to be imported without duplicate PVs ever appearing on the system. The command option --devices <devnames> allows a specific list of devices to be exposed to the lvm command, overriding the devices file.
2069 lines
51 KiB
C
2069 lines
51 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/device/device_id.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[];
|
|
};
|
|
|
|
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->bcache_fd = -1;
|
|
dev->bcache_di = -1;
|
|
dev->read_ahead = -1;
|
|
dev->part = -1;
|
|
|
|
dev->ext.enabled = 0;
|
|
dev->ext.src = DEV_EXT_NONE;
|
|
|
|
dm_list_init(&dev->aliases);
|
|
dm_list_init(&dev->ids);
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
static 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)
|
|
{
|
|
struct device *dev;
|
|
struct dm_hash_node *n;
|
|
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);
|
|
|
|
dm_hash_iterate(n, _cache.names) {
|
|
dev = (struct device *) dm_hash_get_data(_cache.names, n);
|
|
free_dids(&dev->ids);
|
|
}
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
struct device *dev_hash_get(const char *name)
|
|
{
|
|
return (struct device *) dm_hash_lookup(_cache.names, name);
|
|
}
|
|
|
|
static void _remove_alias(struct device *dev, const char *name)
|
|
{
|
|
struct dm_str_list *strl;
|
|
|
|
dm_list_iterate_items(strl, &dev->aliases) {
|
|
if (!strcmp(strl->str, name)) {
|
|
dm_list_del(&strl->list);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Check that paths for this dev still refer to the same dev_t. This is known
|
|
* to drop invalid paths in the case where lvm deactivates an LV, which causes
|
|
* that LV path to go away, but that LV path is not removed from dev-cache (it
|
|
* probably should be). Later a new path to a different LV is added to
|
|
* dev-cache, where the new LV has the same major:minor as the previously
|
|
* deactivated LV. The new LV will find the existing struct dev, and that
|
|
* struct dev will have dev->aliases entries that refer to the name of the old
|
|
* deactivated LV. Those old paths are all invalid and are dropped here.
|
|
*/
|
|
|
|
static void _verify_aliases(struct device *dev, const char *newname)
|
|
{
|
|
struct dm_str_list *strl, *strl2;
|
|
struct stat st;
|
|
|
|
dm_list_iterate_items_safe(strl, strl2, &dev->aliases) {
|
|
/* newname was just stat'd and added by caller */
|
|
if (newname && !strcmp(strl->str, newname))
|
|
continue;
|
|
|
|
if (stat(strl->str, &st) || (st.st_rdev != dev->dev)) {
|
|
log_debug("Drop invalid path %s for %d:%d (new path %s).",
|
|
strl->str, (int)MAJOR(dev->dev), (int)MINOR(dev->dev), newname ?: "");
|
|
dm_hash_remove(_cache.names, strl->str);
|
|
dm_list_del(&strl->list);
|
|
}
|
|
}
|
|
}
|
|
|
|
struct device *dev_cache_get(struct cmd_context *cmd, const char *name, struct dev_filter *f)
|
|
{
|
|
struct device *dev = (struct device *) dm_hash_lookup(_cache.names, name);
|
|
struct stat st;
|
|
int ret;
|
|
|
|
/*
|
|
* DEV_REGULAR means that is "dev" is actually a file, not a device.
|
|
* FIXME: I don't think dev-cache is used for files any more and this
|
|
* can be dropped?
|
|
*/
|
|
if (dev && (dev->flags & DEV_REGULAR))
|
|
return dev;
|
|
|
|
/*
|
|
* The requested path is invalid, remove any dev-cache
|
|
* info for it.
|
|
*/
|
|
if (stat(name, &st)) {
|
|
if (dev) {
|
|
log_print("Device path %s is invalid for %d:%d %s.",
|
|
name, (int)MAJOR(dev->dev), (int)MINOR(dev->dev), dev_name(dev));
|
|
|
|
dm_hash_remove(_cache.names, name);
|
|
|
|
_remove_alias(dev, name);
|
|
|
|
/* Remove any other names in dev->aliases that are incorrect. */
|
|
_verify_aliases(dev, NULL);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
if (!S_ISBLK(st.st_mode)) {
|
|
log_debug("Not a block device %s.", name);
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* dev-cache has incorrect info for the requested path.
|
|
* Remove incorrect info and then add new dev-cache entry.
|
|
*/
|
|
if (dev && (st.st_rdev != dev->dev)) {
|
|
log_print("Device path %s does not match %d:%d %s.",
|
|
name, (int)MAJOR(dev->dev), (int)MINOR(dev->dev), dev_name(dev));
|
|
|
|
dm_hash_remove(_cache.names, name);
|
|
|
|
_remove_alias(dev, name);
|
|
|
|
/* Remove any other names in dev->aliases that are incorrect. */
|
|
_verify_aliases(dev, NULL);
|
|
|
|
/* Add new dev-cache entry next. */
|
|
dev = NULL;
|
|
}
|
|
|
|
/*
|
|
* Either add a new struct dev for st_rdev and name,
|
|
* or add name as a new alias for an existing struct dev
|
|
* for st_rdev.
|
|
*/
|
|
if (!dev) {
|
|
_insert_dev(name, st.st_rdev);
|
|
|
|
/* Get the struct dev that was just added. */
|
|
dev = (struct device *) dm_hash_lookup(_cache.names, name);
|
|
|
|
if (!dev) {
|
|
log_error("Failed to get device %s", name);
|
|
return NULL;
|
|
}
|
|
|
|
_verify_aliases(dev, name);
|
|
}
|
|
|
|
/*
|
|
* The caller passed a filter if they only want the dev if it
|
|
* passes filters.
|
|
*/
|
|
|
|
if (!f)
|
|
return dev;
|
|
|
|
ret = f->passes_filter(cmd, f, dev, NULL);
|
|
|
|
/*
|
|
* This might happen if this function is called before
|
|
* filters can do i/o. I don't think this will happen
|
|
* any longer and this EAGAIN case can be removed.
|
|
*/
|
|
if (ret == -EAGAIN) {
|
|
log_debug_devs("dev_cache_get filter deferred %s", dev_name(dev));
|
|
dev->flags |= DEV_FILTER_AFTER_SCAN;
|
|
ret = 1;
|
|
}
|
|
|
|
if (!ret) {
|
|
log_debug_devs("dev_cache_get filter excludes %s", dev_name(dev));
|
|
return NULL;
|
|
}
|
|
|
|
return dev;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
static int _setup_devices_list(struct cmd_context *cmd)
|
|
{
|
|
struct dm_str_list *strl;
|
|
struct dev_use *du;
|
|
|
|
/*
|
|
* For each --devices arg, add a du to cmd->use_devices.
|
|
* The du has devname is the devices arg value.
|
|
*/
|
|
|
|
dm_list_iterate_items(strl, &cmd->deviceslist) {
|
|
if (!(du = zalloc(sizeof(struct dev_use))))
|
|
return_0;
|
|
|
|
if (!(du->devname = strdup(strl->str)))
|
|
return_0;
|
|
|
|
dm_list_add(&cmd->use_devices, &du->list);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
int setup_devices_file(struct cmd_context *cmd)
|
|
{
|
|
char dirpath[PATH_MAX];
|
|
const char *filename = NULL;
|
|
struct stat st;
|
|
int rv;
|
|
|
|
if (cmd->run_by_dmeventd) {
|
|
/*
|
|
* If /etc/lvm/devices/dmeventd.devices exists, then use that.
|
|
* The optional dmeventd.devices allows the user to control
|
|
* which devices dmeventd will look at and use.
|
|
* Otherwise, disable the devices file because dmeventd should
|
|
* be able to manage LVs in any VG (i.e. LVs in a non-system
|
|
* devices file.)
|
|
*/
|
|
|
|
/* TODO: check for and use dmeventd.devices if it exists */
|
|
|
|
cmd->enable_devices_file = 0;
|
|
return 1;
|
|
}
|
|
|
|
if (cmd->devicesfile) {
|
|
/* --devicesfile <filename> or "" has been set which overrides
|
|
lvm.conf settings use_devicesfile and devicesfile. */
|
|
if (!strlen(cmd->devicesfile))
|
|
cmd->enable_devices_file = 0;
|
|
else {
|
|
cmd->enable_devices_file = 1;
|
|
filename = cmd->devicesfile;
|
|
}
|
|
} else {
|
|
if (!find_config_tree_bool(cmd, devices_use_devicesfile_CFG, NULL))
|
|
cmd->enable_devices_file = 0;
|
|
else {
|
|
cmd->enable_devices_file = 1;
|
|
filename = find_config_tree_str(cmd, devices_devicesfile_CFG, NULL);
|
|
if (!validate_name(filename)) {
|
|
log_error("Invalid devices file name from config setting \"%s\".", filename);
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!cmd->enable_devices_file)
|
|
return 1;
|
|
|
|
if (dm_snprintf(dirpath, sizeof(dirpath), "%s/devices", cmd->system_dir) < 0) {
|
|
log_error("Failed to copy devices dir path");
|
|
return 0;
|
|
}
|
|
|
|
if (stat(dirpath, &st)) {
|
|
log_debug("Creating %s.", dirpath);
|
|
dm_prepare_selinux_context(dirpath, S_IFDIR);
|
|
rv = mkdir(dirpath, 0755);
|
|
dm_prepare_selinux_context(NULL, 0);
|
|
|
|
if ((rv < 0) && stat(dirpath, &st)) {
|
|
log_error("Failed to create %s %d", dirpath, errno);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
if (dm_snprintf(cmd->devices_file_path, sizeof(cmd->devices_file_path),
|
|
"%s/devices/%s", cmd->system_dir, filename) < 0) {
|
|
log_error("Failed to copy devices file path");
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Add all system devices to dev-cache, and attempt to
|
|
* match all devices_file entries to dev-cache entries.
|
|
*/
|
|
int setup_devices(struct cmd_context *cmd)
|
|
{
|
|
int file_exists;
|
|
int lock_mode = 0;
|
|
|
|
if (cmd->enable_devices_list) {
|
|
if (!_setup_devices_list(cmd))
|
|
return_0;
|
|
goto scan;
|
|
}
|
|
|
|
if (!setup_devices_file(cmd))
|
|
return_0;
|
|
|
|
if (!cmd->enable_devices_file)
|
|
goto scan;
|
|
|
|
file_exists = devices_file_exists(cmd);
|
|
|
|
/*
|
|
* Removing the devices file is another way of disabling the use of
|
|
* a devices file, unless the command creates the devices file.
|
|
*/
|
|
if (!file_exists && !cmd->create_edit_devices_file) {
|
|
log_debug("Devices file not found, ignoring.");
|
|
cmd->enable_devices_file = 0;
|
|
goto scan;
|
|
}
|
|
|
|
/*
|
|
* Don't let pvcreate or vgcreate create a new system devices file
|
|
* unless it's specified explicitly with --devicesfile. This avoids
|
|
* a problem where a system is running with existing PVs, and is
|
|
* not using a devices file based on the fact that the system
|
|
* devices file doesn't exist. If the user simply uses pvcreate
|
|
* to create a new PV, they almost certainly do not want that to
|
|
* create a new system devices file containing the new PV and none
|
|
* of the existing PVs that the system is already using.
|
|
* However, if they use the vgimportdevices or lvmdevices command
|
|
* then they are clearly intending to use the devices file, so we
|
|
* can create it. Or, if they specify a non-system devices file
|
|
* with pvcreate/vgcreate, then they clearly want to use a devices
|
|
* file and we can create it (and creating a non-system devices file
|
|
* would not cause existing PVs to disappear from the main system.)
|
|
*
|
|
* TODO: a possible exception here is if we can see that pvcreate is
|
|
* creating the first PV on the system and can conclude the system is
|
|
* being installed. In that case we'd like the the system to be
|
|
* installed with the system devices file in place. Perhaps we
|
|
* could create the system.devices_new file but avoid renaming it
|
|
* to system.devices immediately. If label_scan does not see any
|
|
* other PVs on the system, do the rename to system.devices, and
|
|
* if label_scsan does find other PVs, then unlink system.devices_new.
|
|
*/
|
|
if (!file_exists && !cmd->devicesfile &&
|
|
(!strncmp(cmd->name, "pvcreate", 8) || !strncmp(cmd->name, "vgcreate", 8))) {
|
|
log_debug("Devices file for system not created.");
|
|
cmd->enable_devices_file = 0;
|
|
goto scan;
|
|
}
|
|
|
|
if (!file_exists && cmd->sysinit) {
|
|
cmd->enable_devices_file = 0;
|
|
goto scan;
|
|
}
|
|
|
|
if (!file_exists) {
|
|
/*
|
|
* pvcreate/vgcreate/vgimportdevices/lvmdevices-add create
|
|
* a new devices file here if it doesn't exist.
|
|
* They have the create_edit_devices_file flag set.
|
|
* First they create/lock-ex the devices file lockfile.
|
|
* Other commands will not use a devices file if none exists.
|
|
*/
|
|
lock_mode = LOCK_EX;
|
|
|
|
if (!lock_devices_file(cmd, lock_mode)) {
|
|
log_error("Failed to lock the devices file to create.");
|
|
return 0;
|
|
}
|
|
if (!devices_file_touch(cmd)) {
|
|
log_error("Failed to create the devices file.");
|
|
unlock_devices_file(cmd);
|
|
return 0;
|
|
}
|
|
log_print("Created new devices file %s", cmd->devices_file_path);
|
|
} else {
|
|
/*
|
|
* Commands that intend to edit the devices file have
|
|
* edit_devices_file or create_edit_devices_file set (create if
|
|
* they can also create a new devices file) and lock it ex
|
|
* here prior to reading. Other commands that intend to just
|
|
* read the devices file lock sh.
|
|
*/
|
|
lock_mode = (cmd->create_edit_devices_file || cmd->edit_devices_file) ? LOCK_EX : LOCK_SH;
|
|
|
|
if (!lock_devices_file(cmd, lock_mode)) {
|
|
log_error("Failed to lock the devices file.");
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Read the list of device ids that lvm can use.
|
|
* Adds a struct dev_id to cmd->use_devices for each one.
|
|
*/
|
|
if (!device_ids_read(cmd)) {
|
|
log_error("Failed to read the devices file.");
|
|
unlock_devices_file(cmd);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* When the command is editing the devices file, it acquires
|
|
* the ex lock above, will later call device_ids_write(), and
|
|
* then unlock the lock after writing the file.
|
|
* When the command is just reading the devices file, it's
|
|
* locked sh above just before reading the file, and unlocked
|
|
* here after reading.
|
|
*/
|
|
if (lock_mode == LOCK_SH)
|
|
unlock_devices_file(cmd);
|
|
|
|
scan:
|
|
/*
|
|
* Add a 'struct device' to dev-cache for each device available on the system.
|
|
* This will not open or read any devices, but may look at sysfs properties.
|
|
* This list of devs comes from looking /dev entries, or from asking libudev.
|
|
* TODO: or from /proc/partitions?
|
|
*
|
|
* TODO: dev_cache_scan() optimization: start by looking only at
|
|
* devnames listed in the devices_file, and if the device_ids for
|
|
* those all match we won't need any others.
|
|
* Exceptions: the command wants a new device for pvcreate, or
|
|
* device_ids don't match the devnames.
|
|
*/
|
|
dev_cache_scan();
|
|
|
|
/*
|
|
* Match entries from cmd->use_devices with device structs in dev-cache.
|
|
*/
|
|
device_ids_match(cmd);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* The alternative to setup_devices() when the command is interested
|
|
* in using only one PV.
|
|
*
|
|
* Add one system device to dev-cache, and attempt to
|
|
* match its dev-cache entry to a devices_file entry.
|
|
*/
|
|
int setup_device(struct cmd_context *cmd, const char *devname)
|
|
{
|
|
struct stat buf;
|
|
struct device *dev;
|
|
|
|
if (cmd->enable_devices_list) {
|
|
if (!_setup_devices_list(cmd))
|
|
return_0;
|
|
goto scan;
|
|
}
|
|
|
|
if (!setup_devices_file(cmd))
|
|
return_0;
|
|
|
|
if (!cmd->enable_devices_file)
|
|
goto scan;
|
|
|
|
if (!devices_file_exists(cmd)) {
|
|
log_debug("Devices file not found, ignoring.");
|
|
cmd->enable_devices_file = 0;
|
|
goto scan;
|
|
}
|
|
|
|
if (!lock_devices_file(cmd, LOCK_SH)) {
|
|
log_error("Failed to lock the devices file to read.");
|
|
return 0;
|
|
}
|
|
|
|
if (!device_ids_read(cmd)) {
|
|
log_error("Failed to read the devices file.");
|
|
unlock_devices_file(cmd);
|
|
return 0;
|
|
}
|
|
|
|
unlock_devices_file(cmd);
|
|
|
|
scan:
|
|
if (stat(devname, &buf) < 0) {
|
|
log_error("Cannot access device %s.", devname);
|
|
return 0;
|
|
}
|
|
|
|
if (!S_ISBLK(buf.st_mode)) {
|
|
log_error("Invaild device type %s.", devname);
|
|
return 0;
|
|
}
|
|
|
|
if (!_insert_dev(devname, buf.st_rdev))
|
|
return_0;
|
|
|
|
if (!(dev = (struct device *) dm_hash_lookup(_cache.names, devname)))
|
|
return_0;
|
|
|
|
/* Match this device to an entry in devices_file so it will not
|
|
be rejected by filter-deviceid. */
|
|
if (cmd->enable_devices_file)
|
|
device_ids_match_dev(cmd, dev);
|
|
|
|
return 1;
|
|
}
|
|
|