diff options
author | Mikulas Patocka <mpatocka@redhat.com> | 2013-03-01 17:45:44 -0500 |
---|---|---|
committer | Alasdair G Kergon <agk@redhat.com> | 2013-03-01 17:45:44 -0500 |
commit | fd7c092e711ebab55b2688d3859d95dfd0301f73 (patch) | |
tree | 3cc99f96f4a2de8e22347feb86b0ecd5dd7200d0 | |
parent | 16245bdc9d3e22d1460341a655c8b5288953bc14 (diff) |
dm: fix truncated status strings
Avoid returning a truncated table or status string instead of setting
the DM_BUFFER_FULL_FLAG when the last target of a table fills the
buffer.
When processing a table or status request, the function retrieve_status
calls ti->type->status. If ti->type->status returns non-zero,
retrieve_status assumes that the buffer overflowed and sets
DM_BUFFER_FULL_FLAG.
However, targets don't return non-zero values from their status method
on overflow. Most targets returns always zero.
If a buffer overflow happens in a target that is not the last in the
table, it gets noticed during the next iteration of the loop in
retrieve_status; but if a buffer overflow happens in the last target, it
goes unnoticed and erroneously truncated data is returned.
In the current code, the targets behave in the following way:
* dm-crypt returns -ENOMEM if there is not enough space to store the
key, but it returns 0 on all other overflows.
* dm-thin returns errors from the status method if a disk error happened.
This is incorrect because retrieve_status doesn't check the error
code, it assumes that all non-zero values mean buffer overflow.
* all the other targets always return 0.
This patch changes the ti->type->status function to return void (because
most targets don't use the return code). Overflow is detected in
retrieve_status: if the status method fills up the remaining space
completely, it is assumed that buffer overflow happened.
Cc: stable@vger.kernel.org
Signed-off-by: Mikulas Patocka <mpatocka@redhat.com>
Signed-off-by: Alasdair G Kergon <agk@redhat.com>
-rw-r--r-- | drivers/md/dm-crypt.c | 39 | ||||
-rw-r--r-- | drivers/md/dm-delay.c | 8 | ||||
-rw-r--r-- | drivers/md/dm-flakey.c | 7 | ||||
-rw-r--r-- | drivers/md/dm-ioctl.c | 14 | ||||
-rw-r--r-- | drivers/md/dm-linear.c | 7 | ||||
-rw-r--r-- | drivers/md/dm-mpath.c | 8 | ||||
-rw-r--r-- | drivers/md/dm-raid.c | 8 | ||||
-rw-r--r-- | drivers/md/dm-raid1.c | 8 | ||||
-rw-r--r-- | drivers/md/dm-snap.c | 16 | ||||
-rw-r--r-- | drivers/md/dm-stripe.c | 7 | ||||
-rw-r--r-- | drivers/md/dm-thin.c | 80 | ||||
-rw-r--r-- | drivers/md/dm-verity.c | 8 | ||||
-rw-r--r-- | include/linux/device-mapper.h | 4 |
13 files changed, 99 insertions, 115 deletions
diff --git a/drivers/md/dm-crypt.c b/drivers/md/dm-crypt.c index f7369f9d8595..2ae151e59c0c 100644 --- a/drivers/md/dm-crypt.c +++ b/drivers/md/dm-crypt.c | |||
@@ -1234,20 +1234,6 @@ static int crypt_decode_key(u8 *key, char *hex, unsigned int size) | |||
1234 | return 0; | 1234 | return 0; |
1235 | } | 1235 | } |
1236 | 1236 | ||
1237 | /* | ||
1238 | * Encode key into its hex representation | ||
1239 | */ | ||
1240 | static void crypt_encode_key(char *hex, u8 *key, unsigned int size) | ||
1241 | { | ||
1242 | unsigned int i; | ||
1243 | |||
1244 | for (i = 0; i < size; i++) { | ||
1245 | sprintf(hex, "%02x", *key); | ||
1246 | hex += 2; | ||
1247 | key++; | ||
1248 | } | ||
1249 | } | ||
1250 | |||
1251 | static void crypt_free_tfms(struct crypt_config *cc) | 1237 | static void crypt_free_tfms(struct crypt_config *cc) |
1252 | { | 1238 | { |
1253 | unsigned i; | 1239 | unsigned i; |
@@ -1717,11 +1703,11 @@ static int crypt_map(struct dm_target *ti, struct bio *bio) | |||
1717 | return DM_MAPIO_SUBMITTED; | 1703 | return DM_MAPIO_SUBMITTED; |
1718 | } | 1704 | } |
1719 | 1705 | ||
1720 | static int crypt_status(struct dm_target *ti, status_type_t type, | 1706 | static void crypt_status(struct dm_target *ti, status_type_t type, |
1721 | unsigned status_flags, char *result, unsigned maxlen) | 1707 | unsigned status_flags, char *result, unsigned maxlen) |
1722 | { | 1708 | { |
1723 | struct crypt_config *cc = ti->private; | 1709 | struct crypt_config *cc = ti->private; |
1724 | unsigned int sz = 0; | 1710 | unsigned i, sz = 0; |
1725 | 1711 | ||
1726 | switch (type) { | 1712 | switch (type) { |
1727 | case STATUSTYPE_INFO: | 1713 | case STATUSTYPE_INFO: |
@@ -1731,17 +1717,11 @@ static int crypt_status(struct dm_target *ti, status_type_t type, | |||
1731 | case STATUSTYPE_TABLE: | 1717 | case STATUSTYPE_TABLE: |
1732 | DMEMIT("%s ", cc->cipher_string); | 1718 | DMEMIT("%s ", cc->cipher_string); |
1733 | 1719 | ||
1734 | if (cc->key_size > 0) { | 1720 | if (cc->key_size > 0) |
1735 | if ((maxlen - sz) < ((cc->key_size << 1) + 1)) | 1721 | for (i = 0; i < cc->key_size; i++) |
1736 | return -ENOMEM; | 1722 | DMEMIT("%02x", cc->key[i]); |
1737 | 1723 | else | |
1738 | crypt_encode_key(result + sz, cc->key, cc->key_size); | 1724 | DMEMIT("-"); |
1739 | sz += cc->key_size << 1; | ||
1740 | } else { | ||
1741 | if (sz >= maxlen) | ||
1742 | return -ENOMEM; | ||
1743 | result[sz++] = '-'; | ||
1744 | } | ||
1745 | 1725 | ||
1746 | DMEMIT(" %llu %s %llu", (unsigned long long)cc->iv_offset, | 1726 | DMEMIT(" %llu %s %llu", (unsigned long long)cc->iv_offset, |
1747 | cc->dev->name, (unsigned long long)cc->start); | 1727 | cc->dev->name, (unsigned long long)cc->start); |
@@ -1751,7 +1731,6 @@ static int crypt_status(struct dm_target *ti, status_type_t type, | |||
1751 | 1731 | ||
1752 | break; | 1732 | break; |
1753 | } | 1733 | } |
1754 | return 0; | ||
1755 | } | 1734 | } |
1756 | 1735 | ||
1757 | static void crypt_postsuspend(struct dm_target *ti) | 1736 | static void crypt_postsuspend(struct dm_target *ti) |
@@ -1845,7 +1824,7 @@ static int crypt_iterate_devices(struct dm_target *ti, | |||
1845 | 1824 | ||
1846 | static struct target_type crypt_target = { | 1825 | static struct target_type crypt_target = { |
1847 | .name = "crypt", | 1826 | .name = "crypt", |
1848 | .version = {1, 12, 0}, | 1827 | .version = {1, 12, 1}, |
1849 | .module = THIS_MODULE, | 1828 | .module = THIS_MODULE, |
1850 | .ctr = crypt_ctr, | 1829 | .ctr = crypt_ctr, |
1851 | .dtr = crypt_dtr, | 1830 | .dtr = crypt_dtr, |
diff --git a/drivers/md/dm-delay.c b/drivers/md/dm-delay.c index cc1bd048acb2..c0d03b006e40 100644 --- a/drivers/md/dm-delay.c +++ b/drivers/md/dm-delay.c | |||
@@ -293,8 +293,8 @@ static int delay_map(struct dm_target *ti, struct bio *bio) | |||
293 | return delay_bio(dc, dc->read_delay, bio); | 293 | return delay_bio(dc, dc->read_delay, bio); |
294 | } | 294 | } |
295 | 295 | ||
296 | static int delay_status(struct dm_target *ti, status_type_t type, | 296 | static void delay_status(struct dm_target *ti, status_type_t type, |
297 | unsigned status_flags, char *result, unsigned maxlen) | 297 | unsigned status_flags, char *result, unsigned maxlen) |
298 | { | 298 | { |
299 | struct delay_c *dc = ti->private; | 299 | struct delay_c *dc = ti->private; |
300 | int sz = 0; | 300 | int sz = 0; |
@@ -314,8 +314,6 @@ static int delay_status(struct dm_target *ti, status_type_t type, | |||
314 | dc->write_delay); | 314 | dc->write_delay); |
315 | break; | 315 | break; |
316 | } | 316 | } |
317 | |||
318 | return 0; | ||
319 | } | 317 | } |
320 | 318 | ||
321 | static int delay_iterate_devices(struct dm_target *ti, | 319 | static int delay_iterate_devices(struct dm_target *ti, |
@@ -337,7 +335,7 @@ out: | |||
337 | 335 | ||
338 | static struct target_type delay_target = { | 336 | static struct target_type delay_target = { |
339 | .name = "delay", | 337 | .name = "delay", |
340 | .version = {1, 2, 0}, | 338 | .version = {1, 2, 1}, |
341 | .module = THIS_MODULE, | 339 | .module = THIS_MODULE, |
342 | .ctr = delay_ctr, | 340 | .ctr = delay_ctr, |
343 | .dtr = delay_dtr, | 341 | .dtr = delay_dtr, |
diff --git a/drivers/md/dm-flakey.c b/drivers/md/dm-flakey.c index 9721f2ffb1a2..5d6c04cceee0 100644 --- a/drivers/md/dm-flakey.c +++ b/drivers/md/dm-flakey.c | |||
@@ -337,8 +337,8 @@ static int flakey_end_io(struct dm_target *ti, struct bio *bio, int error) | |||
337 | return error; | 337 | return error; |
338 | } | 338 | } |
339 | 339 | ||
340 | static int flakey_status(struct dm_target *ti, status_type_t type, | 340 | static void flakey_status(struct dm_target *ti, status_type_t type, |
341 | unsigned status_flags, char *result, unsigned maxlen) | 341 | unsigned status_flags, char *result, unsigned maxlen) |
342 | { | 342 | { |
343 | unsigned sz = 0; | 343 | unsigned sz = 0; |
344 | struct flakey_c *fc = ti->private; | 344 | struct flakey_c *fc = ti->private; |
@@ -368,7 +368,6 @@ static int flakey_status(struct dm_target *ti, status_type_t type, | |||
368 | 368 | ||
369 | break; | 369 | break; |
370 | } | 370 | } |
371 | return 0; | ||
372 | } | 371 | } |
373 | 372 | ||
374 | static int flakey_ioctl(struct dm_target *ti, unsigned int cmd, unsigned long arg) | 373 | static int flakey_ioctl(struct dm_target *ti, unsigned int cmd, unsigned long arg) |
@@ -411,7 +410,7 @@ static int flakey_iterate_devices(struct dm_target *ti, iterate_devices_callout_ | |||
411 | 410 | ||
412 | static struct target_type flakey_target = { | 411 | static struct target_type flakey_target = { |
413 | .name = "flakey", | 412 | .name = "flakey", |
414 | .version = {1, 3, 0}, | 413 | .version = {1, 3, 1}, |
415 | .module = THIS_MODULE, | 414 | .module = THIS_MODULE, |
416 | .ctr = flakey_ctr, | 415 | .ctr = flakey_ctr, |
417 | .dtr = flakey_dtr, | 416 | .dtr = flakey_dtr, |
diff --git a/drivers/md/dm-ioctl.c b/drivers/md/dm-ioctl.c index 0666b5d14b88..eee353da3742 100644 --- a/drivers/md/dm-ioctl.c +++ b/drivers/md/dm-ioctl.c | |||
@@ -1067,6 +1067,7 @@ static void retrieve_status(struct dm_table *table, | |||
1067 | num_targets = dm_table_get_num_targets(table); | 1067 | num_targets = dm_table_get_num_targets(table); |
1068 | for (i = 0; i < num_targets; i++) { | 1068 | for (i = 0; i < num_targets; i++) { |
1069 | struct dm_target *ti = dm_table_get_target(table, i); | 1069 | struct dm_target *ti = dm_table_get_target(table, i); |
1070 | size_t l; | ||
1070 | 1071 | ||
1071 | remaining = len - (outptr - outbuf); | 1072 | remaining = len - (outptr - outbuf); |
1072 | if (remaining <= sizeof(struct dm_target_spec)) { | 1073 | if (remaining <= sizeof(struct dm_target_spec)) { |
@@ -1093,14 +1094,17 @@ static void retrieve_status(struct dm_table *table, | |||
1093 | if (ti->type->status) { | 1094 | if (ti->type->status) { |
1094 | if (param->flags & DM_NOFLUSH_FLAG) | 1095 | if (param->flags & DM_NOFLUSH_FLAG) |
1095 | status_flags |= DM_STATUS_NOFLUSH_FLAG; | 1096 | status_flags |= DM_STATUS_NOFLUSH_FLAG; |
1096 | if (ti->type->status(ti, type, status_flags, outptr, remaining)) { | 1097 | ti->type->status(ti, type, status_flags, outptr, remaining); |
1097 | param->flags |= DM_BUFFER_FULL_FLAG; | ||
1098 | break; | ||
1099 | } | ||
1100 | } else | 1098 | } else |
1101 | outptr[0] = '\0'; | 1099 | outptr[0] = '\0'; |
1102 | 1100 | ||
1103 | outptr += strlen(outptr) + 1; | 1101 | l = strlen(outptr) + 1; |
1102 | if (l == remaining) { | ||
1103 | param->flags |= DM_BUFFER_FULL_FLAG; | ||
1104 | break; | ||
1105 | } | ||
1106 | |||
1107 | outptr += l; | ||
1104 | used = param->data_start + (outptr - outbuf); | 1108 | used = param->data_start + (outptr - outbuf); |
1105 | 1109 | ||
1106 | outptr = align_ptr(outptr); | 1110 | outptr = align_ptr(outptr); |
diff --git a/drivers/md/dm-linear.c b/drivers/md/dm-linear.c index 328cad5617ab..5be301c1e809 100644 --- a/drivers/md/dm-linear.c +++ b/drivers/md/dm-linear.c | |||
@@ -95,8 +95,8 @@ static int linear_map(struct dm_target *ti, struct bio *bio) | |||
95 | return DM_MAPIO_REMAPPED; | 95 | return DM_MAPIO_REMAPPED; |
96 | } | 96 | } |
97 | 97 | ||
98 | static int linear_status(struct dm_target *ti, status_type_t type, | 98 | static void linear_status(struct dm_target *ti, status_type_t type, |
99 | unsigned status_flags, char *result, unsigned maxlen) | 99 | unsigned status_flags, char *result, unsigned maxlen) |
100 | { | 100 | { |
101 | struct linear_c *lc = (struct linear_c *) ti->private; | 101 | struct linear_c *lc = (struct linear_c *) ti->private; |
102 | 102 | ||
@@ -110,7 +110,6 @@ static int linear_status(struct dm_target *ti, status_type_t type, | |||
110 | (unsigned long long)lc->start); | 110 | (unsigned long long)lc->start); |
111 | break; | 111 | break; |
112 | } | 112 | } |
113 | return 0; | ||
114 | } | 113 | } |
115 | 114 | ||
116 | static int linear_ioctl(struct dm_target *ti, unsigned int cmd, | 115 | static int linear_ioctl(struct dm_target *ti, unsigned int cmd, |
@@ -155,7 +154,7 @@ static int linear_iterate_devices(struct dm_target *ti, | |||
155 | 154 | ||
156 | static struct target_type linear_target = { | 155 | static struct target_type linear_target = { |
157 | .name = "linear", | 156 | .name = "linear", |
158 | .version = {1, 2, 0}, | 157 | .version = {1, 2, 1}, |
159 | .module = THIS_MODULE, | 158 | .module = THIS_MODULE, |
160 | .ctr = linear_ctr, | 159 | .ctr = linear_ctr, |
161 | .dtr = linear_dtr, | 160 | .dtr = linear_dtr, |
diff --git a/drivers/md/dm-mpath.c b/drivers/md/dm-mpath.c index 573bd04591bf..d267bb5705e9 100644 --- a/drivers/md/dm-mpath.c +++ b/drivers/md/dm-mpath.c | |||
@@ -1378,8 +1378,8 @@ static void multipath_resume(struct dm_target *ti) | |||
1378 | * [priority selector-name num_ps_args [ps_args]* | 1378 | * [priority selector-name num_ps_args [ps_args]* |
1379 | * num_paths num_selector_args [path_dev [selector_args]* ]+ ]+ | 1379 | * num_paths num_selector_args [path_dev [selector_args]* ]+ ]+ |
1380 | */ | 1380 | */ |
1381 | static int multipath_status(struct dm_target *ti, status_type_t type, | 1381 | static void multipath_status(struct dm_target *ti, status_type_t type, |
1382 | unsigned status_flags, char *result, unsigned maxlen) | 1382 | unsigned status_flags, char *result, unsigned maxlen) |
1383 | { | 1383 | { |
1384 | int sz = 0; | 1384 | int sz = 0; |
1385 | unsigned long flags; | 1385 | unsigned long flags; |
@@ -1485,8 +1485,6 @@ static int multipath_status(struct dm_target *ti, status_type_t type, | |||
1485 | } | 1485 | } |
1486 | 1486 | ||
1487 | spin_unlock_irqrestore(&m->lock, flags); | 1487 | spin_unlock_irqrestore(&m->lock, flags); |
1488 | |||
1489 | return 0; | ||
1490 | } | 1488 | } |
1491 | 1489 | ||
1492 | static int multipath_message(struct dm_target *ti, unsigned argc, char **argv) | 1490 | static int multipath_message(struct dm_target *ti, unsigned argc, char **argv) |
@@ -1695,7 +1693,7 @@ out: | |||
1695 | *---------------------------------------------------------------*/ | 1693 | *---------------------------------------------------------------*/ |
1696 | static struct target_type multipath_target = { | 1694 | static struct target_type multipath_target = { |
1697 | .name = "multipath", | 1695 | .name = "multipath", |
1698 | .version = {1, 5, 0}, | 1696 | .version = {1, 5, 1}, |
1699 | .module = THIS_MODULE, | 1697 | .module = THIS_MODULE, |
1700 | .ctr = multipath_ctr, | 1698 | .ctr = multipath_ctr, |
1701 | .dtr = multipath_dtr, | 1699 | .dtr = multipath_dtr, |
diff --git a/drivers/md/dm-raid.c b/drivers/md/dm-raid.c index 9e58dbd8d8cb..5a578d89da2d 100644 --- a/drivers/md/dm-raid.c +++ b/drivers/md/dm-raid.c | |||
@@ -1201,8 +1201,8 @@ static int raid_map(struct dm_target *ti, struct bio *bio) | |||
1201 | return DM_MAPIO_SUBMITTED; | 1201 | return DM_MAPIO_SUBMITTED; |
1202 | } | 1202 | } |
1203 | 1203 | ||
1204 | static int raid_status(struct dm_target *ti, status_type_t type, | 1204 | static void raid_status(struct dm_target *ti, status_type_t type, |
1205 | unsigned status_flags, char *result, unsigned maxlen) | 1205 | unsigned status_flags, char *result, unsigned maxlen) |
1206 | { | 1206 | { |
1207 | struct raid_set *rs = ti->private; | 1207 | struct raid_set *rs = ti->private; |
1208 | unsigned raid_param_cnt = 1; /* at least 1 for chunksize */ | 1208 | unsigned raid_param_cnt = 1; /* at least 1 for chunksize */ |
@@ -1344,8 +1344,6 @@ static int raid_status(struct dm_target *ti, status_type_t type, | |||
1344 | DMEMIT(" -"); | 1344 | DMEMIT(" -"); |
1345 | } | 1345 | } |
1346 | } | 1346 | } |
1347 | |||
1348 | return 0; | ||
1349 | } | 1347 | } |
1350 | 1348 | ||
1351 | static int raid_iterate_devices(struct dm_target *ti, iterate_devices_callout_fn fn, void *data) | 1349 | static int raid_iterate_devices(struct dm_target *ti, iterate_devices_callout_fn fn, void *data) |
@@ -1405,7 +1403,7 @@ static void raid_resume(struct dm_target *ti) | |||
1405 | 1403 | ||
1406 | static struct target_type raid_target = { | 1404 | static struct target_type raid_target = { |
1407 | .name = "raid", | 1405 | .name = "raid", |
1408 | .version = {1, 4, 1}, | 1406 | .version = {1, 4, 2}, |
1409 | .module = THIS_MODULE, | 1407 | .module = THIS_MODULE, |
1410 | .ctr = raid_ctr, | 1408 | .ctr = raid_ctr, |
1411 | .dtr = raid_dtr, | 1409 | .dtr = raid_dtr, |
diff --git a/drivers/md/dm-raid1.c b/drivers/md/dm-raid1.c index fa519185ebba..7f2419099b50 100644 --- a/drivers/md/dm-raid1.c +++ b/drivers/md/dm-raid1.c | |||
@@ -1347,8 +1347,8 @@ static char device_status_char(struct mirror *m) | |||
1347 | } | 1347 | } |
1348 | 1348 | ||
1349 | 1349 | ||
1350 | static int mirror_status(struct dm_target *ti, status_type_t type, | 1350 | static void mirror_status(struct dm_target *ti, status_type_t type, |
1351 | unsigned status_flags, char *result, unsigned maxlen) | 1351 | unsigned status_flags, char *result, unsigned maxlen) |
1352 | { | 1352 | { |
1353 | unsigned int m, sz = 0; | 1353 | unsigned int m, sz = 0; |
1354 | struct mirror_set *ms = (struct mirror_set *) ti->private; | 1354 | struct mirror_set *ms = (struct mirror_set *) ti->private; |
@@ -1383,8 +1383,6 @@ static int mirror_status(struct dm_target *ti, status_type_t type, | |||
1383 | if (ms->features & DM_RAID1_HANDLE_ERRORS) | 1383 | if (ms->features & DM_RAID1_HANDLE_ERRORS) |
1384 | DMEMIT(" 1 handle_errors"); | 1384 | DMEMIT(" 1 handle_errors"); |
1385 | } | 1385 | } |
1386 | |||
1387 | return 0; | ||
1388 | } | 1386 | } |
1389 | 1387 | ||
1390 | static int mirror_iterate_devices(struct dm_target *ti, | 1388 | static int mirror_iterate_devices(struct dm_target *ti, |
@@ -1403,7 +1401,7 @@ static int mirror_iterate_devices(struct dm_target *ti, | |||
1403 | 1401 | ||
1404 | static struct target_type mirror_target = { | 1402 | static struct target_type mirror_target = { |
1405 | .name = "mirror", | 1403 | .name = "mirror", |
1406 | .version = {1, 13, 1}, | 1404 | .version = {1, 13, 2}, |
1407 | .module = THIS_MODULE, | 1405 | .module = THIS_MODULE, |
1408 | .ctr = mirror_ctr, | 1406 | .ctr = mirror_ctr, |
1409 | .dtr = mirror_dtr, | 1407 | .dtr = mirror_dtr, |
diff --git a/drivers/md/dm-snap.c b/drivers/md/dm-snap.c index 10079e07edf4..6e45e3774eab 100644 --- a/drivers/md/dm-snap.c +++ b/drivers/md/dm-snap.c | |||
@@ -1836,8 +1836,8 @@ static void snapshot_merge_resume(struct dm_target *ti) | |||
1836 | start_merge(s); | 1836 | start_merge(s); |
1837 | } | 1837 | } |
1838 | 1838 | ||
1839 | static int snapshot_status(struct dm_target *ti, status_type_t type, | 1839 | static void snapshot_status(struct dm_target *ti, status_type_t type, |
1840 | unsigned status_flags, char *result, unsigned maxlen) | 1840 | unsigned status_flags, char *result, unsigned maxlen) |
1841 | { | 1841 | { |
1842 | unsigned sz = 0; | 1842 | unsigned sz = 0; |
1843 | struct dm_snapshot *snap = ti->private; | 1843 | struct dm_snapshot *snap = ti->private; |
@@ -1883,8 +1883,6 @@ static int snapshot_status(struct dm_target *ti, status_type_t type, | |||
1883 | maxlen - sz); | 1883 | maxlen - sz); |
1884 | break; | 1884 | break; |
1885 | } | 1885 | } |
1886 | |||
1887 | return 0; | ||
1888 | } | 1886 | } |
1889 | 1887 | ||
1890 | static int snapshot_iterate_devices(struct dm_target *ti, | 1888 | static int snapshot_iterate_devices(struct dm_target *ti, |
@@ -2138,8 +2136,8 @@ static void origin_resume(struct dm_target *ti) | |||
2138 | ti->max_io_len = get_origin_minimum_chunksize(dev->bdev); | 2136 | ti->max_io_len = get_origin_minimum_chunksize(dev->bdev); |
2139 | } | 2137 | } |
2140 | 2138 | ||
2141 | static int origin_status(struct dm_target *ti, status_type_t type, | 2139 | static void origin_status(struct dm_target *ti, status_type_t type, |
2142 | unsigned status_flags, char *result, unsigned maxlen) | 2140 | unsigned status_flags, char *result, unsigned maxlen) |
2143 | { | 2141 | { |
2144 | struct dm_dev *dev = ti->private; | 2142 | struct dm_dev *dev = ti->private; |
2145 | 2143 | ||
@@ -2152,8 +2150,6 @@ static int origin_status(struct dm_target *ti, status_type_t type, | |||
2152 | snprintf(result, maxlen, "%s", dev->name); | 2150 | snprintf(result, maxlen, "%s", dev->name); |
2153 | break; | 2151 | break; |
2154 | } | 2152 | } |
2155 | |||
2156 | return 0; | ||
2157 | } | 2153 | } |
2158 | 2154 | ||
2159 | static int origin_merge(struct dm_target *ti, struct bvec_merge_data *bvm, | 2155 | static int origin_merge(struct dm_target *ti, struct bvec_merge_data *bvm, |
@@ -2180,7 +2176,7 @@ static int origin_iterate_devices(struct dm_target *ti, | |||
2180 | 2176 | ||
2181 | static struct target_type origin_target = { | 2177 | static struct target_type origin_target = { |
2182 | .name = "snapshot-origin", | 2178 | .name = "snapshot-origin", |
2183 | .version = {1, 8, 0}, | 2179 | .version = {1, 8, 1}, |
2184 | .module = THIS_MODULE, | 2180 | .module = THIS_MODULE, |
2185 | .ctr = origin_ctr, | 2181 | .ctr = origin_ctr, |
2186 | .dtr = origin_dtr, | 2182 | .dtr = origin_dtr, |
@@ -2193,7 +2189,7 @@ static struct target_type origin_target = { | |||
2193 | 2189 | ||
2194 | static struct target_type snapshot_target = { | 2190 | static struct target_type snapshot_target = { |
2195 | .name = "snapshot", | 2191 | .name = "snapshot", |
2196 | .version = {1, 11, 0}, | 2192 | .version = {1, 11, 1}, |
2197 | .module = THIS_MODULE, | 2193 | .module = THIS_MODULE, |
2198 | .ctr = snapshot_ctr, | 2194 | .ctr = snapshot_ctr, |
2199 | .dtr = snapshot_dtr, | 2195 | .dtr = snapshot_dtr, |
diff --git a/drivers/md/dm-stripe.c b/drivers/md/dm-stripe.c index c89cde86d400..aaecefa63935 100644 --- a/drivers/md/dm-stripe.c +++ b/drivers/md/dm-stripe.c | |||
@@ -312,8 +312,8 @@ static int stripe_map(struct dm_target *ti, struct bio *bio) | |||
312 | * | 312 | * |
313 | */ | 313 | */ |
314 | 314 | ||
315 | static int stripe_status(struct dm_target *ti, status_type_t type, | 315 | static void stripe_status(struct dm_target *ti, status_type_t type, |
316 | unsigned status_flags, char *result, unsigned maxlen) | 316 | unsigned status_flags, char *result, unsigned maxlen) |
317 | { | 317 | { |
318 | struct stripe_c *sc = (struct stripe_c *) ti->private; | 318 | struct stripe_c *sc = (struct stripe_c *) ti->private; |
319 | char buffer[sc->stripes + 1]; | 319 | char buffer[sc->stripes + 1]; |
@@ -340,7 +340,6 @@ static int stripe_status(struct dm_target *ti, status_type_t type, | |||
340 | (unsigned long long)sc->stripe[i].physical_start); | 340 | (unsigned long long)sc->stripe[i].physical_start); |
341 | break; | 341 | break; |
342 | } | 342 | } |
343 | return 0; | ||
344 | } | 343 | } |
345 | 344 | ||
346 | static int stripe_end_io(struct dm_target *ti, struct bio *bio, int error) | 345 | static int stripe_end_io(struct dm_target *ti, struct bio *bio, int error) |
@@ -428,7 +427,7 @@ static int stripe_merge(struct dm_target *ti, struct bvec_merge_data *bvm, | |||
428 | 427 | ||
429 | static struct target_type stripe_target = { | 428 | static struct target_type stripe_target = { |
430 | .name = "striped", | 429 | .name = "striped", |
431 | .version = {1, 5, 0}, | 430 | .version = {1, 5, 1}, |
432 | .module = THIS_MODULE, | 431 | .module = THIS_MODULE, |
433 | .ctr = stripe_ctr, | 432 | .ctr = stripe_ctr, |
434 | .dtr = stripe_dtr, | 433 | .dtr = stripe_dtr, |
diff --git a/drivers/md/dm-thin.c b/drivers/md/dm-thin.c index 5409607d4875..7a66d73148e6 100644 --- a/drivers/md/dm-thin.c +++ b/drivers/md/dm-thin.c | |||
@@ -2299,8 +2299,8 @@ static void emit_flags(struct pool_features *pf, char *result, | |||
2299 | * <transaction id> <used metadata sectors>/<total metadata sectors> | 2299 | * <transaction id> <used metadata sectors>/<total metadata sectors> |
2300 | * <used data sectors>/<total data sectors> <held metadata root> | 2300 | * <used data sectors>/<total data sectors> <held metadata root> |
2301 | */ | 2301 | */ |
2302 | static int pool_status(struct dm_target *ti, status_type_t type, | 2302 | static void pool_status(struct dm_target *ti, status_type_t type, |
2303 | unsigned status_flags, char *result, unsigned maxlen) | 2303 | unsigned status_flags, char *result, unsigned maxlen) |
2304 | { | 2304 | { |
2305 | int r; | 2305 | int r; |
2306 | unsigned sz = 0; | 2306 | unsigned sz = 0; |
@@ -2326,32 +2326,41 @@ static int pool_status(struct dm_target *ti, status_type_t type, | |||
2326 | if (!(status_flags & DM_STATUS_NOFLUSH_FLAG) && !dm_suspended(ti)) | 2326 | if (!(status_flags & DM_STATUS_NOFLUSH_FLAG) && !dm_suspended(ti)) |
2327 | (void) commit_or_fallback(pool); | 2327 | (void) commit_or_fallback(pool); |
2328 | 2328 | ||
2329 | r = dm_pool_get_metadata_transaction_id(pool->pmd, | 2329 | r = dm_pool_get_metadata_transaction_id(pool->pmd, &transaction_id); |
2330 | &transaction_id); | 2330 | if (r) { |
2331 | if (r) | 2331 | DMERR("dm_pool_get_metadata_transaction_id returned %d", r); |
2332 | return r; | 2332 | goto err; |
2333 | } | ||
2333 | 2334 | ||
2334 | r = dm_pool_get_free_metadata_block_count(pool->pmd, | 2335 | r = dm_pool_get_free_metadata_block_count(pool->pmd, &nr_free_blocks_metadata); |
2335 | &nr_free_blocks_metadata); | 2336 | if (r) { |
2336 | if (r) | 2337 | DMERR("dm_pool_get_free_metadata_block_count returned %d", r); |
2337 | return r; | 2338 | goto err; |
2339 | } | ||
2338 | 2340 | ||
2339 | r = dm_pool_get_metadata_dev_size(pool->pmd, &nr_blocks_metadata); | 2341 | r = dm_pool_get_metadata_dev_size(pool->pmd, &nr_blocks_metadata); |
2340 | if (r) | 2342 | if (r) { |
2341 | return r; | 2343 | DMERR("dm_pool_get_metadata_dev_size returned %d", r); |
2344 | goto err; | ||
2345 | } | ||
2342 | 2346 | ||
2343 | r = dm_pool_get_free_block_count(pool->pmd, | 2347 | r = dm_pool_get_free_block_count(pool->pmd, &nr_free_blocks_data); |
2344 | &nr_free_blocks_data); | 2348 | if (r) { |
2345 | if (r) | 2349 | DMERR("dm_pool_get_free_block_count returned %d", r); |
2346 | return r; | 2350 | goto err; |
2351 | } | ||
2347 | 2352 | ||
2348 | r = dm_pool_get_data_dev_size(pool->pmd, &nr_blocks_data); | 2353 | r = dm_pool_get_data_dev_size(pool->pmd, &nr_blocks_data); |
2349 | if (r) | 2354 | if (r) { |
2350 | return r; | 2355 | DMERR("dm_pool_get_data_dev_size returned %d", r); |
2356 | goto err; | ||
2357 | } | ||
2351 | 2358 | ||
2352 | r = dm_pool_get_metadata_snap(pool->pmd, &held_root); | 2359 | r = dm_pool_get_metadata_snap(pool->pmd, &held_root); |
2353 | if (r) | 2360 | if (r) { |
2354 | return r; | 2361 | DMERR("dm_pool_get_metadata_snap returned %d", r); |
2362 | goto err; | ||
2363 | } | ||
2355 | 2364 | ||
2356 | DMEMIT("%llu %llu/%llu %llu/%llu ", | 2365 | DMEMIT("%llu %llu/%llu %llu/%llu ", |
2357 | (unsigned long long)transaction_id, | 2366 | (unsigned long long)transaction_id, |
@@ -2388,8 +2397,10 @@ static int pool_status(struct dm_target *ti, status_type_t type, | |||
2388 | emit_flags(&pt->requested_pf, result, sz, maxlen); | 2397 | emit_flags(&pt->requested_pf, result, sz, maxlen); |
2389 | break; | 2398 | break; |
2390 | } | 2399 | } |
2400 | return; | ||
2391 | 2401 | ||
2392 | return 0; | 2402 | err: |
2403 | DMEMIT("Error"); | ||
2393 | } | 2404 | } |
2394 | 2405 | ||
2395 | static int pool_iterate_devices(struct dm_target *ti, | 2406 | static int pool_iterate_devices(struct dm_target *ti, |
@@ -2468,7 +2479,7 @@ static struct target_type pool_target = { | |||
2468 | .name = "thin-pool", | 2479 | .name = "thin-pool", |
2469 | .features = DM_TARGET_SINGLETON | DM_TARGET_ALWAYS_WRITEABLE | | 2480 | .features = DM_TARGET_SINGLETON | DM_TARGET_ALWAYS_WRITEABLE | |
2470 | DM_TARGET_IMMUTABLE, | 2481 | DM_TARGET_IMMUTABLE, |
2471 | .version = {1, 6, 0}, | 2482 | .version = {1, 6, 1}, |
2472 | .module = THIS_MODULE, | 2483 | .module = THIS_MODULE, |
2473 | .ctr = pool_ctr, | 2484 | .ctr = pool_ctr, |
2474 | .dtr = pool_dtr, | 2485 | .dtr = pool_dtr, |
@@ -2676,8 +2687,8 @@ static void thin_postsuspend(struct dm_target *ti) | |||
2676 | /* | 2687 | /* |
2677 | * <nr mapped sectors> <highest mapped sector> | 2688 | * <nr mapped sectors> <highest mapped sector> |
2678 | */ | 2689 | */ |
2679 | static int thin_status(struct dm_target *ti, status_type_t type, | 2690 | static void thin_status(struct dm_target *ti, status_type_t type, |
2680 | unsigned status_flags, char *result, unsigned maxlen) | 2691 | unsigned status_flags, char *result, unsigned maxlen) |
2681 | { | 2692 | { |
2682 | int r; | 2693 | int r; |
2683 | ssize_t sz = 0; | 2694 | ssize_t sz = 0; |
@@ -2687,7 +2698,7 @@ static int thin_status(struct dm_target *ti, status_type_t type, | |||
2687 | 2698 | ||
2688 | if (get_pool_mode(tc->pool) == PM_FAIL) { | 2699 | if (get_pool_mode(tc->pool) == PM_FAIL) { |
2689 | DMEMIT("Fail"); | 2700 | DMEMIT("Fail"); |
2690 | return 0; | 2701 | return; |
2691 | } | 2702 | } |
2692 | 2703 | ||
2693 | if (!tc->td) | 2704 | if (!tc->td) |
@@ -2696,12 +2707,16 @@ static int thin_status(struct dm_target *ti, status_type_t type, | |||
2696 | switch (type) { | 2707 | switch (type) { |
2697 | case STATUSTYPE_INFO: | 2708 | case STATUSTYPE_INFO: |
2698 | r = dm_thin_get_mapped_count(tc->td, &mapped); | 2709 | r = dm_thin_get_mapped_count(tc->td, &mapped); |
2699 | if (r) | 2710 | if (r) { |
2700 | return r; | 2711 | DMERR("dm_thin_get_mapped_count returned %d", r); |
2712 | goto err; | ||
2713 | } | ||
2701 | 2714 | ||
2702 | r = dm_thin_get_highest_mapped_block(tc->td, &highest); | 2715 | r = dm_thin_get_highest_mapped_block(tc->td, &highest); |
2703 | if (r < 0) | 2716 | if (r < 0) { |
2704 | return r; | 2717 | DMERR("dm_thin_get_highest_mapped_block returned %d", r); |
2718 | goto err; | ||
2719 | } | ||
2705 | 2720 | ||
2706 | DMEMIT("%llu ", mapped * tc->pool->sectors_per_block); | 2721 | DMEMIT("%llu ", mapped * tc->pool->sectors_per_block); |
2707 | if (r) | 2722 | if (r) |
@@ -2721,7 +2736,10 @@ static int thin_status(struct dm_target *ti, status_type_t type, | |||
2721 | } | 2736 | } |
2722 | } | 2737 | } |
2723 | 2738 | ||
2724 | return 0; | 2739 | return; |
2740 | |||
2741 | err: | ||
2742 | DMEMIT("Error"); | ||
2725 | } | 2743 | } |
2726 | 2744 | ||
2727 | static int thin_iterate_devices(struct dm_target *ti, | 2745 | static int thin_iterate_devices(struct dm_target *ti, |
@@ -2748,7 +2766,7 @@ static int thin_iterate_devices(struct dm_target *ti, | |||
2748 | 2766 | ||
2749 | static struct target_type thin_target = { | 2767 | static struct target_type thin_target = { |
2750 | .name = "thin", | 2768 | .name = "thin", |
2751 | .version = {1, 7, 0}, | 2769 | .version = {1, 7, 1}, |
2752 | .module = THIS_MODULE, | 2770 | .module = THIS_MODULE, |
2753 | .ctr = thin_ctr, | 2771 | .ctr = thin_ctr, |
2754 | .dtr = thin_dtr, | 2772 | .dtr = thin_dtr, |
diff --git a/drivers/md/dm-verity.c b/drivers/md/dm-verity.c index 52cde982164a..6ad538375c3c 100644 --- a/drivers/md/dm-verity.c +++ b/drivers/md/dm-verity.c | |||
@@ -508,8 +508,8 @@ static int verity_map(struct dm_target *ti, struct bio *bio) | |||
508 | /* | 508 | /* |
509 | * Status: V (valid) or C (corruption found) | 509 | * Status: V (valid) or C (corruption found) |
510 | */ | 510 | */ |
511 | static int verity_status(struct dm_target *ti, status_type_t type, | 511 | static void verity_status(struct dm_target *ti, status_type_t type, |
512 | unsigned status_flags, char *result, unsigned maxlen) | 512 | unsigned status_flags, char *result, unsigned maxlen) |
513 | { | 513 | { |
514 | struct dm_verity *v = ti->private; | 514 | struct dm_verity *v = ti->private; |
515 | unsigned sz = 0; | 515 | unsigned sz = 0; |
@@ -540,8 +540,6 @@ static int verity_status(struct dm_target *ti, status_type_t type, | |||
540 | DMEMIT("%02x", v->salt[x]); | 540 | DMEMIT("%02x", v->salt[x]); |
541 | break; | 541 | break; |
542 | } | 542 | } |
543 | |||
544 | return 0; | ||
545 | } | 543 | } |
546 | 544 | ||
547 | static int verity_ioctl(struct dm_target *ti, unsigned cmd, | 545 | static int verity_ioctl(struct dm_target *ti, unsigned cmd, |
@@ -860,7 +858,7 @@ bad: | |||
860 | 858 | ||
861 | static struct target_type verity_target = { | 859 | static struct target_type verity_target = { |
862 | .name = "verity", | 860 | .name = "verity", |
863 | .version = {1, 1, 0}, | 861 | .version = {1, 1, 1}, |
864 | .module = THIS_MODULE, | 862 | .module = THIS_MODULE, |
865 | .ctr = verity_ctr, | 863 | .ctr = verity_ctr, |
866 | .dtr = verity_dtr, | 864 | .dtr = verity_dtr, |
diff --git a/include/linux/device-mapper.h b/include/linux/device-mapper.h index bf6afa2fc432..a5cda3ea6b88 100644 --- a/include/linux/device-mapper.h +++ b/include/linux/device-mapper.h | |||
@@ -68,8 +68,8 @@ typedef void (*dm_postsuspend_fn) (struct dm_target *ti); | |||
68 | typedef int (*dm_preresume_fn) (struct dm_target *ti); | 68 | typedef int (*dm_preresume_fn) (struct dm_target *ti); |
69 | typedef void (*dm_resume_fn) (struct dm_target *ti); | 69 | typedef void (*dm_resume_fn) (struct dm_target *ti); |
70 | 70 | ||
71 | typedef int (*dm_status_fn) (struct dm_target *ti, status_type_t status_type, | 71 | typedef void (*dm_status_fn) (struct dm_target *ti, status_type_t status_type, |
72 | unsigned status_flags, char *result, unsigned maxlen); | 72 | unsigned status_flags, char *result, unsigned maxlen); |
73 | 73 | ||
74 | typedef int (*dm_message_fn) (struct dm_target *ti, unsigned argc, char **argv); | 74 | typedef int (*dm_message_fn) (struct dm_target *ti, unsigned argc, char **argv); |
75 | 75 | ||