18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
28c2ecf20Sopenharmony_ci/* Realtek PCI-Express 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 *   Wei WANG <wei_wang@realsil.com.cn>
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/memstick.h>
158c2ecf20Sopenharmony_ci#include <linux/rtsx_pci.h>
168c2ecf20Sopenharmony_ci#include <asm/unaligned.h>
178c2ecf20Sopenharmony_ci
188c2ecf20Sopenharmony_cistruct realtek_pci_ms {
198c2ecf20Sopenharmony_ci	struct platform_device	*pdev;
208c2ecf20Sopenharmony_ci	struct rtsx_pcr		*pcr;
218c2ecf20Sopenharmony_ci	struct memstick_host	*msh;
228c2ecf20Sopenharmony_ci	struct memstick_request	*req;
238c2ecf20Sopenharmony_ci
248c2ecf20Sopenharmony_ci	struct mutex		host_mutex;
258c2ecf20Sopenharmony_ci	struct work_struct	handle_req;
268c2ecf20Sopenharmony_ci
278c2ecf20Sopenharmony_ci	u8			ssc_depth;
288c2ecf20Sopenharmony_ci	unsigned int		clock;
298c2ecf20Sopenharmony_ci	unsigned char           ifmode;
308c2ecf20Sopenharmony_ci	bool			eject;
318c2ecf20Sopenharmony_ci};
328c2ecf20Sopenharmony_ci
338c2ecf20Sopenharmony_cistatic inline struct device *ms_dev(struct realtek_pci_ms *host)
348c2ecf20Sopenharmony_ci{
358c2ecf20Sopenharmony_ci	return &(host->pdev->dev);
368c2ecf20Sopenharmony_ci}
378c2ecf20Sopenharmony_ci
388c2ecf20Sopenharmony_cistatic inline void ms_clear_error(struct realtek_pci_ms *host)
398c2ecf20Sopenharmony_ci{
408c2ecf20Sopenharmony_ci	rtsx_pci_write_register(host->pcr, CARD_STOP,
418c2ecf20Sopenharmony_ci			MS_STOP | MS_CLR_ERR, MS_STOP | MS_CLR_ERR);
428c2ecf20Sopenharmony_ci}
438c2ecf20Sopenharmony_ci
448c2ecf20Sopenharmony_ci#ifdef DEBUG
458c2ecf20Sopenharmony_ci
468c2ecf20Sopenharmony_cistatic void ms_print_debug_regs(struct realtek_pci_ms *host)
478c2ecf20Sopenharmony_ci{
488c2ecf20Sopenharmony_ci	struct rtsx_pcr *pcr = host->pcr;
498c2ecf20Sopenharmony_ci	u16 i;
508c2ecf20Sopenharmony_ci	u8 *ptr;
518c2ecf20Sopenharmony_ci
528c2ecf20Sopenharmony_ci	/* Print MS host internal registers */
538c2ecf20Sopenharmony_ci	rtsx_pci_init_cmd(pcr);
548c2ecf20Sopenharmony_ci	for (i = 0xFD40; i <= 0xFD44; i++)
558c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, READ_REG_CMD, i, 0, 0);
568c2ecf20Sopenharmony_ci	for (i = 0xFD52; i <= 0xFD69; i++)
578c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, READ_REG_CMD, i, 0, 0);
588c2ecf20Sopenharmony_ci	rtsx_pci_send_cmd(pcr, 100);
598c2ecf20Sopenharmony_ci
608c2ecf20Sopenharmony_ci	ptr = rtsx_pci_get_cmd_data(pcr);
618c2ecf20Sopenharmony_ci	for (i = 0xFD40; i <= 0xFD44; i++)
628c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "0x%04X: 0x%02x\n", i, *(ptr++));
638c2ecf20Sopenharmony_ci	for (i = 0xFD52; i <= 0xFD69; i++)
648c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "0x%04X: 0x%02x\n", i, *(ptr++));
658c2ecf20Sopenharmony_ci}
668c2ecf20Sopenharmony_ci
678c2ecf20Sopenharmony_ci#else
688c2ecf20Sopenharmony_ci
698c2ecf20Sopenharmony_ci#define ms_print_debug_regs(host)
708c2ecf20Sopenharmony_ci
718c2ecf20Sopenharmony_ci#endif
728c2ecf20Sopenharmony_ci
738c2ecf20Sopenharmony_cistatic int ms_power_on(struct realtek_pci_ms *host)
748c2ecf20Sopenharmony_ci{
758c2ecf20Sopenharmony_ci	struct rtsx_pcr *pcr = host->pcr;
768c2ecf20Sopenharmony_ci	int err;
778c2ecf20Sopenharmony_ci
788c2ecf20Sopenharmony_ci	rtsx_pci_init_cmd(pcr);
798c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, CARD_SELECT, 0x07, MS_MOD_SEL);
808c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, CARD_SHARE_MODE,
818c2ecf20Sopenharmony_ci			CARD_SHARE_MASK, CARD_SHARE_48_MS);
828c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, CARD_CLK_EN,
838c2ecf20Sopenharmony_ci			MS_CLK_EN, MS_CLK_EN);
848c2ecf20Sopenharmony_ci	err = rtsx_pci_send_cmd(pcr, 100);
858c2ecf20Sopenharmony_ci	if (err < 0)
868c2ecf20Sopenharmony_ci		return err;
878c2ecf20Sopenharmony_ci
888c2ecf20Sopenharmony_ci	err = rtsx_pci_card_pull_ctl_enable(pcr, RTSX_MS_CARD);
898c2ecf20Sopenharmony_ci	if (err < 0)
908c2ecf20Sopenharmony_ci		return err;
918c2ecf20Sopenharmony_ci
928c2ecf20Sopenharmony_ci	err = rtsx_pci_card_power_on(pcr, RTSX_MS_CARD);
938c2ecf20Sopenharmony_ci	if (err < 0)
948c2ecf20Sopenharmony_ci		return err;
958c2ecf20Sopenharmony_ci
968c2ecf20Sopenharmony_ci	/* Wait ms power stable */
978c2ecf20Sopenharmony_ci	msleep(150);
988c2ecf20Sopenharmony_ci
998c2ecf20Sopenharmony_ci	err = rtsx_pci_write_register(pcr, CARD_OE,
1008c2ecf20Sopenharmony_ci			MS_OUTPUT_EN, MS_OUTPUT_EN);
1018c2ecf20Sopenharmony_ci	if (err < 0)
1028c2ecf20Sopenharmony_ci		return err;
1038c2ecf20Sopenharmony_ci
1048c2ecf20Sopenharmony_ci	return 0;
1058c2ecf20Sopenharmony_ci}
1068c2ecf20Sopenharmony_ci
1078c2ecf20Sopenharmony_cistatic int ms_power_off(struct realtek_pci_ms *host)
1088c2ecf20Sopenharmony_ci{
1098c2ecf20Sopenharmony_ci	struct rtsx_pcr *pcr = host->pcr;
1108c2ecf20Sopenharmony_ci	int err;
1118c2ecf20Sopenharmony_ci
1128c2ecf20Sopenharmony_ci	rtsx_pci_init_cmd(pcr);
1138c2ecf20Sopenharmony_ci
1148c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, CARD_CLK_EN, MS_CLK_EN, 0);
1158c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, CARD_OE, MS_OUTPUT_EN, 0);
1168c2ecf20Sopenharmony_ci
1178c2ecf20Sopenharmony_ci	err = rtsx_pci_send_cmd(pcr, 100);
1188c2ecf20Sopenharmony_ci	if (err < 0)
1198c2ecf20Sopenharmony_ci		return err;
1208c2ecf20Sopenharmony_ci
1218c2ecf20Sopenharmony_ci	err = rtsx_pci_card_power_off(pcr, RTSX_MS_CARD);
1228c2ecf20Sopenharmony_ci	if (err < 0)
1238c2ecf20Sopenharmony_ci		return err;
1248c2ecf20Sopenharmony_ci
1258c2ecf20Sopenharmony_ci	return rtsx_pci_card_pull_ctl_disable(pcr, RTSX_MS_CARD);
1268c2ecf20Sopenharmony_ci}
1278c2ecf20Sopenharmony_ci
1288c2ecf20Sopenharmony_cistatic int ms_transfer_data(struct realtek_pci_ms *host, unsigned char data_dir,
1298c2ecf20Sopenharmony_ci		u8 tpc, u8 cfg, struct scatterlist *sg)
1308c2ecf20Sopenharmony_ci{
1318c2ecf20Sopenharmony_ci	struct rtsx_pcr *pcr = host->pcr;
1328c2ecf20Sopenharmony_ci	int err;
1338c2ecf20Sopenharmony_ci	unsigned int length = sg->length;
1348c2ecf20Sopenharmony_ci	u16 sec_cnt = (u16)(length / 512);
1358c2ecf20Sopenharmony_ci	u8 val, trans_mode, dma_dir;
1368c2ecf20Sopenharmony_ci	struct memstick_dev *card = host->msh->card;
1378c2ecf20Sopenharmony_ci	bool pro_card = card->id.type == MEMSTICK_TYPE_PRO;
1388c2ecf20Sopenharmony_ci
1398c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s: tpc = 0x%02x, data_dir = %s, length = %d\n",
1408c2ecf20Sopenharmony_ci			__func__, tpc, (data_dir == READ) ? "READ" : "WRITE",
1418c2ecf20Sopenharmony_ci			length);
1428c2ecf20Sopenharmony_ci
1438c2ecf20Sopenharmony_ci	if (data_dir == READ) {
1448c2ecf20Sopenharmony_ci		dma_dir = DMA_DIR_FROM_CARD;
1458c2ecf20Sopenharmony_ci		trans_mode = pro_card ? MS_TM_AUTO_READ : MS_TM_NORMAL_READ;
1468c2ecf20Sopenharmony_ci	} else {
1478c2ecf20Sopenharmony_ci		dma_dir = DMA_DIR_TO_CARD;
1488c2ecf20Sopenharmony_ci		trans_mode = pro_card ? MS_TM_AUTO_WRITE : MS_TM_NORMAL_WRITE;
1498c2ecf20Sopenharmony_ci	}
1508c2ecf20Sopenharmony_ci
1518c2ecf20Sopenharmony_ci	rtsx_pci_init_cmd(pcr);
1528c2ecf20Sopenharmony_ci
1538c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_TPC, 0xFF, tpc);
1548c2ecf20Sopenharmony_ci	if (pro_card) {
1558c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_SECTOR_CNT_H,
1568c2ecf20Sopenharmony_ci				0xFF, (u8)(sec_cnt >> 8));
1578c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_SECTOR_CNT_L,
1588c2ecf20Sopenharmony_ci				0xFF, (u8)sec_cnt);
1598c2ecf20Sopenharmony_ci	}
1608c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_TRANS_CFG, 0xFF, cfg);
1618c2ecf20Sopenharmony_ci
1628c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, IRQSTAT0,
1638c2ecf20Sopenharmony_ci			DMA_DONE_INT, DMA_DONE_INT);
1648c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, DMATC3, 0xFF, (u8)(length >> 24));
1658c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, DMATC2, 0xFF, (u8)(length >> 16));
1668c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, DMATC1, 0xFF, (u8)(length >> 8));
1678c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, DMATC0, 0xFF, (u8)length);
1688c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, DMACTL,
1698c2ecf20Sopenharmony_ci			0x03 | DMA_PACK_SIZE_MASK, dma_dir | DMA_EN | DMA_512);
1708c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, CARD_DATA_SOURCE,
1718c2ecf20Sopenharmony_ci			0x01, RING_BUFFER);
1728c2ecf20Sopenharmony_ci
1738c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_TRANSFER,
1748c2ecf20Sopenharmony_ci			0xFF, MS_TRANSFER_START | trans_mode);
1758c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, CHECK_REG_CMD, MS_TRANSFER,
1768c2ecf20Sopenharmony_ci			MS_TRANSFER_END, MS_TRANSFER_END);
1778c2ecf20Sopenharmony_ci
1788c2ecf20Sopenharmony_ci	rtsx_pci_send_cmd_no_wait(pcr);
1798c2ecf20Sopenharmony_ci
1808c2ecf20Sopenharmony_ci	err = rtsx_pci_transfer_data(pcr, sg, 1, data_dir == READ, 10000);
1818c2ecf20Sopenharmony_ci	if (err < 0) {
1828c2ecf20Sopenharmony_ci		ms_clear_error(host);
1838c2ecf20Sopenharmony_ci		return err;
1848c2ecf20Sopenharmony_ci	}
1858c2ecf20Sopenharmony_ci
1868c2ecf20Sopenharmony_ci	rtsx_pci_read_register(pcr, MS_TRANS_CFG, &val);
1878c2ecf20Sopenharmony_ci	if (pro_card) {
1888c2ecf20Sopenharmony_ci		if (val & (MS_INT_CMDNK | MS_INT_ERR |
1898c2ecf20Sopenharmony_ci				MS_CRC16_ERR | MS_RDY_TIMEOUT))
1908c2ecf20Sopenharmony_ci			return -EIO;
1918c2ecf20Sopenharmony_ci	} else {
1928c2ecf20Sopenharmony_ci		if (val & (MS_CRC16_ERR | MS_RDY_TIMEOUT))
1938c2ecf20Sopenharmony_ci			return -EIO;
1948c2ecf20Sopenharmony_ci	}
1958c2ecf20Sopenharmony_ci
1968c2ecf20Sopenharmony_ci	return 0;
1978c2ecf20Sopenharmony_ci}
1988c2ecf20Sopenharmony_ci
1998c2ecf20Sopenharmony_cistatic int ms_write_bytes(struct realtek_pci_ms *host, u8 tpc,
2008c2ecf20Sopenharmony_ci		u8 cfg, u8 cnt, u8 *data, u8 *int_reg)
2018c2ecf20Sopenharmony_ci{
2028c2ecf20Sopenharmony_ci	struct rtsx_pcr *pcr = host->pcr;
2038c2ecf20Sopenharmony_ci	int err, i;
2048c2ecf20Sopenharmony_ci
2058c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s: tpc = 0x%02x\n", __func__, tpc);
2068c2ecf20Sopenharmony_ci
2078c2ecf20Sopenharmony_ci	if (!data)
2088c2ecf20Sopenharmony_ci		return -EINVAL;
2098c2ecf20Sopenharmony_ci
2108c2ecf20Sopenharmony_ci	rtsx_pci_init_cmd(pcr);
2118c2ecf20Sopenharmony_ci
2128c2ecf20Sopenharmony_ci	for (i = 0; i < cnt; i++)
2138c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, WRITE_REG_CMD,
2148c2ecf20Sopenharmony_ci				PPBUF_BASE2 + i, 0xFF, data[i]);
2158c2ecf20Sopenharmony_ci	if (cnt % 2)
2168c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, WRITE_REG_CMD,
2178c2ecf20Sopenharmony_ci				PPBUF_BASE2 + i, 0xFF, 0xFF);
2188c2ecf20Sopenharmony_ci
2198c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_TPC, 0xFF, tpc);
2208c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_BYTE_CNT, 0xFF, cnt);
2218c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_TRANS_CFG, 0xFF, cfg);
2228c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, CARD_DATA_SOURCE,
2238c2ecf20Sopenharmony_ci			0x01, PINGPONG_BUFFER);
2248c2ecf20Sopenharmony_ci
2258c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_TRANSFER,
2268c2ecf20Sopenharmony_ci			0xFF, MS_TRANSFER_START | MS_TM_WRITE_BYTES);
2278c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, CHECK_REG_CMD, MS_TRANSFER,
2288c2ecf20Sopenharmony_ci			MS_TRANSFER_END, MS_TRANSFER_END);
2298c2ecf20Sopenharmony_ci	if (int_reg)
2308c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, READ_REG_CMD, MS_TRANS_CFG, 0, 0);
2318c2ecf20Sopenharmony_ci
2328c2ecf20Sopenharmony_ci	err = rtsx_pci_send_cmd(pcr, 5000);
2338c2ecf20Sopenharmony_ci	if (err < 0) {
2348c2ecf20Sopenharmony_ci		u8 val;
2358c2ecf20Sopenharmony_ci
2368c2ecf20Sopenharmony_ci		rtsx_pci_read_register(pcr, MS_TRANS_CFG, &val);
2378c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "MS_TRANS_CFG: 0x%02x\n", val);
2388c2ecf20Sopenharmony_ci
2398c2ecf20Sopenharmony_ci		if (int_reg)
2408c2ecf20Sopenharmony_ci			*int_reg = val & 0x0F;
2418c2ecf20Sopenharmony_ci
2428c2ecf20Sopenharmony_ci		ms_print_debug_regs(host);
2438c2ecf20Sopenharmony_ci
2448c2ecf20Sopenharmony_ci		ms_clear_error(host);
2458c2ecf20Sopenharmony_ci
2468c2ecf20Sopenharmony_ci		if (!(tpc & 0x08)) {
2478c2ecf20Sopenharmony_ci			if (val & MS_CRC16_ERR)
2488c2ecf20Sopenharmony_ci				return -EIO;
2498c2ecf20Sopenharmony_ci		} else {
2508c2ecf20Sopenharmony_ci			if (!(val & 0x80)) {
2518c2ecf20Sopenharmony_ci				if (val & (MS_INT_ERR | MS_INT_CMDNK))
2528c2ecf20Sopenharmony_ci					return -EIO;
2538c2ecf20Sopenharmony_ci			}
2548c2ecf20Sopenharmony_ci		}
2558c2ecf20Sopenharmony_ci
2568c2ecf20Sopenharmony_ci		return -ETIMEDOUT;
2578c2ecf20Sopenharmony_ci	}
2588c2ecf20Sopenharmony_ci
2598c2ecf20Sopenharmony_ci	if (int_reg) {
2608c2ecf20Sopenharmony_ci		u8 *ptr = rtsx_pci_get_cmd_data(pcr) + 1;
2618c2ecf20Sopenharmony_ci		*int_reg = *ptr & 0x0F;
2628c2ecf20Sopenharmony_ci	}
2638c2ecf20Sopenharmony_ci
2648c2ecf20Sopenharmony_ci	return 0;
2658c2ecf20Sopenharmony_ci}
2668c2ecf20Sopenharmony_ci
2678c2ecf20Sopenharmony_cistatic int ms_read_bytes(struct realtek_pci_ms *host, u8 tpc,
2688c2ecf20Sopenharmony_ci		u8 cfg, u8 cnt, u8 *data, u8 *int_reg)
2698c2ecf20Sopenharmony_ci{
2708c2ecf20Sopenharmony_ci	struct rtsx_pcr *pcr = host->pcr;
2718c2ecf20Sopenharmony_ci	int err, i;
2728c2ecf20Sopenharmony_ci	u8 *ptr;
2738c2ecf20Sopenharmony_ci
2748c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s: tpc = 0x%02x\n", __func__, tpc);
2758c2ecf20Sopenharmony_ci
2768c2ecf20Sopenharmony_ci	if (!data)
2778c2ecf20Sopenharmony_ci		return -EINVAL;
2788c2ecf20Sopenharmony_ci
2798c2ecf20Sopenharmony_ci	rtsx_pci_init_cmd(pcr);
2808c2ecf20Sopenharmony_ci
2818c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_TPC, 0xFF, tpc);
2828c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_BYTE_CNT, 0xFF, cnt);
2838c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_TRANS_CFG, 0xFF, cfg);
2848c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, CARD_DATA_SOURCE,
2858c2ecf20Sopenharmony_ci			0x01, PINGPONG_BUFFER);
2868c2ecf20Sopenharmony_ci
2878c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, WRITE_REG_CMD, MS_TRANSFER,
2888c2ecf20Sopenharmony_ci			0xFF, MS_TRANSFER_START | MS_TM_READ_BYTES);
2898c2ecf20Sopenharmony_ci	rtsx_pci_add_cmd(pcr, CHECK_REG_CMD, MS_TRANSFER,
2908c2ecf20Sopenharmony_ci			MS_TRANSFER_END, MS_TRANSFER_END);
2918c2ecf20Sopenharmony_ci	for (i = 0; i < cnt - 1; i++)
2928c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, READ_REG_CMD, PPBUF_BASE2 + i, 0, 0);
2938c2ecf20Sopenharmony_ci	if (cnt % 2)
2948c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, READ_REG_CMD, PPBUF_BASE2 + cnt, 0, 0);
2958c2ecf20Sopenharmony_ci	else
2968c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, READ_REG_CMD,
2978c2ecf20Sopenharmony_ci				PPBUF_BASE2 + cnt - 1, 0, 0);
2988c2ecf20Sopenharmony_ci	if (int_reg)
2998c2ecf20Sopenharmony_ci		rtsx_pci_add_cmd(pcr, READ_REG_CMD, MS_TRANS_CFG, 0, 0);
3008c2ecf20Sopenharmony_ci
3018c2ecf20Sopenharmony_ci	err = rtsx_pci_send_cmd(pcr, 5000);
3028c2ecf20Sopenharmony_ci	if (err < 0) {
3038c2ecf20Sopenharmony_ci		u8 val;
3048c2ecf20Sopenharmony_ci
3058c2ecf20Sopenharmony_ci		rtsx_pci_read_register(pcr, MS_TRANS_CFG, &val);
3068c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "MS_TRANS_CFG: 0x%02x\n", val);
3078c2ecf20Sopenharmony_ci
3088c2ecf20Sopenharmony_ci		if (int_reg)
3098c2ecf20Sopenharmony_ci			*int_reg = val & 0x0F;
3108c2ecf20Sopenharmony_ci
3118c2ecf20Sopenharmony_ci		ms_print_debug_regs(host);
3128c2ecf20Sopenharmony_ci
3138c2ecf20Sopenharmony_ci		ms_clear_error(host);
3148c2ecf20Sopenharmony_ci
3158c2ecf20Sopenharmony_ci		if (!(tpc & 0x08)) {
3168c2ecf20Sopenharmony_ci			if (val & MS_CRC16_ERR)
3178c2ecf20Sopenharmony_ci				return -EIO;
3188c2ecf20Sopenharmony_ci		} else {
3198c2ecf20Sopenharmony_ci			if (!(val & 0x80)) {
3208c2ecf20Sopenharmony_ci				if (val & (MS_INT_ERR | MS_INT_CMDNK))
3218c2ecf20Sopenharmony_ci					return -EIO;
3228c2ecf20Sopenharmony_ci			}
3238c2ecf20Sopenharmony_ci		}
3248c2ecf20Sopenharmony_ci
3258c2ecf20Sopenharmony_ci		return -ETIMEDOUT;
3268c2ecf20Sopenharmony_ci	}
3278c2ecf20Sopenharmony_ci
3288c2ecf20Sopenharmony_ci	ptr = rtsx_pci_get_cmd_data(pcr) + 1;
3298c2ecf20Sopenharmony_ci	for (i = 0; i < cnt; i++)
3308c2ecf20Sopenharmony_ci		data[i] = *ptr++;
3318c2ecf20Sopenharmony_ci
3328c2ecf20Sopenharmony_ci	if (int_reg)
3338c2ecf20Sopenharmony_ci		*int_reg = *ptr & 0x0F;
3348c2ecf20Sopenharmony_ci
3358c2ecf20Sopenharmony_ci	return 0;
3368c2ecf20Sopenharmony_ci}
3378c2ecf20Sopenharmony_ci
3388c2ecf20Sopenharmony_cistatic int rtsx_pci_ms_issue_cmd(struct realtek_pci_ms *host)
3398c2ecf20Sopenharmony_ci{
3408c2ecf20Sopenharmony_ci	struct memstick_request *req = host->req;
3418c2ecf20Sopenharmony_ci	int err = 0;
3428c2ecf20Sopenharmony_ci	u8 cfg = 0, int_reg;
3438c2ecf20Sopenharmony_ci
3448c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s\n", __func__);
3458c2ecf20Sopenharmony_ci
3468c2ecf20Sopenharmony_ci	if (req->need_card_int) {
3478c2ecf20Sopenharmony_ci		if (host->ifmode != MEMSTICK_SERIAL)
3488c2ecf20Sopenharmony_ci			cfg = WAIT_INT;
3498c2ecf20Sopenharmony_ci	}
3508c2ecf20Sopenharmony_ci
3518c2ecf20Sopenharmony_ci	if (req->long_data) {
3528c2ecf20Sopenharmony_ci		err = ms_transfer_data(host, req->data_dir,
3538c2ecf20Sopenharmony_ci				req->tpc, cfg, &(req->sg));
3548c2ecf20Sopenharmony_ci	} else {
3558c2ecf20Sopenharmony_ci		if (req->data_dir == READ) {
3568c2ecf20Sopenharmony_ci			err = ms_read_bytes(host, req->tpc, cfg,
3578c2ecf20Sopenharmony_ci					req->data_len, req->data, &int_reg);
3588c2ecf20Sopenharmony_ci		} else {
3598c2ecf20Sopenharmony_ci			err = ms_write_bytes(host, req->tpc, cfg,
3608c2ecf20Sopenharmony_ci					req->data_len, req->data, &int_reg);
3618c2ecf20Sopenharmony_ci		}
3628c2ecf20Sopenharmony_ci	}
3638c2ecf20Sopenharmony_ci	if (err < 0)
3648c2ecf20Sopenharmony_ci		return err;
3658c2ecf20Sopenharmony_ci
3668c2ecf20Sopenharmony_ci	if (req->need_card_int && (host->ifmode == MEMSTICK_SERIAL)) {
3678c2ecf20Sopenharmony_ci		err = ms_read_bytes(host, MS_TPC_GET_INT,
3688c2ecf20Sopenharmony_ci				NO_WAIT_INT, 1, &int_reg, NULL);
3698c2ecf20Sopenharmony_ci		if (err < 0)
3708c2ecf20Sopenharmony_ci			return err;
3718c2ecf20Sopenharmony_ci	}
3728c2ecf20Sopenharmony_ci
3738c2ecf20Sopenharmony_ci	if (req->need_card_int) {
3748c2ecf20Sopenharmony_ci		dev_dbg(ms_dev(host), "int_reg: 0x%02x\n", int_reg);
3758c2ecf20Sopenharmony_ci
3768c2ecf20Sopenharmony_ci		if (int_reg & MS_INT_CMDNK)
3778c2ecf20Sopenharmony_ci			req->int_reg |= MEMSTICK_INT_CMDNAK;
3788c2ecf20Sopenharmony_ci		if (int_reg & MS_INT_BREQ)
3798c2ecf20Sopenharmony_ci			req->int_reg |= MEMSTICK_INT_BREQ;
3808c2ecf20Sopenharmony_ci		if (int_reg & MS_INT_ERR)
3818c2ecf20Sopenharmony_ci			req->int_reg |= MEMSTICK_INT_ERR;
3828c2ecf20Sopenharmony_ci		if (int_reg & MS_INT_CED)
3838c2ecf20Sopenharmony_ci			req->int_reg |= MEMSTICK_INT_CED;
3848c2ecf20Sopenharmony_ci	}
3858c2ecf20Sopenharmony_ci
3868c2ecf20Sopenharmony_ci	return 0;
3878c2ecf20Sopenharmony_ci}
3888c2ecf20Sopenharmony_ci
3898c2ecf20Sopenharmony_cistatic void rtsx_pci_ms_handle_req(struct work_struct *work)
3908c2ecf20Sopenharmony_ci{
3918c2ecf20Sopenharmony_ci	struct realtek_pci_ms *host = container_of(work,
3928c2ecf20Sopenharmony_ci			struct realtek_pci_ms, handle_req);
3938c2ecf20Sopenharmony_ci	struct rtsx_pcr *pcr = host->pcr;
3948c2ecf20Sopenharmony_ci	struct memstick_host *msh = host->msh;
3958c2ecf20Sopenharmony_ci	int rc;
3968c2ecf20Sopenharmony_ci
3978c2ecf20Sopenharmony_ci	mutex_lock(&pcr->pcr_mutex);
3988c2ecf20Sopenharmony_ci
3998c2ecf20Sopenharmony_ci	rtsx_pci_start_run(pcr);
4008c2ecf20Sopenharmony_ci
4018c2ecf20Sopenharmony_ci	rtsx_pci_switch_clock(host->pcr, host->clock, host->ssc_depth,
4028c2ecf20Sopenharmony_ci			false, true, false);
4038c2ecf20Sopenharmony_ci	rtsx_pci_write_register(pcr, CARD_SELECT, 0x07, MS_MOD_SEL);
4048c2ecf20Sopenharmony_ci	rtsx_pci_write_register(pcr, CARD_SHARE_MODE,
4058c2ecf20Sopenharmony_ci			CARD_SHARE_MASK, CARD_SHARE_48_MS);
4068c2ecf20Sopenharmony_ci
4078c2ecf20Sopenharmony_ci	if (!host->req) {
4088c2ecf20Sopenharmony_ci		do {
4098c2ecf20Sopenharmony_ci			rc = memstick_next_req(msh, &host->req);
4108c2ecf20Sopenharmony_ci			dev_dbg(ms_dev(host), "next req %d\n", rc);
4118c2ecf20Sopenharmony_ci
4128c2ecf20Sopenharmony_ci			if (!rc)
4138c2ecf20Sopenharmony_ci				host->req->error = rtsx_pci_ms_issue_cmd(host);
4148c2ecf20Sopenharmony_ci		} while (!rc);
4158c2ecf20Sopenharmony_ci	}
4168c2ecf20Sopenharmony_ci
4178c2ecf20Sopenharmony_ci	mutex_unlock(&pcr->pcr_mutex);
4188c2ecf20Sopenharmony_ci}
4198c2ecf20Sopenharmony_ci
4208c2ecf20Sopenharmony_cistatic void rtsx_pci_ms_request(struct memstick_host *msh)
4218c2ecf20Sopenharmony_ci{
4228c2ecf20Sopenharmony_ci	struct realtek_pci_ms *host = memstick_priv(msh);
4238c2ecf20Sopenharmony_ci
4248c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "--> %s\n", __func__);
4258c2ecf20Sopenharmony_ci
4268c2ecf20Sopenharmony_ci	if (rtsx_pci_card_exclusive_check(host->pcr, RTSX_MS_CARD))
4278c2ecf20Sopenharmony_ci		return;
4288c2ecf20Sopenharmony_ci
4298c2ecf20Sopenharmony_ci	schedule_work(&host->handle_req);
4308c2ecf20Sopenharmony_ci}
4318c2ecf20Sopenharmony_ci
4328c2ecf20Sopenharmony_cistatic int rtsx_pci_ms_set_param(struct memstick_host *msh,
4338c2ecf20Sopenharmony_ci		enum memstick_param param, int value)
4348c2ecf20Sopenharmony_ci{
4358c2ecf20Sopenharmony_ci	struct realtek_pci_ms *host = memstick_priv(msh);
4368c2ecf20Sopenharmony_ci	struct rtsx_pcr *pcr = host->pcr;
4378c2ecf20Sopenharmony_ci	unsigned int clock = 0;
4388c2ecf20Sopenharmony_ci	u8 ssc_depth = 0;
4398c2ecf20Sopenharmony_ci	int err;
4408c2ecf20Sopenharmony_ci
4418c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "%s: param = %d, value = %d\n",
4428c2ecf20Sopenharmony_ci			__func__, param, value);
4438c2ecf20Sopenharmony_ci
4448c2ecf20Sopenharmony_ci	err = rtsx_pci_card_exclusive_check(host->pcr, RTSX_MS_CARD);
4458c2ecf20Sopenharmony_ci	if (err)
4468c2ecf20Sopenharmony_ci		return err;
4478c2ecf20Sopenharmony_ci
4488c2ecf20Sopenharmony_ci	switch (param) {
4498c2ecf20Sopenharmony_ci	case MEMSTICK_POWER:
4508c2ecf20Sopenharmony_ci		if (value == MEMSTICK_POWER_ON)
4518c2ecf20Sopenharmony_ci			err = ms_power_on(host);
4528c2ecf20Sopenharmony_ci		else if (value == MEMSTICK_POWER_OFF)
4538c2ecf20Sopenharmony_ci			err = ms_power_off(host);
4548c2ecf20Sopenharmony_ci		else
4558c2ecf20Sopenharmony_ci			return -EINVAL;
4568c2ecf20Sopenharmony_ci		break;
4578c2ecf20Sopenharmony_ci
4588c2ecf20Sopenharmony_ci	case MEMSTICK_INTERFACE:
4598c2ecf20Sopenharmony_ci		if (value == MEMSTICK_SERIAL) {
4608c2ecf20Sopenharmony_ci			clock = 19000000;
4618c2ecf20Sopenharmony_ci			ssc_depth = RTSX_SSC_DEPTH_500K;
4628c2ecf20Sopenharmony_ci
4638c2ecf20Sopenharmony_ci			err = rtsx_pci_write_register(pcr, MS_CFG, 0x58,
4648c2ecf20Sopenharmony_ci					MS_BUS_WIDTH_1 | PUSH_TIME_DEFAULT);
4658c2ecf20Sopenharmony_ci			if (err < 0)
4668c2ecf20Sopenharmony_ci				return err;
4678c2ecf20Sopenharmony_ci		} else if (value == MEMSTICK_PAR4) {
4688c2ecf20Sopenharmony_ci			clock = 39000000;
4698c2ecf20Sopenharmony_ci			ssc_depth = RTSX_SSC_DEPTH_1M;
4708c2ecf20Sopenharmony_ci
4718c2ecf20Sopenharmony_ci			err = rtsx_pci_write_register(pcr, MS_CFG,
4728c2ecf20Sopenharmony_ci					0x58, MS_BUS_WIDTH_4 | PUSH_TIME_ODD);
4738c2ecf20Sopenharmony_ci			if (err < 0)
4748c2ecf20Sopenharmony_ci				return err;
4758c2ecf20Sopenharmony_ci		} else {
4768c2ecf20Sopenharmony_ci			return -EINVAL;
4778c2ecf20Sopenharmony_ci		}
4788c2ecf20Sopenharmony_ci
4798c2ecf20Sopenharmony_ci		err = rtsx_pci_switch_clock(pcr, clock,
4808c2ecf20Sopenharmony_ci				ssc_depth, false, true, false);
4818c2ecf20Sopenharmony_ci		if (err < 0)
4828c2ecf20Sopenharmony_ci			return err;
4838c2ecf20Sopenharmony_ci
4848c2ecf20Sopenharmony_ci		host->ssc_depth = ssc_depth;
4858c2ecf20Sopenharmony_ci		host->clock = clock;
4868c2ecf20Sopenharmony_ci		host->ifmode = value;
4878c2ecf20Sopenharmony_ci		break;
4888c2ecf20Sopenharmony_ci	}
4898c2ecf20Sopenharmony_ci
4908c2ecf20Sopenharmony_ci	return 0;
4918c2ecf20Sopenharmony_ci}
4928c2ecf20Sopenharmony_ci
4938c2ecf20Sopenharmony_ci#ifdef CONFIG_PM
4948c2ecf20Sopenharmony_ci
4958c2ecf20Sopenharmony_cistatic int rtsx_pci_ms_suspend(struct platform_device *pdev, pm_message_t state)
4968c2ecf20Sopenharmony_ci{
4978c2ecf20Sopenharmony_ci	struct realtek_pci_ms *host = platform_get_drvdata(pdev);
4988c2ecf20Sopenharmony_ci	struct memstick_host *msh = host->msh;
4998c2ecf20Sopenharmony_ci
5008c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "--> %s\n", __func__);
5018c2ecf20Sopenharmony_ci
5028c2ecf20Sopenharmony_ci	memstick_suspend_host(msh);
5038c2ecf20Sopenharmony_ci	return 0;
5048c2ecf20Sopenharmony_ci}
5058c2ecf20Sopenharmony_ci
5068c2ecf20Sopenharmony_cistatic int rtsx_pci_ms_resume(struct platform_device *pdev)
5078c2ecf20Sopenharmony_ci{
5088c2ecf20Sopenharmony_ci	struct realtek_pci_ms *host = platform_get_drvdata(pdev);
5098c2ecf20Sopenharmony_ci	struct memstick_host *msh = host->msh;
5108c2ecf20Sopenharmony_ci
5118c2ecf20Sopenharmony_ci	dev_dbg(ms_dev(host), "--> %s\n", __func__);
5128c2ecf20Sopenharmony_ci
5138c2ecf20Sopenharmony_ci	memstick_resume_host(msh);
5148c2ecf20Sopenharmony_ci	return 0;
5158c2ecf20Sopenharmony_ci}
5168c2ecf20Sopenharmony_ci
5178c2ecf20Sopenharmony_ci#else /* CONFIG_PM */
5188c2ecf20Sopenharmony_ci
5198c2ecf20Sopenharmony_ci#define rtsx_pci_ms_suspend NULL
5208c2ecf20Sopenharmony_ci#define rtsx_pci_ms_resume NULL
5218c2ecf20Sopenharmony_ci
5228c2ecf20Sopenharmony_ci#endif /* CONFIG_PM */
5238c2ecf20Sopenharmony_ci
5248c2ecf20Sopenharmony_cistatic void rtsx_pci_ms_card_event(struct platform_device *pdev)
5258c2ecf20Sopenharmony_ci{
5268c2ecf20Sopenharmony_ci	struct realtek_pci_ms *host = platform_get_drvdata(pdev);
5278c2ecf20Sopenharmony_ci
5288c2ecf20Sopenharmony_ci	memstick_detect_change(host->msh);
5298c2ecf20Sopenharmony_ci}
5308c2ecf20Sopenharmony_ci
5318c2ecf20Sopenharmony_cistatic int rtsx_pci_ms_drv_probe(struct platform_device *pdev)
5328c2ecf20Sopenharmony_ci{
5338c2ecf20Sopenharmony_ci	struct memstick_host *msh;
5348c2ecf20Sopenharmony_ci	struct realtek_pci_ms *host;
5358c2ecf20Sopenharmony_ci	struct rtsx_pcr *pcr;
5368c2ecf20Sopenharmony_ci	struct pcr_handle *handle = pdev->dev.platform_data;
5378c2ecf20Sopenharmony_ci	int rc;
5388c2ecf20Sopenharmony_ci
5398c2ecf20Sopenharmony_ci	if (!handle)
5408c2ecf20Sopenharmony_ci		return -ENXIO;
5418c2ecf20Sopenharmony_ci
5428c2ecf20Sopenharmony_ci	pcr = handle->pcr;
5438c2ecf20Sopenharmony_ci	if (!pcr)
5448c2ecf20Sopenharmony_ci		return -ENXIO;
5458c2ecf20Sopenharmony_ci
5468c2ecf20Sopenharmony_ci	dev_dbg(&(pdev->dev),
5478c2ecf20Sopenharmony_ci			": Realtek PCI-E Memstick controller found\n");
5488c2ecf20Sopenharmony_ci
5498c2ecf20Sopenharmony_ci	msh = memstick_alloc_host(sizeof(*host), &pdev->dev);
5508c2ecf20Sopenharmony_ci	if (!msh)
5518c2ecf20Sopenharmony_ci		return -ENOMEM;
5528c2ecf20Sopenharmony_ci
5538c2ecf20Sopenharmony_ci	host = memstick_priv(msh);
5548c2ecf20Sopenharmony_ci	host->pcr = pcr;
5558c2ecf20Sopenharmony_ci	host->msh = msh;
5568c2ecf20Sopenharmony_ci	host->pdev = pdev;
5578c2ecf20Sopenharmony_ci	platform_set_drvdata(pdev, host);
5588c2ecf20Sopenharmony_ci	pcr->slots[RTSX_MS_CARD].p_dev = pdev;
5598c2ecf20Sopenharmony_ci	pcr->slots[RTSX_MS_CARD].card_event = rtsx_pci_ms_card_event;
5608c2ecf20Sopenharmony_ci
5618c2ecf20Sopenharmony_ci	mutex_init(&host->host_mutex);
5628c2ecf20Sopenharmony_ci
5638c2ecf20Sopenharmony_ci	INIT_WORK(&host->handle_req, rtsx_pci_ms_handle_req);
5648c2ecf20Sopenharmony_ci	msh->request = rtsx_pci_ms_request;
5658c2ecf20Sopenharmony_ci	msh->set_param = rtsx_pci_ms_set_param;
5668c2ecf20Sopenharmony_ci	msh->caps = MEMSTICK_CAP_PAR4;
5678c2ecf20Sopenharmony_ci
5688c2ecf20Sopenharmony_ci	rc = memstick_add_host(msh);
5698c2ecf20Sopenharmony_ci	if (rc) {
5708c2ecf20Sopenharmony_ci		memstick_free_host(msh);
5718c2ecf20Sopenharmony_ci		return rc;
5728c2ecf20Sopenharmony_ci	}
5738c2ecf20Sopenharmony_ci
5748c2ecf20Sopenharmony_ci	return 0;
5758c2ecf20Sopenharmony_ci}
5768c2ecf20Sopenharmony_ci
5778c2ecf20Sopenharmony_cistatic int rtsx_pci_ms_drv_remove(struct platform_device *pdev)
5788c2ecf20Sopenharmony_ci{
5798c2ecf20Sopenharmony_ci	struct realtek_pci_ms *host = platform_get_drvdata(pdev);
5808c2ecf20Sopenharmony_ci	struct rtsx_pcr *pcr;
5818c2ecf20Sopenharmony_ci	struct memstick_host *msh;
5828c2ecf20Sopenharmony_ci	int rc;
5838c2ecf20Sopenharmony_ci
5848c2ecf20Sopenharmony_ci	if (!host)
5858c2ecf20Sopenharmony_ci		return 0;
5868c2ecf20Sopenharmony_ci
5878c2ecf20Sopenharmony_ci	pcr = host->pcr;
5888c2ecf20Sopenharmony_ci	pcr->slots[RTSX_MS_CARD].p_dev = NULL;
5898c2ecf20Sopenharmony_ci	pcr->slots[RTSX_MS_CARD].card_event = NULL;
5908c2ecf20Sopenharmony_ci	msh = host->msh;
5918c2ecf20Sopenharmony_ci	host->eject = true;
5928c2ecf20Sopenharmony_ci	cancel_work_sync(&host->handle_req);
5938c2ecf20Sopenharmony_ci
5948c2ecf20Sopenharmony_ci	mutex_lock(&host->host_mutex);
5958c2ecf20Sopenharmony_ci	if (host->req) {
5968c2ecf20Sopenharmony_ci		dev_dbg(&(pdev->dev),
5978c2ecf20Sopenharmony_ci			"%s: Controller removed during transfer\n",
5988c2ecf20Sopenharmony_ci			dev_name(&msh->dev));
5998c2ecf20Sopenharmony_ci
6008c2ecf20Sopenharmony_ci		rtsx_pci_complete_unfinished_transfer(pcr);
6018c2ecf20Sopenharmony_ci
6028c2ecf20Sopenharmony_ci		host->req->error = -ENOMEDIUM;
6038c2ecf20Sopenharmony_ci		do {
6048c2ecf20Sopenharmony_ci			rc = memstick_next_req(msh, &host->req);
6058c2ecf20Sopenharmony_ci			if (!rc)
6068c2ecf20Sopenharmony_ci				host->req->error = -ENOMEDIUM;
6078c2ecf20Sopenharmony_ci		} while (!rc);
6088c2ecf20Sopenharmony_ci	}
6098c2ecf20Sopenharmony_ci	mutex_unlock(&host->host_mutex);
6108c2ecf20Sopenharmony_ci
6118c2ecf20Sopenharmony_ci	memstick_remove_host(msh);
6128c2ecf20Sopenharmony_ci	memstick_free_host(msh);
6138c2ecf20Sopenharmony_ci
6148c2ecf20Sopenharmony_ci	dev_dbg(&(pdev->dev),
6158c2ecf20Sopenharmony_ci		": Realtek PCI-E Memstick controller has been removed\n");
6168c2ecf20Sopenharmony_ci
6178c2ecf20Sopenharmony_ci	return 0;
6188c2ecf20Sopenharmony_ci}
6198c2ecf20Sopenharmony_ci
6208c2ecf20Sopenharmony_cistatic struct platform_device_id rtsx_pci_ms_ids[] = {
6218c2ecf20Sopenharmony_ci	{
6228c2ecf20Sopenharmony_ci		.name = DRV_NAME_RTSX_PCI_MS,
6238c2ecf20Sopenharmony_ci	}, {
6248c2ecf20Sopenharmony_ci		/* sentinel */
6258c2ecf20Sopenharmony_ci	}
6268c2ecf20Sopenharmony_ci};
6278c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(platform, rtsx_pci_ms_ids);
6288c2ecf20Sopenharmony_ci
6298c2ecf20Sopenharmony_cistatic struct platform_driver rtsx_pci_ms_driver = {
6308c2ecf20Sopenharmony_ci	.probe		= rtsx_pci_ms_drv_probe,
6318c2ecf20Sopenharmony_ci	.remove		= rtsx_pci_ms_drv_remove,
6328c2ecf20Sopenharmony_ci	.id_table       = rtsx_pci_ms_ids,
6338c2ecf20Sopenharmony_ci	.suspend	= rtsx_pci_ms_suspend,
6348c2ecf20Sopenharmony_ci	.resume		= rtsx_pci_ms_resume,
6358c2ecf20Sopenharmony_ci	.driver		= {
6368c2ecf20Sopenharmony_ci		.name	= DRV_NAME_RTSX_PCI_MS,
6378c2ecf20Sopenharmony_ci	},
6388c2ecf20Sopenharmony_ci};
6398c2ecf20Sopenharmony_cimodule_platform_driver(rtsx_pci_ms_driver);
6408c2ecf20Sopenharmony_ci
6418c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL");
6428c2ecf20Sopenharmony_ciMODULE_AUTHOR("Wei WANG <wei_wang@realsil.com.cn>");
6438c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Realtek PCI-E Memstick Card Host Driver");
644