xref: /rk3399_ARM-atf/drivers/ufs/ufs.c (revision 83ef8698f9d1477c892cad15b4e48574ed634903)
1eb5073f4SHaojian Zhuang /*
2c5ee8588SWing Li  * Copyright (c) 2017-2021, ARM Limited and Contributors. All rights reserved.
3eb5073f4SHaojian Zhuang  *
4eb5073f4SHaojian Zhuang  * SPDX-License-Identifier: BSD-3-Clause
5eb5073f4SHaojian Zhuang  */
6eb5073f4SHaojian Zhuang 
7eb5073f4SHaojian Zhuang #include <assert.h>
8eb5073f4SHaojian Zhuang #include <endian.h>
9eb5073f4SHaojian Zhuang #include <errno.h>
10eb5073f4SHaojian Zhuang #include <stdint.h>
11eb5073f4SHaojian Zhuang #include <string.h>
1209d40e0eSAntonio Nino Diaz 
1309d40e0eSAntonio Nino Diaz #include <platform_def.h>
1409d40e0eSAntonio Nino Diaz 
1509d40e0eSAntonio Nino Diaz #include <arch_helpers.h>
1609d40e0eSAntonio Nino Diaz #include <common/debug.h>
1709d40e0eSAntonio Nino Diaz #include <drivers/delay_timer.h>
1809d40e0eSAntonio Nino Diaz #include <drivers/ufs.h>
1909d40e0eSAntonio Nino Diaz #include <lib/mmio.h>
20eb5073f4SHaojian Zhuang 
21eb5073f4SHaojian Zhuang #define CDB_ADDR_MASK			127
22eb5073f4SHaojian Zhuang #define ALIGN_CDB(x)			(((x) + CDB_ADDR_MASK) & ~CDB_ADDR_MASK)
23eb5073f4SHaojian Zhuang #define ALIGN_8(x)			(((x) + 7) & ~7)
24eb5073f4SHaojian Zhuang 
25eb5073f4SHaojian Zhuang #define UFS_DESC_SIZE			0x400
26eb5073f4SHaojian Zhuang #define MAX_UFS_DESC_SIZE		0x8000		/* 32 descriptors */
27eb5073f4SHaojian Zhuang 
28eb5073f4SHaojian Zhuang #define MAX_PRDT_SIZE			0x40000		/* 256KB */
29eb5073f4SHaojian Zhuang 
30eb5073f4SHaojian Zhuang static ufs_params_t ufs_params;
31eb5073f4SHaojian Zhuang static int nutrs;	/* Number of UTP Transfer Request Slots */
32eb5073f4SHaojian Zhuang 
33eb5073f4SHaojian Zhuang int ufshc_send_uic_cmd(uintptr_t base, uic_cmd_t *cmd)
34eb5073f4SHaojian Zhuang {
35eb5073f4SHaojian Zhuang 	unsigned int data;
36eb5073f4SHaojian Zhuang 
37d68d163dSJorge Troncoso 	if (base == 0 || cmd == NULL)
38d68d163dSJorge Troncoso 		return -EINVAL;
39d68d163dSJorge Troncoso 
40eb5073f4SHaojian Zhuang 	data = mmio_read_32(base + HCS);
41eb5073f4SHaojian Zhuang 	if ((data & HCS_UCRDY) == 0)
42eb5073f4SHaojian Zhuang 		return -EBUSY;
43eb5073f4SHaojian Zhuang 	mmio_write_32(base + IS, ~0);
44eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG1, cmd->arg1);
45eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG2, cmd->arg2);
46eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG3, cmd->arg3);
47eb5073f4SHaojian Zhuang 	mmio_write_32(base + UICCMD, cmd->op);
48eb5073f4SHaojian Zhuang 
49eb5073f4SHaojian Zhuang 	do {
50eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + IS);
51eb5073f4SHaojian Zhuang 	} while ((data & UFS_INT_UCCS) == 0);
52eb5073f4SHaojian Zhuang 	mmio_write_32(base + IS, UFS_INT_UCCS);
53101afa02SHaojian Zhuang 	return mmio_read_32(base + UCMDARG2) & CONFIG_RESULT_CODE_MASK;
54eb5073f4SHaojian Zhuang }
55eb5073f4SHaojian Zhuang 
56eb5073f4SHaojian Zhuang int ufshc_dme_get(unsigned int attr, unsigned int idx, unsigned int *val)
57eb5073f4SHaojian Zhuang {
58eb5073f4SHaojian Zhuang 	uintptr_t base;
59eb5073f4SHaojian Zhuang 	unsigned int data;
60d68d163dSJorge Troncoso 	int result, retries;
61d68d163dSJorge Troncoso 	uic_cmd_t cmd;
62eb5073f4SHaojian Zhuang 
63d68d163dSJorge Troncoso 	assert(ufs_params.reg_base != 0);
64d68d163dSJorge Troncoso 
65d68d163dSJorge Troncoso 	if (val == NULL)
66d68d163dSJorge Troncoso 		return -EINVAL;
67eb5073f4SHaojian Zhuang 
68eb5073f4SHaojian Zhuang 	base = ufs_params.reg_base;
69eb5073f4SHaojian Zhuang 	for (retries = 0; retries < 100; retries++) {
70eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + HCS);
71eb5073f4SHaojian Zhuang 		if ((data & HCS_UCRDY) != 0)
72eb5073f4SHaojian Zhuang 			break;
73eb5073f4SHaojian Zhuang 		mdelay(1);
74eb5073f4SHaojian Zhuang 	}
75eb5073f4SHaojian Zhuang 	if (retries >= 100)
76eb5073f4SHaojian Zhuang 		return -EBUSY;
77eb5073f4SHaojian Zhuang 
78d68d163dSJorge Troncoso 	cmd.arg1 = (attr << 16) | GEN_SELECTOR_IDX(idx);
79d68d163dSJorge Troncoso 	cmd.arg2 = 0;
80d68d163dSJorge Troncoso 	cmd.arg3 = 0;
81d68d163dSJorge Troncoso 	cmd.op = DME_GET;
82d68d163dSJorge Troncoso 	for (retries = 0; retries < UFS_UIC_COMMAND_RETRIES; ++retries) {
83d68d163dSJorge Troncoso 		result = ufshc_send_uic_cmd(base, &cmd);
84d68d163dSJorge Troncoso 		if (result == 0)
85d68d163dSJorge Troncoso 			break;
86eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + IS);
87eb5073f4SHaojian Zhuang 		if (data & UFS_INT_UE)
88eb5073f4SHaojian Zhuang 			return -EINVAL;
89d68d163dSJorge Troncoso 	}
90d68d163dSJorge Troncoso 	if (retries >= UFS_UIC_COMMAND_RETRIES)
91d68d163dSJorge Troncoso 		return -EIO;
92eb5073f4SHaojian Zhuang 
93eb5073f4SHaojian Zhuang 	*val = mmio_read_32(base + UCMDARG3);
94eb5073f4SHaojian Zhuang 	return 0;
95eb5073f4SHaojian Zhuang }
96eb5073f4SHaojian Zhuang 
97eb5073f4SHaojian Zhuang int ufshc_dme_set(unsigned int attr, unsigned int idx, unsigned int val)
98eb5073f4SHaojian Zhuang {
99eb5073f4SHaojian Zhuang 	uintptr_t base;
100eb5073f4SHaojian Zhuang 	unsigned int data;
101d68d163dSJorge Troncoso 	int result, retries;
102d68d163dSJorge Troncoso 	uic_cmd_t cmd;
103eb5073f4SHaojian Zhuang 
104eb5073f4SHaojian Zhuang 	assert((ufs_params.reg_base != 0));
105eb5073f4SHaojian Zhuang 
106eb5073f4SHaojian Zhuang 	base = ufs_params.reg_base;
107d68d163dSJorge Troncoso 	cmd.arg1 = (attr << 16) | GEN_SELECTOR_IDX(idx);
108d68d163dSJorge Troncoso 	cmd.arg2 = 0;
109d68d163dSJorge Troncoso 	cmd.arg3 = val;
110d68d163dSJorge Troncoso 	cmd.op = DME_SET;
111d68d163dSJorge Troncoso 
112d68d163dSJorge Troncoso 	for (retries = 0; retries < UFS_UIC_COMMAND_RETRIES; ++retries) {
113d68d163dSJorge Troncoso 		result = ufshc_send_uic_cmd(base, &cmd);
114d68d163dSJorge Troncoso 		if (result == 0)
115d68d163dSJorge Troncoso 			break;
116eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + IS);
117eb5073f4SHaojian Zhuang 		if (data & UFS_INT_UE)
118eb5073f4SHaojian Zhuang 			return -EINVAL;
119d68d163dSJorge Troncoso 	}
120d68d163dSJorge Troncoso 	if (retries >= UFS_UIC_COMMAND_RETRIES)
121d68d163dSJorge Troncoso 		return -EIO;
122d68d163dSJorge Troncoso 
123eb5073f4SHaojian Zhuang 	return 0;
124eb5073f4SHaojian Zhuang }
125eb5073f4SHaojian Zhuang 
12699ff1a35SJorge Troncoso static int ufshc_hce_enable(uintptr_t base)
127eb5073f4SHaojian Zhuang {
128eb5073f4SHaojian Zhuang 	unsigned int data;
12999ff1a35SJorge Troncoso 	int retries;
130eb5073f4SHaojian Zhuang 
131eb5073f4SHaojian Zhuang 	/* Enable Host Controller */
132eb5073f4SHaojian Zhuang 	mmio_write_32(base + HCE, HCE_ENABLE);
13399ff1a35SJorge Troncoso 
134eb5073f4SHaojian Zhuang 	/* Wait until basic initialization sequence completed */
13599ff1a35SJorge Troncoso 	for (retries = 0; retries < HCE_ENABLE_INNER_RETRIES; ++retries) {
136eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + HCE);
13799ff1a35SJorge Troncoso 		if (data & HCE_ENABLE) {
13899ff1a35SJorge Troncoso 			break;
13999ff1a35SJorge Troncoso 		}
14099ff1a35SJorge Troncoso 		udelay(HCE_ENABLE_TIMEOUT_US);
14199ff1a35SJorge Troncoso 	}
14299ff1a35SJorge Troncoso 	if (retries >= HCE_ENABLE_INNER_RETRIES) {
14399ff1a35SJorge Troncoso 		return -ETIMEDOUT;
14499ff1a35SJorge Troncoso 	}
14599ff1a35SJorge Troncoso 
14699ff1a35SJorge Troncoso 	return 0;
14799ff1a35SJorge Troncoso }
14899ff1a35SJorge Troncoso 
149b3f03b20Sanans static int ufshc_hce_disable(uintptr_t base)
150b3f03b20Sanans {
151b3f03b20Sanans 	unsigned int data;
152b3f03b20Sanans 	int timeout;
153b3f03b20Sanans 
154b3f03b20Sanans 	/* Disable Host Controller */
155b3f03b20Sanans 	mmio_write_32(base + HCE, HCE_DISABLE);
156b3f03b20Sanans 	timeout = HCE_DISABLE_TIMEOUT_US;
157b3f03b20Sanans 	do {
158b3f03b20Sanans 		data = mmio_read_32(base + HCE);
159b3f03b20Sanans 		if ((data & HCE_ENABLE) == HCE_DISABLE) {
160b3f03b20Sanans 			break;
161b3f03b20Sanans 		}
162b3f03b20Sanans 		udelay(1);
163b3f03b20Sanans 	} while (--timeout > 0);
164b3f03b20Sanans 
165b3f03b20Sanans 	if (timeout <= 0) {
166b3f03b20Sanans 		return -ETIMEDOUT;
167b3f03b20Sanans 	}
168b3f03b20Sanans 
169b3f03b20Sanans 	return 0;
170b3f03b20Sanans }
171b3f03b20Sanans 
172b3f03b20Sanans 
17399ff1a35SJorge Troncoso static int ufshc_reset(uintptr_t base)
17499ff1a35SJorge Troncoso {
17599ff1a35SJorge Troncoso 	unsigned int data;
17699ff1a35SJorge Troncoso 	int retries, result;
17799ff1a35SJorge Troncoso 
178b3f03b20Sanans 	/* disable controller if enabled */
179b3f03b20Sanans 	if (mmio_read_32(base + HCE) & HCE_ENABLE) {
180b3f03b20Sanans 		result = ufshc_hce_disable(base);
181b3f03b20Sanans 		if (result != 0) {
182b3f03b20Sanans 			return -EIO;
183b3f03b20Sanans 		}
184b3f03b20Sanans 	}
185b3f03b20Sanans 
18699ff1a35SJorge Troncoso 	for (retries = 0; retries < HCE_ENABLE_OUTER_RETRIES; ++retries) {
18799ff1a35SJorge Troncoso 		result = ufshc_hce_enable(base);
18899ff1a35SJorge Troncoso 		if (result == 0) {
18999ff1a35SJorge Troncoso 			break;
19099ff1a35SJorge Troncoso 		}
19199ff1a35SJorge Troncoso 	}
19299ff1a35SJorge Troncoso 	if (retries >= HCE_ENABLE_OUTER_RETRIES) {
19399ff1a35SJorge Troncoso 		return -EIO;
19499ff1a35SJorge Troncoso 	}
195eb5073f4SHaojian Zhuang 
196eb5073f4SHaojian Zhuang 	/* Enable Interrupts */
197eb5073f4SHaojian Zhuang 	data = UFS_INT_UCCS | UFS_INT_ULSS | UFS_INT_UE | UFS_INT_UTPES |
198eb5073f4SHaojian Zhuang 	       UFS_INT_DFES | UFS_INT_HCFES | UFS_INT_SBFES;
199eb5073f4SHaojian Zhuang 	mmio_write_32(base + IE, data);
20099ff1a35SJorge Troncoso 
20199ff1a35SJorge Troncoso 	return 0;
202eb5073f4SHaojian Zhuang }
203eb5073f4SHaojian Zhuang 
204905635d5SJorge Troncoso static int ufshc_dme_link_startup(uintptr_t base)
205eb5073f4SHaojian Zhuang {
206eb5073f4SHaojian Zhuang 	uic_cmd_t cmd;
207905635d5SJorge Troncoso 
208905635d5SJorge Troncoso 	memset(&cmd, 0, sizeof(cmd));
209905635d5SJorge Troncoso 	cmd.op = DME_LINKSTARTUP;
210905635d5SJorge Troncoso 	return ufshc_send_uic_cmd(base, &cmd);
211905635d5SJorge Troncoso }
212905635d5SJorge Troncoso 
213905635d5SJorge Troncoso static int ufshc_link_startup(uintptr_t base)
214905635d5SJorge Troncoso {
215eb5073f4SHaojian Zhuang 	int data, result;
216eb5073f4SHaojian Zhuang 	int retries;
217eb5073f4SHaojian Zhuang 
218905635d5SJorge Troncoso 	for (retries = DME_LINKSTARTUP_RETRIES; retries > 0; retries--) {
219905635d5SJorge Troncoso 		result = ufshc_dme_link_startup(base);
220905635d5SJorge Troncoso 		if (result != 0) {
221905635d5SJorge Troncoso 			/* Reset controller before trying again */
222905635d5SJorge Troncoso 			result = ufshc_reset(base);
223905635d5SJorge Troncoso 			if (result != 0) {
224905635d5SJorge Troncoso 				return result;
225905635d5SJorge Troncoso 			}
226eb5073f4SHaojian Zhuang 			continue;
227905635d5SJorge Troncoso 		}
22883103d12SJorge Troncoso 		assert(mmio_read_32(base + HCS) & HCS_DP);
229eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + IS);
230eb5073f4SHaojian Zhuang 		if (data & UFS_INT_ULSS)
231eb5073f4SHaojian Zhuang 			mmio_write_32(base + IS, UFS_INT_ULSS);
232eb5073f4SHaojian Zhuang 		return 0;
233eb5073f4SHaojian Zhuang 	}
234eb5073f4SHaojian Zhuang 	return -EIO;
235eb5073f4SHaojian Zhuang }
236eb5073f4SHaojian Zhuang 
23756db7b8bSJorge Troncoso /* Read Door Bell register to check if slot zero is available */
23856db7b8bSJorge Troncoso static int is_slot_available(void)
239eb5073f4SHaojian Zhuang {
24056db7b8bSJorge Troncoso 	if (mmio_read_32(ufs_params.reg_base + UTRLDBR) & 0x1) {
241eb5073f4SHaojian Zhuang 		return -EBUSY;
24256db7b8bSJorge Troncoso 	}
243eb5073f4SHaojian Zhuang 	return 0;
244eb5073f4SHaojian Zhuang }
245eb5073f4SHaojian Zhuang 
246eb5073f4SHaojian Zhuang static void get_utrd(utp_utrd_t *utrd)
247eb5073f4SHaojian Zhuang {
248eb5073f4SHaojian Zhuang 	uintptr_t base;
24956db7b8bSJorge Troncoso 	int result;
250eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
251eb5073f4SHaojian Zhuang 
252eb5073f4SHaojian Zhuang 	assert(utrd != NULL);
25356db7b8bSJorge Troncoso 	result = is_slot_available();
254eb5073f4SHaojian Zhuang 	assert(result == 0);
255eb5073f4SHaojian Zhuang 
256eb5073f4SHaojian Zhuang 	/* clear utrd */
257eb5073f4SHaojian Zhuang 	memset((void *)utrd, 0, sizeof(utp_utrd_t));
25856db7b8bSJorge Troncoso 	base = ufs_params.desc_base;
259eb5073f4SHaojian Zhuang 	/* clear the descriptor */
260eb5073f4SHaojian Zhuang 	memset((void *)base, 0, UFS_DESC_SIZE);
261eb5073f4SHaojian Zhuang 
262eb5073f4SHaojian Zhuang 	utrd->header = base;
26356db7b8bSJorge Troncoso 	utrd->task_tag = 1; /* We always use the first slot */
264eb5073f4SHaojian Zhuang 	/* CDB address should be aligned with 128 bytes */
265eb5073f4SHaojian Zhuang 	utrd->upiu = ALIGN_CDB(utrd->header + sizeof(utrd_header_t));
266eb5073f4SHaojian Zhuang 	utrd->resp_upiu = ALIGN_8(utrd->upiu + sizeof(cmd_upiu_t));
267eb5073f4SHaojian Zhuang 	utrd->size_upiu = utrd->resp_upiu - utrd->upiu;
268eb5073f4SHaojian Zhuang 	utrd->size_resp_upiu = ALIGN_8(sizeof(resp_upiu_t));
269eb5073f4SHaojian Zhuang 	utrd->prdt = utrd->resp_upiu + utrd->size_resp_upiu;
270eb5073f4SHaojian Zhuang 
271eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
272eb5073f4SHaojian Zhuang 	hd->ucdba = utrd->upiu & UINT32_MAX;
273eb5073f4SHaojian Zhuang 	hd->ucdbau = (utrd->upiu >> 32) & UINT32_MAX;
274eb5073f4SHaojian Zhuang 	/* Both RUL and RUO is based on DWORD */
275eb5073f4SHaojian Zhuang 	hd->rul = utrd->size_resp_upiu >> 2;
276eb5073f4SHaojian Zhuang 	hd->ruo = utrd->size_upiu >> 2;
277eb5073f4SHaojian Zhuang 	(void)result;
278eb5073f4SHaojian Zhuang }
279eb5073f4SHaojian Zhuang 
280eb5073f4SHaojian Zhuang /*
281eb5073f4SHaojian Zhuang  * Prepare UTRD, Command UPIU, Response UPIU.
282eb5073f4SHaojian Zhuang  */
283eb5073f4SHaojian Zhuang static int ufs_prepare_cmd(utp_utrd_t *utrd, uint8_t op, uint8_t lun,
284eb5073f4SHaojian Zhuang 			   int lba, uintptr_t buf, size_t length)
285eb5073f4SHaojian Zhuang {
286eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
287eb5073f4SHaojian Zhuang 	cmd_upiu_t *upiu;
288eb5073f4SHaojian Zhuang 	prdt_t *prdt;
289eb5073f4SHaojian Zhuang 	unsigned int ulba;
290eb5073f4SHaojian Zhuang 	unsigned int lba_cnt;
291eb5073f4SHaojian Zhuang 	int prdt_size;
292*83ef8698SJorge Troncoso 	uintptr_t desc_limit;
293*83ef8698SJorge Troncoso 	size_t flush_size;
294eb5073f4SHaojian Zhuang 
295eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
296eb5073f4SHaojian Zhuang 	upiu = (cmd_upiu_t *)utrd->upiu;
297eb5073f4SHaojian Zhuang 
298eb5073f4SHaojian Zhuang 	hd->i = 1;
299eb5073f4SHaojian Zhuang 	hd->ct = CT_UFS_STORAGE;
300eb5073f4SHaojian Zhuang 	hd->ocs = OCS_MASK;
301eb5073f4SHaojian Zhuang 
302eb5073f4SHaojian Zhuang 	upiu->trans_type = CMD_UPIU;
303eb5073f4SHaojian Zhuang 	upiu->task_tag = utrd->task_tag;
304eb5073f4SHaojian Zhuang 	upiu->cdb[0] = op;
305eb5073f4SHaojian Zhuang 	ulba = (unsigned int)lba;
306eb5073f4SHaojian Zhuang 	lba_cnt = (unsigned int)(length >> UFS_BLOCK_SHIFT);
307eb5073f4SHaojian Zhuang 	switch (op) {
308eb5073f4SHaojian Zhuang 	case CDBCMD_TEST_UNIT_READY:
309eb5073f4SHaojian Zhuang 		break;
310eb5073f4SHaojian Zhuang 	case CDBCMD_READ_CAPACITY_10:
311eb5073f4SHaojian Zhuang 		hd->dd = DD_OUT;
312eb5073f4SHaojian Zhuang 		upiu->flags = UPIU_FLAGS_R | UPIU_FLAGS_ATTR_S;
313eb5073f4SHaojian Zhuang 		upiu->lun = lun;
314eb5073f4SHaojian Zhuang 		break;
315eb5073f4SHaojian Zhuang 	case CDBCMD_READ_10:
316eb5073f4SHaojian Zhuang 		hd->dd = DD_OUT;
317eb5073f4SHaojian Zhuang 		upiu->flags = UPIU_FLAGS_R | UPIU_FLAGS_ATTR_S;
318eb5073f4SHaojian Zhuang 		upiu->lun = lun;
319eb5073f4SHaojian Zhuang 		upiu->cdb[1] = RW_WITHOUT_CACHE;
320eb5073f4SHaojian Zhuang 		/* set logical block address */
321eb5073f4SHaojian Zhuang 		upiu->cdb[2] = (ulba >> 24) & 0xff;
322eb5073f4SHaojian Zhuang 		upiu->cdb[3] = (ulba >> 16) & 0xff;
323eb5073f4SHaojian Zhuang 		upiu->cdb[4] = (ulba >> 8) & 0xff;
324eb5073f4SHaojian Zhuang 		upiu->cdb[5] = ulba & 0xff;
325eb5073f4SHaojian Zhuang 		/* set transfer length */
326eb5073f4SHaojian Zhuang 		upiu->cdb[7] = (lba_cnt >> 8) & 0xff;
327eb5073f4SHaojian Zhuang 		upiu->cdb[8] = lba_cnt & 0xff;
328eb5073f4SHaojian Zhuang 		break;
329eb5073f4SHaojian Zhuang 	case CDBCMD_WRITE_10:
330eb5073f4SHaojian Zhuang 		hd->dd = DD_IN;
331eb5073f4SHaojian Zhuang 		upiu->flags = UPIU_FLAGS_W | UPIU_FLAGS_ATTR_S;
332eb5073f4SHaojian Zhuang 		upiu->lun = lun;
333eb5073f4SHaojian Zhuang 		upiu->cdb[1] = RW_WITHOUT_CACHE;
334eb5073f4SHaojian Zhuang 		/* set logical block address */
335eb5073f4SHaojian Zhuang 		upiu->cdb[2] = (ulba >> 24) & 0xff;
336eb5073f4SHaojian Zhuang 		upiu->cdb[3] = (ulba >> 16) & 0xff;
337eb5073f4SHaojian Zhuang 		upiu->cdb[4] = (ulba >> 8) & 0xff;
338eb5073f4SHaojian Zhuang 		upiu->cdb[5] = ulba & 0xff;
339eb5073f4SHaojian Zhuang 		/* set transfer length */
340eb5073f4SHaojian Zhuang 		upiu->cdb[7] = (lba_cnt >> 8) & 0xff;
341eb5073f4SHaojian Zhuang 		upiu->cdb[8] = lba_cnt & 0xff;
342eb5073f4SHaojian Zhuang 		break;
343eb5073f4SHaojian Zhuang 	default:
344eb5073f4SHaojian Zhuang 		assert(0);
3455aa7498aSJonathan Wright 		break;
346eb5073f4SHaojian Zhuang 	}
347*83ef8698SJorge Troncoso 	if (hd->dd == DD_IN) {
348eb5073f4SHaojian Zhuang 		flush_dcache_range(buf, length);
349*83ef8698SJorge Troncoso 	} else if (hd->dd == DD_OUT) {
350eb5073f4SHaojian Zhuang 		inv_dcache_range(buf, length);
351*83ef8698SJorge Troncoso 	}
352*83ef8698SJorge Troncoso 
353*83ef8698SJorge Troncoso 	utrd->size_prdt = 0;
354eb5073f4SHaojian Zhuang 	if (length) {
355eb5073f4SHaojian Zhuang 		upiu->exp_data_trans_len = htobe32(length);
356eb5073f4SHaojian Zhuang 		assert(lba_cnt <= UINT16_MAX);
357eb5073f4SHaojian Zhuang 		prdt = (prdt_t *)utrd->prdt;
358eb5073f4SHaojian Zhuang 
359*83ef8698SJorge Troncoso 		desc_limit = ufs_params.desc_base + ufs_params.desc_size;
360eb5073f4SHaojian Zhuang 		prdt_size = 0;
361eb5073f4SHaojian Zhuang 		while (length > 0) {
362*83ef8698SJorge Troncoso 			if ((uintptr_t)prdt + sizeof(prdt_t) > desc_limit) {
363*83ef8698SJorge Troncoso 				ERROR("UFS: Exceeded descriptor limit. Image is too large\n");
364*83ef8698SJorge Troncoso 				panic();
365*83ef8698SJorge Troncoso 			}
366eb5073f4SHaojian Zhuang 			prdt->dba = (unsigned int)(buf & UINT32_MAX);
367eb5073f4SHaojian Zhuang 			prdt->dbau = (unsigned int)((buf >> 32) & UINT32_MAX);
368eb5073f4SHaojian Zhuang 			/* prdt->dbc counts from 0 */
369eb5073f4SHaojian Zhuang 			if (length > MAX_PRDT_SIZE) {
370eb5073f4SHaojian Zhuang 				prdt->dbc = MAX_PRDT_SIZE - 1;
371eb5073f4SHaojian Zhuang 				length = length - MAX_PRDT_SIZE;
372eb5073f4SHaojian Zhuang 			} else {
373eb5073f4SHaojian Zhuang 				prdt->dbc = length - 1;
374eb5073f4SHaojian Zhuang 				length = 0;
375eb5073f4SHaojian Zhuang 			}
376eb5073f4SHaojian Zhuang 			buf += MAX_PRDT_SIZE;
377eb5073f4SHaojian Zhuang 			prdt++;
378eb5073f4SHaojian Zhuang 			prdt_size += sizeof(prdt_t);
379eb5073f4SHaojian Zhuang 		}
380eb5073f4SHaojian Zhuang 		utrd->size_prdt = ALIGN_8(prdt_size);
381eb5073f4SHaojian Zhuang 		hd->prdtl = utrd->size_prdt >> 2;
382eb5073f4SHaojian Zhuang 		hd->prdto = (utrd->size_upiu + utrd->size_resp_upiu) >> 2;
383eb5073f4SHaojian Zhuang 	}
384eb5073f4SHaojian Zhuang 
385*83ef8698SJorge Troncoso 	flush_size = utrd->prdt + utrd->size_prdt - utrd->header;
386*83ef8698SJorge Troncoso 	flush_dcache_range(utrd->header, flush_size);
387eb5073f4SHaojian Zhuang 	return 0;
388eb5073f4SHaojian Zhuang }
389eb5073f4SHaojian Zhuang 
390eb5073f4SHaojian Zhuang static int ufs_prepare_query(utp_utrd_t *utrd, uint8_t op, uint8_t idn,
391eb5073f4SHaojian Zhuang 			     uint8_t index, uint8_t sel,
392eb5073f4SHaojian Zhuang 			     uintptr_t buf, size_t length)
393eb5073f4SHaojian Zhuang {
394eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
395eb5073f4SHaojian Zhuang 	query_upiu_t *query_upiu;
396eb5073f4SHaojian Zhuang 
397eb5073f4SHaojian Zhuang 
398eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
399eb5073f4SHaojian Zhuang 	query_upiu = (query_upiu_t *)utrd->upiu;
400eb5073f4SHaojian Zhuang 
401eb5073f4SHaojian Zhuang 	hd->i = 1;
402eb5073f4SHaojian Zhuang 	hd->ct = CT_UFS_STORAGE;
403eb5073f4SHaojian Zhuang 	hd->ocs = OCS_MASK;
404eb5073f4SHaojian Zhuang 
405eb5073f4SHaojian Zhuang 	query_upiu->trans_type = QUERY_REQUEST_UPIU;
406eb5073f4SHaojian Zhuang 	query_upiu->task_tag = utrd->task_tag;
407eb5073f4SHaojian Zhuang 	query_upiu->ts.desc.opcode = op;
408eb5073f4SHaojian Zhuang 	query_upiu->ts.desc.idn = idn;
409eb5073f4SHaojian Zhuang 	query_upiu->ts.desc.index = index;
410eb5073f4SHaojian Zhuang 	query_upiu->ts.desc.selector = sel;
411eb5073f4SHaojian Zhuang 	switch (op) {
412eb5073f4SHaojian Zhuang 	case QUERY_READ_DESC:
413eb5073f4SHaojian Zhuang 		query_upiu->query_func = QUERY_FUNC_STD_READ;
414eb5073f4SHaojian Zhuang 		query_upiu->ts.desc.length = htobe16(length);
415eb5073f4SHaojian Zhuang 		break;
416eb5073f4SHaojian Zhuang 	case QUERY_WRITE_DESC:
417eb5073f4SHaojian Zhuang 		query_upiu->query_func = QUERY_FUNC_STD_WRITE;
418eb5073f4SHaojian Zhuang 		query_upiu->ts.desc.length = htobe16(length);
419eb5073f4SHaojian Zhuang 		memcpy((void *)(utrd->upiu + sizeof(query_upiu_t)),
420eb5073f4SHaojian Zhuang 		       (void *)buf, length);
421eb5073f4SHaojian Zhuang 		break;
422eb5073f4SHaojian Zhuang 	case QUERY_READ_ATTR:
423eb5073f4SHaojian Zhuang 	case QUERY_READ_FLAG:
424eb5073f4SHaojian Zhuang 		query_upiu->query_func = QUERY_FUNC_STD_READ;
425eb5073f4SHaojian Zhuang 		break;
426eb5073f4SHaojian Zhuang 	case QUERY_CLEAR_FLAG:
427eb5073f4SHaojian Zhuang 	case QUERY_SET_FLAG:
428eb5073f4SHaojian Zhuang 		query_upiu->query_func = QUERY_FUNC_STD_WRITE;
429eb5073f4SHaojian Zhuang 		break;
430eb5073f4SHaojian Zhuang 	case QUERY_WRITE_ATTR:
431eb5073f4SHaojian Zhuang 		query_upiu->query_func = QUERY_FUNC_STD_WRITE;
432a4755183Sanans 		query_upiu->ts.attr.value = htobe32(*((uint32_t *)buf));
433eb5073f4SHaojian Zhuang 		break;
434eb5073f4SHaojian Zhuang 	default:
435eb5073f4SHaojian Zhuang 		assert(0);
4365aa7498aSJonathan Wright 		break;
437eb5073f4SHaojian Zhuang 	}
438eb5073f4SHaojian Zhuang 	flush_dcache_range((uintptr_t)utrd->header, UFS_DESC_SIZE);
439eb5073f4SHaojian Zhuang 	return 0;
440eb5073f4SHaojian Zhuang }
441eb5073f4SHaojian Zhuang 
442eb5073f4SHaojian Zhuang static void ufs_prepare_nop_out(utp_utrd_t *utrd)
443eb5073f4SHaojian Zhuang {
444eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
445eb5073f4SHaojian Zhuang 	nop_out_upiu_t *nop_out;
446eb5073f4SHaojian Zhuang 
447eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
448eb5073f4SHaojian Zhuang 	nop_out = (nop_out_upiu_t *)utrd->upiu;
449eb5073f4SHaojian Zhuang 
450eb5073f4SHaojian Zhuang 	hd->i = 1;
451eb5073f4SHaojian Zhuang 	hd->ct = CT_UFS_STORAGE;
452eb5073f4SHaojian Zhuang 	hd->ocs = OCS_MASK;
453eb5073f4SHaojian Zhuang 
454eb5073f4SHaojian Zhuang 	nop_out->trans_type = 0;
455eb5073f4SHaojian Zhuang 	nop_out->task_tag = utrd->task_tag;
456eb5073f4SHaojian Zhuang 	flush_dcache_range((uintptr_t)utrd->header, UFS_DESC_SIZE);
457eb5073f4SHaojian Zhuang }
458eb5073f4SHaojian Zhuang 
459eb5073f4SHaojian Zhuang static void ufs_send_request(int task_tag)
460eb5073f4SHaojian Zhuang {
461eb5073f4SHaojian Zhuang 	unsigned int data;
462eb5073f4SHaojian Zhuang 	int slot;
463eb5073f4SHaojian Zhuang 
464eb5073f4SHaojian Zhuang 	slot = task_tag - 1;
465eb5073f4SHaojian Zhuang 	/* clear all interrupts */
466eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + IS, ~0);
467eb5073f4SHaojian Zhuang 
468eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + UTRLRSR, 1);
469660c208dSanans 	assert(mmio_read_32(ufs_params.reg_base + UTRLRSR) == 1);
470eb5073f4SHaojian Zhuang 
471eb5073f4SHaojian Zhuang 	data = UTRIACR_IAEN | UTRIACR_CTR | UTRIACR_IACTH(0x1F) |
472eb5073f4SHaojian Zhuang 	       UTRIACR_IATOVAL(0xFF);
473eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + UTRIACR, data);
474eb5073f4SHaojian Zhuang 	/* send request */
475eb5073f4SHaojian Zhuang 	mmio_setbits_32(ufs_params.reg_base + UTRLDBR, 1 << slot);
476eb5073f4SHaojian Zhuang }
477eb5073f4SHaojian Zhuang 
478eb5073f4SHaojian Zhuang static int ufs_check_resp(utp_utrd_t *utrd, int trans_type)
479eb5073f4SHaojian Zhuang {
480eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
481eb5073f4SHaojian Zhuang 	resp_upiu_t *resp;
4823d309556SAnand Saminathan 	sense_data_t *sense;
483eb5073f4SHaojian Zhuang 	unsigned int data;
484eb5073f4SHaojian Zhuang 	int slot;
485eb5073f4SHaojian Zhuang 
486eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
487eb5073f4SHaojian Zhuang 	resp = (resp_upiu_t *)utrd->resp_upiu;
488eb5073f4SHaojian Zhuang 	do {
489eb5073f4SHaojian Zhuang 		data = mmio_read_32(ufs_params.reg_base + IS);
490eb5073f4SHaojian Zhuang 		if ((data & ~(UFS_INT_UCCS | UFS_INT_UTRCS)) != 0)
491eb5073f4SHaojian Zhuang 			return -EIO;
492eb5073f4SHaojian Zhuang 	} while ((data & UFS_INT_UTRCS) == 0);
493eb5073f4SHaojian Zhuang 	slot = utrd->task_tag - 1;
494eb5073f4SHaojian Zhuang 
495eb5073f4SHaojian Zhuang 	data = mmio_read_32(ufs_params.reg_base + UTRLDBR);
496eb5073f4SHaojian Zhuang 	assert((data & (1 << slot)) == 0);
49738a5ecb7SChannagoud kadabi 	/*
49838a5ecb7SChannagoud kadabi 	 * Invalidate the header after DMA read operation has
49938a5ecb7SChannagoud kadabi 	 * completed to avoid cpu referring to the prefetched
50038a5ecb7SChannagoud kadabi 	 * data brought in before DMA completion.
50138a5ecb7SChannagoud kadabi 	 */
50238a5ecb7SChannagoud kadabi 	inv_dcache_range((uintptr_t)hd, UFS_DESC_SIZE);
503eb5073f4SHaojian Zhuang 	assert(hd->ocs == OCS_SUCCESS);
504eb5073f4SHaojian Zhuang 	assert((resp->trans_type & TRANS_TYPE_CODE_MASK) == trans_type);
5053d309556SAnand Saminathan 
5063d309556SAnand Saminathan 	sense = &resp->sd.sense;
5073d309556SAnand Saminathan 	if (sense->resp_code == SENSE_DATA_VALID &&
5083d309556SAnand Saminathan 	    sense->sense_key == SENSE_KEY_UNIT_ATTENTION && sense->asc == 0x29 &&
5093d309556SAnand Saminathan 	    sense->ascq == 0) {
5103d309556SAnand Saminathan 		WARN("Unit Attention Condition\n");
5113d309556SAnand Saminathan 		return -EAGAIN;
5123d309556SAnand Saminathan 	}
5133d309556SAnand Saminathan 
514eb5073f4SHaojian Zhuang 	(void)resp;
515eb5073f4SHaojian Zhuang 	(void)slot;
516eb5073f4SHaojian Zhuang 	return 0;
517eb5073f4SHaojian Zhuang }
518eb5073f4SHaojian Zhuang 
5196e16f7f0Sanans static void ufs_send_cmd(utp_utrd_t *utrd, uint8_t cmd_op, uint8_t lun, int lba, uintptr_t buf,
5206e16f7f0Sanans 			 size_t length)
5216e16f7f0Sanans {
5223d309556SAnand Saminathan 	int result, i;
5236e16f7f0Sanans 
5243d309556SAnand Saminathan 	for (i = 0; i < UFS_CMD_RETRIES; ++i) {
5256e16f7f0Sanans 		get_utrd(utrd);
5266e16f7f0Sanans 		result = ufs_prepare_cmd(utrd, cmd_op, lun, lba, buf, length);
5276e16f7f0Sanans 		assert(result == 0);
5286e16f7f0Sanans 		ufs_send_request(utrd->task_tag);
5296e16f7f0Sanans 		result = ufs_check_resp(utrd, RESPONSE_UPIU);
5303d309556SAnand Saminathan 		if (result == 0 || result == -EIO) {
5313d309556SAnand Saminathan 			break;
5323d309556SAnand Saminathan 		}
5333d309556SAnand Saminathan 	}
5346e16f7f0Sanans 	assert(result == 0);
5356e16f7f0Sanans 	(void)result;
5366e16f7f0Sanans }
5376e16f7f0Sanans 
538eb5073f4SHaojian Zhuang #ifdef UFS_RESP_DEBUG
539eb5073f4SHaojian Zhuang static void dump_upiu(utp_utrd_t *utrd)
540eb5073f4SHaojian Zhuang {
541eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
542eb5073f4SHaojian Zhuang 	int i;
543eb5073f4SHaojian Zhuang 
544eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
545eb5073f4SHaojian Zhuang 	INFO("utrd:0x%x, ruo:0x%x, rul:0x%x, ocs:0x%x, UTRLDBR:0x%x\n",
546eb5073f4SHaojian Zhuang 		(unsigned int)(uintptr_t)utrd, hd->ruo, hd->rul, hd->ocs,
547eb5073f4SHaojian Zhuang 		mmio_read_32(ufs_params.reg_base + UTRLDBR));
548eb5073f4SHaojian Zhuang 	for (i = 0; i < sizeof(utrd_header_t); i += 4) {
549eb5073f4SHaojian Zhuang 		INFO("[%lx]:0x%x\n",
550eb5073f4SHaojian Zhuang 			(uintptr_t)utrd->header + i,
551eb5073f4SHaojian Zhuang 			*(unsigned int *)((uintptr_t)utrd->header + i));
552eb5073f4SHaojian Zhuang 	}
553eb5073f4SHaojian Zhuang 
554eb5073f4SHaojian Zhuang 	for (i = 0; i < sizeof(cmd_upiu_t); i += 4) {
555eb5073f4SHaojian Zhuang 		INFO("cmd[%lx]:0x%x\n",
556eb5073f4SHaojian Zhuang 			utrd->upiu + i,
557eb5073f4SHaojian Zhuang 			*(unsigned int *)(utrd->upiu + i));
558eb5073f4SHaojian Zhuang 	}
559eb5073f4SHaojian Zhuang 	for (i = 0; i < sizeof(resp_upiu_t); i += 4) {
560eb5073f4SHaojian Zhuang 		INFO("resp[%lx]:0x%x\n",
561eb5073f4SHaojian Zhuang 			utrd->resp_upiu + i,
562eb5073f4SHaojian Zhuang 			*(unsigned int *)(utrd->resp_upiu + i));
563eb5073f4SHaojian Zhuang 	}
564eb5073f4SHaojian Zhuang 	for (i = 0; i < sizeof(prdt_t); i += 4) {
565eb5073f4SHaojian Zhuang 		INFO("prdt[%lx]:0x%x\n",
566eb5073f4SHaojian Zhuang 			utrd->prdt + i,
567eb5073f4SHaojian Zhuang 			*(unsigned int *)(utrd->prdt + i));
568eb5073f4SHaojian Zhuang 	}
569eb5073f4SHaojian Zhuang }
570eb5073f4SHaojian Zhuang #endif
571eb5073f4SHaojian Zhuang 
572eb5073f4SHaojian Zhuang static void ufs_verify_init(void)
573eb5073f4SHaojian Zhuang {
574eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
575eb5073f4SHaojian Zhuang 	int result;
576eb5073f4SHaojian Zhuang 
577eb5073f4SHaojian Zhuang 	get_utrd(&utrd);
578eb5073f4SHaojian Zhuang 	ufs_prepare_nop_out(&utrd);
579eb5073f4SHaojian Zhuang 	ufs_send_request(utrd.task_tag);
580eb5073f4SHaojian Zhuang 	result = ufs_check_resp(&utrd, NOP_IN_UPIU);
581eb5073f4SHaojian Zhuang 	assert(result == 0);
582eb5073f4SHaojian Zhuang 	(void)result;
583eb5073f4SHaojian Zhuang }
584eb5073f4SHaojian Zhuang 
585eb5073f4SHaojian Zhuang static void ufs_verify_ready(void)
586eb5073f4SHaojian Zhuang {
587eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
5886e16f7f0Sanans 	ufs_send_cmd(&utrd, CDBCMD_TEST_UNIT_READY, 0, 0, 0, 0);
589eb5073f4SHaojian Zhuang }
590eb5073f4SHaojian Zhuang 
591eb5073f4SHaojian Zhuang static void ufs_query(uint8_t op, uint8_t idn, uint8_t index, uint8_t sel,
592eb5073f4SHaojian Zhuang 		      uintptr_t buf, size_t size)
593eb5073f4SHaojian Zhuang {
594eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
595eb5073f4SHaojian Zhuang 	query_resp_upiu_t *resp;
596eb5073f4SHaojian Zhuang 	int result;
597eb5073f4SHaojian Zhuang 
598eb5073f4SHaojian Zhuang 	switch (op) {
599eb5073f4SHaojian Zhuang 	case QUERY_READ_FLAG:
600eb5073f4SHaojian Zhuang 	case QUERY_READ_ATTR:
601eb5073f4SHaojian Zhuang 	case QUERY_READ_DESC:
602eb5073f4SHaojian Zhuang 	case QUERY_WRITE_DESC:
603eb5073f4SHaojian Zhuang 	case QUERY_WRITE_ATTR:
604eb5073f4SHaojian Zhuang 		assert(((buf & 3) == 0) && (size != 0));
605eb5073f4SHaojian Zhuang 		break;
6065aa7498aSJonathan Wright 	default:
6075aa7498aSJonathan Wright 		/* Do nothing in default case */
6085aa7498aSJonathan Wright 		break;
609eb5073f4SHaojian Zhuang 	}
610eb5073f4SHaojian Zhuang 	get_utrd(&utrd);
611eb5073f4SHaojian Zhuang 	ufs_prepare_query(&utrd, op, idn, index, sel, buf, size);
612eb5073f4SHaojian Zhuang 	ufs_send_request(utrd.task_tag);
613eb5073f4SHaojian Zhuang 	result = ufs_check_resp(&utrd, QUERY_RESPONSE_UPIU);
614eb5073f4SHaojian Zhuang 	assert(result == 0);
615eb5073f4SHaojian Zhuang 	resp = (query_resp_upiu_t *)utrd.resp_upiu;
616eb5073f4SHaojian Zhuang #ifdef UFS_RESP_DEBUG
617eb5073f4SHaojian Zhuang 	dump_upiu(&utrd);
618eb5073f4SHaojian Zhuang #endif
619eb5073f4SHaojian Zhuang 	assert(resp->query_resp == QUERY_RESP_SUCCESS);
620eb5073f4SHaojian Zhuang 
621eb5073f4SHaojian Zhuang 	switch (op) {
622eb5073f4SHaojian Zhuang 	case QUERY_READ_FLAG:
623eb5073f4SHaojian Zhuang 		*(uint32_t *)buf = (uint32_t)resp->ts.flag.value;
624eb5073f4SHaojian Zhuang 		break;
625eb5073f4SHaojian Zhuang 	case QUERY_READ_DESC:
626eb5073f4SHaojian Zhuang 		memcpy((void *)buf,
627eb5073f4SHaojian Zhuang 		       (void *)(utrd.resp_upiu + sizeof(query_resp_upiu_t)),
628eb5073f4SHaojian Zhuang 		       size);
629eb5073f4SHaojian Zhuang 		break;
630a4755183Sanans 	case QUERY_READ_ATTR:
631a4755183Sanans 		*(uint32_t *)buf = htobe32(resp->ts.attr.value);
632a4755183Sanans 		break;
6335aa7498aSJonathan Wright 	default:
6345aa7498aSJonathan Wright 		/* Do nothing in default case */
6355aa7498aSJonathan Wright 		break;
636eb5073f4SHaojian Zhuang 	}
637eb5073f4SHaojian Zhuang 	(void)result;
638eb5073f4SHaojian Zhuang }
639eb5073f4SHaojian Zhuang 
640eb5073f4SHaojian Zhuang unsigned int ufs_read_attr(int idn)
641eb5073f4SHaojian Zhuang {
642eb5073f4SHaojian Zhuang 	unsigned int value;
643eb5073f4SHaojian Zhuang 
644eb5073f4SHaojian Zhuang 	ufs_query(QUERY_READ_ATTR, idn, 0, 0,
645eb5073f4SHaojian Zhuang 		  (uintptr_t)&value, sizeof(value));
646eb5073f4SHaojian Zhuang 	return value;
647eb5073f4SHaojian Zhuang }
648eb5073f4SHaojian Zhuang 
649eb5073f4SHaojian Zhuang void ufs_write_attr(int idn, unsigned int value)
650eb5073f4SHaojian Zhuang {
651eb5073f4SHaojian Zhuang 	ufs_query(QUERY_WRITE_ATTR, idn, 0, 0,
652eb5073f4SHaojian Zhuang 		  (uintptr_t)&value, sizeof(value));
653eb5073f4SHaojian Zhuang }
654eb5073f4SHaojian Zhuang 
655eb5073f4SHaojian Zhuang unsigned int ufs_read_flag(int idn)
656eb5073f4SHaojian Zhuang {
657eb5073f4SHaojian Zhuang 	unsigned int value;
658eb5073f4SHaojian Zhuang 
659eb5073f4SHaojian Zhuang 	ufs_query(QUERY_READ_FLAG, idn, 0, 0,
660eb5073f4SHaojian Zhuang 		  (uintptr_t)&value, sizeof(value));
661eb5073f4SHaojian Zhuang 	return value;
662eb5073f4SHaojian Zhuang }
663eb5073f4SHaojian Zhuang 
664eb5073f4SHaojian Zhuang void ufs_set_flag(int idn)
665eb5073f4SHaojian Zhuang {
666eb5073f4SHaojian Zhuang 	ufs_query(QUERY_SET_FLAG, idn, 0, 0, 0, 0);
667eb5073f4SHaojian Zhuang }
668eb5073f4SHaojian Zhuang 
669eb5073f4SHaojian Zhuang void ufs_clear_flag(int idn)
670eb5073f4SHaojian Zhuang {
671eb5073f4SHaojian Zhuang 	ufs_query(QUERY_CLEAR_FLAG, idn, 0, 0, 0, 0);
672eb5073f4SHaojian Zhuang }
673eb5073f4SHaojian Zhuang 
674eb5073f4SHaojian Zhuang void ufs_read_desc(int idn, int index, uintptr_t buf, size_t size)
675eb5073f4SHaojian Zhuang {
676eb5073f4SHaojian Zhuang 	ufs_query(QUERY_READ_DESC, idn, index, 0, buf, size);
677eb5073f4SHaojian Zhuang }
678eb5073f4SHaojian Zhuang 
679eb5073f4SHaojian Zhuang void ufs_write_desc(int idn, int index, uintptr_t buf, size_t size)
680eb5073f4SHaojian Zhuang {
681eb5073f4SHaojian Zhuang 	ufs_query(QUERY_WRITE_DESC, idn, index, 0, buf, size);
682eb5073f4SHaojian Zhuang }
683eb5073f4SHaojian Zhuang 
68428645ebdSRohit Ner static int ufs_read_capacity(int lun, unsigned int *num, unsigned int *size)
685eb5073f4SHaojian Zhuang {
686eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
687eb5073f4SHaojian Zhuang 	resp_upiu_t *resp;
688eb5073f4SHaojian Zhuang 	sense_data_t *sense;
689eb5073f4SHaojian Zhuang 	unsigned char data[CACHE_WRITEBACK_GRANULE << 1];
690eb5073f4SHaojian Zhuang 	uintptr_t buf;
69128645ebdSRohit Ner 	int retries = UFS_READ_CAPACITY_RETRIES;
692eb5073f4SHaojian Zhuang 
693eb5073f4SHaojian Zhuang 	assert((ufs_params.reg_base != 0) &&
694eb5073f4SHaojian Zhuang 	       (ufs_params.desc_base != 0) &&
695eb5073f4SHaojian Zhuang 	       (ufs_params.desc_size >= UFS_DESC_SIZE) &&
696eb5073f4SHaojian Zhuang 	       (num != NULL) && (size != NULL));
697eb5073f4SHaojian Zhuang 
698eb5073f4SHaojian Zhuang 	/* align buf address */
699eb5073f4SHaojian Zhuang 	buf = (uintptr_t)data;
700eb5073f4SHaojian Zhuang 	buf = (buf + CACHE_WRITEBACK_GRANULE - 1) &
701eb5073f4SHaojian Zhuang 	      ~(CACHE_WRITEBACK_GRANULE - 1);
702eb5073f4SHaojian Zhuang 	do {
7036e16f7f0Sanans 		ufs_send_cmd(&utrd, CDBCMD_READ_CAPACITY_10, lun, 0,
704eb5073f4SHaojian Zhuang 			    buf, READ_CAPACITY_LENGTH);
705eb5073f4SHaojian Zhuang #ifdef UFS_RESP_DEBUG
706eb5073f4SHaojian Zhuang 		dump_upiu(&utrd);
707eb5073f4SHaojian Zhuang #endif
708eb5073f4SHaojian Zhuang 		resp = (resp_upiu_t *)utrd.resp_upiu;
709eb5073f4SHaojian Zhuang 		sense = &resp->sd.sense;
71028645ebdSRohit Ner 		if (!((sense->resp_code == SENSE_DATA_VALID) &&
71128645ebdSRohit Ner 		    (sense->sense_key == SENSE_KEY_UNIT_ATTENTION) &&
71228645ebdSRohit Ner 		    (sense->asc == 0x29) && (sense->ascq == 0))) {
713eb5073f4SHaojian Zhuang 			inv_dcache_range(buf, CACHE_WRITEBACK_GRANULE);
714eb5073f4SHaojian Zhuang 			/* last logical block address */
715eb5073f4SHaojian Zhuang 			*num = be32toh(*(unsigned int *)buf);
716eb5073f4SHaojian Zhuang 			if (*num)
717eb5073f4SHaojian Zhuang 				*num += 1;
718eb5073f4SHaojian Zhuang 			/* logical block length in bytes */
719eb5073f4SHaojian Zhuang 			*size = be32toh(*(unsigned int *)(buf + 4));
72028645ebdSRohit Ner 
72128645ebdSRohit Ner 			return 0;
72228645ebdSRohit Ner 		}
72328645ebdSRohit Ner 
72428645ebdSRohit Ner 	} while (retries-- > 0);
72528645ebdSRohit Ner 
72628645ebdSRohit Ner 	return -ETIMEDOUT;
727eb5073f4SHaojian Zhuang }
728eb5073f4SHaojian Zhuang 
729eb5073f4SHaojian Zhuang size_t ufs_read_blocks(int lun, int lba, uintptr_t buf, size_t size)
730eb5073f4SHaojian Zhuang {
731eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
732eb5073f4SHaojian Zhuang 	resp_upiu_t *resp;
733eb5073f4SHaojian Zhuang 
734eb5073f4SHaojian Zhuang 	assert((ufs_params.reg_base != 0) &&
735eb5073f4SHaojian Zhuang 	       (ufs_params.desc_base != 0) &&
736eb5073f4SHaojian Zhuang 	       (ufs_params.desc_size >= UFS_DESC_SIZE));
737eb5073f4SHaojian Zhuang 
7386e16f7f0Sanans 	ufs_send_cmd(&utrd, CDBCMD_READ_10, lun, lba, buf, size);
739eb5073f4SHaojian Zhuang #ifdef UFS_RESP_DEBUG
740eb5073f4SHaojian Zhuang 	dump_upiu(&utrd);
741eb5073f4SHaojian Zhuang #endif
74238a5ecb7SChannagoud kadabi 	/*
74338a5ecb7SChannagoud kadabi 	 * Invalidate prefetched cache contents before cpu
74438a5ecb7SChannagoud kadabi 	 * accesses the buf.
74538a5ecb7SChannagoud kadabi 	 */
74638a5ecb7SChannagoud kadabi 	inv_dcache_range(buf, size);
747eb5073f4SHaojian Zhuang 	resp = (resp_upiu_t *)utrd.resp_upiu;
748eb5073f4SHaojian Zhuang 	return size - resp->res_trans_cnt;
749eb5073f4SHaojian Zhuang }
750eb5073f4SHaojian Zhuang 
751eb5073f4SHaojian Zhuang size_t ufs_write_blocks(int lun, int lba, const uintptr_t buf, size_t size)
752eb5073f4SHaojian Zhuang {
753eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
754eb5073f4SHaojian Zhuang 	resp_upiu_t *resp;
755eb5073f4SHaojian Zhuang 
756eb5073f4SHaojian Zhuang 	assert((ufs_params.reg_base != 0) &&
757eb5073f4SHaojian Zhuang 	       (ufs_params.desc_base != 0) &&
758eb5073f4SHaojian Zhuang 	       (ufs_params.desc_size >= UFS_DESC_SIZE));
759eb5073f4SHaojian Zhuang 
7606e16f7f0Sanans 	ufs_send_cmd(&utrd, CDBCMD_WRITE_10, lun, lba, buf, size);
761eb5073f4SHaojian Zhuang #ifdef UFS_RESP_DEBUG
762eb5073f4SHaojian Zhuang 	dump_upiu(&utrd);
763eb5073f4SHaojian Zhuang #endif
764eb5073f4SHaojian Zhuang 	resp = (resp_upiu_t *)utrd.resp_upiu;
765eb5073f4SHaojian Zhuang 	return size - resp->res_trans_cnt;
766eb5073f4SHaojian Zhuang }
767eb5073f4SHaojian Zhuang 
76850593e69Sanans static int ufs_set_fdevice_init(void)
76950593e69Sanans {
77050593e69Sanans 	unsigned int result;
77150593e69Sanans 	int timeout;
77250593e69Sanans 
77350593e69Sanans 	ufs_set_flag(FLAG_DEVICE_INIT);
77450593e69Sanans 
77550593e69Sanans 	timeout = FDEVICEINIT_TIMEOUT_MS;
77650593e69Sanans 	do {
77750593e69Sanans 		result = ufs_read_flag(FLAG_DEVICE_INIT);
77850593e69Sanans 		if (!result) {
77950593e69Sanans 			break;
78050593e69Sanans 		}
78150593e69Sanans 		mdelay(5);
78250593e69Sanans 		timeout -= 5;
78350593e69Sanans 	} while (timeout > 0);
78450593e69Sanans 
78550593e69Sanans 	if (result != 0U) {
78650593e69Sanans 		return -ETIMEDOUT;
78750593e69Sanans 	}
78850593e69Sanans 
78950593e69Sanans 	return 0;
79050593e69Sanans }
79150593e69Sanans 
792eb5073f4SHaojian Zhuang static void ufs_enum(void)
793eb5073f4SHaojian Zhuang {
794eb5073f4SHaojian Zhuang 	unsigned int blk_num, blk_size;
79550593e69Sanans 	int i, result;
796eb5073f4SHaojian Zhuang 
7979d6d1a94Sanans 	mmio_write_32(ufs_params.reg_base + UTRLBA,
7989d6d1a94Sanans 		      ufs_params.desc_base & UINT32_MAX);
7999d6d1a94Sanans 	mmio_write_32(ufs_params.reg_base + UTRLBAU,
8009d6d1a94Sanans 		      (ufs_params.desc_base >> 32) & UINT32_MAX);
8019d6d1a94Sanans 
802eb5073f4SHaojian Zhuang 	ufs_verify_init();
803eb5073f4SHaojian Zhuang 	ufs_verify_ready();
804eb5073f4SHaojian Zhuang 
80550593e69Sanans 	result = ufs_set_fdevice_init();
80650593e69Sanans 	assert(result == 0);
80750593e69Sanans 
80828645ebdSRohit Ner 	blk_num = 0;
80928645ebdSRohit Ner 	blk_size = 0;
81028645ebdSRohit Ner 
811eb5073f4SHaojian Zhuang 	/* dump available LUNs */
812eb5073f4SHaojian Zhuang 	for (i = 0; i < UFS_MAX_LUNS; i++) {
81328645ebdSRohit Ner 		result = ufs_read_capacity(i, &blk_num, &blk_size);
81428645ebdSRohit Ner 		if (result != 0) {
81528645ebdSRohit Ner 			WARN("UFS LUN%d dump failed\n", i);
81628645ebdSRohit Ner 		}
817eb5073f4SHaojian Zhuang 		if (blk_num && blk_size) {
818eb5073f4SHaojian Zhuang 			INFO("UFS LUN%d contains %d blocks with %d-byte size\n",
819eb5073f4SHaojian Zhuang 			     i, blk_num, blk_size);
820eb5073f4SHaojian Zhuang 		}
821eb5073f4SHaojian Zhuang 	}
82250593e69Sanans 
82350593e69Sanans 	(void)result;
824eb5073f4SHaojian Zhuang }
825eb5073f4SHaojian Zhuang 
8265ac25de6Sfengbaopeng static void ufs_get_device_info(struct ufs_dev_desc *card_data)
8275ac25de6Sfengbaopeng {
8285ac25de6Sfengbaopeng 	uint8_t desc_buf[DESC_DEVICE_MAX_SIZE];
8295ac25de6Sfengbaopeng 
8305ac25de6Sfengbaopeng 	ufs_query(QUERY_READ_DESC, DESC_TYPE_DEVICE, 0, 0,
8315ac25de6Sfengbaopeng 				(uintptr_t)desc_buf, DESC_DEVICE_MAX_SIZE);
8325ac25de6Sfengbaopeng 
8335ac25de6Sfengbaopeng 	/*
8345ac25de6Sfengbaopeng 	 * getting vendor (manufacturerID) and Bank Index in big endian
8355ac25de6Sfengbaopeng 	 * format
8365ac25de6Sfengbaopeng 	 */
8375ac25de6Sfengbaopeng 	card_data->wmanufacturerid = (uint16_t)((desc_buf[DEVICE_DESC_PARAM_MANF_ID] << 8) |
8385ac25de6Sfengbaopeng 				     (desc_buf[DEVICE_DESC_PARAM_MANF_ID + 1]));
8395ac25de6Sfengbaopeng }
8405ac25de6Sfengbaopeng 
841eb5073f4SHaojian Zhuang int ufs_init(const ufs_ops_t *ops, ufs_params_t *params)
842eb5073f4SHaojian Zhuang {
843eb5073f4SHaojian Zhuang 	int result;
844eb5073f4SHaojian Zhuang 	unsigned int data;
845eb5073f4SHaojian Zhuang 	uic_cmd_t cmd;
8465ac25de6Sfengbaopeng 	struct ufs_dev_desc card = {0};
847eb5073f4SHaojian Zhuang 
848eb5073f4SHaojian Zhuang 	assert((params != NULL) &&
849eb5073f4SHaojian Zhuang 	       (params->reg_base != 0) &&
850eb5073f4SHaojian Zhuang 	       (params->desc_base != 0) &&
851eb5073f4SHaojian Zhuang 	       (params->desc_size >= UFS_DESC_SIZE));
852eb5073f4SHaojian Zhuang 
853eb5073f4SHaojian Zhuang 	memcpy(&ufs_params, params, sizeof(ufs_params_t));
854eb5073f4SHaojian Zhuang 
8550956319bSanans 	/* 0 means 1 slot */
8560956319bSanans 	nutrs = (mmio_read_32(ufs_params.reg_base + CAP) & CAP_NUTRS_MASK) + 1;
8570956319bSanans 	if (nutrs > (ufs_params.desc_size / UFS_DESC_SIZE)) {
8580956319bSanans 		nutrs = ufs_params.desc_size / UFS_DESC_SIZE;
8590956319bSanans 	}
8600956319bSanans 
8610956319bSanans 
862eb5073f4SHaojian Zhuang 	if (ufs_params.flags & UFS_FLAGS_SKIPINIT) {
8639d6d1a94Sanans 		mmio_write_32(ufs_params.reg_base + UTRLBA,
8649d6d1a94Sanans 			      ufs_params.desc_base & UINT32_MAX);
8659d6d1a94Sanans 		mmio_write_32(ufs_params.reg_base + UTRLBAU,
8669d6d1a94Sanans 			      (ufs_params.desc_base >> 32) & UINT32_MAX);
8679d6d1a94Sanans 
868eb5073f4SHaojian Zhuang 		result = ufshc_dme_get(0x1571, 0, &data);
869eb5073f4SHaojian Zhuang 		assert(result == 0);
870eb5073f4SHaojian Zhuang 		result = ufshc_dme_get(0x41, 0, &data);
871eb5073f4SHaojian Zhuang 		assert(result == 0);
872eb5073f4SHaojian Zhuang 		if (data == 1) {
873eb5073f4SHaojian Zhuang 			/* prepare to exit hibernate mode */
874eb5073f4SHaojian Zhuang 			memset(&cmd, 0, sizeof(uic_cmd_t));
875eb5073f4SHaojian Zhuang 			cmd.op = DME_HIBERNATE_EXIT;
876eb5073f4SHaojian Zhuang 			result = ufshc_send_uic_cmd(ufs_params.reg_base,
877eb5073f4SHaojian Zhuang 						    &cmd);
878eb5073f4SHaojian Zhuang 			assert(result == 0);
879eb5073f4SHaojian Zhuang 			data = mmio_read_32(ufs_params.reg_base + UCMDARG2);
880eb5073f4SHaojian Zhuang 			assert(data == 0);
881eb5073f4SHaojian Zhuang 			do {
882eb5073f4SHaojian Zhuang 				data = mmio_read_32(ufs_params.reg_base + IS);
883eb5073f4SHaojian Zhuang 			} while ((data & UFS_INT_UHXS) == 0);
884eb5073f4SHaojian Zhuang 			mmio_write_32(ufs_params.reg_base + IS, UFS_INT_UHXS);
885eb5073f4SHaojian Zhuang 			data = mmio_read_32(ufs_params.reg_base + HCS);
886eb5073f4SHaojian Zhuang 			assert((data & HCS_UPMCRS_MASK) == HCS_PWR_LOCAL);
887eb5073f4SHaojian Zhuang 		}
888eb5073f4SHaojian Zhuang 		result = ufshc_dme_get(0x1568, 0, &data);
889eb5073f4SHaojian Zhuang 		assert(result == 0);
890eb5073f4SHaojian Zhuang 		assert((data > 0) && (data <= 3));
891eb5073f4SHaojian Zhuang 	} else {
892eb5073f4SHaojian Zhuang 		assert((ops != NULL) && (ops->phy_init != NULL) &&
893eb5073f4SHaojian Zhuang 		       (ops->phy_set_pwr_mode != NULL));
894eb5073f4SHaojian Zhuang 
89599ff1a35SJorge Troncoso 		result = ufshc_reset(ufs_params.reg_base);
89699ff1a35SJorge Troncoso 		assert(result == 0);
897eb5073f4SHaojian Zhuang 		ops->phy_init(&ufs_params);
898eb5073f4SHaojian Zhuang 		result = ufshc_link_startup(ufs_params.reg_base);
899eb5073f4SHaojian Zhuang 		assert(result == 0);
9005ac25de6Sfengbaopeng 
9015ac25de6Sfengbaopeng 		ufs_enum();
9025ac25de6Sfengbaopeng 
9035ac25de6Sfengbaopeng 		ufs_get_device_info(&card);
9045ac25de6Sfengbaopeng 		if (card.wmanufacturerid == UFS_VENDOR_SKHYNIX) {
9055ac25de6Sfengbaopeng 			ufs_params.flags |= UFS_FLAGS_VENDOR_SKHYNIX;
9065ac25de6Sfengbaopeng 		}
9075ac25de6Sfengbaopeng 
908eb5073f4SHaojian Zhuang 		ops->phy_set_pwr_mode(&ufs_params);
909eb5073f4SHaojian Zhuang 	}
910eb5073f4SHaojian Zhuang 
911eb5073f4SHaojian Zhuang 	(void)result;
912eb5073f4SHaojian Zhuang 	return 0;
913eb5073f4SHaojian Zhuang }
914