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