25e7853126
The driver is sending a response to the actual scsi op that was aborted by an abort task TM, while LIO is sending a response to the abort task TM. ibmvscsis_tgt does not send the response to the client until release_cmd time. The reason for this was because if we did it at queue_status time, then the client would be free to reuse the tag for that command, but we're still using the tag until the command is released at release_cmd time, so we chose to delay sending the response until then. That then caused this issue, because release_cmd is always called, even if queue_status is not. SCSI spec says that the initiator that sends the abort task TM NEVER gets a response to the aborted op and with the current code it will send a response. Thus this fix will remove that response if the CMD_T_ABORTED && !CMD_T_TAS. Another case with a small timing window is the case where if LIO sends a TMR_DOES_NOT_EXIST, and the release_cmd callback is called for the TMR Abort cmd before the release_cmd for the (attemped) aborted cmd, then we need to ensure that we send the response for the (attempted) abort cmd to the client before we send the response for the TMR Abort cmd. Cc: <stable@vger.kernel.org> # v4.8+ Signed-off-by: Bryant G. Ly <bryantly@linux.vnet.ibm.com> Signed-off-by: Michael Cyr <mikecyr@linux.vnet.ibm.com> Signed-off-by: Nicholas Bellinger <nab@linux-iscsi.org>
351 lines
10 KiB
C
351 lines
10 KiB
C
/*******************************************************************************
|
|
* IBM Virtual SCSI Target Driver
|
|
* Copyright (C) 2003-2005 Dave Boutcher (boutcher@us.ibm.com) IBM Corp.
|
|
* Santiago Leon (santil@us.ibm.com) IBM Corp.
|
|
* Linda Xie (lxie@us.ibm.com) IBM Corp.
|
|
*
|
|
* Copyright (C) 2005-2011 FUJITA Tomonori <tomof@acm.org>
|
|
* Copyright (C) 2010 Nicholas A. Bellinger <nab@kernel.org>
|
|
* Copyright (C) 2016 Bryant G. Ly <bryantly@linux.vnet.ibm.com> IBM Corp.
|
|
*
|
|
* Authors: Bryant G. Ly <bryantly@linux.vnet.ibm.com>
|
|
* Authors: Michael Cyr <mikecyr@linux.vnet.ibm.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifndef __H_IBMVSCSI_TGT
|
|
#define __H_IBMVSCSI_TGT
|
|
|
|
#include <linux/interrupt.h>
|
|
#include "libsrp.h"
|
|
|
|
#define SYS_ID_NAME_LEN 64
|
|
#define PARTITION_NAMELEN 96
|
|
#define IBMVSCSIS_NAMELEN 32
|
|
|
|
#define MSG_HI 0
|
|
#define MSG_LOW 1
|
|
|
|
#define MAX_CMD_Q_PAGES 4
|
|
#define CRQ_PER_PAGE (PAGE_SIZE / sizeof(struct viosrp_crq))
|
|
/* in terms of number of elements */
|
|
#define DEFAULT_CMD_Q_SIZE CRQ_PER_PAGE
|
|
#define MAX_CMD_Q_SIZE (DEFAULT_CMD_Q_SIZE * MAX_CMD_Q_PAGES)
|
|
|
|
#define SRP_VIOLATION 0x102 /* general error code */
|
|
|
|
/*
|
|
* SRP buffer formats defined as of 16.a supported by this driver.
|
|
*/
|
|
#define SUPPORTED_FORMATS ((SRP_DATA_DESC_DIRECT << 1) | \
|
|
(SRP_DATA_DESC_INDIRECT << 1))
|
|
|
|
#define SCSI_LUN_ADDR_METHOD_FLAT 1
|
|
|
|
struct dma_window {
|
|
u32 liobn; /* Unique per vdevice */
|
|
u64 tce_base; /* Physical location of the TCE table */
|
|
u64 tce_size; /* Size of the TCE table in bytes */
|
|
};
|
|
|
|
struct target_dds {
|
|
u64 unit_id; /* 64 bit will force alignment */
|
|
#define NUM_DMA_WINDOWS 2
|
|
#define LOCAL 0
|
|
#define REMOTE 1
|
|
struct dma_window window[NUM_DMA_WINDOWS];
|
|
|
|
/* root node property "ibm,partition-no" */
|
|
uint partition_num;
|
|
char partition_name[PARTITION_NAMELEN];
|
|
};
|
|
|
|
#define MAX_NUM_PORTS 1
|
|
#define MAX_H_COPY_RDMA (128 * 1024)
|
|
|
|
#define MAX_EYE 64
|
|
|
|
/* Return codes */
|
|
#define ADAPT_SUCCESS 0L
|
|
/* choose error codes that do not conflict with PHYP */
|
|
#define ERROR -40L
|
|
|
|
struct format_code {
|
|
u8 reserved;
|
|
u8 buffers;
|
|
};
|
|
|
|
struct client_info {
|
|
#define SRP_VERSION "16.a"
|
|
char srp_version[8];
|
|
/* root node property ibm,partition-name */
|
|
char partition_name[PARTITION_NAMELEN];
|
|
/* root node property ibm,partition-no */
|
|
u32 partition_number;
|
|
/* initially 1 */
|
|
u32 mad_version;
|
|
u32 os_type;
|
|
};
|
|
|
|
/*
|
|
* Changing this constant changes the number of seconds to wait before
|
|
* considering the client will never service its queue again.
|
|
*/
|
|
#define SECONDS_TO_CONSIDER_FAILED 30
|
|
/*
|
|
* These constants set the polling period used to determine if the client
|
|
* has freed at least one element in the response queue.
|
|
*/
|
|
#define WAIT_SECONDS 1
|
|
#define WAIT_NANO_SECONDS 5000
|
|
#define MAX_TIMER_POPS ((1000000 / WAIT_NANO_SECONDS) * \
|
|
SECONDS_TO_CONSIDER_FAILED)
|
|
/*
|
|
* general purpose timer control block
|
|
* which can be used for multiple functions
|
|
*/
|
|
struct timer_cb {
|
|
struct hrtimer timer;
|
|
/*
|
|
* how long has it been since the client
|
|
* serviced the queue. The variable is incrmented
|
|
* in the service_wait_q routine and cleared
|
|
* in send messages
|
|
*/
|
|
int timer_pops;
|
|
/* the timer is started */
|
|
bool started;
|
|
};
|
|
|
|
struct cmd_queue {
|
|
/* kva */
|
|
struct viosrp_crq *base_addr;
|
|
dma_addr_t crq_token;
|
|
/* used to maintain index */
|
|
uint mask;
|
|
/* current element */
|
|
uint index;
|
|
int size;
|
|
};
|
|
|
|
#define SCSOLNT_RESP_SHIFT 1
|
|
#define UCSOLNT_RESP_SHIFT 2
|
|
|
|
#define SCSOLNT BIT(SCSOLNT_RESP_SHIFT)
|
|
#define UCSOLNT BIT(UCSOLNT_RESP_SHIFT)
|
|
|
|
enum cmd_type {
|
|
SCSI_CDB = 0x01,
|
|
TASK_MANAGEMENT = 0x02,
|
|
/* MAD or addressed to port 0 */
|
|
ADAPTER_MAD = 0x04,
|
|
UNSET_TYPE = 0x08,
|
|
};
|
|
|
|
struct iu_rsp {
|
|
u8 format;
|
|
u8 sol_not;
|
|
u16 len;
|
|
/* tag is just to help client identify cmd, so don't translate be/le */
|
|
u64 tag;
|
|
};
|
|
|
|
struct ibmvscsis_cmd {
|
|
struct list_head list;
|
|
/* Used for TCM Core operations */
|
|
struct se_cmd se_cmd;
|
|
struct iu_entry *iue;
|
|
struct iu_rsp rsp;
|
|
struct work_struct work;
|
|
struct scsi_info *adapter;
|
|
struct ibmvscsis_cmd *abort_cmd;
|
|
/* Sense buffer that will be mapped into outgoing status */
|
|
unsigned char sense_buf[TRANSPORT_SENSE_BUFFER];
|
|
u64 init_time;
|
|
#define CMD_FAST_FAIL BIT(0)
|
|
#define DELAY_SEND BIT(1)
|
|
u32 flags;
|
|
char type;
|
|
};
|
|
|
|
struct ibmvscsis_nexus {
|
|
struct se_session *se_sess;
|
|
};
|
|
|
|
struct ibmvscsis_tport {
|
|
/* SCSI protocol the tport is providing */
|
|
u8 tport_proto_id;
|
|
/* ASCII formatted WWPN for SRP Target port */
|
|
char tport_name[IBMVSCSIS_NAMELEN];
|
|
/* Returned by ibmvscsis_make_tport() */
|
|
struct se_wwn tport_wwn;
|
|
/* Returned by ibmvscsis_make_tpg() */
|
|
struct se_portal_group se_tpg;
|
|
/* ibmvscsis port target portal group tag for TCM */
|
|
u16 tport_tpgt;
|
|
/* Pointer to TCM session for I_T Nexus */
|
|
struct ibmvscsis_nexus *ibmv_nexus;
|
|
bool enabled;
|
|
bool releasing;
|
|
};
|
|
|
|
struct scsi_info {
|
|
struct list_head list;
|
|
char eye[MAX_EYE];
|
|
|
|
/* commands waiting for space on repsonse queue */
|
|
struct list_head waiting_rsp;
|
|
#define NO_QUEUE 0x00
|
|
#define WAIT_ENABLED 0X01
|
|
#define WAIT_CONNECTION 0x04
|
|
/* have established a connection */
|
|
#define CONNECTED 0x08
|
|
/* at least one port is processing SRP IU */
|
|
#define SRP_PROCESSING 0x10
|
|
/* remove request received */
|
|
#define UNCONFIGURING 0x20
|
|
/* disconnect by letting adapter go idle, no error */
|
|
#define WAIT_IDLE 0x40
|
|
/* disconnecting to clear an error */
|
|
#define ERR_DISCONNECT 0x80
|
|
/* disconnect to clear error state, then come back up */
|
|
#define ERR_DISCONNECT_RECONNECT 0x100
|
|
/* disconnected after clearing an error */
|
|
#define ERR_DISCONNECTED 0x200
|
|
/* A series of errors caused unexpected errors */
|
|
#define UNDEFINED 0x400
|
|
u16 state;
|
|
int fast_fail;
|
|
struct target_dds dds;
|
|
char *cmd_pool;
|
|
/* list of free commands */
|
|
struct list_head free_cmd;
|
|
/* command elements ready for scheduler */
|
|
struct list_head schedule_q;
|
|
/* commands sent to TCM */
|
|
struct list_head active_q;
|
|
caddr_t *map_buf;
|
|
/* ioba of map buffer */
|
|
dma_addr_t map_ioba;
|
|
/* allowable number of outstanding SRP requests */
|
|
int request_limit;
|
|
/* extra credit */
|
|
int credit;
|
|
/* outstanding transactions against credit limit */
|
|
int debit;
|
|
|
|
/* allow only one outstanding mad request */
|
|
#define PROCESSING_MAD 0x00002
|
|
/* Waiting to go idle */
|
|
#define WAIT_FOR_IDLE 0x00004
|
|
/* H_REG_CRQ called */
|
|
#define CRQ_CLOSED 0x00010
|
|
/* detected that client has failed */
|
|
#define CLIENT_FAILED 0x00040
|
|
/* detected that transport event occurred */
|
|
#define TRANS_EVENT 0x00080
|
|
/* don't attempt to send anything to the client */
|
|
#define RESPONSE_Q_DOWN 0x00100
|
|
/* request made to schedule disconnect handler */
|
|
#define SCHEDULE_DISCONNECT 0x00400
|
|
/* disconnect handler is scheduled */
|
|
#define DISCONNECT_SCHEDULED 0x00800
|
|
/* remove function is sleeping */
|
|
#define CFG_SLEEPING 0x01000
|
|
u32 flags;
|
|
/* adapter lock */
|
|
spinlock_t intr_lock;
|
|
/* information needed to manage command queue */
|
|
struct cmd_queue cmd_q;
|
|
/* used in hcall to copy response back into srp buffer */
|
|
u64 empty_iu_id;
|
|
/* used in crq, to tag what iu the response is for */
|
|
u64 empty_iu_tag;
|
|
uint new_state;
|
|
/* control block for the response queue timer */
|
|
struct timer_cb rsp_q_timer;
|
|
/* keep last client to enable proper accounting */
|
|
struct client_info client_data;
|
|
/* what can this client do */
|
|
u32 client_cap;
|
|
/*
|
|
* The following two fields capture state and flag changes that
|
|
* can occur when the lock is given up. In the orginal design,
|
|
* the lock was held during calls into phyp;
|
|
* however, phyp did not meet PAPR architecture. This is
|
|
* a work around.
|
|
*/
|
|
u16 phyp_acr_state;
|
|
u32 phyp_acr_flags;
|
|
|
|
struct workqueue_struct *work_q;
|
|
struct completion wait_idle;
|
|
struct completion unconfig;
|
|
struct device dev;
|
|
struct vio_dev *dma_dev;
|
|
struct srp_target target;
|
|
struct ibmvscsis_tport tport;
|
|
struct tasklet_struct work_task;
|
|
struct work_struct proc_work;
|
|
};
|
|
|
|
/*
|
|
* Provide a constant that allows software to detect the adapter is
|
|
* disconnecting from the client from one of several states.
|
|
*/
|
|
#define IS_DISCONNECTING (UNCONFIGURING | ERR_DISCONNECT_RECONNECT | \
|
|
ERR_DISCONNECT)
|
|
|
|
/*
|
|
* Provide a constant that can be used with interrupt handling that
|
|
* essentially lets the interrupt handler know that all requests should
|
|
* be thrown out,
|
|
*/
|
|
#define DONT_PROCESS_STATE (IS_DISCONNECTING | UNDEFINED | \
|
|
ERR_DISCONNECTED | WAIT_IDLE)
|
|
|
|
/*
|
|
* If any of these flag bits are set then do not allow the interrupt
|
|
* handler to schedule the off level handler.
|
|
*/
|
|
#define BLOCK (DISCONNECT_SCHEDULED)
|
|
|
|
/* State and transition events that stop the interrupt handler */
|
|
#define TARGET_STOP(VSCSI) (long)(((VSCSI)->state & DONT_PROCESS_STATE) | \
|
|
((VSCSI)->flags & BLOCK))
|
|
|
|
/* flag bit that are not reset during disconnect */
|
|
#define PRESERVE_FLAG_FIELDS 0
|
|
|
|
#define vio_iu(IUE) ((union viosrp_iu *)((IUE)->sbuf->buf))
|
|
|
|
#define READ_CMD(cdb) (((cdb)[0] & 0x1F) == 8)
|
|
#define WRITE_CMD(cdb) (((cdb)[0] & 0x1F) == 0xA)
|
|
|
|
#ifndef H_GET_PARTNER_INFO
|
|
#define H_GET_PARTNER_INFO 0x0000000000000008LL
|
|
#endif
|
|
|
|
#define h_copy_rdma(l, sa, sb, da, db) \
|
|
plpar_hcall_norets(H_COPY_RDMA, l, sa, sb, da, db)
|
|
#define h_vioctl(u, o, a, u1, u2, u3, u4) \
|
|
plpar_hcall_norets(H_VIOCTL, u, o, a, u1, u2)
|
|
#define h_reg_crq(ua, tok, sz) \
|
|
plpar_hcall_norets(H_REG_CRQ, ua, tok, sz)
|
|
#define h_free_crq(ua) \
|
|
plpar_hcall_norets(H_FREE_CRQ, ua)
|
|
#define h_send_crq(ua, d1, d2) \
|
|
plpar_hcall_norets(H_SEND_CRQ, ua, d1, d2)
|
|
|
|
#endif
|