xref: /rk3399_ARM-atf/drivers/ufs/ufs.c (revision 5ac25de695520bb60b54bbe91a66c58ba28bde42)
1eb5073f4SHaojian Zhuang /*
2eb5073f4SHaojian Zhuang  * Copyright (c) 2017, ARM Limited and Contributors. All rights reserved.
3eb5073f4SHaojian Zhuang  *
4eb5073f4SHaojian Zhuang  * SPDX-License-Identifier: BSD-3-Clause
5eb5073f4SHaojian Zhuang  */
6eb5073f4SHaojian Zhuang 
7eb5073f4SHaojian Zhuang #include <arch_helpers.h>
8eb5073f4SHaojian Zhuang #include <assert.h>
9eb5073f4SHaojian Zhuang #include <debug.h>
10eb5073f4SHaojian Zhuang #include <delay_timer.h>
11eb5073f4SHaojian Zhuang #include <endian.h>
12eb5073f4SHaojian Zhuang #include <errno.h>
13eb5073f4SHaojian Zhuang #include <mmio.h>
14eb5073f4SHaojian Zhuang #include <platform_def.h>
15eb5073f4SHaojian Zhuang #include <stdint.h>
16eb5073f4SHaojian Zhuang #include <string.h>
17eb5073f4SHaojian Zhuang #include <ufs.h>
18eb5073f4SHaojian Zhuang 
19eb5073f4SHaojian Zhuang #define CDB_ADDR_MASK			127
20eb5073f4SHaojian Zhuang #define ALIGN_CDB(x)			(((x) + CDB_ADDR_MASK) & ~CDB_ADDR_MASK)
21eb5073f4SHaojian Zhuang #define ALIGN_8(x)			(((x) + 7) & ~7)
22eb5073f4SHaojian Zhuang 
23eb5073f4SHaojian Zhuang #define UFS_DESC_SIZE			0x400
24eb5073f4SHaojian Zhuang #define MAX_UFS_DESC_SIZE		0x8000		/* 32 descriptors */
25eb5073f4SHaojian Zhuang 
26eb5073f4SHaojian Zhuang #define MAX_PRDT_SIZE			0x40000		/* 256KB */
27eb5073f4SHaojian Zhuang 
28eb5073f4SHaojian Zhuang static ufs_params_t ufs_params;
29eb5073f4SHaojian Zhuang static int nutrs;	/* Number of UTP Transfer Request Slots */
30eb5073f4SHaojian Zhuang 
31eb5073f4SHaojian Zhuang int ufshc_send_uic_cmd(uintptr_t base, uic_cmd_t *cmd)
32eb5073f4SHaojian Zhuang {
33eb5073f4SHaojian Zhuang 	unsigned int data;
34eb5073f4SHaojian Zhuang 
35eb5073f4SHaojian Zhuang 	data = mmio_read_32(base + HCS);
36eb5073f4SHaojian Zhuang 	if ((data & HCS_UCRDY) == 0)
37eb5073f4SHaojian Zhuang 		return -EBUSY;
38eb5073f4SHaojian Zhuang 	mmio_write_32(base + IS, ~0);
39eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG1, cmd->arg1);
40eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG2, cmd->arg2);
41eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG3, cmd->arg3);
42eb5073f4SHaojian Zhuang 	mmio_write_32(base + UICCMD, cmd->op);
43eb5073f4SHaojian Zhuang 
44eb5073f4SHaojian Zhuang 	do {
45eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + IS);
46eb5073f4SHaojian Zhuang 	} while ((data & UFS_INT_UCCS) == 0);
47eb5073f4SHaojian Zhuang 	mmio_write_32(base + IS, UFS_INT_UCCS);
48101afa02SHaojian Zhuang 	return mmio_read_32(base + UCMDARG2) & CONFIG_RESULT_CODE_MASK;
49eb5073f4SHaojian Zhuang }
50eb5073f4SHaojian Zhuang 
51eb5073f4SHaojian Zhuang int ufshc_dme_get(unsigned int attr, unsigned int idx, unsigned int *val)
52eb5073f4SHaojian Zhuang {
53eb5073f4SHaojian Zhuang 	uintptr_t base;
54eb5073f4SHaojian Zhuang 	unsigned int data;
55eb5073f4SHaojian Zhuang 	int retries;
56eb5073f4SHaojian Zhuang 
57eb5073f4SHaojian Zhuang 	assert((ufs_params.reg_base != 0) && (val != NULL));
58eb5073f4SHaojian Zhuang 
59eb5073f4SHaojian Zhuang 	base = ufs_params.reg_base;
60eb5073f4SHaojian Zhuang 	for (retries = 0; retries < 100; retries++) {
61eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + HCS);
62eb5073f4SHaojian Zhuang 		if ((data & HCS_UCRDY) != 0)
63eb5073f4SHaojian Zhuang 			break;
64eb5073f4SHaojian Zhuang 		mdelay(1);
65eb5073f4SHaojian Zhuang 	}
66eb5073f4SHaojian Zhuang 	if (retries >= 100)
67eb5073f4SHaojian Zhuang 		return -EBUSY;
68eb5073f4SHaojian Zhuang 
69eb5073f4SHaojian Zhuang 	mmio_write_32(base + IS, ~0);
70eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG1, (attr << 16) | GEN_SELECTOR_IDX(idx));
71eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG2, 0);
72eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG3, 0);
73eb5073f4SHaojian Zhuang 	mmio_write_32(base + UICCMD, DME_GET);
74eb5073f4SHaojian Zhuang 	do {
75eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + IS);
76eb5073f4SHaojian Zhuang 		if (data & UFS_INT_UE)
77eb5073f4SHaojian Zhuang 			return -EINVAL;
78eb5073f4SHaojian Zhuang 	} while ((data & UFS_INT_UCCS) == 0);
79eb5073f4SHaojian Zhuang 	mmio_write_32(base + IS, UFS_INT_UCCS);
80101afa02SHaojian Zhuang 	data = mmio_read_32(base + UCMDARG2) & CONFIG_RESULT_CODE_MASK;
81eb5073f4SHaojian Zhuang 	assert(data == 0);
82eb5073f4SHaojian Zhuang 
83eb5073f4SHaojian Zhuang 	*val = mmio_read_32(base + UCMDARG3);
84eb5073f4SHaojian Zhuang 	return 0;
85eb5073f4SHaojian Zhuang }
86eb5073f4SHaojian Zhuang 
87eb5073f4SHaojian Zhuang int ufshc_dme_set(unsigned int attr, unsigned int idx, unsigned int val)
88eb5073f4SHaojian Zhuang {
89eb5073f4SHaojian Zhuang 	uintptr_t base;
90eb5073f4SHaojian Zhuang 	unsigned int data;
91eb5073f4SHaojian Zhuang 
92eb5073f4SHaojian Zhuang 	assert((ufs_params.reg_base != 0));
93eb5073f4SHaojian Zhuang 
94eb5073f4SHaojian Zhuang 	base = ufs_params.reg_base;
95eb5073f4SHaojian Zhuang 	data = mmio_read_32(base + HCS);
96eb5073f4SHaojian Zhuang 	if ((data & HCS_UCRDY) == 0)
97eb5073f4SHaojian Zhuang 		return -EBUSY;
98eb5073f4SHaojian Zhuang 	mmio_write_32(base + IS, ~0);
99eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG1, (attr << 16) | GEN_SELECTOR_IDX(idx));
100eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG2, 0);
101eb5073f4SHaojian Zhuang 	mmio_write_32(base + UCMDARG3, val);
102eb5073f4SHaojian Zhuang 	mmio_write_32(base + UICCMD, DME_SET);
103eb5073f4SHaojian Zhuang 	do {
104eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + IS);
105eb5073f4SHaojian Zhuang 		if (data & UFS_INT_UE)
106eb5073f4SHaojian Zhuang 			return -EINVAL;
107eb5073f4SHaojian Zhuang 	} while ((data & UFS_INT_UCCS) == 0);
108eb5073f4SHaojian Zhuang 	mmio_write_32(base + IS, UFS_INT_UCCS);
109101afa02SHaojian Zhuang 	data = mmio_read_32(base + UCMDARG2) & CONFIG_RESULT_CODE_MASK;
110eb5073f4SHaojian Zhuang 	assert(data == 0);
111eb5073f4SHaojian Zhuang 	return 0;
112eb5073f4SHaojian Zhuang }
113eb5073f4SHaojian Zhuang 
114eb5073f4SHaojian Zhuang static void ufshc_reset(uintptr_t base)
115eb5073f4SHaojian Zhuang {
116eb5073f4SHaojian Zhuang 	unsigned int data;
117eb5073f4SHaojian Zhuang 
118eb5073f4SHaojian Zhuang 	/* Enable Host Controller */
119eb5073f4SHaojian Zhuang 	mmio_write_32(base + HCE, HCE_ENABLE);
120eb5073f4SHaojian Zhuang 	/* Wait until basic initialization sequence completed */
121eb5073f4SHaojian Zhuang 	do {
122eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + HCE);
123eb5073f4SHaojian Zhuang 	} while ((data & HCE_ENABLE) == 0);
124eb5073f4SHaojian Zhuang 
125eb5073f4SHaojian Zhuang 	/* Enable Interrupts */
126eb5073f4SHaojian Zhuang 	data = UFS_INT_UCCS | UFS_INT_ULSS | UFS_INT_UE | UFS_INT_UTPES |
127eb5073f4SHaojian Zhuang 	       UFS_INT_DFES | UFS_INT_HCFES | UFS_INT_SBFES;
128eb5073f4SHaojian Zhuang 	mmio_write_32(base + IE, data);
129eb5073f4SHaojian Zhuang }
130eb5073f4SHaojian Zhuang 
131eb5073f4SHaojian Zhuang static int ufshc_link_startup(uintptr_t base)
132eb5073f4SHaojian Zhuang {
133eb5073f4SHaojian Zhuang 	uic_cmd_t cmd;
134eb5073f4SHaojian Zhuang 	int data, result;
135eb5073f4SHaojian Zhuang 	int retries;
136eb5073f4SHaojian Zhuang 
137eb5073f4SHaojian Zhuang 	for (retries = 10; retries > 0; retries--) {
138eb5073f4SHaojian Zhuang 		memset(&cmd, 0, sizeof(cmd));
139eb5073f4SHaojian Zhuang 		cmd.op = DME_LINKSTARTUP;
140eb5073f4SHaojian Zhuang 		result = ufshc_send_uic_cmd(base, &cmd);
141eb5073f4SHaojian Zhuang 		if (result != 0)
142eb5073f4SHaojian Zhuang 			continue;
143eb5073f4SHaojian Zhuang 		while ((mmio_read_32(base + HCS) & HCS_DP) == 0)
144eb5073f4SHaojian Zhuang 			;
145eb5073f4SHaojian Zhuang 		data = mmio_read_32(base + IS);
146eb5073f4SHaojian Zhuang 		if (data & UFS_INT_ULSS)
147eb5073f4SHaojian Zhuang 			mmio_write_32(base + IS, UFS_INT_ULSS);
148eb5073f4SHaojian Zhuang 		return 0;
149eb5073f4SHaojian Zhuang 	}
150eb5073f4SHaojian Zhuang 	return -EIO;
151eb5073f4SHaojian Zhuang }
152eb5073f4SHaojian Zhuang 
153eb5073f4SHaojian Zhuang /* Check Door Bell register to get an empty slot */
154eb5073f4SHaojian Zhuang static int get_empty_slot(int *slot)
155eb5073f4SHaojian Zhuang {
156eb5073f4SHaojian Zhuang 	unsigned int data;
157eb5073f4SHaojian Zhuang 	int i;
158eb5073f4SHaojian Zhuang 
159eb5073f4SHaojian Zhuang 	data = mmio_read_32(ufs_params.reg_base + UTRLDBR);
160eb5073f4SHaojian Zhuang 	for (i = 0; i < nutrs; i++) {
161eb5073f4SHaojian Zhuang 		if ((data & 1) == 0)
162eb5073f4SHaojian Zhuang 			break;
163eb5073f4SHaojian Zhuang 		data = data >> 1;
164eb5073f4SHaojian Zhuang 	}
165eb5073f4SHaojian Zhuang 	if (i >= nutrs)
166eb5073f4SHaojian Zhuang 		return -EBUSY;
167eb5073f4SHaojian Zhuang 	*slot = i;
168eb5073f4SHaojian Zhuang 	return 0;
169eb5073f4SHaojian Zhuang }
170eb5073f4SHaojian Zhuang 
171eb5073f4SHaojian Zhuang static void get_utrd(utp_utrd_t *utrd)
172eb5073f4SHaojian Zhuang {
173eb5073f4SHaojian Zhuang 	uintptr_t base;
174eb5073f4SHaojian Zhuang 	int slot = 0, result;
175eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
176eb5073f4SHaojian Zhuang 
177eb5073f4SHaojian Zhuang 	assert(utrd != NULL);
178eb5073f4SHaojian Zhuang 	result = get_empty_slot(&slot);
179eb5073f4SHaojian Zhuang 	assert(result == 0);
180eb5073f4SHaojian Zhuang 
181eb5073f4SHaojian Zhuang 	/* clear utrd */
182eb5073f4SHaojian Zhuang 	memset((void *)utrd, 0, sizeof(utp_utrd_t));
183eb5073f4SHaojian Zhuang 	base = ufs_params.desc_base + (slot * UFS_DESC_SIZE);
184eb5073f4SHaojian Zhuang 	/* clear the descriptor */
185eb5073f4SHaojian Zhuang 	memset((void *)base, 0, UFS_DESC_SIZE);
186eb5073f4SHaojian Zhuang 
187eb5073f4SHaojian Zhuang 	utrd->header = base;
188eb5073f4SHaojian Zhuang 	utrd->task_tag = slot + 1;
189eb5073f4SHaojian Zhuang 	/* CDB address should be aligned with 128 bytes */
190eb5073f4SHaojian Zhuang 	utrd->upiu = ALIGN_CDB(utrd->header + sizeof(utrd_header_t));
191eb5073f4SHaojian Zhuang 	utrd->resp_upiu = ALIGN_8(utrd->upiu + sizeof(cmd_upiu_t));
192eb5073f4SHaojian Zhuang 	utrd->size_upiu = utrd->resp_upiu - utrd->upiu;
193eb5073f4SHaojian Zhuang 	utrd->size_resp_upiu = ALIGN_8(sizeof(resp_upiu_t));
194eb5073f4SHaojian Zhuang 	utrd->prdt = utrd->resp_upiu + utrd->size_resp_upiu;
195eb5073f4SHaojian Zhuang 
196eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
197eb5073f4SHaojian Zhuang 	hd->ucdba = utrd->upiu & UINT32_MAX;
198eb5073f4SHaojian Zhuang 	hd->ucdbau = (utrd->upiu >> 32) & UINT32_MAX;
199eb5073f4SHaojian Zhuang 	/* Both RUL and RUO is based on DWORD */
200eb5073f4SHaojian Zhuang 	hd->rul = utrd->size_resp_upiu >> 2;
201eb5073f4SHaojian Zhuang 	hd->ruo = utrd->size_upiu >> 2;
202eb5073f4SHaojian Zhuang 	(void)result;
203eb5073f4SHaojian Zhuang }
204eb5073f4SHaojian Zhuang 
205eb5073f4SHaojian Zhuang /*
206eb5073f4SHaojian Zhuang  * Prepare UTRD, Command UPIU, Response UPIU.
207eb5073f4SHaojian Zhuang  */
208eb5073f4SHaojian Zhuang static int ufs_prepare_cmd(utp_utrd_t *utrd, uint8_t op, uint8_t lun,
209eb5073f4SHaojian Zhuang 			   int lba, uintptr_t buf, size_t length)
210eb5073f4SHaojian Zhuang {
211eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
212eb5073f4SHaojian Zhuang 	cmd_upiu_t *upiu;
213eb5073f4SHaojian Zhuang 	prdt_t *prdt;
214eb5073f4SHaojian Zhuang 	unsigned int ulba;
215eb5073f4SHaojian Zhuang 	unsigned int lba_cnt;
216eb5073f4SHaojian Zhuang 	int prdt_size;
217eb5073f4SHaojian Zhuang 
218eb5073f4SHaojian Zhuang 
219eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + UTRLBA,
220eb5073f4SHaojian Zhuang 		      utrd->header & UINT32_MAX);
221eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + UTRLBAU,
222eb5073f4SHaojian Zhuang 		      (utrd->upiu >> 32) & UINT32_MAX);
223eb5073f4SHaojian Zhuang 
224eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
225eb5073f4SHaojian Zhuang 	upiu = (cmd_upiu_t *)utrd->upiu;
226eb5073f4SHaojian Zhuang 
227eb5073f4SHaojian Zhuang 	hd->i = 1;
228eb5073f4SHaojian Zhuang 	hd->ct = CT_UFS_STORAGE;
229eb5073f4SHaojian Zhuang 	hd->ocs = OCS_MASK;
230eb5073f4SHaojian Zhuang 
231eb5073f4SHaojian Zhuang 	upiu->trans_type = CMD_UPIU;
232eb5073f4SHaojian Zhuang 	upiu->task_tag = utrd->task_tag;
233eb5073f4SHaojian Zhuang 	upiu->cdb[0] = op;
234eb5073f4SHaojian Zhuang 	ulba = (unsigned int)lba;
235eb5073f4SHaojian Zhuang 	lba_cnt = (unsigned int)(length >> UFS_BLOCK_SHIFT);
236eb5073f4SHaojian Zhuang 	switch (op) {
237eb5073f4SHaojian Zhuang 	case CDBCMD_TEST_UNIT_READY:
238eb5073f4SHaojian Zhuang 		break;
239eb5073f4SHaojian Zhuang 	case CDBCMD_READ_CAPACITY_10:
240eb5073f4SHaojian Zhuang 		hd->dd = DD_OUT;
241eb5073f4SHaojian Zhuang 		upiu->flags = UPIU_FLAGS_R | UPIU_FLAGS_ATTR_S;
242eb5073f4SHaojian Zhuang 		upiu->lun = lun;
243eb5073f4SHaojian Zhuang 		break;
244eb5073f4SHaojian Zhuang 	case CDBCMD_READ_10:
245eb5073f4SHaojian Zhuang 		hd->dd = DD_OUT;
246eb5073f4SHaojian Zhuang 		upiu->flags = UPIU_FLAGS_R | UPIU_FLAGS_ATTR_S;
247eb5073f4SHaojian Zhuang 		upiu->lun = lun;
248eb5073f4SHaojian Zhuang 		upiu->cdb[1] = RW_WITHOUT_CACHE;
249eb5073f4SHaojian Zhuang 		/* set logical block address */
250eb5073f4SHaojian Zhuang 		upiu->cdb[2] = (ulba >> 24) & 0xff;
251eb5073f4SHaojian Zhuang 		upiu->cdb[3] = (ulba >> 16) & 0xff;
252eb5073f4SHaojian Zhuang 		upiu->cdb[4] = (ulba >> 8) & 0xff;
253eb5073f4SHaojian Zhuang 		upiu->cdb[5] = ulba & 0xff;
254eb5073f4SHaojian Zhuang 		/* set transfer length */
255eb5073f4SHaojian Zhuang 		upiu->cdb[7] = (lba_cnt >> 8) & 0xff;
256eb5073f4SHaojian Zhuang 		upiu->cdb[8] = lba_cnt & 0xff;
257eb5073f4SHaojian Zhuang 		break;
258eb5073f4SHaojian Zhuang 	case CDBCMD_WRITE_10:
259eb5073f4SHaojian Zhuang 		hd->dd = DD_IN;
260eb5073f4SHaojian Zhuang 		upiu->flags = UPIU_FLAGS_W | UPIU_FLAGS_ATTR_S;
261eb5073f4SHaojian Zhuang 		upiu->lun = lun;
262eb5073f4SHaojian Zhuang 		upiu->cdb[1] = RW_WITHOUT_CACHE;
263eb5073f4SHaojian Zhuang 		/* set logical block address */
264eb5073f4SHaojian Zhuang 		upiu->cdb[2] = (ulba >> 24) & 0xff;
265eb5073f4SHaojian Zhuang 		upiu->cdb[3] = (ulba >> 16) & 0xff;
266eb5073f4SHaojian Zhuang 		upiu->cdb[4] = (ulba >> 8) & 0xff;
267eb5073f4SHaojian Zhuang 		upiu->cdb[5] = ulba & 0xff;
268eb5073f4SHaojian Zhuang 		/* set transfer length */
269eb5073f4SHaojian Zhuang 		upiu->cdb[7] = (lba_cnt >> 8) & 0xff;
270eb5073f4SHaojian Zhuang 		upiu->cdb[8] = lba_cnt & 0xff;
271eb5073f4SHaojian Zhuang 		break;
272eb5073f4SHaojian Zhuang 	default:
273eb5073f4SHaojian Zhuang 		assert(0);
274eb5073f4SHaojian Zhuang 	}
275eb5073f4SHaojian Zhuang 	if (hd->dd == DD_IN)
276eb5073f4SHaojian Zhuang 		flush_dcache_range(buf, length);
277eb5073f4SHaojian Zhuang 	else if (hd->dd == DD_OUT)
278eb5073f4SHaojian Zhuang 		inv_dcache_range(buf, length);
279eb5073f4SHaojian Zhuang 	if (length) {
280eb5073f4SHaojian Zhuang 		upiu->exp_data_trans_len = htobe32(length);
281eb5073f4SHaojian Zhuang 		assert(lba_cnt <= UINT16_MAX);
282eb5073f4SHaojian Zhuang 		prdt = (prdt_t *)utrd->prdt;
283eb5073f4SHaojian Zhuang 
284eb5073f4SHaojian Zhuang 		prdt_size = 0;
285eb5073f4SHaojian Zhuang 		while (length > 0) {
286eb5073f4SHaojian Zhuang 			prdt->dba = (unsigned int)(buf & UINT32_MAX);
287eb5073f4SHaojian Zhuang 			prdt->dbau = (unsigned int)((buf >> 32) & UINT32_MAX);
288eb5073f4SHaojian Zhuang 			/* prdt->dbc counts from 0 */
289eb5073f4SHaojian Zhuang 			if (length > MAX_PRDT_SIZE) {
290eb5073f4SHaojian Zhuang 				prdt->dbc = MAX_PRDT_SIZE - 1;
291eb5073f4SHaojian Zhuang 				length = length - MAX_PRDT_SIZE;
292eb5073f4SHaojian Zhuang 			} else {
293eb5073f4SHaojian Zhuang 				prdt->dbc = length - 1;
294eb5073f4SHaojian Zhuang 				length = 0;
295eb5073f4SHaojian Zhuang 			}
296eb5073f4SHaojian Zhuang 			buf += MAX_PRDT_SIZE;
297eb5073f4SHaojian Zhuang 			prdt++;
298eb5073f4SHaojian Zhuang 			prdt_size += sizeof(prdt_t);
299eb5073f4SHaojian Zhuang 		}
300eb5073f4SHaojian Zhuang 		utrd->size_prdt = ALIGN_8(prdt_size);
301eb5073f4SHaojian Zhuang 		hd->prdtl = utrd->size_prdt >> 2;
302eb5073f4SHaojian Zhuang 		hd->prdto = (utrd->size_upiu + utrd->size_resp_upiu) >> 2;
303eb5073f4SHaojian Zhuang 	}
304eb5073f4SHaojian Zhuang 
305eb5073f4SHaojian Zhuang 	flush_dcache_range((uintptr_t)utrd, sizeof(utp_utrd_t));
306eb5073f4SHaojian Zhuang 	flush_dcache_range((uintptr_t)utrd->header, UFS_DESC_SIZE);
307eb5073f4SHaojian Zhuang 	return 0;
308eb5073f4SHaojian Zhuang }
309eb5073f4SHaojian Zhuang 
310eb5073f4SHaojian Zhuang static int ufs_prepare_query(utp_utrd_t *utrd, uint8_t op, uint8_t idn,
311eb5073f4SHaojian Zhuang 			     uint8_t index, uint8_t sel,
312eb5073f4SHaojian Zhuang 			     uintptr_t buf, size_t length)
313eb5073f4SHaojian Zhuang {
314eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
315eb5073f4SHaojian Zhuang 	query_upiu_t *query_upiu;
316eb5073f4SHaojian Zhuang 
317eb5073f4SHaojian Zhuang 
318eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
319eb5073f4SHaojian Zhuang 	query_upiu = (query_upiu_t *)utrd->upiu;
320eb5073f4SHaojian Zhuang 
321eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + UTRLBA,
322eb5073f4SHaojian Zhuang 		      utrd->header & UINT32_MAX);
323eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + UTRLBAU,
324eb5073f4SHaojian Zhuang 		      (utrd->header >> 32) & UINT32_MAX);
325eb5073f4SHaojian Zhuang 
326eb5073f4SHaojian Zhuang 
327eb5073f4SHaojian Zhuang 	hd->i = 1;
328eb5073f4SHaojian Zhuang 	hd->ct = CT_UFS_STORAGE;
329eb5073f4SHaojian Zhuang 	hd->ocs = OCS_MASK;
330eb5073f4SHaojian Zhuang 
331eb5073f4SHaojian Zhuang 	query_upiu->trans_type = QUERY_REQUEST_UPIU;
332eb5073f4SHaojian Zhuang 	query_upiu->task_tag = utrd->task_tag;
333eb5073f4SHaojian Zhuang 	query_upiu->ts.desc.opcode = op;
334eb5073f4SHaojian Zhuang 	query_upiu->ts.desc.idn = idn;
335eb5073f4SHaojian Zhuang 	query_upiu->ts.desc.index = index;
336eb5073f4SHaojian Zhuang 	query_upiu->ts.desc.selector = sel;
337eb5073f4SHaojian Zhuang 	switch (op) {
338eb5073f4SHaojian Zhuang 	case QUERY_READ_DESC:
339eb5073f4SHaojian Zhuang 		query_upiu->query_func = QUERY_FUNC_STD_READ;
340eb5073f4SHaojian Zhuang 		query_upiu->ts.desc.length = htobe16(length);
341eb5073f4SHaojian Zhuang 		break;
342eb5073f4SHaojian Zhuang 	case QUERY_WRITE_DESC:
343eb5073f4SHaojian Zhuang 		query_upiu->query_func = QUERY_FUNC_STD_WRITE;
344eb5073f4SHaojian Zhuang 		query_upiu->ts.desc.length = htobe16(length);
345eb5073f4SHaojian Zhuang 		memcpy((void *)(utrd->upiu + sizeof(query_upiu_t)),
346eb5073f4SHaojian Zhuang 		       (void *)buf, length);
347eb5073f4SHaojian Zhuang 		break;
348eb5073f4SHaojian Zhuang 	case QUERY_READ_ATTR:
349eb5073f4SHaojian Zhuang 	case QUERY_READ_FLAG:
350eb5073f4SHaojian Zhuang 		query_upiu->query_func = QUERY_FUNC_STD_READ;
351eb5073f4SHaojian Zhuang 		break;
352eb5073f4SHaojian Zhuang 	case QUERY_CLEAR_FLAG:
353eb5073f4SHaojian Zhuang 	case QUERY_SET_FLAG:
354eb5073f4SHaojian Zhuang 		query_upiu->query_func = QUERY_FUNC_STD_WRITE;
355eb5073f4SHaojian Zhuang 		break;
356eb5073f4SHaojian Zhuang 	case QUERY_WRITE_ATTR:
357eb5073f4SHaojian Zhuang 		query_upiu->query_func = QUERY_FUNC_STD_WRITE;
358eb5073f4SHaojian Zhuang 		memcpy((void *)&query_upiu->ts.attr.value, (void *)buf, length);
359eb5073f4SHaojian Zhuang 		break;
360eb5073f4SHaojian Zhuang 	default:
361eb5073f4SHaojian Zhuang 		assert(0);
362eb5073f4SHaojian Zhuang 	}
363eb5073f4SHaojian Zhuang 	flush_dcache_range((uintptr_t)utrd, sizeof(utp_utrd_t));
364eb5073f4SHaojian Zhuang 	flush_dcache_range((uintptr_t)utrd->header, UFS_DESC_SIZE);
365eb5073f4SHaojian Zhuang 	return 0;
366eb5073f4SHaojian Zhuang }
367eb5073f4SHaojian Zhuang 
368eb5073f4SHaojian Zhuang static void ufs_prepare_nop_out(utp_utrd_t *utrd)
369eb5073f4SHaojian Zhuang {
370eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
371eb5073f4SHaojian Zhuang 	nop_out_upiu_t *nop_out;
372eb5073f4SHaojian Zhuang 
373eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + UTRLBA,
374eb5073f4SHaojian Zhuang 		      utrd->header & UINT32_MAX);
375eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + UTRLBAU,
376eb5073f4SHaojian Zhuang 		      (utrd->header >> 32) & UINT32_MAX);
377eb5073f4SHaojian Zhuang 
378eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
379eb5073f4SHaojian Zhuang 	nop_out = (nop_out_upiu_t *)utrd->upiu;
380eb5073f4SHaojian Zhuang 
381eb5073f4SHaojian Zhuang 	hd->i = 1;
382eb5073f4SHaojian Zhuang 	hd->ct = CT_UFS_STORAGE;
383eb5073f4SHaojian Zhuang 	hd->ocs = OCS_MASK;
384eb5073f4SHaojian Zhuang 
385eb5073f4SHaojian Zhuang 	nop_out->trans_type = 0;
386eb5073f4SHaojian Zhuang 	nop_out->task_tag = utrd->task_tag;
387eb5073f4SHaojian Zhuang 	flush_dcache_range((uintptr_t)utrd, sizeof(utp_utrd_t));
388eb5073f4SHaojian Zhuang 	flush_dcache_range((uintptr_t)utrd->header, UFS_DESC_SIZE);
389eb5073f4SHaojian Zhuang }
390eb5073f4SHaojian Zhuang 
391eb5073f4SHaojian Zhuang static void ufs_send_request(int task_tag)
392eb5073f4SHaojian Zhuang {
393eb5073f4SHaojian Zhuang 	unsigned int data;
394eb5073f4SHaojian Zhuang 	int slot;
395eb5073f4SHaojian Zhuang 
396eb5073f4SHaojian Zhuang 	slot = task_tag - 1;
397eb5073f4SHaojian Zhuang 	/* clear all interrupts */
398eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + IS, ~0);
399eb5073f4SHaojian Zhuang 
400eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + UTRLRSR, 1);
401eb5073f4SHaojian Zhuang 	do {
402eb5073f4SHaojian Zhuang 		data = mmio_read_32(ufs_params.reg_base + UTRLRSR);
403eb5073f4SHaojian Zhuang 	} while (data == 0);
404eb5073f4SHaojian Zhuang 
405eb5073f4SHaojian Zhuang 	data = UTRIACR_IAEN | UTRIACR_CTR | UTRIACR_IACTH(0x1F) |
406eb5073f4SHaojian Zhuang 	       UTRIACR_IATOVAL(0xFF);
407eb5073f4SHaojian Zhuang 	mmio_write_32(ufs_params.reg_base + UTRIACR, data);
408eb5073f4SHaojian Zhuang 	/* send request */
409eb5073f4SHaojian Zhuang 	mmio_setbits_32(ufs_params.reg_base + UTRLDBR, 1 << slot);
410eb5073f4SHaojian Zhuang }
411eb5073f4SHaojian Zhuang 
412eb5073f4SHaojian Zhuang static int ufs_check_resp(utp_utrd_t *utrd, int trans_type)
413eb5073f4SHaojian Zhuang {
414eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
415eb5073f4SHaojian Zhuang 	resp_upiu_t *resp;
416eb5073f4SHaojian Zhuang 	unsigned int data;
417eb5073f4SHaojian Zhuang 	int slot;
418eb5073f4SHaojian Zhuang 
419eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
420eb5073f4SHaojian Zhuang 	resp = (resp_upiu_t *)utrd->resp_upiu;
421eb5073f4SHaojian Zhuang 	inv_dcache_range((uintptr_t)hd, UFS_DESC_SIZE);
422eb5073f4SHaojian Zhuang 	inv_dcache_range((uintptr_t)utrd, sizeof(utp_utrd_t));
423eb5073f4SHaojian Zhuang 	do {
424eb5073f4SHaojian Zhuang 		data = mmio_read_32(ufs_params.reg_base + IS);
425eb5073f4SHaojian Zhuang 		if ((data & ~(UFS_INT_UCCS | UFS_INT_UTRCS)) != 0)
426eb5073f4SHaojian Zhuang 			return -EIO;
427eb5073f4SHaojian Zhuang 	} while ((data & UFS_INT_UTRCS) == 0);
428eb5073f4SHaojian Zhuang 	slot = utrd->task_tag - 1;
429eb5073f4SHaojian Zhuang 
430eb5073f4SHaojian Zhuang 	data = mmio_read_32(ufs_params.reg_base + UTRLDBR);
431eb5073f4SHaojian Zhuang 	assert((data & (1 << slot)) == 0);
432eb5073f4SHaojian Zhuang 	assert(hd->ocs == OCS_SUCCESS);
433eb5073f4SHaojian Zhuang 	assert((resp->trans_type & TRANS_TYPE_CODE_MASK) == trans_type);
434eb5073f4SHaojian Zhuang 	(void)resp;
435eb5073f4SHaojian Zhuang 	(void)slot;
436eb5073f4SHaojian Zhuang 	return 0;
437eb5073f4SHaojian Zhuang }
438eb5073f4SHaojian Zhuang 
439eb5073f4SHaojian Zhuang #ifdef UFS_RESP_DEBUG
440eb5073f4SHaojian Zhuang static void dump_upiu(utp_utrd_t *utrd)
441eb5073f4SHaojian Zhuang {
442eb5073f4SHaojian Zhuang 	utrd_header_t *hd;
443eb5073f4SHaojian Zhuang 	int i;
444eb5073f4SHaojian Zhuang 
445eb5073f4SHaojian Zhuang 	hd = (utrd_header_t *)utrd->header;
446eb5073f4SHaojian Zhuang 	INFO("utrd:0x%x, ruo:0x%x, rul:0x%x, ocs:0x%x, UTRLDBR:0x%x\n",
447eb5073f4SHaojian Zhuang 		(unsigned int)(uintptr_t)utrd, hd->ruo, hd->rul, hd->ocs,
448eb5073f4SHaojian Zhuang 		mmio_read_32(ufs_params.reg_base + UTRLDBR));
449eb5073f4SHaojian Zhuang 	for (i = 0; i < sizeof(utrd_header_t); i += 4) {
450eb5073f4SHaojian Zhuang 		INFO("[%lx]:0x%x\n",
451eb5073f4SHaojian Zhuang 			(uintptr_t)utrd->header + i,
452eb5073f4SHaojian Zhuang 			*(unsigned int *)((uintptr_t)utrd->header + i));
453eb5073f4SHaojian Zhuang 	}
454eb5073f4SHaojian Zhuang 
455eb5073f4SHaojian Zhuang 	for (i = 0; i < sizeof(cmd_upiu_t); i += 4) {
456eb5073f4SHaojian Zhuang 		INFO("cmd[%lx]:0x%x\n",
457eb5073f4SHaojian Zhuang 			utrd->upiu + i,
458eb5073f4SHaojian Zhuang 			*(unsigned int *)(utrd->upiu + i));
459eb5073f4SHaojian Zhuang 	}
460eb5073f4SHaojian Zhuang 	for (i = 0; i < sizeof(resp_upiu_t); i += 4) {
461eb5073f4SHaojian Zhuang 		INFO("resp[%lx]:0x%x\n",
462eb5073f4SHaojian Zhuang 			utrd->resp_upiu + i,
463eb5073f4SHaojian Zhuang 			*(unsigned int *)(utrd->resp_upiu + i));
464eb5073f4SHaojian Zhuang 	}
465eb5073f4SHaojian Zhuang 	for (i = 0; i < sizeof(prdt_t); i += 4) {
466eb5073f4SHaojian Zhuang 		INFO("prdt[%lx]:0x%x\n",
467eb5073f4SHaojian Zhuang 			utrd->prdt + i,
468eb5073f4SHaojian Zhuang 			*(unsigned int *)(utrd->prdt + i));
469eb5073f4SHaojian Zhuang 	}
470eb5073f4SHaojian Zhuang }
471eb5073f4SHaojian Zhuang #endif
472eb5073f4SHaojian Zhuang 
473eb5073f4SHaojian Zhuang static void ufs_verify_init(void)
474eb5073f4SHaojian Zhuang {
475eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
476eb5073f4SHaojian Zhuang 	int result;
477eb5073f4SHaojian Zhuang 
478eb5073f4SHaojian Zhuang 	get_utrd(&utrd);
479eb5073f4SHaojian Zhuang 	ufs_prepare_nop_out(&utrd);
480eb5073f4SHaojian Zhuang 	ufs_send_request(utrd.task_tag);
481eb5073f4SHaojian Zhuang 	result = ufs_check_resp(&utrd, NOP_IN_UPIU);
482eb5073f4SHaojian Zhuang 	assert(result == 0);
483eb5073f4SHaojian Zhuang 	(void)result;
484eb5073f4SHaojian Zhuang }
485eb5073f4SHaojian Zhuang 
486eb5073f4SHaojian Zhuang static void ufs_verify_ready(void)
487eb5073f4SHaojian Zhuang {
488eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
489eb5073f4SHaojian Zhuang 	int result;
490eb5073f4SHaojian Zhuang 
491eb5073f4SHaojian Zhuang 	get_utrd(&utrd);
492eb5073f4SHaojian Zhuang 	ufs_prepare_cmd(&utrd, CDBCMD_TEST_UNIT_READY, 0, 0, 0, 0);
493eb5073f4SHaojian Zhuang 	ufs_send_request(utrd.task_tag);
494eb5073f4SHaojian Zhuang 	result = ufs_check_resp(&utrd, RESPONSE_UPIU);
495eb5073f4SHaojian Zhuang 	assert(result == 0);
496eb5073f4SHaojian Zhuang 	(void)result;
497eb5073f4SHaojian Zhuang }
498eb5073f4SHaojian Zhuang 
499eb5073f4SHaojian Zhuang static void ufs_query(uint8_t op, uint8_t idn, uint8_t index, uint8_t sel,
500eb5073f4SHaojian Zhuang 		      uintptr_t buf, size_t size)
501eb5073f4SHaojian Zhuang {
502eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
503eb5073f4SHaojian Zhuang 	query_resp_upiu_t *resp;
504eb5073f4SHaojian Zhuang 	int result;
505eb5073f4SHaojian Zhuang 
506eb5073f4SHaojian Zhuang 	switch (op) {
507eb5073f4SHaojian Zhuang 	case QUERY_READ_FLAG:
508eb5073f4SHaojian Zhuang 	case QUERY_READ_ATTR:
509eb5073f4SHaojian Zhuang 	case QUERY_READ_DESC:
510eb5073f4SHaojian Zhuang 	case QUERY_WRITE_DESC:
511eb5073f4SHaojian Zhuang 	case QUERY_WRITE_ATTR:
512eb5073f4SHaojian Zhuang 		assert(((buf & 3) == 0) && (size != 0));
513eb5073f4SHaojian Zhuang 		break;
514eb5073f4SHaojian Zhuang 	}
515eb5073f4SHaojian Zhuang 	get_utrd(&utrd);
516eb5073f4SHaojian Zhuang 	ufs_prepare_query(&utrd, op, idn, index, sel, buf, size);
517eb5073f4SHaojian Zhuang 	ufs_send_request(utrd.task_tag);
518eb5073f4SHaojian Zhuang 	result = ufs_check_resp(&utrd, QUERY_RESPONSE_UPIU);
519eb5073f4SHaojian Zhuang 	assert(result == 0);
520eb5073f4SHaojian Zhuang 	resp = (query_resp_upiu_t *)utrd.resp_upiu;
521eb5073f4SHaojian Zhuang #ifdef UFS_RESP_DEBUG
522eb5073f4SHaojian Zhuang 	dump_upiu(&utrd);
523eb5073f4SHaojian Zhuang #endif
524eb5073f4SHaojian Zhuang 	assert(resp->query_resp == QUERY_RESP_SUCCESS);
525eb5073f4SHaojian Zhuang 
526eb5073f4SHaojian Zhuang 	switch (op) {
527eb5073f4SHaojian Zhuang 	case QUERY_READ_FLAG:
528eb5073f4SHaojian Zhuang 		*(uint32_t *)buf = (uint32_t)resp->ts.flag.value;
529eb5073f4SHaojian Zhuang 		break;
530eb5073f4SHaojian Zhuang 	case QUERY_READ_ATTR:
531eb5073f4SHaojian Zhuang 	case QUERY_READ_DESC:
532eb5073f4SHaojian Zhuang 		memcpy((void *)buf,
533eb5073f4SHaojian Zhuang 		       (void *)(utrd.resp_upiu + sizeof(query_resp_upiu_t)),
534eb5073f4SHaojian Zhuang 		       size);
535eb5073f4SHaojian Zhuang 		break;
536eb5073f4SHaojian Zhuang 	}
537eb5073f4SHaojian Zhuang 	(void)result;
538eb5073f4SHaojian Zhuang }
539eb5073f4SHaojian Zhuang 
540eb5073f4SHaojian Zhuang unsigned int ufs_read_attr(int idn)
541eb5073f4SHaojian Zhuang {
542eb5073f4SHaojian Zhuang 	unsigned int value;
543eb5073f4SHaojian Zhuang 
544eb5073f4SHaojian Zhuang 	ufs_query(QUERY_READ_ATTR, idn, 0, 0,
545eb5073f4SHaojian Zhuang 		  (uintptr_t)&value, sizeof(value));
546eb5073f4SHaojian Zhuang 	return value;
547eb5073f4SHaojian Zhuang }
548eb5073f4SHaojian Zhuang 
549eb5073f4SHaojian Zhuang void ufs_write_attr(int idn, unsigned int value)
550eb5073f4SHaojian Zhuang {
551eb5073f4SHaojian Zhuang 	ufs_query(QUERY_WRITE_ATTR, idn, 0, 0,
552eb5073f4SHaojian Zhuang 		  (uintptr_t)&value, sizeof(value));
553eb5073f4SHaojian Zhuang }
554eb5073f4SHaojian Zhuang 
555eb5073f4SHaojian Zhuang unsigned int ufs_read_flag(int idn)
556eb5073f4SHaojian Zhuang {
557eb5073f4SHaojian Zhuang 	unsigned int value;
558eb5073f4SHaojian Zhuang 
559eb5073f4SHaojian Zhuang 	ufs_query(QUERY_READ_FLAG, idn, 0, 0,
560eb5073f4SHaojian Zhuang 		  (uintptr_t)&value, sizeof(value));
561eb5073f4SHaojian Zhuang 	return value;
562eb5073f4SHaojian Zhuang }
563eb5073f4SHaojian Zhuang 
564eb5073f4SHaojian Zhuang void ufs_set_flag(int idn)
565eb5073f4SHaojian Zhuang {
566eb5073f4SHaojian Zhuang 	ufs_query(QUERY_SET_FLAG, idn, 0, 0, 0, 0);
567eb5073f4SHaojian Zhuang }
568eb5073f4SHaojian Zhuang 
569eb5073f4SHaojian Zhuang void ufs_clear_flag(int idn)
570eb5073f4SHaojian Zhuang {
571eb5073f4SHaojian Zhuang 	ufs_query(QUERY_CLEAR_FLAG, idn, 0, 0, 0, 0);
572eb5073f4SHaojian Zhuang }
573eb5073f4SHaojian Zhuang 
574eb5073f4SHaojian Zhuang void ufs_read_desc(int idn, int index, uintptr_t buf, size_t size)
575eb5073f4SHaojian Zhuang {
576eb5073f4SHaojian Zhuang 	ufs_query(QUERY_READ_DESC, idn, index, 0, buf, size);
577eb5073f4SHaojian Zhuang }
578eb5073f4SHaojian Zhuang 
579eb5073f4SHaojian Zhuang void ufs_write_desc(int idn, int index, uintptr_t buf, size_t size)
580eb5073f4SHaojian Zhuang {
581eb5073f4SHaojian Zhuang 	ufs_query(QUERY_WRITE_DESC, idn, index, 0, buf, size);
582eb5073f4SHaojian Zhuang }
583eb5073f4SHaojian Zhuang 
584eb5073f4SHaojian Zhuang void ufs_read_capacity(int lun, unsigned int *num, unsigned int *size)
585eb5073f4SHaojian Zhuang {
586eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
587eb5073f4SHaojian Zhuang 	resp_upiu_t *resp;
588eb5073f4SHaojian Zhuang 	sense_data_t *sense;
589eb5073f4SHaojian Zhuang 	unsigned char data[CACHE_WRITEBACK_GRANULE << 1];
590eb5073f4SHaojian Zhuang 	uintptr_t buf;
591eb5073f4SHaojian Zhuang 	int result;
592eb5073f4SHaojian Zhuang 	int retry;
593eb5073f4SHaojian Zhuang 
594eb5073f4SHaojian Zhuang 	assert((ufs_params.reg_base != 0) &&
595eb5073f4SHaojian Zhuang 	       (ufs_params.desc_base != 0) &&
596eb5073f4SHaojian Zhuang 	       (ufs_params.desc_size >= UFS_DESC_SIZE) &&
597eb5073f4SHaojian Zhuang 	       (num != NULL) && (size != NULL));
598eb5073f4SHaojian Zhuang 
599eb5073f4SHaojian Zhuang 	/* align buf address */
600eb5073f4SHaojian Zhuang 	buf = (uintptr_t)data;
601eb5073f4SHaojian Zhuang 	buf = (buf + CACHE_WRITEBACK_GRANULE - 1) &
602eb5073f4SHaojian Zhuang 	      ~(CACHE_WRITEBACK_GRANULE - 1);
603eb5073f4SHaojian Zhuang 	memset((void *)buf, 0, CACHE_WRITEBACK_GRANULE);
604eb5073f4SHaojian Zhuang 	flush_dcache_range(buf, CACHE_WRITEBACK_GRANULE);
605eb5073f4SHaojian Zhuang 	do {
606eb5073f4SHaojian Zhuang 		get_utrd(&utrd);
607eb5073f4SHaojian Zhuang 		ufs_prepare_cmd(&utrd, CDBCMD_READ_CAPACITY_10, lun, 0,
608eb5073f4SHaojian Zhuang 				buf, READ_CAPACITY_LENGTH);
609eb5073f4SHaojian Zhuang 		ufs_send_request(utrd.task_tag);
610eb5073f4SHaojian Zhuang 		result = ufs_check_resp(&utrd, RESPONSE_UPIU);
611eb5073f4SHaojian Zhuang 		assert(result == 0);
612eb5073f4SHaojian Zhuang #ifdef UFS_RESP_DEBUG
613eb5073f4SHaojian Zhuang 		dump_upiu(&utrd);
614eb5073f4SHaojian Zhuang #endif
615eb5073f4SHaojian Zhuang 		resp = (resp_upiu_t *)utrd.resp_upiu;
616eb5073f4SHaojian Zhuang 		retry = 0;
617eb5073f4SHaojian Zhuang 		sense = &resp->sd.sense;
618eb5073f4SHaojian Zhuang 		if (sense->resp_code == SENSE_DATA_VALID) {
619eb5073f4SHaojian Zhuang 			if ((sense->sense_key == SENSE_KEY_UNIT_ATTENTION) &&
620eb5073f4SHaojian Zhuang 			    (sense->asc == 0x29) && (sense->ascq == 0)) {
621eb5073f4SHaojian Zhuang 				retry = 1;
622eb5073f4SHaojian Zhuang 			}
623eb5073f4SHaojian Zhuang 		}
624eb5073f4SHaojian Zhuang 		inv_dcache_range(buf, CACHE_WRITEBACK_GRANULE);
625eb5073f4SHaojian Zhuang 		/* last logical block address */
626eb5073f4SHaojian Zhuang 		*num = be32toh(*(unsigned int *)buf);
627eb5073f4SHaojian Zhuang 		if (*num)
628eb5073f4SHaojian Zhuang 			*num += 1;
629eb5073f4SHaojian Zhuang 		/* logical block length in bytes */
630eb5073f4SHaojian Zhuang 		*size = be32toh(*(unsigned int *)(buf + 4));
631eb5073f4SHaojian Zhuang 	} while (retry);
632eb5073f4SHaojian Zhuang 	(void)result;
633eb5073f4SHaojian Zhuang }
634eb5073f4SHaojian Zhuang 
635eb5073f4SHaojian Zhuang size_t ufs_read_blocks(int lun, int lba, uintptr_t buf, size_t size)
636eb5073f4SHaojian Zhuang {
637eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
638eb5073f4SHaojian Zhuang 	resp_upiu_t *resp;
639eb5073f4SHaojian Zhuang 	int result;
640eb5073f4SHaojian Zhuang 
641eb5073f4SHaojian Zhuang 	assert((ufs_params.reg_base != 0) &&
642eb5073f4SHaojian Zhuang 	       (ufs_params.desc_base != 0) &&
643eb5073f4SHaojian Zhuang 	       (ufs_params.desc_size >= UFS_DESC_SIZE));
644eb5073f4SHaojian Zhuang 
645eb5073f4SHaojian Zhuang 	memset((void *)buf, 0, size);
646eb5073f4SHaojian Zhuang 	get_utrd(&utrd);
647eb5073f4SHaojian Zhuang 	ufs_prepare_cmd(&utrd, CDBCMD_READ_10, lun, lba, buf, size);
648eb5073f4SHaojian Zhuang 	ufs_send_request(utrd.task_tag);
649eb5073f4SHaojian Zhuang 	result = ufs_check_resp(&utrd, RESPONSE_UPIU);
650eb5073f4SHaojian Zhuang 	assert(result == 0);
651eb5073f4SHaojian Zhuang #ifdef UFS_RESP_DEBUG
652eb5073f4SHaojian Zhuang 	dump_upiu(&utrd);
653eb5073f4SHaojian Zhuang #endif
654eb5073f4SHaojian Zhuang 	resp = (resp_upiu_t *)utrd.resp_upiu;
655eb5073f4SHaojian Zhuang 	(void)result;
656eb5073f4SHaojian Zhuang 	return size - resp->res_trans_cnt;
657eb5073f4SHaojian Zhuang }
658eb5073f4SHaojian Zhuang 
659eb5073f4SHaojian Zhuang size_t ufs_write_blocks(int lun, int lba, const uintptr_t buf, size_t size)
660eb5073f4SHaojian Zhuang {
661eb5073f4SHaojian Zhuang 	utp_utrd_t utrd;
662eb5073f4SHaojian Zhuang 	resp_upiu_t *resp;
663eb5073f4SHaojian Zhuang 	int result;
664eb5073f4SHaojian Zhuang 
665eb5073f4SHaojian Zhuang 	assert((ufs_params.reg_base != 0) &&
666eb5073f4SHaojian Zhuang 	       (ufs_params.desc_base != 0) &&
667eb5073f4SHaojian Zhuang 	       (ufs_params.desc_size >= UFS_DESC_SIZE));
668eb5073f4SHaojian Zhuang 
669eb5073f4SHaojian Zhuang 	memset((void *)buf, 0, size);
670eb5073f4SHaojian Zhuang 	get_utrd(&utrd);
671eb5073f4SHaojian Zhuang 	ufs_prepare_cmd(&utrd, CDBCMD_WRITE_10, lun, lba, buf, size);
672eb5073f4SHaojian Zhuang 	ufs_send_request(utrd.task_tag);
673eb5073f4SHaojian Zhuang 	result = ufs_check_resp(&utrd, RESPONSE_UPIU);
674eb5073f4SHaojian Zhuang 	assert(result == 0);
675eb5073f4SHaojian Zhuang #ifdef UFS_RESP_DEBUG
676eb5073f4SHaojian Zhuang 	dump_upiu(&utrd);
677eb5073f4SHaojian Zhuang #endif
678eb5073f4SHaojian Zhuang 	resp = (resp_upiu_t *)utrd.resp_upiu;
679eb5073f4SHaojian Zhuang 	(void)result;
680eb5073f4SHaojian Zhuang 	return size - resp->res_trans_cnt;
681eb5073f4SHaojian Zhuang }
682eb5073f4SHaojian Zhuang 
683eb5073f4SHaojian Zhuang static void ufs_enum(void)
684eb5073f4SHaojian Zhuang {
685eb5073f4SHaojian Zhuang 	unsigned int blk_num, blk_size;
686eb5073f4SHaojian Zhuang 	int i;
687eb5073f4SHaojian Zhuang 
688eb5073f4SHaojian Zhuang 	/* 0 means 1 slot */
689eb5073f4SHaojian Zhuang 	nutrs = (mmio_read_32(ufs_params.reg_base + CAP) & CAP_NUTRS_MASK) + 1;
690eb5073f4SHaojian Zhuang 	if (nutrs > (ufs_params.desc_size / UFS_DESC_SIZE))
691eb5073f4SHaojian Zhuang 		nutrs = ufs_params.desc_size / UFS_DESC_SIZE;
692eb5073f4SHaojian Zhuang 
693eb5073f4SHaojian Zhuang 	ufs_verify_init();
694eb5073f4SHaojian Zhuang 	ufs_verify_ready();
695eb5073f4SHaojian Zhuang 
696eb5073f4SHaojian Zhuang 	ufs_set_flag(FLAG_DEVICE_INIT);
697eb5073f4SHaojian Zhuang 	mdelay(100);
698eb5073f4SHaojian Zhuang 	/* dump available LUNs */
699eb5073f4SHaojian Zhuang 	for (i = 0; i < UFS_MAX_LUNS; i++) {
700eb5073f4SHaojian Zhuang 		ufs_read_capacity(i, &blk_num, &blk_size);
701eb5073f4SHaojian Zhuang 		if (blk_num && blk_size) {
702eb5073f4SHaojian Zhuang 			INFO("UFS LUN%d contains %d blocks with %d-byte size\n",
703eb5073f4SHaojian Zhuang 			     i, blk_num, blk_size);
704eb5073f4SHaojian Zhuang 		}
705eb5073f4SHaojian Zhuang 	}
706eb5073f4SHaojian Zhuang }
707eb5073f4SHaojian Zhuang 
708*5ac25de6Sfengbaopeng static void ufs_get_device_info(struct ufs_dev_desc *card_data)
709*5ac25de6Sfengbaopeng {
710*5ac25de6Sfengbaopeng 	uint8_t desc_buf[DESC_DEVICE_MAX_SIZE];
711*5ac25de6Sfengbaopeng 
712*5ac25de6Sfengbaopeng 	ufs_query(QUERY_READ_DESC, DESC_TYPE_DEVICE, 0, 0,
713*5ac25de6Sfengbaopeng 				(uintptr_t)desc_buf, DESC_DEVICE_MAX_SIZE);
714*5ac25de6Sfengbaopeng 
715*5ac25de6Sfengbaopeng 	/*
716*5ac25de6Sfengbaopeng 	 * getting vendor (manufacturerID) and Bank Index in big endian
717*5ac25de6Sfengbaopeng 	 * format
718*5ac25de6Sfengbaopeng 	 */
719*5ac25de6Sfengbaopeng 	card_data->wmanufacturerid = (uint16_t)((desc_buf[DEVICE_DESC_PARAM_MANF_ID] << 8) |
720*5ac25de6Sfengbaopeng 				     (desc_buf[DEVICE_DESC_PARAM_MANF_ID + 1]));
721*5ac25de6Sfengbaopeng }
722*5ac25de6Sfengbaopeng 
723eb5073f4SHaojian Zhuang int ufs_init(const ufs_ops_t *ops, ufs_params_t *params)
724eb5073f4SHaojian Zhuang {
725eb5073f4SHaojian Zhuang 	int result;
726eb5073f4SHaojian Zhuang 	unsigned int data;
727eb5073f4SHaojian Zhuang 	uic_cmd_t cmd;
728*5ac25de6Sfengbaopeng 	struct ufs_dev_desc card = {0};
729eb5073f4SHaojian Zhuang 
730eb5073f4SHaojian Zhuang 	assert((params != NULL) &&
731eb5073f4SHaojian Zhuang 	       (params->reg_base != 0) &&
732eb5073f4SHaojian Zhuang 	       (params->desc_base != 0) &&
733eb5073f4SHaojian Zhuang 	       (params->desc_size >= UFS_DESC_SIZE));
734eb5073f4SHaojian Zhuang 
735eb5073f4SHaojian Zhuang 	memcpy(&ufs_params, params, sizeof(ufs_params_t));
736eb5073f4SHaojian Zhuang 
737eb5073f4SHaojian Zhuang 	if (ufs_params.flags & UFS_FLAGS_SKIPINIT) {
738eb5073f4SHaojian Zhuang 		result = ufshc_dme_get(0x1571, 0, &data);
739eb5073f4SHaojian Zhuang 		assert(result == 0);
740eb5073f4SHaojian Zhuang 		result = ufshc_dme_get(0x41, 0, &data);
741eb5073f4SHaojian Zhuang 		assert(result == 0);
742eb5073f4SHaojian Zhuang 		if (data == 1) {
743eb5073f4SHaojian Zhuang 			/* prepare to exit hibernate mode */
744eb5073f4SHaojian Zhuang 			memset(&cmd, 0, sizeof(uic_cmd_t));
745eb5073f4SHaojian Zhuang 			cmd.op = DME_HIBERNATE_EXIT;
746eb5073f4SHaojian Zhuang 			result = ufshc_send_uic_cmd(ufs_params.reg_base,
747eb5073f4SHaojian Zhuang 						    &cmd);
748eb5073f4SHaojian Zhuang 			assert(result == 0);
749eb5073f4SHaojian Zhuang 			data = mmio_read_32(ufs_params.reg_base + UCMDARG2);
750eb5073f4SHaojian Zhuang 			assert(data == 0);
751eb5073f4SHaojian Zhuang 			do {
752eb5073f4SHaojian Zhuang 				data = mmio_read_32(ufs_params.reg_base + IS);
753eb5073f4SHaojian Zhuang 			} while ((data & UFS_INT_UHXS) == 0);
754eb5073f4SHaojian Zhuang 			mmio_write_32(ufs_params.reg_base + IS, UFS_INT_UHXS);
755eb5073f4SHaojian Zhuang 			data = mmio_read_32(ufs_params.reg_base + HCS);
756eb5073f4SHaojian Zhuang 			assert((data & HCS_UPMCRS_MASK) == HCS_PWR_LOCAL);
757eb5073f4SHaojian Zhuang 		}
758eb5073f4SHaojian Zhuang 		result = ufshc_dme_get(0x1568, 0, &data);
759eb5073f4SHaojian Zhuang 		assert(result == 0);
760eb5073f4SHaojian Zhuang 		assert((data > 0) && (data <= 3));
761eb5073f4SHaojian Zhuang 	} else {
762eb5073f4SHaojian Zhuang 		assert((ops != NULL) && (ops->phy_init != NULL) &&
763eb5073f4SHaojian Zhuang 		       (ops->phy_set_pwr_mode != NULL));
764eb5073f4SHaojian Zhuang 
765eb5073f4SHaojian Zhuang 		ufshc_reset(ufs_params.reg_base);
766eb5073f4SHaojian Zhuang 		ops->phy_init(&ufs_params);
767eb5073f4SHaojian Zhuang 		result = ufshc_link_startup(ufs_params.reg_base);
768eb5073f4SHaojian Zhuang 		assert(result == 0);
769*5ac25de6Sfengbaopeng 
770*5ac25de6Sfengbaopeng 		ufs_enum();
771*5ac25de6Sfengbaopeng 
772*5ac25de6Sfengbaopeng 		ufs_get_device_info(&card);
773*5ac25de6Sfengbaopeng 		if (card.wmanufacturerid == UFS_VENDOR_SKHYNIX) {
774*5ac25de6Sfengbaopeng 			ufs_params.flags |= UFS_FLAGS_VENDOR_SKHYNIX;
775*5ac25de6Sfengbaopeng 		}
776*5ac25de6Sfengbaopeng 
777eb5073f4SHaojian Zhuang 		ops->phy_set_pwr_mode(&ufs_params);
778eb5073f4SHaojian Zhuang 	}
779eb5073f4SHaojian Zhuang 
780eb5073f4SHaojian Zhuang 	(void)result;
781eb5073f4SHaojian Zhuang 	return 0;
782eb5073f4SHaojian Zhuang }
783