MMC
This sample demonstrates the use of an MMC with the chip using the SD library.
The sample initializes the memory card and performs a data write operation to the card. It then performs a data read and checks for consistency with the previously written data.
Requirements
The sample supports the following development kits:
Hardware Platforms |
Board Name |
---|---|
RTL87x2G HDK |
RTL87x2G EVB |
For more requirements, please refer to Quick Start.
Wiring
For the MMC daughterboard and wiring, please refer to WiFi and SD Nand Flash Multiplexed Interface in Model B Evaluation Board.
MMC interface pins are shown in the table below:
Name |
PIN |
Description |
---|---|---|
DAT3 / CS |
P9_5 |
Data Line 3 |
CMD / DI |
P9_3 |
Command Response Line |
VSS1 |
GND |
Supply Power Ground |
VDD |
VCC(3.3V) |
Supply Power 3.3V |
CLK |
P9_4 |
Clock |
VSS2 |
GND |
Supply Power Ground |
DAT0 / DO |
P10_0 |
Data Line 0 |
DAT1 |
P9_7 |
Data Line 1 |
DAT2 |
P9_6 |
Data Line 2 |
Building and Downloading
This sample can be found in the SDK folder:
Project file: samples\module_sample\sd\mmc\proj\mdk
Project file: samples\module_sample\sd\mmc\proj\gcc
To build and run the sample, follow the steps listed below:
Open sample project file.
To build the target, follow the steps listed on the Generating App Image in Quick Start.
After a successful compilation, the app bin
app_MP_xxx.bin
will be generated in the directorymdk\bin
orgcc\bin
.To download app bin into EVB board, follow the steps listed on the MP Tool Download in Quick Start.
Press reset button on EVB board and it will start running.
Experimental Verification
After resetting the EVB, observe the MMC initialization status in the Debug Analyzer. If the card is successfully recognized, the status
Init: 0
is printed.Init: 0
Write the prepared data (the first 30 blocks of data are 0xff, and the next 30 blocks of data are 0x00) to the MMC. If the write is successful, the status
Write: 0
is printed.Write: 0
Read the data from the MMC. If the read is successful, the status
Read: 0
is printed along with the read data.Read: 0 Block 0: 0x0000 -> 0xFF - 0xFF - 0xFF - 0xFF --- 0xFF - 0xFF - 0xFF - 0xFF ... Block 30: 0x0000 -> 0x00 - 0x00 - 0x00 - 0x00 --- 0x00 - 0x00 - 0x00 - 0x00 ...
Check if the read and write data are identical. If they are not, it will print the error data. For example, if the 100th data of the 1st block is an error: 0x33, it will print as follows:
Block 0: 0xA0 -> 0x33 data error
Code Overview
This chapter will be introduced according to the following several parts:
Peripheral initialization will be introduced in chapter Initialization.
Functional implementation after initialization will be introduced in chapter Functional Implementation.
Source Code Directory
Project directory:
sdk\samples\module_sample\sd\mmc\proj
Source code directory:
sdk\samples\module_sample\sd\mmc\src
Source files are currently categorized into several groups as below.
└── Project: mmc
└── secure_only_app
└── Device includes startup code
├── startup_rtl.c
└── system_rtl.c
├── CMSIS includes CMSIS header files
├── CMSE Library Non-secure callable lib
├── lib includes all binary symbol files that user application is built on
├── sd.lib
└── rtl87x2g_io.lib
├── peripheral includes all peripheral drivers and module code used by the application
├── rtl_rcc.c
├── rtl_pinmux.c
├── io_dlps.c
└── rtl_nvic.c
└── APP includes the ble_peripheral user application implementation
├── main_ns.c
├── app_task.c
└── sd_test.c
Initialization
Set the initialization parameters.
Set the Card type to EMMC.
Set the data width to 4BIT.
Set the clock output frequency to 52MHz.
static const SdEmmcInitParm_t Parm =
{
.CardType = CARDTYPE_MMC,
.DataWidth = DATAWIDTH_4BIT,
.ClkOutFreq_kHz = 52 * 1000,
};
...
Functional Implementation
Execute
Emmc_Init
to initialize the MMC.Execute the
os_mem_alloc
function to allocate memory for pBuf.Execute
Emmc_Write
function to write the prepared data (the first 30 blocks of data are 0xff, and the next 30 blocks of data are 0x00) to the MMC.Pre-fill pBuf with 0x77, then execute
Emmc_Read
function to read the data from the MMC into pBuf.Print the read data and check if the read and write data are identical. If they are not, it will print the error data.
Execute the
os_mem_free
function to free the memory for pBuf.
SdEmmcRes_t Res = Init(SD_Demo, &Parm);
uint8_t *pBuf = os_mem_alloc(RAM_TYPE_EXT_DATA_SRAM, BUF_BYTES);
memset(pBuf, 0xff, BUF_BYTES / 2);
memset(pBuf + BUF_BYTES / 2, 0, BUF_BYTES / 2);
Res = Write(SD_Demo, START_BLOCK, BLOCK_CNT, pBuf);
memset(pBuf, 0x77, BUF_BYTES);
Res = Read(SD_Demo, START_BLOCK, BLOCK_CNT, pBuf);
/* Print all read data */
for (size_t i = 0; i < BUF_BYTES; i += 8)
{
DBG_DIRECT("Block %d: 0x%04x -> 0x%02x - 0x%02x - 0x%02x - 0x%02x --- 0x%02x - 0x%02x - 0x%02x - 0x%02x",
i / 512, i % 512,
pBuf[i], pBuf[i + 1], pBuf[i + 2], pBuf[i + 3],
pBuf[i + 4], pBuf[i + 5], pBuf[i + 6], pBuf[i + 7]);
}
/* Compare whether the read data is correct from the write data */
for (uint32_t j = 0; j < BUF_BYTES; j++)
{
if (((j < BUF_BYTES / 2) && ((pBuf[j] != 0xff))) ||
((j >= BUF_BYTES / 2) && (j < BUF_BYTES) && (pBuf[j] != 0x00)))
{
DBG_DIRECT("Block %d: 0x%04x -> 0x%02x data error", j / 512, j % 512, pBuf[j]);
if (pBuf[j] == 0x77) { break; }
}
}
os_mem_free(pBuf);