Skip to content

Commit

Permalink
xen/gntdev: safely unmap grants in case they are still in use
Browse files Browse the repository at this point in the history
Use gnttab_unmap_refs_async() to wait until the mapped pages are no
longer in use before unmapping them.

This allows userspace programs to safely use Direct I/O and AIO to a
network filesystem which may retain refs to pages in queued skbs after
the filesystem I/O has completed.

Signed-off-by: Jennifer Herbert <[email protected]>
Reviewed-by: Stefano Stabellini <[email protected]>
Signed-off-by: David Vrabel <[email protected]>
  • Loading branch information
xennifer authored and David Vrabel committed Jan 28, 2015
1 parent 1401c00 commit 7452822
Showing 1 changed file with 31 additions and 5 deletions.
36 changes: 31 additions & 5 deletions drivers/xen/gntdev.c
Original file line number Diff line number Diff line change
Expand Up @@ -309,9 +309,30 @@ static int map_grant_pages(struct grant_map *map)
return err;
}

struct unmap_grant_pages_callback_data
{
struct completion completion;
int result;
};

static void unmap_grant_callback(int result,
struct gntab_unmap_queue_data *data)
{
struct unmap_grant_pages_callback_data* d = data->data;

d->result = result;
complete(&d->completion);
}

static int __unmap_grant_pages(struct grant_map *map, int offset, int pages)
{
int i, err = 0;
struct gntab_unmap_queue_data unmap_data;
struct unmap_grant_pages_callback_data data;

init_completion(&data.completion);
unmap_data.data = &data;
unmap_data.done= &unmap_grant_callback;

if (map->notify.flags & UNMAP_NOTIFY_CLEAR_BYTE) {
int pgno = (map->notify.addr >> PAGE_SHIFT);
Expand All @@ -323,11 +344,16 @@ static int __unmap_grant_pages(struct grant_map *map, int offset, int pages)
}
}

err = gnttab_unmap_refs(map->unmap_ops + offset,
use_ptemod ? map->kunmap_ops + offset : NULL, map->pages + offset,
pages);
if (err)
return err;
unmap_data.unmap_ops = map->unmap_ops + offset;
unmap_data.kunmap_ops = use_ptemod ? map->kunmap_ops + offset : NULL;
unmap_data.pages = map->pages + offset;
unmap_data.count = pages;

gnttab_unmap_refs_async(&unmap_data);

wait_for_completion(&data.completion);
if (data.result)
return data.result;

for (i = 0; i < pages; i++) {
if (map->unmap_ops[offset+i].status)
Expand Down

0 comments on commit 7452822

Please sign in to comment.