132 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
			
		
		
	
	
			132 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
# File system
 | 
						|
 | 
						|
LVGL has a 'File system' abstraction module that enables you to attach any type of file system.
 | 
						|
A file system is identified by an assigned drive letter.
 | 
						|
For example, if an SD card is associated with the letter `'S'`, a file can be reached using `"S:path/to/file.txt"`.
 | 
						|
 | 
						|
## Ready to use drivers
 | 
						|
The [lv_fs_if](https://github.com/lvgl/lv_fs_if) repository contains prepared drivers using POSIX, standard C and the [FATFS](http://elm-chan.org/fsw/ff/00index_e.html) API.
 | 
						|
See its [README](https://github.com/lvgl/lv_fs_if#readme) for the details.
 | 
						|
 | 
						|
## Adding a driver
 | 
						|
 | 
						|
### Registering a driver
 | 
						|
To add a driver, a `lv_fs_drv_t` needs to be initialized like below. The `lv_fs_drv_t` needs to be static, global or dynamically allocated and not a local variable.
 | 
						|
```c
 | 
						|
static lv_fs_drv_t drv;                   /*Needs to be static or global*/
 | 
						|
lv_fs_drv_init(&drv);                     /*Basic initialization*/
 | 
						|
 | 
						|
drv.letter = 'S';                         /*An uppercase letter to identify the drive */
 | 
						|
drv.cache_size = my_cache_size;           /*Cache size for reading in bytes. 0 to not cache.*/
 | 
						|
 | 
						|
drv.ready_cb = my_ready_cb;               /*Callback to tell if the drive is ready to use */
 | 
						|
drv.open_cb = my_open_cb;                 /*Callback to open a file */
 | 
						|
drv.close_cb = my_close_cb;               /*Callback to close a file */
 | 
						|
drv.read_cb = my_read_cb;                 /*Callback to read a file */
 | 
						|
drv.write_cb = my_write_cb;               /*Callback to write a file */
 | 
						|
drv.seek_cb = my_seek_cb;                 /*Callback to seek in a file (Move cursor) */
 | 
						|
drv.tell_cb = my_tell_cb;                 /*Callback to tell the cursor position  */
 | 
						|
 | 
						|
drv.dir_open_cb = my_dir_open_cb;         /*Callback to open directory to read its content */
 | 
						|
drv.dir_read_cb = my_dir_read_cb;         /*Callback to read a directory's content */
 | 
						|
drv.dir_close_cb = my_dir_close_cb;       /*Callback to close a directory */
 | 
						|
 | 
						|
drv.user_data = my_user_data;             /*Any custom data if required*/
 | 
						|
 | 
						|
lv_fs_drv_register(&drv);                 /*Finally register the drive*/
 | 
						|
 | 
						|
```
 | 
						|
 | 
						|
Any of the callbacks can be `NULL` to indicate that operation is not supported.
 | 
						|
 | 
						|
 | 
						|
### Implementing the callbacks
 | 
						|
 | 
						|
#### Open callback
 | 
						|
The prototype of `open_cb` looks like this:
 | 
						|
```c
 | 
						|
void * (*open_cb)(lv_fs_drv_t * drv, const char * path, lv_fs_mode_t mode);
 | 
						|
```
 | 
						|
 | 
						|
`path` is the path after the drive letter (e.g. "S:path/to/file.txt" -> "path/to/file.txt"). `mode` can be `LV_FS_MODE_WR` or `LV_FS_MODE_RD` to open for writes or reads.
 | 
						|
 | 
						|
The return value is a pointer to a *file object* that describes the opened file or `NULL` if there were any issues (e.g. the file wasn't found).
 | 
						|
The returned file object will be passed to other file system related callbacks. (see below)
 | 
						|
 | 
						|
### Other callbacks
 | 
						|
The other callbacks are quite similar. For example `write_cb` looks like this:
 | 
						|
```c
 | 
						|
lv_fs_res_t (*write_cb)(lv_fs_drv_t * drv, void * file_p, const void * buf, uint32_t btw, uint32_t * bw);
 | 
						|
```
 | 
						|
 | 
						|
For `file_p`, LVGL passes the return value of `open_cb`, `buf` is the data to write, `btw` is the Bytes To Write, `bw` is the actually written bytes.
 | 
						|
 | 
						|
For a template of these callbacks see [lv_fs_template.c](https://github.com/lvgl/lvgl/blob/master/examples/porting/lv_port_fs_template.c).
 | 
						|
 | 
						|
 | 
						|
## Usage example
 | 
						|
 | 
						|
The example below shows how to read from a file:
 | 
						|
```c
 | 
						|
lv_fs_file_t f;
 | 
						|
lv_fs_res_t res;
 | 
						|
res = lv_fs_open(&f, "S:folder/file.txt", LV_FS_MODE_RD);
 | 
						|
if(res != LV_FS_RES_OK) my_error_handling();
 | 
						|
 | 
						|
uint32_t read_num;
 | 
						|
uint8_t buf[8];
 | 
						|
res = lv_fs_read(&f, buf, 8, &read_num);
 | 
						|
if(res != LV_FS_RES_OK || read_num != 8) my_error_handling();
 | 
						|
 | 
						|
lv_fs_close(&f);
 | 
						|
```
 | 
						|
*The mode in `lv_fs_open` can be `LV_FS_MODE_WR` to open for writes only or `LV_FS_MODE_RD | LV_FS_MODE_WR` for both*
 | 
						|
 | 
						|
This example shows how to read a directory's content. It's up to the driver how to mark directories in the result but it can be a good practice to insert a `'/'` in front of each directory name.
 | 
						|
```c
 | 
						|
lv_fs_dir_t dir;
 | 
						|
lv_fs_res_t res;
 | 
						|
res = lv_fs_dir_open(&dir, "S:/folder");
 | 
						|
if(res != LV_FS_RES_OK) my_error_handling();
 | 
						|
 | 
						|
char fn[256];
 | 
						|
while(1) {
 | 
						|
    res = lv_fs_dir_read(&dir, fn);
 | 
						|
    if(res != LV_FS_RES_OK) {
 | 
						|
        my_error_handling();
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    /*fn is empty, if not more files to read*/
 | 
						|
    if(strlen(fn) == 0) {
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    printf("%s\n", fn);
 | 
						|
}
 | 
						|
 | 
						|
lv_fs_dir_close(&dir);
 | 
						|
```
 | 
						|
 | 
						|
## Use drives for images
 | 
						|
 | 
						|
[Image](/widgets/core/img) objects can be opened from files too (besides variables stored in the compiled program).
 | 
						|
 | 
						|
To use files in image widgets the following callbacks are required:
 | 
						|
- open
 | 
						|
- close
 | 
						|
- read
 | 
						|
- seek
 | 
						|
- tell
 | 
						|
 | 
						|
 | 
						|
 | 
						|
## API
 | 
						|
 | 
						|
```eval_rst
 | 
						|
 | 
						|
.. doxygenfile:: lv_fs.h
 | 
						|
  :project: lvgl
 | 
						|
 | 
						|
```
 |