258 lines
8.6 KiB
C
258 lines
8.6 KiB
C
/*
|
||
* Copyright : (C) 2022 Phytium Information Technology, Inc.
|
||
* All Rights Reserved.
|
||
*
|
||
* This program is OPEN SOURCE software: you can redistribute it and/or modify it
|
||
* under the terms of the Phytium Public License as published by the Phytium Technology Co.,Ltd,
|
||
* either version 1.0 of the License, or (at your option) any later version.
|
||
*
|
||
* This program is distributed in the hope that it will be useful,but WITHOUT ANY WARRANTY;
|
||
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||
* See the Phytium Public License for more details.
|
||
*
|
||
*
|
||
* FilePath: fsdio_cmd.c
|
||
* Date: 2022-06-01 14:23:59
|
||
* LastEditTime: 2022-06-01 14:24:00
|
||
* Description: This file is for SDIO command related function
|
||
*
|
||
* Modify History:
|
||
* Ver Who Date Changes
|
||
* ----- ------ -------- --------------------------------------
|
||
* 1.1 zhugengyu 2022/6/6 modify according to tech manual.
|
||
*/
|
||
/***************************** Include Files *********************************/
|
||
|
||
#include "fio.h"
|
||
#include "fdebug.h"
|
||
#include "fassert.h"
|
||
#include "ftypes.h"
|
||
|
||
#include "fcache.h"
|
||
#include "fswap.h"
|
||
|
||
#include "fsdio_hw.h"
|
||
#include "fsdio.h"
|
||
|
||
/************************** Constant Definitions *****************************/
|
||
|
||
/**************************** Type Definitions *******************************/
|
||
#define FSDIO_EXT_APP_CMD 55U
|
||
|
||
/***************** Macros (Inline Functions) Definitions *********************/
|
||
#define FSDIO_DEBUG_TAG "FSDIO-CMD"
|
||
#define FSDIO_ERROR(format, ...) FT_DEBUG_PRINT_E(FSDIO_DEBUG_TAG, format, ##__VA_ARGS__)
|
||
#define FSDIO_WARN(format, ...) FT_DEBUG_PRINT_W(FSDIO_DEBUG_TAG, format, ##__VA_ARGS__)
|
||
#define FSDIO_INFO(format, ...) FT_DEBUG_PRINT_I(FSDIO_DEBUG_TAG, format, ##__VA_ARGS__)
|
||
#define FSDIO_DEBUG(format, ...) FT_DEBUG_PRINT_D(FSDIO_DEBUG_TAG, format, ##__VA_ARGS__)
|
||
|
||
/************************** Function Prototypes ******************************/
|
||
extern FError FSdioPIOReadData(FSdio *const instance_p, FSdioData *data_p);
|
||
|
||
/*****************************************************************************/
|
||
FError FSdioSendPrivateCmd(uintptr base_addr, u32 cmd, u32 arg)
|
||
{
|
||
u32 reg_val;
|
||
int retries = FSDIO_TIMEOUT;
|
||
|
||
do
|
||
{
|
||
reg_val = FSDIO_READ_REG(base_addr, FSDIO_STATUS_OFFSET);
|
||
if (--retries <= 0)
|
||
{
|
||
break;
|
||
}
|
||
}
|
||
while (FSDIO_STATUS_DATA_BUSY & reg_val);
|
||
|
||
if (retries <= 0)
|
||
{
|
||
return FSDIO_ERR_BUSY;
|
||
}
|
||
|
||
FSDIO_WRITE_REG(base_addr, FSDIO_CMD_ARG_OFFSET, arg);
|
||
|
||
FSDIO_DATA_BARRIER(); /* drain writebuffer */
|
||
|
||
FSDIO_WRITE_REG(base_addr, FSDIO_CMD_OFFSET, FSDIO_CMD_START | cmd);
|
||
|
||
retries = FSDIO_TIMEOUT;
|
||
do
|
||
{
|
||
reg_val = FSDIO_READ_REG(base_addr, FSDIO_CMD_OFFSET);
|
||
if (--retries <= 0)
|
||
{
|
||
break;
|
||
}
|
||
}
|
||
while (FSDIO_CMD_START & reg_val); /* wait until command send done */
|
||
|
||
return (retries <= 0) ? FSDIO_ERR_TIMEOUT : FSDIO_SUCCESS;
|
||
}
|
||
|
||
/**
|
||
* @name: FSdioTransferCmd
|
||
* @msg: pack and transfer command
|
||
* @return {FError} FSDIO_SUCCESS if transfer success
|
||
* @param {FSdio} *instance_p, SDIO controller instance
|
||
* @param {FSdioCmdData} *cmd_data_p, contents of transfer command and data
|
||
*/
|
||
FError FSdioTransferCmd(FSdio *const instance_p, FSdioCmdData *const cmd_data_p)
|
||
{
|
||
FASSERT(cmd_data_p);
|
||
FError ret = FSDIO_SUCCESS;
|
||
uintptr base_addr = instance_p->config.base_addr;
|
||
u32 cmd_flag = cmd_data_p->flag;
|
||
u32 raw_cmd = FSDIO_CMD_USE_HOLD_REG; /* USE_HOLD_REG必须为1 */
|
||
|
||
/* 命令需要进行卡初始化,如CMD-0 */
|
||
if (FSDIO_CMD_FLAG_NEED_INIT & cmd_flag)
|
||
{
|
||
raw_cmd |= FSDIO_CMD_INIT;
|
||
}
|
||
|
||
/* 命令涉及电压切换 */
|
||
if (FSDIO_CMD_FLAG_SWITCH_VOLTAGE & cmd_flag)
|
||
{
|
||
raw_cmd |= FSDIO_CMD_VOLT_SWITCH;
|
||
}
|
||
|
||
/* 命令传输过程伴随数据传输 */
|
||
if (FSDIO_CMD_FLAG_EXP_DATA & cmd_flag)
|
||
{
|
||
raw_cmd |= FSDIO_CMD_DAT_EXP;
|
||
|
||
if (FSDIO_CMD_FLAG_WRITE_DATA & cmd_flag) /* 写卡 */
|
||
{
|
||
raw_cmd |= FSDIO_CMD_DAT_WRITE;
|
||
}
|
||
}
|
||
|
||
/* 命令需要进行CRC校验 */
|
||
if (FSDIO_CMD_FLAG_NEED_RESP_CRC & cmd_flag)
|
||
{
|
||
raw_cmd |= FSDIO_CMD_RESP_CRC;
|
||
}
|
||
|
||
/* 命令需要响应回复 */
|
||
if (FSDIO_CMD_FLAG_EXP_RESP & cmd_flag)
|
||
{
|
||
raw_cmd |= FSDIO_CMD_RESP_EXP;
|
||
|
||
if (FSDIO_CMD_FLAG_EXP_LONG_RESP & cmd_flag) /* 命令需要136字节长回复 */
|
||
{
|
||
raw_cmd |= FSDIO_CMD_RESP_LONG;
|
||
}
|
||
}
|
||
|
||
raw_cmd |= FSDIO_CMD_INDX_SET(cmd_data_p->cmdidx);
|
||
|
||
FSDIO_DEBUG("============[%s-%d]@0x%x begin ============",
|
||
(FSDIO_EXT_APP_CMD == instance_p->prev_cmd) ? "ACMD" : "CMD", cmd_data_p->cmdidx,
|
||
base_addr);
|
||
FSDIO_DEBUG(" cmd: 0x%x", raw_cmd);
|
||
FSDIO_DEBUG(" arg: 0x%x", cmd_data_p->cmdarg);
|
||
|
||
/* enable related interrupt */
|
||
FSdioSetInterruptMask(instance_p, FSDIO_GENERAL_INTR,
|
||
FSDIO_INTS_CMD_MASK, TRUE);
|
||
|
||
ret = FSdioSendPrivateCmd(base_addr, raw_cmd, cmd_data_p->cmdarg);
|
||
FSDIO_INFO("cmd send done ...");
|
||
return ret;
|
||
}
|
||
|
||
static void FSdioFlipByteOrder(u32 *response, fsize_t size)
|
||
{
|
||
/*
|
||
swap response and convert byte order
|
||
resp[0] = bswap32(resp[3])
|
||
resp[1] = bswap32(resp[2])
|
||
resp[2] = bswap32(resp[1])
|
||
resp[3] = bswap32(resp[0])
|
||
*/
|
||
FASSERT(size % (2 * sizeof(u32)) == 0);
|
||
const fsize_t n_words = size / sizeof(uint32_t);
|
||
for (int i = 0; i < (int)n_words / 2; ++i)
|
||
{
|
||
u32 left = __builtin_bswap32(response[i]);
|
||
u32 right = __builtin_bswap32(response[n_words - i - 1]);
|
||
response[i] = right;
|
||
response[n_words - i - 1] = left;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @name: FSdioGetCmdResponse
|
||
* @msg: Get cmd response and received data after wait poll status or interrupt signal
|
||
* @return {FError} FSDIO_SUCCESS if get success
|
||
* @param {FSdio} *instance_p, SDIO controller instance
|
||
* @param {FSdioCmdData} *cmd_data_p, contents of transfer command and data
|
||
*/
|
||
FError FSdioGetCmdResponse(FSdio *const instance_p, FSdioCmdData *const cmd_data_p)
|
||
{
|
||
FASSERT(instance_p);
|
||
FASSERT(cmd_data_p);
|
||
FError ret = FSDIO_SUCCESS;
|
||
const boolean read = cmd_data_p->flag & FSDIO_CMD_FLAG_READ_DATA;
|
||
uintptr base_addr = instance_p->config.base_addr;
|
||
|
||
if (FT_COMPONENT_IS_READY != instance_p->is_ready)
|
||
{
|
||
FSDIO_ERROR("Device is not yet initialized!!!");
|
||
return FSDIO_ERR_NOT_INIT;
|
||
}
|
||
|
||
if ((NULL != cmd_data_p->data_p) && (read))
|
||
{
|
||
if (FSDIO_PIO_TRANS_MODE == instance_p->config.trans_mode)
|
||
{
|
||
ret = FSdioPIOReadData(instance_p, cmd_data_p->data_p);
|
||
}
|
||
}
|
||
|
||
/* check response of cmd */
|
||
if (FSDIO_CMD_FLAG_EXP_RESP & cmd_data_p->flag)
|
||
{
|
||
if (FSDIO_CMD_FLAG_EXP_LONG_RESP & cmd_data_p->flag)
|
||
{
|
||
cmd_data_p->response[0] = FSDIO_READ_REG(base_addr, FSDIO_RESP0_OFFSET);
|
||
cmd_data_p->response[1] = FSDIO_READ_REG(base_addr, FSDIO_RESP1_OFFSET);
|
||
cmd_data_p->response[2] = FSDIO_READ_REG(base_addr, FSDIO_RESP2_OFFSET);
|
||
cmd_data_p->response[3] = FSDIO_READ_REG(base_addr, FSDIO_RESP3_OFFSET);
|
||
|
||
/* according to SD spec. 136 bits R2 is send-back in reverse order and big-end,
|
||
some SD protocol will do this reverse and ntol itself, other do not,
|
||
filp_resp_byte_order is to compilant with those not */
|
||
if (instance_p->config.filp_resp_byte_order)
|
||
{
|
||
FSdioFlipByteOrder(cmd_data_p->response, sizeof(u32) * 4);
|
||
}
|
||
|
||
FSDIO_DEBUG(" resp: 0x%x-0x%x-0x%x-0x%x",
|
||
cmd_data_p->response[0], cmd_data_p->response[1],
|
||
cmd_data_p->response[2], cmd_data_p->response[3]);
|
||
}
|
||
else
|
||
{
|
||
cmd_data_p->response[0] = FSDIO_READ_REG(base_addr, FSDIO_RESP0_OFFSET);
|
||
cmd_data_p->response[1] = 0U;
|
||
cmd_data_p->response[2] = 0U;
|
||
cmd_data_p->response[3] = 0U;
|
||
FSDIO_DEBUG(" resp: 0x%x", cmd_data_p->response[0]);
|
||
}
|
||
}
|
||
|
||
cmd_data_p->success = TRUE; /* cmd / data transfer finished successful */
|
||
FSDIO_DEBUG("============[%s-%d]@0x%x end ============",
|
||
(FSDIO_EXT_APP_CMD == instance_p->prev_cmd) ? "ACMD" : "CMD",
|
||
cmd_data_p->cmdidx, base_addr);
|
||
|
||
/* disable related interrupt */
|
||
FSdioSetInterruptMask(instance_p, FSDIO_GENERAL_INTR, FSDIO_INTS_CMD_MASK | FSDIO_INTS_DATA_MASK, FALSE);
|
||
FSdioSetInterruptMask(instance_p, FSDIO_IDMA_INTR, FSDIO_DMAC_INTS_MASK, FALSE);
|
||
|
||
instance_p->prev_cmd = cmd_data_p->cmdidx; /* record previous command */
|
||
|
||
return ret;
|
||
} |