guo ecf2d82159
sync branch rt-smart. (#6641)
* Synchronize the code of the rt mart branch to the master branch.
  * TTY device
  * Add lwP code from rt-smart
  * Add vnode in DFS, but DFS will be re-write for rt-smart
  * There are three libcpu for rt-smart:
    * arm/cortex-a, arm/aarch64
    * riscv64

Co-authored-by: Rbb666 <zhangbingru@rt-thread.com>
Co-authored-by: zhkag <zhkag@foxmail.com>
2022-12-03 12:07:44 +08:00

1132 lines
31 KiB
C

/*
* Copyright (C) 2017 ALLWINNERTECH TECHNOLOGY CO., LTD. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the
* distribution.
* 3. Neither the name of ALLWINNERTECH TECHNOLOGY CO., LTD. nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "sunxi_hal_common.h"
#include "hal_base.h"
#include "sys/endian.h"
#include "hal_sdhost.h"
#include "sdmmc.h"
#include "_sdhost.h"
#include "_core.h"
#ifdef CONFIG_USE_MMC
#include "_mmc.h"
#endif
#include "_sd.h"
#include "_sd_define.h"
#ifdef CONFIG_USE_SD
static const unsigned int tran_exp[] = { /* about KB/S */
10000, 100000, 1000000, 10000000,
0, 0, 0, 0
};
static const unsigned char tran_mant[] = { /* time value*10 */
0, 10, 12, 13, 15, 20, 25, 30,
35, 40, 45, 50, 55, 60, 70, 80,
};
/*
static const unsigned int tacc_exp[] = {
1, 10, 100, 1000, 10000, 100000, 1000000, 10000000,
};
static const unsigned int tacc_mant[] = {
0, 10, 12, 13, 15, 20, 25, 30,
35, 40, 45, 50, 55, 60, 70, 80,
};
*/
static int32_t mmc_send_app_op_cond(struct mmc_host *host, uint32_t ocr, uint32_t *rocr)
{
struct mmc_command cmd = {0};
int32_t i, err;
if (!host) {
SDC_LOGE_RAW(ROM_ERR_MASK, "%s,%d err", __func__, __LINE__);
return -1;
}
cmd.opcode = SD_APP_OP_COND;
cmd.arg = ocr;
cmd.flags = MMC_RSP_SPI_R1 | MMC_RSP_R3 | MMC_CMD_BCR;
for (i = 100; i; i--) {
err = mmc_wait_for_app_cmd(host, NULL, &cmd);
if (err) {
// printf("%s,%d\n",__FUNCTION__,__LINE__);
break;
}
/* otherwise wait until reset completes */
if (cmd.resp[0] & MMC_CARD_BUSY){
// printf("%s,%d\n",__FUNCTION__,__LINE__);
break;
}
if (host->ocr_avail & MMC_VDD_165_195)
cmd.arg = 0x41000000 | (cmd.resp[0] & 0xFF8000);
else
cmd.arg = 0x40000000 | (cmd.resp[0] & 0xFF8000);
err = -1;
mmc_mdelay(20);
// printf("%s,%d %d\n",__FUNCTION__,__LINE__,i);
}
if (rocr)
*rocr = cmd.resp[0];
if(err)
printf("%s,%d %ld\n",__FUNCTION__,__LINE__, HAL_PR_SZ_L(i));
return err;
}
int32_t mmc_app_sd_status(struct mmc_card *card, uint8_t *ssr)
{
struct mmc_request mrq;
struct mmc_command cmd = {0};
struct mmc_data data = {0};
struct scatterlist sg;
if (!ssr) {
SD_LOGE("%s,%d err", __func__, __LINE__);
return -1;
}
if (mmc_app_cmd(card->host, card)) {
return -1;
}
mrq.cmd = &cmd;
mrq.data = &data;
cmd.opcode = SD_APP_SD_STATUS;
cmd.arg = 0;
cmd.flags = MMC_RSP_SPI_R2 | MMC_RSP_R1 | MMC_CMD_ADTC;
data.blksz = 64;
data.blocks = 1;
data.flags = MMC_DATA_READ;
data.sg = &sg;
data.sg_len = 1;
sg.len = 64;
sg.buffer = ssr;
if (mmc_wait_for_req(card->host, &mrq)) {
return -1;
}
SD_LOGN("card raw SD status:\n");
sd_hex_dump_bytes((void *)ssr, 64);
return 0;
}
int32_t mmc_app_send_scr(struct mmc_card *card, uint32_t *raw_scr)
{
struct mmc_command cmd = {0};
struct mmc_data data = {0};
struct mmc_request mrq;
struct scatterlist sg;
if (mmc_app_cmd(card->host, card)) {
return -1;
}
mrq.cmd = &cmd;
mrq.data = &data;
cmd.opcode = SD_APP_SEND_SCR;
cmd.arg = 0;
cmd.flags = MMC_RSP_SPI_R1 | MMC_RSP_R1 | MMC_CMD_ADTC;
data.blksz = 8;
data.blocks = 1;
data.flags = MMC_DATA_READ;
data.sg = &sg;
data.sg_len = 1;
sg.len = 8;
sg.buffer = (void *)raw_scr;
/* get scr, 8 bytes */
if (mmc_wait_for_req(card->host, &mrq)) {
return -1;
}
raw_scr[0] = be32_to_cpu(raw_scr[0]);
raw_scr[1] = be32_to_cpu(raw_scr[1]);
return 0;
}
#ifdef SD_SUPPORT_VERSION3
int32_t mmc_switch_to_1v8(struct mmc_card *card)
{
struct mmc_command cmd = {0};
cmd.opcode = SD_SWITCH_VOLTAGE;
cmd.arg = 0;
cmd.vol_switch = 1;
cmd.flags = MMC_RSP_R1 | MMC_CMD_AC;
if (mmc_wait_for_cmd(card->host, &cmd)) {
SD_LOGW("sd switch 1v8 failed\n");
return -1;
}
return 0;
}
#endif
/*
* Given the decoded CSD structure, decode the raw CID to our CID structure.
*/
void mmc_decode_cid(struct mmc_card *card, uint32_t *resp)
{
SDC_Memset(&card->cid, 0, sizeof(struct mmc_cid));
/*
* SD doesn't currently have a version field so we will
* have to assume we can parse this.
*/
card->cid.manfid = UNSTUFF_BITS(resp, 120, 8);
card->cid.oemid = UNSTUFF_BITS(resp, 104, 8);
card->cid.prod_name[0] = UNSTUFF_BITS(resp, 96, 8);
card->cid.prod_name[1] = UNSTUFF_BITS(resp, 88, 8);
card->cid.prod_name[2] = UNSTUFF_BITS(resp, 80, 8);
card->cid.prod_name[3] = UNSTUFF_BITS(resp, 72, 8);
card->cid.prod_name[4] = UNSTUFF_BITS(resp, 64, 8);
card->cid.hwrev = UNSTUFF_BITS(resp, 60, 4);
card->cid.fwrev = UNSTUFF_BITS(resp, 56, 4);
card->cid.serial = UNSTUFF_BITS(resp, 24, 32);
card->cid.year = UNSTUFF_BITS(resp, 12, 8);
card->cid.month = UNSTUFF_BITS(resp, 8, 4);
card->cid.year += 2000; /* SD cards year offset */
}
int32_t mmc_send_cid(struct mmc_card *card)
{
struct mmc_command cmd = {0};
uint32_t cid[4] = {0};
cmd.opcode = MMC_SEND_CID;
cmd.arg = card->rca << 16;
cmd.flags = MMC_RSP_R2 | MMC_CMD_AC;
if (mmc_wait_for_cmd(card->host, &cmd)) {
return -1;
}
HAL_Memcpy((void *)cid, (void *)cmd.resp, 16);
SD_LOGN("card raw cid:\n");
sd_hex_dump_bytes((void *)cid, 16);
mmc_decode_cid(card, cid);
return 0;
}
/*
* Given a 128-bit response, decode to our card CSD structure.
*/
static int32_t mmc_decode_csd(struct mmc_card *card, uint32_t *raw_csd)
{
int32_t e, m, csd_struct;
uint32_t *resp = raw_csd;
struct mmc_csd *csd = &card->csd;
csd_struct = UNSTUFF_BITS(resp, 126, 2);
card->csd.csd_ver = csd_struct;
switch (csd_struct) {
case 0:
//m = UNSTUFF_BITS(resp, 115, 4);
//e = UNSTUFF_BITS(resp, 112, 3);
//csd->tacc_ns = (tacc_exp[e] * tacc_mant[m] + 9) / 10;
//csd->tacc_clks = UNSTUFF_BITS(resp, 104, 8) * 100;
m = UNSTUFF_BITS(resp, 99, 4);
e = UNSTUFF_BITS(resp, 96, 3);
csd->max_dtr = tran_exp[e] * tran_mant[m];
csd->cmdclass = UNSTUFF_BITS(resp, 84, 12);
m = UNSTUFF_BITS(resp, 80, 4);
csd->read_blk_len = 1 << m;
e = UNSTUFF_BITS(resp, 47, 3);
m = UNSTUFF_BITS(resp, 62, 12);
csd->capacity = (1 + m) * (1 << (e + 2)) * csd->read_blk_len;
csd->capacity >>= 10; /* unit:KB */
//csd->read_blkbits = UNSTUFF_BITS(resp, 80, 4);
//csd->read_partial = UNSTUFF_BITS(resp, 79, 1);
//csd->write_misalign = UNSTUFF_BITS(resp, 78, 1);
//csd->read_misalign = UNSTUFF_BITS(resp, 77, 1);
//csd->r2w_factor = UNSTUFF_BITS(resp, 26, 3);
//csd->write_blkbits = UNSTUFF_BITS(resp, 22, 4);
//csd->write_partial = UNSTUFF_BITS(resp, 21, 1);
//if (UNSTUFF_BITS(resp, 46, 1)) {
// csd->erase_size = 1;
//} else if (csd->write_blkbits >= 9) {
// csd->erase_size = UNSTUFF_BITS(resp, 39, 7) + 1;
// csd->erase_size <<= csd->write_blkbits - 9;
//}
break;
case 1:
/*
* This is a block-addressed SDHC or SDXC card. Most
* interesting fields are unused and have fixed
* values. To avoid getting tripped by buggy cards,
* we assume those fixed values ourselves.
*/
mmc_card_set_blockaddr(card);
//csd->tacc_ns = 0; /* Unused */
//csd->tacc_clks = 0; /* Unused */
m = UNSTUFF_BITS(resp, 99, 4);
e = UNSTUFF_BITS(resp, 96, 3);
csd->max_dtr = tran_exp[e] * tran_mant[m];
csd->cmdclass = UNSTUFF_BITS(resp, 84, 12);
csd->read_blk_len = UNSTUFF_BITS(resp, 80, 4);
m = UNSTUFF_BITS(resp, 48, 22);
csd->capacity = (1 + m) << 9;
/* SDXC cards have a minimum C_SIZE of 0x00FFFF */
if (m >= 0xFFFF)
mmc_card_set_ext_capacity(card);
//csd->read_blkbits = 9;
//csd->read_partial = 0;
//csd->write_misalign = 0;
//csd->read_misalign = 0;
//csd->r2w_factor = 4; /* Unused */
//csd->write_blkbits = 9;
//csd->write_partial = 0;
//csd->erase_size = 1;
break;
default:
SD_LOGE("%s: unrecognised CSD structure version %d\n",
__func__, (unsigned int)csd_struct);
return -1;
}
//card->erase_size = csd->erase_size;
SD_LOGD("%s %d ca:%d\n", __func__, (unsigned int)csd_struct, (unsigned int)csd->capacity);
return 0;
}
#ifdef SD_SUPPORT_VERSION3
static void sd_update_bus_speed_mode(struct mmc_card *card)
{
/*
* If the host doesn't support any of the UHS-I modes, fallback on
* default speed.
*/
if (!(card->host->caps & (MMC_CAP_UHS_SDR12 | MMC_CAP_UHS_SDR25 |
MMC_CAP_UHS_SDR50 | MMC_CAP_UHS_SDR104 | MMC_CAP_UHS_DDR50))) {
card->sd_bus_speed = 0;
return;
}
if ((card->host->caps & MMC_CAP_UHS_SDR104) &&
(card->sw_caps.sd3_bus_mode & SD_MODE_UHS_SDR104)) {
card->sd_bus_speed = UHS_SDR104_BUS_SPEED;
} else if ((card->host->caps & MMC_CAP_UHS_DDR50) &&
(card->sw_caps.sd3_bus_mode & SD_MODE_UHS_DDR50)) {
card->sd_bus_speed = UHS_DDR50_BUS_SPEED;
} else if ((card->host->caps & (MMC_CAP_UHS_SDR104 |
MMC_CAP_UHS_SDR50)) && (card->sw_caps.sd3_bus_mode &
SD_MODE_UHS_SDR50)) {
card->sd_bus_speed = UHS_SDR50_BUS_SPEED;
} else if ((card->host->caps & (MMC_CAP_UHS_SDR104 |
MMC_CAP_UHS_SDR50 | MMC_CAP_UHS_SDR25)) &&
(card->sw_caps.sd3_bus_mode & SD_MODE_UHS_SDR25)) {
card->sd_bus_speed = UHS_SDR25_BUS_SPEED;
} else if ((card->host->caps & (MMC_CAP_UHS_SDR104 |
MMC_CAP_UHS_SDR50 | MMC_CAP_UHS_SDR25 |
MMC_CAP_UHS_SDR12)) && (card->sw_caps.sd3_bus_mode &
SD_MODE_UHS_SDR12)) {
card->sd_bus_speed = UHS_SDR12_BUS_SPEED;
}
}
static int32_t sd_set_bus_speed_mode(struct mmc_card *card, int8_t *status)
{
int32_t err;
uint32_t timing = 0;
switch (card->sd_bus_speed) {
case UHS_SDR104_BUS_SPEED:
timing = MMC_TIMING_UHS_SDR104;
card->sw_caps.uhs_max_dtr = UHS_SDR104_MAX_DTR;
break;
case UHS_DDR50_BUS_SPEED:
timing = MMC_TIMING_UHS_DDR50;
card->sw_caps.uhs_max_dtr = UHS_DDR50_MAX_DTR;
break;
case UHS_SDR50_BUS_SPEED:
timing = MMC_TIMING_UHS_SDR50;
card->sw_caps.uhs_max_dtr = UHS_SDR50_MAX_DTR;
break;
case UHS_SDR25_BUS_SPEED:
timing = MMC_TIMING_UHS_SDR25;
card->sw_caps.uhs_max_dtr = UHS_SDR25_MAX_DTR;
break;
case UHS_SDR12_BUS_SPEED:
timing = MMC_TIMING_UHS_SDR12;
card->sw_caps.uhs_max_dtr = UHS_SDR12_MAX_DTR;
break;
default:
return 0;
}
err = mmc_sd_switch(card, 1, 0, card->sd_bus_speed, status);
if (err)
return err;
if ((status[16] & 0xF) != card->sd_bus_speed)
SD_LOGW("%s: Problem setting bus speed mode!\n", __func__);
else {
mmc_set_timing(card->host, timing);
mmc_set_clock(card->host, card->sw_caps.uhs_max_dtr);
}
return 0;
}
static int32_t sd_set_current_limit(struct mmc_card *card, int8_t *status)
{
int32_t current_limit = 0;
int32_t err;
/*
* Current limit switch is only defined for SDR50, SDR104, and DDR50
* bus speed modes. For other bus speed modes, we set the default
* current limit of 200mA.
*/
if ((card->sd_bus_speed == UHS_SDR50_BUS_SPEED) ||
(card->sd_bus_speed == UHS_SDR104_BUS_SPEED) ||
(card->sd_bus_speed == UHS_DDR50_BUS_SPEED)) {
if (card->host->caps & MMC_CAP_MAX_CURRENT_800) {
if (card->sw_caps.sd3_curr_limit & SD_MAX_CURRENT_800)
current_limit = SD_SET_CURRENT_LIMIT_800;
else if (card->sw_caps.sd3_curr_limit &
SD_MAX_CURRENT_600)
current_limit = SD_SET_CURRENT_LIMIT_600;
else if (card->sw_caps.sd3_curr_limit &
SD_MAX_CURRENT_400)
current_limit = SD_SET_CURRENT_LIMIT_400;
else if (card->sw_caps.sd3_curr_limit &
SD_MAX_CURRENT_200)
current_limit = SD_SET_CURRENT_LIMIT_200;
} else if (card->host->caps & MMC_CAP_MAX_CURRENT_600) {
if (card->sw_caps.sd3_curr_limit & SD_MAX_CURRENT_600)
current_limit = SD_SET_CURRENT_LIMIT_600;
else if (card->sw_caps.sd3_curr_limit &
SD_MAX_CURRENT_400)
current_limit = SD_SET_CURRENT_LIMIT_400;
else if (card->sw_caps.sd3_curr_limit &
SD_MAX_CURRENT_200)
current_limit = SD_SET_CURRENT_LIMIT_200;
} else if (card->host->caps & MMC_CAP_MAX_CURRENT_400) {
if (card->sw_caps.sd3_curr_limit & SD_MAX_CURRENT_400)
current_limit = SD_SET_CURRENT_LIMIT_400;
else if (card->sw_caps.sd3_curr_limit &
SD_MAX_CURRENT_200)
current_limit = SD_SET_CURRENT_LIMIT_200;
} else if (card->host->caps & MMC_CAP_MAX_CURRENT_200) {
if (card->sw_caps.sd3_curr_limit & SD_MAX_CURRENT_200)
current_limit = SD_SET_CURRENT_LIMIT_200;
}
} else
current_limit = SD_SET_CURRENT_LIMIT_200;
err = mmc_sd_switch(card, 1, 3, current_limit, status);
if (err)
return err;
if (((status[15] >> 4) & 0x0F) != current_limit)
SD_LOGW("%s: Problem setting current limit!\n", __func__);
return 0;
}
int32_t mmc_sd_init_uhs_card(struct mmc_card *card)
{
int32_t err;
struct mmc_command cmd = {0};
struct mmc_data data = {0};
struct mmc_request mrq;
struct scatterlist sg = {0};
uint32_t time = 0;
uint8_t pattern[] = {
0xff, 0x0f, 0xff, 0x00, 0xff, 0xcc, 0xc3, 0xcc, 0xc3, 0x3c, 0xcc, 0xff, 0xfe, 0xff, 0xfe, 0xef,
0xff, 0xdf, 0xff, 0xdd, 0xff, 0xfb, 0xff, 0xfb, 0xbf, 0xff, 0x7f, 0xff, 0x77, 0xf7, 0xbd, 0xef,
0xff, 0xf0, 0xff, 0xf0, 0x0f, 0xfc, 0xcc, 0x3c, 0xcc, 0x33, 0xcc, 0xcf, 0xff, 0xef, 0xff, 0xee,
0xff, 0xfd, 0xff, 0xfd, 0xdf, 0xff, 0xbf, 0xff, 0xbb, 0xff, 0xf7, 0xff, 0xf7, 0x7f, 0x7b, 0xde
};
uint32_t status[64/4] = {0};
if (!card->scr.sda_spec3)
return 0;
if (!(card->csd.cmdclass & CCC_SWITCH))
return 0;
/* Set 4-bit bus width */
if ((card->host->caps & MMC_CAP_4_BIT_DATA) &&
(card->scr.bus_widths & SD_SCR_BUS_WIDTH_4)) {
err = mmc_app_set_bus_width(card, MMC_BUS_WIDTH_4);
if (err)
goto out;
HAL_SDC_Set_BusWidth(card->host, MMC_BUS_WIDTH_4);
}
/*
* Select the bus speed mode depending on host
* and card capability.
*/
sd_update_bus_speed_mode(card);
/* Set the driver strength for the card */
err = sd_select_driver_type(card, status);
if (err)
goto out;
/* Set current limit for the card */
err = sd_set_current_limit(card, status);
if (err)
goto out;
/* Set bus speed mode of the card */
err = sd_set_bus_speed_mode(card, status);
if (err)
goto out;
/* SPI mode doesn't define CMD19 */
mmc_host_clk_hold(card->host);
sg.len = 512;
sg.buffer = status;
cmd.opcode = MMC_SEND_TUNING_PATTERN;
cmd.arg = 0;
cmd.flags = MMC_RSP_SPI_R1 | MMC_RSP_R1 | MMC_CMD_ADTC;
cmd.data = &data;
data.blksz = 64;
data.sg_len = 1;
data.sg = &sg;
data.flags = MMC_DATA_READ;
mrq.cmd = &cmd;
mrq.data = &data;
do {
if (mmc_wait_for_req(card->host, &mrq)) {
continue;
}
time++;
} while (time < 40);
if (HAL_Memcmp((void *)pattern, (void *)status, 64))
return -1;
mmc_host_clk_release(card->host);
out:
return 0;
}
#endif
/*
* Given a 64-bit response, decode to our card SCR structure.
*/
static int32_t mmc_decode_scr(struct mmc_card *card, uint32_t *raw_scr)
{
struct sd_scr *scr = &card->scr;
uint32_t scr_struct;
uint32_t resp[4];
resp[3] = raw_scr[1];
resp[2] = raw_scr[0];
scr_struct = UNSTUFF_BITS(resp, 60, 4);
if (scr_struct != 0) {
SD_LOGW("sdc unrecognised SCR structure version %u\n", (unsigned int)scr_struct);
return -1;
}
scr->sda_vsn = UNSTUFF_BITS(resp, 56, 4);
scr->security_sup = UNSTUFF_BITS(resp, 52, 3);
scr->bus_widths = UNSTUFF_BITS(resp, 48, 4);
if (scr->sda_vsn == SCR_SPEC_VER_2)
/* Check if Physical Layer Spec v3.0 is supported */
scr->sda_spec3 = UNSTUFF_BITS(resp, 47, 1);
//if (UNSTUFF_BITS(resp, 55, 1))
// card->erased_byte = 0xFF;
//else
// card->erased_byte = 0x0;
if (scr->sda_spec3) {
scr->cmds = UNSTUFF_BITS(resp, 32, 2);
scr->sda_spec4 = UNSTUFF_BITS(resp, 42, 1);
scr->sda_spec5 = UNSTUFF_BITS(resp, 38, 4);
}
return 0;
}
/*
* Fetch and process SD Status register.
*/
static int32_t mmc_read_ssr(struct mmc_card *card)
{
int32_t err, i;
uint32_t ssr[64/4];
if (!(card->csd.cmdclass & CCC_APP_SPEC)) {
SD_LOGW("%s: card lacks mandatory SD Status function.\n", __func__);
return 0;
}
err = mmc_app_sd_status(card, (uint8_t *)ssr);
if (err) {
SD_LOGW("%s: problem reading SD Status register.\n", __func__);
err = 0;
goto out;
}
for (i = 0; i < 16; i++)
ssr[i] = be32_to_cpu(ssr[i]);
#ifdef SD_SUPPORT_ERASE
uint32_t au, es, et, eo;
/*
* UNSTUFF_BITS only works with four u32s so we have to offset the
* bitfield positions accordingly.
*/
au = UNSTUFF_BITS(ssr, 428 - 384, 4);
if (au > 0 && au <= 9) {
card->ssr.au = 1 << (au + 4);
es = UNSTUFF_BITS(ssr, 408 - 384, 16);
et = UNSTUFF_BITS(ssr, 402 - 384, 6);
eo = UNSTUFF_BITS(ssr, 400 - 384, 2);
if (es && et) {
card->ssr.erase_timeout = (et * 1000) / es;
card->ssr.erase_offset = eo * 1000;
}
} else {
SD_LOGW("%s: SD Status: Invalid Allocation Unit size.\n", __func__);
}
#endif
card->speed_class = UNSTUFF_BITS(ssr, 440 - 384, 8) * 2;
if (card->speed_class == 8)
card->speed_class = 10;
out:
return err;
}
/*
* Fetches and decodes switch information
*/
static int32_t mmc_read_switch(struct mmc_card *card)
{
int32_t err;
uint32_t status[64/sizeof(uint32_t)] = {0};
uint8_t *p_sta = (uint8_t *)status;
if (card->scr.sda_vsn < SCR_SPEC_VER_1) {
SD_LOGN("Card ver. does not support to read switch info!\n");
return 0;
}
if (!(card->csd.cmdclass & CCC_SWITCH)) {
SD_LOGW("card lacks mandatory switch function, performance might suffer.\n");
return 0;
}
/* Find out the supported Bus Speed Modes. */
err = mmc_sd_switch(card, SD_SWITCH_CHECK, SD_SWITCH_GRP_ACCESS_MODE,
SD_SWITCH_ACCESS_HS, p_sta);
if (err) {
SD_LOGW("%s: problem reading Bus Speed modes.\n", __func__);
goto out;
}
if (p_sta[13] & SD_MODE_HIGH_SPEED)
card->sw_caps.hs_max_dtr = HIGH_SPEED_MAX_DTR;
if (card->scr.sda_spec3) {
card->sw_caps.sd3_bus_mode = p_sta[13];
/* Find out Driver Strengths supported by the card */
err = mmc_sd_switch(card, SD_SWITCH_CHECK, SD_SWITCH_GRP_DRV_STRENGTH,
SD_SWITCH_ACCESS_HS, p_sta);
if (err) {
SD_LOGW("%s: problem reading Driver Strength.\n", __func__);
goto out;
}
card->sw_caps.sd3_drv_type = p_sta[9];
/* Find out Current Limits supported by the card */
err = mmc_sd_switch(card, SD_SWITCH_CHECK, SD_SWITCH_GRP_CUR_LIMIT,
SD_SWITCH_ACCESS_HS, p_sta);
if (err) {
SD_LOGW("%s: problem reading Current Limit.\n", __func__);
goto out;
}
card->sw_caps.sd3_curr_limit = p_sta[7];
}
out:
return err;
}
/*
* Test if the card supports high-speed mode and, if so, switch to it.
*/
int32_t mmc_sd_switch_hs(struct mmc_card *card)
{
int32_t err;
uint32_t status[64/sizeof(uint32_t)] = {0};
uint8_t *p_sta = (uint8_t *)status;
if (card->scr.sda_vsn < SCR_SPEC_VER_1) {
SD_LOGN("Card ver. does not support to switch to high speed!\n");
return 0;
}
if (!(card->csd.cmdclass & CCC_SWITCH)) {
SD_LOGN("Card cmdclass not support to switch to high speed!\n");
return 0;
}
if (!(card->host->caps & MMC_CAP_SD_HIGHSPEED))
return 0;
if (card->sw_caps.hs_max_dtr == 0)
return 0;
/* Check function */
err = mmc_sd_switch(card, SD_SWITCH_CHECK, SD_SWITCH_GRP_ACCESS_MODE,
SD_SWITCH_ACCESS_HS, p_sta);
if (err)
return err;
if ((p_sta[16] & 0x0F) != 1) {
SD_LOGW("%s: Problem switching card into high-speed mode!\n", __func__);
err = 0;
} else {
err = 1;
}
return err;
}
static int32_t mmc_send_cxd_native(struct mmc_host *host, uint32_t arg, uint32_t *cxd, int32_t opcode)
{
int32_t err;
struct mmc_command cmd = {0};
if (!host || !cxd) {
SDC_LOGE_RAW(ROM_ERR_MASK, "%s,%d err", __func__, __LINE__);
return -1;
}
cmd.opcode = opcode;
cmd.arg = arg;
cmd.flags = MMC_RSP_R2 | MMC_CMD_AC;
err = mmc_wait_for_cmd(host, &cmd);
if (err)
return err;
HAL_Memcpy(cxd, cmd.resp, sizeof(uint32_t) * 4);
return 0;
}
int32_t mmc_send_csd(struct mmc_card *card, uint32_t *csd)
{
return mmc_send_cxd_native(card->host, card->rca << 16,
csd, MMC_SEND_CSD);
}
int32_t mmc_sd_get_csd(struct mmc_card *card)
{
int32_t err;
uint32_t csd[4] = {0};
/* Fetch CSD from card. */
err = mmc_send_csd(card, csd);
if (err)
return err;
SD_LOGN("card raw csd:\n");
sd_hex_dump_bytes((void *)csd, 16);
err = mmc_decode_csd(card, csd);
if (err)
return err;
return 0;
}
int32_t mmc_sd_setup_card(struct mmc_host *host, struct mmc_card *card)
{
int32_t err;
int32_t retries;
uint32_t raw_scr[2] = {0};
/* Fetch SCR from card. */
err = mmc_app_send_scr(card, raw_scr);
if (err)
return err;
SD_LOGN("card raw scr:\n");
sd_hex_dump_bytes((void *)raw_scr, 8);
err = mmc_decode_scr(card, raw_scr);
if (err)
return err;
/* Fetch and process SD Status register. */
err = mmc_read_ssr(card);
if (err)
return err;
#ifdef SD_SUPPORT_ERASE
/* Erase init depends on CSD and SSR */
mmc_init_erase(card);
#endif
/* Fetch switch information from card. */
for (retries = 1; retries <= 3; retries++) {
err = mmc_read_switch(card);
if (!err) {
if (retries > 1) {
SD_LOGW("%s: recovered\n", __func__);
}
break;
} else {
SD_LOGW("%s: read switch failed (attempt %d)\n",
__func__, (unsigned int)retries);
}
}
if (err)
return err;
#ifdef SD_SUPPORT_WRITEPROTECT
int32_t ro = -1;
/* Check if read-only switch is active. */
mmc_host_clk_hold(card->host);
ro = HAL_SDC_Get_ReadOnly(card->host);
mmc_host_clk_release(card->host);
if (ro < 0) {
SD_LOGW("%s: host does not support reading read-only switch."
" assuming write-enable.\n", __func__);
} else if (ro > 0) {
mmc_card_set_readonly(card);
}
#endif
return 0;
}
uint32_t mmc_sd_get_max_clock(struct mmc_card *card)
{
uint32_t max_dtr = (uint32_t)-1;
if (mmc_card_highspeed(card)) {
if (max_dtr > card->sw_caps.hs_max_dtr)
max_dtr = card->sw_caps.hs_max_dtr;
} else if (max_dtr > card->csd.max_dtr) {
max_dtr = card->csd.max_dtr;
}
return max_dtr;
}
/*
* Handle the detection and initialisation of a card.
*
* In the case of a resume, "oldcard" will contain the card
* we're trying to reinitialise.
*/
static int32_t mmc_sd_init_card(struct mmc_card *card, struct mmc_host *host)
{
int32_t err = 0;
/*
* I/O voltage should be 3.3 V here for the initialization needed.
*/
/* cmd2, send cid, check if card support 3.3V */
err = mmc_all_send_cid(host, card->cidno);
if (err) {
SD_LOGW("Cards all send CID number failed !!\n");
return -1;
} else
SD_LOGN("Card CID number:%x\n", (unsigned int)card->cidno[0]);
card->type = MMC_TYPE_SD;
/* cmd3, For native busses: get card RCA and quit open drain mode. */
err = mmc_send_relative_addr(card->host, &card->rca);
if (err) {
SD_LOGW("Card public new RCA failed !!\n");
return -1;
} else
SD_LOGD("Card public new RCA:%x\n", (unsigned int)card->rca);
/* cmd10, get CID register */
if (mmc_send_cid(card)) {
SD_LOGW("Card send CID reg failed !!\n");
return -1;
}
/* cmd9, get CSD register */
if (mmc_sd_get_csd(card)) {
SD_LOGW("Card send CSD reg failed !!\n");
return -1;
}
/* cmd7, Select card to standby state, as all following commands rely on that. */
if (mmc_select_card(card, 1)) {
SD_LOGW("mmc_select_card failed !!\n");
return -1;
}
err = mmc_sd_setup_card(host, card);
if (err)
goto free_card;
/* Initialization sequence for UHS-I cards */
#ifdef SD_SUPPORT_VERSION3
if (card->ocr.ocr & SD_ROCR_S18A) {
err = mmc_sd_init_uhs_card(card);
if (err)
goto free_card;
/* Card is an ultra-high-speed card */
mmc_card_set_uhs(card);
/*
* Since initialization is now complete, enable preset
* value registers for UHS-I cards.
*/
mmc_host_clk_hold(card->host);
HAL_SDC_Enable_Preset_Value(card->host, true);
mmc_host_clk_release(card->host);
} else
#endif
{
uint32_t clk;
uint32_t retries = 3;
while (retries) {
err = mmc_sd_switch_hs(card);
if (err < 0) {
SD_LOGE("%s: Re-switch hs, err %d (retries = %u)\n",
__func__, (unsigned int)err, (unsigned int)retries);
mmc_mdelay(5);
retries--;
continue;
}
break;
}
if (err <= 0) {
SD_LOGW("switch to high speed error, use DS: 25 MHz\n");
clk = 25000000;
err = HAL_SDC_Update_Clk(card->host, clk);
if (err)
return -1;
} else {
mmc_card_set_highspeed(card);
card->sd_bus_speed = HIGH_SPEED_BUS_SPEED;
clk = mmc_sd_get_max_clock(card);
err = HAL_SDC_Update_Clk(card->host, clk);
if (err)
return -1;
SD_LOGN("card is switched to high speed mode, clk:%u KHz\n", (unsigned int)clk/1000);
}
/* Switch to wider bus (if supported). */
if ((host->caps & MMC_CAP_4_BIT_DATA) &&
(card->scr.bus_widths & SD_SCR_BUS_WIDTH_4)) {
err = mmc_app_set_bus_width(card, MMC_BUS_WIDTH_4);
if (err) {
SD_LOGW("Set bus width error, use default 1 bit !!\n");
return -1;
}
/* config SDMMC controller bus width */
HAL_SDC_Set_BusWidth(card->host, MMC_BUS_WIDTH_4);
SD_LOGN("Set bus width type: %d\n", MMC_BUS_WIDTH_4);
}
}
mmc_add_card(card);
return 0;
free_card:
return err;
}
#ifdef CONFIG_SD_PM
static int32_t mmc_sd_suspend(struct mmc_host *host)
{
struct mmc_card *card = host->card;
if (card == NULL) {
SD_LOGE_RAW(ROM_ERR_MASK, "card open fail\n");
return -1;
}
mmc_card_open(card->id);
card->suspend = 1;
mmc_card_deinit(host->card);
SD_LOGD("%s ok\n", __func__);
mmc_card_close(card->id);
return 0;
}
static int32_t mmc_sd_resume(struct mmc_host *host)
{
struct mmc_card *card = host->card;
if (card == NULL) {
SD_LOGE_RAW(ROM_ERR_MASK, "card open fail\n");
return -1;
}
mmc_card_open(card->id);
mmc_rescan(card, host->sdc_id);
card->suspend = 0;
SD_LOGD("%s ok\n", __func__);
mmc_card_close(card->id);
return 0;
}
static const struct mmc_bus_ops sd_bus_ops = {
.suspend = mmc_sd_suspend,
.resume = mmc_sd_resume,
};
#endif
/*
* Starting point for SD card init.
*/
int32_t mmc_attach_sd(struct mmc_card *card, struct mmc_host *host)
{
int32_t err = 0;
uint32_t ocr = 0;
if (!host) {
SD_LOGE_RAW(ROM_ERR_MASK, "%s,%d err", __func__, __LINE__);
return -1;
}
//SD_LOGE("%s,%d\n",__FUNCTION__,__LINE__);
/* send cmd41/55 to check operation condition */
err = mmc_send_app_op_cond(host, 0, &ocr);
if (err) {
return err;
}
SD_LOGN("card ocr: %08x\n", (unsigned int)ocr);
/*
* Sanity check the voltages that the card claims to
* support.
*/
if (ocr & 0x7F) {
SD_LOGW("%s: card claims to support voltages below the defined range."
" These will be ignored.\n", __func__);
ocr &= ~0x7F;
}
card->ocr.ocr = 0x7fffffff & ocr; /* set card not in busy state */
err = mmc_sd_init_card(card, host);
if (err) {
goto out;
}
host->card = card;
#ifdef CONFIG_SD_PM
if (!card->suspend) {
mmc_attach_bus(host, &sd_bus_ops);
}
#endif
out:
return err;
}
void mmc_deattach_sd(struct mmc_card *card, struct mmc_host *host)
{
mmc_select_card(card, 0);
card->state &= ~MMC_STATE_HIGHSPEED;
#ifdef CONFIG_SD_PM
if (!card->suspend) {
mmc_detach_bus(host);
host->card = NULL;
}
#else
host->card = NULL;
#endif
}
#endif /* CONFIG_USE_SD */