rt-thread-official/bsp/allwinner/libraries/sunxi-hal/hal/source/sdmmc/test.c

440 lines
14 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 <string.h>
#include <stdio.h>
#include "sys/sys_debug.h"
#include "os_time.h"
//#include "FreeRTOS.h"
#include "os_semaphore.h"
#include "hal_def.h"
#include "hal_ccm.h"
#include "hal_sdhost.h"
#include "sdmmc.h"
#include "sdio.h"
#include "_sd_define.h"
#include "_sdhost.h"
#define TEST_SD
#define TEST_SD_WRITE
#ifdef TEST_SD
#define SIZE_1K (1024)
#define SIZE_1M (SIZE_1K*SIZE_1K)
#define READ_WRITE_SINGLE_SIZE (16*SIZE_1K)
#define READ_WRITE_TOTAL_SIZE (8*SIZE_1M)
struct sdmmc_tester {
struct mmc_card *card;
SDCard_InitTypeDef card_param;
SDC_InitTypeDef sdc_param;
uint16_t sdc_id;
uint16_t card_id;
uint8_t wbuf[READ_WRITE_SINGLE_SIZE];
uint8_t rbuf[READ_WRITE_SINGLE_SIZE];
#ifdef CONFIG_DETECT_CARD
OS_Semaphore_t card_present_sem;
#endif
};
static struct sdmmc_tester *sdmmc_test = NULL;
#ifdef CONFIG_DETECT_CARD
static int detect_status = -1;
void card_detect(uint32_t present)
{
if (present) {
detect_status = 0;
rt_kprintf("%s insert\n", __func__);
if (mmc_card_create(sdmmc_test->card_id, &sdmmc_test->card_param) != 0) {
rt_kprintf("mmc create fail\n");
return ;
}
sdmmc_test->card = mmc_card_open(sdmmc_test->card_id);
if (sdmmc_test->card == NULL) {
rt_kprintf("mmc open fail\n");
return ;
}
/* scan card for detect card is exist? */
if (!mmc_card_present(sdmmc_test->card)) {
// if (mmc_rescan(sdmmc_test->card, sdmmc_test->card->id)) {
sdmmc_test->card->type = MMC_TYPE_SD;
if (mmc_rescan(sdmmc_test->card, sdmmc_test->sdc_id)) {
rt_kprintf("Initial card failed!!\n");
mmc_card_close(sdmmc_test->card_id);
detect_status = -1;
return ;
} else {
rt_kprintf("Initial card success. capacity :%dMB\n", sdmmc_test->card->csd.capacity / 1024);
mmc_card_close(sdmmc_test->card_id);
#ifndef CONFIG_KERNEL_FREERTOS
// void mount_sdmmc_filesystem(int card_id);
// mount_sdmmc_filesystem((int)sdmmc_test->card_id);
#endif
SDC_SemPost(&sdmmc_test->card_present_sem);
}
} else {
rt_kprintf("%s not eixst\n", __func__);
mmc_card_close(sdmmc_test->card_id);
return ;
}
} else {
struct mmc_card *card;
rt_kprintf("%s removed\n", __func__);
card = mmc_card_open(sdmmc_test->card_id);
if (card == NULL) {
rt_kprintf("card open fail\n");
} else {
if (mmc_card_present(card)) {
mmc_card_deinit(card);
}
mmc_card_close(sdmmc_test->card_id);
#ifndef CONFIG_KERNEL_FREERTOS
// void unmount_sdmmc_filesystem(void);
// unmount_sdmmc_filesystem();
#endif
mmc_card_delete(sdmmc_test->card_id);
}
}
}
#endif
//#define WIFI_DEBUG 0
// #define DETECT_BY_GPIO 1
int32_t mmc_test_init(uint32_t host_id, SDC_InitTypeDef *sdc_param, uint32_t scan)
{
struct mmc_host *host;
if (!sdmmc_test) {
sdmmc_test = malloc(sizeof(struct sdmmc_tester));
if (!sdmmc_test) {
rt_kprintf("malloc faild!\n");
return -1;
}
memset(sdmmc_test, 0, sizeof(struct sdmmc_tester));
}
if (sdc_param)
memcpy(&sdmmc_test->sdc_param, sdc_param, sizeof(SDC_InitTypeDef));
#ifdef CONFIG_DETECT_CARD
OS_SemaphoreCreate(&sdmmc_test->card_present_sem, 0, OS_SEMAPHORE_MAX_COUNT);
if (!sdc_param) {
#ifndef DETECT_BY_GPIO
sdmmc_test->sdc_param.cd_mode = CARD_ALWAYS_PRESENT;
#else
sdmmc_test->sdc_param.cd_mode = CARD_DETECT_BY_GPIO_IRQ;
#endif
sdmmc_test->sdc_param.cd_cb = &card_detect;
#ifndef WIFI_DEBUG
sdmmc_test->sdc_param.debug_mask = (ROM_INF_MASK | \
ROM_WRN_MASK | ROM_ERR_MASK | ROM_ANY_MASK);
#else
sdmmc_test->sdc_param.debug_mask = (ROM_DUMP_MASK | ROM_DBG_MASK | ROM_INF_MASK | \
ROM_WRN_MASK | ROM_ERR_MASK | ROM_ANY_MASK);
#endif
sdmmc_test->sdc_param.dma_use = 1;
}
#endif
if (!sdc_param)
sdc_param = &sdmmc_test->sdc_param;
#ifdef WIFI_DEBUG
sdmmc_test->card_param.debug_mask = (ROM_DUMP_MASK | ROM_DBG_MASK | ROM_INF_MASK | \
ROM_WRN_MASK | ROM_ERR_MASK | ROM_ANY_MASK);
#else
sdmmc_test->card_param.debug_mask = (ROM_INF_MASK | \
ROM_WRN_MASK | ROM_ERR_MASK | ROM_ANY_MASK);
#endif
host = hal_sdc_create(host_id, sdc_param);
host->State = SDC_STATE_RESET;
hal_sdc_init(host);
//HAL_SDC_SetHighSpeed(host);
sdmmc_test->sdc_id = host_id;
sdmmc_test->card_id = host_id;
if (scan && sdmmc_test->sdc_param.cd_mode == CARD_ALWAYS_PRESENT) {
card_detect(1);
return detect_status;
}
return 0;
}
int32_t mmc_test_exit(uint16_t sd_id, uint16_t host_id)
{
struct mmc_card *card;
struct mmc_host *host;
if (!sdmmc_test->card || !sdmmc_test->card->host ||
sdmmc_test->card->id != sd_id || sdmmc_test->sdc_id != host_id) {
if (!sdmmc_test->card || !sdmmc_test->card->host)
rt_kprintf("no card found\n");
else
rt_kprintf("wrong card id:%d<->%d host id:%ld<->%d\n", sd_id, host_id,
HAL_PR_SZ_L(sdmmc_test->card->id), sdmmc_test->sdc_id);
return 0;
}
card = mmc_card_open(sdmmc_test->card_id);
if (card == NULL) {
rt_kprintf("card open fail\n");
} else {
if (mmc_card_present(card)) {
mmc_card_deinit(card);
}
mmc_card_close(sdmmc_test->card_id);
mmc_card_delete(sdmmc_test->card_id);
}
#if 0
host = HAL_SDC_Open(host_id);
HAL_SDC_Close(host_id);
#endif
hal_sdc_deinit(sd_id);
#if 0
HAL_SDC_Destory(host);
#endif
#ifdef CONFIG_DETECT_CARD
OS_SemaphoreDelete(&sdmmc_test->card_present_sem);
#endif
if (sdmmc_test) {
free(sdmmc_test);
sdmmc_test = NULL;
}
return 0;
}
struct mmc_card *mmc_scan_init(uint16_t sd_id, uint16_t sdc_id, SDCard_InitTypeDef *card_param)
{
struct mmc_card *card;
if (!card_param) {
card_param = &sdmmc_test->card_param;
sdmmc_test->card_param.debug_mask = ROM_WRN_MASK | ROM_ERR_MASK | ROM_ANY_MASK;
}
if (mmc_card_create(sd_id, card_param) != 0) {
rt_kprintf("mmc create fail\n");
return NULL;
}
card = mmc_card_open(sd_id);
if (card == NULL) {
rt_kprintf("mmc open fail\n");
return NULL;
}
if (!mmc_card_present(card)) {
int mmc_ret = mmc_rescan(card, sdc_id);
if (mmc_ret != 0) {
rt_kprintf("mmc scan fail\n");
mmc_card_close(sd_id);
return NULL;
} else {
rt_kprintf("mmc init\n");
}
}
mmc_card_close(sd_id);
sdmmc_test->card = card;
return card;
}
int32_t mmc_test(uint32_t host_id, uint32_t cd_mode, uint32_t sdc_degmask, uint32_t card_dbgmask)
{
int32_t err;
uint32_t i, cnt = 0;
SDC_InitTypeDef sdc_param = { 0 };
sdc_param.cd_mode = cd_mode;
sdc_param.cd_cb = &card_detect;
sdc_param.debug_mask = sdc_degmask;
sdc_param.dma_use = 1;
//if (mmc_test_init(host_id, &sdc_param, 1)) {
if (mmc_test_init(host_id, &sdc_param, 1)) {
return -1;
}
memset((void *)sdmmc_test->wbuf, 0x55, 128);
memset((void *)&sdmmc_test->wbuf[128], 0xaa, 128);
for (i = 0; i < 256; i ++)
sdmmc_test->wbuf[256 + i] = i;
memcpy((void *)&sdmmc_test->wbuf[512], (void *)sdmmc_test->wbuf, 512);
sdmmc_test->card_param.debug_mask = card_dbgmask;
// if (mmc_test_init(host_id, &sdc_param, 1)) {
// return -1;
// }
sdmmc_test->card_param.type = MMC_TYPE_SD;
while (cnt++ < 1) {
uint32_t throuth_mb, throuth_kb;
OS_Time_t tick_use;
rt_kprintf("%s,%d count:%lu\n", __func__, __LINE__, HAL_PR_SZ_L(cnt));
#ifdef CONFIG_DETECT_CARD
if (!mmc_card_present(sdmmc_test->card) || (cd_mode != CARD_ALWAYS_PRESENT))
OS_SemaphoreWait(&sdmmc_test->card_present_sem, OS_WAIT_FOREVER);
#endif
struct mmc_card *card = mmc_card_open(sdmmc_test->card_id);
card->debug_mask = card_dbgmask;
if (card == NULL) {
rt_kprintf("mmc open fail\n");
goto err_out;
}
#ifdef TEST_SD_WRITE
tick_use = OS_GetTicks();
err = mmc_block_write(sdmmc_test->card, sdmmc_test->wbuf, 0, 1);
tick_use = OS_GetTicks() - tick_use;
if (err) {
goto err_out;
} else {
rt_kprintf("%s 1 block write ok, 512B use:%lu ms\n", __func__,
HAL_PR_SZ_L((uint32_t)OS_TicksToMSecs(tick_use)));
}
#endif
memset((void *)sdmmc_test->rbuf, 0, 512);
tick_use = OS_GetTicks();
err = mmc_block_read(sdmmc_test->card, sdmmc_test->rbuf, 0, 1);
tick_use = OS_GetTicks() - tick_use;
if (err) {
goto err_out;
} else {
rt_kprintf("%s 1 block read ok, 512B use:%lu ms\n", __func__,
HAL_PR_SZ_L((uint32_t)OS_TicksToMSecs(tick_use)));
#ifndef TEST_SD_WRITE
print_hex_dump_words(sdmmc_test->rbuf, 512);
#endif
}
#ifdef TEST_SD_WRITE
if (memcmp((void *)sdmmc_test->wbuf, (void *)sdmmc_test->rbuf, 512)) {
goto err_out;
} else
rt_kprintf("%s,%d mmc 1 block rw ok\n", __func__, __LINE__);
tick_use = OS_GetTicks();
for (i = 0; i < READ_WRITE_TOTAL_SIZE/READ_WRITE_SINGLE_SIZE; i++) {
err = mmc_block_write(sdmmc_test->card, sdmmc_test->wbuf, 3 + i * (READ_WRITE_SINGLE_SIZE/512),
READ_WRITE_SINGLE_SIZE/512);
if (err)
break;
if (i % 50 == 0)
rt_kprintf("%s, wirite cnt:%lu\n", __func__, HAL_PR_SZ_L(i));
}
tick_use = OS_GetTicks() - tick_use;
if (err) {
rt_kprintf("%s,%d mmc mult blocks write err!\n", __func__, __LINE__);
goto err_out;
} else {
throuth_kb = READ_WRITE_TOTAL_SIZE/SIZE_1K*1000/(uint32_t)OS_TicksToMSecs(tick_use);
throuth_mb = throuth_kb/1000;
rt_kprintf("%s mult blocks write ok, %d MB use:%lu ms, throughput:%lu.%lu MB/S\n",
__func__, READ_WRITE_TOTAL_SIZE/SIZE_1M, HAL_PR_SZ_L((uint32_t)OS_TicksToMSecs(tick_use)),
HAL_PR_SZ_L(throuth_mb), HAL_PR_SZ_L(throuth_kb - throuth_mb));
}
#endif
tick_use = OS_GetTicks();
for (i = 0; i < READ_WRITE_TOTAL_SIZE/READ_WRITE_SINGLE_SIZE; i++) {
err = mmc_block_read(sdmmc_test->card, sdmmc_test->rbuf, 3 + i * (READ_WRITE_SINGLE_SIZE/512),
READ_WRITE_SINGLE_SIZE/512);
if (err)
break;
}
tick_use = OS_GetTicks() - tick_use;
if (err) {
rt_kprintf("%s,%d mmc mult blocks read err!\n", __func__, __LINE__);
goto err_out;
} else {
throuth_kb = READ_WRITE_TOTAL_SIZE/SIZE_1K*1000/(uint32_t)OS_TicksToMSecs(tick_use);
throuth_mb = throuth_kb/1000;
rt_kprintf("%s mult blocks read ok, %d MB use:%lu ms, throughput:%lu.%lu MB/S\n",
__func__, READ_WRITE_TOTAL_SIZE/SIZE_1M, HAL_PR_SZ_L((uint32_t)OS_TicksToMSecs(tick_use)),
HAL_PR_SZ_L(throuth_mb), HAL_PR_SZ_L(throuth_kb - throuth_mb));
}
memset((void *)sdmmc_test->rbuf, 0, READ_WRITE_SINGLE_SIZE);
err = mmc_block_read(sdmmc_test->card, sdmmc_test->rbuf, 3, READ_WRITE_SINGLE_SIZE/512);
if (err) {
goto err_out;
} else
rt_kprintf("%s %d blocks read ok\n", __func__, READ_WRITE_SINGLE_SIZE/512);
#ifdef TEST_SD_WRITE
if (memcmp((void *)sdmmc_test->wbuf, (void *)sdmmc_test->rbuf, 1024)) { /* check 1024B */
rt_kprintf("%s %d mmc blocks rw failed\n", __func__, READ_WRITE_SINGLE_SIZE/512);
goto err_out;
} else
rt_kprintf("%s %d mmc blocks rw ok\n", __func__, READ_WRITE_SINGLE_SIZE/512);
#endif
mmc_card_close(sdmmc_test->card_id);
OS_MSleep(1000);
}
mmc_test_exit(host_id, sdmmc_test->card_id);
return 0;
err_out:
#ifdef TEST_SD_WRITE
rt_kprintf("%s,%d mmc block rw failed\n", __func__, __LINE__);
rt_kprintf("rbuf:\n");
print_hex_dump_words(sdmmc_test->rbuf, SIZE_1K);
rt_kprintf("wbuf:\n");
print_hex_dump_words(sdmmc_test->wbuf, 512);
#endif
#ifndef CONFIG_DETECT_CARD
out:
#endif
mmc_test_exit(host_id, sdmmc_test->card_id);
return -1;
}
#endif /* TEST_SD */