aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/s390/scsi/zfcp_fsf.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/s390/scsi/zfcp_fsf.c')
-rw-r--r--drivers/s390/scsi/zfcp_fsf.c60
1 files changed, 32 insertions, 28 deletions
diff --git a/drivers/s390/scsi/zfcp_fsf.c b/drivers/s390/scsi/zfcp_fsf.c
index 3b0fc1163f5f..fbc81b012919 100644
--- a/drivers/s390/scsi/zfcp_fsf.c
+++ b/drivers/s390/scsi/zfcp_fsf.c
@@ -2191,13 +2191,10 @@ zfcp_fsf_exchange_port_data(struct zfcp_erp_action *erp_action,
2191 return -EOPNOTSUPP; 2191 return -EOPNOTSUPP;
2192 } 2192 }
2193 2193
2194 timer = kmalloc(sizeof(struct timer_list), GFP_KERNEL);
2195 if (!timer)
2196 return -ENOMEM;
2197
2198 /* setup new FSF request */ 2194 /* setup new FSF request */
2199 retval = zfcp_fsf_req_create(adapter, FSF_QTCB_EXCHANGE_PORT_DATA, 2195 retval = zfcp_fsf_req_create(adapter, FSF_QTCB_EXCHANGE_PORT_DATA,
2200 0, 0, &lock_flags, &fsf_req); 2196 erp_action ? ZFCP_REQ_AUTO_CLEANUP : 0,
2197 0, &lock_flags, &fsf_req);
2201 if (retval < 0) { 2198 if (retval < 0) {
2202 ZFCP_LOG_INFO("error: Out of resources. Could not create an " 2199 ZFCP_LOG_INFO("error: Out of resources. Could not create an "
2203 "exchange port data request for" 2200 "exchange port data request for"
@@ -2205,25 +2202,33 @@ zfcp_fsf_exchange_port_data(struct zfcp_erp_action *erp_action,
2205 zfcp_get_busid_by_adapter(adapter)); 2202 zfcp_get_busid_by_adapter(adapter));
2206 write_unlock_irqrestore(&adapter->request_queue.queue_lock, 2203 write_unlock_irqrestore(&adapter->request_queue.queue_lock,
2207 lock_flags); 2204 lock_flags);
2208 goto out; 2205 return retval;
2209 }
2210
2211 if (erp_action) {
2212 erp_action->fsf_req = fsf_req;
2213 fsf_req->erp_action = erp_action;
2214 } 2206 }
2215 2207
2216 if (data) 2208 if (data)
2217 fsf_req->data = (unsigned long) data; 2209 fsf_req->data = (unsigned long) data;
2218 2210
2219 sbale = zfcp_qdio_sbale_req(fsf_req, fsf_req->sbal_curr, 0); 2211 sbale = zfcp_qdio_sbale_req(fsf_req, fsf_req->sbal_curr, 0);
2220 sbale[0].flags |= SBAL_FLAGS0_TYPE_READ; 2212 sbale[0].flags |= SBAL_FLAGS0_TYPE_READ;
2221 sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY; 2213 sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY;
2222 2214
2223 init_timer(timer); 2215 if (erp_action) {
2224 timer->function = zfcp_fsf_request_timeout_handler; 2216 erp_action->fsf_req = fsf_req;
2225 timer->data = (unsigned long) adapter; 2217 fsf_req->erp_action = erp_action;
2226 timer->expires = ZFCP_FSF_REQUEST_TIMEOUT; 2218 timer = &erp_action->timer;
2219 } else {
2220 timer = kmalloc(sizeof(struct timer_list), GFP_ATOMIC);
2221 if (!timer) {
2222 write_unlock_irqrestore(&adapter->request_queue.queue_lock,
2223 lock_flags);
2224 zfcp_fsf_req_free(fsf_req);
2225 return -ENOMEM;
2226 }
2227 init_timer(timer);
2228 timer->function = zfcp_fsf_request_timeout_handler;
2229 timer->data = (unsigned long) adapter;
2230 timer->expires = ZFCP_FSF_REQUEST_TIMEOUT;
2231 }
2227 2232
2228 retval = zfcp_fsf_req_send(fsf_req, timer); 2233 retval = zfcp_fsf_req_send(fsf_req, timer);
2229 if (retval) { 2234 if (retval) {
@@ -2233,23 +2238,22 @@ zfcp_fsf_exchange_port_data(struct zfcp_erp_action *erp_action,
2233 zfcp_fsf_req_free(fsf_req); 2238 zfcp_fsf_req_free(fsf_req);
2234 if (erp_action) 2239 if (erp_action)
2235 erp_action->fsf_req = NULL; 2240 erp_action->fsf_req = NULL;
2241 else
2242 kfree(timer);
2236 write_unlock_irqrestore(&adapter->request_queue.queue_lock, 2243 write_unlock_irqrestore(&adapter->request_queue.queue_lock,
2237 lock_flags); 2244 lock_flags);
2238 goto out; 2245 return retval;
2239 } 2246 }
2240 2247
2241 ZFCP_LOG_DEBUG("Exchange Port Data request initiated (adapter %s)\n", 2248 write_unlock_irqrestore(&adapter->request_queue.queue_lock, lock_flags);
2242 zfcp_get_busid_by_adapter(adapter));
2243
2244 write_unlock_irqrestore(&adapter->request_queue.queue_lock,
2245 lock_flags);
2246 2249
2247 wait_event(fsf_req->completion_wq, 2250 if (!erp_action) {
2248 fsf_req->status & ZFCP_STATUS_FSFREQ_COMPLETED); 2251 wait_event(fsf_req->completion_wq,
2249 del_timer_sync(timer); 2252 fsf_req->status & ZFCP_STATUS_FSFREQ_COMPLETED);
2250 zfcp_fsf_req_free(fsf_req); 2253 del_timer_sync(timer);
2251 out: 2254 zfcp_fsf_req_free(fsf_req);
2252 kfree(timer); 2255 kfree(timer);
2256 }
2253 return retval; 2257 return retval;
2254} 2258}
2255 2259