本文章介紹使用Raspberry Pi Pico PIO程式製作SD memory Card 4-bit wide bus and 1-bit data bus驅動程式。並與FatFs檔案系統結合,測試在SD card上進行檔案讀寫。
SD memory card主要分為Command, data read and data write三個部份。每個部份我們製作一個相對應的State machine:CMD, READ and WRITE。
一、CMD PIO state machine:
command為48bits:第一個byte:01開頭為HOST to Card, 00開頭為Card to Host, 其餘6bits為Command ID。接下來4 bytes為parameters。最後一個byte為CRC7and 1 bit(stop bit)。response分成0,48 and 136 bits三種。
CMD state machine使用1 CMD line and 1 CLK line:
CMD pin:IN/OUT/SET/JMP。 CLK pin:side-set
詳細程式碼附於文章末尾。
二、READ State machine:
Card output(HOST read): 每個data block為512 bytes後面跟著CRC16。4-bit wide bus DAT line分別需要個別的CRC16 data。
READ state machine: 1 or 4 DAT line, 1 CLK line。
DAT pin(s): IN/SET/JMP, CLK pin:side-set
三、WRITE state machine:
Card input(HOST output):host寫入一個data block後面跟CRC16。Card於3 clocks後在DAT0 (DAT1~3 don't care) 送出CRC status(010為positive, 101為negative)。
WRITE state machine: 1~4 DAT line, 1 CLK line。
DAT pin(s): IN/OUT/SET/JMP, CLK pin:side-set
四、HOST read data:
- 啟動CMD state machine, host向card 送出data read command(CMD17 or CMD18),收到rsponse後,停止state machine(disable)。
- 啟動READ state machine讀取資料。
- CMD與READ state machine使用自己的CLK pin。因此在CMD state machine換成READ state machine時,clock會先暫停。根據SD specification "clock control"章節說明是允許的。
五、HOST DATA write:
- 啟動CMD state machine, host向card 送出data write command(CMD24 or CMD25),收到rsponse後,停止state machine(disable)。
- 啟動WRITE state machine write data block and crc。接著讀取crc status。
參考資料為:
SD Specifications Part 1 Physical Layer Specification(https://www.sdcard.org/)
成果展示:
程式碼:
- sdio_mem_card.c
#include <stdio.h> #include "pico/stdlib.h" #include "hardware/dma.h" #include "hardware/pio.h" #include "sdio_mem_card.h" #include "sdio_mem_card.pio.h" #include "hardware/clocks.h" #include "crc7.h" #include "crc-itu-t.h" #include "string.h" #include "inttypes.h" /* void pio_irq_read_data() { if (pio_interrupt_get(SDIO_MEM_PIO, 0)) { pio_interrupt_clear(SDIO_MEM_PIO, 0); printf("irq 0:\n"); } if (pio_interrupt_get(SDIO_MEM_PIO, 1)) { pio_interrupt_clear(SDIO_MEM_PIO, 1); printf("irq:1\n"); } } */ /*! * \brief sdio memory card pio initialize * \param pio: pio number * \param sm state machine * \param cmd_pin command pin * \param clk_pin CLK pin * \param data_pin_base data 0~4 pins(D0~D3) * \param clk_int Integer part of the divisor * \param clk_frac – Fractional part in 1/256ths */ void sdio_pio_init(sd_memory_card_t* pSD, PIO pio, uint cmd_pin, uint clk_pin, uint data_pin_base, uint16_t clk_int, uint8_t clk_frac) { uint offset=0; pio_sm_config c; static uint8_t data_width; data_width = (pSD->is_wide_bus ? 4:1); pio_gpio_init(pio, cmd_pin); pio_gpio_init(pio, clk_pin); gpio_pull_up(cmd_pin); for (int i=0; i < 4; i++) { pio_gpio_init(pio, data_pin_base+i); gpio_pull_up(data_pin_base+i); } pio_enable_sm_mask_in_sync(pio, (1<<SDIO_CMD_SM)|(1<<SDIO_DATA_READ_SM)|(1<<SDIO_DATA_WRITE_SM)); //== SDIO READ SM == pio_sm_config cr; if (data_width == 1) { pSD->sd_data_rx_offset = pio_add_program(pio, &sdio_1_bit_rx_program); cr = sdio_1_bit_rx_program_get_default_config(pSD->sd_data_rx_offset); } else { pSD->sd_data_rx_offset = pio_add_program(pio, &sdio_4_bit_rx_program); cr = sdio_4_bit_rx_program_get_default_config(pSD->sd_data_rx_offset); } pio_sm_set_consecutive_pindirs(pio, SDIO_DATA_READ_SM, data_pin_base, 4, false); pio_sm_set_consecutive_pindirs(pio, SDIO_DATA_READ_SM, clk_pin, 1, true); //pio_sm_set_consecutive_pindirs(pio, SDIO_DATA_READ_SM, cmd_pin, 1, false); sm_config_set_in_pins(&cr, data_pin_base); sm_config_set_sideset_pins(&cr, clk_pin); sm_config_set_set_pins(&cr, data_pin_base,data_width); sm_config_set_jmp_pin(&cr, data_pin_base); sm_config_set_out_shift(&cr, false, true, 32); sm_config_set_in_shift(&cr, false, true, 32); sm_config_set_clkdiv_int_frac(&cr, clk_int,clk_frac); pio_sm_init(pio, SDIO_DATA_READ_SM, pSD->sd_data_rx_offset, &cr); //== SDIO WRITE SM == pio_sm_config cw; if (data_width == 1) { pSD->sd_data_tx_offset = pio_add_program(pio, &sdio_1_bit_tx_program); cw = sdio_1_bit_rx_program_get_default_config(pSD->sd_data_tx_offset); } else { pSD->sd_data_tx_offset = pio_add_program(pio, &sdio_4_bit_tx_program); cw = sdio_4_bit_rx_program_get_default_config(pSD->sd_data_tx_offset); } pio_sm_set_consecutive_pindirs(pio, SDIO_DATA_WRITE_SM, data_pin_base, 4, true); pio_sm_set_consecutive_pindirs(pio, SDIO_DATA_WRITE_SM, clk_pin, 1, true); //pio_sm_set_consecutive_pindirs(pio, SDIO_DATA_WRITE_SM, cmd_pin, 1, false); sm_config_set_in_pins(&cw, data_pin_base); sm_config_set_out_pins(&cw, data_pin_base, data_width); sm_config_set_sideset_pins(&cw, clk_pin); sm_config_set_set_pins(&cw, data_pin_base,data_width); sm_config_set_jmp_pin(&cw, data_pin_base); sm_config_set_out_shift(&cw, false, true, 32); sm_config_set_in_shift(&cw, false, true, 32); sm_config_set_clkdiv_int_frac(&cw, clk_int,clk_frac); pio_sm_init(pio, SDIO_DATA_WRITE_SM, pSD->sd_data_tx_offset, &cw); //== SDIO command SM == offset = pio_add_program(pio, &sdio_mem_cmd_program); c = sdio_mem_cmd_program_get_default_config(offset); pio_sm_set_consecutive_pindirs(pio, SDIO_CMD_SM, cmd_pin, 1, false); pio_sm_set_consecutive_pindirs(pio, SDIO_CMD_SM, clk_pin, 1, true); pio_sm_set_consecutive_pindirs(pio, SDIO_CMD_SM, data_pin_base, 4, false); sm_config_set_in_pins(&c, cmd_pin); sm_config_set_out_pins(&c, cmd_pin, 1); sm_config_set_set_pins(&c, cmd_pin,1); sm_config_set_sideset_pins(&c, clk_pin); sm_config_set_jmp_pin(&c, cmd_pin); sm_config_set_out_shift(&c, false, true, 32); sm_config_set_in_shift(&c, false, true, 32); sm_config_set_clkdiv_int_frac(&c, clk_int,clk_frac); pio_sm_init(pio, SDIO_CMD_SM, offset, &c); // initial state machine but not run pio_sm_set_enabled(pio, SDIO_CMD_SM, false); pio_sm_set_enabled(pio, SDIO_DATA_READ_SM, false); pio_sm_set_enabled(pio, SDIO_DATA_WRITE_SM, false); //** for test only //uint pio_irq = pio_get_index(pio)? PIO1_IRQ_0:PIO0_IRQ_0; //pio_set_irq0_source_enabled(pio, pis_interrupt0, true); //pio_set_irq0_source_enabled(pio, pis_interrupt1, true); //irq_add_shared_handler(pio_irq, pio_irq_read_data, PICO_SHARED_IRQ_HANDLER_DEFAULT_ORDER_PRIORITY); //irq_set_enabled(pio_irq, true); #if _DMA_TRANS_ /* DMA */ uint pio_base = (pio==pio0)?PIO0_BASE:PIO1_BASE; // ==== Write DMA === pSD->dma_write_channel = dma_claim_unused_channel(true); dma_channel_config dcw = dma_channel_get_default_config(pSD->dma_write_channel); channel_config_set_write_increment(&dcw, false); channel_config_set_read_increment(&dcw, true); channel_config_set_bswap(&dcw, true); channel_config_set_dreq(&dcw, pio_get_dreq(pio, SDIO_DATA_WRITE_SM, true)); channel_config_set_transfer_data_size(&dcw, DMA_SIZE_32); //DMA_SIZE_8,16,32 dma_channel_configure(pSD->dma_write_channel, &dcw, (void*) (pio_base+PIO_TXF2_OFFSET), // SDIO_DATA_WRITE_SM = 2 NULL, MAX_RW_BLOCKS_TRANS>> DMA_SIZE_32, false); //DMA_SIZE_8 or 16 or 32 //==== READ DMA === pSD->dma_read_channel = dma_claim_unused_channel(true); dma_channel_config dcr = dma_channel_get_default_config(pSD->dma_read_channel); channel_config_set_write_increment(&dcr, true); channel_config_set_read_increment(&dcr, false); channel_config_set_bswap(&dcr, true); channel_config_set_dreq(&dcr, pio_get_dreq(pio, SDIO_DATA_READ_SM, false)); channel_config_set_transfer_data_size(&dcr, DMA_SIZE_32); //DMA_SIZE_8,16,32 dma_channel_configure(pSD->dma_read_channel, &dcr, NULL, (void*) (pio_base+PIO_RXF1_OFFSET), // SDIO_DATA_READ_SM = 1 MAX_RW_BLOCKS_TRANS>> DMA_SIZE_32, false); //DMA_SIZE_8 or 16 or 32 /* DMA */ #endif } /*! * \brief send SDIO CMD and recv RESPONSE * \param cmd_id command ID * \param arg args size 4 bytes(32bits) * \param resp RESPONSE Register (48 bits / 136 bites) * \param resp_bits response register length */ uint8_t sdio_send_recv_cmd(uint8_t cmd_id, uint8_t *arg, uint32_t *resp, uint8_t resp_bits) { uint8_t cmd; uint32_t cmd_buf[2]; uint8_t crc = 0; cmd = cmd_id | 0x40; // start 0b01 send from host to card // crc7 crc = crc7_table[crc ^ cmd]; for (int i=0; i < count_of(arg); i++) crc = crc7_table[crc ^ arg[i]]; // crc7 if (resp_bits > 0) resp_bits--; // put to fifo: 1byte command len,1byte command ID, 4bytes arg ,1byte crc ,1bytes RESP cmd_buf[0] = 47 << 24 | cmd << 16 | arg[0] << 8 | arg[1]; cmd_buf[1] = arg[2] << 24 | arg[3] << 16 | (crc | 0x01) << 8 | (resp_bits); memset(resp, 0, sizeof(resp)); pio_sm_clear_fifos(SDIO_MEM_PIO, SDIO_CMD_SM); pio_sm_exec(SDIO_MEM_PIO, SDIO_CMD_SM,pio_encode_set(pio_pins,1)); pio_sm_exec(SDIO_MEM_PIO, SDIO_CMD_SM,pio_encode_set(pio_pindirs,1)); //Output shift counter is reset to 0 by this operation, i.e. full) pio_sm_exec(SDIO_MEM_PIO, SDIO_CMD_SM, pio_encode_mov(pio_osr, pio_null)); for (int i = 0; i < count_of(cmd_buf); i++) { pio_sm_put_blocking(SDIO_MEM_PIO, SDIO_CMD_SM, cmd_buf[i]); } pio_sm_exec(SDIO_MEM_PIO, SDIO_CMD_SM, pio_encode_jmp(sdio_mem_cmd_offset_sdio_cmd_start)); pio_sm_set_enabled(SDIO_MEM_PIO, SDIO_CMD_SM, true); switch (resp_bits) { case 47: resp[0] = pio_sm_get_blocking(SDIO_MEM_PIO, SDIO_CMD_SM); resp[1] = pio_sm_get_blocking(SDIO_MEM_PIO, SDIO_CMD_SM); break; case 135: for (int i=0; i < 5; i++) { resp[i] = pio_sm_get_blocking(SDIO_MEM_PIO, SDIO_CMD_SM); } break; case 0: resp[0] = pio_sm_get_blocking(SDIO_MEM_PIO, SDIO_CMD_SM); //dummy break; } pio_sm_set_enabled(SDIO_MEM_PIO, SDIO_CMD_SM, false); return SD_OK; } bool format_sdio_response_136(uint32_t *resp_32, uint32_t *content) { if ((resp_32[0]>>24) != 0x3f){ printf("get CID/CSD error\n"); return false;} for (int i=0; i < 3;i++) { content[i] = resp_32[i] << 8 | resp_32[i+1] >> 24; } content[3] = (resp_32[3] >> 24) << 8 | resp_32[4]&0xff; } bool format_sdio_response_48(uint32_t *resp_32, uint8_t *resp_8) { resp_8[0] = resp_32[0] >> 24; resp_8[1] = (resp_32[0]&0x00ff0000) >> 16; resp_8[2] = (resp_32[0]&0x0000ff00) >> 8; resp_8[3] = resp_32[0] &0x000000ff; resp_8[4] = (resp_32[1]&0x0000ff00) >> 8; resp_8[5] = (resp_32[1] &0x000000ff) ; uint8_t crc = 0; for (int i=0; i < 5; i++) crc = crc7_table[crc ^ resp_8[i]]; if (crc != (resp_8[5]&0xfe)) return false; return true; } uint32_t get_sd_card_status(sd_memory_card_t* pSD) { uint8_t arg[4]; uint32_t resp[2]; arg[0] = pSD->rca[0]; arg[1] = pSD->rca[1]; arg[2] = 0; arg[3] = 0; sdio_send_recv_cmd(CMD13, arg, resp, 48); return (resp[0])<<8 | (resp[1]>>8); } uint32_t format_R1_response(uint32_t *resp) { return (resp[0]&0x00ffffff)<<8 | (resp[1]>>8); } /*! * \brief initialize sd card * \param wide_bus true:4bit, false 1bit */ uint8_t sd_memory_card_init(sd_memory_card_t* pSD, uint8_t wide_bus) { uint8_t arg[4]; uint32_t resp[5]; uint8_t resp_bytes[17]; uint8_t ret_val; uint32_t CSID[5]; uint8_t ret = SD_OK; pSD->SD_initialized=false; pio_clear_instruction_memory(SDIO_MEM_PIO); pSD->is_wide_bus = (wide_bus)? true:false; sdio_pio_init(pSD, SDIO_MEM_PIO, SDIO_CMD_PIN, SDIO_CLK_PIN, SDIO_D0_PIN, 1, 0); sleep_ms(2); memset(arg, 0, sizeof(arg)); sdio_send_recv_cmd(CMD0, arg, resp, 0); // CMD0: REEST for (int i=0; i < 10; i++) { memset(arg, 0, sizeof(arg)); arg[2]=1; arg[3]=0xaa; sdio_send_recv_cmd(CMD8, arg, resp, 48); //CMD8: SEND_IF_CONF if(format_sdio_response_48(resp, resp_bytes)) { if (resp_bytes[4]== arg[3]) printf("cmd8 ok:%d\n",i); ret = SD_OK; break; } else { ret = SD_INIT_FAILURE; } } absolute_time_t timeout = make_timeout_time_ms(1000); do { memset(arg, 0, sizeof(arg)); sdio_send_recv_cmd(CMD55, arg, resp, 48); // next is APP_CMD arg[0] = 0x40; // HCS, XPC, S18R arg[1] = 0x10; sdio_send_recv_cmd(ACMD41, arg, resp, 48); // R3 format_sdio_response_48(resp, resp_bytes); // crc is always 1111111 //for (int i=0; i < 6; i++) { printf("%02x ",resp_bytes[i]);} printf("\n"); if (resp_bytes[0]==0x3f && (resp_bytes[1] & 0x80)) { ret = SD_OK; break; } else { ret = SD_TIME_OUT; } } while (absolute_time_diff_us(get_absolute_time(), timeout) > 0); // 0x80 initialize completely. R3[1]: Busy CCS UHS-II Reserved(0000) S18A if(ret != SD_OK) return ret; memset(arg, 0, sizeof(arg)); if (resp_bytes[1] & 0x01) { sdio_send_recv_cmd(CMD11, arg, resp, 0); // voltage switch } sdio_send_recv_cmd(CMD2, arg, resp, 136); // CID(R2) sdio_send_recv_cmd(CMD3, arg, resp, 48); if (format_sdio_response_48(resp, resp_bytes)) { pSD->rca[0] = resp_bytes[1]; pSD->rca[1] = resp_bytes[2]; printf("rca received:%02x, %02x\n", pSD->rca[0], pSD->rca[1]); } else { printf("CMD3 response crc error\n"); for (int i=0; i < 6; i++) { printf("%02x ",resp_bytes[i]); } printf("\n"); return SD_CRC_ERROR; } //get CID regisger === memset(arg,0,sizeof(arg)); arg[0] = pSD->rca[0]; arg[1] = pSD->rca[1]; for (int i=0; i<10; i++) { if (sdio_send_recv_cmd(CMD10, arg, resp, 136)== SD_OK) { ret = SD_OK; break; // CID (R2) } else { ret = SD_INIT_FAILURE; } } format_sdio_response_136(resp, CSID); pSD->pnm[0]= (uint8_t)(CSID[0]); pSD->pnm[1]= (uint8_t)((CSID[1]>>24)); pSD->pnm[2]= (uint8_t)((CSID[1]>>16)); pSD->pnm[3]= (uint8_t)((CSID[1])>>8); pSD->pnm[4]= (uint8_t)(CSID[1]); pSD->pnm[5]= '\0'; pSD->prv = (uint8_t)(CSID[2]>>24); pSD->psn = (CSID[2]) << 8 | CSID[3]>>24; // === get CID end //===get CSD register === memset(arg,0,sizeof(arg)); arg[0] = pSD->rca[0]; arg[1] = pSD->rca[1]; for (int i=0; i<10; i++) { if (sdio_send_recv_cmd(CMD9, arg, resp, 136)== SD_OK) { ret = SD_OK; break; // CSD (R2) }else { ret = SD_INIT_FAILURE; } sleep_ms(2); } format_sdio_response_136(resp, CSID); printf("CSD:%08x,%08x,%08x,%08x\n", CSID[0],CSID[1],CSID[2],CSID[3]); pSD->block_size = 1<<((CSID[1]>>16) & 0xf) ; pSD->total_blocks = (CSID[1]&0x1f) << 16 |CSID[2] >> 16; printf("block_size:%d, total_blocks:%u, capacity:%d KB\n", pSD->block_size, pSD->total_blocks, (pSD->total_blocks/(1024/pSD->block_size)) ); // ==get CSD end=== memset(arg,0,sizeof(arg)); arg[0] = pSD->rca[0]; arg[1] = pSD->rca[1]; sdio_send_recv_cmd(CMD7, arg, resp, 48); // R1b, SELECT_CARD //sdio_send_recv_cmd(CMD42, arg, resp, 48); //R1 //format_sdio_response_48(resp, resp_bytes); arg[0] = pSD->rca[0]; arg[1] = pSD->rca[1]; sdio_send_recv_cmd(CMD55, arg, resp, 48); // next is APP_CMD if (wide_bus) arg[3]=0x02; // 4-bit sdio_send_recv_cmd(ACMD6, arg, resp, 48); //R1, set bus width //format_sdio_response_48(resp, resp_bytes); //printf("status:%0x\n", resp_bytes[4]); sdio_send_recv_cmd(CMD13, arg, resp, 48); format_sdio_response_48(resp, resp_bytes); if (resp_bytes[3] == 0x09 && resp_bytes[4]==0x00) { uint8_t test[512]; sd_read_single_block(pSD, (uint32_t*)test, pSD->total_blocks); if (get_sd_card_status(pSD)==0x0900) { pSD->SD_initialized=true; printf("SD card initialize successfully\n"); } } return ret; } uint8_t sd_stop_trans() { uint8_t arg[4]; uint32_t resp[2]; uint8_t resp_bytes[5]; memset(arg, 0, sizeof(arg)); sdio_send_recv_cmd(CMD12, arg, resp, 48); //R1b, return SD_OK; } uint16_t calc_1bit_crc16(uint32_t *buff, uint16_t words_per_block){ uint16_t crc =0; uint32_t temp; for (int i= 0; i< words_per_block; i++) { temp = __builtin_bswap32(buff[i]); crc = crc_itu_t_table[(crc >> 8) ^ (uint8_t)(temp>>24)] ^ (crc << 8); crc = crc_itu_t_table[(crc >> 8) ^ (uint8_t)(temp>>16)] ^ (crc << 8); crc = crc_itu_t_table[(crc >> 8) ^ (uint8_t)(temp>>8)] ^ (crc << 8); crc = crc_itu_t_table[(crc >> 8) ^ (uint8_t)(temp)] ^ (crc << 8); } return crc; } uint64_t calc_4bit_crc16(uint32_t *data, uint32_t num_words) { uint64_t crc = 0; uint32_t *end = data + num_words; while (data < end) { for (int unroll = 0; unroll < 4; unroll++) { // Each 32-bit word contains 8 bits per line. // Reverse the bytes because SDIO protocol is big-endian. uint32_t data_in = __builtin_bswap32(*data++); // Shift out 8 bits for each line uint32_t data_out = crc >> 32; crc <<= 32; // XOR outgoing data to itself with 4 bit delay data_out ^= (data_out >> 16); // XOR incoming data to outgoing data with 4 bit delay data_out ^= (data_in >> 16); // XOR outgoing and incoming data to accumulator at each tap uint64_t xorred = data_out ^ data_in; crc ^= xorred; crc ^= xorred << (5 * 4); crc ^= xorred << (12 * 4); } } return crc; } uint8_t pio_sd_read_blocks(sd_memory_card_t* pSD, uint32_t *buff, uint32_t count, uint32_t* crc) { uint32_t read_cycles; uint read_start_addr; uint8_t ret=SD_OK; uint16_t block_data_bytes; uint8_t crc_len; if (pSD->is_wide_bus) { block_data_bytes=(512+2*4); read_cycles=block_data_bytes*2-1; read_start_addr = pSD->sd_data_rx_offset+sdio_4_bit_rx_offset_rx_start; crc_len=2; } else { block_data_bytes = (512+2); read_cycles = block_data_bytes*8-1; read_start_addr = pSD->sd_data_rx_offset+sdio_1_bit_rx_offset_rx_start; crc_len=1; } pio_sm_clear_fifos(SDIO_MEM_PIO, SDIO_DATA_READ_SM); pio_sm_restart(SDIO_MEM_PIO, SDIO_DATA_READ_SM); pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_READ_SM,pio_encode_set(pio_pindirs, 0)); //pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_READ_SM, pio_encode_mov(pio_isr, pio_null)); pio_sm_put_blocking(SDIO_MEM_PIO, SDIO_DATA_READ_SM, read_cycles); pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_READ_SM, pio_encode_out(pio_x, 32)); pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_READ_SM, pio_encode_jmp(read_start_addr)); pio_sm_set_enabled(SDIO_MEM_PIO, SDIO_DATA_READ_SM, true); for (int block_idx=0; block_idx < count; block_idx++) { #if _DMA_TRANS_ dma_channel_set_trans_count(pSD->dma_read_channel, SDIO_WORDS_PER_BLOCK, false); dma_channel_set_write_addr(pSD->dma_read_channel, buff+block_idx*SDIO_WORDS_PER_BLOCK, false); dma_channel_start(pSD->dma_read_channel); dma_channel_wait_for_finish_blocking(pSD->dma_read_channel); dma_channel_set_trans_count(pSD->dma_read_channel, crc_len, false); dma_channel_set_write_addr(pSD->dma_read_channel, crc+crc_len*block_idx, false); dma_channel_start(pSD->dma_read_channel); dma_channel_wait_for_finish_blocking(pSD->dma_read_channel); #else for (int i=0; i < SDIO_WORDS_PER_BLOCK; i++) { buff[i+SDIO_WORDS_PER_BLOCK*block_idx] = __builtin_bswap32(pio_sm_get_blocking(SDIO_MEM_PIO, SDIO_DATA_READ_SM)); } for (int c=0; c < crc_len; c++) { crc[c+crc_len*block_idx] = __builtin_bswap32(pio_sm_get_blocking(SDIO_MEM_PIO, SDIO_DATA_READ_SM)); } #endif } pio_sm_set_enabled(SDIO_MEM_PIO, SDIO_DATA_READ_SM, false); return SD_OK; } uint8_t pio_sd_write_blocks(sd_memory_card_t* pSD, uint32_t *buff, uint32_t count) { uint32_t write_cycles; uint write_start_addr; uint8_t ret=SD_OK; uint16_t block_data_bytes; uint16_t crc; uint32_t start, end; uint32_t lines=1; if (pSD->is_wide_bus) { block_data_bytes=(4+512+2*4); // start 0xfffffff0, end 0xffffffff write_cycles=block_data_bytes*2; write_start_addr = pSD->sd_data_tx_offset+sdio_4_bit_tx_offset_tx_start; start=0xfffffff0; end = 0xffffffff; lines=4; } else { block_data_bytes = (4+512+2); // start 0xfffffffe, write_cycles = block_data_bytes*8; write_start_addr = pSD->sd_data_tx_offset+sdio_1_bit_tx_offset_tx_start; start = 0xfffffffe; end =0xffffffff; lines=1; } pio_sm_clear_fifos(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM); pio_sm_restart(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM); // set pins output uint pindirs; if (pSD->is_wide_bus){ pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, pio_encode_set(pio_pins, 15)); pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, pio_encode_set(pio_pindirs, 15)); } else { pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, pio_encode_set(pio_pins, 1)); pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, pio_encode_set(pio_pindirs, 1)); } pio_sm_put_blocking(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, write_cycles); pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, pio_encode_out(pio_x, 32)); pio_sm_put_blocking(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, 31);// pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, pio_encode_out(pio_y, 32)); pio_sm_exec(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, pio_encode_jmp(write_start_addr)); uint32_t crc_and_stop_bit; uint32_t crc32_h, crc32_l; if (pSD->is_wide_bus) { uint64_t crc64 = calc_4bit_crc16(buff, SDIO_WORDS_PER_BLOCK); crc32_h = (uint32_t)(crc64>>32); crc32_l = (uint32_t)crc64; } else { crc = calc_1bit_crc16(buff, SDIO_WORDS_PER_BLOCK); crc_and_stop_bit = crc <<16 | 0xffff; } pio_sm_put_blocking(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, start); //start bit pio_sm_set_enabled(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, true); #if _DMA_TRANS_ dma_channel_set_trans_count(pSD->dma_write_channel, SDIO_WORDS_PER_BLOCK, false); dma_channel_set_read_addr(pSD->dma_write_channel, buff, false); dma_channel_start(pSD->dma_write_channel); dma_channel_wait_for_finish_blocking(pSD->dma_write_channel); #else for (int i=0; i < SDIO_WORDS_PER_BLOCK; i++) { pio_sm_put_blocking(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, __builtin_bswap32(buff[i])); } #endif if (pSD->is_wide_bus) { pio_sm_put_blocking(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM,crc32_h); pio_sm_put_blocking(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM,crc32_l); pio_sm_put_blocking(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM,end); } else { pio_sm_put_blocking(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM,crc_and_stop_bit); } uint32_t crc_status; crc_status= pio_sm_get_blocking(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM); pio_sm_set_enabled(SDIO_MEM_PIO, SDIO_DATA_WRITE_SM, false); // card report postive crc:010, neg:101(card status) if (crc_status>>24 != 0xe5){ // EZZS...E(pos:e5, net:e9) //printf("write CRC Error\n"); ret = SD_CRC_ERROR; } return ret; } uint8_t sd_read_single_block(sd_memory_card_t *pSD, uint32_t *buff, uint32_t address) { uint8_t arg[4]; uint32_t resp[2]; uint32_t read_crc[2]; uint8_t ret = SD_OK; arg[0] = address >> 24; arg[1] = (address & 0x00ff0000) >> 16; arg[2] = (address & 0x0000ff00) >> 8; arg[3] = (address & 0x000000ff); sdio_send_recv_cmd(CMD17, arg, resp, 48); //R1, if (format_R1_response(resp) == 0x0900) { ret = pio_sd_read_blocks(pSD, buff, 1, read_crc); } else { printf("read single bolck error:%08x\n", format_R1_response(resp)); return SD_STATE_ERROR; } // check CRC; if (pSD->is_wide_bus) { uint64_t data_crc; uint64_t crc; data_crc = (uint64_t)(__builtin_bswap32(read_crc[0]))<<32|(__builtin_bswap32(read_crc[1])); crc = calc_4bit_crc16(buff,SDIO_WORDS_PER_BLOCK); //printf("data_crc:%016llu\n crc:%016llu\n\n", data_crc, crc); //printf("data_crc:%08x,%08x\n crc:%08x,%08x\n\n", read_crc[0],read_crc[1], (uint32_t)(crc>>32),(uint32_t)crc) ; if (data_crc != crc) { printf("crc error:wide bus\n"); return SD_CRC_ERROR; } } else { uint16_t data_crc; uint16_t crc; data_crc = (uint16_t)(__builtin_bswap32(read_crc[0])>>16); crc = calc_1bit_crc16(buff,SDIO_WORDS_PER_BLOCK); //printf("data_crc:%08x crc:%08x, dd:%08x\n", data_crc, crc, read_crc[0]); if (data_crc!=crc){ printf("crc error:1 bit\n"); return SD_CRC_ERROR; } } return ret; } uint8_t sd_read_multiple_block(sd_memory_card_t* pSD, uint32_t *buff, uint32_t address, uint32_t count) { uint8_t arg[4]; uint32_t resp[2]; uint32_t read_crc[2*count]; uint8_t ret = SD_OK; uint32_t crc_len=(pSD->is_wide_bus)?2:1; arg[0] = address >> 24; arg[1] = (address & 0x00ff0000) >> 16; arg[2] = (address & 0x0000ff00) >> 8; arg[3] = (address & 0x000000ff); sdio_send_recv_cmd(CMD18, arg, resp, 48); //R1, if (format_R1_response(resp) == 0x0900) { ret = pio_sd_read_blocks(pSD, buff, count, read_crc); } else { printf("read multiple error:%08x\n", format_R1_response(resp)); return SD_STATE_ERROR; } sd_stop_trans(); // check CRC; if (pSD->is_wide_bus) { uint64_t data_crc; uint64_t crc; for (int i=0; i < count; i++) { data_crc = (uint64_t)(__builtin_bswap32(read_crc[2*i]))<<32|(__builtin_bswap32(read_crc[2*i+1])); crc = calc_4bit_crc16(buff+(i*SDIO_WORDS_PER_BLOCK),SDIO_WORDS_PER_BLOCK); //printf("data_crc:%016llu\n crc:%016llu\n", data_crc, crc); if (data_crc != crc) { printf("crc error: wide bus\n\n"); return SD_CRC_ERROR; } } } else { uint16_t data_crc; uint16_t crc; for (int i=0; i < count; i++) { data_crc = (uint16_t)(__builtin_bswap32(read_crc[i])>>16); crc = calc_1bit_crc16(buff+(i*(SDIO_WORDS_PER_BLOCK)),SDIO_WORDS_PER_BLOCK); //printf("data_crc:%08x crc:%08x\n", data_crc, crc); if (data_crc!=crc){ printf("crc error: 1bit\n"); return SD_CRC_ERROR; } } } return ret ; } uint8_t sd_write_single_block(sd_memory_card_t* pSD, uint32_t *buff, uint32_t address) { uint8_t arg[4]; uint32_t resp[2]; uint8_t resp_bytes[6]; uint8_t ret=SD_ERROR_CODE; //CMD16 SET_BLOCKLEN memset(arg,0,sizeof(arg)); arg[2]=0x02; sdio_send_recv_cmd(CMD16, arg, resp, 48); //R1, arg[0] = address >> 24; arg[1] = (address & 0x00ff0000) >> 16; arg[2] = (address & 0x0000ff00) >> 8; arg[3] = (address & 0x000000ff); sdio_send_recv_cmd(CMD24, arg, resp, 48); //R1, // uint32_t count=0; if (format_R1_response(resp) == 0x0900) { ret = pio_sd_write_blocks(pSD, buff, 1); for (int idx=0; idx< 10000;idx++) { uint32_t ss= get_sd_card_status(pSD); if (ss == 0x0900) break; sleep_us(10); // count++; }; } else { printf("write single error :%08x\n",format_R1_response(resp) ); return SD_STATE_ERROR; } //printf("error count:%d\n", count); return ret; } uint8_t sd_write_multiple_block(sd_memory_card_t* pSD, uint32_t *buff, uint32_t address, uint32_t count) { uint8_t arg[4]; uint32_t resp[2]; uint8_t resp_bytes[6]; uint8_t ret=SD_OK; //CMD16 SET_BLOCKLEN memset(arg,0,sizeof(arg)); arg[2]=0x02; sdio_send_recv_cmd(CMD16, arg, resp, 48); //R1, // pre-erase arg[0] = pSD->rca[0]; arg[1] = pSD->rca[1]; sdio_send_recv_cmd(CMD55, arg, resp, 48); // next is APP_CMD arg[0] = count >> 24; arg[1] = (count & 0x00ff0000) >> 16; arg[2] = (count & 0x0000ff00) >> 8; arg[3] = (count & 0x000000ff); sdio_send_recv_cmd(ACMD23, arg, resp, 48); //R1, pre-erase blocks arg[0] = address >> 24; arg[1] = (address & 0x00ff0000) >> 16; arg[2] = (address & 0x0000ff00) >> 8; arg[3] = (address & 0x000000ff); sdio_send_recv_cmd(CMD25, arg, resp, 48); //R1, // uint32_t errors=0; for (int i=0; i < count; i++) { for (int idx=0; idx < 100000; idx++) { // max 100ms wait for status : READY_FOR_DATA if (get_sd_card_status(pSD) & 0x0100) {ret=SD_OK;break;} sleep_us(1); ret = SD_TIME_OUT; // errors++; } if (ret == SD_OK) { ret = pio_sd_write_blocks(pSD, buff+i*SDIO_WORDS_PER_BLOCK, 1); if (ret != SD_OK) break; } else { return ret; } } sd_stop_trans(); if (ret != SD_OK) { sleep_ms(1); return ret;} uint32_t cardStatus; // errors=0; for (int timeout=0; timeout<10000;timeout++) { // wait for card ready cardStatus = get_sd_card_status(pSD); if (cardStatus == 0x0900) break; // errors++; sleep_us(10); } //printf("not finish count:%d\n", errors); // send ACMD22 to get how many blocks well writen, no arg, R1 32bit+CRC return ret; } uint32_t sd_get_total_blocks(sd_memory_card_t* pSD) { return pSD->total_blocks; } uint16_t sd_get_block_size(sd_memory_card_t* pSD) { return pSD->block_size; } /* sdio_memory_disk_initizlize, sdio_memory_disk_read, sdio_memory_disk_write, sdio_memory_disk_status, sdio_memory_disk_ioctl*/ /* sdmmc_disk_initialize*/ DSTATUS sdio_memory_disk_initialize(sd_memory_card_t* pSD, uint8_t wide_bus) { DSTATUS stat = sd_memory_card_init(pSD, wide_bus); return RES_OK; } /* sdmmc disk status*/ DSTATUS sdio_memory_disk_status(sd_memory_card_t* pSD) { if (pSD->SD_initialized) return RES_OK; else return RES_PARERR; } /* sdmmc disk read*/ DSTATUS sdio_memory_disk_read( BYTE *buff, /* Pointer to the data buffer to store read data */ LBA_t sector, /* Start sector number (LBA) */ UINT count, /* Number of sectors to read (1..128) */ sd_memory_card_t* pSD) { DSTATUS ret=RES_OK; DWORD sect = (DWORD)(sector); if (!count) return RES_PARERR; /* Check parameter */ if (!pSD->SD_initialized) return RES_NOTRDY; /* Check if drive is ready */ if (count == 1) { for (int retry=0; retry < CRC_ERROR_RETRY_COUNT; retry++) { ret = sd_read_single_block(pSD, (uint32_t*)buff, sector); if (ret == SD_OK) break;//crc error try again if (ret == SD_CRC_ERROR) continue; else break; //led_blinking(); //// LED blinking } } else { /* Multiple sector read */ for (int retry=0; retry < CRC_ERROR_RETRY_COUNT; retry++) { ret = sd_read_multiple_block(pSD, (uint32_t*)buff, sect, count); if (ret == SD_OK) break;//crc error try again if (ret == SD_CRC_ERROR) continue; else break; //led_blinking(); //// LED blinking } } //led_blinking_off(); //// LED blinking off return ret; /* Return result */ } DSTATUS sdio_memory_disk_write( const BYTE *buff, /* Ponter to the data to write */ LBA_t sector, /* Start sector number (LBA) */ UINT count, /* Number of sectors to write (1..128) */ sd_memory_card_t *pSD) { DWORD sect = (DWORD)sector; if (!count) return RES_PARERR; /* Check parameter */ if (!pSD->SD_initialized) return RES_NOTRDY; /* Check if drive is ready */ if (count == 1) { /* Single block write */ return sd_write_single_block(pSD, (uint32_t*)buff, sect); //led_blinking(); //// LED_blinking } else { /* Multiple sector write */ return sd_write_multiple_block(pSD, (uint32_t*)buff, sect, count); } return RES_OK; /* Return result */ } /* sdmmc disk ioctl*/ DSTATUS sdio_memory_disk_ioctl( BYTE cmd, /* Control command code */ void *buff, /* Pointer to the conrtol data */ sd_memory_card_t* pSD) { DRESULT res; BYTE n, csd[16]; DWORD st, ed, csize; LBA_t *dp; BYTE src = 0xFF; if (!pSD->SD_initialized) return RES_NOTRDY; /* Check if drive is ready */ res = RES_ERROR; switch (cmd) { case CTRL_SYNC: /* Wait for end of internal write process of the drive */ if (get_sd_card_status(pSD) == 0x0900) res = RES_OK; break; case GET_SECTOR_COUNT: /* Get drive capacity in unit of sector (DWORD) */ *(LBA_t *)buff = *(LBA_t *)sd_get_total_blocks(pSD); res = RES_OK; break; case GET_SECTOR_SIZE: // FF_MAX_SS != FX_MIN_SS //*(WORD*)buff=512; // SDHC, SDXC sector size is 512 *(WORD *)buff = pSD->block_size; res = RES_OK; break; case GET_BLOCK_SIZE: /* Get erase block size in unit of sector (DWORD) */ //printf("get_block_size\n"); /* ACMD13,CMD9 */ res = RES_OK; break; case CTRL_TRIM: /* Erase a block of sectors (used when _USE_ERASE == 1) */ //printf("CTL_TRIM\n"); /* CMD32 */ res = RES_OK; /* Following commands are never used by FatFs module */ break; case MMC_GET_TYPE: /* Get MMC/SDC type (BYTE) */ //printf("MMC_GET_TYPE\n"); res = RES_OK; break; case MMC_GET_CSD: /* Read CSD (16 bytes) CMD9*/ //printf("MMC_GET_CSD\n"); res = RES_OK; break; case MMC_GET_CID: /* Read CID (16 bytes) CMD10*/ //printf("MMC_GET_CID\n"); res = RES_OK; break; case MMC_GET_OCR: /* Read OCR (4 bytes) CMD58*/ //printf("MMC_GET_OCR\n"); res = RES_OK; break; case MMC_GET_SDSTAT: /* Read SD status (64 bytes) ACMD13*/ //printf("MMC_GET_SDSTAT\n"); res = RES_OK; break; default: //printf("default:%d\n",cmd); res = RES_PARERR; } return res; }
- sdio_mem_card.h
#ifndef __SDIO_MEM_H_ #define __SDIO_MEM_H_ #include "ff.h" #include "diskio.h" #include "pico_storage.h" #define SDIO_CLK_PIN 16 // PIO sideset pin #define SDIO_CMD_PIN 17 // #define SDIO_D0_PIN 18 // D0~D3 must consecutive and D0 PIN number is minium #define SDIO_D1_PIN 19 #define SDIO_D2_PIN 20 #define SDIO_D3_PIN 21 #define SDIO_MEM_PIO pio0 #define SDIO_CMD_SM 0 #define SDIO_DATA_READ_SM 1 #define SDIO_DATA_WRITE_SM 2 #define SDIO_CLK_SM 3 #define _DMA_TRANS_ 1 #define SDIO_WORDS_PER_BLOCK 128 #define SDIO_RW_WORDS_PER_BLOCK 130 //128+2 #define MAX_RW_BLOCKS_TRANS 16 #define CRC_ERROR_RETRY_COUNT 3 typedef struct { uint8_t rca[2]; uint8_t SD_initialized; int sd_data_rx_offset; int sd_data_tx_offset; int sd_data_cmd_offset; uint8_t is_wide_bus; uint8_t pnm[6]; // product name uint8_t prv; // product revision BCD(4.4) uint32_t psn; //product serial number uint16_t block_size; uint32_t total_blocks; int dma_write_channel; int dma_read_channel; } sd_memory_card_t; enum { CLK_FREQ_400K=0, CLK_FREQ_1M, CLK_FREQ_12_5M, CLK_FREQ_25M, CLK_FREQ_50M, } SDIO_CLK_FREQ; enum { CMD0 = 0, CMD2 = 2, CMD3 = 3, //SEND_RELATIVE_ADDR CMD4 = 4, //SET_DSR CMD7 = 7, //SELECT_CARD CMD8 = 8, CMD9 = 9, //SEND_CSD CMD10 = 10, //SEND_CID CMD11 = 11, CMD12 = 12, //STOP_TRANSMISSION CMD13 = 13, //SEND_STATUS CMD16 = 16, // BLOCK LENGTH CMD17 = 17, // read single block CMD18 = 18, // read mutiple block CMD24 = 24, // write single block CMD25 = 25, // write mutiple block CMD42 = 42, CMD55 = 55, // APP_CMD, next is app command ACMD22 = 22, // SEND_NUM_WR_BLOCKS ACMD23 = 23, //SET_WR_BLK_ERASE_COUNT ACMD6 = 6, // SET_BUS_WIDTH, param:00b-1 bit, 10b-4 bits ACMD41 = 41, //SD_SEND_OP_COND } SDIO_CMD_LIST; enum { IDLE_STATE= 0x01, ERASE_RESET= 0x02, ILLEAGAL_COMMAND= 0x04, CRC_ERROR= 0x08, ERASE_SEQUENCE_ERROR= 0x10, ADDRESS_ERROR= 0x20, PARAMETER_ERROR= 0x40, } SDIO_R1; enum { SD_OK=0, SD_INIT_FAILURE, SD_CRC_ERROR, SD_TIME_OUT, SD_STATE_ERROR, SD_READ_ERROR, SD_WRITE_ERROR, } SD_ERROR_CODE; //void sdio_pio_init(PIO pio, uint cmd_pin, uint clk_pin, uint data_pin_base, uint16_t clk_int, uint8_t clk_frac); uint8_t sd_memory_card_init(sd_memory_card_t* sd_mem_card, uint8_t wide_bus); uint8_t sd_read_single_block(sd_memory_card_t* pSD,uint32_t *buff, uint32_t address); uint8_t sd_read_multiple_block(sd_memory_card_t* pSD,uint32_t *buff, uint32_t address, uint32_t count); uint8_t sd_write_single_block(sd_memory_card_t* pSD,uint32_t *buff, uint32_t address); uint8_t sd_write_multiple_block(sd_memory_card_t* pSD,uint32_t *buff, uint32_t address, uint32_t count); uint32_t sd_get_total_blocks(sd_memory_card_t* pSD); uint16_t sd_get_block_size(sd_memory_card_t* pSD); DSTATUS sdio_memory_disk_initialize(sd_memory_card_t* pSD, uint8_t wide_bus); DSTATUS sdio_memory_disk_status(sd_memory_card_t* pSD); DSTATUS sdio_memory_disk_read(BYTE *buff, LBA_t sector, UINT count, sd_memory_card_t *pSD); DSTATUS sdio_memory_disk_write(const BYTE *buff, LBA_t sector,UINT count, sd_memory_card_t *pSD); DSTATUS sdio_memory_disk_ioctl(BYTE cmd, void *buff, sd_memory_card_t* pSD); #endif
- sdio_mem_card.pio
; SD clock frequency: 125M/(DL+DH+2). ; DL=2, DH=1: 25Mhz ; DL=1, DH=1: 31.25Mhz. ; DL=1, DH=0: 41.67Mhz. .define DL 1 .define DH 0 ; ===cmd state machine=== .program sdio_mem_cmd .origin 0 .side_set 1 .wrap_target public sdio_cmd_start: out null, 32 side 0 [1] ;discard data out x, 8 side 1 [1] ;command bits send_cmd_bits: out pins, 1 side 0 [1] jmp x--, send_cmd_bits side 1 [1] set pindirs, 0 side 0 [1] out x, 8 side 1 [1] ;response bits jmp !x, sdio_cmd_stop side 0 [1] wait_recv: nop side 1 [1] jmp pin, wait_recv side 0 [1] recv_resp: in pins, 1 side 1 [1] jmp x--, recv_resp side 0 [1] sdio_cmd_stop: ;nop side 1 set x,7 side 1 [1] wait_Ncr_Ncc: nop side 0 [1] jmp x--, wait_Ncr_Ncc side 1 [1] push side 0 [1] wait_shutdown: jmp wait_shutdown side 0 [1] ;irq nowait 0 side 1 .wrap ;========== 1 bit READ ============ .program sdio_1_bit_rx .side_set 1 public rx_start: .wrap_target wait_start: mov y,x side 0 [DL] jmp pin, wait_start side 1 [DH] nop side 0 [DL] rx_data_bit: in PINS, 1 side 1 [DH] jmp y--, rx_data_bit side 0 [DL] IN NULL,16 side 0 [DL] .wrap ;========== 4 bit READ ============ .program sdio_4_bit_rx .side_set 1 .wrap_target public rx_start: wait_start: mov y,x side 0 [DL] jmp pin, wait_start side 1 [DH] nop side 0 [DL] rx_data_bit: in PINS, 4 side 1 [DH] jmp y--, rx_data_bit side 0 [DL] nop side 0 [DL] .wrap ;========== 1 bit WRITE ============ .program sdio_1_bit_tx .side_set 1 .wrap_target public tx_start: tx_data_bit: out PINS, 1 side 0 [DL] jmp x--, tx_data_bit side 1 [DH] set pindirs, 0 side 0 [DL] response_loop: in PINS, 1 side 1 [DH] jmp Y--, response_loop side 0 [DL] nop side 1 [DH] push side 0 [DL] .wrap ;========== 4 bit WRITE ============ .program sdio_4_bit_tx .side_set 1 .wrap_target public tx_start: tx_data_bit: out PINS, 4 side 0 [DL] jmp x--, tx_data_bit side 1 [DH] set pindirs, 0 side 0 [DL] response_loop: in PINS, 1 side 1 [DH] jmp Y--, response_loop side 0 [DL] nop side 1 [DH] push side 0 [DL] ;nop side 1 .wrap
- CMakeLists.txt(pico_strorage_drv)
add_library(pico_storage_drv INTERFACE) pico_generate_pio_header(pico_storage_drv ${CMAKE_CURRENT_LIST_DIR}/sdio_mem_card/sdio_mem_card.pio) target_sources(pico_storage_drv INTERFACE ${CMAKE_CURRENT_LIST_DIR}/glue.c ${CMAKE_CURRENT_LIST_DIR}/FatFs/ff.c ${CMAKE_CURRENT_LIST_DIR}/FatFs/ffunicode.c ${CMAKE_CURRENT_LIST_DIR}/FatFs/ffsystem.c ${CMAKE_CURRENT_LIST_DIR}/spi_sdmmc/spi_sdmmc.c ${CMAKE_CURRENT_LIST_DIR}/flash/W25Q.c ${CMAKE_CURRENT_LIST_DIR}/usb_msc/usb_msc.c ${CMAKE_CURRENT_LIST_DIR}/sdio_mem_card/sdio_mem_card.c ) target_include_directories(pico_storage_drv INTERFACE ${CMAKE_CURRENT_LIST_DIR} ${CMAKE_CURRENT_LIST_DIR}/FatFs ${CMAKE_CURRENT_LIST_DIR}/flash ${CMAKE_CURRENT_LIST_DIR}/spi_sdmmc ${CMAKE_CURRENT_LIST_DIR}/usb_msc ${CMAKE_CURRENT_LIST_DIR}/sdio_mem_card ) target_link_libraries(pico_storage_drv INTERFACE hardware_spi hardware_dma hardware_pio hardware_rtc pico_stdlib tinyusb_board tinyusb_host )
- pico_storage.h
#ifndef _PICO_STORAGE_H_ #define _PICO_STORAGE_H_ /* used by my project */ #define SD_4_BIT_PATH "0:" #define SPI_SDMMC_PATH "1:" #define SD_1_BIT_PATH "2:" #define W25Q_PATH "3:" #define USB_MSC_PATH "4:" #define SPI_BAUDRATE_LOW (1000*1000) #define SPI_BAUDRATE_HIGH (10*1000*1000) /* ================================= */ #define LED_BLINKING_PIN 25 void led_blinking(void); void led_blinking_off(void); /* ================================= */ #endif
- glue.c
#include "stdio.h" #include "stdlib.h" #include "ff.h" #include "diskio.h" #include "sdio_mem_card.h" #include "spi_sdmmc.h" #include "W25Q.h" #include "usb_msc.h" #include "hardware/rtc.h" #include "inttypes.h" #include "hardware/gpio.h" #define SD_4_BIT_DRV 0 #define SPI_SDMMC_DRV 1 #define SD_1_BIT_DRV 2 #define W25Q_DRV 3 #define USB_MSC_DRV 4 sd_memory_card_t *pSDIO_MEM_CARD_4=NULL; sd_memory_card_t *pSDIO_MEM_CARD_1=NULL; spi_sdmmc_data_t *pSPI_SDMMC=NULL; w25q_data_t *pW25Q = NULL; usb_msc_t *pUSB_MSC=NULL; //==================// DSTATUS disk_initialize (BYTE drv){ DSTATUS stat; switch (drv) { case SD_4_BIT_DRV: if (pSDIO_MEM_CARD_4 == NULL) { pSDIO_MEM_CARD_4 = (sd_memory_card_t*)malloc(sizeof(sd_memory_card_t)); pSDIO_MEM_CARD_4->SD_initialized=false; } stat = sdio_memory_disk_initialize(pSDIO_MEM_CARD_4, true); return stat; break; case SD_1_BIT_DRV: if (pSDIO_MEM_CARD_1 == NULL) { pSDIO_MEM_CARD_1 = (sd_memory_card_t*)malloc(sizeof(sd_memory_card_t)); pSDIO_MEM_CARD_1->SD_initialized=false; } stat = sdio_memory_disk_initialize(pSDIO_MEM_CARD_1, false); return stat; break; case SPI_SDMMC_DRV: if (pSPI_SDMMC == NULL) { pSPI_SDMMC = (spi_sdmmc_data_t*)malloc(sizeof(spi_sdmmc_data_t)); pSPI_SDMMC->csPin = SDMMC_PIN_CS; pSPI_SDMMC->spiPort = SDMMC_SPI_PORT; pSPI_SDMMC->spiInit=false; pSPI_SDMMC->sectSize=512; #ifdef __SPI_SDMMC_DMA pSPI_SDMMC->dmaInit=false; #endif } stat = sdmmc_disk_initialize(pSPI_SDMMC); return stat; break; case W25Q_DRV: if (pW25Q == NULL) { pW25Q = (w25q_data_t*)malloc(sizeof(w25q_data_t)); pW25Q->spiInit=false; pW25Q->Stat=STA_NOINIT; } stat = w25q_disk_initialize(W25Q_SPI_PORT, W25Q_PIN_CS, pW25Q); return stat; break; case USB_MSC_DRV: if (pUSB_MSC == NULL) { pUSB_MSC = (usb_msc_t*)malloc(sizeof(usb_msc_t)); } stat = usb_msc_initialize(pUSB_MSC); return stat; break; } return STA_NOINIT; } /*-----------------------------------------------------------------------*/ /* Get disk status */ /*-----------------------------------------------------------------------*/ DSTATUS disk_status (BYTE drv) { DSTATUS stat; switch (drv) { case SD_4_BIT_DRV: stat = sdio_memory_disk_status(pSDIO_MEM_CARD_4); return stat; break; case SD_1_BIT_DRV: stat = sdio_memory_disk_status(pSDIO_MEM_CARD_1); return stat; break; case SPI_SDMMC_DRV: stat= sdmmc_disk_status(pSPI_SDMMC); /* Return disk status */ return stat; break; case W25Q_DRV: stat = pW25Q->Stat; return stat; break; case USB_MSC_DRV: //uint8_t dev_addr = pdrv + 1; return tuh_msc_mounted(pUSB_MSC->dev_addr) ? 0 : STA_NODISK; break; } return RES_PARERR; } /*-----------------------------------------------------------------------*/ /* Read sector(s) */ /*-----------------------------------------------------------------------*/ DRESULT disk_read ( BYTE drv, /* Physical drive number (0) */ BYTE *buff, /* Pointer to the data buffer to store read data */ LBA_t sector, /* Start sector number (LBA) */ UINT count /* Number of sectors to read (1..128) */ ) { DSTATUS stat; switch (drv) { case SD_4_BIT_DRV: stat = sdio_memory_disk_read(buff, sector, count, pSDIO_MEM_CARD_4); return stat; break; case SD_1_BIT_DRV: stat = sdio_memory_disk_read(buff, sector, count, pSDIO_MEM_CARD_1); return stat; break; case SPI_SDMMC_DRV: stat = sdmmc_disk_read(buff, sector, count, pSPI_SDMMC); return stat; break; case W25Q_DRV: if (pW25Q->Stat & STA_NOINIT) return RES_NOTRDY; w25q_read_sector((uint32_t)sector, 0, buff, count*pW25Q->sectorSize, pW25Q); return pW25Q->Stat; break; case USB_MSC_DRV: stat = usb_msc_disk_read(buff, sector, count, pUSB_MSC); return stat; break; } return RES_PARERR; } /*-----------------------------------------------------------------------*/ /* Write sector(s) */ /*-----------------------------------------------------------------------*/ #if FF_FS_READONLY == 0 DRESULT disk_write ( BYTE drv, /* Physical drive number (0) */ const BYTE *buff, /* Ponter to the data to write */ LBA_t sector, /* Start sector number (LBA) */ UINT count /* Number of sectors to write (1..128) */ ) { DSTATUS stat = STA_NODISK; switch (drv) { case SD_4_BIT_DRV: stat = sdio_memory_disk_write(buff, sector, count, pSDIO_MEM_CARD_4); return stat; break; case SD_1_BIT_DRV: stat = sdio_memory_disk_write(buff, sector, count, pSDIO_MEM_CARD_1); return stat; break; case SPI_SDMMC_DRV: stat = sdmmc_disk_write(buff, sector, count, pSPI_SDMMC); return stat; break; case W25Q_DRV: stat = w25q_disk_write(buff, sector, count, pW25Q); return stat; break; case USB_MSC_DRV: stat = usb_msc_disk_write(buff, sector, count, pUSB_MSC); return stat; break; } return RES_PARERR; } #endif /*-----------------------------------------------------------------------*/ /* Miscellaneous drive controls other than data read/write */ /*-----------------------------------------------------------------------*/ DRESULT disk_ioctl ( BYTE drv, /* Physical drive number (0) */ BYTE cmd, /* Control command code */ void *buff /* Pointer to the conrtol data */ ) { DSTATUS stat; switch (drv) { case SD_4_BIT_DRV: stat = sdio_memory_disk_ioctl(cmd, buff, pSDIO_MEM_CARD_4); return stat; break; case SD_1_BIT_DRV: stat = sdio_memory_disk_ioctl(cmd, buff, pSDIO_MEM_CARD_1); return stat; break; case W25Q_DRV: stat = w25q_disk_ioctl(cmd, buff, pW25Q); return stat; break; case USB_MSC_DRV: stat = usb_msc_ioctl(cmd, buff, pUSB_MSC); return stat; break; } return RES_PARERR; } DWORD get_fattime(void) { datetime_t t = {0, 0, 0, 0, 0, 0, 0}; bool rc = rtc_get_datetime(&t); if (!rc) return 0; DWORD fattime = 0; // bit31:25 // Year origin from the 1980 (0..127, e.g. 37 for 2017) uint32_t yr = t.year - 1980; fattime |= (0b01111111 & yr) << 25; // bit24:21 // Month (1..12) uint32_t mo = t.month; fattime |= (0b00001111 & mo) << 21; // bit20:16 // Day of the month (1..31) uint32_t da = t.day; fattime |= (0b00011111 & da) << 16; // bit15:11 // Hour (0..23) uint32_t hr = t.hour; fattime |= (0b00011111 & hr) << 11; // bit10:5 // Minute (0..59) uint32_t mi = t.min; fattime |= (0b00111111 & mi) << 5; // bit4:0 // Second / 2 (0..29, e.g. 25 for 50) uint32_t sd = t.sec / 2; fattime |= (0b00011111 & sd); return fattime; } void led_blinking(void) { static absolute_time_t t1; static bool state=false; // Blink every interval ms if ( absolute_time_diff_us(t1, get_absolute_time()) < 100000) return; // not enough time t1 = get_absolute_time(); gpio_put(LED_BLINKING_PIN, state); state = !state; } void led_blinking_off(void) { gpio_put(LED_BLINKING_PIN, false); }
- main.c
#include <stdio.h> #include "pico/stdlib.h" #include "hardware/dma.h" #include "hardware/pio.h" //#include "sdio_mem_card/sdio_mem_card.h" #include "string.h" #include "ff.h" #include "diskio.h" #include "pico_storage.h" #include "hardware/rtc.h" //sd_memory_card_t sd; void test_data(uint8_t* buff) { // test write data uint8_t ch='A'; for (int i=0; i < 512;i++){ buff[i] = ch; ch++; if (ch > 'Z') ch='A'; } memcpy(buff+512, buff, 512); buff[511]='\n'; buff[0+512] = 'a'; buff[26+512] = 'b'; buff[52+512] = 'c'; buff[512+511]='\n'; memcpy(buff+1024, buff, 512); buff[0+1024] = '1'; buff[26+1024] = '2'; buff[52+1024] = '3'; buff[1024+511]='\n'; for (uint32_t i = 1536; i< 8192; i+=512) memcpy(buff+i, buff, 512); } uint8_t test_4_bit() { FATFS fs; FIL fil_r, fil_w; FRESULT res; uint bw, br; uint8_t buff[16384]; if (f_mount(&fs, SD_4_BIT_PATH, 1) != FR_OK) { printf("mount error\n"); return 0; } printf("mount ok\n"); printf("\n ==============================\ntest under 41.66Mhz SD clock frequency\n"); res = f_open(&fil_r, SD_4_BIT_PATH"/pico_storage.mp4", FA_READ); if (res != FR_OK) { printf("open read error:\n"); f_unmount(SD_4_BIT_PATH); return 0; } uint64_t r_out=0; uint64_t fsize=f_size(&fil_r); uint32_t i=0; f_close(&fil_r); printf("file size:%llu\n",fsize); uint16_t buff_size[4] = {1024, 4096, 8192, 12288}; for (int r=0; r<4; r++) { res = f_open(&fil_r, SD_4_BIT_PATH"/pico_storage.mp4", FA_READ); if (res != FR_OK) { printf("open read error:\n"); f_unmount(SD_4_BIT_PATH); return 0; } r_out=0; printf("Test file read speed(SD-4bit): using %dK buffer\n", buff_size[r]/1024); absolute_time_t tt=get_absolute_time(); do { res = f_read(&fil_r, buff, buff_size[r], &br); if (res != FR_OK) { printf("read error:%d\n", res); break; } r_out += br; } while(r_out < fsize); int64_t et=absolute_time_diff_us(tt, get_absolute_time())/1000; double dd = (double)(r_out)/et*1000/1024/1024; printf("total time:%llu ms, total_read:%d bytes, = %0.2lfMB/s\n", et, r_out, dd); printf("--------------\n\n"); f_close(&fil_r); } printf("\n======================\n"); printf("Test file write speed(SD-4bit): using 8K buffer\n"); test_data(buff); res = f_open(&fil_w, SD_4_BIT_PATH"/8k_4_buff.txt", FA_CREATE_ALWAYS|FA_WRITE); if (res != FR_OK) { printf("open file error:\n"); f_unmount(SD_4_BIT_PATH); return 0; } absolute_time_t tw=get_absolute_time(); for (int s=0; s < 8192;s++) { // 64MB f_write(&fil_w, buff, 8192, &bw); //printf("write:%d\n", bw); } int64_t et=absolute_time_diff_us(tw, get_absolute_time())/1000; double dd = (double)(8192*8192)/et*1000/1024/1024; printf("total time:%llu ms, total_write:%d bytes, = %0.2lfMB/s\n", et, 8192*8192, dd); f_close(&fil_w); printf("\n======================\n"); printf("Test file read in and write out speed(SD-4bit): using 8K buffer\n"); res = f_open(&fil_r, SD_4_BIT_PATH"/pico_storage.mp4", FA_READ); if (res != FR_OK) { printf("open read error:\n"); f_unmount(SD_4_BIT_PATH); return 0; } res = f_open(&fil_w, SD_4_BIT_PATH"/test_out_4.mp4", FA_CREATE_ALWAYS|FA_WRITE); if (res != FR_OK) { printf("open file error:\n"); f_unmount(SD_4_BIT_PATH); return 0; } r_out=0; absolute_time_t tt=get_absolute_time(); do { res = f_read(&fil_r, buff, 8192, &br); if (res != FR_OK) { printf("read error:%d\n", res); break; } f_write(&fil_w, buff, br, &bw); if (res != FR_OK) { printf("write error:%d\n", res); break; } r_out += br; //if ((i++%100)==0) { //printf("read bytes:%d\n", br); // printf("write :%llu\n", r_out); //} } while(r_out < fsize); printf("total time:%llu ms, total_read:%d bytes\n", absolute_time_diff_us(tt, get_absolute_time())/1000, r_out); f_close(&fil_r); f_close(&fil_w); printf("\n#=============#\n"); f_unmount(SD_4_BIT_PATH); } uint8_t test_1_bit() { FATFS fs; FIL fil_r, fil_w; FRESULT res; uint bw, br; uint8_t buff[16384]; if (f_mount(&fs, SD_1_BIT_PATH, 1) != FR_OK) { printf("mount error\n"); return 0; } printf("mount ok\n"); printf("\n ==============================\ntest under 41.66Mhz SD clock frequency\n"); res = f_open(&fil_r, SD_1_BIT_PATH"/pico_storage.mp4", FA_READ); if (res != FR_OK) { printf("open read error:\n"); f_unmount(SD_1_BIT_PATH); return 0; } uint64_t r_out=0; uint64_t fsize=f_size(&fil_r); uint32_t i=0; f_close(&fil_r); printf("file size:%llu\n",fsize); uint16_t buff_size[4] = {1024, 4096, 8192, 12288}; for (int r=0; r<4; r++) { res = f_open(&fil_r, SD_1_BIT_PATH"/pico_storage.mp4", FA_READ); if (res != FR_OK) { printf("open read error:\n"); f_unmount(SD_1_BIT_PATH); return 0; } r_out=0; printf("Test file read speed(SD-1bit): using %dK buffer\n", buff_size[r]/1024); absolute_time_t tt=get_absolute_time(); do { res = f_read(&fil_r, buff, buff_size[r], &br); if (res != FR_OK) { printf("read error:%d\n", res); break; } r_out += br; } while(r_out < fsize); int64_t et=absolute_time_diff_us(tt, get_absolute_time())/1000; double dd = (double)(r_out)/et*1000/1024/1024; printf("total time:%llu ms, total_read:%d bytes, = %0.2lfMB/s\n", et, r_out, dd); printf("--------------\n\n"); f_close(&fil_r); } printf("\n======================\n"); printf("Test file write speed(SD-1bit): using 8K buffer\n"); test_data(buff); res = f_open(&fil_w, SD_1_BIT_PATH"/8k_1_buff.txt", FA_CREATE_ALWAYS|FA_WRITE); if (res != FR_OK) { printf("open file error:\n"); f_unmount(SD_1_BIT_PATH); return 0; } absolute_time_t tw=get_absolute_time(); for (int s=0; s < 8192;s++) { // 64MB f_write(&fil_w, buff, 8192, &bw); } int64_t et=absolute_time_diff_us(tw, get_absolute_time())/1000; double dd = (double)(8192*8192)/et*1000/1024/1024; printf("total time:%llu ms, total_write:%d bytes, = %0.2lfMB/s\n", et, 8192*8192, dd); f_close(&fil_w); printf("\n======================\n"); printf("Test file read in and write out speed(SD-1bit): using 8K buffer\n"); res = f_open(&fil_r, SD_1_BIT_PATH"/pico_storage.mp4", FA_READ); if (res != FR_OK) { printf("open read error:\n"); f_unmount(SD_1_BIT_PATH); return 0; } res = f_open(&fil_w, SD_1_BIT_PATH"/test_out_1.mp4", FA_CREATE_ALWAYS|FA_WRITE); if (res != FR_OK) { printf("open file error:\n"); f_unmount(SD_1_BIT_PATH); return 0; } r_out=0; absolute_time_t tt=get_absolute_time(); do { res = f_read(&fil_r, buff, 8192, &br); if (res != FR_OK) { printf("read error:%d\n", res); break; } f_write(&fil_w, buff, br, &bw); if (res != FR_OK) { printf("write error:%d\n", res); break; } r_out += br; //if ((i++%100)==0) { //printf("read bytes:%d\n", br); // printf("write :%llu\n", r_out); //} } while(r_out < fsize); printf("total time:%llu ms, total_read:%d bytes\n", absolute_time_diff_us(tt, get_absolute_time())/1000, r_out); f_close(&fil_r); f_close(&fil_w); printf("\n#=============#\n"); f_unmount(SD_1_BIT_PATH); } int main() { stdio_init_all(); printf("start\n"); datetime_t t = {2023, 10, 20, 5, 3, 0, 0}; rtc_set_datetime(&t); sleep_ms(1000); test_1_bit(); test_4_bit(); while(1); puts("Hello, world!"); return 0; }
- CMakeListx.txt(root)
# Generated Cmake Pico project file cmake_minimum_required(VERSION 3.13) set(CMAKE_C_STANDARD 11) set(CMAKE_CXX_STANDARD 17) # Initialise pico_sdk from installed location # (note this can come from environment, CMake cache etc) set(PICO_SDK_PATH "/home/duser/pico/pico-sdk") set(PICO_BOARD pico CACHE STRING "Board type") # Pull in Raspberry Pi Pico SDK (must be before project) include(pico_sdk_import.cmake) if (PICO_SDK_VERSION_STRING VERSION_LESS "1.4.0") message(FATAL_ERROR "Raspberry Pi Pico SDK version 1.4.0 (or later) required. Your version is ${PICO_SDK_VERSION_STRING}") endif() project(SDIO_MEM C CXX ASM) # Initialise the Raspberry Pi Pico SDK pico_sdk_init() # Add executable. Default name is the project name, version 0.1 add_executable(SDIO_MEM main.c) pico_set_program_name(SDIO_MEM "SDIO_MEM") pico_set_program_version(SDIO_MEM "0.1") pico_enable_stdio_uart(SDIO_MEM 1) pico_enable_stdio_usb(SDIO_MEM 0) # Add the standard library to the build target_link_libraries(SDIO_MEM pico_stdlib) # Add the standard include files to the build target_include_directories(SDIO_MEM PRIVATE ${CMAKE_CURRENT_LIST_DIR} ${CMAKE_CURRENT_LIST_DIR}/.. # for our common lwipopts or any other standard includes, if required ) # Add any user requested libraries #target_link_libraries(SDIO_MEM # hardware_dma # hardware_pio # ) add_subdirectory(pico_storage_drv) # Add any user requested libraries target_link_libraries(SDIO_MEM tinyusb_host tinyusb_board pico_storage_drv ) pico_add_extra_outputs(SDIO_MEM)