Created
June 18, 2024 10:21
-
-
Save Spiritdude/da36d2cf064e49094c870e0a8b9f972f to your computer and use it in GitHub Desktop.
LuckFox Pico Pro/Max SD card writer from .env.txt
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| #!/usr/bin/python3 | |
| # == BLKENVFLASH == written by Rene K. Mueller <[email protected]> | |
| # | |
| # Description: | |
| # Writes an image (.img) or to the SD card direct from existing .env.txt for LuckFox Pico Pro/Max | |
| # | |
| # % ./blkenvflash disk.img | |
| # -- inquery with `lsblk` which device is your SD card resides -- | |
| # % sudo dd if=disk.img of=/dev/sdX bs=1M; sync | |
| # | |
| # -- or do in one step -- | |
| # % sudo ./blkenflash /dev/sdX | |
| # | |
| # License: MIT | |
| # | |
| # History: | |
| # 2024/06/18: after the hint by A. Schuetz https://github.com/LuckfoxTECH/luckfox-pico/issues/129 was able to make script functional | |
| # 2024/06/10: start | |
| import os, sys, re, subprocess, shutil | |
| import atexit | |
| VERSION = '0.0.1' | |
| me = sys.argv.pop(0) | |
| path = os.path.dirname(me) | |
| me = os.path.basename(me) | |
| step = 'sdcard' # -- hard-codec to do one sdcard writing | |
| if len(sys.argv) != 1: | |
| print(f'''USAGE {me} <file or device> | |
| examples: | |
| % ./{me} disk.img | |
| % lsblk -- and check output on which /dev/sdX your SD card resides -- | |
| % sudo dd if=disk.img of=/dev/sdX bs=1M; sync | |
| -- or in one step -- | |
| % sudo ./{me} /dev/sdX | |
| ''') | |
| sys.exit(-1) | |
| dev = sys.argv.pop(0) | |
| print(f"== {me} {VERSION} ==") | |
| print(f"writing to {dev}") | |
| # -- a few helpers to en/decode nice numbers | |
| def true_size(s): | |
| u = { 'B': 1, 'K': 1024, 'M': 1024*1024, 'G': 1024*1024*1024 } | |
| if m := re.search('(\d+)([BKMG])',s): | |
| return int(m.group(1)) * u[m.group(2)] | |
| return 0 | |
| def nice_size(n): | |
| u = { 'B': 1, 'K': 1024, 'M': 1024*1024, 'G': 1024*1024*1024 } | |
| for k,v in reversed(u.items()): | |
| if n >= v and n % v == 0: | |
| return f"{n//v:,}{k}" | |
| return f"{n:,}B" | |
| def cleanup(): | |
| if os.path.exists(".env.txt.orig"): | |
| os.rename(".env.txt.orig",".env.txt") | |
| print("done.") | |
| atexit.register(cleanup) | |
| with open(".env.txt") as fh: | |
| env2 = [ ] | |
| for l in fh.readlines(): | |
| if m := re.search('(blkdevparts|sd_parts)=(\w+)',l): | |
| _type = m.group(1) | |
| _dev = m.group(2) | |
| l = re.sub('blkdevparts=\w+:','',l) | |
| p = l.split(',') | |
| c_off = 0 | |
| if step=='sdcard' and _dev == 'mmcblk1': | |
| for e in p: | |
| #print(e) | |
| if (m := re.search('(\d+[KMG])@(\d+[KMG])\((\w+)\)',e)) or (m := re.search('(\d+[KMG])\((\w+)\)',e)): | |
| if len(m.groups())==3: | |
| size = true_size(m.group(1)) | |
| off = true_size(m.group(2)) | |
| name = m.group(3) | |
| else: | |
| size = true_size(m.group(1)) | |
| name = m.group(2) | |
| off = 0 | |
| print(f" {_dev}: {name}.img size:{size:,}/{nice_size(size)} (offset:{off:,}/{nice_size(off)})",end='') | |
| if os.path.exists(f"{name}.img"): | |
| size_ = os.path.getsize(f"{name}.img") # -- actual size | |
| print(f" imgsize:{size_:,} ({nice_size(size_)})") | |
| if step=='sdcard': | |
| # -- it seems the explicit `off` seems not taking effect? | |
| cmd = ['dd',f"if={name}.img",f"of={dev}","bs=1k",f"seek={c_off//1024}"] | |
| try: | |
| res = subprocess.run(cmd,stdout=subprocess.DEVNULL,stderr=subprocess.DEVNULL,check=True) | |
| except subprocess.CalledProcessError as e: | |
| print(f"ERROR: {e}",e.stderr) | |
| sys.exit(-1) | |
| c_off += size | |
| else: | |
| print() | |
| print(f"ERROR: '{name}.img' not found") | |
| sys.exit(-1) | |
| if _dev == 'mmcblk0': | |
| env2.append(l) | |
| if step=='emmc': # -- untested, not needed as 'upgrade_tool' does this already | |
| shutil.copy(".env.txt",".env.txt.orig") | |
| with open(".env.txt","w") as fh: | |
| fh.write("\n".join(env2)) | |
| # -- prepare upgrade_tool call to write to eMMC | |
| print(f"{path}/upgrade_tool") | |
| subprocess.run([f'{path}/upgrade_tool','uf','update.img']) | |
just found this gem, THANK YOU SO MUCH FOR THIS!
im curious... would it also be possible to have it flash the different parts to actual partitions instead?
I dont understand why they went the route they did since it does contain a valid fs
To anyone looking for a .env.txt but cant find it.Here you go.
blkdevparts=mmcblk1:32K(env),512K@32K(idblock),256K(uboot),32M(boot),512M(oem),256M(userdata),6G(rootfs)
sys_bootargs= root=/dev/mmcblk1p7 rootfstype=ext4 rk_dma_heap_cma=1M
sd_parts=mmcblk0:16K@512(env),512K@32K(idblock),4M(uboot)
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
This is super. Saved me hours? days?
I put a few notes together https://github.com/jouellnyc/RV1103_LuckFox_PicoPlus/blob/main/README.md as well.