Commit 76191bb2 authored by Roland Dreier's avatar Roland Dreier Committed by Ben Hutchings

target: Fix possible integer underflow in UNMAP emulation

commit b7fc7f37 upstream.

It's possible for an initiator to send us an UNMAP command with a
descriptor that is less than 8 bytes; in that case it's really bad for
us to set an unsigned int to that value, subtract 8 from it, and then
use that as a limit for our loop (since the value will wrap around to
a huge positive value).

Fix this by making size be signed and only looping if size >= 16 (ie
if we have at least a full descriptor available).

Also remove offset as an obfuscated name for the constant 8.
Signed-off-by: default avatarRoland Dreier <roland@purestorage.com>
Signed-off-by: default avatarNicholas Bellinger <nab@linux-iscsi.org>
[bwh: Backported to 3.2: adjust filename, context]
Signed-off-by: default avatarBen Hutchings <ben@decadent.org.uk>
parent b6718392
...@@ -1115,9 +1115,10 @@ int target_emulate_unmap(struct se_task *task) ...@@ -1115,9 +1115,10 @@ int target_emulate_unmap(struct se_task *task)
struct se_device *dev = cmd->se_dev; struct se_device *dev = cmd->se_dev;
unsigned char *buf, *ptr = NULL; unsigned char *buf, *ptr = NULL;
sector_t lba; sector_t lba;
unsigned int size = cmd->data_length, range; int size = cmd->data_length;
int ret = 0, offset; u32 range;
unsigned short dl, bd_dl; int ret = 0;
int dl, bd_dl;
if (!dev->transport->do_discard) { if (!dev->transport->do_discard) {
pr_err("UNMAP emulation not supported for: %s\n", pr_err("UNMAP emulation not supported for: %s\n",
...@@ -1126,20 +1127,19 @@ int target_emulate_unmap(struct se_task *task) ...@@ -1126,20 +1127,19 @@ int target_emulate_unmap(struct se_task *task)
return -ENOSYS; return -ENOSYS;
} }
/* First UNMAP block descriptor starts at 8 byte offset */
offset = 8;
size -= 8;
buf = transport_kmap_data_sg(cmd); buf = transport_kmap_data_sg(cmd);
dl = get_unaligned_be16(&buf[0]); dl = get_unaligned_be16(&buf[0]);
bd_dl = get_unaligned_be16(&buf[2]); bd_dl = get_unaligned_be16(&buf[2]);
ptr = &buf[offset]; size = min(size - 8, bd_dl);
pr_debug("UNMAP: Sub: %s Using dl: %hu bd_dl: %hu size: %hu"
/* First UNMAP block descriptor starts at 8 byte offset */
ptr = &buf[8];
pr_debug("UNMAP: Sub: %s Using dl: %u bd_dl: %u size: %u"
" ptr: %p\n", dev->transport->name, dl, bd_dl, size, ptr); " ptr: %p\n", dev->transport->name, dl, bd_dl, size, ptr);
while (size) { while (size >= 16) {
lba = get_unaligned_be64(&ptr[0]); lba = get_unaligned_be64(&ptr[0]);
range = get_unaligned_be32(&ptr[8]); range = get_unaligned_be32(&ptr[8]);
pr_debug("UNMAP: Using lba: %llu and range: %u\n", pr_debug("UNMAP: Using lba: %llu and range: %u\n",
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment