Tasmota/lib/libesp32/Berry/src/be_filelib.c
2021-04-12 19:53:35 +02:00

180 lines
4.5 KiB
C

/********************************************************************
** Copyright (c) 2018-2020 Guan Wenliang
** This file is part of the Berry default interpreter.
** skiars@qq.com, https://github.com/Skiars/berry
** See Copyright Notice in the LICENSE file or at
** https://github.com/Skiars/berry/blob/master/LICENSE
********************************************************************/
#include "be_object.h"
#include "be_mem.h"
#include "be_sys.h"
#include "be_gc.h"
#include <string.h>
#define READLINE_STEP 100
static int i_write(bvm *vm)
{
be_getmember(vm, 1, ".p");
if(be_iscomptr(vm, -1) && be_isstring(vm, 2)) {
void *fh = be_tocomptr(vm, -1);
const char *data = be_tostring(vm, 2);
be_fwrite(fh, data, be_strlen(vm, 2));
}
be_return_nil(vm);
}
static size_t readsize(bvm *vm, int argc, void *fh)
{
if (argc >=2 && be_isint(vm, 2)) {
return be_toindex(vm, 2);
}
return be_fsize(fh) - be_ftell(fh);
}
static int i_read(bvm *vm)
{
int argc = be_top(vm);
be_getmember(vm, 1, ".p");
if (be_iscomptr(vm, -1)) {
void *fh = be_tocomptr(vm, -1);
size_t size = readsize(vm, argc, fh);
if (size) {
char *buffer = be_malloc(vm, size);
size = be_fread(fh, buffer, size);
be_pushnstring(vm, buffer, size);
be_free(vm, buffer, size);
} else {
be_pushstring(vm, "");
}
be_return(vm);
}
be_return_nil(vm);
}
static int i_readline(bvm *vm)
{
be_getmember(vm, 1, ".p");
if (be_iscomptr(vm, -1)) {
void *fh = be_tocomptr(vm, -1);
size_t pos = 0, size = READLINE_STEP;
char *buffer = be_malloc(vm, size);
char *res = be_fgets(fh, buffer, (int)size);
while (res) {
pos += strlen(buffer + pos);
if (!pos || buffer[pos - 1] == '\n') {
break;
}
buffer = be_realloc(vm, buffer, size, size + READLINE_STEP);
res = be_fgets(fh, buffer + pos, READLINE_STEP);
size += READLINE_STEP;
}
be_pushnstring(vm, buffer, pos);
be_free(vm, buffer, size);
be_return(vm);
}
be_return_nil(vm);
}
static int i_seek(bvm *vm)
{
be_getmember(vm, 1, ".p");
if (be_iscomptr(vm, -1) && be_isint(vm, 2)) {
void *fh = be_tocomptr(vm, -1);
be_fseek(fh, be_toindex(vm, 2));
}
be_return_nil(vm);
}
static int i_tell(bvm *vm)
{
be_getmember(vm, 1, ".p");
if (be_iscomptr(vm, -1)) {
void *fh = be_tocomptr(vm, -1);
size_t pos = be_ftell(fh);
be_pushint(vm, cast(bint, pos));
be_return(vm);
}
be_return_nil(vm);
}
static int i_size(bvm *vm)
{
be_getmember(vm, 1, ".p");
if (be_iscomptr(vm, -1)) {
void *fh = be_tocomptr(vm, -1);
size_t pos = be_fsize(fh);
be_pushint(vm, cast(bint, pos));
be_return(vm);
}
be_return_nil(vm);
}
static int i_flush(bvm *vm)
{
be_getmember(vm, 1, ".p");
if (be_iscomptr(vm, -1)) {
void *fh = be_tocomptr(vm, -1);
be_fflush(fh);
}
be_return_nil(vm);
}
static int i_close(bvm *vm)
{
be_getmember(vm, 1, ".p");
if (be_iscomptr(vm, -1)) {
void *fh = be_tocomptr(vm, -1);
be_fclose(fh);
be_pushnil(vm);
be_setmember(vm, 1, ".p");
}
be_return_nil(vm);
}
#if !BE_USE_PRECOMPILED_OBJECT
static int m_open(bvm *vm)
#else
int be_nfunc_open(bvm *vm)
#endif
{
int argc = be_top(vm);
const char *fname, *mode;
static const bnfuncinfo members[] = {
{ ".p", NULL },
{ "write", i_write },
{ "read", i_read },
{ "readline", i_readline },
{ "seek", i_seek },
{ "tell", i_tell },
{ "size", i_size },
{ "flush", i_flush },
{ "close", i_close },
{ "deinit", i_close },
{ NULL, NULL }
};
fname = argc >= 1 && be_isstring(vm, 1) ? be_tostring(vm, 1) : NULL;
mode = argc >= 2 && be_isstring(vm, 2) ? be_tostring(vm, 2) : "r";
if (fname) {
void *fh = be_fopen(fname, mode);
if (fh == NULL) {
be_raise(vm, "io_error",
be_pushfstring(vm, "cannot open file '%s'", fname));
}
be_pushclass(vm, "file", members);
be_call(vm, 0);
be_pushcomptr(vm, fh);
be_setmember(vm, -2, ".p");
be_pop(vm, 1);
be_return(vm);
}
be_return_nil(vm);
}
#if !BE_USE_PRECOMPILED_OBJECT
void be_load_filelib(bvm *vm)
{
be_regfunc(vm, "open", m_open);
}
#endif