1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-03 05:18:29 +03:00

vdo: read new sysfs path

New versions of kvdo module exposes statistics at new location:
/sys/block/dm-XXX/vdo/statistics/...

Enhance lvm2 to access this location first.
Also if the statistic info is missing - make it 'debug' level info,
so it is not failing 'lvs' command.
This commit is contained in:
Zdenek Kabelac 2021-09-09 14:59:38 +02:00
parent 89595a3665
commit e6f735d411
4 changed files with 32 additions and 25 deletions

View File

@ -1,5 +1,6 @@
Version 2.03.14 - Version 2.03.14 -
================================== ==================================
Support newer location for VDO statistics.
Add support for VDO async-unsage write policy. Add support for VDO async-unsage write policy.
Improve lvm_import_vdo script. Improve lvm_import_vdo script.
Support VDO LV with lvcreate -ky. Support VDO LV with lvcreate -ky.

View File

@ -157,6 +157,7 @@ out:
static int _get_segment_status_from_target_params(const char *target_name, static int _get_segment_status_from_target_params(const char *target_name,
const char *params, const char *params,
const struct dm_info *dminfo,
struct lv_seg_status *seg_status) struct lv_seg_status *seg_status)
{ {
const struct lv_segment *seg = seg_status->seg; const struct lv_segment *seg = seg_status->seg;
@ -216,7 +217,7 @@ static int _get_segment_status_from_target_params(const char *target_name,
return_0; return_0;
seg_status->type = SEG_STATUS_SNAPSHOT; seg_status->type = SEG_STATUS_SNAPSHOT;
} else if (segtype_is_vdo_pool(segtype)) { } else if (segtype_is_vdo_pool(segtype)) {
if (!parse_vdo_pool_status(seg_status->mem, seg->lv, params, &seg_status->vdo_pool)) if (!parse_vdo_pool_status(seg_status->mem, seg->lv, params, dminfo, &seg_status->vdo_pool))
return_0; return_0;
seg_status->type = SEG_STATUS_VDO_POOL; seg_status->type = SEG_STATUS_VDO_POOL;
} else if (segtype_is_writecache(segtype)) { } else if (segtype_is_writecache(segtype)) {
@ -320,7 +321,7 @@ static int _info_run(const char *dlid, struct dm_info *dminfo,
} while (target); } while (target);
if (!target_name || if (!target_name ||
!_get_segment_status_from_target_params(target_name, target_params, seg_status)) !_get_segment_status_from_target_params(target_name, target_params, dminfo, seg_status))
stack; stack;
} }
@ -1886,7 +1887,7 @@ int dev_manager_vdo_pool_status(struct dev_manager *dm,
goto out; goto out;
} }
if (!parse_vdo_pool_status(dm->mem, lv, params, *status)) if (!parse_vdo_pool_status(dm->mem, lv, params, &info, *status))
goto_out; goto_out;
(*status)->mem = dm->mem; (*status)->mem = dm->mem;

View File

@ -1365,7 +1365,8 @@ const char *get_vdo_write_policy_name(enum dm_vdo_write_policy policy);
uint64_t get_vdo_pool_virtual_size(const struct lv_segment *vdo_pool_seg); uint64_t get_vdo_pool_virtual_size(const struct lv_segment *vdo_pool_seg);
int update_vdo_pool_virtual_size(struct lv_segment *vdo_pool_seg); int update_vdo_pool_virtual_size(struct lv_segment *vdo_pool_seg);
int parse_vdo_pool_status(struct dm_pool *mem, const struct logical_volume *vdo_pool_lv, int parse_vdo_pool_status(struct dm_pool *mem, const struct logical_volume *vdo_pool_lv,
const char *params, struct lv_status_vdo *status); const char *params, const struct dm_info *dminfo,
struct lv_status_vdo *status);
struct logical_volume *convert_vdo_pool_lv(struct logical_volume *data_lv, struct logical_volume *convert_vdo_pool_lv(struct logical_volume *data_lv,
const struct dm_vdo_target_params *vtp, const struct dm_vdo_target_params *vtp,
uint32_t *virtual_extents, uint32_t *virtual_extents,

View File

@ -125,48 +125,56 @@ int update_vdo_pool_virtual_size(struct lv_segment *vdo_pool_seg)
return 1; return 1;
} }
static int _sysfs_get_kvdo_value(const char *dm_name, const char *vdo_param, uint64_t *value) static int _sysfs_get_kvdo_value(const char *dm_name, const struct dm_info *dminfo,
const char *vdo_param, uint64_t *value)
{ {
char path[PATH_MAX]; char path[PATH_MAX];
char temp[64]; char temp[64];
int fd, size, r = 0; int fd, size, r = 0;
if (dm_snprintf(path, sizeof(path), "%skvdo/%s/%s", if (dm_snprintf(path, sizeof(path), "%s/block/dm-%d/vdo/%s",
dm_sysfs_dir(), dm_name, vdo_param) < 0) { dm_sysfs_dir(), dminfo->minor, vdo_param) < 0) {
log_error("Failed to build kmod path."); log_debug("Failed to build kvdo path.");
return 0;
}
if ((fd = open(path, O_RDONLY)) < 0) {
/* try with older location */
if (dm_snprintf(path, sizeof(path), "%skvdo/%s/%s",
dm_sysfs_dir(), dm_name, vdo_param) < 0) {
log_debug("Failed to build kvdo path.");
return 0; return 0;
} }
if ((fd = open(path, O_RDONLY)) < 0) { if ((fd = open(path, O_RDONLY)) < 0) {
if (errno != ENOENT)
log_sys_error("open", path);
else
log_sys_debug("open", path); log_sys_debug("open", path);
goto bad; goto bad;
} }
}
if ((size = read(fd, temp, sizeof(temp) - 1)) < 0) { if ((size = read(fd, temp, sizeof(temp) - 1)) < 0) {
log_sys_error("read", path); log_sys_debug("read", path);
goto bad; goto bad;
} }
temp[size] = 0; temp[size] = 0;
errno = 0; errno = 0;
*value = strtoll(temp, NULL, 0); *value = strtoll(temp, NULL, 0);
if (errno) { if (errno) {
log_sys_error("strtool", path); log_sys_debug("strtool", path);
goto bad; goto bad;
} }
r = 1; r = 1;
bad: bad:
if (fd >= 0 && close(fd)) if (fd >= 0 && close(fd))
log_sys_error("close", path); log_sys_debug("close", path);
return r; return r;
} }
int parse_vdo_pool_status(struct dm_pool *mem, const struct logical_volume *vdo_pool_lv, int parse_vdo_pool_status(struct dm_pool *mem, const struct logical_volume *vdo_pool_lv,
const char *params, struct lv_status_vdo *status) const char *params, const struct dm_info *dminfo,
struct lv_status_vdo *status)
{ {
struct dm_vdo_status_parse_result result; struct dm_vdo_status_parse_result result;
char *dm_name; char *dm_name;
@ -190,15 +198,11 @@ int parse_vdo_pool_status(struct dm_pool *mem, const struct logical_volume *vdo_
status->vdo = result.status; status->vdo = result.status;
if (result.status->operating_mode == DM_VDO_MODE_NORMAL) { if ((result.status->operating_mode == DM_VDO_MODE_NORMAL) &&
if (!_sysfs_get_kvdo_value(dm_name, "statistics/data_blocks_used", _sysfs_get_kvdo_value(dm_name, dminfo, "statistics/data_blocks_used",
&status->data_blocks_used)) &status->data_blocks_used) &&
return_0; _sysfs_get_kvdo_value(dm_name, dminfo, "statistics/logical_blocks_used",
&status->logical_blocks_used)) {
if (!_sysfs_get_kvdo_value(dm_name, "statistics/logical_blocks_used",
&status->logical_blocks_used))
return_0;
status->usage = dm_make_percent(result.status->used_blocks, status->usage = dm_make_percent(result.status->used_blocks,
result.status->total_blocks); result.status->total_blocks);
status->saving = dm_make_percent(status->logical_blocks_used - status->data_blocks_used, status->saving = dm_make_percent(status->logical_blocks_used - status->data_blocks_used,