mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-01 10:42:11 +00:00
Merge branch 'for-next' of git://git.kernel.org/pub/scm/linux/kernel/git/mkp/scsi.git
This commit is contained in:
commit
46ab0e9d85
@ -5403,7 +5403,7 @@ W: http://www.chelsio.com
|
||||
F: drivers/net/ethernet/chelsio/cxgb3/
|
||||
|
||||
CXGB3 ISCSI DRIVER (CXGB3I)
|
||||
M: Karen Xie <kxie@chelsio.com>
|
||||
M: Varun Prakash <varun@chelsio.com>
|
||||
L: linux-scsi@vger.kernel.org
|
||||
S: Supported
|
||||
W: http://www.chelsio.com
|
||||
@ -5435,7 +5435,7 @@ W: http://www.chelsio.com
|
||||
F: drivers/net/ethernet/chelsio/cxgb4/
|
||||
|
||||
CXGB4 ISCSI DRIVER (CXGB4I)
|
||||
M: Karen Xie <kxie@chelsio.com>
|
||||
M: Varun Prakash <varun@chelsio.com>
|
||||
L: linux-scsi@vger.kernel.org
|
||||
S: Supported
|
||||
W: http://www.chelsio.com
|
||||
|
@ -169,7 +169,7 @@ static void bsg_device_release(struct device *dev)
|
||||
{
|
||||
struct bsg_device *bd = container_of(dev, struct bsg_device, device);
|
||||
|
||||
ida_simple_remove(&bsg_minor_ida, MINOR(bd->device.devt));
|
||||
ida_free(&bsg_minor_ida, MINOR(bd->device.devt));
|
||||
kfree(bd);
|
||||
}
|
||||
|
||||
@ -196,7 +196,7 @@ struct bsg_device *bsg_register_queue(struct request_queue *q,
|
||||
bd->queue = q;
|
||||
bd->sg_io_fn = sg_io_fn;
|
||||
|
||||
ret = ida_simple_get(&bsg_minor_ida, 0, BSG_MAX_DEVS, GFP_KERNEL);
|
||||
ret = ida_alloc_max(&bsg_minor_ida, BSG_MAX_DEVS - 1, GFP_KERNEL);
|
||||
if (ret < 0) {
|
||||
if (ret == -ENOSPC)
|
||||
dev_err(parent, "bsg: too many bsg devices\n");
|
||||
|
@ -568,7 +568,7 @@ static void iscsi_iser_session_destroy(struct iscsi_cls_session *cls_session)
|
||||
struct Scsi_Host *shost = iscsi_session_to_shost(cls_session);
|
||||
|
||||
iscsi_session_teardown(cls_session);
|
||||
iscsi_host_remove(shost);
|
||||
iscsi_host_remove(shost, false);
|
||||
iscsi_host_free(shost);
|
||||
}
|
||||
|
||||
@ -685,7 +685,7 @@ iscsi_iser_session_create(struct iscsi_endpoint *ep,
|
||||
return cls_session;
|
||||
|
||||
remove_host:
|
||||
iscsi_host_remove(shost);
|
||||
iscsi_host_remove(shost, false);
|
||||
free_host:
|
||||
iscsi_host_free(shost);
|
||||
return NULL;
|
||||
|
@ -101,7 +101,7 @@ static u8 mptspiInternalCtx = MPT_MAX_PROTOCOL_DRIVERS; /* Used only for interna
|
||||
* @target: per target private data
|
||||
* @sdev: SCSI device
|
||||
*
|
||||
* Update the target negotiation parameters based on the the Inquiry
|
||||
* Update the target negotiation parameters based on the Inquiry
|
||||
* data, adapter capabilities, and NVRAM settings.
|
||||
**/
|
||||
static void
|
||||
|
@ -1050,7 +1050,7 @@ static void get_container_serial_callback(void *context, struct fib * fibptr)
|
||||
vpdpage83data.type1.productid));
|
||||
|
||||
/* Convert to ascii based serial number.
|
||||
* The LSB is the the end.
|
||||
* The LSB is the end.
|
||||
*/
|
||||
for (i = 0; i < 8; i++) {
|
||||
u8 temp =
|
||||
|
@ -231,6 +231,7 @@ static int beiscsi_eh_abort(struct scsi_cmnd *sc)
|
||||
cls_session = starget_to_session(scsi_target(sc->device));
|
||||
session = cls_session->dd_data;
|
||||
|
||||
completion_check:
|
||||
/* check if we raced, task just got cleaned up under us */
|
||||
spin_lock_bh(&session->back_lock);
|
||||
if (!abrt_task || !abrt_task->sc) {
|
||||
@ -238,7 +239,13 @@ static int beiscsi_eh_abort(struct scsi_cmnd *sc)
|
||||
return SUCCESS;
|
||||
}
|
||||
/* get a task ref till FW processes the req for the ICD used */
|
||||
__iscsi_get_task(abrt_task);
|
||||
if (!iscsi_get_task(abrt_task)) {
|
||||
spin_unlock(&session->back_lock);
|
||||
/* We are just about to call iscsi_free_task so wait for it. */
|
||||
udelay(5);
|
||||
goto completion_check;
|
||||
}
|
||||
|
||||
abrt_io_task = abrt_task->dd_data;
|
||||
conn = abrt_task->conn;
|
||||
beiscsi_conn = conn->dd_data;
|
||||
@ -323,7 +330,15 @@ static int beiscsi_eh_device_reset(struct scsi_cmnd *sc)
|
||||
}
|
||||
|
||||
/* get a task ref till FW processes the req for the ICD used */
|
||||
__iscsi_get_task(task);
|
||||
if (!iscsi_get_task(task)) {
|
||||
/*
|
||||
* The task has completed in the driver and is
|
||||
* completing in libiscsi. Just ignore it here. When we
|
||||
* call iscsi_eh_device_reset, it will wait for us.
|
||||
*/
|
||||
continue;
|
||||
}
|
||||
|
||||
io_task = task->dd_data;
|
||||
/* mark WRB invalid which have been not processed by FW yet */
|
||||
if (is_chip_be2_be3r(phba)) {
|
||||
@ -5745,7 +5760,7 @@ static void beiscsi_remove(struct pci_dev *pcidev)
|
||||
cancel_work_sync(&phba->sess_work);
|
||||
|
||||
beiscsi_iface_destroy_default(phba);
|
||||
iscsi_host_remove(phba->shost);
|
||||
iscsi_host_remove(phba->shost, false);
|
||||
beiscsi_disable_port(phba, 1);
|
||||
|
||||
/* after cancelling boot_work */
|
||||
|
@ -909,7 +909,7 @@ void bnx2i_free_hba(struct bnx2i_hba *hba)
|
||||
{
|
||||
struct Scsi_Host *shost = hba->shost;
|
||||
|
||||
iscsi_host_remove(shost);
|
||||
iscsi_host_remove(shost, false);
|
||||
INIT_LIST_HEAD(&hba->ep_ofld_list);
|
||||
INIT_LIST_HEAD(&hba->ep_active_list);
|
||||
INIT_LIST_HEAD(&hba->ep_destroy_list);
|
||||
|
@ -328,7 +328,7 @@ void cxgbi_hbas_remove(struct cxgbi_device *cdev)
|
||||
chba = cdev->hbas[i];
|
||||
if (chba) {
|
||||
cdev->hbas[i] = NULL;
|
||||
iscsi_host_remove(chba->shost);
|
||||
iscsi_host_remove(chba->shost, false);
|
||||
pci_dev_put(cdev->pdev);
|
||||
iscsi_host_free(chba->shost);
|
||||
}
|
||||
@ -1455,7 +1455,7 @@ void cxgbi_conn_tx_open(struct cxgbi_sock *csk)
|
||||
if (conn) {
|
||||
log_debug(1 << CXGBI_DBG_SOCK,
|
||||
"csk 0x%p, cid %d.\n", csk, conn->id);
|
||||
iscsi_conn_queue_work(conn);
|
||||
iscsi_conn_queue_xmit(conn);
|
||||
}
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(cxgbi_conn_tx_open);
|
||||
|
@ -1488,7 +1488,6 @@ static int fcoe_xmit(struct fc_lport *lport, struct fc_frame *fp)
|
||||
|
||||
fh = fc_frame_header_get(fp);
|
||||
skb = fp_skb(fp);
|
||||
wlen = skb->len / FCOE_WORD_TO_BYTE;
|
||||
|
||||
if (!lport->link_up) {
|
||||
kfree_skb(skb);
|
||||
|
@ -805,8 +805,8 @@ slot_index_alloc_quirk_v2_hw(struct hisi_hba *hisi_hba,
|
||||
return -SAS_QUEUE_FULL;
|
||||
}
|
||||
/*
|
||||
* SAS IPTT bit0 should be 1, and SATA IPTT bit0 should be 0.
|
||||
*/
|
||||
* SAS IPTT bit0 should be 1, and SATA IPTT bit0 should be 0.
|
||||
*/
|
||||
if (sata_dev ^ (start & 1))
|
||||
break;
|
||||
start++;
|
||||
|
@ -52,6 +52,10 @@ static struct iscsi_transport iscsi_sw_tcp_transport;
|
||||
static unsigned int iscsi_max_lun = ~0;
|
||||
module_param_named(max_lun, iscsi_max_lun, uint, S_IRUGO);
|
||||
|
||||
static bool iscsi_recv_from_iscsi_q;
|
||||
module_param_named(recv_from_iscsi_q, iscsi_recv_from_iscsi_q, bool, 0644);
|
||||
MODULE_PARM_DESC(recv_from_iscsi_q, "Set to true to read iSCSI data/headers from the iscsi_q workqueue. The default is false which will perform reads from the network softirq context.");
|
||||
|
||||
static int iscsi_sw_tcp_dbg;
|
||||
module_param_named(debug_iscsi_tcp, iscsi_sw_tcp_dbg, int,
|
||||
S_IRUGO | S_IWUSR);
|
||||
@ -122,20 +126,13 @@ static inline int iscsi_sw_sk_state_check(struct sock *sk)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void iscsi_sw_tcp_data_ready(struct sock *sk)
|
||||
static void iscsi_sw_tcp_recv_data(struct iscsi_conn *conn)
|
||||
{
|
||||
struct iscsi_conn *conn;
|
||||
struct iscsi_tcp_conn *tcp_conn;
|
||||
struct iscsi_tcp_conn *tcp_conn = conn->dd_data;
|
||||
struct iscsi_sw_tcp_conn *tcp_sw_conn = tcp_conn->dd_data;
|
||||
struct sock *sk = tcp_sw_conn->sock->sk;
|
||||
read_descriptor_t rd_desc;
|
||||
|
||||
read_lock_bh(&sk->sk_callback_lock);
|
||||
conn = sk->sk_user_data;
|
||||
if (!conn) {
|
||||
read_unlock_bh(&sk->sk_callback_lock);
|
||||
return;
|
||||
}
|
||||
tcp_conn = conn->dd_data;
|
||||
|
||||
/*
|
||||
* Use rd_desc to pass 'conn' to iscsi_tcp_recv.
|
||||
* We set count to 1 because we want the network layer to
|
||||
@ -144,13 +141,48 @@ static void iscsi_sw_tcp_data_ready(struct sock *sk)
|
||||
*/
|
||||
rd_desc.arg.data = conn;
|
||||
rd_desc.count = 1;
|
||||
tcp_read_sock(sk, &rd_desc, iscsi_sw_tcp_recv);
|
||||
|
||||
iscsi_sw_sk_state_check(sk);
|
||||
tcp_read_sock(sk, &rd_desc, iscsi_sw_tcp_recv);
|
||||
|
||||
/* If we had to (atomically) map a highmem page,
|
||||
* unmap it now. */
|
||||
iscsi_tcp_segment_unmap(&tcp_conn->in.segment);
|
||||
|
||||
iscsi_sw_sk_state_check(sk);
|
||||
}
|
||||
|
||||
static void iscsi_sw_tcp_recv_data_work(struct work_struct *work)
|
||||
{
|
||||
struct iscsi_conn *conn = container_of(work, struct iscsi_conn,
|
||||
recvwork);
|
||||
struct iscsi_tcp_conn *tcp_conn = conn->dd_data;
|
||||
struct iscsi_sw_tcp_conn *tcp_sw_conn = tcp_conn->dd_data;
|
||||
struct sock *sk = tcp_sw_conn->sock->sk;
|
||||
|
||||
lock_sock(sk);
|
||||
iscsi_sw_tcp_recv_data(conn);
|
||||
release_sock(sk);
|
||||
}
|
||||
|
||||
static void iscsi_sw_tcp_data_ready(struct sock *sk)
|
||||
{
|
||||
struct iscsi_sw_tcp_conn *tcp_sw_conn;
|
||||
struct iscsi_tcp_conn *tcp_conn;
|
||||
struct iscsi_conn *conn;
|
||||
|
||||
read_lock_bh(&sk->sk_callback_lock);
|
||||
conn = sk->sk_user_data;
|
||||
if (!conn) {
|
||||
read_unlock_bh(&sk->sk_callback_lock);
|
||||
return;
|
||||
}
|
||||
tcp_conn = conn->dd_data;
|
||||
tcp_sw_conn = tcp_conn->dd_data;
|
||||
|
||||
if (tcp_sw_conn->queue_recv)
|
||||
iscsi_conn_queue_recv(conn);
|
||||
else
|
||||
iscsi_sw_tcp_recv_data(conn);
|
||||
read_unlock_bh(&sk->sk_callback_lock);
|
||||
}
|
||||
|
||||
@ -205,7 +237,7 @@ static void iscsi_sw_tcp_write_space(struct sock *sk)
|
||||
old_write_space(sk);
|
||||
|
||||
ISCSI_SW_TCP_DBG(conn, "iscsi_write_space\n");
|
||||
iscsi_conn_queue_work(conn);
|
||||
iscsi_conn_queue_xmit(conn);
|
||||
}
|
||||
|
||||
static void iscsi_sw_tcp_conn_set_callbacks(struct iscsi_conn *conn)
|
||||
@ -274,7 +306,10 @@ static int iscsi_sw_tcp_xmit_segment(struct iscsi_tcp_conn *tcp_conn,
|
||||
copy = segment->size - offset;
|
||||
|
||||
if (segment->total_copied + segment->size < segment->total_size)
|
||||
flags |= MSG_MORE;
|
||||
flags |= MSG_MORE | MSG_SENDPAGE_NOTLAST;
|
||||
|
||||
if (tcp_sw_conn->queue_recv)
|
||||
flags |= MSG_DONTWAIT;
|
||||
|
||||
/* Use sendpage if we can; else fall back to sendmsg */
|
||||
if (!segment->data) {
|
||||
@ -557,6 +592,8 @@ iscsi_sw_tcp_conn_create(struct iscsi_cls_session *cls_session,
|
||||
conn = cls_conn->dd_data;
|
||||
tcp_conn = conn->dd_data;
|
||||
tcp_sw_conn = tcp_conn->dd_data;
|
||||
INIT_WORK(&conn->recvwork, iscsi_sw_tcp_recv_data_work);
|
||||
tcp_sw_conn->queue_recv = iscsi_recv_from_iscsi_q;
|
||||
|
||||
tfm = crypto_alloc_ahash("crc32c", 0, CRYPTO_ALG_ASYNC);
|
||||
if (IS_ERR(tfm))
|
||||
@ -610,6 +647,8 @@ static void iscsi_sw_tcp_release_conn(struct iscsi_conn *conn)
|
||||
iscsi_sw_tcp_conn_restore_callbacks(conn);
|
||||
sock_put(sock->sk);
|
||||
|
||||
iscsi_suspend_rx(conn);
|
||||
|
||||
spin_lock_bh(&session->frwd_lock);
|
||||
tcp_sw_conn->sock = NULL;
|
||||
spin_unlock_bh(&session->frwd_lock);
|
||||
@ -898,7 +937,7 @@ iscsi_sw_tcp_session_create(struct iscsi_endpoint *ep, uint16_t cmds_max,
|
||||
remove_session:
|
||||
iscsi_session_teardown(cls_session);
|
||||
remove_host:
|
||||
iscsi_host_remove(shost);
|
||||
iscsi_host_remove(shost, false);
|
||||
free_host:
|
||||
iscsi_host_free(shost);
|
||||
return NULL;
|
||||
@ -915,7 +954,7 @@ static void iscsi_sw_tcp_session_destroy(struct iscsi_cls_session *cls_session)
|
||||
iscsi_tcp_r2tpool_free(cls_session->dd_data);
|
||||
iscsi_session_teardown(cls_session);
|
||||
|
||||
iscsi_host_remove(shost);
|
||||
iscsi_host_remove(shost, false);
|
||||
iscsi_host_free(shost);
|
||||
}
|
||||
|
||||
@ -1003,7 +1042,6 @@ static struct scsi_host_template iscsi_sw_tcp_sht = {
|
||||
.eh_target_reset_handler = iscsi_eh_recover_target,
|
||||
.dma_boundary = PAGE_SIZE - 1,
|
||||
.slave_configure = iscsi_sw_tcp_slave_configure,
|
||||
.target_alloc = iscsi_target_alloc,
|
||||
.proc_name = "iscsi_tcp",
|
||||
.this_id = -1,
|
||||
.track_queue_depth = 1,
|
||||
|
@ -28,6 +28,8 @@ struct iscsi_sw_tcp_send {
|
||||
|
||||
struct iscsi_sw_tcp_conn {
|
||||
struct socket *sock;
|
||||
struct work_struct recvwork;
|
||||
bool queue_recv;
|
||||
|
||||
struct iscsi_sw_tcp_send out;
|
||||
/* old values for socket callbacks */
|
||||
|
@ -83,7 +83,9 @@ MODULE_PARM_DESC(debug_libiscsi_eh,
|
||||
"%s " dbg_fmt, __func__, ##arg); \
|
||||
} while (0);
|
||||
|
||||
inline void iscsi_conn_queue_work(struct iscsi_conn *conn)
|
||||
#define ISCSI_CMD_COMPL_WAIT 5
|
||||
|
||||
inline void iscsi_conn_queue_xmit(struct iscsi_conn *conn)
|
||||
{
|
||||
struct Scsi_Host *shost = conn->session->host;
|
||||
struct iscsi_host *ihost = shost_priv(shost);
|
||||
@ -91,7 +93,17 @@ inline void iscsi_conn_queue_work(struct iscsi_conn *conn)
|
||||
if (ihost->workq)
|
||||
queue_work(ihost->workq, &conn->xmitwork);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(iscsi_conn_queue_work);
|
||||
EXPORT_SYMBOL_GPL(iscsi_conn_queue_xmit);
|
||||
|
||||
inline void iscsi_conn_queue_recv(struct iscsi_conn *conn)
|
||||
{
|
||||
struct Scsi_Host *shost = conn->session->host;
|
||||
struct iscsi_host *ihost = shost_priv(shost);
|
||||
|
||||
if (ihost->workq && !test_bit(ISCSI_CONN_FLAG_SUSPEND_RX, &conn->flags))
|
||||
queue_work(ihost->workq, &conn->recvwork);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(iscsi_conn_queue_recv);
|
||||
|
||||
static void __iscsi_update_cmdsn(struct iscsi_session *session,
|
||||
uint32_t exp_cmdsn, uint32_t max_cmdsn)
|
||||
@ -472,12 +484,18 @@ static void iscsi_free_task(struct iscsi_task *task)
|
||||
}
|
||||
}
|
||||
|
||||
void __iscsi_get_task(struct iscsi_task *task)
|
||||
bool iscsi_get_task(struct iscsi_task *task)
|
||||
{
|
||||
refcount_inc(&task->refcount);
|
||||
return refcount_inc_not_zero(&task->refcount);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(__iscsi_get_task);
|
||||
EXPORT_SYMBOL_GPL(iscsi_get_task);
|
||||
|
||||
/**
|
||||
* __iscsi_put_task - drop the refcount on a task
|
||||
* @task: iscsi_task to drop the refcount on
|
||||
*
|
||||
* The back_lock must be held when calling in case it frees the task.
|
||||
*/
|
||||
void __iscsi_put_task(struct iscsi_task *task)
|
||||
{
|
||||
if (refcount_dec_and_test(&task->refcount))
|
||||
@ -489,10 +507,11 @@ void iscsi_put_task(struct iscsi_task *task)
|
||||
{
|
||||
struct iscsi_session *session = task->conn->session;
|
||||
|
||||
/* regular RX path uses back_lock */
|
||||
spin_lock_bh(&session->back_lock);
|
||||
__iscsi_put_task(task);
|
||||
spin_unlock_bh(&session->back_lock);
|
||||
if (refcount_dec_and_test(&task->refcount)) {
|
||||
spin_lock_bh(&session->back_lock);
|
||||
iscsi_free_task(task);
|
||||
spin_unlock_bh(&session->back_lock);
|
||||
}
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(iscsi_put_task);
|
||||
|
||||
@ -557,16 +576,19 @@ static bool cleanup_queued_task(struct iscsi_task *task)
|
||||
struct iscsi_conn *conn = task->conn;
|
||||
bool early_complete = false;
|
||||
|
||||
/* Bad target might have completed task while it was still running */
|
||||
/*
|
||||
* We might have raced where we handled a R2T early and got a response
|
||||
* but have not yet taken the task off the requeue list, then a TMF or
|
||||
* recovery happened and so we can still see it here.
|
||||
*/
|
||||
if (task->state == ISCSI_TASK_COMPLETED)
|
||||
early_complete = true;
|
||||
|
||||
if (!list_empty(&task->running)) {
|
||||
list_del_init(&task->running);
|
||||
/*
|
||||
* If it's on a list but still running, this could be from
|
||||
* a bad target sending a rsp early, cleanup from a TMF, or
|
||||
* session recovery.
|
||||
* If it's on a list but still running this could be cleanup
|
||||
* from a TMF or session recovery.
|
||||
*/
|
||||
if (task->state == ISCSI_TASK_RUNNING ||
|
||||
task->state == ISCSI_TASK_COMPLETED)
|
||||
@ -587,20 +609,17 @@ static bool cleanup_queued_task(struct iscsi_task *task)
|
||||
}
|
||||
|
||||
/*
|
||||
* session frwd lock must be held and if not called for a task that is still
|
||||
* pending or from the xmit thread, then xmit thread must be suspended
|
||||
* session back and frwd lock must be held and if not called for a task that
|
||||
* is still pending or from the xmit thread, then xmit thread must be suspended
|
||||
*/
|
||||
static void fail_scsi_task(struct iscsi_task *task, int err)
|
||||
static void __fail_scsi_task(struct iscsi_task *task, int err)
|
||||
{
|
||||
struct iscsi_conn *conn = task->conn;
|
||||
struct scsi_cmnd *sc;
|
||||
int state;
|
||||
|
||||
spin_lock_bh(&conn->session->back_lock);
|
||||
if (cleanup_queued_task(task)) {
|
||||
spin_unlock_bh(&conn->session->back_lock);
|
||||
if (cleanup_queued_task(task))
|
||||
return;
|
||||
}
|
||||
|
||||
if (task->state == ISCSI_TASK_PENDING) {
|
||||
/*
|
||||
@ -619,7 +638,15 @@ static void fail_scsi_task(struct iscsi_task *task, int err)
|
||||
sc->result = err << 16;
|
||||
scsi_set_resid(sc, scsi_bufflen(sc));
|
||||
iscsi_complete_task(task, state);
|
||||
spin_unlock_bh(&conn->session->back_lock);
|
||||
}
|
||||
|
||||
static void fail_scsi_task(struct iscsi_task *task, int err)
|
||||
{
|
||||
struct iscsi_session *session = task->conn->session;
|
||||
|
||||
spin_lock_bh(&session->back_lock);
|
||||
__fail_scsi_task(task, err);
|
||||
spin_unlock_bh(&session->back_lock);
|
||||
}
|
||||
|
||||
static int iscsi_prep_mgmt_task(struct iscsi_conn *conn,
|
||||
@ -668,12 +695,18 @@ static int iscsi_prep_mgmt_task(struct iscsi_conn *conn,
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* iscsi_alloc_mgmt_task - allocate and setup a mgmt task.
|
||||
* @conn: iscsi conn that the task will be sent on.
|
||||
* @hdr: iscsi pdu that will be sent.
|
||||
* @data: buffer for data segment if needed.
|
||||
* @data_size: length of data in bytes.
|
||||
*/
|
||||
static struct iscsi_task *
|
||||
__iscsi_conn_send_pdu(struct iscsi_conn *conn, struct iscsi_hdr *hdr,
|
||||
iscsi_alloc_mgmt_task(struct iscsi_conn *conn, struct iscsi_hdr *hdr,
|
||||
char *data, uint32_t data_size)
|
||||
{
|
||||
struct iscsi_session *session = conn->session;
|
||||
struct iscsi_host *ihost = shost_priv(session->host);
|
||||
uint8_t opcode = hdr->opcode & ISCSI_OPCODE_MASK;
|
||||
struct iscsi_task *task;
|
||||
itt_t itt;
|
||||
@ -754,30 +787,59 @@ __iscsi_conn_send_pdu(struct iscsi_conn *conn, struct iscsi_hdr *hdr,
|
||||
task->conn->session->age);
|
||||
}
|
||||
|
||||
if (unlikely(READ_ONCE(conn->ping_task) == INVALID_SCSI_TASK))
|
||||
WRITE_ONCE(conn->ping_task, task);
|
||||
|
||||
if (!ihost->workq) {
|
||||
if (iscsi_prep_mgmt_task(conn, task))
|
||||
goto free_task;
|
||||
|
||||
if (session->tt->xmit_task(task))
|
||||
goto free_task;
|
||||
} else {
|
||||
list_add_tail(&task->running, &conn->mgmtqueue);
|
||||
iscsi_conn_queue_work(conn);
|
||||
}
|
||||
|
||||
return task;
|
||||
|
||||
free_task:
|
||||
/* regular RX path uses back_lock */
|
||||
spin_lock(&session->back_lock);
|
||||
__iscsi_put_task(task);
|
||||
spin_unlock(&session->back_lock);
|
||||
iscsi_put_task(task);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/**
|
||||
* iscsi_send_mgmt_task - Send task created with iscsi_alloc_mgmt_task.
|
||||
* @task: iscsi task to send.
|
||||
*
|
||||
* On failure this returns a non-zero error code, and the driver must free
|
||||
* the task with iscsi_put_task;
|
||||
*/
|
||||
static int iscsi_send_mgmt_task(struct iscsi_task *task)
|
||||
{
|
||||
struct iscsi_conn *conn = task->conn;
|
||||
struct iscsi_session *session = conn->session;
|
||||
struct iscsi_host *ihost = shost_priv(conn->session->host);
|
||||
int rc = 0;
|
||||
|
||||
if (!ihost->workq) {
|
||||
rc = iscsi_prep_mgmt_task(conn, task);
|
||||
if (rc)
|
||||
return rc;
|
||||
|
||||
rc = session->tt->xmit_task(task);
|
||||
if (rc)
|
||||
return rc;
|
||||
} else {
|
||||
list_add_tail(&task->running, &conn->mgmtqueue);
|
||||
iscsi_conn_queue_xmit(conn);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int __iscsi_conn_send_pdu(struct iscsi_conn *conn, struct iscsi_hdr *hdr,
|
||||
char *data, uint32_t data_size)
|
||||
{
|
||||
struct iscsi_task *task;
|
||||
int rc;
|
||||
|
||||
task = iscsi_alloc_mgmt_task(conn, hdr, data, data_size);
|
||||
if (!task)
|
||||
return -ENOMEM;
|
||||
|
||||
rc = iscsi_send_mgmt_task(task);
|
||||
if (rc)
|
||||
iscsi_put_task(task);
|
||||
return rc;
|
||||
}
|
||||
|
||||
int iscsi_conn_send_pdu(struct iscsi_cls_conn *cls_conn, struct iscsi_hdr *hdr,
|
||||
char *data, uint32_t data_size)
|
||||
{
|
||||
@ -786,7 +848,7 @@ int iscsi_conn_send_pdu(struct iscsi_cls_conn *cls_conn, struct iscsi_hdr *hdr,
|
||||
int err = 0;
|
||||
|
||||
spin_lock_bh(&session->frwd_lock);
|
||||
if (!__iscsi_conn_send_pdu(conn, hdr, data, data_size))
|
||||
if (__iscsi_conn_send_pdu(conn, hdr, data, data_size))
|
||||
err = -EPERM;
|
||||
spin_unlock_bh(&session->frwd_lock);
|
||||
return err;
|
||||
@ -959,7 +1021,6 @@ static int iscsi_send_nopout(struct iscsi_conn *conn, struct iscsi_nopin *rhdr)
|
||||
if (!rhdr) {
|
||||
if (READ_ONCE(conn->ping_task))
|
||||
return -EINVAL;
|
||||
WRITE_ONCE(conn->ping_task, INVALID_SCSI_TASK);
|
||||
}
|
||||
|
||||
memset(&hdr, 0, sizeof(struct iscsi_nopout));
|
||||
@ -973,10 +1034,18 @@ static int iscsi_send_nopout(struct iscsi_conn *conn, struct iscsi_nopin *rhdr)
|
||||
} else
|
||||
hdr.ttt = RESERVED_ITT;
|
||||
|
||||
task = __iscsi_conn_send_pdu(conn, (struct iscsi_hdr *)&hdr, NULL, 0);
|
||||
if (!task) {
|
||||
task = iscsi_alloc_mgmt_task(conn, (struct iscsi_hdr *)&hdr, NULL, 0);
|
||||
if (!task)
|
||||
return -ENOMEM;
|
||||
|
||||
if (!rhdr)
|
||||
WRITE_ONCE(conn->ping_task, task);
|
||||
|
||||
if (iscsi_send_mgmt_task(task)) {
|
||||
if (!rhdr)
|
||||
WRITE_ONCE(conn->ping_task, NULL);
|
||||
iscsi_put_task(task);
|
||||
|
||||
iscsi_conn_printk(KERN_ERR, conn, "Could not send nopout\n");
|
||||
return -EIO;
|
||||
} else if (!rhdr) {
|
||||
@ -1434,11 +1503,17 @@ static int iscsi_xmit_task(struct iscsi_conn *conn, struct iscsi_task *task,
|
||||
{
|
||||
int rc;
|
||||
|
||||
spin_lock_bh(&conn->session->back_lock);
|
||||
|
||||
if (!conn->task) {
|
||||
/* Take a ref so we can access it after xmit_task() */
|
||||
__iscsi_get_task(task);
|
||||
/*
|
||||
* Take a ref so we can access it after xmit_task().
|
||||
*
|
||||
* This should never fail because the failure paths will have
|
||||
* stopped the xmit thread.
|
||||
*/
|
||||
if (!iscsi_get_task(task)) {
|
||||
WARN_ON_ONCE(1);
|
||||
return 0;
|
||||
}
|
||||
} else {
|
||||
/* Already have a ref from when we failed to send it last call */
|
||||
conn->task = NULL;
|
||||
@ -1449,7 +1524,7 @@ static int iscsi_xmit_task(struct iscsi_conn *conn, struct iscsi_task *task,
|
||||
* case a bad target sends a cmd rsp before we have handled the task.
|
||||
*/
|
||||
if (was_requeue)
|
||||
__iscsi_put_task(task);
|
||||
iscsi_put_task(task);
|
||||
|
||||
/*
|
||||
* Do this after dropping the extra ref because if this was a requeue
|
||||
@ -1461,10 +1536,8 @@ static int iscsi_xmit_task(struct iscsi_conn *conn, struct iscsi_task *task,
|
||||
* task and get woken up again.
|
||||
*/
|
||||
conn->task = task;
|
||||
spin_unlock_bh(&conn->session->back_lock);
|
||||
return -ENODATA;
|
||||
}
|
||||
spin_unlock_bh(&conn->session->back_lock);
|
||||
|
||||
spin_unlock_bh(&conn->session->frwd_lock);
|
||||
rc = conn->session->tt->xmit_task(task);
|
||||
@ -1472,20 +1545,16 @@ static int iscsi_xmit_task(struct iscsi_conn *conn, struct iscsi_task *task,
|
||||
if (!rc) {
|
||||
/* done with this task */
|
||||
task->last_xfer = jiffies;
|
||||
}
|
||||
/* regular RX path uses back_lock */
|
||||
spin_lock(&conn->session->back_lock);
|
||||
if (rc && task->state == ISCSI_TASK_RUNNING) {
|
||||
} else {
|
||||
/*
|
||||
* get an extra ref that is released next time we access it
|
||||
* as conn->task above.
|
||||
*/
|
||||
__iscsi_get_task(task);
|
||||
iscsi_get_task(task);
|
||||
conn->task = task;
|
||||
}
|
||||
|
||||
__iscsi_put_task(task);
|
||||
spin_unlock(&conn->session->back_lock);
|
||||
iscsi_put_task(task);
|
||||
return rc;
|
||||
}
|
||||
|
||||
@ -1513,7 +1582,7 @@ void iscsi_requeue_task(struct iscsi_task *task)
|
||||
*/
|
||||
iscsi_put_task(task);
|
||||
}
|
||||
iscsi_conn_queue_work(conn);
|
||||
iscsi_conn_queue_xmit(conn);
|
||||
spin_unlock_bh(&conn->session->frwd_lock);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(iscsi_requeue_task);
|
||||
@ -1786,7 +1855,7 @@ int iscsi_queuecommand(struct Scsi_Host *host, struct scsi_cmnd *sc)
|
||||
}
|
||||
} else {
|
||||
list_add_tail(&task->running, &conn->cmdqueue);
|
||||
iscsi_conn_queue_work(conn);
|
||||
iscsi_conn_queue_xmit(conn);
|
||||
}
|
||||
|
||||
session->queued_cmdsn++;
|
||||
@ -1847,11 +1916,8 @@ static int iscsi_exec_task_mgmt_fn(struct iscsi_conn *conn,
|
||||
__must_hold(&session->frwd_lock)
|
||||
{
|
||||
struct iscsi_session *session = conn->session;
|
||||
struct iscsi_task *task;
|
||||
|
||||
task = __iscsi_conn_send_pdu(conn, (struct iscsi_hdr *)hdr,
|
||||
NULL, 0);
|
||||
if (!task) {
|
||||
if (__iscsi_conn_send_pdu(conn, (struct iscsi_hdr *)hdr, NULL, 0)) {
|
||||
spin_unlock_bh(&session->frwd_lock);
|
||||
iscsi_conn_printk(KERN_ERR, conn, "Could not send TMF.\n");
|
||||
iscsi_conn_failure(conn, ISCSI_ERR_CONN_FAILED);
|
||||
@ -1899,6 +1965,7 @@ static void fail_scsi_tasks(struct iscsi_conn *conn, u64 lun, int error)
|
||||
struct iscsi_task *task;
|
||||
int i;
|
||||
|
||||
restart_cmd_loop:
|
||||
spin_lock_bh(&session->back_lock);
|
||||
for (i = 0; i < session->cmds_max; i++) {
|
||||
task = session->cmds[i];
|
||||
@ -1907,22 +1974,25 @@ static void fail_scsi_tasks(struct iscsi_conn *conn, u64 lun, int error)
|
||||
|
||||
if (lun != -1 && lun != task->sc->device->lun)
|
||||
continue;
|
||||
|
||||
__iscsi_get_task(task);
|
||||
spin_unlock_bh(&session->back_lock);
|
||||
/*
|
||||
* The cmd is completing but if this is called from an eh
|
||||
* callout path then when we return scsi-ml owns the cmd. Wait
|
||||
* for the completion path to finish freeing the cmd.
|
||||
*/
|
||||
if (!iscsi_get_task(task)) {
|
||||
spin_unlock_bh(&session->back_lock);
|
||||
spin_unlock_bh(&session->frwd_lock);
|
||||
udelay(ISCSI_CMD_COMPL_WAIT);
|
||||
spin_lock_bh(&session->frwd_lock);
|
||||
goto restart_cmd_loop;
|
||||
}
|
||||
|
||||
ISCSI_DBG_SESSION(session,
|
||||
"failing sc %p itt 0x%x state %d\n",
|
||||
task->sc, task->itt, task->state);
|
||||
fail_scsi_task(task, error);
|
||||
|
||||
spin_unlock_bh(&session->frwd_lock);
|
||||
iscsi_put_task(task);
|
||||
spin_lock_bh(&session->frwd_lock);
|
||||
|
||||
spin_lock_bh(&session->back_lock);
|
||||
__fail_scsi_task(task, error);
|
||||
__iscsi_put_task(task);
|
||||
}
|
||||
|
||||
spin_unlock_bh(&session->back_lock);
|
||||
}
|
||||
|
||||
@ -1947,7 +2017,7 @@ EXPORT_SYMBOL_GPL(iscsi_suspend_queue);
|
||||
|
||||
/**
|
||||
* iscsi_suspend_tx - suspend iscsi_data_xmit
|
||||
* @conn: iscsi conn tp stop processing IO on.
|
||||
* @conn: iscsi conn to stop processing IO on.
|
||||
*
|
||||
* This function sets the suspend bit to prevent iscsi_data_xmit
|
||||
* from sending new IO, and if work is queued on the xmit thread
|
||||
@ -1960,16 +2030,31 @@ void iscsi_suspend_tx(struct iscsi_conn *conn)
|
||||
|
||||
set_bit(ISCSI_CONN_FLAG_SUSPEND_TX, &conn->flags);
|
||||
if (ihost->workq)
|
||||
flush_workqueue(ihost->workq);
|
||||
flush_work(&conn->xmitwork);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(iscsi_suspend_tx);
|
||||
|
||||
static void iscsi_start_tx(struct iscsi_conn *conn)
|
||||
{
|
||||
clear_bit(ISCSI_CONN_FLAG_SUSPEND_TX, &conn->flags);
|
||||
iscsi_conn_queue_work(conn);
|
||||
iscsi_conn_queue_xmit(conn);
|
||||
}
|
||||
|
||||
/**
|
||||
* iscsi_suspend_rx - Prevent recvwork from running again.
|
||||
* @conn: iscsi conn to stop.
|
||||
*/
|
||||
void iscsi_suspend_rx(struct iscsi_conn *conn)
|
||||
{
|
||||
struct Scsi_Host *shost = conn->session->host;
|
||||
struct iscsi_host *ihost = shost_priv(shost);
|
||||
|
||||
set_bit(ISCSI_CONN_FLAG_SUSPEND_RX, &conn->flags);
|
||||
if (ihost->workq)
|
||||
flush_work(&conn->recvwork);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(iscsi_suspend_rx);
|
||||
|
||||
/*
|
||||
* We want to make sure a ping is in flight. It has timed out.
|
||||
* And we are not busy processing a pdu that is making
|
||||
@ -2012,7 +2097,16 @@ enum blk_eh_timer_return iscsi_eh_cmd_timed_out(struct scsi_cmnd *sc)
|
||||
spin_unlock(&session->back_lock);
|
||||
goto done;
|
||||
}
|
||||
__iscsi_get_task(task);
|
||||
if (!iscsi_get_task(task)) {
|
||||
/*
|
||||
* Racing with the completion path right now, so give it more
|
||||
* time so that path can complete it like normal.
|
||||
*/
|
||||
rc = BLK_EH_RESET_TIMER;
|
||||
task = NULL;
|
||||
spin_unlock(&session->back_lock);
|
||||
goto done;
|
||||
}
|
||||
spin_unlock(&session->back_lock);
|
||||
|
||||
if (session->state != ISCSI_STATE_LOGGED_IN) {
|
||||
@ -2261,6 +2355,7 @@ int iscsi_eh_abort(struct scsi_cmnd *sc)
|
||||
|
||||
ISCSI_DBG_EH(session, "aborting sc %p\n", sc);
|
||||
|
||||
completion_check:
|
||||
mutex_lock(&session->eh_mutex);
|
||||
spin_lock_bh(&session->frwd_lock);
|
||||
/*
|
||||
@ -2300,13 +2395,20 @@ int iscsi_eh_abort(struct scsi_cmnd *sc)
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
if (!iscsi_get_task(task)) {
|
||||
spin_unlock(&session->back_lock);
|
||||
spin_unlock_bh(&session->frwd_lock);
|
||||
mutex_unlock(&session->eh_mutex);
|
||||
/* We are just about to call iscsi_free_task so wait for it. */
|
||||
udelay(ISCSI_CMD_COMPL_WAIT);
|
||||
goto completion_check;
|
||||
}
|
||||
|
||||
ISCSI_DBG_EH(session, "aborting [sc %p itt 0x%x]\n", sc, task->itt);
|
||||
conn = session->leadconn;
|
||||
iscsi_get_conn(conn->cls_conn);
|
||||
conn->eh_abort_cnt++;
|
||||
age = session->age;
|
||||
|
||||
ISCSI_DBG_EH(session, "aborting [sc %p itt 0x%x]\n", sc, task->itt);
|
||||
__iscsi_get_task(task);
|
||||
spin_unlock(&session->back_lock);
|
||||
|
||||
if (task->state == ISCSI_TASK_PENDING) {
|
||||
@ -2832,11 +2934,12 @@ static void iscsi_notify_host_removed(struct iscsi_cls_session *cls_session)
|
||||
/**
|
||||
* iscsi_host_remove - remove host and sessions
|
||||
* @shost: scsi host
|
||||
* @is_shutdown: true if called from a driver shutdown callout
|
||||
*
|
||||
* If there are any sessions left, this will initiate the removal and wait
|
||||
* for the completion.
|
||||
*/
|
||||
void iscsi_host_remove(struct Scsi_Host *shost)
|
||||
void iscsi_host_remove(struct Scsi_Host *shost, bool is_shutdown)
|
||||
{
|
||||
struct iscsi_host *ihost = shost_priv(shost);
|
||||
unsigned long flags;
|
||||
@ -2845,7 +2948,11 @@ void iscsi_host_remove(struct Scsi_Host *shost)
|
||||
ihost->state = ISCSI_HOST_REMOVED;
|
||||
spin_unlock_irqrestore(&ihost->lock, flags);
|
||||
|
||||
iscsi_host_for_each_session(shost, iscsi_notify_host_removed);
|
||||
if (!is_shutdown)
|
||||
iscsi_host_for_each_session(shost, iscsi_notify_host_removed);
|
||||
else
|
||||
iscsi_host_for_each_session(shost, iscsi_force_destroy_session);
|
||||
|
||||
wait_event_interruptible(ihost->session_removal_wq,
|
||||
ihost->num_sessions == 0);
|
||||
if (signal_pending(current))
|
||||
|
@ -558,7 +558,11 @@ static int iscsi_tcp_r2t_rsp(struct iscsi_conn *conn, struct iscsi_hdr *hdr)
|
||||
return 0;
|
||||
}
|
||||
task->last_xfer = jiffies;
|
||||
__iscsi_get_task(task);
|
||||
if (!iscsi_get_task(task)) {
|
||||
spin_unlock(&session->back_lock);
|
||||
/* Let the path that got the early rsp complete it */
|
||||
return 0;
|
||||
}
|
||||
|
||||
tcp_conn = conn->dd_data;
|
||||
rhdr = (struct iscsi_r2t_rsp *)tcp_conn->in.hdr;
|
||||
|
@ -873,7 +873,7 @@ mpt3sas_base_stop_watchdog(struct MPT3SAS_ADAPTER *ioc)
|
||||
* @fault_code: fault code
|
||||
*/
|
||||
void
|
||||
mpt3sas_base_fault_info(struct MPT3SAS_ADAPTER *ioc , u16 fault_code)
|
||||
mpt3sas_base_fault_info(struct MPT3SAS_ADAPTER *ioc, u16 fault_code)
|
||||
{
|
||||
ioc_err(ioc, "fault_state(0x%04x)!\n", fault_code);
|
||||
}
|
||||
@ -1057,7 +1057,7 @@ _base_sas_ioc_info(struct MPT3SAS_ADAPTER *ioc, MPI2DefaultReply_t *mpi_reply,
|
||||
desc = "config no defaults";
|
||||
break;
|
||||
case MPI2_IOCSTATUS_CONFIG_CANT_COMMIT:
|
||||
desc = "config cant commit";
|
||||
desc = "config can't commit";
|
||||
break;
|
||||
|
||||
/****************************************************************************
|
||||
@ -1321,7 +1321,7 @@ _base_display_event_data(struct MPT3SAS_ADAPTER *ioc,
|
||||
* @log_info: log info
|
||||
*/
|
||||
static void
|
||||
_base_sas_log_info(struct MPT3SAS_ADAPTER *ioc , u32 log_info)
|
||||
_base_sas_log_info(struct MPT3SAS_ADAPTER *ioc, u32 log_info)
|
||||
{
|
||||
union loginfo_type {
|
||||
u32 loginfo;
|
||||
@ -1393,7 +1393,7 @@ _base_display_reply_info(struct MPT3SAS_ADAPTER *ioc, u16 smid, u8 msix_index,
|
||||
|
||||
if ((ioc_status & MPI2_IOCSTATUS_MASK) &&
|
||||
(ioc->logging_level & MPT_DEBUG_REPLY)) {
|
||||
_base_sas_ioc_info(ioc , mpi_reply,
|
||||
_base_sas_ioc_info(ioc, mpi_reply,
|
||||
mpt3sas_base_get_msg_frame(ioc, smid));
|
||||
}
|
||||
|
||||
|
@ -5294,7 +5294,7 @@ _scsih_normalize_sense(char *sense_buffer, struct sense_info *data)
|
||||
}
|
||||
|
||||
/**
|
||||
* _scsih_scsi_ioc_info - translated non-succesfull SCSI_IO request
|
||||
* _scsih_scsi_ioc_info - translated non-successful SCSI_IO request
|
||||
* @ioc: per adapter object
|
||||
* @scmd: pointer to scsi command object
|
||||
* @mpi_reply: reply mf payload returned from firmware
|
||||
@ -12409,7 +12409,6 @@ scsih_suspend(struct device *dev)
|
||||
return rc;
|
||||
|
||||
mpt3sas_base_stop_watchdog(ioc);
|
||||
flush_scheduled_work();
|
||||
scsi_block_requests(shost);
|
||||
_scsih_nvme_shutdown(ioc);
|
||||
ioc_info(ioc, "pdev=0x%p, slot=%s, entering operating state\n",
|
||||
|
@ -2414,9 +2414,12 @@ static void __qedi_remove(struct pci_dev *pdev, int mode)
|
||||
int rval;
|
||||
u16 retry = 10;
|
||||
|
||||
if (mode == QEDI_MODE_NORMAL || mode == QEDI_MODE_SHUTDOWN) {
|
||||
iscsi_host_remove(qedi->shost);
|
||||
if (mode == QEDI_MODE_NORMAL)
|
||||
iscsi_host_remove(qedi->shost, false);
|
||||
else if (mode == QEDI_MODE_SHUTDOWN)
|
||||
iscsi_host_remove(qedi->shost, true);
|
||||
|
||||
if (mode == QEDI_MODE_NORMAL || mode == QEDI_MODE_SHUTDOWN) {
|
||||
if (qedi->tmf_thread) {
|
||||
destroy_workqueue(qedi->tmf_thread);
|
||||
qedi->tmf_thread = NULL;
|
||||
@ -2491,7 +2494,7 @@ static void qedi_board_disable_work(struct work_struct *work)
|
||||
if (test_and_set_bit(QEDI_IN_SHUTDOWN, &qedi->flags))
|
||||
return;
|
||||
|
||||
__qedi_remove(qedi->pdev, QEDI_MODE_SHUTDOWN);
|
||||
__qedi_remove(qedi->pdev, QEDI_MODE_NORMAL);
|
||||
}
|
||||
|
||||
static void qedi_shutdown(struct pci_dev *pdev)
|
||||
@ -2791,7 +2794,7 @@ static int __qedi_probe(struct pci_dev *pdev, int mode)
|
||||
#ifdef CONFIG_DEBUG_FS
|
||||
qedi_dbg_host_exit(&qedi->dbg_ctx);
|
||||
#endif
|
||||
iscsi_host_remove(qedi->shost);
|
||||
iscsi_host_remove(qedi->shost, false);
|
||||
stop_iscsi_func:
|
||||
qedi_ops->stop(qedi->cdev);
|
||||
stop_slowpath:
|
||||
|
@ -2143,8 +2143,6 @@ static int iscsi_iter_destroy_conn_fn(struct device *dev, void *data)
|
||||
return 0;
|
||||
|
||||
iscsi_remove_conn(iscsi_dev_to_conn(dev));
|
||||
iscsi_put_conn(iscsi_dev_to_conn(dev));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -2264,17 +2262,19 @@ static void iscsi_if_disconnect_bound_ep(struct iscsi_cls_conn *conn,
|
||||
}
|
||||
}
|
||||
|
||||
static int iscsi_if_stop_conn(struct iscsi_transport *transport,
|
||||
struct iscsi_uevent *ev)
|
||||
static int iscsi_if_stop_conn(struct iscsi_cls_conn *conn, int flag)
|
||||
{
|
||||
int flag = ev->u.stop_conn.flag;
|
||||
struct iscsi_cls_conn *conn;
|
||||
|
||||
conn = iscsi_conn_lookup(ev->u.stop_conn.sid, ev->u.stop_conn.cid);
|
||||
if (!conn)
|
||||
return -EINVAL;
|
||||
|
||||
ISCSI_DBG_TRANS_CONN(conn, "iscsi if conn stop.\n");
|
||||
/*
|
||||
* For offload, iscsid may not know about the ep like when iscsid is
|
||||
* restarted or for kernel based session shutdown iscsid is not even
|
||||
* up. For these cases, we do the disconnect now.
|
||||
*/
|
||||
mutex_lock(&conn->ep_mutex);
|
||||
if (conn->ep)
|
||||
iscsi_if_disconnect_bound_ep(conn, conn->ep, true);
|
||||
mutex_unlock(&conn->ep_mutex);
|
||||
|
||||
/*
|
||||
* If this is a termination we have to call stop_conn with that flag
|
||||
* so the correct states get set. If we haven't run the work yet try to
|
||||
@ -2284,16 +2284,6 @@ static int iscsi_if_stop_conn(struct iscsi_transport *transport,
|
||||
cancel_work_sync(&conn->cleanup_work);
|
||||
iscsi_stop_conn(conn, flag);
|
||||
} else {
|
||||
/*
|
||||
* For offload, when iscsid is restarted it won't know about
|
||||
* existing endpoints so it can't do a ep_disconnect. We clean
|
||||
* it up here for userspace.
|
||||
*/
|
||||
mutex_lock(&conn->ep_mutex);
|
||||
if (conn->ep)
|
||||
iscsi_if_disconnect_bound_ep(conn, conn->ep, true);
|
||||
mutex_unlock(&conn->ep_mutex);
|
||||
|
||||
/*
|
||||
* Figure out if it was the kernel or userspace initiating this.
|
||||
*/
|
||||
@ -2349,6 +2339,55 @@ static void iscsi_cleanup_conn_work_fn(struct work_struct *work)
|
||||
ISCSI_DBG_TRANS_CONN(conn, "cleanup done.\n");
|
||||
}
|
||||
|
||||
static int iscsi_iter_force_destroy_conn_fn(struct device *dev, void *data)
|
||||
{
|
||||
struct iscsi_transport *transport;
|
||||
struct iscsi_cls_conn *conn;
|
||||
|
||||
if (!iscsi_is_conn_dev(dev))
|
||||
return 0;
|
||||
|
||||
conn = iscsi_dev_to_conn(dev);
|
||||
transport = conn->transport;
|
||||
|
||||
if (READ_ONCE(conn->state) != ISCSI_CONN_DOWN)
|
||||
iscsi_if_stop_conn(conn, STOP_CONN_TERM);
|
||||
|
||||
transport->destroy_conn(conn);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* iscsi_force_destroy_session - destroy a session from the kernel
|
||||
* @session: session to destroy
|
||||
*
|
||||
* Force the destruction of a session from the kernel. This should only be
|
||||
* used when userspace is no longer running during system shutdown.
|
||||
*/
|
||||
void iscsi_force_destroy_session(struct iscsi_cls_session *session)
|
||||
{
|
||||
struct iscsi_transport *transport = session->transport;
|
||||
unsigned long flags;
|
||||
|
||||
WARN_ON_ONCE(system_state == SYSTEM_RUNNING);
|
||||
|
||||
spin_lock_irqsave(&sesslock, flags);
|
||||
if (list_empty(&session->sess_list)) {
|
||||
spin_unlock_irqrestore(&sesslock, flags);
|
||||
/*
|
||||
* Conn/ep is already freed. Session is being torn down via
|
||||
* async path. For shutdown we don't care about it so return.
|
||||
*/
|
||||
return;
|
||||
}
|
||||
spin_unlock_irqrestore(&sesslock, flags);
|
||||
|
||||
device_for_each_child(&session->dev, NULL,
|
||||
iscsi_iter_force_destroy_conn_fn);
|
||||
transport->destroy_session(session);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(iscsi_force_destroy_session);
|
||||
|
||||
void iscsi_free_session(struct iscsi_cls_session *session)
|
||||
{
|
||||
ISCSI_DBG_TRANS_SESSION(session, "Freeing session\n");
|
||||
@ -3720,7 +3759,12 @@ static int iscsi_if_transport_conn(struct iscsi_transport *transport,
|
||||
case ISCSI_UEVENT_DESTROY_CONN:
|
||||
return iscsi_if_destroy_conn(transport, ev);
|
||||
case ISCSI_UEVENT_STOP_CONN:
|
||||
return iscsi_if_stop_conn(transport, ev);
|
||||
conn = iscsi_conn_lookup(ev->u.stop_conn.sid,
|
||||
ev->u.stop_conn.cid);
|
||||
if (!conn)
|
||||
return -EINVAL;
|
||||
|
||||
return iscsi_if_stop_conn(conn, ev->u.stop_conn.flag);
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -145,7 +145,7 @@ struct snic_exch_ver_req {
|
||||
* HBA Capabilities
|
||||
* Bit 1: Reserved.
|
||||
* Bit 2: Dynamic Discovery of LUNs.
|
||||
* Bit 3: Async event notifications on on tgt online/offline events.
|
||||
* Bit 3: Async event notifications on tgt online/offline events.
|
||||
* Bit 4: IO timeout support in FW.
|
||||
* Bit 5-31: Reserved.
|
||||
*/
|
||||
|
@ -3598,7 +3598,7 @@ static void sym_sir_task_recovery(struct sym_hcb *np, int num)
|
||||
}
|
||||
|
||||
/*
|
||||
* Gerard's alchemy:) that deals with with the data
|
||||
* Gerard's alchemy:) that deals with the data
|
||||
* pointer for both MDP and the residual calculation.
|
||||
*
|
||||
* I didn't want to bloat the code by more than 200
|
||||
|
@ -385,7 +385,7 @@ target_emulate_set_target_port_groups(struct se_cmd *cmd)
|
||||
|
||||
/*
|
||||
* Extract the RELATIVE TARGET PORT IDENTIFIER to identify
|
||||
* the Target Port in question for the the incoming
|
||||
* the Target Port in question for the incoming
|
||||
* SET_TARGET_PORT_GROUPS op.
|
||||
*/
|
||||
rtpi = get_unaligned_be16(ptr + 2);
|
||||
|
@ -215,7 +215,7 @@ static inline void ufshcd_vops_config_scaling_param(struct ufs_hba *hba,
|
||||
hba->vops->config_scaling_param(hba, p, data);
|
||||
}
|
||||
|
||||
extern struct ufs_pm_lvl_states ufs_pm_lvl_states[];
|
||||
extern const struct ufs_pm_lvl_states ufs_pm_lvl_states[];
|
||||
|
||||
/**
|
||||
* ufshcd_scsi_to_upiu_lun - maps scsi LUN to UPIU LUN
|
||||
@ -234,8 +234,8 @@ static inline u8 ufshcd_scsi_to_upiu_lun(unsigned int scsi_lun)
|
||||
|
||||
int __ufshcd_write_ee_control(struct ufs_hba *hba, u32 ee_ctrl_mask);
|
||||
int ufshcd_write_ee_control(struct ufs_hba *hba);
|
||||
int ufshcd_update_ee_control(struct ufs_hba *hba, u16 *mask, u16 *other_mask,
|
||||
u16 set, u16 clr);
|
||||
int ufshcd_update_ee_control(struct ufs_hba *hba, u16 *mask,
|
||||
const u16 *other_mask, u16 set, u16 clr);
|
||||
|
||||
static inline int ufshcd_update_ee_drv_mask(struct ufs_hba *hba,
|
||||
u16 set, u16 clr)
|
||||
|
@ -175,7 +175,7 @@ enum {
|
||||
#define ufshcd_clear_eh_in_progress(h) \
|
||||
((h)->eh_flags &= ~UFSHCD_EH_IN_PROGRESS)
|
||||
|
||||
struct ufs_pm_lvl_states ufs_pm_lvl_states[] = {
|
||||
const struct ufs_pm_lvl_states ufs_pm_lvl_states[] = {
|
||||
[UFS_PM_LVL_0] = {UFS_ACTIVE_PWR_MODE, UIC_LINK_ACTIVE_STATE},
|
||||
[UFS_PM_LVL_1] = {UFS_ACTIVE_PWR_MODE, UIC_LINK_HIBERN8_STATE},
|
||||
[UFS_PM_LVL_2] = {UFS_SLEEP_PWR_MODE, UIC_LINK_ACTIVE_STATE},
|
||||
@ -363,7 +363,7 @@ static void ufshcd_add_tm_upiu_trace(struct ufs_hba *hba, unsigned int tag,
|
||||
}
|
||||
|
||||
static void ufshcd_add_uic_command_trace(struct ufs_hba *hba,
|
||||
struct uic_command *ucmd,
|
||||
const struct uic_command *ucmd,
|
||||
enum ufs_trace_str_t str_t)
|
||||
{
|
||||
u32 cmd;
|
||||
@ -443,11 +443,11 @@ static void ufshcd_print_clk_freqs(struct ufs_hba *hba)
|
||||
}
|
||||
|
||||
static void ufshcd_print_evt(struct ufs_hba *hba, u32 id,
|
||||
char *err_name)
|
||||
const char *err_name)
|
||||
{
|
||||
int i;
|
||||
bool found = false;
|
||||
struct ufs_event_hist *e;
|
||||
const struct ufs_event_hist *e;
|
||||
|
||||
if (id >= UFS_EVT_CNT)
|
||||
return;
|
||||
@ -497,7 +497,7 @@ static void ufshcd_print_evt_hist(struct ufs_hba *hba)
|
||||
static
|
||||
void ufshcd_print_trs(struct ufs_hba *hba, unsigned long bitmap, bool pr_prdt)
|
||||
{
|
||||
struct ufshcd_lrb *lrbp;
|
||||
const struct ufshcd_lrb *lrbp;
|
||||
int prdt_length;
|
||||
int tag;
|
||||
|
||||
@ -553,7 +553,7 @@ static void ufshcd_print_tmrs(struct ufs_hba *hba, unsigned long bitmap)
|
||||
|
||||
static void ufshcd_print_host_state(struct ufs_hba *hba)
|
||||
{
|
||||
struct scsi_device *sdev_ufs = hba->ufs_device_wlun;
|
||||
const struct scsi_device *sdev_ufs = hba->ufs_device_wlun;
|
||||
|
||||
dev_err(hba->dev, "UFS Host state=%d\n", hba->ufshcd_state);
|
||||
dev_err(hba->dev, "outstanding reqs=0x%lx tasks=0x%lx\n",
|
||||
@ -1109,7 +1109,7 @@ static bool ufshcd_is_devfreq_scaling_required(struct ufs_hba *hba,
|
||||
*/
|
||||
static u32 ufshcd_pending_cmds(struct ufs_hba *hba)
|
||||
{
|
||||
struct scsi_device *sdev;
|
||||
const struct scsi_device *sdev;
|
||||
u32 pending = 0;
|
||||
|
||||
lockdep_assert_held(hba->host->host_lock);
|
||||
@ -2080,14 +2080,15 @@ static inline int ufshcd_monitor_opcode2dir(u8 opcode)
|
||||
static inline bool ufshcd_should_inform_monitor(struct ufs_hba *hba,
|
||||
struct ufshcd_lrb *lrbp)
|
||||
{
|
||||
struct ufs_hba_monitor *m = &hba->monitor;
|
||||
const struct ufs_hba_monitor *m = &hba->monitor;
|
||||
|
||||
return (m->enabled && lrbp && lrbp->cmd &&
|
||||
(!m->chunk_size || m->chunk_size == lrbp->cmd->sdb.length) &&
|
||||
ktime_before(hba->monitor.enabled_ts, lrbp->issue_time_stamp));
|
||||
}
|
||||
|
||||
static void ufshcd_start_monitor(struct ufs_hba *hba, struct ufshcd_lrb *lrbp)
|
||||
static void ufshcd_start_monitor(struct ufs_hba *hba,
|
||||
const struct ufshcd_lrb *lrbp)
|
||||
{
|
||||
int dir = ufshcd_monitor_opcode2dir(*lrbp->cmd->cmnd);
|
||||
unsigned long flags;
|
||||
@ -2098,14 +2099,14 @@ static void ufshcd_start_monitor(struct ufs_hba *hba, struct ufshcd_lrb *lrbp)
|
||||
spin_unlock_irqrestore(hba->host->host_lock, flags);
|
||||
}
|
||||
|
||||
static void ufshcd_update_monitor(struct ufs_hba *hba, struct ufshcd_lrb *lrbp)
|
||||
static void ufshcd_update_monitor(struct ufs_hba *hba, const struct ufshcd_lrb *lrbp)
|
||||
{
|
||||
int dir = ufshcd_monitor_opcode2dir(*lrbp->cmd->cmnd);
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(hba->host->host_lock, flags);
|
||||
if (dir >= 0 && hba->monitor.nr_queued[dir] > 0) {
|
||||
struct request *req = scsi_cmd_to_rq(lrbp->cmd);
|
||||
const struct request *req = scsi_cmd_to_rq(lrbp->cmd);
|
||||
struct ufs_hba_monitor *m = &hba->monitor;
|
||||
ktime_t now, inc, lat;
|
||||
|
||||
@ -3075,7 +3076,7 @@ static int ufshcd_query_flag_retry(struct ufs_hba *hba,
|
||||
|
||||
if (ret)
|
||||
dev_err(hba->dev,
|
||||
"%s: query attribute, opcode %d, idn %d, failed with error %d after %d retires\n",
|
||||
"%s: query attribute, opcode %d, idn %d, failed with error %d after %d retries\n",
|
||||
__func__, opcode, idn, ret, retries);
|
||||
return ret;
|
||||
}
|
||||
@ -3243,7 +3244,7 @@ int ufshcd_query_attr_retry(struct ufs_hba *hba,
|
||||
|
||||
if (ret)
|
||||
dev_err(hba->dev,
|
||||
"%s: query attribute, idn %d, failed with error %d after %d retires\n",
|
||||
"%s: query attribute, idn %d, failed with error %d after %d retries\n",
|
||||
__func__, idn, ret, QUERY_REQ_RETRIES);
|
||||
return ret;
|
||||
}
|
||||
@ -4917,7 +4918,7 @@ static int ufshcd_get_lu_wp(struct ufs_hba *hba,
|
||||
*
|
||||
*/
|
||||
static inline void ufshcd_get_lu_power_on_wp_status(struct ufs_hba *hba,
|
||||
struct scsi_device *sdev)
|
||||
const struct scsi_device *sdev)
|
||||
{
|
||||
if (hba->dev_info.f_power_on_wp_en &&
|
||||
!hba->dev_info.is_lu_power_on_wp) {
|
||||
@ -5436,8 +5437,8 @@ int ufshcd_write_ee_control(struct ufs_hba *hba)
|
||||
return err;
|
||||
}
|
||||
|
||||
int ufshcd_update_ee_control(struct ufs_hba *hba, u16 *mask, u16 *other_mask,
|
||||
u16 set, u16 clr)
|
||||
int ufshcd_update_ee_control(struct ufs_hba *hba, u16 *mask,
|
||||
const u16 *other_mask, u16 set, u16 clr)
|
||||
{
|
||||
u16 new_mask, ee_ctrl_mask;
|
||||
int err = 0;
|
||||
@ -7374,7 +7375,8 @@ static int ufshcd_eh_host_reset_handler(struct scsi_cmnd *cmd)
|
||||
*
|
||||
* Returns calculated max ICC level for specific regulator
|
||||
*/
|
||||
static u32 ufshcd_get_max_icc_level(int sup_curr_uA, u32 start_scan, char *buff)
|
||||
static u32 ufshcd_get_max_icc_level(int sup_curr_uA, u32 start_scan,
|
||||
const char *buff)
|
||||
{
|
||||
int i;
|
||||
int curr_uA;
|
||||
@ -7421,7 +7423,7 @@ static u32 ufshcd_get_max_icc_level(int sup_curr_uA, u32 start_scan, char *buff)
|
||||
* Returns calculated ICC level
|
||||
*/
|
||||
static u32 ufshcd_find_max_sup_active_icc_level(struct ufs_hba *hba,
|
||||
u8 *desc_buf, int len)
|
||||
const u8 *desc_buf, int len)
|
||||
{
|
||||
u32 icc_level = 0;
|
||||
|
||||
@ -7571,7 +7573,7 @@ static int ufshcd_scsi_add_wlus(struct ufs_hba *hba)
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void ufshcd_wb_probe(struct ufs_hba *hba, u8 *desc_buf)
|
||||
static void ufshcd_wb_probe(struct ufs_hba *hba, const u8 *desc_buf)
|
||||
{
|
||||
struct ufs_dev_info *dev_info = &hba->dev_info;
|
||||
u8 lun;
|
||||
@ -7642,7 +7644,7 @@ static void ufshcd_wb_probe(struct ufs_hba *hba, u8 *desc_buf)
|
||||
hba->caps &= ~UFSHCD_CAP_WB_EN;
|
||||
}
|
||||
|
||||
static void ufshcd_temp_notif_probe(struct ufs_hba *hba, u8 *desc_buf)
|
||||
static void ufshcd_temp_notif_probe(struct ufs_hba *hba, const u8 *desc_buf)
|
||||
{
|
||||
struct ufs_dev_info *dev_info = &hba->dev_info;
|
||||
u32 ext_ufs_feature;
|
||||
@ -7876,7 +7878,7 @@ static int ufshcd_quirk_tune_host_pa_tactivate(struct ufs_hba *hba)
|
||||
u32 granularity, peer_granularity;
|
||||
u32 pa_tactivate, peer_pa_tactivate;
|
||||
u32 pa_tactivate_us, peer_pa_tactivate_us;
|
||||
u8 gran_to_us_table[] = {1, 4, 8, 16, 32, 100};
|
||||
static const u8 gran_to_us_table[] = {1, 4, 8, 16, 32, 100};
|
||||
|
||||
ret = ufshcd_dme_get(hba, UIC_ARG_MIB(PA_GRANULARITY),
|
||||
&granularity);
|
||||
@ -7993,7 +7995,7 @@ struct ufs_ref_clk {
|
||||
enum ufs_ref_clk_freq val;
|
||||
};
|
||||
|
||||
static struct ufs_ref_clk ufs_ref_clk_freqs[] = {
|
||||
static const struct ufs_ref_clk ufs_ref_clk_freqs[] = {
|
||||
{19200000, REF_CLK_FREQ_19_2_MHZ},
|
||||
{26000000, REF_CLK_FREQ_26_MHZ},
|
||||
{38400000, REF_CLK_FREQ_38_4_MHZ},
|
||||
|
@ -16,6 +16,7 @@
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/phy/phy.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/pm_qos.h>
|
||||
#include <linux/regulator/consumer.h>
|
||||
#include <linux/reset.h>
|
||||
#include <linux/sched/clock.h>
|
||||
@ -31,9 +32,10 @@
|
||||
#include "ufs-mediatek-trace.h"
|
||||
|
||||
static const struct ufs_dev_quirk ufs_mtk_dev_fixups[] = {
|
||||
{ .wmanufacturerid = UFS_VENDOR_MICRON,
|
||||
{ .wmanufacturerid = UFS_ANY_VENDOR,
|
||||
.model = UFS_ANY_MODEL,
|
||||
.quirk = UFS_DEVICE_QUIRK_DELAY_AFTER_LPM },
|
||||
.quirk = UFS_DEVICE_QUIRK_DELAY_AFTER_LPM |
|
||||
UFS_DEVICE_QUIRK_DELAY_BEFORE_LPM },
|
||||
{ .wmanufacturerid = UFS_VENDOR_SKHYNIX,
|
||||
.model = "H9HQ21AFAMZDAR",
|
||||
.quirk = UFS_DEVICE_QUIRK_SUPPORT_EXTENDED_FEATURES },
|
||||
@ -182,6 +184,14 @@ static int ufs_mtk_hce_enable_notify(struct ufs_hba *hba,
|
||||
hba->capabilities &= ~MASK_AUTO_HIBERN8_SUPPORT;
|
||||
hba->ahit = 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Turn on CLK_CG early to bypass abnormal ERR_CHK signal
|
||||
* to prevent host hang issue
|
||||
*/
|
||||
ufshcd_writel(hba,
|
||||
ufshcd_readl(hba, REG_UFS_XOUFS_CTRL) | 0x80,
|
||||
REG_UFS_XOUFS_CTRL);
|
||||
}
|
||||
|
||||
return 0;
|
||||
@ -235,8 +245,9 @@ static int ufs_mtk_setup_ref_clk(struct ufs_hba *hba, bool on)
|
||||
if (host->ref_clk_enabled == on)
|
||||
return 0;
|
||||
|
||||
ufs_mtk_ref_clk_notify(on, PRE_CHANGE, res);
|
||||
|
||||
if (on) {
|
||||
ufs_mtk_ref_clk_notify(on, res);
|
||||
ufshcd_writel(hba, REFCLK_REQUEST, REG_UFS_REFCLK_CTRL);
|
||||
} else {
|
||||
ufshcd_delay_us(host->ref_clk_gating_wait_us, 10);
|
||||
@ -258,7 +269,7 @@ static int ufs_mtk_setup_ref_clk(struct ufs_hba *hba, bool on)
|
||||
|
||||
dev_err(hba->dev, "missing ack of refclk req, reg: 0x%x\n", value);
|
||||
|
||||
ufs_mtk_ref_clk_notify(host->ref_clk_enabled, res);
|
||||
ufs_mtk_ref_clk_notify(host->ref_clk_enabled, POST_CHANGE, res);
|
||||
|
||||
return -ETIMEDOUT;
|
||||
|
||||
@ -266,8 +277,8 @@ static int ufs_mtk_setup_ref_clk(struct ufs_hba *hba, bool on)
|
||||
host->ref_clk_enabled = on;
|
||||
if (on)
|
||||
ufshcd_delay_us(host->ref_clk_ungating_wait_us, 10);
|
||||
else
|
||||
ufs_mtk_ref_clk_notify(on, res);
|
||||
|
||||
ufs_mtk_ref_clk_notify(on, POST_CHANGE, res);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -576,17 +587,32 @@ static void ufs_mtk_init_host_caps(struct ufs_hba *hba)
|
||||
dev_info(hba->dev, "caps: 0x%x", host->caps);
|
||||
}
|
||||
|
||||
static void ufs_mtk_scale_perf(struct ufs_hba *hba, bool up)
|
||||
static void ufs_mtk_boost_pm_qos(struct ufs_hba *hba, bool boost)
|
||||
{
|
||||
struct ufs_mtk_host *host = ufshcd_get_variant(hba);
|
||||
|
||||
ufs_mtk_boost_crypt(hba, up);
|
||||
ufs_mtk_setup_ref_clk(hba, up);
|
||||
if (!host || !host->pm_qos_init)
|
||||
return;
|
||||
|
||||
if (up)
|
||||
cpu_latency_qos_update_request(&host->pm_qos_req,
|
||||
boost ? 0 : PM_QOS_DEFAULT_VALUE);
|
||||
}
|
||||
|
||||
static void ufs_mtk_pwr_ctrl(struct ufs_hba *hba, bool on)
|
||||
{
|
||||
struct ufs_mtk_host *host = ufshcd_get_variant(hba);
|
||||
|
||||
if (on) {
|
||||
phy_power_on(host->mphy);
|
||||
else
|
||||
ufs_mtk_setup_ref_clk(hba, on);
|
||||
ufs_mtk_boost_crypt(hba, on);
|
||||
ufs_mtk_boost_pm_qos(hba, on);
|
||||
} else {
|
||||
ufs_mtk_boost_pm_qos(hba, on);
|
||||
ufs_mtk_boost_crypt(hba, on);
|
||||
ufs_mtk_setup_ref_clk(hba, on);
|
||||
phy_power_off(host->mphy);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
@ -631,9 +657,9 @@ static int ufs_mtk_setup_clocks(struct ufs_hba *hba, bool on,
|
||||
}
|
||||
|
||||
if (clk_pwr_off)
|
||||
ufs_mtk_scale_perf(hba, false);
|
||||
ufs_mtk_pwr_ctrl(hba, false);
|
||||
} else if (on && status == POST_CHANGE) {
|
||||
ufs_mtk_scale_perf(hba, true);
|
||||
ufs_mtk_pwr_ctrl(hba, true);
|
||||
}
|
||||
|
||||
return ret;
|
||||
@ -685,7 +711,7 @@ static int ufs_mtk_vreg_fix_vcc(struct ufs_hba *hba)
|
||||
if (of_property_read_bool(np, "mediatek,ufs-vcc-by-num")) {
|
||||
ufs_mtk_get_vcc_num(res);
|
||||
if (res.a1 > UFS_VCC_NONE && res.a1 < UFS_VCC_MAX)
|
||||
snprintf(vcc_name, MAX_VCC_NAME, "vcc-opt%u", res.a1);
|
||||
snprintf(vcc_name, MAX_VCC_NAME, "vcc-opt%lu", res.a1);
|
||||
else
|
||||
return -ENODEV;
|
||||
} else if (of_property_read_bool(np, "mediatek,ufs-vcc-by-ver")) {
|
||||
@ -732,6 +758,7 @@ static void ufs_mtk_vreg_fix_vccqx(struct ufs_hba *hba)
|
||||
regulator_disable((*vreg_off)->reg);
|
||||
devm_kfree(hba->dev, (*vreg_off)->name);
|
||||
devm_kfree(hba->dev, *vreg_off);
|
||||
*vreg_off = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
@ -1069,6 +1096,11 @@ static int ufs_mtk_link_set_lpm(struct ufs_hba *hba)
|
||||
{
|
||||
int err;
|
||||
|
||||
/* Disable reset confirm feature by UniPro */
|
||||
ufshcd_writel(hba,
|
||||
(ufshcd_readl(hba, REG_UFS_XOUFS_CTRL) & ~0x100),
|
||||
REG_UFS_XOUFS_CTRL);
|
||||
|
||||
err = ufs_mtk_unipro_set_lpm(hba, true);
|
||||
if (err) {
|
||||
/* Resume UniPro state for following error recovery */
|
||||
@ -1175,6 +1207,8 @@ static int ufs_mtk_suspend(struct ufs_hba *hba, enum ufs_pm_op pm_op,
|
||||
if (ufshcd_is_link_off(hba))
|
||||
ufs_mtk_device_reset_ctrl(0, res);
|
||||
|
||||
ufs_mtk_host_pwr_ctrl(HOST_PWR_HCI, false, res);
|
||||
|
||||
return 0;
|
||||
fail:
|
||||
/*
|
||||
@ -1189,10 +1223,13 @@ static int ufs_mtk_suspend(struct ufs_hba *hba, enum ufs_pm_op pm_op,
|
||||
static int ufs_mtk_resume(struct ufs_hba *hba, enum ufs_pm_op pm_op)
|
||||
{
|
||||
int err;
|
||||
struct arm_smccc_res res;
|
||||
|
||||
if (hba->ufshcd_state != UFSHCD_STATE_OPERATIONAL)
|
||||
ufs_mtk_dev_vreg_set_lpm(hba, false);
|
||||
|
||||
ufs_mtk_host_pwr_ctrl(HOST_PWR_HCI, true, res);
|
||||
|
||||
err = ufs_mtk_mphy_power_on(hba, true);
|
||||
if (err)
|
||||
goto fail;
|
||||
@ -1228,8 +1265,10 @@ static int ufs_mtk_apply_dev_quirks(struct ufs_hba *hba)
|
||||
struct ufs_dev_info *dev_info = &hba->dev_info;
|
||||
u16 mid = dev_info->wmanufacturerid;
|
||||
|
||||
if (mid == UFS_VENDOR_SAMSUNG)
|
||||
if (mid == UFS_VENDOR_SAMSUNG) {
|
||||
ufshcd_dme_set(hba, UIC_ARG_MIB(PA_TACTIVATE), 6);
|
||||
ufshcd_dme_set(hba, UIC_ARG_MIB(PA_HIBERN8TIME), 10);
|
||||
}
|
||||
|
||||
/*
|
||||
* Decide waiting time before gating reference clock and
|
||||
@ -1363,7 +1402,8 @@ static int ufs_mtk_remove(struct platform_device *pdev)
|
||||
return 0;
|
||||
}
|
||||
|
||||
int ufs_mtk_system_suspend(struct device *dev)
|
||||
#ifdef CONFIG_PM_SLEEP
|
||||
static int ufs_mtk_system_suspend(struct device *dev)
|
||||
{
|
||||
struct ufs_hba *hba = dev_get_drvdata(dev);
|
||||
int ret;
|
||||
@ -1377,7 +1417,7 @@ int ufs_mtk_system_suspend(struct device *dev)
|
||||
return 0;
|
||||
}
|
||||
|
||||
int ufs_mtk_system_resume(struct device *dev)
|
||||
static int ufs_mtk_system_resume(struct device *dev)
|
||||
{
|
||||
struct ufs_hba *hba = dev_get_drvdata(dev);
|
||||
|
||||
@ -1385,8 +1425,9 @@ int ufs_mtk_system_resume(struct device *dev)
|
||||
|
||||
return ufshcd_system_resume(dev);
|
||||
}
|
||||
#endif
|
||||
|
||||
int ufs_mtk_runtime_suspend(struct device *dev)
|
||||
static int ufs_mtk_runtime_suspend(struct device *dev)
|
||||
{
|
||||
struct ufs_hba *hba = dev_get_drvdata(dev);
|
||||
int ret = 0;
|
||||
@ -1400,7 +1441,7 @@ int ufs_mtk_runtime_suspend(struct device *dev)
|
||||
return 0;
|
||||
}
|
||||
|
||||
int ufs_mtk_runtime_resume(struct device *dev)
|
||||
static int ufs_mtk_runtime_resume(struct device *dev)
|
||||
{
|
||||
struct ufs_hba *hba = dev_get_drvdata(dev);
|
||||
|
||||
|
@ -7,11 +7,13 @@
|
||||
#define _UFS_MEDIATEK_H
|
||||
|
||||
#include <linux/bitops.h>
|
||||
#include <linux/pm_qos.h>
|
||||
#include <linux/soc/mediatek/mtk_sip_svc.h>
|
||||
|
||||
/*
|
||||
* Vendor specific UFSHCI Registers
|
||||
*/
|
||||
#define REG_UFS_XOUFS_CTRL 0x140
|
||||
#define REG_UFS_REFCLK_CTRL 0x144
|
||||
#define REG_UFS_EXTREG 0x2100
|
||||
#define REG_UFS_MPHYCTRL 0x2200
|
||||
@ -83,6 +85,7 @@ enum {
|
||||
#define UFS_MTK_SIP_DEVICE_RESET BIT(1)
|
||||
#define UFS_MTK_SIP_CRYPTO_CTRL BIT(2)
|
||||
#define UFS_MTK_SIP_REF_CLK_NOTIFICATION BIT(3)
|
||||
#define UFS_MTK_SIP_HOST_PWR_CTRL BIT(5)
|
||||
#define UFS_MTK_SIP_GET_VCC_NUM BIT(6)
|
||||
#define UFS_MTK_SIP_DEVICE_PWR_CTRL BIT(7)
|
||||
|
||||
@ -129,6 +132,7 @@ struct ufs_mtk_hw_ver {
|
||||
|
||||
struct ufs_mtk_host {
|
||||
struct phy *mphy;
|
||||
struct pm_qos_request pm_qos_req;
|
||||
struct regulator *reg_va09;
|
||||
struct reset_control *hci_reset;
|
||||
struct reset_control *unipro_reset;
|
||||
@ -138,6 +142,7 @@ struct ufs_mtk_host {
|
||||
struct ufs_mtk_hw_ver hw_ver;
|
||||
enum ufs_mtk_host_caps caps;
|
||||
bool mphy_powered_on;
|
||||
bool pm_qos_init;
|
||||
bool unipro_lpm;
|
||||
bool ref_clk_enabled;
|
||||
u16 ref_clk_ungating_wait_us;
|
||||
@ -155,6 +160,14 @@ enum ufs_mtk_vcc_num {
|
||||
UFS_VCC_MAX
|
||||
};
|
||||
|
||||
/*
|
||||
* Host Power Control options
|
||||
*/
|
||||
enum {
|
||||
HOST_PWR_HCI = 0,
|
||||
HOST_PWR_MPHY
|
||||
};
|
||||
|
||||
/*
|
||||
* SMC call wrapper function
|
||||
*/
|
||||
@ -188,12 +201,15 @@ static void _ufs_mtk_smc(struct ufs_mtk_smc_arg s)
|
||||
#define ufs_mtk_crypto_ctrl(res, enable) \
|
||||
ufs_mtk_smc(UFS_MTK_SIP_CRYPTO_CTRL, &(res), enable)
|
||||
|
||||
#define ufs_mtk_ref_clk_notify(on, res) \
|
||||
ufs_mtk_smc(UFS_MTK_SIP_REF_CLK_NOTIFICATION, &(res), on)
|
||||
#define ufs_mtk_ref_clk_notify(on, stage, res) \
|
||||
ufs_mtk_smc(UFS_MTK_SIP_REF_CLK_NOTIFICATION, &(res), on, stage)
|
||||
|
||||
#define ufs_mtk_device_reset_ctrl(high, res) \
|
||||
ufs_mtk_smc(UFS_MTK_SIP_DEVICE_RESET, &(res), high)
|
||||
|
||||
#define ufs_mtk_host_pwr_ctrl(opt, on, res) \
|
||||
ufs_mtk_smc(UFS_MTK_SIP_HOST_PWR_CTRL, &(res), opt, on)
|
||||
|
||||
#define ufs_mtk_get_vcc_num(res) \
|
||||
ufs_mtk_smc(UFS_MTK_SIP_GET_VCC_NUM, &(res))
|
||||
|
||||
|
@ -58,19 +58,6 @@ static void ufs_qcom_dump_regs_wrapper(struct ufs_hba *hba, int offset, int len,
|
||||
ufshcd_dump_regs(hba, offset, len * 4, prefix);
|
||||
}
|
||||
|
||||
static int ufs_qcom_get_connected_tx_lanes(struct ufs_hba *hba, u32 *tx_lanes)
|
||||
{
|
||||
int err = 0;
|
||||
|
||||
err = ufshcd_dme_get(hba,
|
||||
UIC_ARG_MIB(PA_CONNECTEDTXDATALANES), tx_lanes);
|
||||
if (err)
|
||||
dev_err(hba->dev, "%s: couldn't read PA_CONNECTEDTXDATALANES %d\n",
|
||||
__func__, err);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static int ufs_qcom_host_clk_get(struct device *dev,
|
||||
const char *name, struct clk **clk_out, bool optional)
|
||||
{
|
||||
@ -194,13 +181,6 @@ static int ufs_qcom_init_lane_clks(struct ufs_qcom_host *host)
|
||||
return err;
|
||||
}
|
||||
|
||||
static int ufs_qcom_link_startup_post_change(struct ufs_hba *hba)
|
||||
{
|
||||
u32 tx_lanes;
|
||||
|
||||
return ufs_qcom_get_connected_tx_lanes(hba, &tx_lanes);
|
||||
}
|
||||
|
||||
static int ufs_qcom_check_hibern8(struct ufs_hba *hba)
|
||||
{
|
||||
int err;
|
||||
@ -569,9 +549,6 @@ static int ufs_qcom_link_startup_notify(struct ufs_hba *hba,
|
||||
if (ufshcd_get_local_unipro_ver(hba) != UFS_UNIPRO_VER_1_41)
|
||||
err = ufshcd_disable_host_tx_lcc(hba);
|
||||
|
||||
break;
|
||||
case POST_CHANGE:
|
||||
ufs_qcom_link_startup_post_change(hba);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
|
@ -26,7 +26,7 @@ static int ufshcd_parse_clock_info(struct ufs_hba *hba)
|
||||
int i;
|
||||
struct device *dev = hba->dev;
|
||||
struct device_node *np = dev->of_node;
|
||||
char *name;
|
||||
const char *name;
|
||||
u32 *clkfreq = NULL;
|
||||
struct ufs_clk_info *clki;
|
||||
int len = 0;
|
||||
@ -79,8 +79,8 @@ static int ufshcd_parse_clock_info(struct ufs_hba *hba)
|
||||
}
|
||||
|
||||
for (i = 0; i < sz; i += 2) {
|
||||
ret = of_property_read_string_index(np,
|
||||
"clock-names", i/2, (const char **)&name);
|
||||
ret = of_property_read_string_index(np, "clock-names", i/2,
|
||||
&name);
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
@ -209,8 +209,8 @@ static void ufshcd_init_lanes_per_dir(struct ufs_hba *hba)
|
||||
*
|
||||
* Returns 0 on success, non-zero value on failure
|
||||
*/
|
||||
int ufshcd_get_pwr_dev_param(struct ufs_dev_params *pltfrm_param,
|
||||
struct ufs_pa_layer_attr *dev_max,
|
||||
int ufshcd_get_pwr_dev_param(const struct ufs_dev_params *pltfrm_param,
|
||||
const struct ufs_pa_layer_attr *dev_max,
|
||||
struct ufs_pa_layer_attr *agreed_pwr)
|
||||
{
|
||||
int min_pltfrm_gear;
|
||||
|
@ -25,8 +25,8 @@ struct ufs_dev_params {
|
||||
u32 desired_working_mode;
|
||||
};
|
||||
|
||||
int ufshcd_get_pwr_dev_param(struct ufs_dev_params *dev_param,
|
||||
struct ufs_pa_layer_attr *dev_max,
|
||||
int ufshcd_get_pwr_dev_param(const struct ufs_dev_params *dev_param,
|
||||
const struct ufs_pa_layer_attr *dev_max,
|
||||
struct ufs_pa_layer_attr *agreed_pwr);
|
||||
void ufshcd_init_pwr_dev_param(struct ufs_dev_params *dev_param);
|
||||
int ufshcd_pltfrm_init(struct platform_device *pdev,
|
||||
|
@ -135,9 +135,6 @@ struct iscsi_task {
|
||||
void *dd_data; /* driver/transport data */
|
||||
};
|
||||
|
||||
/* invalid scsi_task pointer */
|
||||
#define INVALID_SCSI_TASK (struct iscsi_task *)-1l
|
||||
|
||||
static inline int iscsi_task_has_unsol_data(struct iscsi_task *task)
|
||||
{
|
||||
return task->unsol_r2t.data_length > task->unsol_r2t.sent;
|
||||
@ -213,6 +210,8 @@ struct iscsi_conn {
|
||||
struct list_head cmdqueue; /* data-path cmd queue */
|
||||
struct list_head requeue; /* tasks needing another run */
|
||||
struct work_struct xmitwork; /* per-conn. xmit workqueue */
|
||||
/* recv */
|
||||
struct work_struct recvwork;
|
||||
unsigned long flags; /* ISCSI_CONN_FLAGs */
|
||||
|
||||
/* negotiated params */
|
||||
@ -411,7 +410,7 @@ extern int iscsi_host_add(struct Scsi_Host *shost, struct device *pdev);
|
||||
extern struct Scsi_Host *iscsi_host_alloc(struct scsi_host_template *sht,
|
||||
int dd_data_size,
|
||||
bool xmit_can_sleep);
|
||||
extern void iscsi_host_remove(struct Scsi_Host *shost);
|
||||
extern void iscsi_host_remove(struct Scsi_Host *shost, bool is_shutdown);
|
||||
extern void iscsi_host_free(struct Scsi_Host *shost);
|
||||
extern int iscsi_target_alloc(struct scsi_target *starget);
|
||||
extern int iscsi_host_get_max_scsi_cmds(struct Scsi_Host *shost,
|
||||
@ -452,8 +451,10 @@ extern int iscsi_conn_get_param(struct iscsi_cls_conn *cls_conn,
|
||||
extern int iscsi_conn_get_addr_param(struct sockaddr_storage *addr,
|
||||
enum iscsi_param param, char *buf);
|
||||
extern void iscsi_suspend_tx(struct iscsi_conn *conn);
|
||||
extern void iscsi_suspend_rx(struct iscsi_conn *conn);
|
||||
extern void iscsi_suspend_queue(struct iscsi_conn *conn);
|
||||
extern void iscsi_conn_queue_work(struct iscsi_conn *conn);
|
||||
extern void iscsi_conn_queue_xmit(struct iscsi_conn *conn);
|
||||
extern void iscsi_conn_queue_recv(struct iscsi_conn *conn);
|
||||
|
||||
#define iscsi_conn_printk(prefix, _c, fmt, a...) \
|
||||
iscsi_cls_conn_printk(prefix, ((struct iscsi_conn *)_c)->cls_conn, \
|
||||
@ -478,7 +479,7 @@ extern struct iscsi_task *iscsi_itt_to_task(struct iscsi_conn *, itt_t);
|
||||
extern void iscsi_requeue_task(struct iscsi_task *task);
|
||||
extern void iscsi_put_task(struct iscsi_task *task);
|
||||
extern void __iscsi_put_task(struct iscsi_task *task);
|
||||
extern void __iscsi_get_task(struct iscsi_task *task);
|
||||
extern bool iscsi_get_task(struct iscsi_task *task);
|
||||
extern void iscsi_complete_scsi_task(struct iscsi_task *task,
|
||||
uint32_t exp_cmdsn, uint32_t max_cmdsn);
|
||||
|
||||
|
@ -442,6 +442,7 @@ extern struct iscsi_cls_session *iscsi_create_session(struct Scsi_Host *shost,
|
||||
struct iscsi_transport *t,
|
||||
int dd_size,
|
||||
unsigned int target_id);
|
||||
extern void iscsi_force_destroy_session(struct iscsi_cls_session *session);
|
||||
extern void iscsi_remove_session(struct iscsi_cls_session *session);
|
||||
extern void iscsi_free_session(struct iscsi_cls_session *session);
|
||||
extern struct iscsi_cls_conn *iscsi_alloc_conn(struct iscsi_cls_session *sess,
|
||||
|
@ -166,6 +166,8 @@ TRACE_EVENT(scsi_dispatch_cmd_start,
|
||||
__field( unsigned int, lun )
|
||||
__field( unsigned int, opcode )
|
||||
__field( unsigned int, cmd_len )
|
||||
__field( int, driver_tag)
|
||||
__field( int, scheduler_tag)
|
||||
__field( unsigned int, data_sglen )
|
||||
__field( unsigned int, prot_sglen )
|
||||
__field( unsigned char, prot_op )
|
||||
@ -179,6 +181,8 @@ TRACE_EVENT(scsi_dispatch_cmd_start,
|
||||
__entry->lun = cmd->device->lun;
|
||||
__entry->opcode = cmd->cmnd[0];
|
||||
__entry->cmd_len = cmd->cmd_len;
|
||||
__entry->driver_tag = scsi_cmd_to_rq(cmd)->tag;
|
||||
__entry->scheduler_tag = scsi_cmd_to_rq(cmd)->internal_tag;
|
||||
__entry->data_sglen = scsi_sg_count(cmd);
|
||||
__entry->prot_sglen = scsi_prot_sg_count(cmd);
|
||||
__entry->prot_op = scsi_get_prot_op(cmd);
|
||||
@ -186,11 +190,11 @@ TRACE_EVENT(scsi_dispatch_cmd_start,
|
||||
),
|
||||
|
||||
TP_printk("host_no=%u channel=%u id=%u lun=%u data_sgl=%u prot_sgl=%u" \
|
||||
" prot_op=%s cmnd=(%s %s raw=%s)",
|
||||
" prot_op=%s driver_tag=%d scheduler_tag=%d cmnd=(%s %s raw=%s)",
|
||||
__entry->host_no, __entry->channel, __entry->id,
|
||||
__entry->lun, __entry->data_sglen, __entry->prot_sglen,
|
||||
show_prot_op_name(__entry->prot_op),
|
||||
show_opcode_name(__entry->opcode),
|
||||
show_prot_op_name(__entry->prot_op), __entry->driver_tag,
|
||||
__entry->scheduler_tag, show_opcode_name(__entry->opcode),
|
||||
__parse_cdb(__get_dynamic_array(cmnd), __entry->cmd_len),
|
||||
__print_hex(__get_dynamic_array(cmnd), __entry->cmd_len))
|
||||
);
|
||||
@ -209,6 +213,8 @@ TRACE_EVENT(scsi_dispatch_cmd_error,
|
||||
__field( int, rtn )
|
||||
__field( unsigned int, opcode )
|
||||
__field( unsigned int, cmd_len )
|
||||
__field( int, driver_tag)
|
||||
__field( int, scheduler_tag)
|
||||
__field( unsigned int, data_sglen )
|
||||
__field( unsigned int, prot_sglen )
|
||||
__field( unsigned char, prot_op )
|
||||
@ -223,6 +229,8 @@ TRACE_EVENT(scsi_dispatch_cmd_error,
|
||||
__entry->rtn = rtn;
|
||||
__entry->opcode = cmd->cmnd[0];
|
||||
__entry->cmd_len = cmd->cmd_len;
|
||||
__entry->driver_tag = scsi_cmd_to_rq(cmd)->tag;
|
||||
__entry->scheduler_tag = scsi_cmd_to_rq(cmd)->internal_tag;
|
||||
__entry->data_sglen = scsi_sg_count(cmd);
|
||||
__entry->prot_sglen = scsi_prot_sg_count(cmd);
|
||||
__entry->prot_op = scsi_get_prot_op(cmd);
|
||||
@ -230,11 +238,12 @@ TRACE_EVENT(scsi_dispatch_cmd_error,
|
||||
),
|
||||
|
||||
TP_printk("host_no=%u channel=%u id=%u lun=%u data_sgl=%u prot_sgl=%u" \
|
||||
" prot_op=%s cmnd=(%s %s raw=%s) rtn=%d",
|
||||
" prot_op=%s driver_tag=%d scheduler_tag=%d cmnd=(%s %s raw=%s)" \
|
||||
" rtn=%d",
|
||||
__entry->host_no, __entry->channel, __entry->id,
|
||||
__entry->lun, __entry->data_sglen, __entry->prot_sglen,
|
||||
show_prot_op_name(__entry->prot_op),
|
||||
show_opcode_name(__entry->opcode),
|
||||
show_prot_op_name(__entry->prot_op), __entry->driver_tag,
|
||||
__entry->scheduler_tag, show_opcode_name(__entry->opcode),
|
||||
__parse_cdb(__get_dynamic_array(cmnd), __entry->cmd_len),
|
||||
__print_hex(__get_dynamic_array(cmnd), __entry->cmd_len),
|
||||
__entry->rtn)
|
||||
@ -254,6 +263,8 @@ DECLARE_EVENT_CLASS(scsi_cmd_done_timeout_template,
|
||||
__field( int, result )
|
||||
__field( unsigned int, opcode )
|
||||
__field( unsigned int, cmd_len )
|
||||
__field( int, driver_tag)
|
||||
__field( int, scheduler_tag)
|
||||
__field( unsigned int, data_sglen )
|
||||
__field( unsigned int, prot_sglen )
|
||||
__field( unsigned char, prot_op )
|
||||
@ -268,19 +279,21 @@ DECLARE_EVENT_CLASS(scsi_cmd_done_timeout_template,
|
||||
__entry->result = cmd->result;
|
||||
__entry->opcode = cmd->cmnd[0];
|
||||
__entry->cmd_len = cmd->cmd_len;
|
||||
__entry->driver_tag = scsi_cmd_to_rq(cmd)->tag;
|
||||
__entry->scheduler_tag = scsi_cmd_to_rq(cmd)->internal_tag;
|
||||
__entry->data_sglen = scsi_sg_count(cmd);
|
||||
__entry->prot_sglen = scsi_prot_sg_count(cmd);
|
||||
__entry->prot_op = scsi_get_prot_op(cmd);
|
||||
memcpy(__get_dynamic_array(cmnd), cmd->cmnd, cmd->cmd_len);
|
||||
),
|
||||
|
||||
TP_printk("host_no=%u channel=%u id=%u lun=%u data_sgl=%u " \
|
||||
"prot_sgl=%u prot_op=%s cmnd=(%s %s raw=%s) result=(driver=" \
|
||||
"%s host=%s message=%s status=%s)",
|
||||
TP_printk("host_no=%u channel=%u id=%u lun=%u data_sgl=%u prot_sgl=%u " \
|
||||
"prot_op=%s driver_tag=%d scheduler_tag=%d cmnd=(%s %s raw=%s) " \
|
||||
"result=(driver=%s host=%s message=%s status=%s)",
|
||||
__entry->host_no, __entry->channel, __entry->id,
|
||||
__entry->lun, __entry->data_sglen, __entry->prot_sglen,
|
||||
show_prot_op_name(__entry->prot_op),
|
||||
show_opcode_name(__entry->opcode),
|
||||
show_prot_op_name(__entry->prot_op), __entry->driver_tag,
|
||||
__entry->scheduler_tag, show_opcode_name(__entry->opcode),
|
||||
__parse_cdb(__get_dynamic_array(cmnd), __entry->cmd_len),
|
||||
__print_hex(__get_dynamic_array(cmnd), __entry->cmd_len),
|
||||
"DRIVER_OK",
|
||||
|
@ -1232,14 +1232,14 @@ static inline int ufshcd_vops_phy_initialization(struct ufs_hba *hba)
|
||||
return 0;
|
||||
}
|
||||
|
||||
extern struct ufs_pm_lvl_states ufs_pm_lvl_states[];
|
||||
extern const struct ufs_pm_lvl_states ufs_pm_lvl_states[];
|
||||
|
||||
int ufshcd_dump_regs(struct ufs_hba *hba, size_t offset, size_t len,
|
||||
const char *prefix);
|
||||
|
||||
int __ufshcd_write_ee_control(struct ufs_hba *hba, u32 ee_ctrl_mask);
|
||||
int ufshcd_write_ee_control(struct ufs_hba *hba);
|
||||
int ufshcd_update_ee_control(struct ufs_hba *hba, u16 *mask, u16 *other_mask,
|
||||
u16 set, u16 clr);
|
||||
int ufshcd_update_ee_control(struct ufs_hba *hba, u16 *mask,
|
||||
const u16 *other_mask, u16 set, u16 clr);
|
||||
|
||||
#endif /* End of Header */
|
||||
|
Loading…
Reference in New Issue
Block a user