Commit c5bca9df authored by Greg Hackmann's avatar Greg Hackmann Committed by Kleber Sacilotto de Souza

staging: android: ion: fix ION_IOC_{MAP,SHARE} use-after-free

BugLink: https://bugs.launchpad.net/bugs/1797563

The ION_IOC_{MAP,SHARE} ioctls drop and reacquire client->lock several
times while operating on one of the client's ion_handles.  This creates
windows where userspace can call ION_IOC_FREE on the same client with
the same handle, and effectively make the kernel drop its own reference.
For example:

- thread A: ION_IOC_ALLOC creates an ion_handle with refcount 1
- thread A: starts ION_IOC_MAP and increments the refcount to 2
- thread B: ION_IOC_FREE decrements the refcount to 1
- thread B: ION_IOC_FREE decrements the refcount to 0 and frees the
            handle
- thread A: continues ION_IOC_MAP with a dangling ion_handle * to
            freed memory

Fix this by holding client->lock for the duration of
ION_IOC_{MAP,SHARE}, preventing the concurrent ION_IOC_FREE.  Also
remove ion_handle_get_by_id(), since there's literally no way to use it
safely.

This patch is applied on top of 4.4.y, and applies to older kernels
too.  4.9.y was fixed separately.  Kernels 4.12 and later are
unaffected, since all the underlying ion_handle infrastructure has been
ripped out.

Cc: stable@vger.kernel.org # v4.4-
Signed-off-by: default avatarGreg Hackmann <ghackmann@google.com>
Acked-by: default avatarLaura Abbott <labbott@redhat.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: default avatarStefan Bader <stefan.bader@canonical.com>
Signed-off-by: default avatarKleber Sacilotto de Souza <kleber.souza@canonical.com>
parent ace965b1
...@@ -449,18 +449,6 @@ static struct ion_handle *ion_handle_get_by_id_nolock(struct ion_client *client, ...@@ -449,18 +449,6 @@ static struct ion_handle *ion_handle_get_by_id_nolock(struct ion_client *client,
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);
} }
struct ion_handle *ion_handle_get_by_id(struct ion_client *client,
int id)
{
struct ion_handle *handle;
mutex_lock(&client->lock);
handle = ion_handle_get_by_id_nolock(client, id);
mutex_unlock(&client->lock);
return handle;
}
static bool ion_handle_validate(struct ion_client *client, static bool ion_handle_validate(struct ion_client *client,
struct ion_handle *handle) struct ion_handle *handle)
{ {
...@@ -1138,23 +1126,27 @@ static struct dma_buf_ops dma_buf_ops = { ...@@ -1138,23 +1126,27 @@ static struct dma_buf_ops dma_buf_ops = {
.kunmap = ion_dma_buf_kunmap, .kunmap = ion_dma_buf_kunmap,
}; };
struct dma_buf *ion_share_dma_buf(struct ion_client *client, static struct dma_buf *__ion_share_dma_buf(struct ion_client *client,
struct ion_handle *handle) struct ion_handle *handle,
bool lock_client)
{ {
DEFINE_DMA_BUF_EXPORT_INFO(exp_info); DEFINE_DMA_BUF_EXPORT_INFO(exp_info);
struct ion_buffer *buffer; struct ion_buffer *buffer;
struct dma_buf *dmabuf; struct dma_buf *dmabuf;
bool valid_handle; bool valid_handle;
if (lock_client)
mutex_lock(&client->lock); mutex_lock(&client->lock);
valid_handle = ion_handle_validate(client, handle); valid_handle = ion_handle_validate(client, handle);
if (!valid_handle) { if (!valid_handle) {
WARN(1, "%s: invalid handle passed to share.\n", __func__); WARN(1, "%s: invalid handle passed to share.\n", __func__);
if (lock_client)
mutex_unlock(&client->lock); mutex_unlock(&client->lock);
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);
} }
buffer = handle->buffer; buffer = handle->buffer;
ion_buffer_get(buffer); ion_buffer_get(buffer);
if (lock_client)
mutex_unlock(&client->lock); mutex_unlock(&client->lock);
exp_info.ops = &dma_buf_ops; exp_info.ops = &dma_buf_ops;
...@@ -1170,14 +1162,21 @@ struct dma_buf *ion_share_dma_buf(struct ion_client *client, ...@@ -1170,14 +1162,21 @@ struct dma_buf *ion_share_dma_buf(struct ion_client *client,
return dmabuf; return dmabuf;
} }
struct dma_buf *ion_share_dma_buf(struct ion_client *client,
struct ion_handle *handle)
{
return __ion_share_dma_buf(client, handle, true);
}
EXPORT_SYMBOL(ion_share_dma_buf); EXPORT_SYMBOL(ion_share_dma_buf);
int ion_share_dma_buf_fd(struct ion_client *client, struct ion_handle *handle) static int __ion_share_dma_buf_fd(struct ion_client *client,
struct ion_handle *handle, bool lock_client)
{ {
struct dma_buf *dmabuf; struct dma_buf *dmabuf;
int fd; int fd;
dmabuf = ion_share_dma_buf(client, handle); dmabuf = __ion_share_dma_buf(client, handle, lock_client);
if (IS_ERR(dmabuf)) if (IS_ERR(dmabuf))
return PTR_ERR(dmabuf); return PTR_ERR(dmabuf);
...@@ -1187,8 +1186,19 @@ int ion_share_dma_buf_fd(struct ion_client *client, struct ion_handle *handle) ...@@ -1187,8 +1186,19 @@ int ion_share_dma_buf_fd(struct ion_client *client, struct ion_handle *handle)
return fd; return fd;
} }
int ion_share_dma_buf_fd(struct ion_client *client, struct ion_handle *handle)
{
return __ion_share_dma_buf_fd(client, handle, true);
}
EXPORT_SYMBOL(ion_share_dma_buf_fd); EXPORT_SYMBOL(ion_share_dma_buf_fd);
static int ion_share_dma_buf_fd_nolock(struct ion_client *client,
struct ion_handle *handle)
{
return __ion_share_dma_buf_fd(client, handle, false);
}
struct ion_handle *ion_import_dma_buf(struct ion_client *client, int fd) struct ion_handle *ion_import_dma_buf(struct ion_client *client, int fd)
{ {
struct dma_buf *dmabuf; struct dma_buf *dmabuf;
...@@ -1335,11 +1345,15 @@ static long ion_ioctl(struct file *filp, unsigned int cmd, unsigned long arg) ...@@ -1335,11 +1345,15 @@ static long ion_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
{ {
struct ion_handle *handle; struct ion_handle *handle;
handle = ion_handle_get_by_id(client, data.handle.handle); mutex_lock(&client->lock);
if (IS_ERR(handle)) handle = ion_handle_get_by_id_nolock(client, data.handle.handle);
if (IS_ERR(handle)) {
mutex_unlock(&client->lock);
return PTR_ERR(handle); return PTR_ERR(handle);
data.fd.fd = ion_share_dma_buf_fd(client, handle); }
ion_handle_put(handle); data.fd.fd = ion_share_dma_buf_fd_nolock(client, handle);
ion_handle_put_nolock(handle);
mutex_unlock(&client->lock);
if (data.fd.fd < 0) if (data.fd.fd < 0)
ret = data.fd.fd; ret = data.fd.fd;
break; break;
......
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