18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
28c2ecf20Sopenharmony_ci/* Realtek USB Memstick Card Interface driver
38c2ecf20Sopenharmony_ci *
48c2ecf20Sopenharmony_ci * Copyright(c) 2009-2013 Realtek Semiconductor Corp. All rights reserved.
58c2ecf20Sopenharmony_ci *
68c2ecf20Sopenharmony_ci * Author:
78c2ecf20Sopenharmony_ci *   Roger Tseng <rogerable@realtek.com>
88c2ecf20Sopenharmony_ci */
98c2ecf20Sopenharmony_ci
108c2ecf20Sopenharmony_ci#include <linux/module.h>
118c2ecf20Sopenharmony_ci#include <linux/highmem.h>
128c2ecf20Sopenharmony_ci#include <linux/delay.h>
138c2ecf20Sopenharmony_ci#include <linux/platform_device.h>
148c2ecf20Sopenharmony_ci#include <linux/workqueue.h>
158c2ecf20Sopenharmony_ci#include <linux/memstick.h>
168c2ecf20Sopenharmony_ci#include <linux/kthread.h>
178c2ecf20Sopenharmony_ci#include <linux/rtsx_usb.h>
188c2ecf20Sopenharmony_ci#include <linux/pm_runtime.h>
198c2ecf20Sopenharmony_ci#include <linux/mutex.h>
208c2ecf20Sopenharmony_ci#include <linux/sched.h>
218c2ecf20Sopenharmony_ci#include <linux/completion.h>
228c2ecf20Sopenharmony_ci#include <asm/unaligned.h>
238c2ecf20Sopenharmony_ci
248c2ecf20Sopenharmony_cistruct rtsx_usb_ms {
258c2ecf20Sopenharmony_ci	struct platform_device	*pdev;
268c2ecf20Sopenharmony_ci	struct rtsx_ucr	*ucr;
278c2ecf20Sopenharmony_ci	struct memstick_host	*msh;
288c2ecf20Sopenharmony_ci	struct memstick_request	*req;
298c2ecf20Sopenharmony_ci
308c2ecf20Sopenharmony_ci	struct mutex		host_mutex;
318c2ecf20Sopenharmony_ci	struct work_struct	handle_req;
328c2ecf20Sopenharmony_ci	struct delayed_work	poll_card;
338c2ecf20Sopenharmony_ci
348c2ecf20Sopenharmony_ci	u8			ssc_depth;
358c2ecf20Sopenharmony_ci	unsigned int		clock;
368c2ecf20Sopenharmony_ci	int			power_mode;
378c2ecf20Sopenharmony_ci	unsigned char           ifmode;
388c2ecf20Sopenharmony_ci	bool			eject;
398c2ecf20Sopenharmony_ci	bool			system_suspending;
408c2ecf20Sopenharmony_ci};
418c2ecf20Sopenharmony_ci
428c2ecf20Sopenharmony_cistatic inline struct device *ms_dev(struct rtsx_usb_ms *host)
438c2ecf20Sopenharmony_ci{
448c2ecf20Sopenharmony_ci	return &(host->pdev->dev);
458c2ecf20Sopenharmony_ci}
468c2ecf20Sopenharmony_ci
478c2ecf20Sopenharmony_cistatic inline void ms_clear_error(struct rtsx_usb_ms *host)
488c2ecf20Sopenharmony_ci{
498c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr = host->ucr;
508c2ecf20Sopenharmony_ci	rtsx_usb_ep0_write_register(ucr, CARD_STOP,
518c2ecf20Sopenharmony_ci				  MS_STOP | MS_CLR_ERR,
528c2ecf20Sopenharmony_ci				  MS_STOP | MS_CLR_ERR);
538c2ecf20Sopenharmony_ci
548c2ecf20Sopenharmony_ci	rtsx_usb_clear_dma_err(ucr);
558c2ecf20Sopenharmony_ci	rtsx_usb_clear_fsm_err(ucr);
568c2ecf20Sopenharmony_ci}
578c2ecf20Sopenharmony_ci
588c2ecf20Sopenharmony_ci#ifdef DEBUG
598c2ecf20Sopenharmony_ci
608c2ecf20Sopenharmony_cistatic void ms_print_debug_regs(struct rtsx_usb_ms *host)
618c2ecf20Sopenharmony_ci{
628c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr = host->ucr;
638c2ecf20Sopenharmony_ci	u16 i;
648c2ecf20Sopenharmony_ci	u8 *ptr;
658c2ecf20Sopenharmony_ci
668c2ecf20Sopenharmony_ci	/* Print MS host internal registers */
678c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
688c2ecf20Sopenharmony_ci
698c2ecf20Sopenharmony_ci	/* MS_CFG to MS_INT_REG */
708c2ecf20Sopenharmony_ci	for (i = 0xFD40; i <= 0xFD44; i++)
718c2ecf20Sopenharmony_ci		rtsx_usb_add_cmd(ucr, READ_REG_CMD, i, 0, 0);
728c2ecf20Sopenharmony_ci
738c2ecf20Sopenharmony_ci	/* CARD_SHARE_MODE to CARD_GPIO */
748c2ecf20Sopenharmony_ci	for (i = 0xFD51; i <= 0xFD56; i++)
758c2ecf20Sopenharmony_ci		rtsx_usb_add_cmd(ucr, READ_REG_CMD, i, 0, 0);
768c2ecf20Sopenharmony_ci
778c2ecf20Sopenharmony_ci	/* CARD_PULL_CTLx */
788c2ecf20Sopenharmony_ci	for (i = 0xFD60; i <= 0xFD65; i++)
798c2ecf20Sopenharmony_ci		rtsx_usb_add_cmd(ucr, READ_REG_CMD, i, 0, 0);
808c2ecf20Sopenharmony_ci
818c2ecf20Sopenharmony_ci	/* CARD_DATA_SOURCE, CARD_SELECT, CARD_CLK_EN, CARD_PWR_CTL */
828c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, READ_REG_CMD, CARD_DATA_SOURCE, 0, 0);
838c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, READ_REG_CMD, CARD_SELECT, 0, 0);
848c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, READ_REG_CMD, CARD_CLK_EN, 0, 0);
858c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, READ_REG_CMD, CARD_PWR_CTL, 0, 0);
868c2ecf20Sopenharmony_ci
878c2ecf20Sopenharmony_ci	rtsx_usb_send_cmd(ucr, MODE_CR, 100);
888c2ecf20Sopenharmony_ci	rtsx_usb_get_rsp(ucr, 21, 100);
898c2ecf20Sopenharmony_ci
908c2ecf20Sopenharmony_ci	ptr = ucr->rsp_buf;
918c2ecf20Sopenharmony_ci	for (i = 0xFD40; i <= 0xFD44; i++)
928c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "0x%04X: 0x%02x\n", i, *(ptr++));
938c2ecf20Sopenharmony_ci	for (i = 0xFD51; i <= 0xFD56; i++)
948c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "0x%04X: 0x%02x\n", i, *(ptr++));
958c2ecf20Sopenharmony_ci	for (i = 0xFD60; i <= 0xFD65; i++)
968c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "0x%04X: 0x%02x\n", i, *(ptr++));
978c2ecf20Sopenharmony_ci
988c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "0x%04X: 0x%02x\n", CARD_DATA_SOURCE, *(ptr++));
998c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "0x%04X: 0x%02x\n", CARD_SELECT, *(ptr++));
1008c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "0x%04X: 0x%02x\n", CARD_CLK_EN, *(ptr++));
1018c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "0x%04X: 0x%02x\n", CARD_PWR_CTL, *(ptr++));
1028c2ecf20Sopenharmony_ci}
1038c2ecf20Sopenharmony_ci
1048c2ecf20Sopenharmony_ci#else
1058c2ecf20Sopenharmony_ci
1068c2ecf20Sopenharmony_cistatic void ms_print_debug_regs(struct rtsx_usb_ms *host)
1078c2ecf20Sopenharmony_ci{
1088c2ecf20Sopenharmony_ci}
1098c2ecf20Sopenharmony_ci
1108c2ecf20Sopenharmony_ci#endif
1118c2ecf20Sopenharmony_ci
1128c2ecf20Sopenharmony_cistatic int ms_pull_ctl_disable_lqfp48(struct rtsx_ucr *ucr)
1138c2ecf20Sopenharmony_ci{
1148c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
1158c2ecf20Sopenharmony_ci
1168c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL1, 0xFF, 0x55);
1178c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL2, 0xFF, 0x55);
1188c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL3, 0xFF, 0x95);
1198c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL4, 0xFF, 0x55);
1208c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL5, 0xFF, 0x55);
1218c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL6, 0xFF, 0xA5);
1228c2ecf20Sopenharmony_ci
1238c2ecf20Sopenharmony_ci	return rtsx_usb_send_cmd(ucr, MODE_C, 100);
1248c2ecf20Sopenharmony_ci}
1258c2ecf20Sopenharmony_ci
1268c2ecf20Sopenharmony_cistatic int ms_pull_ctl_disable_qfn24(struct rtsx_ucr *ucr)
1278c2ecf20Sopenharmony_ci{
1288c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
1298c2ecf20Sopenharmony_ci
1308c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL1, 0xFF, 0x65);
1318c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL2, 0xFF, 0x55);
1328c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL3, 0xFF, 0x95);
1338c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL4, 0xFF, 0x55);
1348c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL5, 0xFF, 0x56);
1358c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL6, 0xFF, 0x59);
1368c2ecf20Sopenharmony_ci
1378c2ecf20Sopenharmony_ci	return rtsx_usb_send_cmd(ucr, MODE_C, 100);
1388c2ecf20Sopenharmony_ci}
1398c2ecf20Sopenharmony_ci
1408c2ecf20Sopenharmony_cistatic int ms_pull_ctl_enable_lqfp48(struct rtsx_ucr *ucr)
1418c2ecf20Sopenharmony_ci{
1428c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
1438c2ecf20Sopenharmony_ci
1448c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL1, 0xFF, 0x55);
1458c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL2, 0xFF, 0x55);
1468c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL3, 0xFF, 0x95);
1478c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL4, 0xFF, 0x55);
1488c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL5, 0xFF, 0x55);
1498c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL6, 0xFF, 0xA5);
1508c2ecf20Sopenharmony_ci
1518c2ecf20Sopenharmony_ci	return rtsx_usb_send_cmd(ucr, MODE_C, 100);
1528c2ecf20Sopenharmony_ci}
1538c2ecf20Sopenharmony_ci
1548c2ecf20Sopenharmony_cistatic int ms_pull_ctl_enable_qfn24(struct rtsx_ucr *ucr)
1558c2ecf20Sopenharmony_ci{
1568c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
1578c2ecf20Sopenharmony_ci
1588c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL1, 0xFF, 0x65);
1598c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL2, 0xFF, 0x55);
1608c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL3, 0xFF, 0x95);
1618c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL4, 0xFF, 0x55);
1628c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL5, 0xFF, 0x55);
1638c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PULL_CTL6, 0xFF, 0x59);
1648c2ecf20Sopenharmony_ci
1658c2ecf20Sopenharmony_ci	return rtsx_usb_send_cmd(ucr, MODE_C, 100);
1668c2ecf20Sopenharmony_ci}
1678c2ecf20Sopenharmony_ci
1688c2ecf20Sopenharmony_cistatic int ms_power_on(struct rtsx_usb_ms *host)
1698c2ecf20Sopenharmony_ci{
1708c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr = host->ucr;
1718c2ecf20Sopenharmony_ci	int err;
1728c2ecf20Sopenharmony_ci
1738c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s\n", __func__);
1748c2ecf20Sopenharmony_ci
1758c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
1768c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_SELECT, 0x07, MS_MOD_SEL);
1778c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_SHARE_MODE,
1788c2ecf20Sopenharmony_ci			CARD_SHARE_MASK, CARD_SHARE_MS);
1798c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_CLK_EN,
1808c2ecf20Sopenharmony_ci			MS_CLK_EN, MS_CLK_EN);
1818c2ecf20Sopenharmony_ci	err = rtsx_usb_send_cmd(ucr, MODE_C, 100);
1828c2ecf20Sopenharmony_ci	if (err < 0)
1838c2ecf20Sopenharmony_ci		return err;
1848c2ecf20Sopenharmony_ci
1858c2ecf20Sopenharmony_ci	if (CHECK_PKG(ucr, LQFP48))
1868c2ecf20Sopenharmony_ci		err = ms_pull_ctl_enable_lqfp48(ucr);
1878c2ecf20Sopenharmony_ci	else
1888c2ecf20Sopenharmony_ci		err = ms_pull_ctl_enable_qfn24(ucr);
1898c2ecf20Sopenharmony_ci	if (err < 0)
1908c2ecf20Sopenharmony_ci		return err;
1918c2ecf20Sopenharmony_ci
1928c2ecf20Sopenharmony_ci	err = rtsx_usb_write_register(ucr, CARD_PWR_CTL,
1938c2ecf20Sopenharmony_ci			POWER_MASK, PARTIAL_POWER_ON);
1948c2ecf20Sopenharmony_ci	if (err)
1958c2ecf20Sopenharmony_ci		return err;
1968c2ecf20Sopenharmony_ci
1978c2ecf20Sopenharmony_ci	usleep_range(800, 1000);
1988c2ecf20Sopenharmony_ci
1998c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
2008c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_PWR_CTL,
2018c2ecf20Sopenharmony_ci			POWER_MASK, POWER_ON);
2028c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_OE,
2038c2ecf20Sopenharmony_ci			MS_OUTPUT_EN, MS_OUTPUT_EN);
2048c2ecf20Sopenharmony_ci
2058c2ecf20Sopenharmony_ci	return rtsx_usb_send_cmd(ucr, MODE_C, 100);
2068c2ecf20Sopenharmony_ci}
2078c2ecf20Sopenharmony_ci
2088c2ecf20Sopenharmony_cistatic int ms_power_off(struct rtsx_usb_ms *host)
2098c2ecf20Sopenharmony_ci{
2108c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr = host->ucr;
2118c2ecf20Sopenharmony_ci	int err;
2128c2ecf20Sopenharmony_ci
2138c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s\n", __func__);
2148c2ecf20Sopenharmony_ci
2158c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
2168c2ecf20Sopenharmony_ci
2178c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_CLK_EN, MS_CLK_EN, 0);
2188c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_OE, MS_OUTPUT_EN, 0);
2198c2ecf20Sopenharmony_ci
2208c2ecf20Sopenharmony_ci	err = rtsx_usb_send_cmd(ucr, MODE_C, 100);
2218c2ecf20Sopenharmony_ci	if (err < 0)
2228c2ecf20Sopenharmony_ci		return err;
2238c2ecf20Sopenharmony_ci
2248c2ecf20Sopenharmony_ci	if (CHECK_PKG(ucr, LQFP48))
2258c2ecf20Sopenharmony_ci		return ms_pull_ctl_disable_lqfp48(ucr);
2268c2ecf20Sopenharmony_ci
2278c2ecf20Sopenharmony_ci	return ms_pull_ctl_disable_qfn24(ucr);
2288c2ecf20Sopenharmony_ci}
2298c2ecf20Sopenharmony_ci
2308c2ecf20Sopenharmony_cistatic int ms_transfer_data(struct rtsx_usb_ms *host, unsigned char data_dir,
2318c2ecf20Sopenharmony_ci		u8 tpc, u8 cfg, struct scatterlist *sg)
2328c2ecf20Sopenharmony_ci{
2338c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr = host->ucr;
2348c2ecf20Sopenharmony_ci	int err;
2358c2ecf20Sopenharmony_ci	unsigned int length = sg->length;
2368c2ecf20Sopenharmony_ci	u16 sec_cnt = (u16)(length / 512);
2378c2ecf20Sopenharmony_ci	u8 trans_mode, dma_dir, flag;
2388c2ecf20Sopenharmony_ci	unsigned int pipe;
2398c2ecf20Sopenharmony_ci	struct memstick_dev *card = host->msh->card;
2408c2ecf20Sopenharmony_ci
2418c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s: tpc = 0x%02x, data_dir = %s, length = %d\n",
2428c2ecf20Sopenharmony_ci			__func__, tpc, (data_dir == READ) ? "READ" : "WRITE",
2438c2ecf20Sopenharmony_ci			length);
2448c2ecf20Sopenharmony_ci
2458c2ecf20Sopenharmony_ci	if (data_dir == READ) {
2468c2ecf20Sopenharmony_ci		flag = MODE_CDIR;
2478c2ecf20Sopenharmony_ci		dma_dir = DMA_DIR_FROM_CARD;
2488c2ecf20Sopenharmony_ci		if (card->id.type != MEMSTICK_TYPE_PRO)
2498c2ecf20Sopenharmony_ci			trans_mode = MS_TM_NORMAL_READ;
2508c2ecf20Sopenharmony_ci		else
2518c2ecf20Sopenharmony_ci			trans_mode = MS_TM_AUTO_READ;
2528c2ecf20Sopenharmony_ci		pipe = usb_rcvbulkpipe(ucr->pusb_dev, EP_BULK_IN);
2538c2ecf20Sopenharmony_ci	} else {
2548c2ecf20Sopenharmony_ci		flag = MODE_CDOR;
2558c2ecf20Sopenharmony_ci		dma_dir = DMA_DIR_TO_CARD;
2568c2ecf20Sopenharmony_ci		if (card->id.type != MEMSTICK_TYPE_PRO)
2578c2ecf20Sopenharmony_ci			trans_mode = MS_TM_NORMAL_WRITE;
2588c2ecf20Sopenharmony_ci		else
2598c2ecf20Sopenharmony_ci			trans_mode = MS_TM_AUTO_WRITE;
2608c2ecf20Sopenharmony_ci		pipe = usb_sndbulkpipe(ucr->pusb_dev, EP_BULK_OUT);
2618c2ecf20Sopenharmony_ci	}
2628c2ecf20Sopenharmony_ci
2638c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
2648c2ecf20Sopenharmony_ci
2658c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_TPC, 0xFF, tpc);
2668c2ecf20Sopenharmony_ci	if (card->id.type == MEMSTICK_TYPE_PRO) {
2678c2ecf20Sopenharmony_ci		rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_SECTOR_CNT_H,
2688c2ecf20Sopenharmony_ci				0xFF, (u8)(sec_cnt >> 8));
2698c2ecf20Sopenharmony_ci		rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_SECTOR_CNT_L,
2708c2ecf20Sopenharmony_ci				0xFF, (u8)sec_cnt);
2718c2ecf20Sopenharmony_ci	}
2728c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_TRANS_CFG, 0xFF, cfg);
2738c2ecf20Sopenharmony_ci
2748c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MC_DMA_TC3,
2758c2ecf20Sopenharmony_ci			0xFF, (u8)(length >> 24));
2768c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MC_DMA_TC2,
2778c2ecf20Sopenharmony_ci			0xFF, (u8)(length >> 16));
2788c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MC_DMA_TC1,
2798c2ecf20Sopenharmony_ci			0xFF, (u8)(length >> 8));
2808c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MC_DMA_TC0, 0xFF,
2818c2ecf20Sopenharmony_ci			(u8)length);
2828c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MC_DMA_CTL,
2838c2ecf20Sopenharmony_ci			0x03 | DMA_PACK_SIZE_MASK, dma_dir | DMA_EN | DMA_512);
2848c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_DATA_SOURCE,
2858c2ecf20Sopenharmony_ci			0x01, RING_BUFFER);
2868c2ecf20Sopenharmony_ci
2878c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_TRANSFER,
2888c2ecf20Sopenharmony_ci			0xFF, MS_TRANSFER_START | trans_mode);
2898c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, CHECK_REG_CMD, MS_TRANSFER,
2908c2ecf20Sopenharmony_ci			MS_TRANSFER_END, MS_TRANSFER_END);
2918c2ecf20Sopenharmony_ci
2928c2ecf20Sopenharmony_ci	err = rtsx_usb_send_cmd(ucr, flag | STAGE_MS_STATUS, 100);
2938c2ecf20Sopenharmony_ci	if (err)
2948c2ecf20Sopenharmony_ci		return err;
2958c2ecf20Sopenharmony_ci
2968c2ecf20Sopenharmony_ci	err = rtsx_usb_transfer_data(ucr, pipe, sg, length,
2978c2ecf20Sopenharmony_ci			1, NULL, 10000);
2988c2ecf20Sopenharmony_ci	if (err)
2998c2ecf20Sopenharmony_ci		goto err_out;
3008c2ecf20Sopenharmony_ci
3018c2ecf20Sopenharmony_ci	err = rtsx_usb_get_rsp(ucr, 3, 15000);
3028c2ecf20Sopenharmony_ci	if (err)
3038c2ecf20Sopenharmony_ci		goto err_out;
3048c2ecf20Sopenharmony_ci
3058c2ecf20Sopenharmony_ci	if (ucr->rsp_buf[0] & MS_TRANSFER_ERR ||
3068c2ecf20Sopenharmony_ci	    ucr->rsp_buf[1] & (MS_CRC16_ERR | MS_RDY_TIMEOUT)) {
3078c2ecf20Sopenharmony_ci		err = -EIO;
3088c2ecf20Sopenharmony_ci		goto err_out;
3098c2ecf20Sopenharmony_ci	}
3108c2ecf20Sopenharmony_ci	return 0;
3118c2ecf20Sopenharmony_cierr_out:
3128c2ecf20Sopenharmony_ci	ms_clear_error(host);
3138c2ecf20Sopenharmony_ci	return err;
3148c2ecf20Sopenharmony_ci}
3158c2ecf20Sopenharmony_ci
3168c2ecf20Sopenharmony_cistatic int ms_write_bytes(struct rtsx_usb_ms *host, u8 tpc,
3178c2ecf20Sopenharmony_ci		u8 cfg, u8 cnt, u8 *data, u8 *int_reg)
3188c2ecf20Sopenharmony_ci{
3198c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr = host->ucr;
3208c2ecf20Sopenharmony_ci	int err, i;
3218c2ecf20Sopenharmony_ci
3228c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s: tpc = 0x%02x\n", __func__, tpc);
3238c2ecf20Sopenharmony_ci
3248c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
3258c2ecf20Sopenharmony_ci
3268c2ecf20Sopenharmony_ci	for (i = 0; i < cnt; i++)
3278c2ecf20Sopenharmony_ci		rtsx_usb_add_cmd(ucr, WRITE_REG_CMD,
3288c2ecf20Sopenharmony_ci				PPBUF_BASE2 + i, 0xFF, data[i]);
3298c2ecf20Sopenharmony_ci
3308c2ecf20Sopenharmony_ci	if (cnt % 2)
3318c2ecf20Sopenharmony_ci		rtsx_usb_add_cmd(ucr, WRITE_REG_CMD,
3328c2ecf20Sopenharmony_ci				PPBUF_BASE2 + i, 0xFF, 0xFF);
3338c2ecf20Sopenharmony_ci
3348c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_TPC, 0xFF, tpc);
3358c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_BYTE_CNT, 0xFF, cnt);
3368c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_TRANS_CFG, 0xFF, cfg);
3378c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_DATA_SOURCE,
3388c2ecf20Sopenharmony_ci			0x01, PINGPONG_BUFFER);
3398c2ecf20Sopenharmony_ci
3408c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_TRANSFER,
3418c2ecf20Sopenharmony_ci			0xFF, MS_TRANSFER_START | MS_TM_WRITE_BYTES);
3428c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, CHECK_REG_CMD, MS_TRANSFER,
3438c2ecf20Sopenharmony_ci			MS_TRANSFER_END, MS_TRANSFER_END);
3448c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, READ_REG_CMD, MS_TRANS_CFG, 0, 0);
3458c2ecf20Sopenharmony_ci
3468c2ecf20Sopenharmony_ci	err = rtsx_usb_send_cmd(ucr, MODE_CR, 100);
3478c2ecf20Sopenharmony_ci	if (err)
3488c2ecf20Sopenharmony_ci		return err;
3498c2ecf20Sopenharmony_ci
3508c2ecf20Sopenharmony_ci	err = rtsx_usb_get_rsp(ucr, 2, 5000);
3518c2ecf20Sopenharmony_ci	if (err || (ucr->rsp_buf[0] & MS_TRANSFER_ERR)) {
3528c2ecf20Sopenharmony_ci		u8 val;
3538c2ecf20Sopenharmony_ci
3548c2ecf20Sopenharmony_ci		rtsx_usb_ep0_read_register(ucr, MS_TRANS_CFG, &val);
3558c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "MS_TRANS_CFG: 0x%02x\n", val);
3568c2ecf20Sopenharmony_ci
3578c2ecf20Sopenharmony_ci		if (int_reg)
3588c2ecf20Sopenharmony_ci			*int_reg = val & 0x0F;
3598c2ecf20Sopenharmony_ci
3608c2ecf20Sopenharmony_ci		ms_print_debug_regs(host);
3618c2ecf20Sopenharmony_ci
3628c2ecf20Sopenharmony_ci		ms_clear_error(host);
3638c2ecf20Sopenharmony_ci
3648c2ecf20Sopenharmony_ci		if (!(tpc & 0x08)) {
3658c2ecf20Sopenharmony_ci			if (val & MS_CRC16_ERR)
3668c2ecf20Sopenharmony_ci				return -EIO;
3678c2ecf20Sopenharmony_ci		} else {
3688c2ecf20Sopenharmony_ci			if (!(val & 0x80)) {
3698c2ecf20Sopenharmony_ci				if (val & (MS_INT_ERR | MS_INT_CMDNK))
3708c2ecf20Sopenharmony_ci					return -EIO;
3718c2ecf20Sopenharmony_ci			}
3728c2ecf20Sopenharmony_ci		}
3738c2ecf20Sopenharmony_ci
3748c2ecf20Sopenharmony_ci		return -ETIMEDOUT;
3758c2ecf20Sopenharmony_ci	}
3768c2ecf20Sopenharmony_ci
3778c2ecf20Sopenharmony_ci	if (int_reg)
3788c2ecf20Sopenharmony_ci		*int_reg = ucr->rsp_buf[1] & 0x0F;
3798c2ecf20Sopenharmony_ci
3808c2ecf20Sopenharmony_ci	return 0;
3818c2ecf20Sopenharmony_ci}
3828c2ecf20Sopenharmony_ci
3838c2ecf20Sopenharmony_cistatic int ms_read_bytes(struct rtsx_usb_ms *host, u8 tpc,
3848c2ecf20Sopenharmony_ci		u8 cfg, u8 cnt, u8 *data, u8 *int_reg)
3858c2ecf20Sopenharmony_ci{
3868c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr = host->ucr;
3878c2ecf20Sopenharmony_ci	int err, i;
3888c2ecf20Sopenharmony_ci	u8 *ptr;
3898c2ecf20Sopenharmony_ci
3908c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s: tpc = 0x%02x\n", __func__, tpc);
3918c2ecf20Sopenharmony_ci
3928c2ecf20Sopenharmony_ci	rtsx_usb_init_cmd(ucr);
3938c2ecf20Sopenharmony_ci
3948c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_TPC, 0xFF, tpc);
3958c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_BYTE_CNT, 0xFF, cnt);
3968c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_TRANS_CFG, 0xFF, cfg);
3978c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, CARD_DATA_SOURCE,
3988c2ecf20Sopenharmony_ci			0x01, PINGPONG_BUFFER);
3998c2ecf20Sopenharmony_ci
4008c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, WRITE_REG_CMD, MS_TRANSFER,
4018c2ecf20Sopenharmony_ci			0xFF, MS_TRANSFER_START | MS_TM_READ_BYTES);
4028c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, CHECK_REG_CMD, MS_TRANSFER,
4038c2ecf20Sopenharmony_ci			MS_TRANSFER_END, MS_TRANSFER_END);
4048c2ecf20Sopenharmony_ci	for (i = 0; i < cnt - 1; i++)
4058c2ecf20Sopenharmony_ci		rtsx_usb_add_cmd(ucr, READ_REG_CMD, PPBUF_BASE2 + i, 0, 0);
4068c2ecf20Sopenharmony_ci	if (cnt % 2)
4078c2ecf20Sopenharmony_ci		rtsx_usb_add_cmd(ucr, READ_REG_CMD, PPBUF_BASE2 + cnt, 0, 0);
4088c2ecf20Sopenharmony_ci	else
4098c2ecf20Sopenharmony_ci		rtsx_usb_add_cmd(ucr, READ_REG_CMD,
4108c2ecf20Sopenharmony_ci				PPBUF_BASE2 + cnt - 1, 0, 0);
4118c2ecf20Sopenharmony_ci
4128c2ecf20Sopenharmony_ci	rtsx_usb_add_cmd(ucr, READ_REG_CMD, MS_TRANS_CFG, 0, 0);
4138c2ecf20Sopenharmony_ci
4148c2ecf20Sopenharmony_ci	err = rtsx_usb_send_cmd(ucr, MODE_CR, 100);
4158c2ecf20Sopenharmony_ci	if (err)
4168c2ecf20Sopenharmony_ci		return err;
4178c2ecf20Sopenharmony_ci
4188c2ecf20Sopenharmony_ci	err = rtsx_usb_get_rsp(ucr, cnt + 2, 5000);
4198c2ecf20Sopenharmony_ci	if (err || (ucr->rsp_buf[0] & MS_TRANSFER_ERR)) {
4208c2ecf20Sopenharmony_ci		u8 val;
4218c2ecf20Sopenharmony_ci
4228c2ecf20Sopenharmony_ci		rtsx_usb_ep0_read_register(ucr, MS_TRANS_CFG, &val);
4238c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "MS_TRANS_CFG: 0x%02x\n", val);
4248c2ecf20Sopenharmony_ci
4258c2ecf20Sopenharmony_ci		if (int_reg && (host->ifmode != MEMSTICK_SERIAL))
4268c2ecf20Sopenharmony_ci			*int_reg = val & 0x0F;
4278c2ecf20Sopenharmony_ci
4288c2ecf20Sopenharmony_ci		ms_print_debug_regs(host);
4298c2ecf20Sopenharmony_ci
4308c2ecf20Sopenharmony_ci		ms_clear_error(host);
4318c2ecf20Sopenharmony_ci
4328c2ecf20Sopenharmony_ci		if (!(tpc & 0x08)) {
4338c2ecf20Sopenharmony_ci			if (val & MS_CRC16_ERR)
4348c2ecf20Sopenharmony_ci				return -EIO;
4358c2ecf20Sopenharmony_ci		} else {
4368c2ecf20Sopenharmony_ci			if (!(val & 0x80)) {
4378c2ecf20Sopenharmony_ci				if (val & (MS_INT_ERR | MS_INT_CMDNK))
4388c2ecf20Sopenharmony_ci					return -EIO;
4398c2ecf20Sopenharmony_ci			}
4408c2ecf20Sopenharmony_ci		}
4418c2ecf20Sopenharmony_ci
4428c2ecf20Sopenharmony_ci		return -ETIMEDOUT;
4438c2ecf20Sopenharmony_ci	}
4448c2ecf20Sopenharmony_ci
4458c2ecf20Sopenharmony_ci	ptr = ucr->rsp_buf + 1;
4468c2ecf20Sopenharmony_ci	for (i = 0; i < cnt; i++)
4478c2ecf20Sopenharmony_ci		data[i] = *ptr++;
4488c2ecf20Sopenharmony_ci
4498c2ecf20Sopenharmony_ci
4508c2ecf20Sopenharmony_ci	if (int_reg && (host->ifmode != MEMSTICK_SERIAL))
4518c2ecf20Sopenharmony_ci		*int_reg = *ptr & 0x0F;
4528c2ecf20Sopenharmony_ci
4538c2ecf20Sopenharmony_ci	return 0;
4548c2ecf20Sopenharmony_ci}
4558c2ecf20Sopenharmony_ci
4568c2ecf20Sopenharmony_cistatic int rtsx_usb_ms_issue_cmd(struct rtsx_usb_ms *host)
4578c2ecf20Sopenharmony_ci{
4588c2ecf20Sopenharmony_ci	struct memstick_request *req = host->req;
4598c2ecf20Sopenharmony_ci	int err = 0;
4608c2ecf20Sopenharmony_ci	u8 cfg = 0, int_reg;
4618c2ecf20Sopenharmony_ci
4628c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s\n", __func__);
4638c2ecf20Sopenharmony_ci
4648c2ecf20Sopenharmony_ci	if (req->need_card_int) {
4658c2ecf20Sopenharmony_ci		if (host->ifmode != MEMSTICK_SERIAL)
4668c2ecf20Sopenharmony_ci			cfg = WAIT_INT;
4678c2ecf20Sopenharmony_ci	}
4688c2ecf20Sopenharmony_ci
4698c2ecf20Sopenharmony_ci	if (req->long_data) {
4708c2ecf20Sopenharmony_ci		err = ms_transfer_data(host, req->data_dir,
4718c2ecf20Sopenharmony_ci				req->tpc, cfg, &(req->sg));
4728c2ecf20Sopenharmony_ci	} else {
4738c2ecf20Sopenharmony_ci		if (req->data_dir == READ)
4748c2ecf20Sopenharmony_ci			err = ms_read_bytes(host, req->tpc, cfg,
4758c2ecf20Sopenharmony_ci					req->data_len, req->data, &int_reg);
4768c2ecf20Sopenharmony_ci		else
4778c2ecf20Sopenharmony_ci			err = ms_write_bytes(host, req->tpc, cfg,
4788c2ecf20Sopenharmony_ci					req->data_len, req->data, &int_reg);
4798c2ecf20Sopenharmony_ci	}
4808c2ecf20Sopenharmony_ci	if (err < 0)
4818c2ecf20Sopenharmony_ci		return err;
4828c2ecf20Sopenharmony_ci
4838c2ecf20Sopenharmony_ci	if (req->need_card_int) {
4848c2ecf20Sopenharmony_ci		if (host->ifmode == MEMSTICK_SERIAL) {
4858c2ecf20Sopenharmony_ci			err = ms_read_bytes(host, MS_TPC_GET_INT,
4868c2ecf20Sopenharmony_ci					NO_WAIT_INT, 1, &req->int_reg, NULL);
4878c2ecf20Sopenharmony_ci			if (err < 0)
4888c2ecf20Sopenharmony_ci				return err;
4898c2ecf20Sopenharmony_ci		} else {
4908c2ecf20Sopenharmony_ci
4918c2ecf20Sopenharmony_ci			if (int_reg & MS_INT_CMDNK)
4928c2ecf20Sopenharmony_ci				req->int_reg |= MEMSTICK_INT_CMDNAK;
4938c2ecf20Sopenharmony_ci			if (int_reg & MS_INT_BREQ)
4948c2ecf20Sopenharmony_ci				req->int_reg |= MEMSTICK_INT_BREQ;
4958c2ecf20Sopenharmony_ci			if (int_reg & MS_INT_ERR)
4968c2ecf20Sopenharmony_ci				req->int_reg |= MEMSTICK_INT_ERR;
4978c2ecf20Sopenharmony_ci			if (int_reg & MS_INT_CED)
4988c2ecf20Sopenharmony_ci				req->int_reg |= MEMSTICK_INT_CED;
4998c2ecf20Sopenharmony_ci		}
5008c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "int_reg: 0x%02x\n", req->int_reg);
5018c2ecf20Sopenharmony_ci	}
5028c2ecf20Sopenharmony_ci
5038c2ecf20Sopenharmony_ci	return 0;
5048c2ecf20Sopenharmony_ci}
5058c2ecf20Sopenharmony_ci
5068c2ecf20Sopenharmony_cistatic void rtsx_usb_ms_handle_req(struct work_struct *work)
5078c2ecf20Sopenharmony_ci{
5088c2ecf20Sopenharmony_ci	struct rtsx_usb_ms *host = container_of(work,
5098c2ecf20Sopenharmony_ci			struct rtsx_usb_ms, handle_req);
5108c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr = host->ucr;
5118c2ecf20Sopenharmony_ci	struct memstick_host *msh = host->msh;
5128c2ecf20Sopenharmony_ci	int rc;
5138c2ecf20Sopenharmony_ci
5148c2ecf20Sopenharmony_ci	if (!host->req) {
5158c2ecf20Sopenharmony_ci		pm_runtime_get_sync(ms_dev(host));
5168c2ecf20Sopenharmony_ci		do {
5178c2ecf20Sopenharmony_ci			rc = memstick_next_req(msh, &host->req);
5188c2ecf20Sopenharmony_ci			dev_dbg(ms_dev(host), "next req %d\n", rc);
5198c2ecf20Sopenharmony_ci
5208c2ecf20Sopenharmony_ci			if (!rc) {
5218c2ecf20Sopenharmony_ci				mutex_lock(&ucr->dev_mutex);
5228c2ecf20Sopenharmony_ci
5238c2ecf20Sopenharmony_ci				if (rtsx_usb_card_exclusive_check(ucr,
5248c2ecf20Sopenharmony_ci							RTSX_USB_MS_CARD))
5258c2ecf20Sopenharmony_ci					host->req->error = -EIO;
5268c2ecf20Sopenharmony_ci				else
5278c2ecf20Sopenharmony_ci					host->req->error =
5288c2ecf20Sopenharmony_ci						rtsx_usb_ms_issue_cmd(host);
5298c2ecf20Sopenharmony_ci
5308c2ecf20Sopenharmony_ci				mutex_unlock(&ucr->dev_mutex);
5318c2ecf20Sopenharmony_ci
5328c2ecf20Sopenharmony_ci				dev_dbg(ms_dev(host), "req result %d\n",
5338c2ecf20Sopenharmony_ci						host->req->error);
5348c2ecf20Sopenharmony_ci			}
5358c2ecf20Sopenharmony_ci		} while (!rc);
5368c2ecf20Sopenharmony_ci		pm_runtime_put_sync(ms_dev(host));
5378c2ecf20Sopenharmony_ci	}
5388c2ecf20Sopenharmony_ci
5398c2ecf20Sopenharmony_ci}
5408c2ecf20Sopenharmony_ci
5418c2ecf20Sopenharmony_cistatic void rtsx_usb_ms_request(struct memstick_host *msh)
5428c2ecf20Sopenharmony_ci{
5438c2ecf20Sopenharmony_ci	struct rtsx_usb_ms *host = memstick_priv(msh);
5448c2ecf20Sopenharmony_ci
5458c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "--> %s\n", __func__);
5468c2ecf20Sopenharmony_ci
5478c2ecf20Sopenharmony_ci	if (!host->eject)
5488c2ecf20Sopenharmony_ci		schedule_work(&host->handle_req);
5498c2ecf20Sopenharmony_ci}
5508c2ecf20Sopenharmony_ci
5518c2ecf20Sopenharmony_cistatic int rtsx_usb_ms_set_param(struct memstick_host *msh,
5528c2ecf20Sopenharmony_ci		enum memstick_param param, int value)
5538c2ecf20Sopenharmony_ci{
5548c2ecf20Sopenharmony_ci	struct rtsx_usb_ms *host = memstick_priv(msh);
5558c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr = host->ucr;
5568c2ecf20Sopenharmony_ci	unsigned int clock = 0;
5578c2ecf20Sopenharmony_ci	u8 ssc_depth = 0;
5588c2ecf20Sopenharmony_ci	int err;
5598c2ecf20Sopenharmony_ci
5608c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s: param = %d, value = %d\n",
5618c2ecf20Sopenharmony_ci			__func__, param, value);
5628c2ecf20Sopenharmony_ci
5638c2ecf20Sopenharmony_ci	pm_runtime_get_sync(ms_dev(host));
5648c2ecf20Sopenharmony_ci	mutex_lock(&ucr->dev_mutex);
5658c2ecf20Sopenharmony_ci
5668c2ecf20Sopenharmony_ci	err = rtsx_usb_card_exclusive_check(ucr, RTSX_USB_MS_CARD);
5678c2ecf20Sopenharmony_ci	if (err)
5688c2ecf20Sopenharmony_ci		goto out;
5698c2ecf20Sopenharmony_ci
5708c2ecf20Sopenharmony_ci	switch (param) {
5718c2ecf20Sopenharmony_ci	case MEMSTICK_POWER:
5728c2ecf20Sopenharmony_ci		if (value == host->power_mode)
5738c2ecf20Sopenharmony_ci			break;
5748c2ecf20Sopenharmony_ci
5758c2ecf20Sopenharmony_ci		if (value == MEMSTICK_POWER_ON) {
5768c2ecf20Sopenharmony_ci			pm_runtime_get_noresume(ms_dev(host));
5778c2ecf20Sopenharmony_ci			err = ms_power_on(host);
5788c2ecf20Sopenharmony_ci			if (err)
5798c2ecf20Sopenharmony_ci				pm_runtime_put_noidle(ms_dev(host));
5808c2ecf20Sopenharmony_ci		} else if (value == MEMSTICK_POWER_OFF) {
5818c2ecf20Sopenharmony_ci			err = ms_power_off(host);
5828c2ecf20Sopenharmony_ci			if (!err)
5838c2ecf20Sopenharmony_ci				pm_runtime_put_noidle(ms_dev(host));
5848c2ecf20Sopenharmony_ci		} else
5858c2ecf20Sopenharmony_ci			err = -EINVAL;
5868c2ecf20Sopenharmony_ci		if (!err)
5878c2ecf20Sopenharmony_ci			host->power_mode = value;
5888c2ecf20Sopenharmony_ci		break;
5898c2ecf20Sopenharmony_ci
5908c2ecf20Sopenharmony_ci	case MEMSTICK_INTERFACE:
5918c2ecf20Sopenharmony_ci		if (value == MEMSTICK_SERIAL) {
5928c2ecf20Sopenharmony_ci			clock = 19000000;
5938c2ecf20Sopenharmony_ci			ssc_depth = SSC_DEPTH_512K;
5948c2ecf20Sopenharmony_ci			err = rtsx_usb_write_register(ucr, MS_CFG, 0x5A,
5958c2ecf20Sopenharmony_ci				       MS_BUS_WIDTH_1 | PUSH_TIME_DEFAULT);
5968c2ecf20Sopenharmony_ci			if (err < 0)
5978c2ecf20Sopenharmony_ci				break;
5988c2ecf20Sopenharmony_ci		} else if (value == MEMSTICK_PAR4) {
5998c2ecf20Sopenharmony_ci			clock = 39000000;
6008c2ecf20Sopenharmony_ci			ssc_depth = SSC_DEPTH_1M;
6018c2ecf20Sopenharmony_ci
6028c2ecf20Sopenharmony_ci			err = rtsx_usb_write_register(ucr, MS_CFG, 0x5A,
6038c2ecf20Sopenharmony_ci					MS_BUS_WIDTH_4 | PUSH_TIME_ODD |
6048c2ecf20Sopenharmony_ci					MS_NO_CHECK_INT);
6058c2ecf20Sopenharmony_ci			if (err < 0)
6068c2ecf20Sopenharmony_ci				break;
6078c2ecf20Sopenharmony_ci		} else {
6088c2ecf20Sopenharmony_ci			err = -EINVAL;
6098c2ecf20Sopenharmony_ci			break;
6108c2ecf20Sopenharmony_ci		}
6118c2ecf20Sopenharmony_ci
6128c2ecf20Sopenharmony_ci		err = rtsx_usb_switch_clock(ucr, clock,
6138c2ecf20Sopenharmony_ci				ssc_depth, false, true, false);
6148c2ecf20Sopenharmony_ci		if (err < 0) {
6158c2ecf20Sopenharmony_ci			dev_dbg(ms_dev(host), "switch clock failed\n");
6168c2ecf20Sopenharmony_ci			break;
6178c2ecf20Sopenharmony_ci		}
6188c2ecf20Sopenharmony_ci
6198c2ecf20Sopenharmony_ci		host->ssc_depth = ssc_depth;
6208c2ecf20Sopenharmony_ci		host->clock = clock;
6218c2ecf20Sopenharmony_ci		host->ifmode = value;
6228c2ecf20Sopenharmony_ci		break;
6238c2ecf20Sopenharmony_ci	default:
6248c2ecf20Sopenharmony_ci		err = -EINVAL;
6258c2ecf20Sopenharmony_ci		break;
6268c2ecf20Sopenharmony_ci	}
6278c2ecf20Sopenharmony_ciout:
6288c2ecf20Sopenharmony_ci	mutex_unlock(&ucr->dev_mutex);
6298c2ecf20Sopenharmony_ci	pm_runtime_put_sync(ms_dev(host));
6308c2ecf20Sopenharmony_ci
6318c2ecf20Sopenharmony_ci	/* power-on delay */
6328c2ecf20Sopenharmony_ci	if (param == MEMSTICK_POWER && value == MEMSTICK_POWER_ON) {
6338c2ecf20Sopenharmony_ci		usleep_range(10000, 12000);
6348c2ecf20Sopenharmony_ci
6358c2ecf20Sopenharmony_ci		if (!host->eject)
6368c2ecf20Sopenharmony_ci			schedule_delayed_work(&host->poll_card, 100);
6378c2ecf20Sopenharmony_ci	}
6388c2ecf20Sopenharmony_ci
6398c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s: return = %d\n", __func__, err);
6408c2ecf20Sopenharmony_ci	return err;
6418c2ecf20Sopenharmony_ci}
6428c2ecf20Sopenharmony_ci
6438c2ecf20Sopenharmony_ci#ifdef CONFIG_PM_SLEEP
6448c2ecf20Sopenharmony_cistatic int rtsx_usb_ms_suspend(struct device *dev)
6458c2ecf20Sopenharmony_ci{
6468c2ecf20Sopenharmony_ci	struct rtsx_usb_ms *host = dev_get_drvdata(dev);
6478c2ecf20Sopenharmony_ci	struct memstick_host *msh = host->msh;
6488c2ecf20Sopenharmony_ci
6498c2ecf20Sopenharmony_ci	/* Since we use rtsx_usb's resume callback to runtime resume its
6508c2ecf20Sopenharmony_ci	 * children to implement remote wakeup signaling, this causes
6518c2ecf20Sopenharmony_ci	 * rtsx_usb_ms' runtime resume callback runs after its suspend
6528c2ecf20Sopenharmony_ci	 * callback:
6538c2ecf20Sopenharmony_ci	 * rtsx_usb_ms_suspend()
6548c2ecf20Sopenharmony_ci	 * rtsx_usb_resume()
6558c2ecf20Sopenharmony_ci	 *   -> rtsx_usb_ms_runtime_resume()
6568c2ecf20Sopenharmony_ci	 *     -> memstick_detect_change()
6578c2ecf20Sopenharmony_ci	 *
6588c2ecf20Sopenharmony_ci	 * rtsx_usb_suspend()
6598c2ecf20Sopenharmony_ci	 *
6608c2ecf20Sopenharmony_ci	 * To avoid this, skip runtime resume/suspend if system suspend is
6618c2ecf20Sopenharmony_ci	 * underway.
6628c2ecf20Sopenharmony_ci	 */
6638c2ecf20Sopenharmony_ci
6648c2ecf20Sopenharmony_ci	host->system_suspending = true;
6658c2ecf20Sopenharmony_ci	memstick_suspend_host(msh);
6668c2ecf20Sopenharmony_ci
6678c2ecf20Sopenharmony_ci	return 0;
6688c2ecf20Sopenharmony_ci}
6698c2ecf20Sopenharmony_ci
6708c2ecf20Sopenharmony_cistatic int rtsx_usb_ms_resume(struct device *dev)
6718c2ecf20Sopenharmony_ci{
6728c2ecf20Sopenharmony_ci	struct rtsx_usb_ms *host = dev_get_drvdata(dev);
6738c2ecf20Sopenharmony_ci	struct memstick_host *msh = host->msh;
6748c2ecf20Sopenharmony_ci
6758c2ecf20Sopenharmony_ci	memstick_resume_host(msh);
6768c2ecf20Sopenharmony_ci	host->system_suspending = false;
6778c2ecf20Sopenharmony_ci
6788c2ecf20Sopenharmony_ci	return 0;
6798c2ecf20Sopenharmony_ci}
6808c2ecf20Sopenharmony_ci#endif /* CONFIG_PM_SLEEP */
6818c2ecf20Sopenharmony_ci
6828c2ecf20Sopenharmony_ci#ifdef CONFIG_PM
6838c2ecf20Sopenharmony_cistatic int rtsx_usb_ms_runtime_suspend(struct device *dev)
6848c2ecf20Sopenharmony_ci{
6858c2ecf20Sopenharmony_ci	struct rtsx_usb_ms *host = dev_get_drvdata(dev);
6868c2ecf20Sopenharmony_ci
6878c2ecf20Sopenharmony_ci	if (host->system_suspending)
6888c2ecf20Sopenharmony_ci		return 0;
6898c2ecf20Sopenharmony_ci
6908c2ecf20Sopenharmony_ci	if (host->msh->card || host->power_mode != MEMSTICK_POWER_OFF)
6918c2ecf20Sopenharmony_ci		return -EAGAIN;
6928c2ecf20Sopenharmony_ci
6938c2ecf20Sopenharmony_ci	return 0;
6948c2ecf20Sopenharmony_ci}
6958c2ecf20Sopenharmony_ci
6968c2ecf20Sopenharmony_cistatic int rtsx_usb_ms_runtime_resume(struct device *dev)
6978c2ecf20Sopenharmony_ci{
6988c2ecf20Sopenharmony_ci	struct rtsx_usb_ms *host = dev_get_drvdata(dev);
6998c2ecf20Sopenharmony_ci
7008c2ecf20Sopenharmony_ci
7018c2ecf20Sopenharmony_ci	if (host->system_suspending)
7028c2ecf20Sopenharmony_ci		return 0;
7038c2ecf20Sopenharmony_ci
7048c2ecf20Sopenharmony_ci	memstick_detect_change(host->msh);
7058c2ecf20Sopenharmony_ci
7068c2ecf20Sopenharmony_ci	return 0;
7078c2ecf20Sopenharmony_ci}
7088c2ecf20Sopenharmony_ci#endif /* CONFIG_PM */
7098c2ecf20Sopenharmony_ci
7108c2ecf20Sopenharmony_cistatic const struct dev_pm_ops rtsx_usb_ms_pm_ops = {
7118c2ecf20Sopenharmony_ci	SET_SYSTEM_SLEEP_PM_OPS(rtsx_usb_ms_suspend, rtsx_usb_ms_resume)
7128c2ecf20Sopenharmony_ci	SET_RUNTIME_PM_OPS(rtsx_usb_ms_runtime_suspend, rtsx_usb_ms_runtime_resume, NULL)
7138c2ecf20Sopenharmony_ci};
7148c2ecf20Sopenharmony_ci
7158c2ecf20Sopenharmony_ci
7168c2ecf20Sopenharmony_cistatic void rtsx_usb_ms_poll_card(struct work_struct *work)
7178c2ecf20Sopenharmony_ci{
7188c2ecf20Sopenharmony_ci	struct rtsx_usb_ms *host = container_of(work, struct rtsx_usb_ms,
7198c2ecf20Sopenharmony_ci			poll_card.work);
7208c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr = host->ucr;
7218c2ecf20Sopenharmony_ci	int err;
7228c2ecf20Sopenharmony_ci	u8 val;
7238c2ecf20Sopenharmony_ci
7248c2ecf20Sopenharmony_ci	if (host->eject || host->power_mode != MEMSTICK_POWER_ON)
7258c2ecf20Sopenharmony_ci		return;
7268c2ecf20Sopenharmony_ci
7278c2ecf20Sopenharmony_ci	pm_runtime_get_sync(ms_dev(host));
7288c2ecf20Sopenharmony_ci	mutex_lock(&ucr->dev_mutex);
7298c2ecf20Sopenharmony_ci
7308c2ecf20Sopenharmony_ci	/* Check pending MS card changes */
7318c2ecf20Sopenharmony_ci	err = rtsx_usb_read_register(ucr, CARD_INT_PEND, &val);
7328c2ecf20Sopenharmony_ci	if (err) {
7338c2ecf20Sopenharmony_ci		mutex_unlock(&ucr->dev_mutex);
7348c2ecf20Sopenharmony_ci		goto poll_again;
7358c2ecf20Sopenharmony_ci	}
7368c2ecf20Sopenharmony_ci
7378c2ecf20Sopenharmony_ci	/* Clear the pending */
7388c2ecf20Sopenharmony_ci	rtsx_usb_write_register(ucr, CARD_INT_PEND,
7398c2ecf20Sopenharmony_ci			XD_INT | MS_INT | SD_INT,
7408c2ecf20Sopenharmony_ci			XD_INT | MS_INT | SD_INT);
7418c2ecf20Sopenharmony_ci
7428c2ecf20Sopenharmony_ci	mutex_unlock(&ucr->dev_mutex);
7438c2ecf20Sopenharmony_ci
7448c2ecf20Sopenharmony_ci	if (val & MS_INT) {
7458c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "MS slot change detected\n");
7468c2ecf20Sopenharmony_ci		memstick_detect_change(host->msh);
7478c2ecf20Sopenharmony_ci	}
7488c2ecf20Sopenharmony_ci
7498c2ecf20Sopenharmony_cipoll_again:
7508c2ecf20Sopenharmony_ci	pm_runtime_put_sync(ms_dev(host));
7518c2ecf20Sopenharmony_ci
7528c2ecf20Sopenharmony_ci	if (!host->eject && host->power_mode == MEMSTICK_POWER_ON)
7538c2ecf20Sopenharmony_ci		schedule_delayed_work(&host->poll_card, 100);
7548c2ecf20Sopenharmony_ci}
7558c2ecf20Sopenharmony_ci
7568c2ecf20Sopenharmony_cistatic int rtsx_usb_ms_drv_probe(struct platform_device *pdev)
7578c2ecf20Sopenharmony_ci{
7588c2ecf20Sopenharmony_ci	struct memstick_host *msh;
7598c2ecf20Sopenharmony_ci	struct rtsx_usb_ms *host;
7608c2ecf20Sopenharmony_ci	struct rtsx_ucr *ucr;
7618c2ecf20Sopenharmony_ci	int err;
7628c2ecf20Sopenharmony_ci
7638c2ecf20Sopenharmony_ci	ucr = usb_get_intfdata(to_usb_interface(pdev->dev.parent));
7648c2ecf20Sopenharmony_ci	if (!ucr)
7658c2ecf20Sopenharmony_ci		return -ENXIO;
7668c2ecf20Sopenharmony_ci
7678c2ecf20Sopenharmony_ci	dev_dbg(&(pdev->dev),
7688c2ecf20Sopenharmony_ci			"Realtek USB Memstick controller found\n");
7698c2ecf20Sopenharmony_ci
7708c2ecf20Sopenharmony_ci	msh = memstick_alloc_host(sizeof(*host), &pdev->dev);
7718c2ecf20Sopenharmony_ci	if (!msh)
7728c2ecf20Sopenharmony_ci		return -ENOMEM;
7738c2ecf20Sopenharmony_ci
7748c2ecf20Sopenharmony_ci	host = memstick_priv(msh);
7758c2ecf20Sopenharmony_ci	host->ucr = ucr;
7768c2ecf20Sopenharmony_ci	host->msh = msh;
7778c2ecf20Sopenharmony_ci	host->pdev = pdev;
7788c2ecf20Sopenharmony_ci	host->power_mode = MEMSTICK_POWER_OFF;
7798c2ecf20Sopenharmony_ci	platform_set_drvdata(pdev, host);
7808c2ecf20Sopenharmony_ci
7818c2ecf20Sopenharmony_ci	mutex_init(&host->host_mutex);
7828c2ecf20Sopenharmony_ci	INIT_WORK(&host->handle_req, rtsx_usb_ms_handle_req);
7838c2ecf20Sopenharmony_ci
7848c2ecf20Sopenharmony_ci	INIT_DELAYED_WORK(&host->poll_card, rtsx_usb_ms_poll_card);
7858c2ecf20Sopenharmony_ci
7868c2ecf20Sopenharmony_ci	msh->request = rtsx_usb_ms_request;
7878c2ecf20Sopenharmony_ci	msh->set_param = rtsx_usb_ms_set_param;
7888c2ecf20Sopenharmony_ci	msh->caps = MEMSTICK_CAP_PAR4;
7898c2ecf20Sopenharmony_ci
7908c2ecf20Sopenharmony_ci	pm_runtime_get_noresume(ms_dev(host));
7918c2ecf20Sopenharmony_ci	pm_runtime_set_active(ms_dev(host));
7928c2ecf20Sopenharmony_ci	pm_runtime_enable(ms_dev(host));
7938c2ecf20Sopenharmony_ci
7948c2ecf20Sopenharmony_ci	err = memstick_add_host(msh);
7958c2ecf20Sopenharmony_ci	if (err)
7968c2ecf20Sopenharmony_ci		goto err_out;
7978c2ecf20Sopenharmony_ci
7988c2ecf20Sopenharmony_ci	pm_runtime_put(ms_dev(host));
7998c2ecf20Sopenharmony_ci
8008c2ecf20Sopenharmony_ci	return 0;
8018c2ecf20Sopenharmony_cierr_out:
8028c2ecf20Sopenharmony_ci	pm_runtime_disable(ms_dev(host));
8038c2ecf20Sopenharmony_ci	pm_runtime_put_noidle(ms_dev(host));
8048c2ecf20Sopenharmony_ci	memstick_free_host(msh);
8058c2ecf20Sopenharmony_ci	return err;
8068c2ecf20Sopenharmony_ci}
8078c2ecf20Sopenharmony_ci
8088c2ecf20Sopenharmony_cistatic int rtsx_usb_ms_drv_remove(struct platform_device *pdev)
8098c2ecf20Sopenharmony_ci{
8108c2ecf20Sopenharmony_ci	struct rtsx_usb_ms *host = platform_get_drvdata(pdev);
8118c2ecf20Sopenharmony_ci	struct memstick_host *msh = host->msh;
8128c2ecf20Sopenharmony_ci	int err;
8138c2ecf20Sopenharmony_ci
8148c2ecf20Sopenharmony_ci	host->eject = true;
8158c2ecf20Sopenharmony_ci	cancel_work_sync(&host->handle_req);
8168c2ecf20Sopenharmony_ci
8178c2ecf20Sopenharmony_ci	mutex_lock(&host->host_mutex);
8188c2ecf20Sopenharmony_ci	if (host->req) {
8198c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host),
8208c2ecf20Sopenharmony_ci			"%s: Controller removed during transfer\n",
8218c2ecf20Sopenharmony_ci			dev_name(&msh->dev));
8228c2ecf20Sopenharmony_ci		host->req->error = -ENOMEDIUM;
8238c2ecf20Sopenharmony_ci		do {
8248c2ecf20Sopenharmony_ci			err = memstick_next_req(msh, &host->req);
8258c2ecf20Sopenharmony_ci			if (!err)
8268c2ecf20Sopenharmony_ci				host->req->error = -ENOMEDIUM;
8278c2ecf20Sopenharmony_ci		} while (!err);
8288c2ecf20Sopenharmony_ci	}
8298c2ecf20Sopenharmony_ci	mutex_unlock(&host->host_mutex);
8308c2ecf20Sopenharmony_ci
8318c2ecf20Sopenharmony_ci	/* Balance possible unbalanced usage count
8328c2ecf20Sopenharmony_ci	 * e.g. unconditional module removal
8338c2ecf20Sopenharmony_ci	 */
8348c2ecf20Sopenharmony_ci	if (pm_runtime_active(ms_dev(host)))
8358c2ecf20Sopenharmony_ci		pm_runtime_put(ms_dev(host));
8368c2ecf20Sopenharmony_ci
8378c2ecf20Sopenharmony_ci	pm_runtime_disable(ms_dev(host));
8388c2ecf20Sopenharmony_ci	memstick_remove_host(msh);
8398c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host),
8408c2ecf20Sopenharmony_ci		": Realtek USB Memstick controller has been removed\n");
8418c2ecf20Sopenharmony_ci	memstick_free_host(msh);
8428c2ecf20Sopenharmony_ci	platform_set_drvdata(pdev, NULL);
8438c2ecf20Sopenharmony_ci
8448c2ecf20Sopenharmony_ci	return 0;
8458c2ecf20Sopenharmony_ci}
8468c2ecf20Sopenharmony_ci
8478c2ecf20Sopenharmony_cistatic struct platform_device_id rtsx_usb_ms_ids[] = {
8488c2ecf20Sopenharmony_ci	{
8498c2ecf20Sopenharmony_ci		.name = "rtsx_usb_ms",
8508c2ecf20Sopenharmony_ci	}, {
8518c2ecf20Sopenharmony_ci		/* sentinel */
8528c2ecf20Sopenharmony_ci	}
8538c2ecf20Sopenharmony_ci};
8548c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(platform, rtsx_usb_ms_ids);
8558c2ecf20Sopenharmony_ci
8568c2ecf20Sopenharmony_cistatic struct platform_driver rtsx_usb_ms_driver = {
8578c2ecf20Sopenharmony_ci	.probe		= rtsx_usb_ms_drv_probe,
8588c2ecf20Sopenharmony_ci	.remove		= rtsx_usb_ms_drv_remove,
8598c2ecf20Sopenharmony_ci	.id_table       = rtsx_usb_ms_ids,
8608c2ecf20Sopenharmony_ci	.driver		= {
8618c2ecf20Sopenharmony_ci		.name	= "rtsx_usb_ms",
8628c2ecf20Sopenharmony_ci		.pm	= &rtsx_usb_ms_pm_ops,
8638c2ecf20Sopenharmony_ci	},
8648c2ecf20Sopenharmony_ci};
8658c2ecf20Sopenharmony_cimodule_platform_driver(rtsx_usb_ms_driver);
8668c2ecf20Sopenharmony_ci
8678c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2");
8688c2ecf20Sopenharmony_ciMODULE_AUTHOR("Roger Tseng <rogerable@realtek.com>");
8698c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Realtek USB Memstick Card Host Driver");
870