提交 0a6ac4ee 编写于 作者: C Christoph Hellwig 提交者: Jens Axboe

scsi: respect unchecked_isa_dma for blk-mq

Currently blk-mq always allocates the sense buffer using normal GFP_KERNEL
allocation.  Refactor the cmd pool code to split the cmd and sense allocation
and share the code to allocate the sense buffers as well as the sense buffer
slab caches between the legacy and blk-mq path.

Note that this switches to lazy allocation of the sense slab caches - the
slab caches (not the actual allocations) won't be destroy until the scsi
module is unloaded instead of keeping track of hosts using them.
Signed-off-by: NChristoph Hellwig <hch@lst.de>
Reviewed-by: NHannes Reinecke <hare@suse.com>
Acked-by: NMartin K. Petersen <martin.petersen@oracle.com>
Signed-off-by: NJens Axboe <axboe@fb.com>
上级 0fbc3e0f
...@@ -213,6 +213,10 @@ int scsi_add_host_with_dma(struct Scsi_Host *shost, struct device *dev, ...@@ -213,6 +213,10 @@ int scsi_add_host_with_dma(struct Scsi_Host *shost, struct device *dev,
goto fail; goto fail;
} }
error = scsi_init_sense_cache(shost);
if (error)
goto fail;
if (shost_use_blk_mq(shost)) { if (shost_use_blk_mq(shost)) {
error = scsi_mq_setup_tags(shost); error = scsi_mq_setup_tags(shost);
if (error) if (error)
......
...@@ -100,22 +100,18 @@ EXPORT_SYMBOL(scsi_sd_pm_domain); ...@@ -100,22 +100,18 @@ EXPORT_SYMBOL(scsi_sd_pm_domain);
struct scsi_host_cmd_pool { struct scsi_host_cmd_pool {
struct kmem_cache *cmd_slab; struct kmem_cache *cmd_slab;
struct kmem_cache *sense_slab;
unsigned int users; unsigned int users;
char *cmd_name; char *cmd_name;
char *sense_name;
unsigned int slab_flags; unsigned int slab_flags;
}; };
static struct scsi_host_cmd_pool scsi_cmd_pool = { static struct scsi_host_cmd_pool scsi_cmd_pool = {
.cmd_name = "scsi_cmd_cache", .cmd_name = "scsi_cmd_cache",
.sense_name = "scsi_sense_cache",
.slab_flags = SLAB_HWCACHE_ALIGN, .slab_flags = SLAB_HWCACHE_ALIGN,
}; };
static struct scsi_host_cmd_pool scsi_cmd_dma_pool = { static struct scsi_host_cmd_pool scsi_cmd_dma_pool = {
.cmd_name = "scsi_cmd_cache(DMA)", .cmd_name = "scsi_cmd_cache(DMA)",
.sense_name = "scsi_sense_cache(DMA)",
.slab_flags = SLAB_HWCACHE_ALIGN|SLAB_CACHE_DMA, .slab_flags = SLAB_HWCACHE_ALIGN|SLAB_CACHE_DMA,
}; };
...@@ -136,7 +132,7 @@ scsi_host_free_command(struct Scsi_Host *shost, struct scsi_cmnd *cmd) ...@@ -136,7 +132,7 @@ scsi_host_free_command(struct Scsi_Host *shost, struct scsi_cmnd *cmd)
if (cmd->prot_sdb) if (cmd->prot_sdb)
kmem_cache_free(scsi_sdb_cache, cmd->prot_sdb); kmem_cache_free(scsi_sdb_cache, cmd->prot_sdb);
kmem_cache_free(pool->sense_slab, cmd->sense_buffer); scsi_free_sense_buffer(shost, cmd->sense_buffer);
kmem_cache_free(pool->cmd_slab, cmd); kmem_cache_free(pool->cmd_slab, cmd);
} }
...@@ -158,7 +154,8 @@ scsi_host_alloc_command(struct Scsi_Host *shost, gfp_t gfp_mask) ...@@ -158,7 +154,8 @@ scsi_host_alloc_command(struct Scsi_Host *shost, gfp_t gfp_mask)
if (!cmd) if (!cmd)
goto fail; goto fail;
cmd->sense_buffer = kmem_cache_alloc(pool->sense_slab, gfp_mask); cmd->sense_buffer = scsi_alloc_sense_buffer(shost, gfp_mask,
NUMA_NO_NODE);
if (!cmd->sense_buffer) if (!cmd->sense_buffer)
goto fail_free_cmd; goto fail_free_cmd;
...@@ -171,7 +168,7 @@ scsi_host_alloc_command(struct Scsi_Host *shost, gfp_t gfp_mask) ...@@ -171,7 +168,7 @@ scsi_host_alloc_command(struct Scsi_Host *shost, gfp_t gfp_mask)
return cmd; return cmd;
fail_free_sense: fail_free_sense:
kmem_cache_free(pool->sense_slab, cmd->sense_buffer); scsi_free_sense_buffer(shost, cmd->sense_buffer);
fail_free_cmd: fail_free_cmd:
kmem_cache_free(pool->cmd_slab, cmd); kmem_cache_free(pool->cmd_slab, cmd);
fail: fail:
...@@ -301,7 +298,6 @@ scsi_find_host_cmd_pool(struct Scsi_Host *shost) ...@@ -301,7 +298,6 @@ scsi_find_host_cmd_pool(struct Scsi_Host *shost)
static void static void
scsi_free_host_cmd_pool(struct scsi_host_cmd_pool *pool) scsi_free_host_cmd_pool(struct scsi_host_cmd_pool *pool)
{ {
kfree(pool->sense_name);
kfree(pool->cmd_name); kfree(pool->cmd_name);
kfree(pool); kfree(pool);
} }
...@@ -317,8 +313,7 @@ scsi_alloc_host_cmd_pool(struct Scsi_Host *shost) ...@@ -317,8 +313,7 @@ scsi_alloc_host_cmd_pool(struct Scsi_Host *shost)
return NULL; return NULL;
pool->cmd_name = kasprintf(GFP_KERNEL, "%s_cmd", hostt->proc_name); pool->cmd_name = kasprintf(GFP_KERNEL, "%s_cmd", hostt->proc_name);
pool->sense_name = kasprintf(GFP_KERNEL, "%s_sense", hostt->proc_name); if (!pool->cmd_name) {
if (!pool->cmd_name || !pool->sense_name) {
scsi_free_host_cmd_pool(pool); scsi_free_host_cmd_pool(pool);
return NULL; return NULL;
} }
...@@ -357,12 +352,6 @@ scsi_get_host_cmd_pool(struct Scsi_Host *shost) ...@@ -357,12 +352,6 @@ scsi_get_host_cmd_pool(struct Scsi_Host *shost)
pool->slab_flags, NULL); pool->slab_flags, NULL);
if (!pool->cmd_slab) if (!pool->cmd_slab)
goto out_free_pool; goto out_free_pool;
pool->sense_slab = kmem_cache_create(pool->sense_name,
SCSI_SENSE_BUFFERSIZE, 0,
pool->slab_flags, NULL);
if (!pool->sense_slab)
goto out_free_slab;
} }
pool->users++; pool->users++;
...@@ -371,8 +360,6 @@ scsi_get_host_cmd_pool(struct Scsi_Host *shost) ...@@ -371,8 +360,6 @@ scsi_get_host_cmd_pool(struct Scsi_Host *shost)
mutex_unlock(&host_cmd_pool_mutex); mutex_unlock(&host_cmd_pool_mutex);
return retval; return retval;
out_free_slab:
kmem_cache_destroy(pool->cmd_slab);
out_free_pool: out_free_pool:
if (hostt->cmd_size) { if (hostt->cmd_size) {
scsi_free_host_cmd_pool(pool); scsi_free_host_cmd_pool(pool);
...@@ -398,7 +385,6 @@ static void scsi_put_host_cmd_pool(struct Scsi_Host *shost) ...@@ -398,7 +385,6 @@ static void scsi_put_host_cmd_pool(struct Scsi_Host *shost)
if (!--pool->users) { if (!--pool->users) {
kmem_cache_destroy(pool->cmd_slab); kmem_cache_destroy(pool->cmd_slab);
kmem_cache_destroy(pool->sense_slab);
if (hostt->cmd_size) { if (hostt->cmd_size) {
scsi_free_host_cmd_pool(pool); scsi_free_host_cmd_pool(pool);
hostt->cmd_pool = NULL; hostt->cmd_pool = NULL;
......
...@@ -39,6 +39,58 @@ ...@@ -39,6 +39,58 @@
struct kmem_cache *scsi_sdb_cache; struct kmem_cache *scsi_sdb_cache;
static struct kmem_cache *scsi_sense_cache;
static struct kmem_cache *scsi_sense_isadma_cache;
static DEFINE_MUTEX(scsi_sense_cache_mutex);
static inline struct kmem_cache *
scsi_select_sense_cache(struct Scsi_Host *shost)
{
return shost->unchecked_isa_dma ?
scsi_sense_isadma_cache : scsi_sense_cache;
}
void scsi_free_sense_buffer(struct Scsi_Host *shost,
unsigned char *sense_buffer)
{
kmem_cache_free(scsi_select_sense_cache(shost), sense_buffer);
}
unsigned char *scsi_alloc_sense_buffer(struct Scsi_Host *shost, gfp_t gfp_mask,
int numa_node)
{
return kmem_cache_alloc_node(scsi_select_sense_cache(shost), gfp_mask,
numa_node);
}
int scsi_init_sense_cache(struct Scsi_Host *shost)
{
struct kmem_cache *cache;
int ret = 0;
cache = scsi_select_sense_cache(shost);
if (cache)
return 0;
mutex_lock(&scsi_sense_cache_mutex);
if (shost->unchecked_isa_dma) {
scsi_sense_isadma_cache =
kmem_cache_create("scsi_sense_cache(DMA)",
SCSI_SENSE_BUFFERSIZE, 0,
SLAB_HWCACHE_ALIGN | SLAB_CACHE_DMA, NULL);
if (!scsi_sense_isadma_cache)
ret = -ENOMEM;
} else {
scsi_sense_cache =
kmem_cache_create("scsi_sense_cache",
SCSI_SENSE_BUFFERSIZE, 0, SLAB_HWCACHE_ALIGN, NULL);
if (!scsi_sense_cache)
ret = -ENOMEM;
}
mutex_unlock(&scsi_sense_cache_mutex);
return ret;
}
/* /*
* When to reinvoke queueing after a resource shortage. It's 3 msecs to * When to reinvoke queueing after a resource shortage. It's 3 msecs to
...@@ -1981,10 +2033,11 @@ static int scsi_init_request(void *data, struct request *rq, ...@@ -1981,10 +2033,11 @@ static int scsi_init_request(void *data, struct request *rq,
unsigned int hctx_idx, unsigned int request_idx, unsigned int hctx_idx, unsigned int request_idx,
unsigned int numa_node) unsigned int numa_node)
{ {
struct Scsi_Host *shost = data;
struct scsi_cmnd *cmd = blk_mq_rq_to_pdu(rq); struct scsi_cmnd *cmd = blk_mq_rq_to_pdu(rq);
cmd->sense_buffer = kzalloc_node(SCSI_SENSE_BUFFERSIZE, GFP_KERNEL, cmd->sense_buffer =
numa_node); scsi_alloc_sense_buffer(shost, GFP_KERNEL, numa_node);
if (!cmd->sense_buffer) if (!cmd->sense_buffer)
return -ENOMEM; return -ENOMEM;
return 0; return 0;
...@@ -1993,9 +2046,10 @@ static int scsi_init_request(void *data, struct request *rq, ...@@ -1993,9 +2046,10 @@ static int scsi_init_request(void *data, struct request *rq,
static void scsi_exit_request(void *data, struct request *rq, static void scsi_exit_request(void *data, struct request *rq,
unsigned int hctx_idx, unsigned int request_idx) unsigned int hctx_idx, unsigned int request_idx)
{ {
struct Scsi_Host *shost = data;
struct scsi_cmnd *cmd = blk_mq_rq_to_pdu(rq); struct scsi_cmnd *cmd = blk_mq_rq_to_pdu(rq);
kfree(cmd->sense_buffer); scsi_free_sense_buffer(shost, cmd->sense_buffer);
} }
static int scsi_map_queues(struct blk_mq_tag_set *set) static int scsi_map_queues(struct blk_mq_tag_set *set)
...@@ -2208,6 +2262,8 @@ int __init scsi_init_queue(void) ...@@ -2208,6 +2262,8 @@ int __init scsi_init_queue(void)
void scsi_exit_queue(void) void scsi_exit_queue(void)
{ {
kmem_cache_destroy(scsi_sense_cache);
kmem_cache_destroy(scsi_sense_isadma_cache);
kmem_cache_destroy(scsi_sdb_cache); kmem_cache_destroy(scsi_sdb_cache);
} }
......
...@@ -30,6 +30,11 @@ extern void scsi_exit_hosts(void); ...@@ -30,6 +30,11 @@ extern void scsi_exit_hosts(void);
/* scsi.c */ /* scsi.c */
extern bool scsi_use_blk_mq; extern bool scsi_use_blk_mq;
void scsi_free_sense_buffer(struct Scsi_Host *shost,
unsigned char *sense_buffer);
unsigned char *scsi_alloc_sense_buffer(struct Scsi_Host *shost, gfp_t gfp_mask,
int numa_node);
int scsi_init_sense_cache(struct Scsi_Host *shost);
extern int scsi_setup_command_freelist(struct Scsi_Host *shost); extern int scsi_setup_command_freelist(struct Scsi_Host *shost);
extern void scsi_destroy_command_freelist(struct Scsi_Host *shost); extern void scsi_destroy_command_freelist(struct Scsi_Host *shost);
#ifdef CONFIG_SCSI_LOGGING #ifdef CONFIG_SCSI_LOGGING
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册