Skip to content

Latest commit

 

History

History
278 lines (175 loc) · 12.2 KB

README.md

File metadata and controls

278 lines (175 loc) · 12.2 KB

pico-sd-card

SD card drivers for Raspberry Pi Pico. If you are intending to use an SD card as a serious file system for your pico, you may also be interested in my simple-cli module.

tested with micropython versions:

  • v1.14-137-g172fb5230-dirty
  • v1.15-88-g7b923d6c7-dirty

Community:

To officially file a bug report or feature request you can use these templates: bug report | feature request

To discus features, bugs or share your own project that utilize code in this repo: join the discussion




Ports:

sdcard.py

This can be uploaded directly to the board, but is intended to be used as a frozen module. For information regarding how to setup the sdk and freeze a module you can refer to this post on the Raspberry Pi forum.

sdcard.mpy

This is a cross-compiled version of sdcard.py. It is intended to be uploaded to your board as you would any normal .py script.

modules/

This contains a pure C port of sdcard.py packaged as a USER_C_MODULE. If you have the sdk setup it can be compiled into firware by navigating to ./ports/rp2/ and running the following command:

make USER_C_MODULES=/path/to/modules/micropython.cmake all




Differences:

The 2 python versions of the port are identical. The C port differs slightly in that it cannot detect when a card is inserted/removed in real time, and as such it does not support the callback argument in the SDCard module's constructor or detect method. The reason behind the difference is due to how interrupt requests are handled in the Pin class, coupled with the fact that the Pin class does not have a public interface exposed to C. Every interrupt request is sent to one function in Pin. That function then does a lookup on the actual pin that created the interrupt, finds it's handler and executes it. Without an interface for me to include the detect pin irq in the main look-up table the irq handler I create will, at best be overwritten, and at worst, break everything. Trying to create some hack to force my irq handler into the table makes my script hacky, and that is not part of my plan. I tried very hard to find some legitimate way to gel with the Pin class and failed. I'm sure there is some genius way to do it. If I ever figure it out I'll bring the C version up to speed.




Docs:

SDCard(spi, sck, mosi, miso, cs, baudrate, automount, drive, led, detect, wait, callback)

Main SDCard interface

Args Type Description Default
spi int which SPI to use (0 or 1) REQUIRED
sck int SPI clock pin id REQUIRED
mosi int SPI mosi pin id REQUIRED
miso int SPI miso pin id REQUIRED
cs int SPI chip select pin id REQUIRED
baudrate int the desired speed to read/write 5mhz
automount bool whether to automatically mount the drive True
drive str drive-name to represent the drive "/sd"
led int pin id for a connected LED. LED is on during read/write -1 (no pin)
detect int pin id for a detect feature -1 (no pin)
wait bool whether to wait for card insertion. Used with detect (blocks) False
callback func detection callback (python versions only) None

.detect(automount, wait, maxwait, interval, callback)

Manual setup option for use with the detect arguments of the constructor. This is done automatically if a card was present or wait was True when SDCard was instantiated

Args Type Description Default
automount bool whether to automatically mount the drive True
wait bool whether to wait for card insertion. Used with detect (blocks) False
maxwait int max amount of intervals to wait (0 = forever) 0
interval int amount of milliseconds to sleep between checks 500
callback func detection callback (python versions only) None

.mount()

Manually mount a card. If a card is already mounted nothing happens.


.eject()

Eject a mounted card. If no card is mounted nothing happens.


.state()

Prints the detected, connected and mounted state of the sdcard.


.type

Returns the type of sdcard that is inserted (ver. 1 or ver. 2)


.sectors

Returns the number of sectors on the insterted sdcard. Divide by 2048 to get volume size in mb.


.drive

Returns the drive letter that is assigned to the card.


.detected

Returns whether an sdcard is currently detected (True|False).


.ready

Returns if the sdcard is 100% ready (True|False).


------

Usage:

The below scripts reflect a basic SPI1 setup (SPI0 is also supported). The pin ids you use will depend on the pins your card is connected to. If you are unsure of your options, you can use this pinout as a reference.


basic

This is an example of the absolute bare minimum it takes to connect and mount an sdcard. All omitted arguments will default to the values in the above table.

import sdcard

sd = sdcard.SDCard(1, 10, 11, 8, 9)

detection and waiting

In this example we use a baudrate of 16mhz, and the on-board LED. It is implied that our sdcard reader has a detect feature and it is connected to pin 15. Since wait is True the script will sit in an infinite loop waiting for an sdcard to be inserted (if one is not already), and will automatically connect once one is. Then our json file will load. This feature may be handy if it is mandatory for an sdcard to be inserted in order for the system to proceed. In this case, it is implied that our system depends on log information to continue.

import sdcard, ujson

sd = sdcard.SDCard(1, 10, 11, 8, 9, baudrate=0x10<<20, led=25, detect=15, wait=True)

log = ujson.load(open("{}/log.json".format(sd.drive), 'r'))

If you are using one of the python versions you can leave wait false and allow the built-in interrupt request to catch when a card is inserted, using a callback to load the json file.

import sdcard, ujson

log = dict()
sd  = None

def get_logs(ready:bool):
    global log
    if ready:
        log = ujson.load(open("{}/log.json".format(sd.drive), 'r'))

sd = sdcard.SDCard(1, 10, 11, 8, 9, baudrate=0x10<<20, led=25, detect=15, callback=get_logs)

mount() / eject()

You can turn off automount, and mount the sdcard manually at a later time. You can also eject the sdcard whenever you like. Conditions are in place that wont run the mount feature if the sdcard is already mounted, and wont run the eject feature if the card is already ejected. When a card is ejected it is also removed from the system path.

import sdcard, ujson

sd = sdcard.SDCard(1, 10, 11, 8, 9, mount=False)

# this line represents your operations between initializing and mountiing the sdcard

sd.mount()

#this line represents an itermittent need for the sdcard

sd.eject()

detect(automount, wait, maxwait, interval, callback)

callback is NOT available in the C port

If you used the detect arguments in the constructor, did not wait, and did not have an sdcard inserted, you can manually call the card setup at a later time to establish a connection. You can also designate whether you want to automount the card and/or wait for a card to be inserted. This feature can be used anywhere that it is mandatory for an sdcard to be connected and mounted. If an sdcard is already connected and mounted when this is called the request is simply ignored.

import sdcard

#init sd card
sd = sdcard.SDCard(1, 10, 11, 8, 9, baudrate=0x10<<20, led=25, detect=15, wait=False)

# this line represents your operations between instantiating and initializing the sdcard

sd.detect(wait=True)

# this line represents your operations once a card is inserted and detected 
# if wait is True this will not be reached until a card is inserted

If you are using one of the python versions you can leave wait false and allow the built-in interrupt request to catch when a card is inserted, using a callback to perform proceeding operations.

import sdcard

#init sd card
sd = sdcard.SDCard(1, 10, 11, 8, 9, baudrate=0x10<<20, led=25, detect=15, wait=False)

# this line represents your operations between instantiating and initializing the sdcard

def my_func(ready:bool):
    if ready:
        # this line represents your operations once a card is inserted and detected
    pass

sd.detect(callback=my_func)

loading external scripts

When a card has been successfully connected and mounted it is also automatically added to the system path. Below is a simple example of running a pyhon script from the sdcard. So the test is easy to perform, this will write a simple script to the sdcard first.

import sdcard

sd = sdcard.SDCard(1, 10, 11, 8, 9)

#write a simple test script to the card
with open("{}/test.py".format(sd.drive), 'w') as f:
    f.write('class Test(object):\n\tdef __init__(self):\n\t\tprint("Hello From SD Card")')

#import the script
test = __import__('test', globals(), locals(), ['Test'], 0)

#call it's class
test.Test()



Tips:

5v Voltage Regulated Card Readers

If you have a card reader that automatically converts 5v to 3.3v, apparently the miso line will output 5v to the Pico, which is not tolerant of 5v. A solution to this is to completely remove the level shifter from the card and short Vin to Vout where the level shifter was (as illustrated below). This will allow you to connect the reader directly to 3.3v.

Modified 5v Voltage Regulated SD Card Reader


Using A MicroSD Card Adapter

Instructions on how to use a card adapter as a reader can be found here


Pimoroni Pico Explorer

If you are using a Pimoroni Pico Explorer and the type of card reader depicted in the image above, using the Explorer's SPI breakout pins will be futile. Your display will stop working while the reader is plugged in. The complications are multiplied because the Explorer uses all of the SPI1 capable pins for the motor drivers and it's buttons. My solution to this requires some hacking. By soldering female headers across pins 8 through 11 you can bypass the motor drivers and use SPI1 for the card reader. It's not illustrated in the below image, but I also broke off the male header pins just to make sure that there was no connection at all to the motor drivers. I soldered a female header onto the 3.3v pin because I'm already using the Explorer's breakout for something else. I just unplugged it all to simplify the image.

Pimoroni Pico Explorer with SD Card Reader Hacked In