Mercurial > ~darius > hgwebdir.cgi > stm32temp
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); +} + + +