mirror of
				https://github.com/eledio-devices/thirdparty-littlefs.git
				synced 2025-11-01 00:38:29 +01:00 
			
		
		
		
	This provides a limited form of wear leveling. While wear is not actually balanced across blocks, the filesystem can recover from corrupted blocks and extend the lifetime of a device nearly as much as dynamic wear leveling. For use-cases where wear is important, it would be better to use a full form of dynamic wear-leveling at the block level. (or consider a logging filesystem). Corrupted block handling was simply added on top of the existing logic in place for the filesystem, so it's a bit more noodly than it may have to be, but it gets the work done.
		
			
				
	
	
		
			107 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			107 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
	
	
	
| #!/bin/bash
 | |
| set -eu
 | |
| 
 | |
| echo "=== Corrupt tests ==="
 | |
| 
 | |
| NAMEMULT=64
 | |
| FILEMULT=1
 | |
| 
 | |
| lfs_mktree() {
 | |
| tests/test.py ${1:-} << TEST
 | |
|     lfs_format(&lfs, &cfg) => 0;
 | |
| 
 | |
|     lfs_mount(&lfs, &cfg) => 0;
 | |
|     for (int i = 1; i < 10; i++) {
 | |
|         for (int j = 0; j < $NAMEMULT; j++) {
 | |
|             buffer[j] = '0'+i;
 | |
|         }
 | |
|         buffer[$NAMEMULT] = '\0';
 | |
|         lfs_mkdir(&lfs, (char*)buffer) => 0;
 | |
| 
 | |
|         buffer[$NAMEMULT] = '/';
 | |
|         for (int j = 0; j < $NAMEMULT; j++) {
 | |
|             buffer[j+$NAMEMULT+1] = '0'+i;
 | |
|         }
 | |
|         buffer[2*$NAMEMULT+1] = '\0';
 | |
|         lfs_file_open(&lfs, &file[0], (char*)buffer,
 | |
|                 LFS_O_WRONLY | LFS_O_CREAT) => 0;
 | |
|         
 | |
|         size = $NAMEMULT;
 | |
|         for (int j = 0; j < i*$FILEMULT; j++) {
 | |
|             lfs_file_write(&lfs, &file[0], buffer, size) => size;
 | |
|         }
 | |
| 
 | |
|         lfs_file_close(&lfs, &file[0]) => 0;
 | |
|     }
 | |
|     lfs_unmount(&lfs) => 0;
 | |
| TEST
 | |
| }
 | |
| 
 | |
| lfs_chktree() {
 | |
| tests/test.py ${1:-} << TEST
 | |
|     lfs_mount(&lfs, &cfg) => 0;
 | |
|     for (int i = 1; i < 10; i++) {
 | |
|         for (int j = 0; j < $NAMEMULT; j++) {
 | |
|             buffer[j] = '0'+i;
 | |
|         }
 | |
|         buffer[$NAMEMULT] = '\0';
 | |
|         lfs_stat(&lfs, (char*)buffer, &info) => 0;
 | |
|         info.type => LFS_TYPE_DIR;
 | |
| 
 | |
|         buffer[$NAMEMULT] = '/';
 | |
|         for (int j = 0; j < $NAMEMULT; j++) {
 | |
|             buffer[j+$NAMEMULT+1] = '0'+i;
 | |
|         }
 | |
|         buffer[2*$NAMEMULT+1] = '\0';
 | |
|         lfs_file_open(&lfs, &file[0], (char*)buffer, LFS_O_RDONLY) => 0;
 | |
|         
 | |
|         size = $NAMEMULT;
 | |
|         for (int j = 0; j < i*$FILEMULT; j++) {
 | |
|             lfs_file_read(&lfs, &file[0], rbuffer, size) => size;
 | |
|             memcmp(buffer, rbuffer, size) => 0;
 | |
|         }
 | |
| 
 | |
|         lfs_file_close(&lfs, &file[0]) => 0;
 | |
|     }
 | |
|     lfs_unmount(&lfs) => 0;
 | |
| TEST
 | |
| }
 | |
| 
 | |
| echo "--- Sanity check ---"
 | |
| rm -rf blocks
 | |
| lfs_mktree
 | |
| lfs_chktree
 | |
| 
 | |
| echo "--- Block corruption ---"
 | |
| for i in {0..33}
 | |
| do 
 | |
|     rm -rf blocks
 | |
|     mkdir blocks
 | |
|     ln -s /dev/zero blocks/$(printf '%x' $i)
 | |
|     lfs_mktree
 | |
|     lfs_chktree
 | |
| done
 | |
| 
 | |
| echo "--- Big region corruption ---"
 | |
| rm -rf blocks
 | |
| mkdir blocks
 | |
| for i in {2..255}
 | |
| do
 | |
|     ln -s /dev/zero blocks/$(printf '%x' $i)
 | |
| done
 | |
| lfs_mktree
 | |
| lfs_chktree
 | |
| 
 | |
| echo "--- Alternating corruption ---"
 | |
| rm -rf blocks
 | |
| mkdir blocks
 | |
| for i in {2..511..2}
 | |
| do
 | |
|     ln -s /dev/zero blocks/$(printf '%x' $i)
 | |
| done
 | |
| lfs_mktree
 | |
| lfs_chktree
 | |
| 
 | |
| echo "--- Results ---"
 | |
| tests/stats.py
 |