[SCSI] be2iscsi: Fix memory leak in control path of driver
In contorl path of the driver the task was mapped using pci_map_single which was not unmapped when the completion for the task had come. Signed-off-by: John Soni Jose <sony.john-n@emulex.com> Signed-off-by: Jayamohan Kallickal <jayamohan.kallickal@emulex.com> Reviewed-by: Mike Christie <michaelc@cs.wisc.edu> Signed-off-by: James Bottomley <JBottomley@Parallels.com>
This commit is contained in:
parent
6763daae8f
commit
d629c47171
|
@ -2242,10 +2242,14 @@ hwi_write_sgl(struct iscsi_wrb *pwrb, struct scatterlist *sg,
|
||||||
AMAP_SET_BITS(struct amap_iscsi_sge, last_sge, psgl, 1);
|
AMAP_SET_BITS(struct amap_iscsi_sge, last_sge, psgl, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* hwi_write_buffer()- Populate the WRB with task info
|
||||||
|
* @pwrb: ptr to the WRB entry
|
||||||
|
* @task: iscsi task which is to be executed
|
||||||
|
**/
|
||||||
static void hwi_write_buffer(struct iscsi_wrb *pwrb, struct iscsi_task *task)
|
static void hwi_write_buffer(struct iscsi_wrb *pwrb, struct iscsi_task *task)
|
||||||
{
|
{
|
||||||
struct iscsi_sge *psgl;
|
struct iscsi_sge *psgl;
|
||||||
unsigned long long addr;
|
|
||||||
struct beiscsi_io_task *io_task = task->dd_data;
|
struct beiscsi_io_task *io_task = task->dd_data;
|
||||||
struct beiscsi_conn *beiscsi_conn = io_task->conn;
|
struct beiscsi_conn *beiscsi_conn = io_task->conn;
|
||||||
struct beiscsi_hba *phba = beiscsi_conn->phba;
|
struct beiscsi_hba *phba = beiscsi_conn->phba;
|
||||||
|
@ -2259,24 +2263,27 @@ static void hwi_write_buffer(struct iscsi_wrb *pwrb, struct iscsi_task *task)
|
||||||
if (task->data) {
|
if (task->data) {
|
||||||
if (task->data_count) {
|
if (task->data_count) {
|
||||||
AMAP_SET_BITS(struct amap_iscsi_wrb, dsp, pwrb, 1);
|
AMAP_SET_BITS(struct amap_iscsi_wrb, dsp, pwrb, 1);
|
||||||
addr = (u64) pci_map_single(phba->pcidev,
|
io_task->mtask_addr = pci_map_single(phba->pcidev,
|
||||||
task->data,
|
task->data,
|
||||||
task->data_count, 1);
|
task->data_count,
|
||||||
|
PCI_DMA_TODEVICE);
|
||||||
|
|
||||||
|
io_task->mtask_data_count = task->data_count;
|
||||||
} else {
|
} else {
|
||||||
AMAP_SET_BITS(struct amap_iscsi_wrb, dsp, pwrb, 0);
|
AMAP_SET_BITS(struct amap_iscsi_wrb, dsp, pwrb, 0);
|
||||||
addr = 0;
|
io_task->mtask_addr = 0;
|
||||||
}
|
}
|
||||||
AMAP_SET_BITS(struct amap_iscsi_wrb, sge0_addr_lo, pwrb,
|
AMAP_SET_BITS(struct amap_iscsi_wrb, sge0_addr_lo, pwrb,
|
||||||
((u32)(addr & 0xFFFFFFFF)));
|
lower_32_bits(io_task->mtask_addr));
|
||||||
AMAP_SET_BITS(struct amap_iscsi_wrb, sge0_addr_hi, pwrb,
|
AMAP_SET_BITS(struct amap_iscsi_wrb, sge0_addr_hi, pwrb,
|
||||||
((u32)(addr >> 32)));
|
upper_32_bits(io_task->mtask_addr));
|
||||||
AMAP_SET_BITS(struct amap_iscsi_wrb, sge0_len, pwrb,
|
AMAP_SET_BITS(struct amap_iscsi_wrb, sge0_len, pwrb,
|
||||||
task->data_count);
|
task->data_count);
|
||||||
|
|
||||||
AMAP_SET_BITS(struct amap_iscsi_wrb, sge0_last, pwrb, 1);
|
AMAP_SET_BITS(struct amap_iscsi_wrb, sge0_last, pwrb, 1);
|
||||||
} else {
|
} else {
|
||||||
AMAP_SET_BITS(struct amap_iscsi_wrb, dsp, pwrb, 0);
|
AMAP_SET_BITS(struct amap_iscsi_wrb, dsp, pwrb, 0);
|
||||||
addr = 0;
|
io_task->mtask_addr = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
psgl = (struct iscsi_sge *)io_task->psgl_handle->pfrag;
|
psgl = (struct iscsi_sge *)io_task->psgl_handle->pfrag;
|
||||||
|
@ -2299,9 +2306,9 @@ static void hwi_write_buffer(struct iscsi_wrb *pwrb, struct iscsi_task *task)
|
||||||
psgl++;
|
psgl++;
|
||||||
if (task->data) {
|
if (task->data) {
|
||||||
AMAP_SET_BITS(struct amap_iscsi_sge, addr_lo, psgl,
|
AMAP_SET_BITS(struct amap_iscsi_sge, addr_lo, psgl,
|
||||||
((u32)(addr & 0xFFFFFFFF)));
|
lower_32_bits(io_task->mtask_addr));
|
||||||
AMAP_SET_BITS(struct amap_iscsi_sge, addr_hi, psgl,
|
AMAP_SET_BITS(struct amap_iscsi_sge, addr_hi, psgl,
|
||||||
((u32)(addr >> 32)));
|
upper_32_bits(io_task->mtask_addr));
|
||||||
}
|
}
|
||||||
AMAP_SET_BITS(struct amap_iscsi_sge, len, psgl, 0x106);
|
AMAP_SET_BITS(struct amap_iscsi_sge, len, psgl, 0x106);
|
||||||
}
|
}
|
||||||
|
@ -3893,6 +3900,11 @@ static void beiscsi_clean_port(struct beiscsi_hba *phba)
|
||||||
kfree(phba->ep_array);
|
kfree(phba->ep_array);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* beiscsi_cleanup_task()- Free driver resources of the task
|
||||||
|
* @task: ptr to the iscsi task
|
||||||
|
*
|
||||||
|
**/
|
||||||
static void beiscsi_cleanup_task(struct iscsi_task *task)
|
static void beiscsi_cleanup_task(struct iscsi_task *task)
|
||||||
{
|
{
|
||||||
struct beiscsi_io_task *io_task = task->dd_data;
|
struct beiscsi_io_task *io_task = task->dd_data;
|
||||||
|
@ -3940,6 +3952,13 @@ static void beiscsi_cleanup_task(struct iscsi_task *task)
|
||||||
spin_unlock(&phba->mgmt_sgl_lock);
|
spin_unlock(&phba->mgmt_sgl_lock);
|
||||||
io_task->psgl_handle = NULL;
|
io_task->psgl_handle = NULL;
|
||||||
}
|
}
|
||||||
|
if (io_task->mtask_addr) {
|
||||||
|
pci_unmap_single(phba->pcidev,
|
||||||
|
io_task->mtask_addr,
|
||||||
|
io_task->mtask_data_count,
|
||||||
|
PCI_DMA_TODEVICE);
|
||||||
|
io_task->mtask_addr = 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -410,6 +410,8 @@ struct beiscsi_io_task {
|
||||||
struct be_cmd_bhs *cmd_bhs;
|
struct be_cmd_bhs *cmd_bhs;
|
||||||
struct be_bus_address bhs_pa;
|
struct be_bus_address bhs_pa;
|
||||||
unsigned short bhs_len;
|
unsigned short bhs_len;
|
||||||
|
dma_addr_t mtask_addr;
|
||||||
|
uint32_t mtask_data_count;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct be_nonio_bhs {
|
struct be_nonio_bhs {
|
||||||
|
|
Loading…
Reference in New Issue