diff sd.c @ 49:ace431a0d0f5

Add SDIO code poached from STM. Use FatFS to read from SD card. LFN doesn't work reliably so it's disabled for now.
author Daniel O'Connor <darius@dons.net.au>
date Wed, 03 Apr 2013 23:34:20 +1030
parents
children d7207a9d3c3b
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/sd.c	Wed Apr 03 23:34:20 2013 +1030
@@ -0,0 +1,298 @@
+/*
+ * SD card code
+ *
+ * Copyright (c) 2013
+ *      Daniel O'Connor <darius@dons.net.au>.  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.
+ *
+ * THIS SOFTWARE IS PROVIDED BY AUTHOR 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 AUTHOR 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 <assert.h>
+#include <inttypes.h>
+#include <stdio.h>
+#include <stdint.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include "ff.h"
+#include "stm32f10x.h"
+#include "sd.h"
+#include "stm32_eval_sdio_sd.h"
+
+#define FAT_YEAR(d)	((((d) & 0xfe00) >> 9) + 1980)
+#define FAT_MONTH(d)	(((d) & 0x01e0) >> 5)
+#define FAT_DAY(d)	(((d) & 0x001f) >> 0)
+#define FAT_HOUR(t)	(((t) & 0xf800) >> 11)
+#define FAT_MIN(t)	(((t) & 0x07e0) >> 5)
+#define FAT_SEC(t)	(((t) & 0x001f) << 1)
+
+static FATFS fsh;
+
+static void sd_domount(void);
+
+void
+sd_init(void) {
+    sd_domount();
+}
+
+void
+sd_domount(void) {
+    SD_Error err;
+    FRESULT fserr;
+
+    if ((err = SD_Init()) != SD_OK) {
+	printf("Failed init: %d\n", err);
+	return;
+    }
+    
+    if ((fserr = f_mount(0, &fsh)) != FR_OK) {
+	printf("Unable to mount FS\n");
+	return;
+    }
+}
+
+void
+sd_cmd(int argc, char **argv) {
+    SD_Error err;
+
+    if (argc < 1) {
+	printf("No command specified\n");
+	return;
+    }
+    
+    if (!strcmp(argv[0], "info")) {
+	SD_CardInfo cardinfo;
+	if ((err = SD_GetCardInfo(&cardinfo)) != SD_OK) {
+	    printf("Get card info failed: %d\r\b", err);
+	} else {
+	    printf("Mfg ID %" PRIu8 ", OEM %" PRIu16 ", Prod1 %" PRIu32 ", Prod2 %" PRIu8 ", Rev %" PRIu8 ", SN %" PRIu32 ", Mfg %" PRIu16 "\n",
+		   cardinfo.SD_cid.ManufacturerID, cardinfo.SD_cid.OEM_AppliID,
+		   cardinfo.SD_cid.ProdName1, cardinfo.SD_cid.ProdName2,
+		   cardinfo.SD_cid.ProdRev, cardinfo.SD_cid.ProdSN,
+		   cardinfo.SD_cid.ManufactDate);
+	    printf("Capacity %" PRIu32 ", blocksize %" PRIu32 "\n", cardinfo.CardCapacity, cardinfo.CardBlockSize);
+	}
+    } else if (!strcmp(argv[0], "mount")) {
+	sd_domount();
+    } else if (!strcmp(argv[0], "read")) {
+	uint32_t addr;
+	uint8_t *buf;
+	uint32_t i;
+	
+	if (argc != 2) {
+	    printf("Block to read not specified\n");
+	    return;
+	}
+	addr = atoi(argv[1]);
+
+	if ((buf = malloc(SD_BLOCK_SIZE)) == NULL) {
+	    printf("Unable to allocate buffer\n");
+	    return;
+	}
+	
+	/* If passed values not modulo 512 it hangs */
+	if ((err = SD_ReadBlock(buf, addr * SD_BLOCK_SIZE, SD_BLOCK_SIZE)) != SD_OK) {
+	    printf("Read block returned %d\n", err);
+	    goto read_exit;
+	}
+	
+#ifdef SD_DMA_MODE
+	if ((err = SD_WaitReadOperation()) != SD_OK) {
+	    printf("Wait returned %d\n", err);
+	    goto read_exit;
+	}
+#endif
+	while(SD_GetStatus() != SD_TRANSFER_OK)
+	    ;
+
+	for (i = 0; i < SD_BLOCK_SIZE; i += 16) {
+	    printf("0x%04lx: %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x\n", i,
+		   buf[i + 0], buf[i + 1], buf[i + 2], buf[i + 3], buf[i + 4], buf[i + 5], buf[i + 6], buf[i + 7],
+		   buf[i + 8], buf[i + 9], buf[i + 10], buf[i + 11], buf[i + 12], buf[i + 13], buf[i + 14], buf[i + 15]);
+	}
+	
+      read_exit:
+	free(buf);
+    } else if (!strcmp(argv[0], "erase")) {
+	uint32_t addr;
+	
+	if (argc != 2) {
+	    printf("Block to erase not specified\n");
+	    return;
+	}
+	addr = atoi(argv[1]);
+	
+	if ((err = SD_Erase(addr * SD_BLOCK_SIZE, (addr + 1) * SD_BLOCK_SIZE)) != SD_OK) {
+	    printf("Erase block returned %d\n", err);
+	    goto read_exit;
+	}
+	
+#ifdef SD_DMA_MODE
+	if ((err = SD_WaitReadOperation()) != SD_OK) {
+	    printf("Wait returned %d\n", err);
+	    goto read_exit;
+	}
+#endif
+	while(SD_GetStatus() != SD_TRANSFER_OK)
+	    ;
+	printf("Erased OK\n");
+    } else if (!strcmp(argv[0], "ls")) {
+	FRESULT fserr;
+	DIR d;
+	FILINFO fno;
+	char *fn;
+	char label[12];
+	DWORD sn;
+	
+	if ((fserr = f_getlabel("", label, &sn)) != FR_OK) {
+	    printf("Unable to read label\n");
+	} else {
+	    printf("Label: %s Serial: %d\n", label, sn);
+	}
+	
+	if ((fserr = f_opendir(&d, argc == 1 ? "\\" : argv[1])) != FR_OK) {
+	    printf("Unable to opendir: %d\n", fserr);
+	    return;
+	}
+	
+	for (;;) {
+	    fserr = f_readdir(&d, &fno);
+	    if (fserr != FR_OK) {
+		printf("Readdir failed: %d\n", fserr);
+		break;
+	    }
+	    if (fno.fname[0] == '\0')
+		break;
+#if _USE_LFN
+            fn = *fno.lfname ? fno.lfname : fno.fname;
+#else
+            fn = fno.fname;
+#endif
+	    printf("%-12s %5u %d/%d/%d %02d:%02d:%02d %s%s%s%s%s\n", fn, fno.fsize,
+		   FAT_YEAR(fno.fdate), FAT_MONTH(fno.fdate), FAT_DAY(fno.fdate),
+		   FAT_HOUR(fno.ftime), FAT_MIN(fno.ftime), FAT_SEC(fno.ftime),
+		   fno.fattrib & AM_DIR ? "D" : "", fno.fattrib & AM_RDO ? "R" : "",
+		   fno.fattrib & AM_HID ? "H" : "", fno.fattrib & AM_SYS ? "S" : "", 
+		   fno.fattrib & AM_ARC ? "A" : "");
+	}
+    } else if (!strcmp(argv[0], "cat")) {
+	FRESULT fserr;
+	FIL f;
+	char buf[128], *tmp;
+	
+	if (argc != 2) {
+	    printf("No file given\n");
+	    return;
+	}
+	
+	if ((fserr = f_open(&f, argv[1], FA_READ)) != FR_OK) {
+	    printf("Failed to open file: %d\n", fserr);
+	    return;
+	}
+	
+	while ((tmp = f_gets(buf, sizeof(buf), &f)) != NULL) {
+	    puts(tmp);
+	}
+    } else {
+	printf("Unknown command\n");
+    }
+}
+
+void
+SD_LowLevel_DeInit(void) {
+    GPIO_InitTypeDef  GPIO_InitStructure;
+  
+    /*!< Disable SDIO Clock */
+    SDIO_ClockCmd(DISABLE);
+  
+    /*!< Set Power State to OFF */
+    SDIO_SetPowerState(SDIO_PowerState_OFF);
+
+    /*!< DeInitializes the SDIO peripheral */
+    SDIO_DeInit();
+  
+    /*!< Disable the SDIO AHB Clock */
+    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_SDIO, DISABLE);
+
+    /*!< Configure PC.08, PC.09, PC.10, PC.11, PC.12 pin: D0, D1, D2, D3, CLK pin */
+    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9 | GPIO_Pin_10 | GPIO_Pin_11 | GPIO_Pin_12;
+    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
+    GPIO_Init(GPIOC, &GPIO_InitStructure);
+
+    /*!< Configure PD.02 CMD line */
+    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
+    GPIO_Init(GPIOD, &GPIO_InitStructure);
+}
+
+void
+SD_LowLevel_Init(void) {
+    GPIO_InitTypeDef  GPIO_InitStructure;
+
+    /*!< GPIOC and GPIOD Periph clock enable */
+    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD, ENABLE);
+
+    /*!< Configure PC.08, PC.09, PC.10, PC.11, PC.12 pin: D0, D1, D2, D3, CLK pin */
+    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9 | GPIO_Pin_10 | GPIO_Pin_11 | GPIO_Pin_12;
+    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
+    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
+    GPIO_Init(GPIOC, &GPIO_InitStructure);
+
+    /*!< Configure PD.02 CMD line */
+    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
+    GPIO_Init(GPIOD, &GPIO_InitStructure);
+
+#ifdef SD_HAVE_DETECT
+    RCC_APB2PeriphClockCmd(SD_DETECT_GPIO_CLK, ENABLE);
+
+    /*!< Configure SD_SPI_DETECT_PIN pin: SD Card detect pin */
+    GPIO_InitStructure.GPIO_Pin = SD_DETECT_PIN;
+    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
+    GPIO_Init(SD_DETECT_GPIO_PORT, &GPIO_InitStructure);
+#endif
+  
+    /*!< Enable the SDIO AHB Clock */
+    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_SDIO, ENABLE);
+
+#ifdef SD_DMA_MODE
+    /*!< Enable the DMA2 Clock */
+    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA2, ENABLE);
+#endif
+}
+
+void
+SD_LowLevel_DMA_RxConfig(uint32_t *BufferDST, uint32_t BufferSize) {
+    printf("SD_LowLevel_DMA_RxConfig unimplemented\n");
+    abort();
+}
+
+void
+SD_LowLevel_DMA_TxConfig(uint32_t *BufferSRC, uint32_t BufferSize) {
+    printf("SD_LowLevel_DMA_TxConfig unimplemented\n");
+    abort();
+}
+
+uint32_t SD_DMAEndOfTransferStatus(void) {
+  return (uint32_t)DMA_GetFlagStatus(DMA2_FLAG_TC4);
+}
+
+
+