2014-12-22 12:35:05 +01:00
|
|
|
// Module for interfacing with file system
|
|
|
|
|
2015-12-16 06:04:58 +01:00
|
|
|
#include "module.h"
|
2014-12-22 12:35:05 +01:00
|
|
|
#include "lauxlib.h"
|
|
|
|
#include "platform.h"
|
|
|
|
|
|
|
|
#include "c_types.h"
|
|
|
|
#include "flash_fs.h"
|
2016-05-26 10:36:20 +02:00
|
|
|
#include <string.h>
|
2014-12-22 12:35:05 +01:00
|
|
|
|
2015-01-07 08:57:17 +01:00
|
|
|
static volatile int file_fd = FS_OPEN_OK - 1;
|
2014-12-22 12:35:05 +01:00
|
|
|
|
|
|
|
// Lua: open(filename, mode)
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_open( lua_State* L )
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
if((FS_OPEN_OK - 1)!=file_fd){
|
|
|
|
fs_close(file_fd);
|
|
|
|
file_fd = FS_OPEN_OK - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *fname = luaL_checklstring( L, 1, &len );
|
2016-05-26 10:36:20 +02:00
|
|
|
luaL_argcheck(L, len < FS_NAME_MAX_LENGTH && strlen(fname) == len, 1, "filename invalid");
|
2016-02-14 00:58:04 +01:00
|
|
|
|
2014-12-22 12:35:05 +01:00
|
|
|
const char *mode = luaL_optstring(L, 2, "r");
|
|
|
|
|
|
|
|
file_fd = fs_open(fname, fs_mode2flag(mode));
|
|
|
|
|
|
|
|
if(file_fd < FS_OPEN_OK){
|
|
|
|
file_fd = FS_OPEN_OK - 1;
|
|
|
|
lua_pushnil(L);
|
|
|
|
} else {
|
|
|
|
lua_pushboolean(L, 1);
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lua: close()
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_close( lua_State* L )
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
|
|
|
if((FS_OPEN_OK - 1)!=file_fd){
|
|
|
|
fs_close(file_fd);
|
|
|
|
file_fd = FS_OPEN_OK - 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lua: format()
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_format( lua_State* L )
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
file_close(L);
|
|
|
|
if( !fs_format() )
|
|
|
|
{
|
|
|
|
NODE_ERR( "\ni*** ERROR ***: unable to format. FS might be compromised.\n" );
|
2015-01-07 08:57:17 +01:00
|
|
|
NODE_ERR( "It is advised to re-flash the NodeMCU image.\n" );
|
2014-12-22 12:35:05 +01:00
|
|
|
}
|
|
|
|
else{
|
|
|
|
NODE_ERR( "format done.\n" );
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-02-26 17:37:06 +01:00
|
|
|
#if defined(BUILD_SPIFFS)
|
2014-12-22 12:35:05 +01:00
|
|
|
|
|
|
|
extern spiffs fs;
|
|
|
|
|
|
|
|
// Lua: list()
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_list( lua_State* L )
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
|
|
|
spiffs_DIR d;
|
|
|
|
struct spiffs_dirent e;
|
|
|
|
struct spiffs_dirent *pe = &e;
|
|
|
|
|
|
|
|
lua_newtable( L );
|
|
|
|
SPIFFS_opendir(&fs, "/", &d);
|
|
|
|
while ((pe = SPIFFS_readdir(&d, pe))) {
|
|
|
|
// NODE_ERR(" %s size:%i\n", pe->name, pe->size);
|
|
|
|
lua_pushinteger(L, pe->size);
|
|
|
|
lua_setfield( L, -2, pe->name );
|
|
|
|
}
|
|
|
|
SPIFFS_closedir(&d);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_seek (lua_State *L)
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
|
|
|
static const int mode[] = {FS_SEEK_SET, FS_SEEK_CUR, FS_SEEK_END};
|
|
|
|
static const char *const modenames[] = {"set", "cur", "end", NULL};
|
|
|
|
if((FS_OPEN_OK - 1)==file_fd)
|
|
|
|
return luaL_error(L, "open a file first");
|
|
|
|
int op = luaL_checkoption(L, 1, "cur", modenames);
|
|
|
|
long offset = luaL_optlong(L, 2, 0);
|
|
|
|
op = fs_seek(file_fd, offset, mode[op]);
|
2015-07-16 08:11:22 +02:00
|
|
|
if (op < 0)
|
2015-02-25 20:34:07 +01:00
|
|
|
lua_pushnil(L); /* error */
|
2014-12-22 12:35:05 +01:00
|
|
|
else
|
|
|
|
lua_pushinteger(L, fs_tell(file_fd));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-02-13 23:37:04 +01:00
|
|
|
// Lua: exists(filename)
|
|
|
|
static int file_exists( lua_State* L )
|
|
|
|
{
|
|
|
|
size_t len;
|
2016-02-14 00:58:04 +01:00
|
|
|
const char *fname = luaL_checklstring( L, 1, &len );
|
|
|
|
luaL_argcheck(L, len <= FS_NAME_MAX_LENGTH, 1, "filename too long");
|
2016-02-13 23:37:04 +01:00
|
|
|
|
|
|
|
spiffs_stat stat;
|
|
|
|
int rc = SPIFFS_stat(&fs, (char *)fname, &stat);
|
|
|
|
|
|
|
|
lua_pushboolean(L, (rc == SPIFFS_OK ? 1 : 0));
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-12-22 12:35:05 +01:00
|
|
|
// Lua: remove(filename)
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_remove( lua_State* L )
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
|
|
|
size_t len;
|
2016-02-14 00:58:04 +01:00
|
|
|
const char *fname = luaL_checklstring( L, 1, &len );
|
|
|
|
luaL_argcheck(L, len <= FS_NAME_MAX_LENGTH, 1, "filename too long");
|
2014-12-22 12:35:05 +01:00
|
|
|
file_close(L);
|
2015-02-11 14:20:54 +01:00
|
|
|
SPIFFS_remove(&fs, (char *)fname);
|
2014-12-22 12:35:05 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lua: flush()
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_flush( lua_State* L )
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
|
|
|
if((FS_OPEN_OK - 1)==file_fd)
|
|
|
|
return luaL_error(L, "open a file first");
|
|
|
|
if(fs_flush(file_fd) == 0)
|
|
|
|
lua_pushboolean(L, 1);
|
|
|
|
else
|
|
|
|
lua_pushnil(L);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
#if 0
|
|
|
|
// Lua: check()
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_check( lua_State* L )
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
|
|
|
file_close(L);
|
|
|
|
lua_pushinteger(L, fs_check());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-02-11 14:20:54 +01:00
|
|
|
// Lua: rename("oldname", "newname")
|
|
|
|
static int file_rename( lua_State* L )
|
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
if((FS_OPEN_OK - 1)!=file_fd){
|
|
|
|
fs_close(file_fd);
|
|
|
|
file_fd = FS_OPEN_OK - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *oldname = luaL_checklstring( L, 1, &len );
|
2016-02-14 00:58:04 +01:00
|
|
|
luaL_argcheck(L, len <= FS_NAME_MAX_LENGTH, 1, "filename too long");
|
|
|
|
|
|
|
|
const char *newname = luaL_checklstring( L, 2, &len );
|
|
|
|
luaL_argcheck(L, len <= FS_NAME_MAX_LENGTH, 2, "filename too long");
|
2015-02-11 14:20:54 +01:00
|
|
|
|
|
|
|
if(SPIFFS_OK==myspiffs_rename( oldname, newname )){
|
|
|
|
lua_pushboolean(L, 1);
|
|
|
|
} else {
|
|
|
|
lua_pushboolean(L, 0);
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-03-09 18:12:36 +01:00
|
|
|
// Lua: fsinfo()
|
|
|
|
static int file_fsinfo( lua_State* L )
|
|
|
|
{
|
2015-12-20 04:39:17 +01:00
|
|
|
u32_t total, used;
|
2015-03-09 18:12:36 +01:00
|
|
|
SPIFFS_info(&fs, &total, &used);
|
|
|
|
NODE_DBG("total: %d, used:%d\n", total, used);
|
|
|
|
if(total>0x7FFFFFFF || used>0x7FFFFFFF || used > total)
|
|
|
|
{
|
|
|
|
return luaL_error(L, "file system error");;
|
|
|
|
}
|
|
|
|
lua_pushinteger(L, total-used);
|
|
|
|
lua_pushinteger(L, used);
|
|
|
|
lua_pushinteger(L, total);
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
2014-12-22 12:35:05 +01:00
|
|
|
#endif
|
|
|
|
|
2014-12-31 06:53:26 +01:00
|
|
|
// g_read()
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_g_read( lua_State* L, int n, int16_t end_char )
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
2015-07-31 08:29:29 +02:00
|
|
|
if(n <= 0 || n > LUAL_BUFFERSIZE)
|
2014-12-31 06:53:26 +01:00
|
|
|
n = LUAL_BUFFERSIZE;
|
|
|
|
if(end_char < 0 || end_char >255)
|
|
|
|
end_char = EOF;
|
2015-01-08 09:03:21 +01:00
|
|
|
|
2014-12-22 12:35:05 +01:00
|
|
|
luaL_Buffer b;
|
|
|
|
if((FS_OPEN_OK - 1)==file_fd)
|
|
|
|
return luaL_error(L, "open a file first");
|
|
|
|
|
|
|
|
luaL_buffinit(L, &b);
|
|
|
|
char *p = luaL_prepbuffer(&b);
|
2015-07-31 08:29:29 +02:00
|
|
|
int i;
|
2014-12-22 12:35:05 +01:00
|
|
|
|
2015-07-31 08:29:29 +02:00
|
|
|
n = fs_read(file_fd, p, n);
|
|
|
|
for (i = 0; i < n; ++i)
|
|
|
|
if (p[i] == end_char)
|
|
|
|
{
|
|
|
|
++i;
|
2014-12-22 12:35:05 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(i==0){
|
|
|
|
luaL_pushresult(&b); /* close buffer */
|
|
|
|
return (lua_objlen(L, -1) > 0); /* check whether read something */
|
|
|
|
}
|
|
|
|
|
2015-07-31 08:29:29 +02:00
|
|
|
fs_seek(file_fd, -(n - i), SEEK_CUR);
|
2014-12-22 12:35:05 +01:00
|
|
|
luaL_addsize(&b, i);
|
|
|
|
luaL_pushresult(&b); /* close buffer */
|
|
|
|
return 1; /* read at least an `eol' */
|
|
|
|
}
|
|
|
|
|
2014-12-31 06:53:26 +01:00
|
|
|
// Lua: read()
|
|
|
|
// file.read() will read all byte in file
|
|
|
|
// file.read(10) will read 10 byte from file, or EOF is reached.
|
|
|
|
// file.read('q') will read until 'q' or EOF is reached.
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_read( lua_State* L )
|
2014-12-31 06:53:26 +01:00
|
|
|
{
|
|
|
|
unsigned need_len = LUAL_BUFFERSIZE;
|
|
|
|
int16_t end_char = EOF;
|
|
|
|
size_t el;
|
|
|
|
if( lua_type( L, 1 ) == LUA_TNUMBER )
|
|
|
|
{
|
|
|
|
need_len = ( unsigned )luaL_checkinteger( L, 1 );
|
|
|
|
if( need_len > LUAL_BUFFERSIZE ){
|
|
|
|
need_len = LUAL_BUFFERSIZE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(lua_isstring(L, 1))
|
|
|
|
{
|
|
|
|
const char *end = luaL_checklstring( L, 1, &el );
|
|
|
|
if(el!=1){
|
|
|
|
return luaL_error( L, "wrong arg range" );
|
|
|
|
}
|
|
|
|
end_char = (int16_t)end[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
return file_g_read(L, need_len, end_char);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lua: readline()
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_readline( lua_State* L )
|
2014-12-31 06:53:26 +01:00
|
|
|
{
|
|
|
|
return file_g_read(L, LUAL_BUFFERSIZE, '\n');
|
|
|
|
}
|
|
|
|
|
2014-12-22 12:35:05 +01:00
|
|
|
// Lua: write("string")
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_write( lua_State* L )
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
|
|
|
if((FS_OPEN_OK - 1)==file_fd)
|
|
|
|
return luaL_error(L, "open a file first");
|
|
|
|
size_t l, rl;
|
|
|
|
const char *s = luaL_checklstring(L, 1, &l);
|
|
|
|
rl = fs_write(file_fd, s, l);
|
|
|
|
if(rl==l)
|
|
|
|
lua_pushboolean(L, 1);
|
|
|
|
else
|
|
|
|
lua_pushnil(L);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lua: writeline("string")
|
2015-01-05 03:09:51 +01:00
|
|
|
static int file_writeline( lua_State* L )
|
2014-12-22 12:35:05 +01:00
|
|
|
{
|
|
|
|
if((FS_OPEN_OK - 1)==file_fd)
|
|
|
|
return luaL_error(L, "open a file first");
|
|
|
|
size_t l, rl;
|
|
|
|
const char *s = luaL_checklstring(L, 1, &l);
|
|
|
|
rl = fs_write(file_fd, s, l);
|
|
|
|
if(rl==l){
|
|
|
|
rl = fs_write(file_fd, "\n", 1);
|
|
|
|
if(rl==1)
|
|
|
|
lua_pushboolean(L, 1);
|
|
|
|
else
|
|
|
|
lua_pushnil(L);
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
lua_pushnil(L);
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-01-12 07:53:45 +01:00
|
|
|
static int file_fscfg (lua_State *L)
|
|
|
|
{
|
|
|
|
lua_pushinteger (L, fs.cfg.phys_addr);
|
|
|
|
lua_pushinteger (L, fs.cfg.phys_size);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
2014-12-22 12:35:05 +01:00
|
|
|
// Module function map
|
2015-12-16 06:04:58 +01:00
|
|
|
static const LUA_REG_TYPE file_map[] = {
|
2015-12-13 03:29:37 +01:00
|
|
|
{ LSTRKEY( "list" ), LFUNCVAL( file_list ) },
|
|
|
|
{ LSTRKEY( "open" ), LFUNCVAL( file_open ) },
|
|
|
|
{ LSTRKEY( "close" ), LFUNCVAL( file_close ) },
|
|
|
|
{ LSTRKEY( "write" ), LFUNCVAL( file_write ) },
|
2014-12-22 12:35:05 +01:00
|
|
|
{ LSTRKEY( "writeline" ), LFUNCVAL( file_writeline ) },
|
2015-12-13 03:29:37 +01:00
|
|
|
{ LSTRKEY( "read" ), LFUNCVAL( file_read ) },
|
|
|
|
{ LSTRKEY( "readline" ), LFUNCVAL( file_readline ) },
|
|
|
|
{ LSTRKEY( "format" ), LFUNCVAL( file_format ) },
|
2016-02-26 17:37:06 +01:00
|
|
|
#if defined(BUILD_SPIFFS)
|
2015-12-13 03:29:37 +01:00
|
|
|
{ LSTRKEY( "remove" ), LFUNCVAL( file_remove ) },
|
|
|
|
{ LSTRKEY( "seek" ), LFUNCVAL( file_seek ) },
|
|
|
|
{ LSTRKEY( "flush" ), LFUNCVAL( file_flush ) },
|
|
|
|
{ LSTRKEY( "rename" ), LFUNCVAL( file_rename ) },
|
|
|
|
{ LSTRKEY( "fsinfo" ), LFUNCVAL( file_fsinfo ) },
|
2016-02-26 17:37:06 +01:00
|
|
|
{ LSTRKEY( "fscfg" ), LFUNCVAL( file_fscfg ) },
|
2016-02-13 23:37:04 +01:00
|
|
|
{ LSTRKEY( "exists" ), LFUNCVAL( file_exists ) },
|
2014-12-22 12:35:05 +01:00
|
|
|
#endif
|
|
|
|
{ LNILKEY, LNILVAL }
|
|
|
|
};
|
|
|
|
|
2015-12-16 06:04:58 +01:00
|
|
|
NODEMCU_MODULE(FILE, "file", file_map, NULL);
|