/*
* New collapsed (a)synchronous interface.
*
* If the IO is asynchronous (i.e. it has notify.fn), you must either unplug
* the queue with blk_unplug() some time later or set REQ_SYNC in
io_req->bi_rw. If you fail to do one of these, the IO will be submitted to
* the disk after q->unplug_delay, which defaults to 3ms in blk-settings.c.
*/intdm_io(structdm_io_request*io_req,unsignednum_regions,structdm_io_region*where,unsignedlong*sync_error_bits){intr;structdpagesdp;r=dp_init(io_req,&dp,(unsignedlong)where->count<<SECTOR_SHIFT);if(r)returnr;if(!io_req->notify.fn)returnsync_io(io_req->client,num_regions,where,io_req->bi_rw,&dp,sync_error_bits);returnasync_io(io_req->client,num_regions,where,io_req->bi_rw,&dp,io_req->notify.fn,io_req->notify.context);}EXPORT_SYMBOL(dm_io);
structdm_io_request{intbi_rw;/* READ|WRITE - not READA */structdm_io_memorymem;/* Memory to use for io */structdm_io_notifynotify;/* Synchronous if notify.fn is NULL */structdm_io_client*client;/* Client memory handler */};enumdm_io_mem_type{DM_IO_PAGE_LIST,/* Page list */DM_IO_BVEC,/* Bio vector */DM_IO_VMA,/* Virtual memory area */DM_IO_KMEM,/* Kernel memory */};structdm_io_notify{io_notify_fnfn;/* Callback for asynchronous requests */void*context;/* Passed to callback */};typedefvoid(*io_notify_fn)(unsignedlongerror,void*context);structdm_io_region{structblock_device*bdev;sector_tsector;sector_tcount;/* If this is zero the region is ignored. */};
staticvoiddispatch_io(intrw,unsignedintnum_regions,structdm_io_region*where,structdpages*dp,structio*io,intsync){inti;structdpagesold_pages=*dp;BUG_ON(num_regions>DM_IO_MAX_REGIONS);if(sync)rw|=REQ_SYNC;/*
* For multiple regions we need to be careful to rewind
* the dp object for each call to do_region.
*/for(i=0;i<num_regions;i++){*dp=old_pages;if(where[i].count||(rw&REQ_FLUSH))do_region(rw,i,where+i,dp,io);}/*
* Drop the extra reference that we were holding to avoid
* the io being completed too early.
*/dec_count(io,0,0);}
*1:not a few people misunderstand it*2:実際に発行するかは, num_regionsで与えた数と一致しません. APIの上から1つでも, 実際には2個発行されるかも知れません. 実際に計算しなければ分からないためこのようなことになってしまっています*3:dm-lcではこのようなテクニックが多様されているため, コードがシンプルになっています