1 #include "../../../lvgl.h"
2 #if LV_USE_FS_LITTLEFS
3 
4 #include "lfs.h"
5 #include "../../core/lv_global.h"
6 
7 #if !LV_FS_IS_VALID_LETTER(LV_FS_LITTLEFS_LETTER)
8     #error "Invalid drive letter"
9 #endif
10 
11 typedef struct LittleFile {
12     lfs_file_t file;
13 } LittleFile;
14 
15 typedef struct LittleDirectory {
16     lfs_dir_t dir;
17 } LittleDirectory;
18 
19 /**********************
20  *  STATIC PROTOTYPES
21  **********************/
22 static void * fs_open(lv_fs_drv_t * drv, const char * path, lv_fs_mode_t mode);
23 static lv_fs_res_t fs_close(lv_fs_drv_t * drv, void * file_p);
24 static lv_fs_res_t fs_read(lv_fs_drv_t * drv, void * file_p, void * buf, uint32_t btr, uint32_t * br);
25 static lv_fs_res_t fs_write(lv_fs_drv_t * drv, void * file_p, const void * buf, uint32_t btw, uint32_t * bw);
26 static lv_fs_res_t fs_seek(lv_fs_drv_t * drv, void * file_p, uint32_t pos, lv_fs_whence_t whence);
27 static lv_fs_res_t fs_tell(lv_fs_drv_t * drv, void * file_p, uint32_t * pos_p);
28 static void * fs_dir_open(lv_fs_drv_t * drv, const char * path);
29 static lv_fs_res_t fs_dir_close(lv_fs_drv_t * drv, void * dir_p);
30 static lv_fs_res_t fs_dir_read(lv_fs_drv_t * drv, void * dir_p, char * fn, uint32_t fn_len);
31 
lv_littlefs_set_handler(lfs_t * lfs)32 void lv_littlefs_set_handler(lfs_t * lfs)
33 {
34     lv_fs_drv_t * drv = lv_fs_get_drv(LV_FS_LITTLEFS_LETTER);
35     drv->user_data = lfs;
36 }
37 
38 /**
39  * Register a driver for the LittleFS File System interface
40  */
lv_fs_littlefs_init(void)41 void lv_fs_littlefs_init(void)
42 {
43     lv_fs_drv_t * fs_drv = &(LV_GLOBAL_DEFAULT()->littlefs_fs_drv);
44     lv_fs_drv_init(fs_drv);
45 
46     fs_drv->letter = LV_FS_LITTLEFS_LETTER;
47     fs_drv->open_cb = fs_open;
48     fs_drv->close_cb = fs_close;
49     fs_drv->read_cb = fs_read;
50     fs_drv->write_cb = fs_write;
51     fs_drv->seek_cb = fs_seek;
52     fs_drv->tell_cb = fs_tell;
53 
54     fs_drv->dir_open_cb = fs_dir_open;
55     fs_drv->dir_close_cb = fs_dir_close;
56     fs_drv->dir_read_cb = fs_dir_read;
57 
58     lv_fs_drv_register(fs_drv);
59 }
60 
61 /**********************
62  *   STATIC FUNCTIONS
63  **********************/
64 
65 /**
66  * Open a file
67  * @param drv       pointer to a driver where this function belongs
68  * @param path      path to the file beginning with the driver letter (e.g. S:/folder/file.txt)
69  * @param mode      read: FS_MODE_RD, write: FS_MODE_WR, both: FS_MODE_RD | FS_MODE_WR
70  * @return          a file descriptor or NULL on error
71  */
fs_open(lv_fs_drv_t * drv,const char * path,lv_fs_mode_t mode)72 static void * fs_open(lv_fs_drv_t * drv, const char * path, lv_fs_mode_t mode)
73 {
74     int flags = 0;
75     if(mode == LV_FS_MODE_WR)
76         flags = LFS_O_WRONLY;
77     else if(mode == LV_FS_MODE_RD)
78         flags = LFS_O_RDONLY;
79     else if(mode == (LV_FS_MODE_WR | LV_FS_MODE_RD))
80         flags = LFS_O_RDWR;
81 
82     LittleFile * lf = lv_malloc(sizeof(LittleFile));
83     LV_ASSERT_MALLOC(lf);
84 
85     char buf[LV_FS_MAX_PATH_LEN];
86     lv_snprintf(buf, sizeof(buf), LV_FS_LITTLEFS_PATH "%s", path);
87 
88     lfs_t * lfs = drv->user_data;
89     int err = lfs_file_open(lfs, &lf->file, buf, flags);
90     if(err) {
91         return NULL;
92     }
93 
94     return (void *)lf;
95 }
96 
97 /**
98  * Close an opened file
99  * @param drv       pointer to a driver where this function belongs
100  * @param file_p    pointer to a file_t variable. (opened with fs_open)
101  * @return          LV_FS_RES_OK: no error or  any error from @lv_fs_res_t enum
102  */
fs_close(lv_fs_drv_t * drv,void * file_p)103 static lv_fs_res_t fs_close(lv_fs_drv_t * drv, void * file_p)
104 {
105     LittleFile * lf = file_p;
106 
107     lfs_t * lfs = drv->user_data;
108     lfs_file_close(lfs, &lf->file);
109     lv_free(lf);
110 
111     return LV_FS_RES_OK;
112 }
113 
114 /**
115  * Read data from an opened file
116  * @param drv       pointer to a driver where this function belongs
117  * @param file_p    pointer to a file_t variable.
118  * @param buf       pointer to a memory block where to store the read data
119  * @param btr       number of Bytes To Read
120  * @param br        the real number of read bytes (Byte Read)
121  * @return          LV_FS_RES_OK: no error or any error from @lv_fs_res_t enum
122  */
fs_read(lv_fs_drv_t * drv,void * file_p,void * buf,uint32_t btr,uint32_t * br)123 static lv_fs_res_t fs_read(lv_fs_drv_t * drv, void * file_p, void * buf, uint32_t btr, uint32_t * br)
124 {
125     LittleFile * lf = file_p;
126 
127     lfs_t * lfs = drv->user_data;
128     *br = lfs_file_read(lfs, &lf->file, (uint8_t *)buf, btr);
129 
130     return (int32_t)(*br) < 0 ? LV_FS_RES_UNKNOWN : LV_FS_RES_OK;
131 }
132 
133 /**
134  * Write into a file
135  * @param drv       pointer to a driver where this function belongs
136  * @param file_p    pointer to a file_t variable
137  * @param buf       pointer to a buffer with the bytes to write
138  * @param btw       Bytes To Write
139  * @param bw        the number of real written bytes (Bytes Written)
140  * @return          LV_FS_RES_OK: no error or  any error from @lv_fs_res_t enum
141  */
fs_write(lv_fs_drv_t * drv,void * file_p,const void * buf,uint32_t btw,uint32_t * bw)142 static lv_fs_res_t fs_write(lv_fs_drv_t * drv, void * file_p, const void * buf, uint32_t btw, uint32_t * bw)
143 {
144     LittleFile * lf = file_p;
145 
146     lfs_t * lfs = drv->user_data;
147     *bw = lfs_file_write(lfs, &lf->file, (uint8_t *)buf, btw);
148 
149     return (int32_t)(*bw) < 0 ? LV_FS_RES_UNKNOWN : LV_FS_RES_OK;
150 }
151 
152 /**
153  * Set the read write pointer. Also expand the file size if necessary.
154  * @param drv       pointer to a driver where this function belongs
155  * @param file_p    pointer to a file_t variable. (opened with fs_open)
156  * @param pos       the new position of read write pointer
157  * @param whence    tells from where to interpret the `pos`. See @lv_fs_whence_t
158  * @return          LV_FS_RES_OK: no error or any error from @lv_fs_res_t enum
159  */
fs_seek(lv_fs_drv_t * drv,void * file_p,uint32_t pos,lv_fs_whence_t whence)160 static lv_fs_res_t fs_seek(lv_fs_drv_t * drv, void * file_p, uint32_t pos, lv_fs_whence_t whence)
161 {
162     int mode = 0;
163     if(whence == LV_FS_SEEK_SET)
164         mode = LFS_SEEK_SET;
165     else if(whence == LV_FS_SEEK_CUR)
166         mode = LFS_SEEK_CUR;
167     else if(whence == LV_FS_SEEK_END)
168         mode = LFS_SEEK_END;
169 
170     LittleFile * lf = file_p;
171 
172     lfs_t * lfs = drv->user_data;
173     int rc = lfs_file_seek(lfs, &lf->file, pos, mode);
174 
175     return rc < 0 ? LV_FS_RES_UNKNOWN : LV_FS_RES_OK;
176 }
177 
178 /**
179  * Give the position of the read write pointer
180  * @param drv       pointer to a driver where this function belongs
181  * @param file_p    pointer to a file_p variable
182  * @param pos_p     pointer to store the result
183  * @return          LV_FS_RES_OK: no error or any error from @lv_fs_res_t enum
184  */
fs_tell(lv_fs_drv_t * drv,void * file_p,uint32_t * pos_p)185 static lv_fs_res_t fs_tell(lv_fs_drv_t * drv, void * file_p, uint32_t * pos_p)
186 {
187     LittleFile * lf = file_p;
188 
189     lfs_t * lfs = drv->user_data;
190     *pos_p = lfs_file_tell(lfs, &lf->file);
191 
192     return (int32_t)(*pos_p) < 0 ? LV_FS_RES_UNKNOWN : LV_FS_RES_OK;
193 }
194 
195 /**
196  * Open a directory
197  * @param drv       pointer to a driver where this function belongs
198  * @param path      path to the directory beginning with the driver letter (e.g. S:/folder)
199  * @return          a directory descriptor or NULL on error
200  */
fs_dir_open(lv_fs_drv_t * drv,const char * path)201 static void * fs_dir_open(lv_fs_drv_t * drv, const char * path)
202 {
203     LittleDirectory * ld = lv_malloc(sizeof(LittleDirectory));
204     LV_ASSERT_MALLOC(ld);
205 
206     char buf[LV_FS_MAX_PATH_LEN];
207     lv_snprintf(buf, sizeof(buf), LV_FS_LITTLEFS_PATH "%s", path);
208 
209     lfs_t * lfs = drv->user_data;
210     int err = lfs_dir_open(lfs, &ld->dir, buf);
211     if(err != LFS_ERR_OK) {
212         lv_free(ld);
213         return NULL;
214     }
215 
216     return (void *)ld;
217 }
218 
219 /**
220  * Close an opened directory
221  * @param drv      pointer to a driver where this function belongs
222  * @param dir_p    pointer to a dir_p variable. (opened with fs_dir_open)
223  * @return         LV_FS_RES_OK: no error or any error from @lv_fs_res_t enum
224  */
fs_dir_close(lv_fs_drv_t * drv,void * dir_p)225 static lv_fs_res_t fs_dir_close(lv_fs_drv_t * drv, void * dir_p)
226 {
227     LittleDirectory * ld = dir_p;
228 
229     lfs_t * lfs = drv->user_data;
230     int rc = lfs_dir_close(lfs, &ld->dir);
231 
232     if(rc < 0) return LV_FS_RES_UNKNOWN;
233     lv_free(ld);
234 
235     return LV_FS_RES_OK;
236 }
237 
238 /**
239  * Read data from an opened directory
240  * @param drv      pointer to a driver where this function belongs
241  * @param dir_p    pointer to a file_t variable.
242  * @param fn       pointer to a buffer to store the filename
243  * @param fn_len   length of the buffer to store the filename
244  * @return         LV_FS_RES_OK: no error or any error from @lv_fs_res_t enum
245  */
fs_dir_read(lv_fs_drv_t * drv,void * dir_p,char * fn,uint32_t fn_len)246 static lv_fs_res_t fs_dir_read(lv_fs_drv_t * drv, void * dir_p, char * fn, uint32_t fn_len)
247 {
248     if(fn_len == 0) return LV_FS_RES_INV_PARAM;
249 
250     LittleDirectory * lf = dir_p;
251 
252     lfs_t * lfs = drv->user_data;
253 
254     fn[0] = '\0';
255     do {
256         struct lfs_info info;
257         int res = lfs_dir_read(lfs, &lf->dir, &info);
258 
259         if(res < 0) return LV_FS_RES_UNKNOWN;
260         if(res == 0) { /* End of the directory */
261             fn[0] = '\0';
262             break;
263         }
264 
265         if(info.type != LFS_TYPE_DIR) {
266             lv_strlcpy(fn, info.name, fn_len);
267         }
268         else {
269             lv_snprintf(fn, fn_len, "/%s", info.name);
270         }
271 
272     } while(lv_strcmp(fn, "/.") == 0 || lv_strcmp(fn, "/..") == 0);
273 
274     return LV_FS_RES_OK;
275 }
276 
277 #else /*LV_USE_FS_LITTLEFS == 0*/
278 
279 #if defined(LV_FS_LITTLEFS_LETTER) && LV_FS_LITTLEFS_LETTER != '\0'
280     #warning "LV_USE_FS_LITTLEFS is not enabled but LV_FS_LITTLEFS_LETTER is set"
281 #endif
282 
283 #endif /*LV_USE_FS_LITTLEFS*/
284