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

report: add dm_report_field_string_list_unsorted

This commit is contained in:
Peter Rajnoha 2014-08-25 10:04:08 +02:00
parent 993f8d1b3f
commit 02dc3c773e
3 changed files with 94 additions and 42 deletions

View File

@ -1,5 +1,6 @@
Version 1.02.89 - Version 1.02.89 -
================================= =================================
Add dm_report_field_string_list_unsorted for str. list report without sorting.
Support --deferred with dmsetup remove to defer removal of open devices. Support --deferred with dmsetup remove to defer removal of open devices.
Update dm-ioctl.h to include DM_DEFERRED_REMOVE flag. Update dm-ioctl.h to include DM_DEFERRED_REMOVE flag.
Add support for selection to match string list subset, recognize { } operator. Add support for selection to match string list subset, recognize { } operator.

View File

@ -1755,6 +1755,8 @@ int dm_report_field_string(struct dm_report *rh, struct dm_report_field *field,
const char *const *data); const char *const *data);
int dm_report_field_string_list(struct dm_report *rh, struct dm_report_field *field, int dm_report_field_string_list(struct dm_report *rh, struct dm_report_field *field,
const struct dm_list *data, const char *delimiter); const struct dm_list *data, const char *delimiter);
int dm_report_field_string_list_unsorted(struct dm_report *rh, struct dm_report_field *field,
const struct dm_list *data, const char *delimiter);
int dm_report_field_int32(struct dm_report *rh, struct dm_report_field *field, int dm_report_field_int32(struct dm_report *rh, struct dm_report_field *field,
const int32_t *data); const int32_t *data);
int dm_report_field_uint32(struct dm_report *rh, struct dm_report_field *field, int dm_report_field_uint32(struct dm_report *rh, struct dm_report_field *field,

View File

@ -283,24 +283,6 @@ int dm_report_field_string(struct dm_report *rh,
return 1; return 1;
} }
static int _str_cmp(const void *a, const void *b)
{
const char **str_a = (const char **) a;
const char **str_b = (const char **) b;
return strcmp(*str_a, *str_b);
}
struct str_list_sort_value_item {
unsigned pos;
size_t len;
};
struct str_list_sort_value {
const char *value;
struct str_list_sort_value_item *items;
};
int dm_report_field_percent(struct dm_report *rh, int dm_report_field_percent(struct dm_report *rh,
struct dm_report_field *field, struct dm_report_field *field,
const dm_percent_t *data) const dm_percent_t *data)
@ -336,15 +318,39 @@ int dm_report_field_percent(struct dm_report *rh,
return 1; return 1;
} }
int dm_report_field_string_list(struct dm_report *rh, struct str_list_sort_value_item {
struct dm_report_field *field, unsigned pos;
const struct dm_list *data, size_t len;
const char *delimiter) };
struct str_list_sort_value {
const char *value;
struct str_list_sort_value_item *items;
};
struct str_list_sort_item {
const char *str;
struct str_list_sort_value_item item;
};
static int _str_list_sort_item_cmp(const void *a, const void *b)
{
const struct str_list_sort_item *slsi_a = (const struct str_list_sort_item *) a;
const struct str_list_sort_item *slsi_b = (const struct str_list_sort_item *) b;
return strcmp(slsi_a->str, slsi_b->str);
}
static int _report_field_string_list(struct dm_report *rh,
struct dm_report_field *field,
const struct dm_list *data,
const char *delimiter,
int sort)
{ {
static const char _string_list_grow_object_failed_msg[] = "dm_report_field_string_list: dm_pool_grow_object_failed"; static const char _string_list_grow_object_failed_msg[] = "dm_report_field_string_list: dm_pool_grow_object_failed";
struct str_list_sort_value *sort_value = NULL; struct str_list_sort_value *sort_value = NULL;
unsigned int list_size, pos, i; unsigned int list_size, pos, i;
const char **arr = NULL; struct str_list_sort_item *arr = NULL;
struct dm_str_list *sl; struct dm_str_list *sl;
size_t delimiter_len, len; size_t delimiter_len, len;
void *object; void *object;
@ -392,14 +398,10 @@ int dm_report_field_string_list(struct dm_report *rh,
} }
/* more than one item - sort the list */ /* more than one item - sort the list */
if (!(arr = dm_malloc(sizeof(char *) * list_size))) { if (!(arr = dm_malloc(sizeof(struct str_list_sort_item) * list_size))) {
log_error("dm_report_field_string_list: dm_malloc failed"); log_error("dm_report_field_string_list: dm_malloc failed");
goto out; goto out;
} }
i = 0;
dm_list_iterate_items(sl, data)
arr[i++] = sl->str;
qsort(arr, i, sizeof(char *), _str_cmp);
if (!(dm_pool_begin_object(rh->mem, 256))) { if (!(dm_pool_begin_object(rh->mem, 256))) {
log_error(_string_list_grow_object_failed_msg); log_error(_string_list_grow_object_failed_msg);
@ -410,21 +412,47 @@ int dm_report_field_string_list(struct dm_report *rh,
delimiter = ","; delimiter = ",";
delimiter_len = strlen(delimiter); delimiter_len = strlen(delimiter);
/* start from 1 - the item 0 stores the list size! */ i = pos = len = 0;
for (i = 1, pos = 0; i <= list_size; i++) { dm_list_iterate_items(sl, data) {
len = strlen(arr[i-1]); arr[i].str = sl->str;
if (!dm_pool_grow_object(rh->mem, arr[i-1], len) || if (!sort) {
(i != list_size && !dm_pool_grow_object(rh->mem, delimiter, delimiter_len))) { /* sorted outpud not required - report the list as it is */
log_error(_string_list_grow_object_failed_msg); len = strlen(sl->str);
goto out; if (!dm_pool_grow_object(rh->mem, arr[i].str, len) ||
(i+1 != list_size && !dm_pool_grow_object(rh->mem, delimiter, delimiter_len))) {
log_error(_string_list_grow_object_failed_msg);
goto out;
}
arr[i].item.pos = pos;
arr[i].item.len = len;
pos = i+1 == list_size ? pos+len : pos+len+delimiter_len;
}
i++;
}
qsort(arr, i, sizeof(struct str_list_sort_item), _str_list_sort_item_cmp);
for (i = 0, pos = 0; i < list_size; i++) {
if (sort) {
/* sorted output required - report the list as sorted */
len = strlen(arr[i].str);
if (!dm_pool_grow_object(rh->mem, arr[i].str, len) ||
(i+1 != list_size && !dm_pool_grow_object(rh->mem, delimiter, delimiter_len))) {
log_error(_string_list_grow_object_failed_msg);
goto out;
}
/*
* Save position and length of the string
* element in report_string for sort_value.
* Use i+1 here since items[0] stores list size!!!
*/
sort_value->items[i+1].pos = pos;
sort_value->items[i+1].len = len;
pos = i+1 == list_size ? pos+len : pos+len+delimiter_len;
} else {
sort_value->items[i+1].pos = arr[i].item.pos;
sort_value->items[i+1].len = arr[i].item.len;
} }
/*
* save position and length of the string
* element in report_string for sort_value
*/
sort_value->items[i].pos = pos;
sort_value->items[i].len = len;
pos = i == list_size ? pos+len : pos+len+delimiter_len;
} }
if (!dm_pool_grow_object(rh->mem, "\0", 1)) { if (!dm_pool_grow_object(rh->mem, "\0", 1)) {
@ -445,6 +473,27 @@ out:
return r; return r;
} }
int dm_report_field_string_list(struct dm_report *rh,
struct dm_report_field *field,
const struct dm_list *data,
const char *delimiter)
{
return _report_field_string_list(rh, field, data, delimiter, 1);
}
int dm_report_field_string_list_unsorted(struct dm_report *rh,
struct dm_report_field *field,
const struct dm_list *data,
const char *delimiter)
{
/*
* The raw value is always sorted, just the string reported is unsorted.
* Having the raw value always sorted helps when matching selection list
* with selection criteria.
*/
return _report_field_string_list(rh, field, data, delimiter, 0);
}
int dm_report_field_int(struct dm_report *rh, int dm_report_field_int(struct dm_report *rh,
struct dm_report_field *field, const int *data) struct dm_report_field *field, const int *data)
{ {