[PATCH 5/8] BBB: Add am335x_mmc.c and am335x_mmc.h that implement "mmc" command with initialization functionality

Jarielle Catbagan jcatbagan93 at gmail.com
Tue Aug 4 18:03:25 UTC 2015


---
 ports/beagleboneblack/am335x_mmc.c | 347 +++++++++++++++++++++++++++++++++++++
 ports/beagleboneblack/am335x_mmc.h |  13 ++
 2 files changed, 360 insertions(+)
 create mode 100644 ports/beagleboneblack/am335x_mmc.c
 create mode 100644 ports/beagleboneblack/am335x_mmc.h

diff --git a/ports/beagleboneblack/am335x_mmc.c b/ports/beagleboneblack/am335x_mmc.c
new file mode 100644
index 0000000..f045fec
--- /dev/null
+++ b/ports/beagleboneblack/am335x_mmc.c
@@ -0,0 +1,347 @@
+/*
+ * Copyright (c) 2015 Jarielle Catbagan <jcatbagan93 at gmail.com>
+ *
+ * The license and distribution terms for this file may be
+ * found in the file LICENSE in this distribution or at
+ * http://www.apache.org/licenses/LICENSE-2.0
+ */
+
+#include "genlib.h"
+#include "cli.h"
+#include "stddefs.h"
+#include "am335x.h"
+#include "am335x_mmc.h"
+
+uint16_t mmcrca;
+int mmcInum;
+
+char *mmcHelp[] = {
+	"MultiMediaCard Interface",
+	"[options] {operation} [args]...",
+#if INCLUDE_VERBOSEHELP
+	"",
+	"Options:",
+	" -i ##	interface # (default is 0)",
+	" -v 	additive verbosity",
+	"",
+	"Operations:",
+	" init",
+	" read {dest} {blk} {blktot}",
+	" write {source} {blk} {blktot}",
+#endif /* INCLUDE_VERBOSEHELP */
+	0
+};
+
+int
+mmccmd(uint32_t cmd, uint32_t arg, uint32_t resp[4])
+{
+	/* Clear the SD_STAT register for proper update of status bits after CMD invocation */
+	MMC1_REG(SD_STAT) = 0xFFFFFFFF;
+
+	MMC1_REG(SD_ARG) = arg;
+	MMC1_REG(SD_CMD) = cmd;
+
+	/* CMDx complete? */
+	while (!(MMC1_REG(SD_STAT) & (SD_STAT_CC | SD_STAT_ERRI)));
+
+	resp[0] = MMC1_REG(SD_RSP10);
+	resp[1] = MMC1_REG(SD_RSP32);
+	resp[2] = MMC1_REG(SD_RSP54);
+	resp[3] = MMC1_REG(SD_RSP76);
+
+	/* CMDx error? */
+	if (MMC1_REG(SD_STAT) & SD_STAT_ERRI)
+		return(-1);
+	else
+		return(0);
+}
+
+int
+mmc(int argc, char *argv[])
+{
+	char *cmd, *buf;
+	int opt, verbose, mmcret, blknum, blkcnt;
+
+	verbose = 0;
+
+	while ((opt = getopt(argc, argv, "i:v")) != -1) {
+		switch (opt) {
+			case 'i':
+				mmcInum = atoi(optarg);
+				break;
+			case 'v':
+				verbose++;
+				break;
+			default:
+				return(CMD_PARAM_ERROR);
+		}
+	}
+
+	if (argc < optind + 1)
+		return(CMD_PARAM_ERROR);
+
+	cmd = argv[optind];
+
+	if (mmcInstalled(mmcInum) == 0) {
+		printf("MMC not installed\n");
+		return(CMD_FAILURE);
+	}
+
+	if (strcmp(cmd, "init") == 0) {
+		mmcret = mmcInit(mmcInum, verbose);
+		if(mmcret < 0) {
+			printf("mmcInit returned %d\n", mmcret);
+			return(CMD_FAILURE);
+		}
+	}
+	else if (strcmp(cmd, "read") == 0) {
+		if (argc != (optind + 4))
+			return(CMD_PARAM_ERROR);
+
+		buf = (char *)strtoul(argv[optind + 1], 0, 0);
+		blknum = strtoul(argv[optind + 2], 0, 0);
+		blkcnt = strtoul(argv[optind + 3], 0, 0);
+
+		mmcret = mmcRead(mmcInum, buf, blknum, blkcnt);
+		if (mmcret < 0) {
+			printf("mmcRead returned %d\n", mmcret);
+			return(CMD_FAILURE);
+		}
+	}
+	else if (strcmp(cmd, "write") == 0) {
+		if (argc != (optind + 4))
+			return(CMD_PARAM_ERROR);
+
+		buf = (char *)strtoul(argv[optind + 1], 0, 0);
+		blknum = strtoul(argv[optind + 2], 0, 0);
+		blkcnt = strtoul(argv[optind + 3], 0, 0);
+
+		mmcret = mmcWrite(mmcInum, buf, blknum, blkcnt);
+		if (mmcret < 0) {
+			printf("mmcWrite returned %d\n", mmcret);
+			return(CMD_FAILURE);
+		}
+	}
+	else {
+		printf("mmc op <%s> not found\n", cmd);
+		return(CMD_FAILURE);
+	}
+
+	return(CMD_SUCCESS);
+}
+
+int
+mmcInit(int interface, int verbose)
+{
+	uint32_t cmd, arg, resp[4];
+
+	/* Enable MMC1 clocks */
+	CM_PER_REG(CM_PER_MMC1_CLKCTRL) |= CM_PER_MMC1_CLKCTRL_MODULEMODE_ENABLE;
+	while (CM_PER_REG(CM_PER_MMC1_CLKCTRL) & CM_PER_MMC0_CLKCTRL_IDLEST);
+
+	/* Reset the MMC1 Controller */
+	MMC1_REG(SD_SYSCONFIG) = SD_SYSCONFIG_SOFTRESET;
+	while (!(MMC1_REG(SD_SYSSTATUS) & SD_SYSSTATUS_RESETDONE));
+
+	/* Reset the command and data lines */
+	MMC1_REG(SD_SYSCTL) |= SD_SYSCTL_SRA;
+	while (MMC1_REG(SD_SYSCTL) & SD_SYSCTL_SRA);
+
+	/* Configure the MMC1 controller capabilities to enable 3.0 V operating voltage */
+	MMC1_REG(SD_CAPA) |= SD_CAPA_VS30;
+
+	/* Configure SD_IE register to update certain status bits in SD_STAT */
+	MMC1_REG(SD_IE) = SD_IE_BADA_ENABLE | SD_IE_CERR_ENABLE | SD_IE_ACE_ENABLE |
+		SD_IE_DEB_ENABLE | SD_IE_DCRC_ENABLE | SD_IE_DTO_ENABLE | SD_IE_CIE_ENABLE |
+		SD_IE_CEB_ENABLE | SD_IE_CCRC_ENABLE | SD_IE_CIRQ_ENABLE | SD_IE_CREM_ENABLE |
+		SD_IE_CINS_ENABLE | SD_IE_BRR_ENABLE | SD_IE_BWR_ENABLE |
+		SD_IE_TC_ENABLE | SD_IE_CC_ENABLE;
+
+	/* Configure the operating voltage to 3.0 V */
+	MMC1_REG(SD_HCTL) &= ~(SD_HCTL_SDVS);
+	MMC1_REG(SD_HCTL) |= SD_HCTL_SDVS_VS30;
+
+	/* Turn on the bus */
+	MMC1_REG(SD_HCTL) |= SD_HCTL_SDBP;
+	while (!(MMC1_REG(SD_HCTL) & SD_HCTL_SDBP));
+
+	/* Enable the internal clock */
+	MMC1_REG(SD_SYSCTL) |= SD_SYSCTL_ICE;
+
+	/* Configure Clock Frequency Select to 100 KHz */
+	MMC1_REG(SD_SYSCTL) = (MMC1_REG(SD_SYSCTL) & ~SD_SYSCTL_CLKD) | (960 << 6);
+
+	/* Wait for clock to stabilize */
+	while (!(MMC1_REG(SD_SYSCTL) & SD_SYSCTL_ICS));
+
+	/* Configure SD_SYSCONFIG */
+	MMC1_REG(SD_SYSCONFIG) &= ~(SD_SYSCONFIG_CLOCKACTIVITY | SD_SYSCONFIG_SIDLEMODE);
+	MMC1_REG(SD_SYSCONFIG) |= SD_SYSCONFIG_SIDLEMODE_WKUP | SD_SYSCONFIG_ENAWAKEUP_ENABLE |
+		SD_SYSCONFIG_AUTOIDLE_AUTOGATE;
+
+	/* Enable the clock to the eMMC */
+	MMC1_REG(SD_SYSCTL) |= SD_SYSCTL_CEN;
+
+	/* Perform the Initialization Stream as specified in the AM335x TRM, Section 18.3.3.2
+	   "Card Detection, Identification, and Selection" */
+	MMC1_REG(SD_CON) |= SD_CON_INIT;
+	/* Clear the SD_STAT register */
+	MMC1_REG(SD_STAT) = 0xFFFFFFFF;
+	MMC1_REG(SD_ARG) = 0x00000000;
+	MMC1_REG(SD_CMD) = 0x00000000;
+	while (!(MMC1_REG(SD_STAT) & SD_STAT_CC));
+	/* Clear CC flag in SD_STAT */
+	MMC1_REG(SD_STAT) |= SD_STAT_CC;
+	MMC1_REG(SD_CON) &= ~SD_CON_INIT;
+
+	/* Clear the SD_STAT register */
+	MMC1_REG(SD_STAT) = 0xFFFFFFFF;
+
+	/* Enable open-drain mode until we enter Stand-by State as illustrated in the
+	   JEDEC JESD84-A43 Embedded MultiMediaCard Product Standard specification, Table 5 */
+	MMC1_REG(SD_CON) |= SD_CON_OD;
+
+	/* Send CMD0/GO_IDLE_STATE to reset the eMMC on MMC1 interface */
+	arg = 0x00000000;
+	cmd = SD_CMD_CMD0_GO_IDLE_STATE | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+		SD_CMD_CICE_DISABLE | SD_CMD_CCCE_DISABLE | SD_CMD_RSP_TYPE_NO_RESPONSE;
+	if (mmccmd(cmd, arg, resp) == -1)
+		return(-1);
+
+	/* Send CMD1 and poll busy bit in response */
+	do {
+		arg = 0x40FF8000;
+		cmd = SD_CMD_CMD1_SEND_OP_COND | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+			SD_CMD_CICE_DISABLE | SD_CMD_CCCE_DISABLE | SD_CMD_RSP_TYPE_R3;
+		if (mmccmd(cmd, arg, resp) == -1)
+			return(-1);
+	} while (!(MMC1_REG(SD_RSP10) & 0x80000000));
+
+	/* Send CMD2, i.e. ALL_SEND_CID */
+	arg = 0x00000000;
+	cmd = SD_CMD_CMD2_ALL_SEND_CID | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+		SD_CMD_CICE_DISABLE | SD_CMD_CCCE_ENABLE | SD_CMD_RSP_TYPE_R2;
+	if (mmccmd(cmd, arg, resp) == -1)
+		return(-1);
+
+	/* Set RCA of eMMC */
+	mmcrca = 0x3A3A;
+
+	/* Send CMD3 to set the relative card address (RCA) of the eMMC */
+	arg = (mmcrca << 16) & 0xFFFF0000;
+	cmd = SD_CMD_CMD3_SET_RELATIVE_ADDR | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+		SD_CMD_CICE_ENABLE | SD_CMD_CCCE_ENABLE | SD_CMD_RSP_TYPE_R1;
+	if (mmccmd(cmd, arg, resp) == -1)
+		return(-1);
+
+	/* Wait for the eMMC to enter Stand-by State */
+	do {
+		/* Send CMD13 to get the status of the MMC */
+		arg = (mmcrca << 16) & 0xFFFF0000;
+		cmd = SD_CMD_CMD13_SEND_STATUS | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+			SD_CMD_CICE_ENABLE | SD_CMD_CCCE_ENABLE | SD_CMD_RSP_TYPE_R1;
+		if (mmccmd(cmd, arg, resp) == -1)
+			return(-1);
+	} while ((resp[0] & SD_RSP10_R1_CURRENT_STATE) != SD_RSP10_R1_CURRENT_STATE_STANDBY);
+
+	/* Disable open-drain mode */
+	MMC1_REG(SD_CON) &= ~SD_CON_OD;
+
+	/* Send CMD7 to put the eMMC into Transfer State */
+	arg = (mmcrca << 16) & 0xFFFF0000;
+	cmd = SD_CMD_CMD7_SELECT_DESELECT_CARD | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+		SD_CMD_CICE_ENABLE | SD_CMD_CCCE_ENABLE | SD_CMD_RSP_TYPE_R1;
+	if (mmccmd(cmd, arg, resp) == -1)
+		return(-1);
+
+	/* Wait for eMMC to enter Transfer State */
+	do {
+		/* Send CMD13 to get the status of the eMMC */
+		arg = (mmcrca << 16) & 0xFFFF0000;
+		cmd = SD_CMD_CMD13_SEND_STATUS | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+			SD_CMD_CICE_ENABLE | SD_CMD_CCCE_ENABLE | SD_CMD_RSP_TYPE_R1;
+		if (mmccmd(cmd, arg, resp) == -1)
+			return(-1);
+	} while ((resp[0] & SD_RSP10_R1_CURRENT_STATE) != SD_RSP10_R1_CURRENT_STATE_TRANSFER);
+
+	/* Send CMD6 to change bus-width to 8-bits */
+	arg = (3 << 24) | (183 << 16) | (2 << 8);
+	cmd = SD_CMD_CMD6_SWITCH | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+		SD_CMD_CICE_ENABLE | SD_CMD_CCCE_ENABLE | SD_CMD_RSP_TYPE_R1B;
+	if (mmccmd(cmd, arg, resp) == -1)
+		return(-1);
+	while (!(MMC1_REG(SD_STAT) & SD_STAT_TC));
+
+	/* Wait while CMD6 is still in effect, i.e. while eMMC is not in Transfer State */
+	do {
+		arg = (mmcrca << 16) & 0xFFFF0000;
+		cmd = SD_CMD_CMD13_SEND_STATUS | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+			SD_CMD_CICE_ENABLE | SD_CMD_CCCE_ENABLE | SD_CMD_RSP_TYPE_R1;
+		if (mmccmd(cmd, arg, resp) == -1)
+			return(-1);
+	} while ((resp[0] & SD_RSP10_R1_CURRENT_STATE) != SD_RSP10_R1_CURRENT_STATE_TRANSFER);
+
+	/* Configure the MMC1 controller to use an 8-bit data width */
+	MMC1_REG(SD_CON) |= SD_CON_DW8_8BIT;
+
+	/* Send CMD6 to change to high-speed mode */
+	arg = 0x03B90100;
+	cmd = SD_CMD_CMD6_SWITCH | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+		SD_CMD_CICE_ENABLE | SD_CMD_CCCE_ENABLE | SD_CMD_RSP_TYPE_R1B;
+	if (mmccmd(cmd, arg, resp) == -1)
+		return(-1);
+	while (!(MMC1_REG(SD_STAT) & SD_STAT_TC));
+
+	/* Wait while CMD6 is still in effect, i.e. while eMMC is not in Transfer State */
+	do {
+		arg = (mmcrca << 16) & 0xFFFF0000;
+		cmd = SD_CMD_CMD13_SEND_STATUS | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+			SD_CMD_CICE_ENABLE | SD_CMD_CCCE_ENABLE | SD_CMD_RSP_TYPE_R1;
+		if (mmccmd(cmd, arg, resp) == -1)
+			return(-1);
+	} while ((resp[0] & SD_RSP10_R1_CURRENT_STATE) != SD_RSP10_R1_CURRENT_STATE_TRANSFER);
+
+	/* Change the clock frequency to 48 MHz and set the DTO to the maximum value setting */
+	MMC1_REG(SD_SYSCTL) &= ~SD_SYSCTL_DTO;
+	MMC1_REG(SD_SYSCTL) |= SD_SYSCTL_DTO_TCF_2_27;
+	MMC1_REG(SD_SYSCTL) = (MMC1_REG(SD_SYSCTL) & ~SD_SYSCTL_CLKD) | (2 << 6);
+
+	/* Wait for clock to stabilize */
+	while ((MMC1_REG(SD_SYSCTL) & SD_SYSCTL_ICS) != SD_SYSCTL_ICS);
+
+	/* Put the eMMC into Stand-by State */
+	arg = 0x00000000;
+	cmd = SD_CMD_CMD7_SELECT_DESELECT_CARD | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+		SD_CMD_CICE_DISABLE | SD_CMD_CCCE_DISABLE | SD_CMD_RSP_TYPE_NO_RESPONSE;
+	if (mmccmd(cmd, arg, resp) == -1)
+		return(-1);
+
+	/* Wait for the eMMC to enter Stand-by State */
+	do {
+		arg = (mmcrca << 16) & 0xFFFF0000;
+		cmd = SD_CMD_CMD13_SEND_STATUS | SD_CMD_CMD_TYPE_NORMAL | SD_CMD_DP_NO_DATA_PRESENT |
+			SD_CMD_CICE_ENABLE | SD_CMD_CCCE_ENABLE | SD_CMD_RSP_TYPE_R1;
+		if (mmccmd(cmd, arg, resp) == -1)
+			return(-1);
+	} while ((resp[0] & SD_RSP10_R1_CURRENT_STATE) != SD_RSP10_R1_CURRENT_STATE_STANDBY);
+
+	return(0);
+}
+
+int
+mmcRead(int interface, char *buf, int blknum, int blkcnt)
+{
+	return(-1);
+}
+
+int
+mmcWrite(int interface, char *buf, int blknum, int blkcnt)
+{
+	return(-1);
+}
+
+int
+mmcInstalled(int interface)
+{
+	return(1);
+}
diff --git a/ports/beagleboneblack/am335x_mmc.h b/ports/beagleboneblack/am335x_mmc.h
new file mode 100644
index 0000000..aae1b16
--- /dev/null
+++ b/ports/beagleboneblack/am335x_mmc.h
@@ -0,0 +1,13 @@
+/*
+ * Copyright (c) 2015 Jarielle Catbagan <jcatbagan93 at gmail.com>
+ *
+ * The license and distribution terms for this file may be
+ * found in the file LICENSE in this distribution or at
+ * http://www.apache.org/licenses/LICENSE-2.0
+ */
+
+int mmc(int argc, char *argv[]);
+int mmcInit(int interface, int verbose);
+int mmcRead(int interface, char *buf, int blknum, int blkcnt);
+int mmcWrite(int interface, char *buf, int blknum, int blkcnt);
+int mmcInstalled(int interface);
-- 
2.5.0




More information about the umon-devel mailing list