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