diff options
Diffstat (limited to 'drivers/md/dm-io.c')
-rw-r--r-- | drivers/md/dm-io.c | 122 |
1 files changed, 122 insertions, 0 deletions
diff --git a/drivers/md/dm-io.c b/drivers/md/dm-io.c index 66db79208c1d..0c63809ab70e 100644 --- a/drivers/md/dm-io.c +++ b/drivers/md/dm-io.c | |||
@@ -103,6 +103,51 @@ void dm_io_put(unsigned int num_pages) | |||
103 | resize_pool(_num_ios - pages_to_ios(num_pages)); | 103 | resize_pool(_num_ios - pages_to_ios(num_pages)); |
104 | } | 104 | } |
105 | 105 | ||
106 | /* | ||
107 | * Create a client with mempool and bioset. | ||
108 | */ | ||
109 | struct dm_io_client *dm_io_client_create(unsigned num_pages) | ||
110 | { | ||
111 | unsigned ios = pages_to_ios(num_pages); | ||
112 | struct dm_io_client *client; | ||
113 | |||
114 | client = kmalloc(sizeof(*client), GFP_KERNEL); | ||
115 | if (!client) | ||
116 | return ERR_PTR(-ENOMEM); | ||
117 | |||
118 | client->pool = mempool_create_kmalloc_pool(ios, sizeof(struct io)); | ||
119 | if (!client->pool) | ||
120 | goto bad; | ||
121 | |||
122 | client->bios = bioset_create(16, 16); | ||
123 | if (!client->bios) | ||
124 | goto bad; | ||
125 | |||
126 | return client; | ||
127 | |||
128 | bad: | ||
129 | if (client->pool) | ||
130 | mempool_destroy(client->pool); | ||
131 | kfree(client); | ||
132 | return ERR_PTR(-ENOMEM); | ||
133 | } | ||
134 | EXPORT_SYMBOL(dm_io_client_create); | ||
135 | |||
136 | int dm_io_client_resize(unsigned num_pages, struct dm_io_client *client) | ||
137 | { | ||
138 | return mempool_resize(client->pool, pages_to_ios(num_pages), | ||
139 | GFP_KERNEL); | ||
140 | } | ||
141 | EXPORT_SYMBOL(dm_io_client_resize); | ||
142 | |||
143 | void dm_io_client_destroy(struct dm_io_client *client) | ||
144 | { | ||
145 | mempool_destroy(client->pool); | ||
146 | bioset_free(client->bios); | ||
147 | kfree(client); | ||
148 | } | ||
149 | EXPORT_SYMBOL(dm_io_client_destroy); | ||
150 | |||
106 | /*----------------------------------------------------------------- | 151 | /*----------------------------------------------------------------- |
107 | * We need to keep track of which region a bio is doing io for. | 152 | * We need to keep track of which region a bio is doing io for. |
108 | * In order to save a memory allocation we store this the last | 153 | * In order to save a memory allocation we store this the last |
@@ -236,6 +281,9 @@ static void bvec_dp_init(struct dpages *dp, struct bio_vec *bvec) | |||
236 | dp->context_ptr = bvec; | 281 | dp->context_ptr = bvec; |
237 | } | 282 | } |
238 | 283 | ||
284 | /* | ||
285 | * Functions for getting the pages from a VMA. | ||
286 | */ | ||
239 | static void vm_get_page(struct dpages *dp, | 287 | static void vm_get_page(struct dpages *dp, |
240 | struct page **p, unsigned long *len, unsigned *offset) | 288 | struct page **p, unsigned long *len, unsigned *offset) |
241 | { | 289 | { |
@@ -265,6 +313,31 @@ static void dm_bio_destructor(struct bio *bio) | |||
265 | bio_free(bio, bios(io->client)); | 313 | bio_free(bio, bios(io->client)); |
266 | } | 314 | } |
267 | 315 | ||
316 | /* | ||
317 | * Functions for getting the pages from kernel memory. | ||
318 | */ | ||
319 | static void km_get_page(struct dpages *dp, struct page **p, unsigned long *len, | ||
320 | unsigned *offset) | ||
321 | { | ||
322 | *p = virt_to_page(dp->context_ptr); | ||
323 | *offset = dp->context_u; | ||
324 | *len = PAGE_SIZE - dp->context_u; | ||
325 | } | ||
326 | |||
327 | static void km_next_page(struct dpages *dp) | ||
328 | { | ||
329 | dp->context_ptr += PAGE_SIZE - dp->context_u; | ||
330 | dp->context_u = 0; | ||
331 | } | ||
332 | |||
333 | static void km_dp_init(struct dpages *dp, void *data) | ||
334 | { | ||
335 | dp->get_page = km_get_page; | ||
336 | dp->next_page = km_next_page; | ||
337 | dp->context_u = ((unsigned long) data) & (PAGE_SIZE - 1); | ||
338 | dp->context_ptr = data; | ||
339 | } | ||
340 | |||
268 | /*----------------------------------------------------------------- | 341 | /*----------------------------------------------------------------- |
269 | * IO routines that accept a list of pages. | 342 | * IO routines that accept a list of pages. |
270 | *---------------------------------------------------------------*/ | 343 | *---------------------------------------------------------------*/ |
@@ -451,6 +524,55 @@ int dm_io_async_vm(unsigned int num_regions, struct io_region *where, int rw, | |||
451 | return async_io(NULL, num_regions, where, rw, &dp, fn, context); | 524 | return async_io(NULL, num_regions, where, rw, &dp, fn, context); |
452 | } | 525 | } |
453 | 526 | ||
527 | static int dp_init(struct dm_io_request *io_req, struct dpages *dp) | ||
528 | { | ||
529 | /* Set up dpages based on memory type */ | ||
530 | switch (io_req->mem.type) { | ||
531 | case DM_IO_PAGE_LIST: | ||
532 | list_dp_init(dp, io_req->mem.ptr.pl, io_req->mem.offset); | ||
533 | break; | ||
534 | |||
535 | case DM_IO_BVEC: | ||
536 | bvec_dp_init(dp, io_req->mem.ptr.bvec); | ||
537 | break; | ||
538 | |||
539 | case DM_IO_VMA: | ||
540 | vm_dp_init(dp, io_req->mem.ptr.vma); | ||
541 | break; | ||
542 | |||
543 | case DM_IO_KMEM: | ||
544 | km_dp_init(dp, io_req->mem.ptr.addr); | ||
545 | break; | ||
546 | |||
547 | default: | ||
548 | return -EINVAL; | ||
549 | } | ||
550 | |||
551 | return 0; | ||
552 | } | ||
553 | |||
554 | /* | ||
555 | * New collapsed (a)synchronous interface | ||
556 | */ | ||
557 | int dm_io(struct dm_io_request *io_req, unsigned num_regions, | ||
558 | struct io_region *where, unsigned long *sync_error_bits) | ||
559 | { | ||
560 | int r; | ||
561 | struct dpages dp; | ||
562 | |||
563 | r = dp_init(io_req, &dp); | ||
564 | if (r) | ||
565 | return r; | ||
566 | |||
567 | if (!io_req->notify.fn) | ||
568 | return sync_io(io_req->client, num_regions, where, | ||
569 | io_req->bi_rw, &dp, sync_error_bits); | ||
570 | |||
571 | return async_io(io_req->client, num_regions, where, io_req->bi_rw, | ||
572 | &dp, io_req->notify.fn, io_req->notify.context); | ||
573 | } | ||
574 | EXPORT_SYMBOL(dm_io); | ||
575 | |||
454 | EXPORT_SYMBOL(dm_io_get); | 576 | EXPORT_SYMBOL(dm_io_get); |
455 | EXPORT_SYMBOL(dm_io_put); | 577 | EXPORT_SYMBOL(dm_io_put); |
456 | EXPORT_SYMBOL(dm_io_sync); | 578 | EXPORT_SYMBOL(dm_io_sync); |