162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci// Copyright 2014 Cisco Systems, Inc.  All rights reserved.
362306a36Sopenharmony_ci
462306a36Sopenharmony_ci#include <linux/errno.h>
562306a36Sopenharmony_ci#include <linux/pci.h>
662306a36Sopenharmony_ci#include <linux/slab.h>
762306a36Sopenharmony_ci
862306a36Sopenharmony_ci#include <linux/interrupt.h>
962306a36Sopenharmony_ci#include <linux/workqueue.h>
1062306a36Sopenharmony_ci#include <linux/spinlock.h>
1162306a36Sopenharmony_ci#include <linux/mempool.h>
1262306a36Sopenharmony_ci#include <scsi/scsi_tcq.h>
1362306a36Sopenharmony_ci#include <linux/ctype.h>
1462306a36Sopenharmony_ci
1562306a36Sopenharmony_ci#include "snic_io.h"
1662306a36Sopenharmony_ci#include "snic.h"
1762306a36Sopenharmony_ci#include "cq_enet_desc.h"
1862306a36Sopenharmony_ci#include "snic_fwint.h"
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_ci/*
2162306a36Sopenharmony_ci * snic_handle_link : Handles link flaps.
2262306a36Sopenharmony_ci */
2362306a36Sopenharmony_civoid
2462306a36Sopenharmony_cisnic_handle_link(struct work_struct *work)
2562306a36Sopenharmony_ci{
2662306a36Sopenharmony_ci	struct snic *snic = container_of(work, struct snic, link_work);
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci	if (snic->config.xpt_type == SNIC_DAS)
2962306a36Sopenharmony_ci		return;
3062306a36Sopenharmony_ci
3162306a36Sopenharmony_ci	snic->link_status = svnic_dev_link_status(snic->vdev);
3262306a36Sopenharmony_ci	snic->link_down_cnt = svnic_dev_link_down_cnt(snic->vdev);
3362306a36Sopenharmony_ci	SNIC_HOST_INFO(snic->shost, "Link Event: Link %s.\n",
3462306a36Sopenharmony_ci		       ((snic->link_status) ? "Up" : "Down"));
3562306a36Sopenharmony_ci
3662306a36Sopenharmony_ci	SNIC_ASSERT_NOT_IMPL(1);
3762306a36Sopenharmony_ci}
3862306a36Sopenharmony_ci
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_ci/*
4162306a36Sopenharmony_ci * snic_ver_enc : Encodes version str to int
4262306a36Sopenharmony_ci * version string is similar to netmask string
4362306a36Sopenharmony_ci */
4462306a36Sopenharmony_cistatic int
4562306a36Sopenharmony_cisnic_ver_enc(const char *s)
4662306a36Sopenharmony_ci{
4762306a36Sopenharmony_ci	int v[4] = {0};
4862306a36Sopenharmony_ci	int  i = 0, x = 0;
4962306a36Sopenharmony_ci	char c;
5062306a36Sopenharmony_ci	const char *p = s;
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci	/* validate version string */
5362306a36Sopenharmony_ci	if ((strlen(s) > 15) || (strlen(s) < 7))
5462306a36Sopenharmony_ci		goto end;
5562306a36Sopenharmony_ci
5662306a36Sopenharmony_ci	while ((c = *p++)) {
5762306a36Sopenharmony_ci		if (c == '.') {
5862306a36Sopenharmony_ci			i++;
5962306a36Sopenharmony_ci			continue;
6062306a36Sopenharmony_ci		}
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_ci		if (i > 3 || !isdigit(c))
6362306a36Sopenharmony_ci			goto end;
6462306a36Sopenharmony_ci
6562306a36Sopenharmony_ci		v[i] = v[i] * 10 + (c - '0');
6662306a36Sopenharmony_ci	}
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_ci	/* validate sub version numbers */
6962306a36Sopenharmony_ci	for (i = 3; i >= 0; i--)
7062306a36Sopenharmony_ci		if (v[i] > 0xff)
7162306a36Sopenharmony_ci			goto end;
7262306a36Sopenharmony_ci
7362306a36Sopenharmony_ci	x |= (v[0] << 24) | v[1] << 16 | v[2] << 8 | v[3];
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ciend:
7662306a36Sopenharmony_ci	if (x == 0) {
7762306a36Sopenharmony_ci		SNIC_ERR("Invalid version string [%s].\n", s);
7862306a36Sopenharmony_ci
7962306a36Sopenharmony_ci		return -1;
8062306a36Sopenharmony_ci	}
8162306a36Sopenharmony_ci
8262306a36Sopenharmony_ci	return x;
8362306a36Sopenharmony_ci} /* end of snic_ver_enc */
8462306a36Sopenharmony_ci
8562306a36Sopenharmony_ci/*
8662306a36Sopenharmony_ci * snic_qeueue_exch_ver_req :
8762306a36Sopenharmony_ci *
8862306a36Sopenharmony_ci * Queues Exchange Version Request, to communicate host information
8962306a36Sopenharmony_ci * in return, it gets firmware version details
9062306a36Sopenharmony_ci */
9162306a36Sopenharmony_ciint
9262306a36Sopenharmony_cisnic_queue_exch_ver_req(struct snic *snic)
9362306a36Sopenharmony_ci{
9462306a36Sopenharmony_ci	struct snic_req_info *rqi = NULL;
9562306a36Sopenharmony_ci	struct snic_host_req *req = NULL;
9662306a36Sopenharmony_ci	u32 ver = 0;
9762306a36Sopenharmony_ci	int ret = 0;
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ci	SNIC_HOST_INFO(snic->shost, "Exch Ver Req Preparing...\n");
10062306a36Sopenharmony_ci
10162306a36Sopenharmony_ci	rqi = snic_req_init(snic, 0);
10262306a36Sopenharmony_ci	if (!rqi) {
10362306a36Sopenharmony_ci		SNIC_HOST_ERR(snic->shost, "Init Exch Ver Req failed\n");
10462306a36Sopenharmony_ci		ret = -ENOMEM;
10562306a36Sopenharmony_ci		goto error;
10662306a36Sopenharmony_ci	}
10762306a36Sopenharmony_ci
10862306a36Sopenharmony_ci	req = rqi_to_req(rqi);
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_ci	/* Initialize snic_host_req */
11162306a36Sopenharmony_ci	snic_io_hdr_enc(&req->hdr, SNIC_REQ_EXCH_VER, 0, SCSI_NO_TAG,
11262306a36Sopenharmony_ci			snic->config.hid, 0, (ulong)rqi);
11362306a36Sopenharmony_ci	ver = snic_ver_enc(SNIC_DRV_VERSION);
11462306a36Sopenharmony_ci	req->u.exch_ver.drvr_ver = cpu_to_le32(ver);
11562306a36Sopenharmony_ci	req->u.exch_ver.os_type = cpu_to_le32(SNIC_OS_LINUX);
11662306a36Sopenharmony_ci
11762306a36Sopenharmony_ci	snic_handle_untagged_req(snic, rqi);
11862306a36Sopenharmony_ci
11962306a36Sopenharmony_ci	ret = snic_queue_wq_desc(snic, req, sizeof(*req));
12062306a36Sopenharmony_ci	if (ret) {
12162306a36Sopenharmony_ci		snic_release_untagged_req(snic, rqi);
12262306a36Sopenharmony_ci		SNIC_HOST_ERR(snic->shost,
12362306a36Sopenharmony_ci			      "Queuing Exch Ver Req failed, err = %d\n",
12462306a36Sopenharmony_ci			      ret);
12562306a36Sopenharmony_ci		goto error;
12662306a36Sopenharmony_ci	}
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci	SNIC_HOST_INFO(snic->shost, "Exch Ver Req is issued. ret = %d\n", ret);
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_cierror:
13162306a36Sopenharmony_ci	return ret;
13262306a36Sopenharmony_ci} /* end of snic_queue_exch_ver_req */
13362306a36Sopenharmony_ci
13462306a36Sopenharmony_ci/*
13562306a36Sopenharmony_ci * snic_io_exch_ver_cmpl_handler
13662306a36Sopenharmony_ci */
13762306a36Sopenharmony_civoid
13862306a36Sopenharmony_cisnic_io_exch_ver_cmpl_handler(struct snic *snic, struct snic_fw_req *fwreq)
13962306a36Sopenharmony_ci{
14062306a36Sopenharmony_ci	struct snic_req_info *rqi = NULL;
14162306a36Sopenharmony_ci	struct snic_exch_ver_rsp *exv_cmpl = &fwreq->u.exch_ver_cmpl;
14262306a36Sopenharmony_ci	u8 typ, hdr_stat;
14362306a36Sopenharmony_ci	u32 cmnd_id, hid, max_sgs;
14462306a36Sopenharmony_ci	ulong ctx = 0;
14562306a36Sopenharmony_ci	unsigned long flags;
14662306a36Sopenharmony_ci
14762306a36Sopenharmony_ci	SNIC_HOST_INFO(snic->shost, "Exch Ver Compl Received.\n");
14862306a36Sopenharmony_ci	snic_io_hdr_dec(&fwreq->hdr, &typ, &hdr_stat, &cmnd_id, &hid, &ctx);
14962306a36Sopenharmony_ci	SNIC_BUG_ON(snic->config.hid != hid);
15062306a36Sopenharmony_ci	rqi = (struct snic_req_info *) ctx;
15162306a36Sopenharmony_ci
15262306a36Sopenharmony_ci	if (hdr_stat) {
15362306a36Sopenharmony_ci		SNIC_HOST_ERR(snic->shost,
15462306a36Sopenharmony_ci			      "Exch Ver Completed w/ err status %d\n",
15562306a36Sopenharmony_ci			      hdr_stat);
15662306a36Sopenharmony_ci
15762306a36Sopenharmony_ci		goto exch_cmpl_end;
15862306a36Sopenharmony_ci	}
15962306a36Sopenharmony_ci
16062306a36Sopenharmony_ci	spin_lock_irqsave(&snic->snic_lock, flags);
16162306a36Sopenharmony_ci	snic->fwinfo.fw_ver = le32_to_cpu(exv_cmpl->version);
16262306a36Sopenharmony_ci	snic->fwinfo.hid = le32_to_cpu(exv_cmpl->hid);
16362306a36Sopenharmony_ci	snic->fwinfo.max_concur_ios = le32_to_cpu(exv_cmpl->max_concur_ios);
16462306a36Sopenharmony_ci	snic->fwinfo.max_sgs_per_cmd = le32_to_cpu(exv_cmpl->max_sgs_per_cmd);
16562306a36Sopenharmony_ci	snic->fwinfo.max_io_sz = le32_to_cpu(exv_cmpl->max_io_sz);
16662306a36Sopenharmony_ci	snic->fwinfo.max_tgts = le32_to_cpu(exv_cmpl->max_tgts);
16762306a36Sopenharmony_ci	snic->fwinfo.io_tmo = le16_to_cpu(exv_cmpl->io_timeout);
16862306a36Sopenharmony_ci
16962306a36Sopenharmony_ci	SNIC_HOST_INFO(snic->shost,
17062306a36Sopenharmony_ci		       "vers %u hid %u max_concur_ios %u max_sgs_per_cmd %u max_io_sz %u max_tgts %u fw tmo %u\n",
17162306a36Sopenharmony_ci		       snic->fwinfo.fw_ver,
17262306a36Sopenharmony_ci		       snic->fwinfo.hid,
17362306a36Sopenharmony_ci		       snic->fwinfo.max_concur_ios,
17462306a36Sopenharmony_ci		       snic->fwinfo.max_sgs_per_cmd,
17562306a36Sopenharmony_ci		       snic->fwinfo.max_io_sz,
17662306a36Sopenharmony_ci		       snic->fwinfo.max_tgts,
17762306a36Sopenharmony_ci		       snic->fwinfo.io_tmo);
17862306a36Sopenharmony_ci
17962306a36Sopenharmony_ci	SNIC_HOST_INFO(snic->shost,
18062306a36Sopenharmony_ci		       "HBA Capabilities = 0x%x\n",
18162306a36Sopenharmony_ci		       le32_to_cpu(exv_cmpl->hba_cap));
18262306a36Sopenharmony_ci
18362306a36Sopenharmony_ci	/* Updating SGList size */
18462306a36Sopenharmony_ci	max_sgs = snic->fwinfo.max_sgs_per_cmd;
18562306a36Sopenharmony_ci	if (max_sgs && max_sgs < SNIC_MAX_SG_DESC_CNT) {
18662306a36Sopenharmony_ci		snic->shost->sg_tablesize = max_sgs;
18762306a36Sopenharmony_ci		SNIC_HOST_INFO(snic->shost, "Max SGs set to %d\n",
18862306a36Sopenharmony_ci			       snic->shost->sg_tablesize);
18962306a36Sopenharmony_ci	} else if (max_sgs > snic->shost->sg_tablesize) {
19062306a36Sopenharmony_ci		SNIC_HOST_INFO(snic->shost,
19162306a36Sopenharmony_ci			       "Target type %d Supports Larger Max SGList %d than driver's Max SG List %d.\n",
19262306a36Sopenharmony_ci			       snic->config.xpt_type, max_sgs,
19362306a36Sopenharmony_ci			       snic->shost->sg_tablesize);
19462306a36Sopenharmony_ci	}
19562306a36Sopenharmony_ci
19662306a36Sopenharmony_ci	if (snic->shost->can_queue > snic->fwinfo.max_concur_ios)
19762306a36Sopenharmony_ci		snic->shost->can_queue = snic->fwinfo.max_concur_ios;
19862306a36Sopenharmony_ci
19962306a36Sopenharmony_ci	snic->shost->max_sectors = snic->fwinfo.max_io_sz >> 9;
20062306a36Sopenharmony_ci	if (snic->fwinfo.wait)
20162306a36Sopenharmony_ci		complete(snic->fwinfo.wait);
20262306a36Sopenharmony_ci
20362306a36Sopenharmony_ci	spin_unlock_irqrestore(&snic->snic_lock, flags);
20462306a36Sopenharmony_ci
20562306a36Sopenharmony_ciexch_cmpl_end:
20662306a36Sopenharmony_ci	snic_release_untagged_req(snic, rqi);
20762306a36Sopenharmony_ci
20862306a36Sopenharmony_ci	SNIC_HOST_INFO(snic->shost, "Exch_cmpl Done, hdr_stat %d.\n", hdr_stat);
20962306a36Sopenharmony_ci} /* end of snic_io_exch_ver_cmpl_handler */
21062306a36Sopenharmony_ci
21162306a36Sopenharmony_ci/*
21262306a36Sopenharmony_ci * snic_get_conf
21362306a36Sopenharmony_ci *
21462306a36Sopenharmony_ci * Synchronous call, and Retrieves snic params.
21562306a36Sopenharmony_ci */
21662306a36Sopenharmony_ciint
21762306a36Sopenharmony_cisnic_get_conf(struct snic *snic)
21862306a36Sopenharmony_ci{
21962306a36Sopenharmony_ci	DECLARE_COMPLETION_ONSTACK(wait);
22062306a36Sopenharmony_ci	unsigned long flags;
22162306a36Sopenharmony_ci	int ret;
22262306a36Sopenharmony_ci	int nr_retries = 3;
22362306a36Sopenharmony_ci
22462306a36Sopenharmony_ci	SNIC_HOST_INFO(snic->shost, "Retrieving snic params.\n");
22562306a36Sopenharmony_ci	spin_lock_irqsave(&snic->snic_lock, flags);
22662306a36Sopenharmony_ci	memset(&snic->fwinfo, 0, sizeof(snic->fwinfo));
22762306a36Sopenharmony_ci	snic->fwinfo.wait = &wait;
22862306a36Sopenharmony_ci	spin_unlock_irqrestore(&snic->snic_lock, flags);
22962306a36Sopenharmony_ci
23062306a36Sopenharmony_ci	/* Additional delay to handle HW Resource initialization. */
23162306a36Sopenharmony_ci	msleep(50);
23262306a36Sopenharmony_ci
23362306a36Sopenharmony_ci	/*
23462306a36Sopenharmony_ci	 * Exch ver req can be ignored by FW, if HW Resource initialization
23562306a36Sopenharmony_ci	 * is in progress, Hence retry.
23662306a36Sopenharmony_ci	 */
23762306a36Sopenharmony_ci	do {
23862306a36Sopenharmony_ci		ret = snic_queue_exch_ver_req(snic);
23962306a36Sopenharmony_ci		if (ret)
24062306a36Sopenharmony_ci			return ret;
24162306a36Sopenharmony_ci
24262306a36Sopenharmony_ci		wait_for_completion_timeout(&wait, msecs_to_jiffies(2000));
24362306a36Sopenharmony_ci		spin_lock_irqsave(&snic->snic_lock, flags);
24462306a36Sopenharmony_ci		ret = (snic->fwinfo.fw_ver != 0) ? 0 : -ETIMEDOUT;
24562306a36Sopenharmony_ci		if (ret)
24662306a36Sopenharmony_ci			SNIC_HOST_ERR(snic->shost,
24762306a36Sopenharmony_ci				      "Failed to retrieve snic params,\n");
24862306a36Sopenharmony_ci
24962306a36Sopenharmony_ci		/* Unset fwinfo.wait, on success or on last retry */
25062306a36Sopenharmony_ci		if (ret == 0 || nr_retries == 1)
25162306a36Sopenharmony_ci			snic->fwinfo.wait = NULL;
25262306a36Sopenharmony_ci
25362306a36Sopenharmony_ci		spin_unlock_irqrestore(&snic->snic_lock, flags);
25462306a36Sopenharmony_ci	} while (ret && --nr_retries);
25562306a36Sopenharmony_ci
25662306a36Sopenharmony_ci	return ret;
25762306a36Sopenharmony_ci} /* end of snic_get_info */
258