diff --git a/.gdbinit b/.gdbinit new file mode 100644 index 0000000000..c27be95090 --- /dev/null +++ b/.gdbinit @@ -0,0 +1,34 @@ +# +# This is very much a work in progress to show how we can use macros to make the +# GDB interface a lot more useable. For example the next / step commands only +# work if the stepper doesn't leave the current scope. Beyond that you have a +# single hardware breakpoint which can be used as an hb or a wa. You have to +# remember to delete the previous one, so the br macro does this for you. +# +file app/.output/eagle/debug/image/eagle.app.v6.out +#set remotedebug 1 +set remotelogfile gdb_rsp_logfile.txt +set serial baud 115200 +set remote hardware-breakpoint-limit 1 +set remote hardware-watchpoint-limit 1 +#set debug xtensa 4 +target remote /dev/ttyUSB0 + +set confirm off +set print null-stop +define br + d + hb $arg0 +end + +define upto + d + hb $arg0 + c +end + +set pagination off +set history filename ~/.gdb_history +set history save on +set history size 1000 + diff --git a/.gdbinitlua b/.gdbinitlua new file mode 100644 index 0000000000..b832fab5f1 --- /dev/null +++ b/.gdbinitlua @@ -0,0 +1,182 @@ + +set pagination off +set print null-stop + +define prTS + set $o = &(((TString *)($arg0))->tsv) + printf "Common header: next = %p, marked = 0x%01x\n", $o->next, $o->marked + printf "String: hash = 0x%08x, len = %u : %s\n", $o->hash, $o->len, (char *)(&$o[1]) +end + +define prTnodes + set $o = (Table *)($arg0) + set $n = 1<<($o->lsizenode) + set $i = 0 + while $i < $n + set $nd = ($o->node) + $i + if $nd->i_key.nk.tt && $nd->i_val.tt + if $nd->i_key.nk.tt == 6 + printf "%4u: %s %2i\n", $i, $nd->i_key.nk.tt , $nd->i_val.tt + else + printf "%4u: %2i %2i\n", $i, $nd->i_key.nk.tt , $nd->i_val.tt + end + end + set $i = $i +1 + end +end +define prTV + if $arg0 + set $type = ($arg0).tt + set $val = ($arg0).value + + if $type == 0 + # NIL + printf "Nil\n" + end + if $type == 1 + # Boolean + printf "Boolean: %u\n", $val.n + end + if $type == 2 + # ROTable + printf "ROTable: %p\n", $val.p + end + if $type == 3 + # Light Function + printf "Light Func: %p\n", $val.p + end + if $type == 4 + # Light User Data + printf "Light Udata: %p\n", $val.p + end + if $type == 5 + # Number + printf "Number: %u\n", $val.n + end + if $type == 6 + prTS $arg0 + end + if $type == 7 + # Table + set $o = &($val->gc.h) + printf "Common header: next = %p, marked = 0x%01x\n", $o->next, $o->marked + printf "Nodes: %4i %p\n", 2<<($o->lsizenode), $o->node + printf "Arry: %4i %p\n", $o->sizearray, $o->array + end + if $type == 8 + # Function + set $o = &($val->gc.cl.c) + printf "Common header: next = %p, marked = 0x%01x\n", $o->next, $o->marked + if $o->isC == 0 + set $o = &($val->gc.cl.l) + printf "LClosure: nupvalues = %u, gclist = %p, env = %p, p = %p\n", \ + $o->nupvalues, $o->gclist, $o->env, $o->p + else + printf "CClosure: nupvalues = %u, gclist = %p, env = %p, f = %p\np", \ + $o->nupvalues, $o->gclist, $o->env, $o->f + end + end + if $type == 9 + # UserData + end + if $type == 10 + # Thread + end + end +end + +define prT + print *(Table*)($arg0) +end + +define prL + if L > 0 + printf " stack: %u\n", L->top-L->base + printf " hooking: %u, %u, %u, %u, %p\n", L->hookmask, L->allowhook, L->basehookcount, L->hookcount, L->hook + end +end + +define dumpstrt + set $st = $arg0 + set $i = 0 + while $i< $st->size + set $o = &(((TString *)($st->hash[$i]))->tsv) + while $o + if $o->next + printf "Slot: %5i %p %p %08x %02x %4u", \ + $i, $o, $o->next, $o->hash, $o->marked, $o->len + else + printf "Slot: %5i %p %08x %02x %4u", \ + $i, $o, $o->hash, $o->marked, $o->len + end + if $o->marked & 0x80 + printf "* %s\n", *(char **)($o+1) + else + printf " %s\n", (char *)($o+1) + end + set $o = &(((TString *)($o->next))->tsv) + end + set $i = $i + 1 + end +end + +define dumpRAMstrt + dumpstrt &(L->l_G->strt) +end + +define dumpROstrt + dumpstrt &(L->l_G->ROstrt) +end + +define graylist + set $n = $arg0 + while $n + printf "%p %2u %02X\n",$n, $n->gch.tt, $n->gch.marked + set $n=$n->gch.next + end +end + +define prPC + printf "Excuting instruction %i: %08x\n", (pc - cl->p->code)+1-1, i +end + + +define where + set $f=cl->p + printf "<%s:%u,%u>, opcode %u\n",\ + (char *)$f->source+17, $f->linedefined, $f->lastlinedefined, pc - $f->code +end + +define callinfo + printf "%p: ", L->ci + print *L->ci +end + +define luastack + set $i = 0 + set $ci = L->base_ci + set $s = L->stack + set $last = L->stack_last - L->stack + printf "stack = %p, last: %i, size: %i, " , $s, $last, L->stacksize + if $last+6==L->stacksize + printf "(OK)\n" + else + printf "(MISMATCH)\n" + end + printf " Ndx top base func\n" + while $ci <= L->ci + printf "%3u %6i %6i %6i\n", $i++, $ci->top-$s, $ci->base-$s, ($ci++)->func-$s + end +end + +define stacklen + printf "%i top: %p, base: %p\n", \ + L->ci->top - L->base, L->ci->top, L->base +end + +define stackcheck + set $ci = L->ci + printf "Used: %i, Headroom: %i, Total: %i\n", \ + L->top-$ci->base-1, $ci->top-L->top+1, $ci->top-$ci->base +end + diff --git a/.gitignore b/.gitignore index 337f056c4f..8ad76014d7 100644 --- a/.gitignore +++ b/.gitignore @@ -1,6 +1,8 @@ sdk/ cache/ .ccache/ +local/ +luac.cross user_config.h server-ca.crt diff --git a/.travis.yml b/.travis.yml index bea38bfc4d..9cd033ea6b 100644 --- a/.travis.yml +++ b/.travis.yml @@ -5,21 +5,13 @@ addons: packages: - python-serial - srecord - - lua5.1 cache: - - pip - directories: - cache -before_install: -- pip install --user hererocks esptool -- hererocks env --lua 5.1 -rlatest -- source env/bin/activate -- luarocks install luafilesystem install: - tar -Jxvf tools/esp-open-sdk.tar.xz - export PATH=$PATH:$PWD/esp-open-sdk/xtensa-lx106-elf/bin script: -- lua tools/cross-lua.lua || exit 1 - export BUILD_DATE=$(date +%Y%m%d) - make EXTRA_CCFLAGS="-DBUILD_DATE='\"'$BUILD_DATE'\"'" all - cd bin/ diff --git a/Makefile b/Makefile index 07a2ff9dee..e02d41f12a 100644 --- a/Makefile +++ b/Makefile @@ -28,7 +28,7 @@ ifdef DEBUG CCFLAGS += -ggdb -O0 LDFLAGS += -ggdb else - CCFLAGS += -Os + CCFLAGS += -O2 endif ############################################################# diff --git a/app/Makefile b/app/Makefile index 51c50ccd29..711cf7a2bd 100644 --- a/app/Makefile +++ b/app/Makefile @@ -20,89 +20,64 @@ FLAVOR = debug ifndef PDIR # { GEN_IMAGES= eagle.app.v6.out GEN_BINS= eagle.app.v6.bin -SPECIAL_MKTARGETS=$(APP_MKTARGETS) +OPT_MKTARGETS := coap crypto dht http mqtt pcm sjson sqlite3 tsl2561 websocket +OPT_MKLIBTARGETS := u8g ucg +SEL_MKTARGETS := $(shell $(CC) -E -dM include/user_modules.h | sed -n '/^\#define LUA_USE_MODULES_/{s/.\{24\}\(.*\)/\L\1/; p}') +OPT_SEL_MKLIBTARGETS := $(foreach tgt,$(OPT_MKLIBTARGETS),$(findstring $(tgt), $(SEL_MKTARGETS))) +OPT_SEL_MKTARGETS := $(foreach tgt,$(OPT_MKTARGETS),$(findstring $(tgt), $(SEL_MKTARGETS))) \ + $(foreach tgt,$(OPT_SEL_MKLIBTARGETS),$(tgt)lib) +OPT_SEL_COMPONENTS := $(foreach tgt,$(OPT_SEL_MKTARGETS),$(tgt)/lib$(tgt).a) +SPECIAL_MKTARGETS :=$(APP_MKTARGETS) + SUBDIRS= \ user \ driver \ - pcm \ mbedtls \ platform \ libc \ lua \ lwip \ - coap \ - mqtt \ task \ - u8glib \ - ucglib \ smart \ modules \ spiffs \ - crypto \ - dhtlib \ - tsl2561 \ net \ - http \ fatfs \ esp-gdbstub \ - websocket \ pm \ - sjson \ - sqlite3 \ - + $(OPT_SEL_MKTARGETS) endif # } PDIR APPDIR = . LDDIR = ../ld -CCFLAGS += -Os - TARGET_LDFLAGS = \ -nostdlib \ -Wl,-EL \ --longcalls \ --text-section-literals -ifeq ($(FLAVOR),debug) - TARGET_LDFLAGS += -g -Os -endif - -ifeq ($(FLAVOR),release) - TARGET_LDFLAGS += -Os -endif - LD_FILE = $(LDDIR)/nodemcu.ld COMPONENTS_eagle.app.v6 = \ user/libuser.a \ driver/libdriver.a \ - pcm/pcm.a \ platform/libplatform.a \ task/libtask.a \ libc/liblibc.a \ lua/liblua.a \ lwip/liblwip.a \ - coap/coap.a \ - mqtt/mqtt.a \ - u8glib/u8glib.a \ - ucglib/ucglib.a \ smart/smart.a \ spiffs/spiffs.a \ fatfs/libfatfs.a \ - crypto/libcrypto.a \ - dhtlib/libdhtlib.a \ - tsl2561/tsl2561lib.a \ - http/libhttp.a \ pm/libpm.a \ - websocket/libwebsocket.a \ esp-gdbstub/libgdbstub.a \ net/libnodemcu_net.a \ mbedtls/libmbedtls.a \ modules/libmodules.a \ - sjson/libsjson.a \ - sqlite3/libsqlite3.a \ - + $(OPT_SEL_COMPONENTS) + # Inspect the modules library and work out which modules need to be linked. # For each enabled module, a symbol name of the form XYZ_module_selected is @@ -162,7 +137,7 @@ CONFIGURATION_DEFINES = -D__ets__ \ -DLWIP_OPEN_SRC \ -DPBUF_RSV_FOR_WLAN \ -DEBUF_LWIP \ - -DUSE_OPTIMIZE_PRINTF \ + -DUSE_OPTIMIZE_PRINTF \ -DMBEDTLS_USER_CONFIG_FILE=\"user_mbedtls.h\" \ DEFINES += \ @@ -191,6 +166,5 @@ INCLUDES += -I ./ PDIR := ../$(PDIR) sinclude $(PDIR)Makefile - .PHONY: FORCE FORCE: diff --git a/app/coap/Makefile b/app/coap/Makefile index 3947069c06..a0c55e0d54 100644 --- a/app/coap/Makefile +++ b/app/coap/Makefile @@ -12,7 +12,7 @@ # a generated lib/image xxx.a () # ifndef PDIR -GEN_LIBS = coap.a +GEN_LIBS = libcoap.a endif ############################################################# diff --git a/app/coap/endpoints.c b/app/coap/endpoints.c index 81e124ccc4..013cef7f70 100644 --- a/app/coap/endpoints.c +++ b/app/coap/endpoints.c @@ -162,7 +162,8 @@ static int handle_post_function(const coap_endpoint_t *ep, coap_rw_buffer_t *scr return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_NOT_FOUND, COAP_CONTENTTYPE_NONE); } -extern lua_Load gLoad; +extern int lua_put_line(const char *s, size_t l); + static const coap_endpoint_path_t path_command = {2, {"v1", "c"}}; static int handle_post_command(const coap_endpoint_t *ep, coap_rw_buffer_t *scratch, const coap_packet_t *inpkt, coap_packet_t *outpkt, uint8_t id_hi, uint8_t id_lo) { @@ -170,16 +171,9 @@ static int handle_post_command(const coap_endpoint_t *ep, coap_rw_buffer_t *scra return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_BAD_REQUEST, COAP_CONTENTTYPE_TEXT_PLAIN); if (inpkt->payload.len > 0) { - lua_Load *load = &gLoad; - if(load->line_position == 0){ - coap_buffer_to_string(load->line, load->len,&inpkt->payload); - load->line_position = c_strlen(load->line)+1; - // load->line[load->line_position-1] = '\n'; - // load->line[load->line_position] = 0; - // load->line_position++; - load->done = 1; - NODE_DBG("Get command:\n"); - NODE_DBG(load->line); // buggy here + char line[LUA_MAXINPUT]; + if (!coap_buffer_to_string(line, LUA_MAXINPUT, &inpkt->payload) && + lua_put_line(line, c_strlen(line))) { NODE_DBG("\nResult(if any):\n"); system_os_post (LUA_TASK_PRIO, LUA_PROCESS_LINE_SIG, 0); } diff --git a/app/dhtlib/Makefile b/app/dht/Makefile similarity index 98% rename from app/dhtlib/Makefile rename to app/dht/Makefile index 9ef52484ed..9c3024450e 100644 --- a/app/dhtlib/Makefile +++ b/app/dht/Makefile @@ -12,7 +12,7 @@ # a generated lib/image xxx.a () # ifndef PDIR -GEN_LIBS = libdhtlib.a +GEN_LIBS = libdht.a endif diff --git a/app/dhtlib/dht.c b/app/dht/dht.c similarity index 100% rename from app/dhtlib/dht.c rename to app/dht/dht.c diff --git a/app/dhtlib/dht.h b/app/dht/dht.h similarity index 100% rename from app/dhtlib/dht.h rename to app/dht/dht.h diff --git a/app/driver/uart.c b/app/driver/uart.c old mode 100755 new mode 100644 diff --git a/app/esp-gdbstub/Makefile b/app/esp-gdbstub/Makefile index 4e6e8b44b0..871b00ffc3 100644 --- a/app/esp-gdbstub/Makefile +++ b/app/esp-gdbstub/Makefile @@ -23,7 +23,7 @@ endif # makefile at its root level - these are then overridden # for a subtree within the makefile rooted therein # -#DEFINES += +#DEFINES += -DGDBSTUB_REDIRECT_CONSOLE_OUTPUT ############################################################# # Recursion Magic - Don't touch this!! diff --git a/app/include/arch/cc.h b/app/include/arch/cc.h old mode 100755 new mode 100644 diff --git a/app/include/driver/uart.h b/app/include/driver/uart.h old mode 100755 new mode 100644 diff --git a/app/include/driver/uart_register.h b/app/include/driver/uart_register.h old mode 100755 new mode 100644 diff --git a/app/include/lwipopts.h b/app/include/lwipopts.h old mode 100755 new mode 100644 diff --git a/app/include/module.h b/app/include/module.h index 5a399f47cc..b727fa9b97 100644 --- a/app/include/module.h +++ b/app/include/module.h @@ -38,8 +38,11 @@ #define MODULE_PASTE_(x,y) x##y #define MODULE_EXPAND_PASTE_(x,y) MODULE_PASTE_(x,y) -#define LOCK_IN_SECTION(s) __attribute__((used,unused,section(s))) - +#ifdef LUA_CROSS_COMPILER +#define LOCK_IN_SECTION(s) __attribute__((used,unused,section(".rodata1." #s))) +#else +#define LOCK_IN_SECTION(s) __attribute__((used,unused,section(".lua_" #s))) +#endif /* For the ROM table, we name the variable according to ( | denotes concat): * cfgname | _module_selected | LUA_USE_MODULES_##cfgname * where the LUA_USE_MODULES_XYZ macro is first expanded to yield either @@ -51,20 +54,20 @@ * to be linked in. */ #define NODEMCU_MODULE(cfgname, luaname, map, initfunc) \ - const LOCK_IN_SECTION(".lua_libs") \ + const LOCK_IN_SECTION(libs) \ luaL_Reg MODULE_PASTE_(lua_lib_,cfgname) = { luaname, initfunc }; \ - const LOCK_IN_SECTION(".lua_rotable") \ + const LOCK_IN_SECTION(rotable) \ luaR_table MODULE_EXPAND_PASTE_(cfgname,MODULE_EXPAND_PASTE_(_module_selected,MODULE_PASTE_(LUA_USE_MODULES_,cfgname))) \ = { luaname, map } /* System module registration support, not using LUA_USE_MODULES_XYZ. */ #define BUILTIN_LIB_INIT(name, luaname, initfunc) \ - const LOCK_IN_SECTION(".lua_libs") \ + const LOCK_IN_SECTION(libs) \ luaL_Reg MODULE_PASTE_(lua_lib_,name) = { luaname, initfunc } #define BUILTIN_LIB(name, luaname, map) \ - const LOCK_IN_SECTION(".lua_rotable") \ + const LOCK_IN_SECTION(rotable) \ luaR_table MODULE_PASTE_(lua_rotable_,name) = { luaname, map } #if !defined(LUA_CROSS_COMPILER) && !(MIN_OPT_LEVEL==2 && LUA_OPTIMIZE_MEMORY==2) diff --git a/app/include/user_config.h b/app/include/user_config.h index e0abceca38..27568233fa 100644 --- a/app/include/user_config.h +++ b/app/include/user_config.h @@ -1,125 +1,250 @@ #ifndef __USER_CONFIG_H__ #define __USER_CONFIG_H__ -// #define FLASH_512K -// #define FLASH_1M -// #define FLASH_2M -// #define FLASH_4M -// #define FLASH_8M -// #define FLASH_16M +// The firmware supports a range of Flash sizes, though 4 Mbyte seems to be +// the most common currently. NodeMCU builds include a discovery function +// which is enabled by FLASH_AUTOSIZE, but you can override this by commenting +// this out and enabling the explicitly size, e.g. FLASH_4M. Valid sizes are +// FLASH_512K, FLASH_1M, FLASH_2M, FLASH_4M, FLASH_8M, FLASH_16M. + #define FLASH_AUTOSIZE +//#define FLASH_4M + + +// The firmware now selects a baudrate of 115,200 by default, but the driver +// also includes automatic baud rate detection at start-up. If you want to change +// the default rate then vaild rates are 300, 600, 1200, 2400, 4800, 9600, 19200, +// 31250, 38400, 57600, 74880, 115200, 230400, 256000, 460800 [, 921600, 1843200, +// 368640]. Note that the last 3 rates are not recommended as these might be +// unreliable, but 460800 seems to work well for most USB-serial devices. + +#define BIT_RATE_DEFAULT BIT_RATE_115200 +//#define BIT_RATE_AUTOBAUD + + +// Three separate build variants are now supported. The main difference is in the +// processing of numeric data types. If LUA_NUMBER_INTEGRAL is defined, then +// all numeric calculations are done in integer, with divide being an integer +// operations, and decimal fraction constants are illegal. Otherwise all +// numeric operations use floating point, though they are exact for integer +// expressions < 2^53. + +// The main advantage of INTEGRAL builds is that the basic internal storage unit, +// the TValue, is 8 bytes long. We have now reduced the size of FP TValues to +// 12 bytes rather than the previous 16 as this gives a material RAM saving with +// no performance loss. However, you can define LUA_DWORD_ALIGNED_TVALUES and +// this will force 16 byte TValues on FP builds. + +//#define LUA_NUMBER_INTEGRAL +//#define LUA_DWORD_ALIGNED_TVALUES + + +// The Lua Flash Store (LFS) allows you to store Lua code in Flash memory and +// the Lua VMS will execute this code directly from flash without needing any +// RAM overhead. If you want to enable LFS then set the following define to +// the size of the store that you need. This can be any multiple of 4kB up to +// a maximum 256Kb. + +//#define LUA_FLASH_STORE 0x10000 + + +// By default Lua executes the file init.lua at start up. The following +// define allows you to replace this with an alternative startup. Warning: +// you must protect this execution otherwise you will enter a panic loop; +// the simplest way is to wrap the action in a function invoked by a pcall. +// The example provided executes the LFS module "_init" at startup or fails +// through to the interactive prompt. + +//#define LUA_INIT_STRING "pcall(function() node.flashindex'_init'() end)" + + +// NodeMCU supports two file systems: SPIFFS and FATFS, the first is available +// on all ESP8266 modules. The latter requires extra H/W so is less common. +// If you use SPIFFS then there are a number of options which impact the +// RAM overhead and performance of the file system. + +// If you use the spiffsimg tool to create your own FS images on your dev PC +// then we recommend that you fix the location and size of the FS, allowing +// some headroom for rebuilding flash images and LFS. As an alternative to +// fixing the size of the FS, you can force the SPIFFS file system to end on +// the next 1Mb boundary. This is useful for certain OTA scenarios. In +// general, limiting the size of the FS only to what your application needs +// gives the fastest start-up and imaging times. + +#define BUILD_SPIFFS +//#define SPIFFS_FIXED_LOCATION 0x100000 +//#define SPIFFS_MAX_FILESYSTEM_SIZE 0x20000 +//#define SPIFFS_SIZE_1M_BOUNDARY +#define SPIFFS_CACHE 1 // Enable if you use you SPIFFS in R/W mode +#define SPIFFS_MAX_OPEN_FILES 4 // maximum number of open files for SPIFFS +#define FS_OBJ_NAME_LEN 31 // maximum length of a filename + +//#define BUILD_FATFS + + +// The HTTPS stack requires client SSL to be enabled. The SSL buffer size is +// used only for espconn-layer secure connections, and is ignored otherwise. +// Some HTTPS applications require a larger buffer size to work. See +// https://github.com/nodemcu/nodemcu-firmware/issues/1457 for details. +// The SHA2 and MD2 libraries are also optionally used by the crypto functions. +// The SHA1 and MD5 function are implemented in the ROM BIOS. The MD2 and SHA2 +// are by firmware code, and can be enabled if you need this functionality. + +//#define CLIENT_SSL_ENABLE +//#define MD2_ENABLE +#define SHA2_ENABLE +#define SSL_BUFFER_SIZE 5120 + + +// GPIO_INTERRUPT_ENABLE needs to be defined if your application uses the +// gpio.trig() or related GPIO interrupt service routine code. Likewise the +// GPIO interrupt hook is requited for a few modules such as rotary. If you +// don't require this functionality, then commenting out these options out +// will remove any associated runtime overhead. + +#define GPIO_INTERRUPT_ENABLE +#define GPIO_INTERRUPT_HOOK_ENABLE + + +// If your application uses the light sleep functions and you wish the +// firmware to manage timer rescheduling over sleeps (the CPU clock is +// suspended so timers get out of sync) then enable the following options + +//#define ENABLE_TIMER_SUSPEND +//#define PMSLEEP_ENABLE + + +// The WiFi module optionally offers an enhanced level of WiFi connection +// management, using internal timer callbacks. Whilst many Lua developers +// prefer to implement equivalent features in Lua, others will prefer the +// Wifi module to do this for them. Uncomment the following to enable +// this functionality. See the relevant WiFi module documentation for +// further details, as the scope of these changes is not obvious. + +// Enable the wifi.startsmart() and wifi.stopsmart() +//#define WIFI_SMART_ENABLE + +// Enable wifi.sta.config() event callbacks +#define WIFI_SDK_EVENT_MONITOR_ENABLE + +// Enable creation on the wifi.eventmon.reason table +#define WIFI_EVENT_MONITOR_DISCONNECT_REASON_LIST_ENABLE + +// Enable use of the WiFi.monitor sub-module +//#define LUA_USE_MODULES_WIFI_MONITOR + + +// Whilst the DNS client details can be configured through the WiFi API, +// the defaults can be exposed temporarily during start-up. The following +// WIFI_STA options allow you to configure this in the firmware. If the +// WIFI_STA_HOSTNAME is not defined then the hostname will default to +// to the last 3 octets (6 hexadecimal digits) of MAC address with the +// prefix "NODE-". If it is defined then the hostname must only contain +// alphanumeric characters. If you are imaging multiple modules with this +// firmware then you must also define WIFI_STA_HOSTNAME_APPEND_MAC to +// append the last 3 octets of the MAC address. Note that the total +// Hostname MUST be 32 chars or less. + +//#define WIFI_STA_HOSTNAME "NodeMCU" +//#define WIFI_STA_HOSTNAME_APPEND_MAC + + +// If you use the enduser_setup module, then you can also set the default +// SSID when this module is running in AP mode. + +#define ENDUSER_SETUP_AP_SSID "SetupGadget" + + +// The following sections are only relevent for those developers who are +// developing modules or core Lua changes and configure how extra diagnostics +// are enabled in the firmware. These should only be configured if you are +// building your own custom firmware and have full access to the firmware +// source code. + +// Enabling DEVELOPMENT_TOOLS adds the asserts in LUA and also some useful +// extras to the node module. These are silent in normal operation and so can +// be enabled without any harm (except for the code size increase and slight +// slowdown). If you want to use the remote GDB to handle breaks and failed +// assertions then enable the DEVELOPMENT_USE GDB option. A supplimentary +// define DEVELOPMENT_BREAK_ON_STARTUP_PIN allows you to define a GPIO pin, +// which if pulled low at start-up will immediately initiate a GDB session. + +// The DEVELOP_VERSION option enables lots of debug output, and is normally +// only used by hardcore developers. + +// These options can be enabled globally here or you can alternatively use +// the DEFINES variable in the relevant Makefile to set these on a per +// directory basis. If you do this then you can also set the corresponding +// compile options (-O0 -ggdb) on a per directory as well. -// This adds the asserts in LUA. It also adds some useful extras to the -// node module. This is all silent in normal operation and so can be enabled -// without any harm (except for the code size increase and slight slowdown) //#define DEVELOPMENT_TOOLS +//#define DEVELOPMENT_USE_GDB +//#define DEVELOPMENT_BREAK_ON_STARTUP_PIN 1 +//#define DEVELOP_VERSION + + +// *** Heareafter, there be demons *** + +// The remaining options are advanced configuration options and you should only +// change this if you have tracked the implications through the Firmware sources +// and understand the these. + +#define LUA_TASK_PRIO USER_TASK_PRIO_0 +#define LUA_PROCESS_LINE_SIG 2 +#define LUA_OPTIMIZE_DEBUG 2 +#define READLINE_INTERVAL 80 +#define STRBUF_DEFAULT_INCREMENT 3 +#define LUA_USE_BUILTIN_DEBUG_MINIMAL // for debug.getregistry() and debug.traceback() #ifdef DEVELOPMENT_TOOLS +#if defined(LUA_CROSS_COMPILER) || !defined(DEVELOPMENT_USE_GDB) extern void luaL_assertfail(const char *file, int line, const char *message); #define lua_assert(x) ((x) ? (void) 0 : luaL_assertfail(__FILE__, __LINE__, #x)) +#else +extern void luaL_dbgbreak(void); +#define lua_assert(x) ((x) ? (void) 0 : luaL_dbgbreak()) +#endif +#endif + +#if !defined(LUA_NUMBER_INTEGRAL) && defined (LUA_DWORD_ALIGNED_TVALUES) + #define LUA_PACK_TVALUES +#else + #undef LUA_PACK_TVALUES #endif -// This enables lots of debug output and changes the serial bit rate. This -// is normally only used by hardcore developers -// #define DEVELOP_VERSION #ifdef DEVELOP_VERSION #define NODE_DEBUG #define COAP_DEBUG #endif /* DEVELOP_VERSION */ -#define BIT_RATE_DEFAULT BIT_RATE_115200 - -// This enables automatic baud rate detection at startup -#define BIT_RATE_AUTOBAUD - -#define NODE_ERROR - #ifdef NODE_DEBUG #define NODE_DBG dbg_printf #else #define NODE_DBG #endif /* NODE_DEBUG */ +#define NODE_ERROR #ifdef NODE_ERROR #define NODE_ERR dbg_printf #else #define NODE_ERR #endif /* NODE_ERROR */ -#define GPIO_INTERRUPT_ENABLE -#define GPIO_INTERRUPT_HOOK_ENABLE // #define GPIO_SAFE_NO_INTR_ENABLE - #define ICACHE_STORE_TYPEDEF_ATTR __attribute__((aligned(4),packed)) #define ICACHE_STORE_ATTR __attribute__((aligned(4))) -#define ICACHE_RAM_STRING(x) ICACHE_RAM_STRING2(x) -#define ICACHE_RAM_STRING2(x) #x -#define ICACHE_RAM_ATTR __attribute__((section(".iram0.text." __FILE__ "." ICACHE_RAM_STRING(__LINE__)))) +#define ICACHE_STRING(x) ICACHE_STRING2(x) +#define ICACHE_STRING2(x) #x +#define ICACHE_RAM_ATTR \ + __attribute__((section(".iram0.text." __FILE__ "." ICACHE_STRING(__LINE__)))) +#define ICACHE_FLASH_RESERVED_ATTR \ + __attribute__((section(".irom.reserved." __FILE__ "." ICACHE_STRING(__LINE__)),\ + used,unused,aligned(INTERNAL_FLASH_SECTOR_SIZE))) + #ifdef GPIO_SAFE_NO_INTR_ENABLE #define NO_INTR_CODE ICACHE_RAM_ATTR __attribute__ ((noinline)) #else #define NO_INTR_CODE inline #endif -// SSL buffer size used only for espconn-layer secure connections. -// See https://github.com/nodemcu/nodemcu-firmware/issues/1457 for conversation details. -#define SSL_BUFFER_SIZE 5120 - -//#define CLIENT_SSL_ENABLE -//#define MD2_ENABLE -#define SHA2_ENABLE - -#define BUILD_SPIFFS -#define SPIFFS_CACHE 1 - -//#define BUILD_FATFS - -// maximum length of a filename -#define FS_OBJ_NAME_LEN 31 - -// maximum number of open files for SPIFFS -#define SPIFFS_MAX_OPEN_FILES 4 - -// Uncomment this next line for fastest startup -// It reduces the format time dramatically -// #define SPIFFS_MAX_FILESYSTEM_SIZE 32768 -// -// You can force the spiffs file system to be at a fixed location -// #define SPIFFS_FIXED_LOCATION 0x100000 -// -// You can force the SPIFFS file system to end on the next !M boundary -// (minus the 16k parameter space). THis is useful for certain OTA scenarios -// #define SPIFFS_SIZE_1M_BOUNDARY - -// #define LUA_NUMBER_INTEGRAL - -#define READLINE_INTERVAL 80 -#define LUA_TASK_PRIO USER_TASK_PRIO_0 -#define LUA_PROCESS_LINE_SIG 2 -#define LUA_OPTIMIZE_DEBUG 2 - -#define ENDUSER_SETUP_AP_SSID "SetupGadget" - -/* - * A valid hostname only contains alphanumeric and hyphen(-) characters, with no hyphens at first or last char - * if WIFI_STA_HOSTNAME not defined: hostname will default to NODE-xxxxxx (xxxxxx being last 3 octets of MAC address) - * if WIFI_STA_HOSTNAME defined: hostname must only contain alphanumeric characters - * if WIFI_STA_HOSTNAME_APPEND_MAC not defined: Hostname MUST be 32 chars or less - * if WIFI_STA_HOSTNAME_APPEND_MAC defined: Hostname MUST be 26 chars or less, since last 3 octets of MAC address will be appended - * if defined hostname is invalid: hostname will default to NODE-xxxxxx (xxxxxx being last 3 octets of MAC address) -*/ -//#define WIFI_STA_HOSTNAME "NodeMCU" -//#define WIFI_STA_HOSTNAME_APPEND_MAC - -//#define WIFI_SMART_ENABLE - -#define WIFI_SDK_EVENT_MONITOR_ENABLE -#define WIFI_EVENT_MONITOR_DISCONNECT_REASON_LIST_ENABLE - -//#define PMSLEEP_ENABLE // Enable wifi.suspend() and node.sleep() (NOTE: node.sleep() is dependent on TIMER_SUSPEND_ENABLE) -//#define TIMER_SUSPEND_ENABLE //Required by node.sleep() - - -#define STRBUF_DEFAULT_INCREMENT 32 - #endif /* __USER_CONFIG_H__ */ diff --git a/app/include/user_modules.h b/app/include/user_modules.h index 0d4b9e918e..a3e6fab8e8 100644 --- a/app/include/user_modules.h +++ b/app/include/user_modules.h @@ -1,16 +1,6 @@ #ifndef __USER_MODULES_H__ #define __USER_MODULES_H__ -#define LUA_USE_BUILTIN_STRING // for string.xxx() -#define LUA_USE_BUILTIN_TABLE // for table.xxx() -#define LUA_USE_BUILTIN_COROUTINE // for coroutine.xxx() -#define LUA_USE_BUILTIN_MATH // for math.xxx(), partially work -// #define LUA_USE_BUILTIN_IO // for io.xxx(), partially work - -// #define LUA_USE_BUILTIN_OS // for os.xxx(), not work -// #define LUA_USE_BUILTIN_DEBUG -#define LUA_USE_BUILTIN_DEBUG_MINIMAL // for debug.getregistry() and debug.traceback() - #ifndef LUA_CROSS_COMPILER // The default configuration is designed to run on all ESP modules including the 512 KB modules like ESP-01 and only @@ -69,9 +59,9 @@ //#define LUA_USE_MODULES_SQLITE3 //#define LUA_USE_MODULES_STRUCT //#define LUA_USE_MODULES_SWITEC -// #define LUA_USE_MODULES_TCS34725 +//#define LUA_USE_MODULES_TCS34725 //#define LUA_USE_MODULES_TM1829 -#define LUA_USE_MODULES_TLS +//#define LUA_USE_MODULES_TLS #define LUA_USE_MODULES_TMR //#define LUA_USE_MODULES_TSL2561 //#define LUA_USE_MODULES_U8G diff --git a/app/libc/c_stdlib.c b/app/libc/c_stdlib.c index 50c685c222..4f39cc8631 100644 --- a/app/libc/c_stdlib.c +++ b/app/libc/c_stdlib.c @@ -15,55 +15,6 @@ #include "c_stdlib.h" #include "c_types.h" #include "c_string.h" - -// const char *lua_init_value = "print(\"Hello world\")"; -const char *lua_init_value = "@init.lua"; - -// int c_abs(int x){ -// return x>0?x:0-x; -// } -// void c_exit(int e){ -// } -const char *c_getenv(const char *__string) -{ - if (c_strcmp(__string, "LUA_INIT") == 0) - { - return lua_init_value; - } - return NULL; -} -// make sure there is enough memory before real malloc, otherwise malloc will panic and reset -// void *c_malloc(size_t __size){ -// if(__size>system_get_free_heap_size()){ -// NODE_ERR("malloc: not enough memory\n"); -// return NULL; -// } -// return (void *)os_malloc(__size); -// } - -// void *c_zalloc(size_t __size){ -// if(__size>system_get_free_heap_size()){ -// NODE_ERR("zalloc: not enough memory\n"); -// return NULL; -// } -// return (void *)os_zalloc(__size); -// } - -// void c_free(void *p){ -// // NODE_ERR("free1: %d\n", system_get_free_heap_size()); -// os_free(p); -// // NODE_ERR("-free1: %d\n", system_get_free_heap_size()); -// }c_stdlib.s - - - -// int c_rand(void){ -// } -// void c_srand(unsigned int __seed){ -// } - -// int c_atoi(const char *__nptr){ -// } #include <_ansi.h> //#include //#include "mprec.h" diff --git a/app/libc/c_stdlib.h b/app/libc/c_stdlib.h index bb2836d53a..171c7fbb2e 100644 --- a/app/libc/c_stdlib.h +++ b/app/libc/c_stdlib.h @@ -46,8 +46,7 @@ // void c_exit(int); -// c_getenv() get env "LUA_INIT" string for lua initialization. -const char *c_getenv(const char *__string); +//const char *c_getenv(const char *__string); // void *c_malloc(size_t __size); // void *c_zalloc(size_t __size); diff --git a/app/lua/Makefile b/app/lua/Makefile index a2ff3635e0..ff7291bde6 100644 --- a/app/lua/Makefile +++ b/app/lua/Makefile @@ -12,6 +12,7 @@ # a generated lib/image xxx.a () # ifndef PDIR +SUBDIRS = luac_cross GEN_LIBS = liblua.a endif @@ -24,7 +25,8 @@ STD_CFLAGS=-std=gnu11 -Wimplicit # makefile at its root level - these are then overridden # for a subtree within the makefile rooted therein # -#DEFINES += +#DEFINES += -DDEVELOPMENT_TOOLS -DDEVELOPMENT_USE_GDB -DDEVELOPMENT_BREAK_ON_STARTUP_PIN=1 +#EXTRA_CCFLAGS += -ggdb -O0 ############################################################# # Recursion Magic - Don't touch this!! diff --git a/app/lua/lapi.c b/app/lua/lapi.c index 3f783cbaea..00c3746aa0 100644 --- a/app/lua/lapi.c +++ b/app/lua/lapi.c @@ -414,7 +414,7 @@ LUA_API const void *lua_topointer (lua_State *L, int idx) { case LUA_TROTABLE: return rvalue(o); case LUA_TLIGHTFUNCTION: - return pvalue(o); + return fvalue(o); default: return NULL; } } @@ -459,15 +459,6 @@ LUA_API void lua_pushlstring (lua_State *L, const char *s, size_t len) { } -LUA_API void lua_pushrolstring (lua_State *L, const char *s, size_t len) { - lua_lock(L); - luaC_checkGC(L); - setsvalue2s(L, L->top, luaS_newrolstr(L, s, len)); - api_incr_top(L); - lua_unlock(L); -} - - LUA_API void lua_pushstring (lua_State *L, const char *s) { if (s == NULL) lua_pushnil(L); diff --git a/app/lua/lauxlib.c b/app/lua/lauxlib.c index ce82ee2951..c6d6f5ba07 100644 --- a/app/lua/lauxlib.c +++ b/app/lua/lauxlib.c @@ -45,6 +45,149 @@ #define LUA_USECCLOSURES 0 #define LUA_USELIGHTFUNCTIONS 1 +//#define DEBUG_ALLOCATOR +#ifdef DEBUG_ALLOCATOR +#ifdef LUA_CROSS_COMPILER +static void break_hook(void) {} +#define ASSERT(s) if (!(s)) {break_hook();} +#else +#define ASSERT(s) if (!(s)) {asm ("break 0,0" ::);} +#endif + +/* +** {====================================================================== +** Diagnosticd version for realloc. This is enabled only if the +** DEBUG_ALLOCATOR is defined. It is a cutdown version of the allocator +** used in the Lua Test Suite -- a compromise between the ability catch +** most alloc/free errors and overruns and working within the RAM limits +** of the ESP8266 architecture. ONLY FOR HEAVY HACKERS +** ======================================================================= +*/ +#define this_realloc debug_realloc +#define MARK 0x55 /* 01010101 (a nice pattern) */ +#define MARKSIZE 2*sizeof(size_t) /* size of marks after each block */ +#define fillmem(mem,size) memset(mem, ~MARK, size) + +typedef union MemHeader MemHeader; +union MemHeader { + L_Umaxalign a; /* ensures maximum alignment for Header */ + struct { + size_t size; + MemHeader *next; + size_t mark[2]; + }; +}; + +typedef struct Memcontrol { /* memory-allocator control variables */ + MemHeader *start; + lu_int32 numblocks; + lu_int32 total; + lu_int32 maxmem; + lu_int32 memlimit; +} Memcontrol; +static Memcontrol mc = {NULL,0,0,0,32768*64}; +static size_t marker[2] = {0,0}; + +static void scanBlocks (void) { + MemHeader *p = mc.start; + int i; + char s,e; + for (i=0; p ;i++) { + s = memcmp(p->mark, marker, MARKSIZE) ? '<' : ' '; + e = memcmp(cast(char *, p+1) + p->size, marker, MARKSIZE) ? '>' : ' '; + c_printf("%4u %p %8lu %c %c\n", i, p, p->size, s, e); + ASSERT(p->next); + p = p->next; + } +} + +static int checkBlocks (void) { + MemHeader *p = mc.start; + while(p) { + if (memcmp(p->mark, marker, MARKSIZE) || + memcmp(cast(char *, p+1) + p->size, marker, MARKSIZE)) { + scanBlocks(); + return 0; + } + p = p->next; + } + return 1; +} + + +static void freeblock (MemHeader *block) { + if (block) { + MemHeader *p = mc.start; + MemHeader *next = block->next; + size_t size = block->size; + ASSERT(checkBlocks()); + if (p == block) { + mc.start = next; + } else { + while (p->next != block) { + ASSERT(p); + p = p->next; + } + p->next = next; + } + fillmem(block, sizeof(MemHeader) + size + MARKSIZE); /* erase block */ + c_free(block); /* actually free block */ + mc.numblocks--; /* update counts */ + mc.total -= size; + } +} + +void *debug_realloc (void *b, size_t oldsize, size_t size) { + MemHeader *block = cast(MemHeader *, b); + ASSERT(checkBlocks()); + if (!marker[0]) memset(marker, MARK, MARKSIZE); + if (block == NULL) { + oldsize = 0; + } else { + block--; /* go to real header */ + ASSERT(!memcmp(block->mark, marker, MARKSIZE)) + ASSERT(oldsize == block->size); + ASSERT(!memcmp(cast(char *, b)+oldsize, marker, MARKSIZE)); + } + if (size == 0) { + freeblock(block); + return NULL; + } else if (size > oldsize && mc.total+size-oldsize > mc.memlimit) + return NULL; /* fake a memory allocation error */ + else { + MemHeader *newblock; + size_t commonsize = (oldsize < size) ? oldsize : size; + size_t realsize = sizeof(MemHeader) + size + MARKSIZE; + newblock = cast(MemHeader *, c_malloc(realsize)); /* alloc a new block */ + if (newblock == NULL) + return NULL; /* really out of memory? */ + if (block) { + memcpy(newblock + 1, block + 1, commonsize); /* copy old contents */ + freeblock(block); /* erase (and check) old copy */ + } + /* initialize new part of the block with something weird */ + if (size > commonsize) + fillmem(cast(char *, newblock + 1) + commonsize, size - commonsize); + /* initialize marks after block */ + memset(newblock->mark, MARK, MARKSIZE); + newblock->size = size; + newblock->next = mc.start; + mc.start = newblock; + memset(cast(char *, newblock + 1)+ size, MARK, MARKSIZE); + mc.total += size; + if (mc.total > mc.maxmem) + mc.maxmem = mc.total; + mc.numblocks++; + return (newblock + 1); + } +} + + +/* }====================================================================== */ +#else +#define this_realloc(p,os,s) c_realloc(p,s) +#endif /* DEBUG_ALLOCATOR */ + /* ** {====================================================== ** Error-report functions @@ -633,8 +776,9 @@ LUALIB_API int luaL_loadfile (lua_State *L, const char *filename) { lf.f = c_freopen(filename, "rb", lf.f); /* reopen in binary mode */ if (lf.f == NULL) return errfile(L, "reopen", fnameindex); /* skip eventual `#!...' */ - while ((c = c_getc(lf.f)) != EOF && c != LUA_SIGNATURE[0]) ; - lf.extraline = 0; + while ((c = c_getc(lf.f)) != EOF && c != LUA_SIGNATURE[0]) {} + + lf.extraline = 0; } c_ungetc(c, lf.f); status = lua_load(L, getF, &lf, lua_tostring(L, -1)); @@ -787,8 +931,12 @@ static void *l_alloc (void *ud, void *ptr, size_t osize, size_t nsize) { void *nptr; if (nsize == 0) { +#ifdef DEBUG_ALLOCATOR + return (void *)this_realloc(ptr, osize, nsize); +#else c_free(ptr); return NULL; +#endif } if (L != NULL && (mode & EGC_ALWAYS)) /* always collect memory if requested */ luaC_fullgc(L); @@ -804,17 +952,44 @@ static void *l_alloc (void *ud, void *ptr, size_t osize, size_t nsize) { if(G(L)->memlimit > 0 && (mode & EGC_ON_MEM_LIMIT) && l_check_memlimit(L, nsize - osize)) return NULL; } - nptr = (void *)c_realloc(ptr, nsize); + nptr = (void *)this_realloc(ptr, osize, nsize); if (nptr == NULL && L != NULL && (mode & EGC_ON_ALLOC_FAILURE)) { luaC_fullgc(L); /* emergency full collection. */ - nptr = (void *)c_realloc(ptr, nsize); /* try allocation again */ + nptr = (void *)this_realloc(ptr, osize, nsize); /* try allocation again */ } return nptr; } LUALIB_API void luaL_assertfail(const char *file, int line, const char *message) { dbg_printf("ASSERT@%s(%d): %s\n", file, line, message); +#if defined(LUA_CROSS_COMPILER) + exit(1); +#endif +} + +#ifdef DEVELOPMENT_USE_GDB +/* + * This is a simple stub used by lua_assert() if DEVELOPMENT_USE_GDB is defined. + * Instead of crashing out with an assert error, this hook starts the GDB remote + * stub if not already running and then issues a break. The rationale here is + * that when testing the developer migght be using screen/PuTTY to work ineractively + * with the Lua Interpreter via UART0. However if an assert triggers, then there + * is the option to exit the interactive session and start the Xtensa remote GDB + * which will then sync up with the remote GDB client to allow forensics of the error. + */ +extern void gdbstub_init(void); + +LUALIB_API void luaL_dbgbreak(void) { + static int repeat_entry = 0; + if (repeat_entry == 0) { + dbg_printf("Start up the gdb stub if not already started\n"); + gdbstub_init(); + repeat_entry = 1; + } + asm("break 0,0" ::); } +#endif + static int panic (lua_State *L) { (void)L; /* to avoid warnings */ diff --git a/app/lua/lbaselib.c b/app/lua/lbaselib.c index 02dad4635f..a08f3eaa3d 100644 --- a/app/lua/lbaselib.c +++ b/app/lua/lbaselib.c @@ -712,10 +712,5 @@ static void base_open (lua_State *L) { LUALIB_API int luaopen_base (lua_State *L) { base_open(L); -#if LUA_OPTIMIZE_MEMORY == 0 - luaL_register(L, LUA_COLIBNAME, co_funcs); - return 2; -#else return 1; -#endif } diff --git a/app/lua/ldblib.c b/app/lua/ldblib.c index 1097897077..e084e23374 100644 --- a/app/lua/ldblib.c +++ b/app/lua/ldblib.c @@ -17,6 +17,8 @@ #include "lauxlib.h" #include "lualib.h" #include "lrotable.h" +#include "lstring.h" +#include "lflash.h" #include "user_modules.h" @@ -26,6 +28,39 @@ static int db_getregistry (lua_State *L) { return 1; } +static int db_getstrings (lua_State *L) { + size_t i,n; + stringtable *tb; + GCObject *o; +#if defined(LUA_FLASH_STORE) && !defined(LUA_CROSS_COMPILER) + const char *opt = lua_tolstring (L, 1, &n); + if (n==3 && memcmp(opt, "ROM", 4) == 0) { + if (G(L)->ROstrt.hash == NULL) + return 0; + tb = &G(L)->ROstrt; + } + else +#endif + tb = &G(L)->strt; + lua_settop(L, 0); + lua_createtable(L, tb->nuse, 0); /* create table the same size as the strt */ + for (i=0, n=1; isize; i++) { + for(o = tb->hash[i]; o; o=o->gch.next) { + TString *ts =cast(TString *, o); + lua_pushnil(L); + setsvalue2s(L, L->top-1, ts); + lua_rawseti(L, -2, n++); /* enumerate the strt, adding elements */ + } + } + lua_getfield(L, LUA_GLOBALSINDEX, "table"); + lua_getfield(L, -1, "sort"); /* look up table.sort function */ + lua_replace(L, -2); /* dump the table table */ + lua_pushvalue(L, -2); /* duplicate the strt_copy ref */ + lua_call(L, 1, 0); /* table.sort(strt_copy) */ + return 1; +} + + #ifndef LUA_USE_BUILTIN_DEBUG_MINIMAL static int db_getmetatable (lua_State *L) { @@ -395,6 +430,7 @@ const LUA_REG_TYPE dblib[] = { {LSTRKEY("getlocal"), LFUNCVAL(db_getlocal)}, #endif {LSTRKEY("getregistry"), LFUNCVAL(db_getregistry)}, + {LSTRKEY("getstrings"), LFUNCVAL(db_getstrings)}, #ifndef LUA_USE_BUILTIN_DEBUG_MINIMAL {LSTRKEY("getmetatable"), LFUNCVAL(db_getmetatable)}, {LSTRKEY("getupvalue"), LFUNCVAL(db_getupvalue)}, diff --git a/app/lua/lflash.c b/app/lua/lflash.c new file mode 100644 index 0000000000..f5c4d9085a --- /dev/null +++ b/app/lua/lflash.c @@ -0,0 +1,274 @@ +/* +** $Id: lflash.c +** See Copyright Notice in lua.h +*/ + +#define lflash_c +#define LUA_CORE +#define LUAC_CROSS_FILE +#include "lua.h" + +#ifdef LUA_FLASH_STORE +#include "lobject.h" +#include "lauxlib.h" +#include "lstate.h" +#include "lfunc.h" +#include "lflash.h" +#include "platform.h" +#include "vfs.h" + +#include "c_fcntl.h" +#include "c_stdio.h" +#include "c_stdlib.h" +#include "c_string.h" + +/* + * Flash memory is a fixed memory addressable block that is serially allocated by the + * luac build process and the out image can be downloaded into SPIFSS and loaded into + * flash with a node.flash.load() command. See luac_cross/lflashimg.c for the build + * process. + */ + +static char *flashAddr; +static uint32_t flashAddrPhys; +static uint32_t flashSector; +static uint32_t curOffset; + +#define ALIGN(s) (((s)+sizeof(size_t)-1) & ((size_t) (- (signed) sizeof(size_t)))) +#define ALIGN_BITS(s) (((uint32_t)s) & (sizeof(size_t)-1)) +#define ALL_SET cast(uint32_t, -1) +#define FLASH_SIZE LUA_FLASH_STORE +#define FLASH_PAGE_SIZE INTERNAL_FLASH_SECTOR_SIZE +#define FLASH_PAGES (FLASH_SIZE/FLASH_PAGE_SIZE) + +char flash_region_base[FLASH_SIZE] ICACHE_FLASH_RESERVED_ATTR; + +#ifdef NODE_DEBUG +extern void dbg_printf(const char *fmt, ...) __attribute__ ((format (printf, 1, 2))); +void dumpStrt(stringtable *tb, const char *type) { + int i,j; + GCObject *o; + + NODE_DBG("\nDumping %s String table\n\n========================\n", type); + NODE_DBG("No of elements: %d\nSize of table: %d\n", tb->nuse, tb->size); + for (i=0; isize; i++) + for(o = tb->hash[i], j=0; o; (o=o->gch.next), j++ ) { + TString *ts =cast(TString *, o); + NODE_DBG("%5d %5d %08x %08x %5d %1s %s\n", + i, j, (size_t) ts, ts->tsv.hash, ts->tsv.len, + ts_isreadonly(ts) ? "R" : " ", getstr(ts)); + } +} + +LUA_API void dumpStrings(lua_State *L) { + dumpStrt(&G(L)->strt, "RAM"); + if (G(L)->ROstrt.hash) + dumpStrt(&G(L)->ROstrt, "ROM"); +} +#endif + +/* ===================================================================================== + * The next 4 functions: flashPosition, flashSetPosition, flashBlock and flashErase + * wrap writing to flash. The last two are platform dependent. Also note that any + * writes are suppressed if the global writeToFlash is false. This is used in + * phase I where the pass is used to size the structures in flash. + */ +static char *flashPosition(void){ + return flashAddr + curOffset; +} + + +static char *flashSetPosition(uint32_t offset){ + NODE_DBG("flashSetPosition(%04x)\n", offset); + curOffset = offset; + return flashPosition(); +} + + +static char *flashBlock(const void* b, size_t size) { + void *cur = flashPosition(); + NODE_DBG("flashBlock((%04x),%08x,%04x)\n", curOffset,b,size); + lua_assert(ALIGN_BITS(b) == 0 && ALIGN_BITS(size) == 0); + platform_flash_write(b, flashAddrPhys+curOffset, size); + curOffset += size; + return cur; +} + + +static void flashErase(uint32_t start, uint32_t end){ + int i; + if (start == -1) start = FLASH_PAGES - 1; + if (end == -1) end = FLASH_PAGES - 1; + NODE_DBG("flashErase(%04x,%04x)\n", flashSector+start, flashSector+end); + for (i = start; i<=end; i++) + platform_flash_erase_sector( flashSector + i ); +} + + +/* + * Hook in lstate.c:f_luaopen() to set up ROstrt and ROpvmain if needed + */ +LUAI_FUNC void luaN_init (lua_State *L) { +// luaL_dbgbreak(); + curOffset = 0; + flashAddr = flash_region_base; + flashAddrPhys = platform_flash_mapped2phys((uint32_t)flashAddr); + flashSector = platform_flash_get_sector_of_address(flashAddrPhys); + FlashHeader *fh = cast(FlashHeader *, flashAddr); + + /* + * For the LFS to be valid, its signature has to be correct for this build variant, + * thr ROhash and main proto fields must be defined and the main proto address + * be within the LFS address bounds. (This last check is primarily to detect the + * direct imaging of an absolute LFS with the wrong base address. + */ + + if ((fh->flash_sig & (~FLASH_SIG_ABSOLUTE)) != FLASH_SIG ) { + NODE_ERR("Flash sig not correct: %p vs %p\n", + fh->flash_sig & (~FLASH_SIG_ABSOLUTE), FLASH_SIG); + return; + } + + if (fh->pROhash == ALL_SET || + ((fh->mainProto - cast(FlashAddr, fh)) >= fh->flash_size)) { + NODE_ERR("Flash size check failed: %p vs 0xFFFFFFFF; %p >= %p\n", + fh->mainProto - cast(FlashAddr, fh), fh->flash_size); + return; + } + + G(L)->ROstrt.hash = cast(GCObject **, fh->pROhash); + G(L)->ROstrt.nuse = fh->nROuse ; + G(L)->ROstrt.size = fh->nROsize; + G(L)->ROpvmain = cast(Proto *,fh->mainProto); +} + +#define BYTE_OFFSET(t,f) cast(size_t, &(cast(t *, NULL)->f)) +/* + * Rehook address chain to correct Flash byte addressed within the mapped adress space + * Note that on input each 32-bit address field is split into 2×16-bit subfields + * - the lu_int16 offset of the target address being referenced + * - the lu_int16 offset of the next address pointer. + */ + +static int rebuild_core (int fd, uint32_t size, lu_int32 *buf, int is_absolute) { + int bi; /* byte offset into memory mapped LFS of current buffer */ + int wNextOffset = BYTE_OFFSET(FlashHeader,mainProto)/sizeof(lu_int32); + int wj; /* word offset into current input buffer */ + for (bi = 0; bi < size; bi += FLASH_PAGE_SIZE) { + int wi = bi / sizeof(lu_int32); + int blen = ((bi + FLASH_PAGE_SIZE) < size) ? FLASH_PAGE_SIZE : size - bi; + int wlen = blen / sizeof(lu_int32); + if (vfs_read(fd, buf , blen) != blen) + return 0; + + if (!is_absolute) { + for (wj = 0; wj < wlen; wj++) { + if ((wi + wj) == wNextOffset) { /* this word is the next linked address */ + int wTargetOffset = buf[wj]&0xFFFF; + wNextOffset = buf[wj]>>16; + lua_assert(!wNextOffset || (wNextOffset>(wi+wj) && wNextOffsetROpvmain)) { + lua_settop(L, 0); + lua_pushnil(L); + lua_pushinteger(L, (lua_Integer) flashAddr); + lua_pushinteger(L, flashAddrPhys); + return 3; + } + + /* Push the LClosure of the LFS index function */ + Closure *cl = luaF_newLclosure(L, 0, hvalue(gt(L))); + cl->l.p = G(L)->ROpvmain; + lua_settop(L, n+1); + setclvalue(L, L->top-1, cl); + + /* Move it infront of the arguments and call the index function */ + lua_insert(L, 1); + lua_call(L, n, LUA_MULTRET); + + /* Return it if the response if a single value (the function) */ + if (lua_gettop(L) == 1) + return 1; + + lua_assert(lua_gettop(L) == 2); + + /* Otherwise add the base address of the LFS, and its size bewteen the */ + /* Unix time and the module list, then return all 4 params. */ + lua_pushinteger(L, (lua_Integer) flashAddr); + lua_insert(L, 2); + lua_pushinteger(L, flashAddrPhys); + lua_insert(L, 3); + lua_pushinteger(L, cast(FlashHeader *, flashAddr)->flash_size); + lua_insert(L, 4); + return 5; +} + +#endif diff --git a/app/lua/lflash.h b/app/lua/lflash.h new file mode 100644 index 0000000000..d3ba862b08 --- /dev/null +++ b/app/lua/lflash.h @@ -0,0 +1,48 @@ +/* +** lflashe.h +** See Copyright Notice in lua.h +*/ + +#if defined(LUA_FLASH_STORE) && !defined(lflash_h) +#define lflash_h + +#include "lobject.h" +#include "lstate.h" +#include "lzio.h" + +#ifdef LUA_NUMBER_INTEGRAL +# define FLASH_SIG_B1 0x02 +#else +# define FLASH_SIG_B1 0x00 +#endif + +#ifdef LUA_PACK_TVALUES +#ifdef LUA_NUMBER_INTEGRAL +#error "LUA_PACK_TVALUES is only valid for Floating point builds" +#endif +# define FLASH_SIG_B2 0x04 +#else +# define FLASH_SIG_B2 0x00 +#endif +#define FLASH_SIG_ABSOLUTE 0x01 +#define FLASH_SIG_IN_PROGRESS 0x08 +#define FLASH_SIG (0xfafaaf50 | FLASH_SIG_B2 | FLASH_SIG_B1) + +typedef lu_int32 FlashAddr; +typedef struct { + lu_int32 flash_sig; /* a stabdard fingerprint identifying an LFS image */ + lu_int32 flash_size; /* Size of LFS image */ + FlashAddr mainProto; /* address of main Proto in Proto hierarchy */ + FlashAddr pROhash; /* address of ROstrt hash */ + lu_int32 nROuse; /* number of elements in ROstrt */ + int nROsize; /* size of ROstrt */ + lu_int32 fill1; /* reserved */ + lu_int32 fill2; /* reserved */ +} FlashHeader; + +LUAI_FUNC void luaN_init (lua_State *L); +LUAI_FUNC int luaN_flashSetup (lua_State *L); +LUAI_FUNC int luaN_reload_reboot (lua_State *L); +LUAI_FUNC int luaN_index (lua_State *L); +#endif + diff --git a/app/lua/lfunc.c b/app/lua/lfunc.c index 3d0dd13395..fe502f6bd9 100644 --- a/app/lua/lfunc.c +++ b/app/lua/lfunc.c @@ -146,7 +146,7 @@ void luaF_freeproto (lua_State *L, Proto *f) { luaM_freearray(L, f->k, f->sizek, TValue); luaM_freearray(L, f->locvars, f->sizelocvars, struct LocVar); luaM_freearray(L, f->upvalues, f->sizeupvalues, TString *); - if (!proto_is_readonly(f)) { + if (!proto_isreadonly(f)) { luaM_freearray(L, f->code, f->sizecode, Instruction); #ifdef LUA_OPTIMIZE_DEBUG if (f->packedlineinfo) { diff --git a/app/lua/lfunc.h b/app/lua/lfunc.h index 1450bb7d91..2dc7266b6a 100644 --- a/app/lua/lfunc.h +++ b/app/lua/lfunc.h @@ -18,9 +18,6 @@ #define sizeLclosure(n) (cast(int, sizeof(LClosure)) + \ cast(int, sizeof(TValue *)*((n)-1))) -#define proto_readonly(p) l_setbit((p)->marked, READONLYBIT) -#define proto_is_readonly(p) testbit((p)->marked, READONLYBIT) - LUAI_FUNC Proto *luaF_newproto (lua_State *L); LUAI_FUNC Closure *luaF_newCclosure (lua_State *L, int nelems, Table *e); LUAI_FUNC Closure *luaF_newLclosure (lua_State *L, int nelems, Table *e); diff --git a/app/lua/lgc.c b/app/lua/lgc.c index e41a779d53..939c4d29b3 100644 --- a/app/lua/lgc.c +++ b/app/lua/lgc.c @@ -28,6 +28,9 @@ #define GCSWEEPCOST 10 #define GCFINALIZECOST 100 +#if READONLYMASK != (1<gch.marked, WHITE0BIT, WHITE1BIT) #define black2gray(x) resetbit((x)->gch.marked, BLACKBIT) -#define stringmark(s) reset2bits((s)->tsv.marked, WHITE0BIT, WHITE1BIT) +#define stringmark(s) if (!isLFSobject(&(s)->tsv)) {reset2bits((s)->tsv.marked, WHITE0BIT, WHITE1BIT);} #define isfinalized(u) testbit((u)->marked, FINALIZEDBIT) @@ -61,12 +64,18 @@ static void removeentry (Node *n) { lua_assert(ttisnil(gval(n))); - if (iscollectable(gkey(n))) + if (ttype(gkey(n)) != LUA_TDEADKEY && iscollectable(gkey(n))) +// The gkey is always in RAM so it can be marked as DEAD even though it +// refers to an LFS object. setttype(gkey(n), LUA_TDEADKEY); /* dead key; remove it */ } static void reallymarkobject (global_State *g, GCObject *o) { + /* don't mark LFS Protos (or strings) */ + if (o->gch.tt == LUA_TPROTO && isLFSobject(&(o->gch))) + return; + lua_assert(iswhite(o) && !isdead(g, o)); white2gray(o); switch (o->gch.tt) { @@ -180,6 +189,8 @@ static int traversetable (global_State *g, Table *h) { while (i--) markvalue(g, &h->array[i]); } + if (luaH_isdummy (h->node)) + return weakkey || weakvalue; i = sizenode(h); while (i--) { Node *n = gnode(h, i); @@ -202,6 +213,8 @@ static int traversetable (global_State *g, Table *h) { */ static void traverseproto (global_State *g, Proto *f) { int i; + if (isLFSobject(f)) + return; /* don't traverse Protos in LFS */ if (f->source) stringmark(f->source); for (i=0; isizek; i++) /* mark literals */ markvalue(g, &f->k[i]); @@ -317,7 +330,7 @@ static l_mem propagatemark (global_State *g) { sizeof(TValue) * p->sizek + sizeof(LocVar) * p->sizelocvars + sizeof(TString *) * p->sizeupvalues + - (proto_is_readonly(p) ? 0 : sizeof(Instruction) * p->sizecode + + (proto_isreadonly(p) ? 0 : sizeof(Instruction) * p->sizecode + #ifdef LUA_OPTIMIZE_DEBUG (p->packedlineinfo ? c_strlen(cast(char *, p->packedlineinfo))+1 : @@ -388,7 +401,10 @@ static void cleartable (GCObject *l) { static void freeobj (lua_State *L, GCObject *o) { switch (o->gch.tt) { - case LUA_TPROTO: luaF_freeproto(L, gco2p(o)); break; + case LUA_TPROTO: + lua_assert(!isLFSobject(&(o->gch))); + luaF_freeproto(L, gco2p(o)); + break; case LUA_TFUNCTION: luaF_freeclosure(L, gco2cl(o)); break; case LUA_TUPVAL: luaF_freeupval(L, gco2uv(o)); break; case LUA_TTABLE: luaH_free(L, gco2h(o)); break; @@ -398,6 +414,7 @@ static void freeobj (lua_State *L, GCObject *o) { break; } case LUA_TSTRING: { + lua_assert(!isLFSobject(&(o->gch))); G(L)->strt.nuse--; luaM_freemem(L, o, sizestring(gco2ts(o))); break; @@ -420,6 +437,7 @@ static GCObject **sweeplist (lua_State *L, GCObject **p, lu_mem count) { global_State *g = G(L); int deadmask = otherwhite(g); while ((curr = *p) != NULL && count-- > 0) { + lua_assert(!isLFSobject(&(curr->gch)) || curr->gch.tt == LUA_TTHREAD); if (curr->gch.tt == LUA_TTHREAD) /* sweep open upvalues of each thread */ sweepwholelist(L, &gco2th(curr)->openupval); if ((curr->gch.marked ^ WHITEBITS) & deadmask) { /* not dead? */ @@ -538,7 +556,7 @@ static void atomic (lua_State *L) { size_t udsize; /* total size of userdata to be finalized */ /* remark occasional upvalues of (maybe) dead threads */ remarkupvals(g); - /* traverse objects cautch by write barrier and by 'remarkupvals' */ + /* traverse objects caucht by write barrier and by 'remarkupvals' */ propagateall(g); /* remark weak tables */ g->gray = g->weak; @@ -694,10 +712,10 @@ void luaC_barrierf (lua_State *L, GCObject *o, GCObject *v) { global_State *g = G(L); lua_assert(isblack(o) && iswhite(v) && !isdead(g, v) && !isdead(g, o)); lua_assert(g->gcstate != GCSfinalize && g->gcstate != GCSpause); - lua_assert(ttype(&o->gch) != LUA_TTABLE); + lua_assert(o->gch.tt != LUA_TTABLE); /* must keep invariant? */ if (g->gcstate == GCSpropagate) - reallymarkobject(g, v); /* restore invariant */ + reallymarkobject(g, v); /* Restore invariant */ else /* don't mind */ makewhite(g, o); /* mark as white just to avoid other barriers */ } diff --git a/app/lua/lgc.h b/app/lua/lgc.h index 9c26932b46..ed994d09ea 100644 --- a/app/lua/lgc.h +++ b/app/lua/lgc.h @@ -79,6 +79,7 @@ #define VALUEWEAKBIT 4 #define FIXEDBIT 5 #define SFIXEDBIT 6 +#define LFSBIT 6 #define READONLYBIT 7 #define WHITEBITS bit2mask(WHITE0BIT, WHITE1BIT) @@ -100,6 +101,13 @@ #define isfixedstack(x) testbit((x)->marked, FIXEDSTACKBIT) #define fixedstack(x) l_setbit((x)->marked, FIXEDSTACKBIT) #define unfixedstack(x) resetbit((x)->marked, FIXEDSTACKBIT) +#ifdef LUA_FLASH_STORE +#define isLFSobject(x) testbit((x)->marked, LFSBIT) +#define stringfix(s) if (!test2bits((s)->tsv.marked, FIXEDBIT, LFSBIT)) {l_setbit((s)->tsv.marked, FIXEDBIT);} +#else +#define isLFSobject(x) (0) +#define stringfix(s) {l_setbit((s)->tsv.marked, FIXEDBIT);} +#endif #define luaC_checkGC(L) { \ condhardstacktests(luaD_reallocstack(L, L->stacksize - EXTRA_STACK - 1)); \ diff --git a/app/lua/linit.c b/app/lua/linit.c new file mode 100644 index 0000000000..1a7c4b5813 --- /dev/null +++ b/app/lua/linit.c @@ -0,0 +1,78 @@ +/* +** $Id: linit.c,v 1.14.1.1 2007/12/27 13:02:25 roberto Exp $ +** Initialization of libraries for lua.c +** See Copyright Notice in lua.h +*/ + + +#define linit_c +#define LUA_LIB +#define LUAC_CROSS_FILE + +#include "lua.h" + +#include "lualib.h" +#include "lauxlib.h" +#include "luaconf.h" +#include "module.h" +#if defined(LUA_CROSS_COMPILER) +BUILTIN_LIB( start_list, NULL, NULL); +BUILTIN_LIB_INIT( start_list, NULL, NULL); +#endif +extern const luaR_entry strlib[], tab_funcs[], dblib[], + co_funcs[], math_map[], syslib[]; + +BUILTIN_LIB_INIT( BASE, "", luaopen_base); +BUILTIN_LIB_INIT( LOADLIB, LUA_LOADLIBNAME, luaopen_package); + +BUILTIN_LIB( STRING, LUA_STRLIBNAME, strlib); +BUILTIN_LIB_INIT( STRING, LUA_STRLIBNAME, luaopen_string); + +BUILTIN_LIB( TABLE, LUA_TABLIBNAME, tab_funcs); +BUILTIN_LIB_INIT( TABLE, LUA_TABLIBNAME, luaopen_table); + +BUILTIN_LIB( DBG, LUA_DBLIBNAME, dblib); +BUILTIN_LIB_INIT( DBG, LUA_DBLIBNAME, luaopen_debug); + +BUILTIN_LIB( CO, LUA_COLIBNAME, co_funcs); + +BUILTIN_LIB( MATH, LUA_MATHLIBNAME, math_map); + +#if defined(LUA_CROSS_COMPILER) +extern const luaR_entry syslib[], iolib[]; +BUILTIN_LIB( OS, LUA_OSLIBNAME, syslib); +BUILTIN_LIB_INIT( IO, LUA_IOLIBNAME, luaopen_io); +BUILTIN_LIB( end_list, NULL, NULL); +BUILTIN_LIB_INIT( end_list, NULL, NULL); +/* + * These base addresses are internal to this module for cross compile builds + * This also exploits feature of the GCC code generator that the variables are + * emitted in either normal OR reverse order within PSECT. + */ +#define isascending(n) ((&(n ## _end_list)-&(n ## _start_list))>0) +static const luaL_Reg *lua_libs; +const luaR_table *lua_rotable; +#else +/* These base addresses are Xtensa toolchain linker constants for Firmware builds */ +extern const luaL_Reg lua_libs_base[]; +extern const luaR_table lua_rotable_base[]; +static const luaL_Reg *lua_libs = lua_libs_base; +const luaR_table *lua_rotable = lua_rotable_base; +#endif + +void luaL_openlibs (lua_State *L) { +#if defined(LUA_CROSS_COMPILER) +lua_libs = (isascending(lua_lib) ? &lua_lib_start_list : &lua_lib_end_list) + 1; +lua_rotable = (isascending(lua_rotable) ? &lua_rotable_start_list : &lua_rotable_end_list) + 1; +#endif + const luaL_Reg *lib = lua_libs; + for (; lib->name; lib++) { + if (lib->func) + { + lua_pushcfunction(L, lib->func); + lua_pushstring(L, lib->name); + lua_call(L, 1, 0); + } + } +} + diff --git a/app/lua/lmathlib.c b/app/lua/lmathlib.c index e3fda445fd..8a3b975461 100644 --- a/app/lua/lmathlib.c +++ b/app/lua/lmathlib.c @@ -326,7 +326,7 @@ const LUA_REG_TYPE math_map[] = { {LSTRKEY("randomseed"), LFUNCVAL(math_randomseed)}, {LSTRKEY("sqrt"), LFUNCVAL(math_sqrt)}, #if LUA_OPTIMIZE_MEMORY > 0 - {LSTRKEY("huge"), LNUMVAL(LONG_MAX)}, + {LSTRKEY("huge"), LNUMVAL(INT_MAX)}, #endif #else {LSTRKEY("abs"), LFUNCVAL(math_abs)}, @@ -374,7 +374,7 @@ const LUA_REG_TYPE math_map[] = { */ #if defined LUA_NUMBER_INTEGRAL -# include "c_limits.h" /* for LONG_MAX */ +# include "c_limits.h" /* for INT_MAX */ #endif LUALIB_API int luaopen_math (lua_State *L) { @@ -383,7 +383,7 @@ LUALIB_API int luaopen_math (lua_State *L) { #else luaL_register(L, LUA_MATHLIBNAME, math_map); # if defined LUA_NUMBER_INTEGRAL - lua_pushnumber(L, LONG_MAX); + lua_pushnumber(L, INT_MAX); lua_setfield(L, -2, "huge"); # else lua_pushnumber(L, PI); diff --git a/app/lua/loadlib.c b/app/lua/loadlib.c index a697b74c6f..f884c3ca3a 100644 --- a/app/lua/loadlib.c +++ b/app/lua/loadlib.c @@ -613,7 +613,7 @@ static int ll_seeall (lua_State *L) { static void setpath (lua_State *L, const char *fieldname, const char *envname, const char *def) { - const char *path = c_getenv(envname); + const char *path = NULL; /* getenv(envname) not used in NodeMCU */; if (path == NULL) /* no environment variable? */ lua_pushstring(L, def); /* use default */ else { diff --git a/app/lua/lobject.c b/app/lua/lobject.c index 6a68455943..bca1f8a843 100644 --- a/app/lua/lobject.c +++ b/app/lua/lobject.c @@ -53,7 +53,8 @@ int luaO_fb2int (int x) { int luaO_log2 (unsigned int x) { - static const lu_byte log_2[256] ICACHE_STORE_ATTR ICACHE_RODATA_ATTR = { +#ifdef LUA_CROSS_COMPILER + static const lu_byte log_2[256] = { 0,1,2,2,3,3,3,3,4,4,4,4,4,4,4,4,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5, 6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6, 7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7, @@ -65,10 +66,12 @@ int luaO_log2 (unsigned int x) { }; int l = -1; while (x >= 256) { l += 8; x >>= 8; } -#ifdef LUA_CROSS_COMPILER return l + log_2[x]; #else - return l + byte_of_aligned_array(log_2,x); + /* Use Normalization Shift Amount Unsigned: 0x1=>31 up to 0xffffffff =>0 + * See Xtensa Instruction Set Architecture (ISA) Refman P 462 */ + asm volatile ("nsau %0, %1;" :"=r"(x) : "r"(x)); + return 31 - x; #endif } @@ -103,7 +106,7 @@ int luaO_str2d (const char *s, lua_Number *result) { #if defined(LUA_CROSS_COMPILER) { long lres = strtoul(s, &endptr, 16); -#if LONG_MAX != 2147483647L +#if INT_MAX != 2147483647L if (lres & ~0xffffffffL) *result = cast_num(-1); else if (lres & 0x80000000L) diff --git a/app/lua/lobject.h b/app/lua/lobject.h index 5b020953da..e1f1962d4f 100644 --- a/app/lua/lobject.h +++ b/app/lua/lobject.h @@ -23,10 +23,10 @@ #define NUM_TAGS (LAST_TAG+1) -/* mask for 'read-only' objects. must match READONLYBIT in lgc.h' */ -#define READONLYMASK 128 - - +#define READONLYMASK (1<<7) /* denormalised bitmask for READONLYBIT and */ +#ifdef LUA_FLASH_STORE +#define LFSMASK (1<<6) /* LFSBIT to avoid include proliferation */ +#endif /* ** Extra tags for non-values */ @@ -55,86 +55,39 @@ typedef struct GCheader { CommonHeader; } GCheader; - +#if defined(LUA_PACK_VALUE) || defined(ELUA_ENDIAN_BIG) || defined(ELUA_ENDIAN_SMALL) +# error "NodeMCU does not support the eLua LUA_PACK_VALUE and ELUA_ENDIAN defines" +#endif /* ** Union of all Lua values */ -#if defined( LUA_PACK_VALUE ) && defined( ELUA_ENDIAN_BIG ) -typedef union { - struct { - int _pad0; - GCObject *gc; - }; - struct { - int _pad1; - void *p; - }; - lua_Number n; - struct { - int _pad2; - int b; - }; -} Value; -#else // #if defined( LUA_PACK_VALUE ) && defined( ELUA_ENDIAN_BIG ) typedef union { GCObject *gc; void *p; lua_Number n; int b; } Value; -#endif // #if defined( LUA_PACK_VALUE ) && defined( ELUA_ENDIAN_BIG ) /* ** Tagged Values */ -#ifndef LUA_PACK_VALUE #define TValuefields Value value; int tt #define LUA_TVALUE_NIL {NULL}, LUA_TNIL +#if defined(LUA_PACK_TVALUES) && !defined(LUA_CROSS_COMPILER) +#pragma pack(4) +#endif typedef struct lua_TValue { TValuefields; } TValue; -#else // #ifndef LUA_PACK_VALUE -#ifdef ELUA_ENDIAN_LITTLE -#define TValuefields union { \ - struct { \ - int _pad0; \ - int tt_sig; \ - } _ts; \ - struct { \ - int _pad; \ - short tt; \ - short sig; \ - } _t; \ - Value value; \ -} -#define LUA_TVALUE_NIL {0, add_sig(LUA_TNIL)} -#else // #ifdef ELUA_ENDIAN_LITTLE -#define TValuefields union { \ - struct { \ - int tt_sig; \ - int _pad0; \ - } _ts; \ - struct { \ - short sig; \ - short tt; \ - int _pad; \ - } _t; \ - Value value; \ -} -#define LUA_TVALUE_NIL {add_sig(LUA_TNIL), 0} -#endif // #ifdef ELUA_ENDIAN_LITTLE -#define LUA_NOTNUMBER_SIG (-1) -#define add_sig(tt) ( 0xffff0000 | (tt) ) - -typedef TValuefields TValue; -#endif // #ifndef LUA_PACK_VALUE +#if defined(LUA_PACK_TVALUES) && !defined(LUA_CROSS_COMPILER) +#pragma pack() +#endif /* Macros to test type */ -#ifndef LUA_PACK_VALUE #define ttisnil(o) (ttype(o) == LUA_TNIL) #define ttisnumber(o) (ttype(o) == LUA_TNUMBER) #define ttisstring(o) (ttype(o) == LUA_TSTRING) @@ -146,27 +99,11 @@ typedef TValuefields TValue; #define ttislightuserdata(o) (ttype(o) == LUA_TLIGHTUSERDATA) #define ttisrotable(o) (ttype(o) == LUA_TROTABLE) #define ttislightfunction(o) (ttype(o) == LUA_TLIGHTFUNCTION) -#else // #ifndef LUA_PACK_VALUE -#define ttisnil(o) (ttype_sig(o) == add_sig(LUA_TNIL)) -#define ttisnumber(o) ((o)->_t.sig != LUA_NOTNUMBER_SIG) -#define ttisstring(o) (ttype_sig(o) == add_sig(LUA_TSTRING)) -#define ttistable(o) (ttype_sig(o) == add_sig(LUA_TTABLE)) -#define ttisfunction(o) (ttype_sig(o) == add_sig(LUA_TFUNCTION)) -#define ttisboolean(o) (ttype_sig(o) == add_sig(LUA_TBOOLEAN)) -#define ttisuserdata(o) (ttype_sig(o) == add_sig(LUA_TUSERDATA)) -#define ttisthread(o) (ttype_sig(o) == add_sig(LUA_TTHREAD)) -#define ttislightuserdata(o) (ttype_sig(o) == add_sig(LUA_TLIGHTUSERDATA)) -#define ttisrotable(o) (ttype_sig(o) == add_sig(LUA_TROTABLE)) -#define ttislightfunction(o) (ttype_sig(o) == add_sig(LUA_TLIGHTFUNCTION)) -#endif // #ifndef LUA_PACK_VALUE + /* Macros to access values */ -#ifndef LUA_PACK_VALUE -#define ttype(o) ((o)->tt) -#else // #ifndef LUA_PACK_VALUE -#define ttype(o) ((o)->_t.sig == LUA_NOTNUMBER_SIG ? (o)->_t.tt : LUA_TNUMBER) -#define ttype_sig(o) ((o)->_ts.tt_sig) -#endif // #ifndef LUA_PACK_VALUE + +#define ttype(o) ((void) (o)->value, (o)->tt) #define gcvalue(o) check_exp(iscollectable(o), (o)->value.gc) #define pvalue(o) check_exp(ttislightuserdata(o), (o)->value.p) #define rvalue(o) check_exp(ttisrotable(o), (o)->value.p) @@ -186,24 +123,15 @@ typedef TValuefields TValue; /* ** for internal debug only */ -#ifndef LUA_PACK_VALUE + #define checkconsistency(obj) \ lua_assert(!iscollectable(obj) || (ttype(obj) == (obj)->value.gc->gch.tt)) #define checkliveness(g,obj) \ lua_assert(!iscollectable(obj) || \ ((ttype(obj) == (obj)->value.gc->gch.tt) && !isdead(g, (obj)->value.gc))) -#else // #ifndef LUA_PACK_VALUE -#define checkconsistency(obj) \ - lua_assert(!iscollectable(obj) || (ttype(obj) == (obj)->value.gc->gch._t.tt)) - -#define checkliveness(g,obj) \ - lua_assert(!iscollectable(obj) || \ - ((ttype(obj) == (obj)->value.gc->gch._t.tt) && !isdead(g, (obj)->value.gc))) -#endif // #ifndef LUA_PACK_VALUE /* Macros to set values */ -#ifndef LUA_PACK_VALUE #define setnilvalue(obj) ((obj)->tt=LUA_TNIL) #define setnvalue(obj,x) \ @@ -257,69 +185,10 @@ typedef TValuefields TValue; i_o->value.gc=i_x; i_o->tt=LUA_TPROTO; \ checkliveness(G(L),i_o); } - - - #define setobj(L,obj1,obj2) \ { const TValue *o2=(obj2); TValue *o1=(obj1); \ o1->value = o2->value; o1->tt=o2->tt; \ checkliveness(G(L),o1); } -#else // #ifndef LUA_PACK_VALUE -#define setnilvalue(obj) ( ttype_sig(obj) = add_sig(LUA_TNIL) ) - -#define setnvalue(obj,x) \ - { TValue *i_o=(obj); i_o->value.n=(x); } - -#define setpvalue(obj,x) \ - { TValue *i_o=(obj); i_o->value.p=(x); i_o->_ts.tt_sig=add_sig(LUA_TLIGHTUSERDATA);} - -#define setrvalue(obj,x) \ - { TValue *i_o=(obj); i_o->value.p=(x); i_o->_ts.tt_sig=add_sig(LUA_TROTABLE);} - -#define setfvalue(obj,x) \ - { TValue *i_o=(obj); i_o->value.p=(x); i_o->_ts.tt_sig=add_sig(LUA_TLIGHTFUNCTION);} - -#define setbvalue(obj,x) \ - { TValue *i_o=(obj); i_o->value.b=(x); i_o->_ts.tt_sig=add_sig(LUA_TBOOLEAN);} - -#define setsvalue(L,obj,x) \ - { TValue *i_o=(obj); \ - i_o->value.gc=cast(GCObject *, (x)); i_o->_ts.tt_sig=add_sig(LUA_TSTRING); \ - checkliveness(G(L),i_o); } - -#define setuvalue(L,obj,x) \ - { TValue *i_o=(obj); \ - i_o->value.gc=cast(GCObject *, (x)); i_o->_ts.tt_sig=add_sig(LUA_TUSERDATA); \ - checkliveness(G(L),i_o); } - -#define setthvalue(L,obj,x) \ - { TValue *i_o=(obj); \ - i_o->value.gc=cast(GCObject *, (x)); i_o->_ts.tt_sig=add_sig(LUA_TTHREAD); \ - checkliveness(G(L),i_o); } - -#define setclvalue(L,obj,x) \ - { TValue *i_o=(obj); \ - i_o->value.gc=cast(GCObject *, (x)); i_o->_ts.tt_sig=add_sig(LUA_TFUNCTION); \ - checkliveness(G(L),i_o); } - -#define sethvalue(L,obj,x) \ - { TValue *i_o=(obj); \ - i_o->value.gc=cast(GCObject *, (x)); i_o->_ts.tt_sig=add_sig(LUA_TTABLE); \ - checkliveness(G(L),i_o); } - -#define setptvalue(L,obj,x) \ - { TValue *i_o=(obj); \ - i_o->value.gc=cast(GCObject *, (x)); i_o->_ts.tt_sig=add_sig(LUA_TPROTO); \ - checkliveness(G(L),i_o); } - - - - -#define setobj(L,obj1,obj2) \ - { const TValue *o2=(obj2); TValue *o1=(obj1); \ - o1->value = o2->value; \ - checkliveness(G(L),o1); } -#endif // #ifndef LUA_PACK_VALUE /* ** different types of sets, according to destination @@ -340,13 +209,7 @@ typedef TValuefields TValue; #define setobj2n setobj #define setsvalue2n setsvalue -#ifndef LUA_PACK_VALUE -#define setttype(obj, tt) (ttype(obj) = (tt)) -#else // #ifndef LUA_PACK_VALUE -/* considering it used only in lgc to set LUA_TDEADKEY */ -/* we could define it this way */ -#define setttype(obj, _tt) ( ttype_sig(obj) = add_sig(_tt) ) -#endif // #ifndef LUA_PACK_VALUE +#define setttype(obj, stt) ((void) (obj)->value, (obj)->tt = (stt)) #define iscollectable(o) (ttype(o) >= LUA_TSTRING) @@ -367,9 +230,16 @@ typedef union TString { } tsv; } TString; - -#define getstr(ts) (((ts)->tsv.marked & READONLYMASK) ? cast(const char *, *(const char**)((ts) + 1)) : cast(const char *, (ts) + 1)) -#define svalue(o) getstr(rawtsvalue(o)) +#ifdef LUA_CROSS_COMPILER +#define isreadonly(o) (0) +#else +#define isreadonly(o) ((o).marked & READONLYMASK) +#endif +#define ts_isreadonly(ts) isreadonly((ts)->tsv) +#define getstr(ts) (ts_isreadonly(ts) ? \ + cast(const char *, *(const char**)((ts) + 1)) : \ + cast(const char *, (ts) + 1)) +#define svalue(o) getstr(rawtsvalue(o)) @@ -418,6 +288,7 @@ typedef struct Proto { lu_byte is_vararg; lu_byte maxstacksize; } Proto; +#define proto_isreadonly(p) isreadonly(*(p)) /* masks for new-style vararg */ @@ -487,7 +358,6 @@ typedef union Closure { ** Tables */ -#ifndef LUA_PACK_VALUE typedef union TKey { struct { TValuefields; @@ -497,16 +367,6 @@ typedef union TKey { } TKey; #define LUA_TKEY_NIL {LUA_TVALUE_NIL, NULL} -#else // #ifndef LUA_PACK_VALUE -typedef struct TKey { - TValue tvk; - struct { - struct Node *next; /* for chaining */ - } nk; -} TKey; - -#define LUA_TKEY_NIL {LUA_TVALUE_NIL}, {NULL} -#endif // #ifndef LUA_PACK_VALUE typedef struct Node { TValue i_val; diff --git a/app/lua/lparser.c b/app/lua/lparser.c index 7b14c3d25d..ae95535983 100644 --- a/app/lua/lparser.c +++ b/app/lua/lparser.c @@ -916,12 +916,11 @@ static int block_follow (int token) { static void block (LexState *ls) { /* block -> chunk */ FuncState *fs = ls->fs; - BlockCnt *pbl = (BlockCnt*)luaM_malloc(ls->L,sizeof(BlockCnt)); - enterblock(fs, pbl, 0); + BlockCnt bl; + enterblock(fs, &bl, 0); chunk(ls); - lua_assert(pbl->breaklist == NO_JUMP); + lua_assert(bl.breaklist == NO_JUMP); leaveblock(fs); - luaM_free(ls->L,pbl); } @@ -1081,13 +1080,13 @@ static int exp1 (LexState *ls) { static void forbody (LexState *ls, int base, int line, int nvars, int isnum) { /* forbody -> DO block */ - BlockCnt *pbl = (BlockCnt*)luaM_malloc(ls->L,sizeof(BlockCnt)); + BlockCnt bl; FuncState *fs = ls->fs; int prep, endfor; adjustlocalvars(ls, 3); /* control variables */ checknext(ls, TK_DO); prep = isnum ? luaK_codeAsBx(fs, OP_FORPREP, base, NO_JUMP) : luaK_jump(fs); - enterblock(fs, pbl, 0); /* scope for declared variables */ + enterblock(fs, &bl, 0); /* scope for declared variables */ adjustlocalvars(ls, nvars); luaK_reserveregs(fs, nvars); block(ls); @@ -1097,7 +1096,6 @@ static void forbody (LexState *ls, int base, int line, int nvars, int isnum) { luaK_codeABC(fs, OP_TFORLOOP, base, 0, nvars); luaK_fixline(fs, line); /* pretend that `OP_FOR' starts the loop */ luaK_patchlist(fs, (isnum ? endfor : luaK_jump(fs)), prep + 1); - luaM_free(ls->L,pbl); } diff --git a/app/lua/lrodefs.h b/app/lua/lrodefs.h index e9bbe7feed..451b36216e 100644 --- a/app/lua/lrodefs.h +++ b/app/lua/lrodefs.h @@ -15,7 +15,7 @@ #undef LNILVAL #undef LREGISTER -#if (MIN_OPT_LEVEL > 0) && (LUA_OPTIMIZE_MEMORY >= MIN_OPT_LEVEL) +#if LUA_OPTIMIZE_MEMORY >=1 #define LUA_REG_TYPE luaR_entry #define LSTRKEY LRO_STRKEY #define LNUMKEY LRO_NUMKEY diff --git a/app/lua/lrotable.c b/app/lua/lrotable.c index 33e0b45d11..a5cbb7cae9 100644 --- a/app/lua/lrotable.c +++ b/app/lua/lrotable.c @@ -14,7 +14,7 @@ #define LUAR_FINDVALUE 1 /* Externally defined read-only table array */ -extern const luaR_table lua_rotable[]; +extern const luaR_table *lua_rotable; /* Find a global "read only table" in the constant lua_rotable array */ void* luaR_findglobal(const char *name, unsigned len) { @@ -85,7 +85,7 @@ static void luaR_next_helper(lua_State *L, const luaR_entry *pentries, int pos, if (pentries[pos].key.type != LUA_TNIL) { /* Found an entry */ if (pentries[pos].key.type == LUA_TSTRING) - setsvalue(L, key, luaS_newro(L, pentries[pos].key.id.strkey)) + setsvalue(L, key, luaS_new(L, pentries[pos].key.id.strkey)) else setnvalue(key, (lua_Number)pentries[pos].key.id.numkey) setobj2s(L, val, &pentries[pos].value); @@ -127,10 +127,15 @@ void luaR_getcstr(char *dest, const TString *src, size_t maxsize) { /* Return 1 if the given pointer is a rotable */ #ifdef LUA_META_ROTABLES - +#ifdef LUA_CROSS_COMPILER +extern char edata[]; +int luaR_isrotable(void *p) { + return (char*)p <= edata; +} +#else #include "compiler.h" - int luaR_isrotable(void *p) { return RODATA_START_ADDRESS <= (char*)p && (char*)p <= RODATA_END_ADDRESS; } #endif +#endif diff --git a/app/lua/lrotable.h b/app/lua/lrotable.h index e8963e3b96..aba263527f 100644 --- a/app/lua/lrotable.h +++ b/app/lua/lrotable.h @@ -4,32 +4,16 @@ #define lrotable_h #include "lua.h" -#include "llimits.h" -#include "lobject.h" #include "luaconf.h" +#include "lobject.h" +#include "llimits.h" /* Macros one can use to define rotable entries */ -#ifndef LUA_PACK_VALUE #define LRO_FUNCVAL(v) {{.p = v}, LUA_TLIGHTFUNCTION} #define LRO_LUDATA(v) {{.p = v}, LUA_TLIGHTUSERDATA} #define LRO_NUMVAL(v) {{.n = v}, LUA_TNUMBER} #define LRO_ROVAL(v) {{.p = (void*)v}, LUA_TROTABLE} #define LRO_NILVAL {{.p = NULL}, LUA_TNIL} -#else // #ifndef LUA_PACK_VALUE -#define LRO_NUMVAL(v) {.value.n = v} -#ifdef ELUA_ENDIAN_LITTLE -#define LRO_FUNCVAL(v) {{(int)v, add_sig(LUA_TLIGHTFUNCTION)}} -#define LRO_LUDATA(v) {{(int)v, add_sig(LUA_TLIGHTUSERDATA)}} -#define LRO_ROVAL(v) {{(int)v, add_sig(LUA_TROTABLE)}} -#define LRO_NILVAL {{0, add_sig(LUA_TNIL)}} -#else // #ifdef ELUA_ENDIAN_LITTLE -#define LRO_FUNCVAL(v) {{add_sig(LUA_TLIGHTFUNCTION), (int)v}} -#define LRO_LUDATA(v) {{add_sig(LUA_TLIGHTUSERDATA), (int)v}} -#define LRO_ROVAL(v) {{add_sig(LUA_TROTABLE), (int)v}} -#define LRO_NILVAL {{add_sig(LUA_TNIL), 0}} -#endif // #ifdef ELUA_ENDIAN_LITTLE -#endif // #ifndef LUA_PACK_VALUE - #define LRO_STRKEY(k) {LUA_TSTRING, {.strkey = k}} #define LRO_NUMKEY(k) {LUA_TNUMBER, {.numkey = k}} #define LRO_NILKEY {LUA_TNIL, {.strkey=NULL}} diff --git a/app/lua/lstate.c b/app/lua/lstate.c index b428974210..a0cb60dcdf 100644 --- a/app/lua/lstate.c +++ b/app/lua/lstate.c @@ -13,6 +13,7 @@ #include "ldebug.h" #include "ldo.h" +#include "lflash.h" #include "lfunc.h" #include "lgc.h" #include "llex.h" @@ -72,9 +73,12 @@ static void f_luaopen (lua_State *L, void *ud) { sethvalue(L, gt(L), luaH_new(L, 0, 2)); /* table of globals */ sethvalue(L, registry(L), luaH_new(L, 0, 2)); /* registry */ luaS_resize(L, MINSTRTABSIZE); /* initial size of string table */ +#if defined(LUA_FLASH_STORE) && !defined(LUA_CROSS_COMPILER) + luaN_init(L); /* optionally map RO string table */ +#endif luaT_init(L); luaX_init(L); - luaS_fix(luaS_newliteral(L, MEMERRMSG)); + stringfix(luaS_newliteral(L, MEMERRMSG)); g->GCthreshold = 4*g->totalbytes; } @@ -191,6 +195,12 @@ LUA_API lua_State *lua_newstate (lua_Alloc f, void *ud) { g->memlimit = EGC_INITIAL_MEMLIMIT; #else g->memlimit = 0; +#endif +#if defined(LUA_FLASH_STORE) && !defined(LUA_CROSS_COMPILER) + g->ROstrt.size = 0; + g->ROstrt.nuse = 0; + g->ROstrt.hash = NULL; + g->ROpvmain = NULL; #endif for (i=0; imt[i] = NULL; if (luaD_rawrunprotected(L, f_luaopen, NULL) != 0) { diff --git a/app/lua/lstate.h b/app/lua/lstate.h index 97a1a4b20c..55d3d89c4e 100644 --- a/app/lua/lstate.h +++ b/app/lua/lstate.h @@ -94,6 +94,10 @@ typedef struct global_State { UpVal uvhead; /* head of double-linked list of all open upvalues */ struct Table *mt[NUM_TAGS]; /* metatables for basic types */ TString *tmname[TM_N]; /* array with tag-method names */ +#if defined(LUA_FLASH_STORE) && !defined(LUA_CROSS_COMPILER) + stringtable ROstrt; /* Flash-based hash table for RO strings */ + Proto *ROpvmain; /* Flash-based Proto main */ +#endif } global_State; diff --git a/app/lua/lstring.c b/app/lua/lstring.c index 8f03e52a72..2257c1464d 100644 --- a/app/lua/lstring.c +++ b/app/lua/lstring.c @@ -61,7 +61,7 @@ static TString *newlstr (lua_State *L, const char *str, size_t l, tb = &G(L)->strt; if ((tb->nuse + 1) > cast(lu_int32, tb->size) && tb->size <= MAX_INT/2) luaS_resize(L, tb->size*2); /* too crowded */ - ts = cast(TString *, luaM_malloc(L, readonly ? sizeof(char**)+sizeof(TString) : (l+1)*sizeof(char)+sizeof(TString))); + ts = cast(TString *, luaM_malloc(L, sizeof(TString) + (readonly ? sizeof(char**) : (l+1)*sizeof(char)))); ts->tsv.len = l; ts->tsv.hash = h; ts->tsv.marked = luaC_white(G(L)); @@ -71,7 +71,7 @@ static TString *newlstr (lua_State *L, const char *str, size_t l, ((char *)(ts+1))[l] = '\0'; /* ending 0 */ } else { *(char **)(ts+1) = (char *)str; - luaS_readonly(ts); + l_setbit((ts)->tsv.marked, READONLYBIT); } h = lmod(h, tb->size); ts->tsv.next = tb->hash[h]; /* chain new entry */ @@ -80,14 +80,29 @@ static TString *newlstr (lua_State *L, const char *str, size_t l, return ts; } +#include "compiler.h" +static int lua_is_ptr_in_ro_area(const char *p) { +#ifdef LUA_CROSS_COMPILER + return 0; +#else + return p >= RODATA_START_ADDRESS && p <= RODATA_END_ADDRESS; +#endif +} -static TString *luaS_newlstr_helper (lua_State *L, const char *str, size_t l, int readonly) { +/* + * The string algorithm has been modified to be LFS-friendly. The previous eLua + * algo used the address of the string was in flash and the string was >4 bytes + * This creates miminal savings and prevents the use of LFS based strings + */ + +LUAI_FUNC TString *luaS_newlstr (lua_State *L, const char *str, size_t l) { GCObject *o; unsigned int h = cast(unsigned int, l); /* seed */ size_t step = (l>>5)+1; /* if string is too long, don't hash all its chars */ size_t l1; for (l1=l; l1>=step; l1-=step) /* compute hash */ h = h ^ ((h<<5)+(h>>2)+cast(unsigned char, str[l1-1])); + for (o = G(L)->strt.hash[lmod(h, G(L)->strt.size)]; o != NULL; o = o->gch.next) { @@ -98,35 +113,27 @@ static TString *luaS_newlstr_helper (lua_State *L, const char *str, size_t l, in return ts; } } - return newlstr(L, str, l, h, readonly); /* not found */ -} - -static int lua_is_ptr_in_ro_area(const char *p) { -#ifdef LUA_CROSS_COMPILER - return 0; -#else - -#include "compiler.h" - - return p >= RODATA_START_ADDRESS && p <= RODATA_END_ADDRESS; +#if defined(LUA_FLASH_STORE) && !defined(LUA_CROSS_COMPILER) + /* + * The RAM strt is searched first since RAM access is faster tham Flash access. + * If a miss, then search the RO string table. + */ + if (G(L)->ROstrt.hash) { + for (o = G(L)->ROstrt.hash[lmod(h, G(L)->ROstrt.size)]; + o != NULL; + o = o->gch.next) { + TString *ts = rawgco2ts(o); + if (ts->tsv.len == l && (memcmp(str, getstr(ts), l) == 0)) { + return ts; + } + } + } #endif -} - -TString *luaS_newlstr (lua_State *L, const char *str, size_t l) { - // If the pointer is in a read-only memory and the string is at least 4 chars in length, - // create it as a read-only string instead - if(lua_is_ptr_in_ro_area(str) && l+1 > sizeof(char**) && l == c_strlen(str)) - return luaS_newlstr_helper(L, str, l, LUAS_READONLY_STRING); - else - return luaS_newlstr_helper(L, str, l, LUAS_REGULAR_STRING); -} - - -LUAI_FUNC TString *luaS_newrolstr (lua_State *L, const char *str, size_t l) { - if(l+1 > sizeof(char**) && l == c_strlen(str)) - return luaS_newlstr_helper(L, str, l, LUAS_READONLY_STRING); - else // no point in creating a RO string, as it would actually be larger - return luaS_newlstr_helper(L, str, l, LUAS_REGULAR_STRING); + /* New additions to the RAM strt are tagged as readonly if the string address + * is in the CTEXT segment (target only, not luac.cross) */ + int readonly = (lua_is_ptr_in_ro_area(str) && l+1 > sizeof(char**) && + l == c_strlen(str) ? LUAS_READONLY_STRING : LUAS_REGULAR_STRING); + return newlstr(L, str, l, h, readonly); /* not found */ } diff --git a/app/lua/lstring.h b/app/lua/lstring.h index b4cc3dda0e..6c5d8f6665 100644 --- a/app/lua/lstring.h +++ b/app/lua/lstring.h @@ -13,22 +13,16 @@ #include "lstate.h" -#define sizestring(s) (sizeof(union TString)+(luaS_isreadonly(s) ? sizeof(char **) : ((s)->len+1)*sizeof(char))) +#define sizestring(s) (sizeof(union TString)+(testbit((s)->marked, READONLYBIT) ? sizeof(char **) : ((s)->len+1)*sizeof(char))) #define sizeudata(u) (sizeof(union Udata)+(u)->len) #define luaS_new(L, s) (luaS_newlstr(L, s, c_strlen(s))) -#define luaS_newro(L, s) (luaS_newrolstr(L, s, c_strlen(s))) #define luaS_newliteral(L, s) (luaS_newlstr(L, "" s, \ (sizeof(s)/sizeof(char))-1)) -#define luaS_fix(s) l_setbit((s)->tsv.marked, FIXEDBIT) -#define luaS_readonly(s) l_setbit((s)->tsv.marked, READONLYBIT) -#define luaS_isreadonly(s) testbit((s)->marked, READONLYBIT) - LUAI_FUNC void luaS_resize (lua_State *L, int newsize); LUAI_FUNC Udata *luaS_newudata (lua_State *L, size_t s, Table *e); LUAI_FUNC TString *luaS_newlstr (lua_State *L, const char *str, size_t l); -LUAI_FUNC TString *luaS_newrolstr (lua_State *L, const char *str, size_t l); #endif diff --git a/app/lua/ltable.c b/app/lua/ltable.c index 2e486512ed..73f64a0a4d 100644 --- a/app/lua/ltable.c +++ b/app/lua/ltable.c @@ -445,7 +445,8 @@ static void resize (lua_State *L, Table *t, int nasize, int nhsize) { int oldasize = t->sizearray; if (nasize > oldasize) /* array part must grow? */ setarrayvector(L, t, nasize); - resize_hashpart(L, t, nhsize); + if (t->node != dummynode || nhsize>0) + resize_hashpart(L, t, nhsize); if (nasize < oldasize) { /* array part must shrink? */ t->sizearray = nasize; /* re-insert elements from vanishing slice */ @@ -749,12 +750,12 @@ int luaH_getn_ro (void *t) { return len; } -#if defined(LUA_DEBUG) +int luaH_isdummy (Node *n) { return n == dummynode; } +#if defined(LUA_DEBUG) Node *luaH_mainposition (const Table *t, const TValue *key) { return mainposition(t, key); } +#endif -int luaH_isdummy (Node *n) { return n == dummynode; } -#endif diff --git a/app/lua/ltable.h b/app/lua/ltable.h index 4835f2c3fb..d8c26d9af8 100644 --- a/app/lua/ltable.h +++ b/app/lua/ltable.h @@ -34,11 +34,9 @@ LUAI_FUNC int luaH_next (lua_State *L, Table *t, StkId key); LUAI_FUNC int luaH_next_ro (lua_State *L, void *t, StkId key); LUAI_FUNC int luaH_getn (Table *t); LUAI_FUNC int luaH_getn_ro (void *t); +LUAI_FUNC int luaH_isdummy (Node *n); #if defined(LUA_DEBUG) LUAI_FUNC Node *luaH_mainposition (const Table *t, const TValue *key); -LUAI_FUNC int luaH_isdummy (Node *n); #endif - - #endif diff --git a/app/lua/ltablib.c b/app/lua/ltablib.c index 8b0a810b3b..30eff7dfea 100644 --- a/app/lua/ltablib.c +++ b/app/lua/ltablib.c @@ -137,7 +137,7 @@ static void addfield (lua_State *L, luaL_Buffer *b, int i) { if (!lua_isstring(L, -1)) luaL_error(L, "invalid value (%s) at index %d in table for " LUA_QL("concat"), luaL_typename(L, -1), i); - luaL_addvalue(b); + luaL_addvalue(b); } diff --git a/app/lua/ltm.c b/app/lua/ltm.c index fccbbe6727..6b27ccc0d1 100644 --- a/app/lua/ltm.c +++ b/app/lua/ltm.c @@ -14,6 +14,7 @@ #include "lobject.h" #include "lstate.h" +#include "lgc.h" #include "lstring.h" #include "ltable.h" #include "ltm.h" @@ -39,7 +40,7 @@ void luaT_init (lua_State *L) { int i; for (i=0; itmname[i] = luaS_new(L, luaT_eventname[i]); - luaS_fix(G(L)->tmname[i]); /* never collect these names */ + stringfix(G(L)->tmname[i]); /* never collect these names */ } } diff --git a/app/lua/lua.c b/app/lua/lua.c index 3082bb288b..6071a9f4ff 100644 --- a/app/lua/lua.c +++ b/app/lua/lua.c @@ -13,6 +13,7 @@ #include "user_version.h" #include "driver/readline.h" #include "driver/uart.h" +#include "platform.h" #define lua_c @@ -21,54 +22,16 @@ #include "lauxlib.h" #include "lualib.h" #include "legc.h" - +#ifdef LUA_FLASH_STORE +#include "lflash.h" +#endif #include "os_type.h" lua_State *globalL = NULL; -lua_Load gLoad; - +static lua_Load gLoad; static const char *progname = LUA_PROGNAME; -#if 0 -static void lstop (lua_State *L, lua_Debug *ar) { - (void)ar; /* unused arg. */ - lua_sethook(L, NULL, 0, 0); - luaL_error(L, "interrupted!"); -} - - -static void laction (int i) { - // signal(i, SIG_DFL); - /* if another SIGINT happens before lstop, - terminate process (default action) */ - lua_sethook(globalL, lstop, LUA_MASKCALL | LUA_MASKRET | LUA_MASKCOUNT, 1); -} - - -static void print_usage (void) { -#if defined(LUA_USE_STDIO) - c_fprintf(c_stderr, -#else - luai_writestringerror( -#endif - "usage: %s [options] [script [args]].\n" - "Available options are:\n" - " -e stat execute string " LUA_QL("stat") "\n" - " -l name require library " LUA_QL("name") "\n" - " -m limit set memory limit. (units are in Kbytes)\n" - " -i enter interactive mode after executing " LUA_QL("script") "\n" - " -v show version information\n" - " -- stop handling options\n" - " - execute stdin and stop handling options\n" - , - progname); -#if defined(LUA_USE_STDIO) - c_fflush(c_stderr); -#endif -} -#endif - static void l_message (const char *pname, const char *msg) { #if defined(LUA_USE_STDIO) if (pname) c_fprintf(c_stderr, "%s: ", pname); @@ -154,17 +117,11 @@ static int getargs (lua_State *L, char **argv, int n) { return narg; } -#if 0 -static int dofile (lua_State *L, const char *name) { - int status = luaL_loadfile(L, name) || docall(L, 0, 1); - return report(L, status); -} -#else static int dofsfile (lua_State *L, const char *name) { int status = luaL_loadfsfile(L, name) || docall(L, 0, 1); return report(L, status); } -#endif + static int dostring (lua_State *L, const char *s, const char *name) { int status = luaL_loadbuffer(L, s, c_strlen(s), name) || docall(L, 0, 1); @@ -201,92 +158,6 @@ static int incomplete (lua_State *L, int status) { return 0; /* else... */ } -#if 0 -static int pushline (lua_State *L, int firstline) { - char buffer[LUA_MAXINPUT]; - char *b = buffer; - size_t l; - const char *prmt = get_prompt(L, firstline); - if (lua_readline(L, b, prmt) == 0) - return 0; /* no input */ - l = c_strlen(b); - if (l > 0 && b[l-1] == '\n') /* line ends with newline? */ - b[l-1] = '\0'; /* remove it */ - if (firstline && b[0] == '=') /* first line starts with `=' ? */ - lua_pushfstring(L, "return %s", b+1); /* change it to `return' */ - else - lua_pushstring(L, b); - lua_freeline(L, b); - return 1; -} - - -static int loadline (lua_State *L) { - int status; - lua_settop(L, 0); - if (!pushline(L, 1)) - return -1; /* no input */ - for (;;) { /* repeat until gets a complete line */ - status = luaL_loadbuffer(L, lua_tostring(L, 1), lua_strlen(L, 1), "=stdin"); - if (!incomplete(L, status)) break; /* cannot try to add lines? */ - if (!pushline(L, 0)) /* no more input? */ - return -1; - lua_pushliteral(L, "\n"); /* add a new line... */ - lua_insert(L, -2); /* ...between the two lines */ - lua_concat(L, 3); /* join them */ - } - lua_saveline(L, 1); - lua_remove(L, 1); /* remove line */ - return status; -} - - -static void dotty (lua_State *L) { - int status; - const char *oldprogname = progname; - progname = NULL; - while ((status = loadline(L)) != -1) { - if (status == 0) status = docall(L, 0, 0); - report(L, status); - if (status == 0 && lua_gettop(L) > 0) { /* any result to print? */ - lua_getglobal(L, "print"); - lua_insert(L, 1); - if (lua_pcall(L, lua_gettop(L)-1, 0, 0) != 0) - l_message(progname, lua_pushfstring(L, - "error calling " LUA_QL("print") " (%s)", - lua_tostring(L, -1))); - } - } - lua_settop(L, 0); /* clear stack */ - -#if defined(LUA_USE_STDIO) - c_fputs("\n", c_stdout); - c_fflush(c_stdout); -#else - luai_writeline(); -#endif - - progname = oldprogname; -} - - -static int handle_script (lua_State *L, char **argv, int n) { - int status; - const char *fname; - int narg = getargs(L, argv, n); /* collect arguments */ - lua_setglobal(L, "arg"); - fname = argv[n]; - if (c_strcmp(fname, "-") == 0 && c_strcmp(argv[n-1], "--") != 0) - fname = NULL; /* stdin */ - status = luaL_loadfile(L, fname); - lua_insert(L, -(narg+1)); - if (status == 0) - status = docall(L, narg, 0); - else - lua_pop(L, narg); - return report(L, status); -} -#endif /* check that argument has no extra characters at the end */ #define notail(x) {if ((x)[2] != '\0') return -1;} @@ -364,17 +235,16 @@ static int runargs (lua_State *L, char **argv, int n) { } +#ifndef LUA_INIT_STRING +#define LUA_INIT_STRING "@init.lua" +#endif + static int handle_luainit (lua_State *L) { - const char *init = c_getenv(LUA_INIT); - if (init == NULL) return 0; /* status OK */ - else if (init[0] == '@') -#if 0 - return dofile(L, init+1); -#else + const char *init = LUA_INIT_STRING; + if (init[0] == '@') return dofsfile(L, init+1); -#endif else - return dostring(L, init, "=" LUA_INIT); + return dostring(L, init, LUA_INIT); } @@ -397,40 +267,18 @@ static int pmain (lua_State *L) { lua_gc(L, LUA_GCRESTART, 0); print_version(L); s->status = handle_luainit(L); -#if 0 - if (s->status != 0) return 0; -#endif script = collectargs(argv, &has_i, &has_v, &has_e); if (script < 0) { /* invalid args? */ -#if 0 - print_usage(); -#endif s->status = 1; return 0; } - // if (has_v) print_version(); s->status = runargs(L, argv, (script > 0) ? script : s->argc); if (s->status != 0) return 0; -#if 0 - if (script) - s->status = handle_script(L, argv, script); - if (s->status != 0) return 0; - if (has_i) - dotty(L); - else if (script == 0 && !has_e && !has_v) { - if (lua_stdin_is_tty()) { - print_version(); - dotty(L); - } - else dofile(L, NULL); /* executes stdin as a file */ - } -#endif return 0; } static void dojob(lua_Load *load); static bool readline(lua_Load *load); -char line_buffer[LUA_MAXINPUT]; #ifdef LUA_RPC int main (int argc, char **argv) { @@ -439,6 +287,13 @@ int lua_main (int argc, char **argv) { #endif int status; struct Smain s; + +#if defined(NODE_DEBUG) && defined(DEVELOPMENT_USE_GDB) && \ + defined(DEVELOPMENT_BREAK_ON_STARTUP_PIN) && DEVELOPMENT_BREAK_ON_STARTUP_PIN > 0 + platform_gpio_mode( DEVELOPMENT_BREAK_ON_STARTUP_PIN, PLATFORM_GPIO_INPUT, PLATFORM_GPIO_PULLUP ); + lua_assert(platform_gpio_read(DEVELOPMENT_BREAK_ON_STARTUP_PIN)); // Break if pin pulled low +#endif + lua_State *L = lua_open(); /* create state */ if (L == NULL) { l_message(argv[0], "cannot create state: not enough memory"); @@ -446,30 +301,44 @@ int lua_main (int argc, char **argv) { } s.argc = argc; s.argv = argv; + status = lua_cpcall(L, &pmain, &s); + report(L, status); gLoad.L = L; gLoad.firstline = 1; gLoad.done = 0; - gLoad.line = line_buffer; + gLoad.line = c_malloc(LUA_MAXINPUT); gLoad.len = LUA_MAXINPUT; gLoad.line_position = 0; gLoad.prmt = get_prompt(L, 1); dojob(&gLoad); - NODE_DBG("Heap size::%d.\n",system_get_free_heap_size()); + NODE_DBG("Heap size:%d.\n",system_get_free_heap_size()); legc_set_mode( L, EGC_ALWAYS, 4096 ); // legc_set_mode( L, EGC_ON_MEM_LIMIT, 4096 ); // lua_close(L); return (status || s.status) ? EXIT_FAILURE : EXIT_SUCCESS; } +int lua_put_line(const char *s, size_t l) { + if (s == NULL || ++l < LUA_MAXINPUT || gLoad.line_position > 0) + return 0; + c_memcpy(gLoad.line, s, l); + gLoad.line[l] = '\0'; + gLoad.line_position = l; + gLoad.done = 1; + NODE_DBG("Get command: %s\n", gLoad.line); + return 1; +} + void lua_handle_input (bool force) { - while (gLoad.L && (force || readline (&gLoad))) - { + while (gLoad.L && (force || readline (&gLoad))) { + NODE_DBG("Handle Input: first=%u, pos=%u, len=%u, actual=%u, line=%s\n", gLoad.firstline, + gLoad.line_position, gLoad.len, c_strlen(gLoad.line), gLoad.line); dojob (&gLoad); force = false; } diff --git a/app/lua/lua.h b/app/lua/lua.h index 69fcc95bb0..b6f6c413b1 100644 --- a/app/lua/lua.h +++ b/app/lua/lua.h @@ -173,7 +173,6 @@ LUA_API void (lua_pushnil) (lua_State *L); LUA_API void (lua_pushnumber) (lua_State *L, lua_Number n); LUA_API void (lua_pushinteger) (lua_State *L, lua_Integer n); LUA_API void (lua_pushlstring) (lua_State *L, const char *s, size_t l); -LUA_API void (lua_pushrolstring) (lua_State *L, const char *s, size_t l); LUA_API void (lua_pushstring) (lua_State *L, const char *s); LUA_API const char *(lua_pushvfstring) (lua_State *L, const char *fmt, va_list argp); diff --git a/app/lua/luac_cross.h b/app/lua/luac_cross.h index 2eea25ce7f..f4a4da83ed 100644 --- a/app/lua/luac_cross.h +++ b/app/lua/luac_cross.h @@ -31,6 +31,7 @@ #define c_freopen freopen #define c_getc getc #define c_getenv getenv +#define c_malloc malloc #define c_memcmp memcmp #define c_memcpy memcpy #define c_printf printf @@ -56,9 +57,10 @@ #define c_strrchr strrchr #define c_strstr strstr double c_strtod(const char *__n, char **__end_PTR); -#define c_strtoul strtoul #define c_ungetc ungetc - +#define c_strtol strtol +#define c_strtoul strtoul +#define dbg_printf printf #else #define C_HEADER_ASSERT "c_assert.h" diff --git a/app/lua/luac_cross/Makefile b/app/lua/luac_cross/Makefile new file mode 100644 index 0000000000..fe66c8b2a2 --- /dev/null +++ b/app/lua/luac_cross/Makefile @@ -0,0 +1,86 @@ +# +# This Make file is called from the core Makefile hierarchy with is a hierarchical +# make wwhich uses parent callbacks to implement inheritance. However is luac_cross +# build stands outside this and uses the host toolchain to implement a separate +# host build of the luac.cross image. +# +.NOTPARALLEL: + +CCFLAGS:= -I.. -I../../include -I../../../include -I ../../libc +LDFLAGS:= -L$(SDK_DIR)/lib -L$(SDK_DIR)/ld -lm -ldl -Wl,-Map=mapfile + +CCFLAGS += -Wall + +DEFINES += -DLUA_CROSS_COMPILER -DLUA_OPTIMIZE_MEMORY=2 + +TARGET = host + +ifeq ($(FLAVOR),debug) + CCFLAGS += -O0 -g + TARGET_LDFLAGS += -O0 -g +else + FLAVOR = release + CCFLAGS += -O2 + TARGET_LDFLAGS += -O2 +endif + +LUACSRC := luac.c lflashimg.c liolib.c loslib.c print.c +LUASRC := lapi.c lauxlib.c lbaselib.c lcode.c ldblib.c ldebug.c \ + ldo.c ldump.c lfunc.c lgc.c linit.c llex.c \ + lmathlib.c lmem.c loadlib.c lobject.c lopcodes.c lparser.c \ + lrotable.c lstate.c lstring.c lstrlib.c ltable.c ltablib.c \ + ltm.c lundump.c lvm.c lzio.c +LIBCSRC := c_stdlib.c + +# +# This relies on the files being unique on the vpath +# +SRC := $(LUACSRC) $(LUASRC) $(LIBCSRC) +vpath %.c .:..:../../libc + + +ODIR := .output/$(TARGET)/$(FLAVOR)/obj + +OBJS := $(SRC:%.c=$(ODIR)/%.o) +DEPS := $(SRC:%.c=$(ODIR)/%.d) + +CFLAGS = $(CCFLAGS) $(DEFINES) $(EXTRA_CCFLAGS) $(STD_CFLAGS) $(INCLUDES) +DFLAGS = $(CCFLAGS) $(DDEFINES) $(EXTRA_CCFLAGS) $(STD_CFLAGS) $(INCLUDES) + +CC := gcc + +ECHO := echo + +IMAGE := ../../../luac.cross + +.PHONY: test clean all + +all: $(DEPS) $(IMAGE) + +$(IMAGE) : $(OBJS) + $(CC) $(OBJS) -o $@ $(LDFLAGS) + +test : + @echo CC: $(CC) + @echo SRC: $(SRC) + @echo OBJS: $(OBJS) + @echo DEPS: $(DEPS) + +clean : + $(RM) -r $(ODIR) + +ifneq ($(MAKECMDGOALS),clean) +-include $(DEPS) +endif + +$(ODIR)/%.o: %.c + @mkdir -p $(ODIR); + $(CC) $(if $(findstring $<,$(DSRCS)),$(DFLAGS),$(CFLAGS)) $(COPTS_$(*F)) -o $@ -c $< + +$(ODIR)/%.d: %.c + @mkdir -p $(ODIR); + @echo DEPEND: $(CC) -M $(CFLAGS) $< + @set -e; rm -f $@; \ + $(CC) -M $(CFLAGS) $< > $@.$$$$; \ + sed 's,\($*\.o\)[ :]*,$(ODIR)/\1 $@ : ,g' < $@.$$$$ > $@; \ + rm -f $@.$$$$ diff --git a/app/lua/luac_cross/lflashimg.c b/app/lua/luac_cross/lflashimg.c new file mode 100644 index 0000000000..9b47d4d1b2 --- /dev/null +++ b/app/lua/luac_cross/lflashimg.c @@ -0,0 +1,421 @@ +/* +** lflashimg.c +** Dump a compiled Proto hiearchy to a RO (FLash) image file +** See Copyright Notice in lua.h +*/ + +#define LUAC_CROSS_FILE + +#include "luac_cross.h" +#include C_HEADER_CTYPE +#include C_HEADER_STDIO +#include C_HEADER_STDLIB +#include C_HEADER_STRING + +#define lflashimg_c +#define LUA_CORE +#include "lobject.h" +#include "lstring.h" +#undef LUA_FLASH_STORE +#define LUA_FLASH_STORE +#include "lflash.h" + +//#define LOCAL_DEBUG + +#if INT_MAX != 2147483647 +# error "luac.cross requires C toolchain with 4 byte word size" +#endif +#define WORDSIZE ((int) sizeof(int)) +#define ALIGN(s) (((s)+(WORDSIZE-1)) & (-(signed) WORDSIZE)) +#define WORDSHIFT 2 +typedef unsigned int uint; +#define FLASH_WORDS(t) (sizeof(t)/sizeof(FlashAddr)) +/* + * + * This dumper is a variant of the standard ldump, in that instead of producing a + * binary loader format that lundump can load, it produces an image file that can be + * directly mapped or copied into addressable memory. The typical application is on + * small memory IoT devices which support programmable flash storage such as the + * ESP8266. A 64 Kb LFS image has 16Kb words and will enable all program-related + * storage to be accessed directly from flash, leaving the RAM for true R/W + * application data. + * + * The start address of the Lua Flash Store (LFS) is build-dependent, and the cross + * compiler '-a' option allows the developer to fix the LFS at a defined flash memory + * address. Alternatively and by default the cross compilation adopts a position + * independent image format, which permits the on-device image loader to load the LFS + * image at an appropriate base within the flash address space. As all objects in the + * LFS can be treated as multiples of 4-byte words, also all address fields are both + * word aligned, and any address references within the LFS are also word-aligned, + * such addresses are stored in a special format, where each PI address is two + * 16-bit unsigned offsets: + * + * Bits 0-15 is the offset into the LFS that this address refers to + * Bits 16-31 is the offset linking to the PIC next address. + * + * Hence the LFS can be up to 256Kb in length and the flash loader can use the forward + * links to chain down PI address from the mainProto address at offet 3 to all image + * addresses during load and convert them to the corresponding correct absolute memory + * addresses. This reloation process is skipped for absolute addressed images (which + * are identified by the FLASH_SIG_ABSOLUTE bit setting in the flash signature. + * + * The flash image has a standard header detailed in lflash.h + * + * Note that luac.cross may be compiled on any little-endian machine with 32 or 64 bit + * word length so Flash addresses can't be handled as standard C pointers as size_t + * and int may not have the same size. Hence addresses with the must be declared as + * the FlashAddr type rather than typed C pointers and must be accessed through macros. + * + * ALso note that image built with a given LUA_PACK_TVALUES / LUA_NUNBER_INTEGRAL + * combination must be loaded into a corresponding firmware build. Hence these + * configuration options are also included in the FLash Signature. + * + * The Flash image is assembled up by first building the RO stringtable containing + * all strings used in the compiled proto hierarchy. This is followed by the Protos. + * + * The storage is allocated bottom up using a serial allocator and the algortihm for + * building the image essentially does a bottom-uo serial enumeration so that any + * referenced storage has already been allocated in the image, and therefore (with the + * exception of the Flash Header) all pointer references are backwards. + * + * As addresses are 4 byte on the target and either 4 or (typically) 8 bytes on the + * host so any structures containing address fields (TStrings, TValues, Protos, other + * address vectors) need repacking. + */ + +typedef struct flashts { /* This is the fixed 32-bit equivalent of TString */ + FlashAddr next; + lu_byte tt; + lu_byte marked; + int hash; + int len; +} FlashTS; + +#ifndef LUA_MAX_FLASH_SIZE +#define LUA_MAX_FLASH_SIZE 0x10000 //in words +#endif + +static uint curOffset = 0; +static uint flashImage[LUA_MAX_FLASH_SIZE]; +static unsigned char flashAddrTag[LUA_MAX_FLASH_SIZE/WORDSIZE]; + +#define fatal luac_fatal +extern void __attribute__((noreturn)) luac_fatal(const char* message); + +#ifdef LOCAL_DEBUG +#define DBG_PRINT(...) printf(__VA_ARGS__) +#else +#define DBG_PRINT(...) ((void)0) +#endif + +/* + * Serial allocator. Throw a luac-style out of memory error is allocaiton fails. + */ +static void *flashAlloc(lua_State* L, size_t n) { + void *p = (void *)(flashImage + curOffset); + curOffset += ALIGN(n)>>WORDSHIFT; + if (curOffset > LUA_MAX_FLASH_SIZE) { + fatal("Out of Flash memmory"); + } + return p; +} + +/* + * Convert an absolute address pointing inside the flash image to offset form. + * This macro form also takes the lvalue destination so that this can be tagged + * as a relocatable address. + */ +#define toFlashAddr(l, pd, s) _toFlashAddr(l, &(pd), s) +static void _toFlashAddr(lua_State* L, FlashAddr *a, void *p) { + uint doffset = cast(char *, a) - cast(char *,flashImage); + lua_assert(!(doffset & (WORDSIZE-1))); + doffset >>= WORDSHIFT; + lua_assert(doffset <= curOffset); + if (p) { + uint poffset = cast(char *, p) - cast(char *,flashImage); + lua_assert(!(poffset & (WORDSIZE-1))); + poffset >>= WORDSHIFT; + lua_assert(poffset <= curOffset); + flashImage[doffset] = poffset; // Set the pointer to the offset + flashAddrTag[doffset] = 1; // And tag as an address + } else { // Special case for NULL pointer + flashImage[doffset] = 0; + } +} + +/* + * Convert an image address in offset form back to (host) absolute form + */ +static void *fromFashAddr(FlashAddr a) { + return a ? cast(void *, flashImage + a) : NULL; +} + + +/* + * Add a TS found in the Proto Load to the table at the ToS + */ +static void addTS(lua_State *L, TString *ts) { + lua_assert(ts->tsv.tt==LUA_TSTRING); + lua_pushnil(L); + setsvalue(L, L->top-1, ts); + lua_pushinteger(L, 1); + lua_rawset(L, -3); + DBG_PRINT("Adding string: %s\n",getstr(ts)); +} + + +/* + * Enumerate all of the Protos in the Proto hiearchy and scan contents to collect + * all referenced strings in a Lua Array at ToS. + */ +static void scanProtoStrings(lua_State *L, const Proto* f) { + /* Table at L->Top[-1] is used to collect the strings */ + int i; + + if (f->source) + addTS(L, f->source); + +#ifdef LUA_OPTIMIZE_DEBUG + if (f->packedlineinfo) + addTS(L, luaS_new(L, cast(const char *, f->packedlineinfo))); +#endif + + for (i = 0; i < f->sizek; i++) { + if (ttisstring(f->k + i)) + addTS(L, rawtsvalue(f->k + i)); + } + for (i = 0; i < f->sizeupvalues; i++) addTS(L, f->upvalues[i]); + for (i = 0; i < f->sizelocvars; i++) addTS(L, f->locvars[i].varname); + for (i = 0; i < f->sizep; i++) scanProtoStrings(L, f->p[i]); +} + + +/* + * Use the collected strings table to build the new ROstrt in the Flash Image + * + * The input is an array of {"SomeString" = 1, ...} on the ToS. + * The output is an array of {"SomeString" = FlashOffset("SomeString"), ...} on ToS + */ +static void createROstrt(lua_State *L, FlashHeader *fh) { + + /* Table at L->Top[-1] on input is hash used to collect the strings */ + /* Count the number of strings. Can't use objlen as this is a hash */ + fh->nROuse = 0; + lua_pushnil(L); /* first key */ + while (lua_next(L, -2) != 0) { + fh->nROuse++; + DBG_PRINT("Found: %s\n",getstr(rawtsvalue(L->top-2))); + lua_pop(L, 1); // dump the value + } + fh->nROsize = 2<nROuse); + FlashAddr *hashTab = flashAlloc(L, fh->nROsize * WORDSIZE); + toFlashAddr(L, fh->pROhash, hashTab); + + /* Now iterate over the strings to be added to the RO string table and build it */ + lua_newtable(L); // add output table + lua_pushnil(L); // First key + while (lua_next(L, -3) != 0) { // replaces key, pushes value + TString *ts = rawtsvalue(L->top - 2); // key.ts + const char *p = getstr(ts); // C string of key + uint hash = ts->tsv.hash; // hash of key + size_t len = ts->tsv.len; // and length + + DBG_PRINT("2nd pass: %s\n",p); + + FlashAddr *e = hashTab + lmod(hash, fh->nROsize); + FlashTS *last = cast(FlashTS *, fromFashAddr(*e)); + FlashTS *fts = cast(FlashTS *, flashAlloc(L, sizeof(FlashTS))); + toFlashAddr(L, *e, fts); // add reference to TS to lookup vector + toFlashAddr(L, fts->next, last); // and chain to previous entry if any + fts->tt = LUA_TSTRING; // Set as String + fts->marked = bitmask(LFSBIT); // LFS string with no Whitebits set + fts->hash = hash; // add hash + fts->len = len; // and length + memcpy(flashAlloc(L, ALIGN(len+1)), p, ALIGN(len+1)); // copy string + // include the trailing null char + lua_pop(L, 1); // Junk the value + lua_pushvalue(L, -1); // Dup the key as rawset dumps its copy + lua_pushinteger(L, cast(FlashAddr*,fts)-flashImage); // Value is new TS offset. + lua_rawset(L, -4); // Add to new table + } + /* At this point the old hash is done to derefence for GC */ + lua_remove(L, -2); +} + +/* + * Convert a TString reference in the host G(L)->strt entry into the corresponding + * TString address in the flashImage using the lookup table at ToS + */ +static void *resolveTString(lua_State* L, TString *s) { + if (!s) + return NULL; + lua_pushnil(L); + setsvalue(L, L->top-1, s); + lua_rawget(L, -2); + lua_assert(!lua_isnil(L, -1)); + void *ts = fromFashAddr(lua_tointeger(L, -1)); + lua_pop(L, 1); + return ts; +} + +/* + * In order to simplify repacking of structures from the host format to that target + * format, this simple copy routine is data-driven by a simple format specifier. + * n Number of consecutive records to be processed + * fmt A string of A, I, S, V specifiers spanning the record. + * src Source of record + * returns Address of destination record + */ +#if defined(LUA_PACK_TVALUES) +#define TARGET_TV_SIZE (sizeof(lua_Number)+sizeof(lu_int32)) +#else +#define TARGET_TV_SIZE (2*sizeof(lua_Number)) +#endif + +static void *flashCopy(lua_State* L, int n, const char *fmt, void *src) { + /* ToS is the string address mapping table */ + if (n == 0) + return NULL; + int i, recsize; + void *newts; + /* A bit of a botch because fmt is either "V" or a string of WORDSIZE specifiers */ + /* The size 8 / 12 / 16 bytes for integer builds, packed TV and default TVs resp */ + + if (fmt[0]=='V') { + lua_assert(fmt[1] == 0); /* V formats must be singetons */ + recsize = TARGET_TV_SIZE; + } else { + recsize = WORDSIZE * strlen(fmt); + } + + uint *d = cast(uint *, flashAlloc(L, n * recsize)); + uint *dest = d; + uint *s = cast(uint *, src); + + for (i = 0; i < n; i++) { + const char *p = fmt; + while (*p) { + /* All input address types (A,S,V) are aligned to size_t boundaries */ + if (*p != 'I' && ((size_t)s)&(sizeof(size_t)-1)) + s++; + + switch (*p++) { + case 'A': + toFlashAddr(L, *d, *cast(void**, s)); + s += FLASH_WORDS(size_t); + d++; + break; + case 'I': + *d++ = *s++; + break; + case 'S': + newts = resolveTString(L, *cast(TString **, s)); + toFlashAddr(L, *d, newts); + s += FLASH_WORDS(size_t); + d++; + break; + case 'V': + /* This code has to work for both Integer and Float build variants */ + memset(d, 0, TARGET_TV_SIZE); + TValue *sv = cast(TValue *, s); + if (ttisstring(sv)) { + toFlashAddr(L, *d, resolveTString(L, rawtsvalue(sv))); + } else { /* non-collectable types all of size lua_Number */ + lua_assert(!iscollectable(sv)); + *cast(lua_Number*,d) = *cast(lua_Number*,s); + } + *cast(int *,cast(lua_Number*,d)+1) = ttype(sv); + s += FLASH_WORDS(TValue); + d += TARGET_TV_SIZE/WORDSIZE; + break; + default: + lua_assert (0); + } + } + } + return dest; +} + +/* The debug optimised version has a different Proto layout */ +#ifdef LUA_OPTIMIZE_DEBUG +#define PROTO_COPY_MASK "AIAAAAAASIIIIIIIAI" +#else +#define PROTO_COPY_MASK "AIAAAAAASIIIIIIIIAI" +#endif + +/* + * Do the actual prototype copy. + */ +static void *functionToFlash(lua_State* L, const Proto* orig) { + Proto f; + int i; + + memcpy (&f, orig, sizeof(Proto)); + f.gclist = NULL; + f.next = NULL; + l_setbit(f.marked, LFSBIT); /* OK to set the LFSBIT on a stack-cloned copy */ + + if (f.sizep) { /* clone included Protos */ + Proto **p = luaM_newvector(L, f.sizep, Proto *); + for (i=0; i= 0; i--) { + if (flashAddrTag[i]) { + lua_assert(flashImage[i]mainProto, functionToFlash(L, main)); + fh->flash_sig = FLASH_SIG + (address ? FLASH_SIG_ABSOLUTE : 0); + fh->flash_size = curOffset*WORDSIZE; + linkAddresses(address); + lua_unlock(L); + int status = w(L, flashImage, curOffset * sizeof(uint), data); + lua_lock(L); + return status; +} diff --git a/app/lua/liolib.c b/app/lua/luac_cross/liolib.c similarity index 52% rename from app/lua/liolib.c rename to app/lua/luac_cross/liolib.c index 4ac31fe448..e333a557a8 100644 --- a/app/lua/liolib.c +++ b/app/lua/luac_cross/liolib.c @@ -5,14 +5,14 @@ */ -// #include "c_errno.h" -#include "c_stdio.h" -#include "c_stdlib.h" -#include "c_string.h" -#include "vfs.h" +#include +#include +#include +#include #define liolib_c #define LUA_LIB +#define LUA_OPTIMIZE_MEMORY 2 #include "lua.h" @@ -20,26 +20,21 @@ #include "lualib.h" #include "lrotable.h" - #define IO_INPUT 1 #define IO_OUTPUT 2 #define IO_STDERR 0 -#if LUA_OPTIMIZE_MEMORY != 2 -#define LUA_IO_GETFIELD(f) lua_rawgeti(L, LUA_ENVIRONINDEX, f) -#define LUA_IO_SETFIELD(f) lua_rawseti(L, LUA_ENVIRONINDEX, f) -#else -#define LUA_IO_GETFIELD(f) lua_rawgeti(L, LUA_REGISTRYINDEX, liolib_keys[f]) -#define LUA_IO_SETFIELD(f) lua_rawseti(L, LUA_REGISTRYINDEX, liolib_keys[f]) +#define LUA_IO_GETFIELD(f) lua_rawgeti(L, LUA_REGISTRYINDEX,(int)(liolib_keys[f])) +#define LUA_IO_SETFIELD(f) lua_rawseti(L, LUA_REGISTRYINDEX,(int)(liolib_keys[f])) /* "Pseudo-random" keys for the registry */ -static const int liolib_keys[] = {(int)&luaL_callmeta, (int)&luaL_typerror, (int)&luaL_argerror}; -#endif +static const size_t liolib_keys[] = {(size_t)&luaL_callmeta, (size_t)&luaL_typerror, (size_t)&luaL_argerror}; static const char *const fnames[] = {"input", "output"}; + static int pushresult (lua_State *L, int i, const char *filename) { - int en = vfs_ferrno(0); /* calls to Lua API may change this value */ + int en = errno; /* calls to Lua API may change this value */ if (i) { lua_pushboolean(L, 1); return 1; @@ -47,9 +42,9 @@ static int pushresult (lua_State *L, int i, const char *filename) { else { lua_pushnil(L); if (filename) - lua_pushfstring(L, "%s: err(%d)", filename, en); + lua_pushfstring(L, "%s: %s", filename, strerror(en)); else - lua_pushfstring(L, "err(%d)", en); + lua_pushfstring(L, "%s", strerror(en)); lua_pushinteger(L, en); return 3; } @@ -57,12 +52,12 @@ static int pushresult (lua_State *L, int i, const char *filename) { static void fileerror (lua_State *L, int arg, const char *filename) { - lua_pushfstring(L, "%s: err(%d)", filename, vfs_ferrno(0)); + lua_pushfstring(L, "%s: %s", filename, strerror(errno)); luaL_argerror(L, arg, lua_tostring(L, -1)); } -#define tofilep(L) ((int *)luaL_checkudata(L, 1, LUA_FILEHANDLE)) +#define tofilep(L) ((FILE **)luaL_checkudata(L, 1, LUA_FILEHANDLE)) static int io_type (lua_State *L) { @@ -72,7 +67,7 @@ static int io_type (lua_State *L) { lua_getfield(L, LUA_REGISTRYINDEX, LUA_FILEHANDLE); if (ud == NULL || !lua_getmetatable(L, 1) || !lua_rawequal(L, -2, -1)) lua_pushnil(L); /* not a file */ - else if (*((int *)ud) < FS_OPEN_OK) + else if (*((FILE **)ud) == NULL) lua_pushliteral(L, "closed file"); else lua_pushliteral(L, "file"); @@ -80,9 +75,9 @@ static int io_type (lua_State *L) { } -static int tofile (lua_State *L) { - int *f = tofilep(L); - if (*f < FS_OPEN_OK) +static FILE *tofile (lua_State *L) { + FILE **f = tofilep(L); + if (*f == NULL) luaL_error(L, "attempt to use a closed file"); return *f; } @@ -94,91 +89,52 @@ static int tofile (lua_State *L) { ** before opening the actual file; so, if there is a memory error, the ** file is not left opened. */ -static int *newfile (lua_State *L) { - int *pf = (int *)lua_newuserdata(L, sizeof(int)); - *pf = FS_OPEN_OK - 1; /* file handle is currently `closed' */ +static FILE **newfile (lua_State *L) { + FILE **pf = (FILE **)lua_newuserdata(L, sizeof(FILE *)); + *pf = NULL; /* file handle is currently `closed' */ luaL_getmetatable(L, LUA_FILEHANDLE); lua_setmetatable(L, -2); return pf; } -#if LUA_OPTIMIZE_MEMORY != 2 -/* -** function to (not) close the standard files stdin, stdout, and stderr -*/ -static int io_noclose (lua_State *L) { - lua_pushnil(L); - lua_pushliteral(L, "cannot close standard file"); - return 2; -} - -#if 0 -/* -** function to close 'popen' files -*/ -static int io_pclose (lua_State *L) { - int *p = tofilep(L); - int ok = lua_pclose(L, *p); - *p = FS_OPEN_OK - 1; - return pushresult(L, ok, NULL); -} -#endif - -/* -** function to close regular files -*/ -static int io_fclose (lua_State *L) { - int *p = tofilep(L); - int ok = (vfs_close(*p) == 0); - *p = FS_OPEN_OK - 1; - return pushresult(L, ok, NULL); -} -#endif - static int aux_close (lua_State *L) { -#if LUA_OPTIMIZE_MEMORY != 2 - lua_getfenv(L, 1); - lua_getfield(L, -1, "__close"); - return (lua_tocfunction(L, -1))(L); -#else - int *p = tofilep(L); - if(*p == c_stdin || *p == c_stdout || *p == c_stderr) + FILE **p = tofilep(L); + if(*p == stdin || *p == stdout || *p == stderr) { lua_pushnil(L); lua_pushliteral(L, "cannot close standard file"); - return 2; + return 2; } - int ok = (vfs_close(*p) == 0); - *p = FS_OPEN_OK - 1; + int ok = (fclose(*p) == 0); + *p = NULL; return pushresult(L, ok, NULL); -#endif } static int io_close (lua_State *L) { if (lua_isnone(L, 1)) - LUA_IO_GETFIELD(IO_OUTPUT); + lua_rawgeti(L, LUA_ENVIRONINDEX, IO_OUTPUT); tofile(L); /* make sure argument is a file */ return aux_close(L); } static int io_gc (lua_State *L) { - int f = *tofilep(L); + FILE *f = *tofilep(L); /* ignore closed files */ - if (f != FS_OPEN_OK - 1) + if (f != NULL) aux_close(L); return 0; } static int io_tostring (lua_State *L) { - int f = *tofilep(L); - if (f == FS_OPEN_OK - 1) + FILE *f = *tofilep(L); + if (f == NULL) lua_pushliteral(L, "file (closed)"); else - lua_pushfstring(L, "file (%i)", f); + lua_pushfstring(L, "file (%p)", f); return 1; } @@ -186,42 +142,19 @@ static int io_tostring (lua_State *L) { static int io_open (lua_State *L) { const char *filename = luaL_checkstring(L, 1); const char *mode = luaL_optstring(L, 2, "r"); - int *pf = newfile(L); - *pf = vfs_open(filename, mode); - return (*pf == FS_OPEN_OK - 1) ? pushresult(L, 0, filename) : 1; + FILE **pf = newfile(L); + *pf = fopen(filename, mode); + return (*pf == NULL) ? pushresult(L, 0, filename) : 1; } -/* -** this function has a separated environment, which defines the -** correct __close for 'popen' files -*/ -#if 0 -static int io_popen (lua_State *L) { - const char *filename = luaL_checkstring(L, 1); - const char *mode = luaL_optstring(L, 2, "r"); - int *pf = newfile(L); - *pf = lua_popen(L, filename, fs_mode2flags(mode)); - return (*pf == FS_OPEN_OK - 1) ? pushresult(L, 0, filename) : 1; -} - - -static int io_tmpfile (lua_State *L) { - int *pf = newfile(L); - *pf = tmpfile(); - return (*pf == FS_OPEN_OK - 1) ? pushresult(L, 0, NULL) : 1; -} -#endif - -static int getiofile (lua_State *L, int findex) { - int *pf; - LUA_IO_GETFIELD(findex); - pf = (int *)lua_touserdata(L, -1); - if (pf == NULL || *pf == FS_OPEN_OK - 1){ - luaL_error(L, "default %s file is closed", fnames[findex - 1]); - return FS_OPEN_OK - 1; - } - return *pf; +static FILE *getiofile (lua_State *L, int findex) { + FILE *f; + lua_rawgeti(L, LUA_ENVIRONINDEX, findex); + f = *(FILE **)lua_touserdata(L, -1); + if (f == NULL) + luaL_error(L, "standard %s file is closed", fnames[findex - 1]); + return f; } @@ -229,19 +162,19 @@ static int g_iofile (lua_State *L, int f, const char *mode) { if (!lua_isnoneornil(L, 1)) { const char *filename = lua_tostring(L, 1); if (filename) { - int *pf = newfile(L); - *pf = vfs_open(filename, mode); - if (*pf == FS_OPEN_OK - 1) + FILE **pf = newfile(L); + *pf = fopen(filename, mode); + if (*pf == NULL) fileerror(L, 1, filename); } else { tofile(L); /* check that it's a valid file handle */ lua_pushvalue(L, 1); } - LUA_IO_SETFIELD(f); + lua_rawseti(L, LUA_ENVIRONINDEX, f); } /* return current value */ - LUA_IO_GETFIELD(f); + lua_rawgeti(L, LUA_ENVIRONINDEX, f); return 1; } @@ -276,14 +209,14 @@ static int f_lines (lua_State *L) { static int io_lines (lua_State *L) { if (lua_isnoneornil(L, 1)) { /* no arguments? */ /* will iterate over default input */ - LUA_IO_GETFIELD(IO_INPUT); + lua_rawgeti(L, LUA_ENVIRONINDEX, IO_INPUT); return f_lines(L); } else { const char *filename = luaL_checkstring(L, 1); - int *pf = newfile(L); - *pf = vfs_open(filename, "r"); - if (*pf == FS_OPEN_OK - 1) + FILE **pf = newfile(L); + *pf = fopen(filename, "r"); + if (*pf == NULL) fileerror(L, 1, filename); aux_lines(L, lua_gettop(L), 1); return 1; @@ -297,10 +230,10 @@ static int io_lines (lua_State *L) { ** ======================================================= */ -#if 0 -static int read_number (lua_State *L, int f) { + +static int read_number (lua_State *L, FILE *f) { lua_Number d; - if (fs_scanf(f, LUA_NUMBER_SCAN, &d) == 1) { + if (fscanf(f, LUA_NUMBER_SCAN, &d) == 1) { lua_pushnumber(L, d); return 1; } @@ -309,27 +242,27 @@ static int read_number (lua_State *L, int f) { return 0; /* read fails */ } } -#endif -static int test_eof (lua_State *L, int f) { - int c = vfs_getc(f); - vfs_ungetc(c, f); + +static int test_eof (lua_State *L, FILE *f) { + int c = getc(f); + ungetc(c, f); lua_pushlstring(L, NULL, 0); return (c != EOF); } -#if 0 -static int read_line (lua_State *L, int f) { + +static int read_line (lua_State *L, FILE *f) { luaL_Buffer b; luaL_buffinit(L, &b); for (;;) { size_t l; char *p = luaL_prepbuffer(&b); - if (fs_gets(p, LUAL_BUFFERSIZE, f) == NULL) { /* eof? */ + if (fgets(p, LUAL_BUFFERSIZE, f) == NULL) { /* eof? */ luaL_pushresult(&b); /* close buffer */ return (lua_objlen(L, -1) > 0); /* check whether read something */ } - l = c_strlen(p); + l = strlen(p); if (l == 0 || p[l-1] != '\n') luaL_addsize(&b, l); else { @@ -339,27 +272,9 @@ static int read_line (lua_State *L, int f) { } } } -#else -static int read_line (lua_State *L, int f) { - luaL_Buffer b; - luaL_buffinit(L, &b); - signed char c; - do { - c = (signed char)vfs_getc(f); - if (c==EOF) { - break; - } - if (c != '\n') { - luaL_addchar(&b, c); - } - } while (c != '\n'); - luaL_pushresult(&b); /* close buffer */ - return (lua_objlen(L, -1) > 0); /* check whether read something */ -} -#endif -static int read_chars (lua_State *L, int f, size_t n) { +static int read_chars (lua_State *L, FILE *f, size_t n) { size_t rlen; /* how much to read */ size_t nr; /* number of chars actually read */ luaL_Buffer b; @@ -368,7 +283,7 @@ static int read_chars (lua_State *L, int f, size_t n) { do { char *p = luaL_prepbuffer(&b); if (rlen > n) rlen = n; /* cannot read more than asked */ - nr = vfs_read(f, p, rlen); + nr = fread(p, sizeof(char), rlen, f); luaL_addsize(&b, nr); n -= nr; /* still have to read `n' chars */ } while (n > 0 && nr == rlen); /* until end of count or eof */ @@ -377,11 +292,11 @@ static int read_chars (lua_State *L, int f, size_t n) { } -static int g_read (lua_State *L, int f, int first) { +static int g_read (lua_State *L, FILE *f, int first) { int nargs = lua_gettop(L) - 1; int success; int n; - //vfs_clearerr(f); + clearerr(f); if (nargs == 0) { /* no arguments? */ success = read_line(L, f); n = first+1; /* to return 1 result */ @@ -398,11 +313,9 @@ static int g_read (lua_State *L, int f, int first) { const char *p = lua_tostring(L, n); luaL_argcheck(L, p && p[0] == '*', n, "invalid option"); switch (p[1]) { -#if 0 case 'n': /* number */ success = read_number(L, f); break; -#endif case 'l': /* line */ success = read_line(L, f); break; @@ -416,7 +329,7 @@ static int g_read (lua_State *L, int f, int first) { } } } - if (vfs_ferrno(f)) + if (ferror(f)) return pushresult(L, 0, NULL); if (!success) { lua_pop(L, 1); /* remove last result */ @@ -437,15 +350,13 @@ static int f_read (lua_State *L) { static int io_readline (lua_State *L) { - int *pf = (int *)lua_touserdata(L, lua_upvalueindex(1)); + FILE *f = *(FILE **)lua_touserdata(L, lua_upvalueindex(1)); int sucess; - if (pf == NULL || *pf == FS_OPEN_OK - 1){ /* file is already closed? */ + if (f == NULL) /* file is already closed? */ luaL_error(L, "file is already closed"); - return 0; - } - sucess = read_line(L, *pf); - if (vfs_ferrno(*pf)) - return luaL_error(L, "err(%d)", vfs_ferrno(*pf)); + sucess = read_line(L, f); + if (ferror(f)) + return luaL_error(L, "%s", strerror(errno)); if (sucess) return 1; else { /* EOF */ if (lua_toboolean(L, lua_upvalueindex(2))) { /* generator created file? */ @@ -460,22 +371,19 @@ static int io_readline (lua_State *L) { /* }====================================================== */ -static int g_write (lua_State *L, int f, int arg) { +static int g_write (lua_State *L, FILE *f, int arg) { int nargs = lua_gettop(L) - 1; int status = 1; for (; nargs--; arg++) { -#if 0 if (lua_type(L, arg) == LUA_TNUMBER) { /* optimization: could be done exactly as for strings */ status = status && - fs_printf(f, LUA_NUMBER_FMT, lua_tonumber(L, arg)) > 0; + fprintf(f, LUA_NUMBER_FMT, lua_tonumber(L, arg)) > 0; } - else -#endif - { + else { size_t l; const char *s = luaL_checklstring(L, arg, &l); - status = status && (vfs_write(f, s, l) == l); + status = status && (fwrite(s, sizeof(char), l, f) == l); } } return pushresult(L, status, NULL); @@ -493,159 +401,103 @@ static int f_write (lua_State *L) { static int f_seek (lua_State *L) { - static const int mode[] = {VFS_SEEK_SET, VFS_SEEK_CUR, VFS_SEEK_END}; + static const int mode[] = {SEEK_SET, SEEK_CUR, SEEK_END}; static const char *const modenames[] = {"set", "cur", "end", NULL}; - int f = tofile(L); + FILE *f = tofile(L); int op = luaL_checkoption(L, 2, "cur", modenames); long offset = luaL_optlong(L, 3, 0); - op = vfs_lseek(f, offset, mode[op]); + op = fseek(f, offset, mode[op]); if (op) return pushresult(L, 0, NULL); /* error */ else { - lua_pushinteger(L, vfs_tell(f)); + lua_pushinteger(L, ftell(f)); return 1; } } -#if 0 + static int f_setvbuf (lua_State *L) { static const int mode[] = {_IONBF, _IOFBF, _IOLBF}; static const char *const modenames[] = {"no", "full", "line", NULL}; - int f = tofile(L); + FILE *f = tofile(L); int op = luaL_checkoption(L, 2, NULL, modenames); lua_Integer sz = luaL_optinteger(L, 3, LUAL_BUFFERSIZE); int res = setvbuf(f, NULL, mode[op], sz); return pushresult(L, res == 0, NULL); } -#endif static int io_flush (lua_State *L) { - return pushresult(L, vfs_flush(getiofile(L, IO_OUTPUT)) == 0, NULL); + return pushresult(L, fflush(getiofile(L, IO_OUTPUT)) == 0, NULL); } static int f_flush (lua_State *L) { - return pushresult(L, vfs_flush(tofile(L)) == 0, NULL); + return pushresult(L, fflush(tofile(L)) == 0, NULL); } -#undef MIN_OPT_LEVEL #define MIN_OPT_LEVEL 2 #include "lrodefs.h" -#if LUA_OPTIMIZE_MEMORY == 2 const LUA_REG_TYPE iolib_funcs[] = { -#else -const LUA_REG_TYPE iolib[] = { -#endif - {LSTRKEY("close"), LFUNCVAL(io_close)}, - {LSTRKEY("flush"), LFUNCVAL(io_flush)}, - {LSTRKEY("input"), LFUNCVAL(io_input)}, - {LSTRKEY("lines"), LFUNCVAL(io_lines)}, - {LSTRKEY("open"), LFUNCVAL(io_open)}, - {LSTRKEY("output"), LFUNCVAL(io_output)}, - // {LSTRKEY("popen"), LFUNCVAL(io_popen)}, - {LSTRKEY("read"), LFUNCVAL(io_read)}, - // {LSTRKEY("tmpfile"), LFUNCVAL(io_tmpfile)}, - {LSTRKEY("type"), LFUNCVAL(io_type)}, - {LSTRKEY("write"), LFUNCVAL(io_write)}, + {LSTRKEY("close"), LFUNCVAL(io_close)}, + {LSTRKEY("flush"), LFUNCVAL(io_flush)}, + {LSTRKEY("input"), LFUNCVAL(io_input)}, + {LSTRKEY("lines"), LFUNCVAL(io_lines)}, + {LSTRKEY("open"), LFUNCVAL(io_open)}, + {LSTRKEY("output"), LFUNCVAL(io_output)}, + {LSTRKEY("read"), LFUNCVAL(io_read)}, + {LSTRKEY("type"), LFUNCVAL(io_type)}, + {LSTRKEY("write"), LFUNCVAL(io_write)}, + {LSTRKEY("__index"), LROVAL(iolib_funcs)}, {LNILKEY, LNILVAL} }; -#if LUA_OPTIMIZE_MEMORY == 2 -static int luaL_index(lua_State *L) +/* Note that IO objects use a RAM metatable created to allow extensibility */ + +static int io_index(lua_State *L) { return luaR_findfunction(L, iolib_funcs); } - + const luaL_Reg iolib[] = { - {"__index", luaL_index}, + {"__index", io_index}, {NULL, NULL} }; -#endif #undef MIN_OPT_LEVEL #define MIN_OPT_LEVEL 1 #include "lrodefs.h" const LUA_REG_TYPE flib[] = { - {LSTRKEY("close"), LFUNCVAL(io_close)}, - {LSTRKEY("flush"), LFUNCVAL(f_flush)}, - {LSTRKEY("lines"), LFUNCVAL(f_lines)}, - {LSTRKEY("read"), LFUNCVAL(f_read)}, - {LSTRKEY("seek"), LFUNCVAL(f_seek)}, - // {LSTRKEY("setvbuf"), LFUNCVAL(f_setvbuf)}, - {LSTRKEY("write"), LFUNCVAL(f_write)}, - {LSTRKEY("__gc"), LFUNCVAL(io_gc)}, + {LSTRKEY("close"), LFUNCVAL(io_close)}, + {LSTRKEY("flush"), LFUNCVAL(f_flush)}, + {LSTRKEY("lines"), LFUNCVAL(f_lines)}, + {LSTRKEY("read"), LFUNCVAL(f_read)}, + {LSTRKEY("seek"), LFUNCVAL(f_seek)}, + {LSTRKEY("setvbuf"), LFUNCVAL(f_setvbuf)}, + {LSTRKEY("write"), LFUNCVAL(f_write)}, + {LSTRKEY("__gc"), LFUNCVAL(io_gc)}, {LSTRKEY("__tostring"), LFUNCVAL(io_tostring)}, -#if LUA_OPTIMIZE_MEMORY > 0 - {LSTRKEY("__index"), LROVAL(flib)}, -#endif + {LSTRKEY("__index"), LROVAL(flib)}, {LNILKEY, LNILVAL} }; -static void createmeta (lua_State *L) { -#if LUA_OPTIMIZE_MEMORY == 0 - luaL_newmetatable(L, LUA_FILEHANDLE); /* create metatable for file handles */ - lua_pushvalue(L, -1); /* push metatable */ - lua_setfield(L, -2, "__index"); /* metatable.__index = metatable */ - luaL_register(L, NULL, flib); /* file methods */ -#else - luaL_rometatable(L, LUA_FILEHANDLE, (void*)flib); /* create metatable for file handles */ -#endif -} - -static void createstdfile (lua_State *L, int f, int k, const char *fname) { +static void createstdfile (lua_State *L, FILE *f, int k, const char *fname) { *newfile(L) = f; -#if LUA_OPTIMIZE_MEMORY != 2 - if (k > 0) { - lua_pushvalue(L, -1); - lua_rawseti(L, LUA_ENVIRONINDEX, k); - } - lua_pushvalue(L, -2); /* copy environment */ - lua_setfenv(L, -2); /* set it */ - lua_setfield(L, -3, fname); -#else + lua_pushvalue(L, -1); - lua_rawseti(L, LUA_REGISTRYINDEX, liolib_keys[k]); + lua_rawseti(L, LUA_REGISTRYINDEX, (int)(liolib_keys[k])); lua_setfield(L, -2, fname); -#endif } -#if LUA_OPTIMIZE_MEMORY != 2 -static void newfenv (lua_State *L, lua_CFunction cls) { - lua_createtable(L, 0, 1); - lua_pushcfunction(L, cls); - lua_setfield(L, -2, "__close"); -} -#endif - LUALIB_API int luaopen_io (lua_State *L) { - createmeta(L); -#if LUA_OPTIMIZE_MEMORY != 2 - /* create (private) environment (with fields IO_INPUT, IO_OUTPUT, __close) */ - newfenv(L, io_fclose); - lua_replace(L, LUA_ENVIRONINDEX); - /* open library */ - luaL_register(L, LUA_IOLIBNAME, iolib); - newfenv(L, io_noclose); /* close function for default files */ -#else + luaL_rometatable(L, LUA_FILEHANDLE, (void*)flib); /* create metatable for file handles */ luaL_register_light(L, LUA_IOLIBNAME, iolib); lua_pushvalue(L, -1); lua_setmetatable(L, -2); -#endif -#if 0 /* create (and set) default files */ - createstdfile(L, c_stdin, IO_INPUT, "stdin"); - createstdfile(L, c_stdout, IO_OUTPUT, "stdout"); - createstdfile(L, c_stderr, IO_STDERR, "stderr"); - -#if LUA_OPTIMIZE_MEMORY != 2 - lua_pop(L, 1); /* pop environment for default files */ - lua_getfield(L, -1, "popen"); - newfenv(L, io_pclose); /* create environment for 'popen' */ - lua_setfenv(L, -2); /* set fenv for 'popen' */ - lua_pop(L, 1); /* pop 'popen' */ -#endif -#endif + createstdfile(L, stdin, IO_INPUT, "stdin"); + createstdfile(L, stdout, IO_OUTPUT, "stdout"); + createstdfile(L, stderr, 0, "stderr"); return 1; } diff --git a/app/lua/luac_cross/luac.c b/app/lua/luac_cross/luac.c index bdd3e01a4f..039d9670ad 100644 --- a/app/lua/luac_cross/luac.c +++ b/app/lua/luac_cross/luac.c @@ -11,13 +11,14 @@ #include C_HEADER_STDIO #include C_HEADER_STDLIB #include C_HEADER_STRING +#include #define luac_c #define LUA_CORE #include "lua.h" #include "lauxlib.h" - +#include "lualib.h" #include "ldo.h" #include "lfunc.h" #include "lmem.h" @@ -31,17 +32,23 @@ static int listing=0; /* list bytecodes? */ static int dumping=1; /* dump bytecodes? */ -static int stripping=0; /* strip debug information? */ +static int stripping=0; /* strip debug information? */ +static int flash=0; /* output flash image */ +static lu_int32 address=0; /* output flash image at absolute location */ +static int lookup=0; /* output lookup-style master combination header */ static char Output[]={ OUTPUT }; /* default output file name */ static const char* output=Output; /* actual output file name */ +static const char* execute; /* executed a Lua file */ static const char* progname=PROGNAME; /* actual program name */ static DumpTargetInfo target; -static void fatal(const char* message) +void luac_fatal(const char* message) { fprintf(stderr,"%s: %s\n",progname,message); exit(EXIT_FAILURE); } +#define fatal(s) luac_fatal(s) + static void cannot(const char* what) { @@ -61,18 +68,21 @@ static void usage(const char* message) " - process stdin\n" " -l list\n" " -o name output to file " LUA_QL("name") " (default is \"%s\")\n" + " -e name execute a lua source file\n" + " -f output a flash image file\n" + " -a addr generate an absolute, rather than position independent flash image file\n" + " -i generate lookup combination master (default with option -f)\n" " -p parse only\n" " -s strip debug information\n" " -v show version information\n" - " -cci bits cross-compile with given integer size\n" - " -ccn type bits cross-compile with given lua_Number type and size\n" - " -cce endian cross-compile with given endianness ('big' or 'little')\n" " -- stop handling options\n", progname,Output); exit(EXIT_FAILURE); } #define IS(s) (strcmp(argv[i],s)==0) +#define IROM0_SEG 0x40210000ul +#define IROM0_SEGMAX 0x00100000ul static int doargs(int argc, char* argv[]) { @@ -89,8 +99,28 @@ static int doargs(int argc, char* argv[]) if (version) ++version; break; } - else if (IS("-")) /* end of options; use stdin */ + else if (IS("-")) /* end of options; use stdin */ break; + else if (IS("-e")) /* execute a lua source file file */ + { + execute=argv[++i]; + if (execute ==NULL || *execute==0 || *execute=='-' ) + usage(LUA_QL("-e") " needs argument"); + } + else if (IS("-f")) /* Flash image file */ + { + flash=lookup=1; + } + else if (IS("-a")) /* Absolue flash image file */ + { + flash=lookup=1; + address=strtol(argv[++i],NULL,0); + size_t offset = (unsigned) (address -IROM0_SEG); + if (offset > IROM0_SEGMAX) + usage(LUA_QL("-e") " absolute address must be valid flash address"); + } + else if (IS("-i")) /* lookup */ + lookup = 1; else if (IS("-l")) /* list */ ++listing; else if (IS("-o")) /* output file */ @@ -99,42 +129,17 @@ static int doargs(int argc, char* argv[]) if (output==NULL || *output==0) usage(LUA_QL("-o") " needs argument"); if (IS("-")) output=NULL; } + else if (IS("-p")) /* parse only */ - dumping=0; + dumping=0; else if (IS("-s")) /* strip debug information */ stripping=1; else if (IS("-v")) /* show version */ ++version; - else if (IS("-cci")) /* target integer size */ - { - int s = target.sizeof_int = atoi(argv[++i])/8; - if (!(s==1 || s==2 || s==4)) fatal(LUA_QL("-cci") " must be 8, 16 or 32"); - } - else if (IS("-ccn")) /* target lua_Number type and size */ - { - const char *type=argv[++i]; - if (strcmp(type,"int")==0) target.lua_Number_integral=1; - else if (strcmp(type,"float")==0) target.lua_Number_integral=0; - else if (strcmp(type,"float_arm")==0) - { - target.lua_Number_integral=0; - target.is_arm_fpa=1; - } - else fatal(LUA_QL("-ccn") " type must be " LUA_QL("int") " or " LUA_QL("float") " or " LUA_QL("float_arm")); - int s = target.sizeof_lua_Number = atoi(argv[++i])/8; - if (target.lua_Number_integral && !(s==1 || s==2 || s==4)) fatal(LUA_QL("-ccn") " size must be 8, 16, or 32 for int"); - if (!target.lua_Number_integral && !(s==4 || s==8)) fatal(LUA_QL("-ccn") " size must be 32 or 64 for float"); - } - else if (IS("-cce")) /* target endianness */ - { - const char *val=argv[++i]; - if (strcmp(val,"big")==0) target.little_endian=0; - else if (strcmp(val,"little")==0) target.little_endian=1; - else fatal(LUA_QL("-cce") " must be " LUA_QL("big") " or " LUA_QL("little")); - } else /* unknown option */ - usage(argv[i]); + usage(argv[i]); } + if (i==argc && (listing || !dumping)) { dumping=0; @@ -150,30 +155,99 @@ static int doargs(int argc, char* argv[]) #define toproto(L,i) (clvalue(L->top+(i))->l.p) -static const Proto* combine(lua_State* L, int n) +static TString *corename(lua_State *L, const TString *filename) { - if (n==1) + const char *fn = getstr(filename)+1; + const char *s = strrchr(fn, '/'); + s = s ? s + 1 : fn; + while (*s == '.') s++; + const char *e = strchr(s, '.'); + int l = e ? e - s: strlen(s); + return l ? luaS_newlstr (L, s, l) : luaS_new(L, fn); +} +/* + * If the luac command line includes multiple files or has the -f option + * then luac generates a main function to reference all sub-main prototypes. + * This is one of two types: + * Type 0 The standard luac combination main + * Type 1 A lookup wrapper that facilitates indexing into the generated protos + */ +static const Proto* combine(lua_State* L, int n, int type) +{ + if (n==1 && type == 0) return toproto(L,-1); else { - int i,pc; + int i; + Instruction *pc; Proto* f=luaF_newproto(L); setptvalue2s(L,L->top,f); incr_top(L); f->source=luaS_newliteral(L,"=(" PROGNAME ")"); - f->maxstacksize=1; - pc=2*n+1; - f->code=luaM_newvector(L,pc,Instruction); - f->sizecode=pc; f->p=luaM_newvector(L,n,Proto*); f->sizep=n; + for (i=0; ip[i]=toproto(L,i-n-1); pc=0; - for (i=0; ip[i]=toproto(L,i-n-1); - f->code[pc++]=CREATE_ABx(OP_CLOSURE,0,i); - f->code[pc++]=CREATE_ABC(OP_CALL,0,1,1); + + if (type == 0) { + /* + * Type 0 is as per the standard luac, which is just a main routine which + * invokes all of the compiled functions sequentially. This is fine if + * they are self registering modules, but useless otherwise. + */ + f->numparams = 0; + f->maxstacksize = 1; + f->sizecode = 2*n + 1 ; + f->sizek = 0; + f->code = luaM_newvector(L, f->sizecode , Instruction); + f->k = luaM_newvector(L,f->sizek,TValue); + + for (i=0, pc = f->code; i LFIELDS_PER_FLUSH) { +#define NO_MOD_ERR_(n) ": Number of modules > " #n +#define NO_MOD_ERR(n) NO_MOD_ERR_(n) + usage(LUA_QL("-f") NO_MOD_ERR(LFIELDS_PER_FLUSH)); + } + f->numparams = 1; + f->maxstacksize = n + 3; + f->sizecode = 5*n + 5 ; + f->sizek = n + 1; + f->sizelocvars = 0; + f->code = luaM_newvector(L, f->sizecode , Instruction); + f->k = luaM_newvector(L,f->sizek,TValue); + for (i=0, pc = f->code; ik+i,corename(L, f->p[i]->source)); + *pc++ = CREATE_ABC(OP_EQ,0,0,RKASK(i)); + *pc++ = CREATE_ABx(OP_JMP,0,MAXARG_sBx+2); + *pc++ = CREATE_ABx(OP_CLOSURE,1,i); + *pc++ = CREATE_ABC(OP_RETURN,1,2,0); + } + + setnvalue(f->k+n, (lua_Number) time(NULL)); + + *pc++ = CREATE_ABx(OP_LOADK,1,n); + *pc++ = CREATE_ABC(OP_NEWTABLE,2,luaO_int2fb(i),0); + for (i=0; icode[pc++]=CREATE_ABC(OP_RETURN,0,1,0); + lua_assert((pc-f->code) == f->sizecode); + return f; } } @@ -189,6 +263,9 @@ struct Smain { char** argv; }; +extern uint dumpToFlashImage (lua_State* L,const Proto *main, lua_Writer w, + void* data, int strip, lu_int32 address); + static int pmain(lua_State* L) { struct Smain* s = (struct Smain*)lua_touserdata(L, 1); @@ -197,19 +274,39 @@ static int pmain(lua_State* L) const Proto* f; int i; if (!lua_checkstack(L,argc)) fatal("too many input files"); + if (execute) + { + if (luaL_loadfile(L,execute)!=0) fatal(lua_tostring(L,-1)); + luaL_openlibs(L); + lua_pushstring(L, execute); + if (lua_pcall(L, 1, 1, 0)) fatal(lua_tostring(L,-1)); + if (!lua_isfunction(L, -1)) + { + lua_pop(L,1); + if(argc == 0) return 0; + execute = NULL; + } + } for (i=0; i1); if (dumping) { + int result; FILE* D= (output==NULL) ? stdout : fopen(output,"wb"); if (D==NULL) cannot("open"); lua_lock(L); - int result=luaU_dump_crosscompile(L,f,writer,D,stripping,target); + if (flash) + { + result=dumpToFlashImage(L,f,writer, D, stripping, address); + } else + { + result=luaU_dump_crosscompile(L,f,writer,D,stripping,target); + } lua_unlock(L); if (result==LUA_ERR_CC_INTOVERFLOW) fatal("value too big or small for target integer type"); if (result==LUA_ERR_CC_NOTINTEGER) fatal("target lua_Number is integral but fractional value found"); @@ -234,7 +331,7 @@ int main(int argc, char* argv[]) int i=doargs(argc,argv); argc-=i; argv+=i; - if (argc<=0) usage("no input files given"); + if (argc<=0 && execute==0) usage("no input files given"); L=lua_open(); if (L==NULL) fatal("not enough memory for state"); s.argc=argc; diff --git a/app/lua/luaconf.h b/app/lua/luaconf.h index d039b18f4f..d6c144cb6e 100644 --- a/app/lua/luaconf.h +++ b/app/lua/luaconf.h @@ -38,6 +38,11 @@ #define LUA_WIN #endif + +#if defined(LUA_CROSS_COMPILER) +#define LUA_USE_LINUX +#endif + #if defined(LUA_USE_LINUX) #define LUA_USE_POSIX #define LUA_USE_DLOPEN /* needs an extra library: -ldl */ @@ -59,7 +64,7 @@ #if defined(LUA_USE_POSIX) #define LUA_USE_MKSTEMP #define LUA_USE_ISATTY -#define LUA_USE_POPEN +//#define LUA_USE_POPEN #define LUA_USE_ULONGJMP #endif @@ -167,7 +172,7 @@ #define LUA_INTEGER ptrdiff_t #else #if !defined LUA_INTEGRAL_LONGLONG - #define LUA_INTEGER long + #define LUA_INTEGER int #else #define LUA_INTEGER long long #endif // #if !defined LUA_INTEGRAL_LONGLONG @@ -487,7 +492,7 @@ extern int readline4lua(const char *prompt, char *buffer, int length); /* 16-bit ints */ #define LUAI_UINT32 unsigned long #define LUAI_INT32 long -#define LUAI_MAXINT32 LONG_MAX +#define LUAI_MAXINT32 INT_MAX #define LUAI_UMEM unsigned long #define LUAI_MEM long #endif @@ -607,8 +612,8 @@ extern int readline4lua(const char *prompt, char *buffer, int length); */ #if defined LUA_NUMBER_INTEGRAL #if !defined LUA_INTEGRAL_LONGLONG - #define LUA_NUMBER_SCAN "%ld" - #define LUA_NUMBER_FMT "%ld" + #define LUA_NUMBER_SCAN "%d" + #define LUA_NUMBER_FMT "%d" #else #define LUA_NUMBER_SCAN "%lld" #define LUA_NUMBER_FMT "%lld" @@ -894,7 +899,7 @@ union luai_Cast { double l_d; long l_l; }; /* If you define the next macro you'll get the ability to set rotables as metatables for tables/userdata/types (but the VM might run slower) */ -#if (LUA_OPTIMIZE_MEMORY == 2) && !defined(LUA_CROSS_COMPILER) +#if (LUA_OPTIMIZE_MEMORY == 2) #define LUA_META_ROTABLES #endif diff --git a/app/lua/lundump.c b/app/lua/lundump.c index 76178c200b..0baedf2939 100644 --- a/app/lua/lundump.c +++ b/app/lua/lundump.c @@ -172,7 +172,7 @@ static TString* LoadString(LoadState* S) } else { s = (char*)luaZ_get_crt_address(S->Z); LoadBlock(S,NULL,size); - return luaS_newrolstr(S->L,s,size-1); + return luaS_newlstr(S->L,s,size-1); } } } @@ -280,7 +280,7 @@ static Proto* LoadFunction(LoadState* S, TString* p) Proto* f; if (++S->L->nCcalls > LUAI_MAXCCALLS) error(S,"code too deep"); f=luaF_newproto(S->L); - if (luaZ_direct_mode(S->Z)) proto_readonly(f); + if (luaZ_direct_mode(S->Z)) l_setbit((f)->marked, READONLYBIT); setptvalue2s(S->L,S->L->top,f); incr_top(S->L); f->source=LoadString(S); if (f->source==NULL) f->source=p; f->linedefined=LoadInt(S); diff --git a/app/lua/lvm.c b/app/lua/lvm.c index 51cf9f215e..0e256d8f62 100644 --- a/app/lua/lvm.c +++ b/app/lua/lvm.c @@ -41,7 +41,7 @@ LUA_NUMBER luai_ipow(LUA_NUMBER a, LUA_NUMBER b) { LUA_NUMBER c = 1; for (;;) { if (b & 1) - c *= a; + c *= a; b = b >> 1; if (b == 0) return c; diff --git a/app/lwip/app/dhcpserver.c b/app/lwip/app/dhcpserver.c old mode 100755 new mode 100644 diff --git a/app/modules/Makefile b/app/modules/Makefile index 316bd011e6..23d298b74e 100644 --- a/app/modules/Makefile +++ b/app/modules/Makefile @@ -50,7 +50,7 @@ INCLUDES += -I ../pcm INCLUDES += -I ../platform INCLUDES += -I ../spiffs INCLUDES += -I ../smart -INCLUDES += -I ../dhtlib +INCLUDES += -I ../dht INCLUDES += -I ../fatfs INCLUDES += -I ../http INCLUDES += -I ../sjson diff --git a/app/modules/gdbstub.c b/app/modules/gdbstub.c index 9c13861a5e..8b0c4ebdd4 100644 --- a/app/modules/gdbstub.c +++ b/app/modules/gdbstub.c @@ -2,7 +2,8 @@ * This module, when enabled with the LUA_USE_MODULES_GDBSTUB define causes * the gdbstub code to be included and enabled to handle all fatal exceptions. * This allows you to use the lx106 gdb to catch the exception and then poke - * around. You can't continue from an exception (at least not easily). + * around. You can continue from a break, but attempting to continue from an + * exception usually fails. * * This should not be included in production builds as any exception will * put the nodemcu into a state where it is waiting for serial input and it has @@ -41,8 +42,8 @@ static int lgdbstub_open(lua_State *L) { static const LUA_REG_TYPE gdbstub_map[] = { { LSTRKEY( "brk" ), LFUNCVAL( lgdbstub_break ) }, { LSTRKEY( "gdboutput" ), LFUNCVAL( lgdbstub_gdboutput) }, - + { LSTRKEY( "open" ), LFUNCVAL( lgdbstub_open) }, { LNILKEY, LNILVAL } }; -NODEMCU_MODULE(GDBSTUB, "gdbstub", gdbstub_map, lgdbstub_open); +NODEMCU_MODULE(GDBSTUB, "gdbstub", gdbstub_map, NULL); diff --git a/app/modules/linit.c b/app/modules/linit.c deleted file mode 100644 index da14b73b14..0000000000 --- a/app/modules/linit.c +++ /dev/null @@ -1,78 +0,0 @@ -/* -** $Id: linit.c,v 1.14.1.1 2007/12/27 13:02:25 roberto Exp $ -** Initialization of libraries for lua.c -** See Copyright Notice in lua.h -*/ - - -#define linit_c -#define LUA_LIB -#define LUAC_CROSS_FILE - -#include "lua.h" - -#include "lualib.h" -#include "lauxlib.h" -#include "luaconf.h" -#include "module.h" - - -BUILTIN_LIB_INIT( BASE, "", luaopen_base); -BUILTIN_LIB_INIT( LOADLIB, LUA_LOADLIBNAME, luaopen_package); - -#if defined(LUA_USE_BUILTIN_IO) -BUILTIN_LIB_INIT( IO, LUA_IOLIBNAME, luaopen_io); -#endif - -#if defined (LUA_USE_BUILTIN_STRING) -extern const luaR_entry strlib[]; -BUILTIN_LIB_INIT( STRING, LUA_STRLIBNAME, luaopen_string); -BUILTIN_LIB( STRING, LUA_STRLIBNAME, strlib); -#endif - -#if defined(LUA_USE_BUILTIN_TABLE) -extern const luaR_entry tab_funcs[]; -BUILTIN_LIB_INIT( TABLE, LUA_TABLIBNAME, luaopen_table); -BUILTIN_LIB( TABLE, LUA_TABLIBNAME, tab_funcs); -#endif - -#if defined(LUA_USE_BUILTIN_DEBUG) || defined(LUA_USE_BUILTIN_DEBUG_MINIMAL) -extern const luaR_entry dblib[]; -BUILTIN_LIB_INIT( DBG, LUA_DBLIBNAME, luaopen_debug); -BUILTIN_LIB( DBG, LUA_DBLIBNAME, dblib); -#endif - -#if defined(LUA_USE_BUILTIN_OS) -extern const luaR_entry syslib[]; -BUILTIN_LIB( OS, LUA_OSLIBNAME, syslib); -#endif - -#if defined(LUA_USE_BUILTIN_COROUTINE) -extern const luaR_entry co_funcs[]; -BUILTIN_LIB( CO, LUA_COLIBNAME, co_funcs); -#endif - -#if defined(LUA_USE_BUILTIN_MATH) -extern const luaR_entry math_map[]; -BUILTIN_LIB( MATH, LUA_MATHLIBNAME, math_map); -#endif - -#ifdef LUA_CROSS_COMPILER -const luaL_Reg lua_libs[] = {{NULL, NULL}}; -const luaR_table lua_rotable[] = {{NULL, NULL}}; -#else -extern const luaL_Reg lua_libs[]; -#endif - -void luaL_openlibs (lua_State *L) { - const luaL_Reg *lib = lua_libs; - for (; lib->name; lib++) { - if (lib->func) - { - lua_pushcfunction(L, lib->func); - lua_pushstring(L, lib->name); - lua_call(L, 1, 0); - } - } -} - diff --git a/app/modules/node.c b/app/modules/node.c index 3df88b8c15..b20502e301 100644 --- a/app/modules/node.c +++ b/app/modules/node.c @@ -17,7 +17,9 @@ #include "platform.h" #include "lrodefs.h" - +#ifdef LUA_FLASH_STORE +#include "lflash.h" +#endif #include "c_types.h" #include "c_string.h" #include "driver/uart.h" @@ -149,7 +151,6 @@ static int node_chipid( lua_State* L ) // lua_pushinteger(L, vdd33); // return 1; // } - // Lua: flashid() static int node_flashid( lua_State* L ) { @@ -178,26 +179,16 @@ static int node_heap( lua_State* L ) return 1; } -extern lua_Load gLoad; +extern int lua_put_line(const char *s, size_t l); extern bool user_process_input(bool force); + // Lua: input("string") -static int node_input( lua_State* L ) -{ +static int node_input( lua_State* L ) { size_t l = 0; const char *s = luaL_checklstring(L, 1, &l); - if (s != NULL && l > 0 && l < LUA_MAXINPUT - 1) - { - lua_Load *load = &gLoad; - if (load->line_position == 0) { - c_memcpy(load->line, s, l); - load->line[l + 1] = '\0'; - load->line_position = c_strlen(load->line) + 1; - load->done = 1; - NODE_DBG("Get command:\n"); - NODE_DBG(load->line); // buggy here - NODE_DBG("\nResult(if any):\n"); - user_process_input(true); - } + if (lua_put_line(s, l)) { + NODE_DBG("Result (if any):\n"); + user_process_input(true); } return 0; } @@ -609,6 +600,13 @@ static const LUA_REG_TYPE node_task_map[] = { static const LUA_REG_TYPE node_map[] = { + { LSTRKEY( "heap" ), LFUNCVAL( node_heap ) }, + { LSTRKEY( "info" ), LFUNCVAL( node_info ) }, + { LSTRKEY( "task" ), LROVAL( node_task_map ) }, +#ifdef LUA_FLASH_STORE + { LSTRKEY( "flashreload" ), LFUNCVAL( luaN_reload_reboot ) }, + { LSTRKEY( "flashindex" ), LFUNCVAL( luaN_index ) }, +#endif { LSTRKEY( "restart" ), LFUNCVAL( node_restart ) }, { LSTRKEY( "dsleep" ), LFUNCVAL( node_deepsleep ) }, { LSTRKEY( "dsleepMax" ), LFUNCVAL( dsleepMax ) }, @@ -616,11 +614,9 @@ static const LUA_REG_TYPE node_map[] = #ifdef PMSLEEP_ENABLE PMSLEEP_INT_MAP, #endif - { LSTRKEY( "info" ), LFUNCVAL( node_info ) }, { LSTRKEY( "chipid" ), LFUNCVAL( node_chipid ) }, { LSTRKEY( "flashid" ), LFUNCVAL( node_flashid ) }, { LSTRKEY( "flashsize" ), LFUNCVAL( node_flashsize) }, - { LSTRKEY( "heap" ), LFUNCVAL( node_heap ) }, { LSTRKEY( "input" ), LFUNCVAL( node_input ) }, { LSTRKEY( "output" ), LFUNCVAL( node_output ) }, // Moved to adc module, use adc.readvdd33() @@ -637,7 +633,6 @@ static const LUA_REG_TYPE node_map[] = { LSTRKEY( "stripdebug" ), LFUNCVAL( node_stripdebug ) }, #endif { LSTRKEY( "egc" ), LROVAL( node_egc_map ) }, - { LSTRKEY( "task" ), LROVAL( node_task_map ) }, #ifdef DEVELOPMENT_TOOLS { LSTRKEY( "osprint" ), LFUNCVAL( node_osprint ) }, #endif diff --git a/app/modules/tmr.c b/app/modules/tmr.c old mode 100755 new mode 100644 diff --git a/app/modules/uart.c b/app/modules/uart.c old mode 100755 new mode 100644 diff --git a/app/mqtt/Makefile b/app/mqtt/Makefile index 925cb5b7e3..efa5055cb5 100644 --- a/app/mqtt/Makefile +++ b/app/mqtt/Makefile @@ -12,7 +12,7 @@ # a generated lib/image xxx.a () # ifndef PDIR -GEN_LIBS = mqtt.a +GEN_LIBS = libmqtt.a endif STD_CFLAGS=-std=gnu11 -Wimplicit diff --git a/app/pcm/Makefile b/app/pcm/Makefile index efa7e3bdd6..18ae9c4428 100644 --- a/app/pcm/Makefile +++ b/app/pcm/Makefile @@ -12,7 +12,7 @@ # a generated lib/image xxx.a () # ifndef PDIR -GEN_LIBS = pcm.a +GEN_LIBS = libpcm.a endif STD_CFLAGS=-std=gnu11 -Wimplicit diff --git a/app/platform/common.c b/app/platform/common.c index aa6fdba7f6..7de2a9484a 100644 --- a/app/platform/common.c +++ b/app/platform/common.c @@ -24,7 +24,7 @@ extern char _flash_used_end[]; // Helper function: find the flash sector in which an address resides // Return the sector number, as well as the start and end address of the sector -static uint32_t flashh_find_sector( uint32_t address, uint32_t *pstart, uint32_t *pend ) +static uint32_t flash_find_sector( uint32_t address, uint32_t *pstart, uint32_t *pend ) { #ifdef INTERNAL_FLASH_SECTOR_SIZE // All the sectors in the flash have the same size, so just align the address @@ -53,7 +53,7 @@ static uint32_t flashh_find_sector( uint32_t address, uint32_t *pstart, uint32_t uint32_t platform_flash_get_sector_of_address( uint32_t addr ) { - return flashh_find_sector( addr, NULL, NULL ); + return flash_find_sector( addr, NULL, NULL ); } uint32_t platform_flash_get_num_sectors(void) @@ -73,12 +73,12 @@ uint32_t platform_flash_get_first_free_block_address( uint32_t *psect ) uint32_t start, end, sect; NODE_DBG("_flash_used_end:%08x\n", (uint32_t)_flash_used_end); if(_flash_used_end>0){ // find the used sector - sect = flashh_find_sector( platform_flash_mapped2phys ( (uint32_t)_flash_used_end - 1), NULL, &end ); + sect = flash_find_sector( platform_flash_mapped2phys ( (uint32_t)_flash_used_end - 1), NULL, &end ); if( psect ) *psect = sect + 1; return end + 1; - }else{ - sect = flashh_find_sector( 0, &start, NULL ); // find the first free sector + } else { + sect = flash_find_sector( 0, &start, NULL ); // find the first free sector if( psect ) *psect = sect; return start; diff --git a/app/platform/flash_api.c b/app/platform/flash_api.c index 05f1239043..6901afc059 100644 --- a/app/platform/flash_api.c +++ b/app/platform/flash_api.c @@ -116,10 +116,11 @@ bool flash_rom_set_size_type(uint8_t size) uint8_t data[SPI_FLASH_SEC_SIZE] ICACHE_STORE_ATTR; if (SPI_FLASH_RESULT_OK == spi_flash_read(0, (uint32 *)data, SPI_FLASH_SEC_SIZE)) { - ((SPIFlashInfo *)(&data[0]))->size = size; + NODE_DBG("\nflash_rom_set_size_type(%u), was %u\n", size, ((SPIFlashInfo *)data)->size ); + ((SPIFlashInfo *)data)->size = size; if (SPI_FLASH_RESULT_OK == spi_flash_erase_sector(0 * SPI_FLASH_SEC_SIZE)) { - NODE_DBG("\nERASE SUCCESS\n"); + NODE_DBG("\nSECTOR 0 ERASE SUCCESS\n"); } if (SPI_FLASH_RESULT_OK == spi_flash_write(0, (uint32 *)data, SPI_FLASH_SEC_SIZE)) { @@ -266,6 +267,7 @@ bool flash_rom_set_speed(uint32_t speed) if (SPI_FLASH_RESULT_OK == spi_flash_read(0, (uint32 *)data, SPI_FLASH_SEC_SIZE)) { ((SPIFlashInfo *)(&data[0]))->speed = speed_type; + NODE_DBG("\nflash_rom_set_speed(%u), was %u\n", speed_type, ((SPIFlashInfo *)(&data[0]))->speed ); if (SPI_FLASH_RESULT_OK == spi_flash_erase_sector(0 * SPI_FLASH_SEC_SIZE)) { NODE_DBG("\nERASE SUCCESS\n"); diff --git a/app/platform/platform.c b/app/platform/platform.c old mode 100755 new mode 100644 index 89d5070e6f..789c31ffdb --- a/app/platform/platform.c +++ b/app/platform/platform.c @@ -879,7 +879,7 @@ uint32_t platform_s_flash_write( const void *from, uint32_t toaddr, uint32_t siz if(SPI_FLASH_RESULT_OK == r) return size; else{ - NODE_ERR( "ERROR in flash_write: r=%d at %08X\n", ( int )r, ( unsigned )toaddr); + NODE_ERR( "ERROR in flash_write: r=%d at %p\n", r, toaddr); return 0; } } @@ -917,26 +917,35 @@ uint32_t platform_s_flash_read( void *to, uint32_t fromaddr, uint32_t size ) if(SPI_FLASH_RESULT_OK == r) return size; else{ - NODE_ERR( "ERROR in flash_read: r=%d at %08X\n", ( int )r, ( unsigned )fromaddr); + NODE_ERR( "ERROR in flash_read: r=%d at %p\n", r, fromaddr); return 0; } } int platform_flash_erase_sector( uint32_t sector_id ) { + NODE_DBG( "flash_erase_sector(%u)\n", sector_id); system_soft_wdt_feed (); return flash_erase( sector_id ) == SPI_FLASH_RESULT_OK ? PLATFORM_OK : PLATFORM_ERR; } -uint32_t platform_flash_mapped2phys (uint32_t mapped_addr) -{ +static uint32_t flash_map_meg_offset (void) { uint32_t cache_ctrl = READ_PERI_REG(CACHE_FLASH_CTRL_REG); if (!(cache_ctrl & CACHE_FLASH_ACTIVE)) return -1; - bool b0 = (cache_ctrl & CACHE_FLASH_MAPPED0) ? 1 : 0; - bool b1 = (cache_ctrl & CACHE_FLASH_MAPPED1) ? 1 : 0; - uint32_t meg = (b1 << 1) | b0; - return mapped_addr - INTERNAL_FLASH_MAPPED_ADDRESS + meg * 0x100000; + uint32_t m0 = (cache_ctrl & CACHE_FLASH_MAPPED0) ? 0x100000 : 0; + uint32_t m1 = (cache_ctrl & CACHE_FLASH_MAPPED1) ? 0x200000 : 0; + return m0 + m1; +} + +uint32_t platform_flash_mapped2phys (uint32_t mapped_addr) { + uint32_t meg = flash_map_meg_offset(); + return (meg&1) ? -1 : mapped_addr - INTERNAL_FLASH_MAPPED_ADDRESS + meg ; +} + +uint32_t platform_flash_phys2mapped (uint32_t phys_addr) { + uint32_t meg = flash_map_meg_offset(); + return (meg&1) ? -1 : phys_addr + INTERNAL_FLASH_MAPPED_ADDRESS - meg; } void* platform_print_deprecation_note( const char *msg, const char *time_frame) diff --git a/app/platform/platform.h b/app/platform/platform.h index a1ab812b96..50503bc4c2 100644 --- a/app/platform/platform.h +++ b/app/platform/platform.h @@ -279,13 +279,14 @@ int platform_flash_erase_sector( uint32_t sector_id ); /** * Translated a mapped address to a physical flash address, based on the - * current flash cache mapping. + * current flash cache mapping, and v.v. * @param mapped_addr Address to translate (>= INTERNAL_FLASH_MAPPED_ADDRESS) * @return the corresponding physical flash address, or -1 if flash cache is * not currently active. * @see Cache_Read_Enable. */ uint32_t platform_flash_mapped2phys (uint32_t mapped_addr); +uint32_t platform_flash_phys2mapped (uint32_t phys_addr); // ***************************************************************************** // Allocator support diff --git a/app/tsl2561/Makefile b/app/tsl2561/Makefile index 7963bd5dd9..762105c019 100644 --- a/app/tsl2561/Makefile +++ b/app/tsl2561/Makefile @@ -12,7 +12,7 @@ # a generated lib/image xxx.a () # ifndef PDIR -GEN_LIBS = tsl2561lib.a +GEN_LIBS = libtsl2561.a endif STD_CFLAGS=-std=gnu11 -Wimplicit diff --git a/app/u8glib/Makefile b/app/u8glib/Makefile index 7495a1d76b..773f40bb95 100644 --- a/app/u8glib/Makefile +++ b/app/u8glib/Makefile @@ -12,7 +12,7 @@ # a generated lib/image xxx.a () # ifndef PDIR -GEN_LIBS = u8glib.a +GEN_LIBS = libu8glib.a endif STD_CFLAGS=-std=gnu11 -Wimplicit diff --git a/app/ucglib/Makefile b/app/ucglib/Makefile index 6a9c929b9c..ddd3d94fee 100644 --- a/app/ucglib/Makefile +++ b/app/ucglib/Makefile @@ -12,7 +12,7 @@ # a generated lib/image xxx.a () # ifndef PDIR -GEN_LIBS = ucglib.a +GEN_LIBS = libucglib.a endif STD_CFLAGS=-std=gnu11 -Wimplicit diff --git a/app/user/user_exceptions.c b/app/user/user_exceptions.c index d95c84d872..a3d6333394 100644 --- a/app/user/user_exceptions.c +++ b/app/user/user_exceptions.c @@ -40,7 +40,6 @@ static exception_handler_fn load_store_handler; - void load_non_32_wide_handler (struct exception_frame *ef, uint32_t cause) { /* If this is not EXCCAUSE_LOAD_STORE_ERROR you're doing it wrong! */ @@ -73,13 +72,15 @@ void load_non_32_wide_handler (struct exception_frame *ef, uint32_t cause) { die: /* Turns out we couldn't fix this, so try and chain to the handler - * that was set by the SDK. If none then trigger a system break instead - * and hang if the break doesn't get handled. This is effectively - * what would happen if the default handler was installed. */ + * that was set. (This is typically a remote GDB break). If none + * then trigger a system break instead and hang if the break doesn't + * get handled. This is effectively what would happen if the default + * handler was installed. */ if (load_store_handler) { load_store_handler(ef, cause); + return; } - asm ("break 1, 1"); + asm ("break 1, 1"); while (1) {} } diff --git a/app/user/user_main.c b/app/user/user_main.c index 97c1836514..c49a304250 100644 --- a/app/user/user_main.c +++ b/app/user/user_main.c @@ -129,8 +129,7 @@ bool user_process_input(bool force) { return task_post_low(input_sig, force); } -void nodemcu_init(void) -{ +void nodemcu_init(void) { NODE_ERR("\n"); // Initialize platform first for lua modules. if( platform_init() != PLATFORM_OK ) @@ -139,11 +138,13 @@ void nodemcu_init(void) NODE_DBG("Can not init platform for modules.\n"); return; } - - if( flash_detect_size_byte() != flash_rom_get_size_byte() ) { - NODE_ERR("Self adjust flash size.\n"); + uint32_t size_detected = flash_detect_size_byte(); + uint32_t size_from_rom = flash_rom_get_size_byte(); + if( size_detected != size_from_rom ) { + NODE_ERR("Self adjust flash size. 0x%x (ROM) -> 0x%x (Detected)\n", + size_from_rom, size_detected); // Fit hardware real flash size. - flash_rom_set_size_byte(flash_detect_size_byte()); + flash_rom_set_size_byte(size_detected); system_restart (); // Don't post the start_lua task, we're about to reboot... @@ -260,6 +261,5 @@ void user_init(void) #ifndef NODE_DEBUG system_set_os_print(0); #endif - system_init_done_cb(nodemcu_init); } diff --git a/docs/en/lcd.md b/docs/en/lcd.md new file mode 100644 index 0000000000..5e167fd902 --- /dev/null +++ b/docs/en/lcd.md @@ -0,0 +1,101 @@ +## Lua Compact Debug (LCD) + +LCD (Lua Compact Debug) was developed in Sept 2015 by Terry Ellison as a patch to the Lua system to decrease the RAM usage of Lua scripts. This makes it possible to run larger Lua scripts on systems with limited RAM. Its use is most typically for eLua-type applications, and in this version it targets the **NodeMCU** implementation for the ESP8266 chipsets. + +This section gives a full description of LCD. If you are writing **NodeMCU** Lua modules, then this paper will be of interest to you, as it shows how to use LCD in an easy to configure way. *Note that the default `user_config.h` has enabled LCD at a level 2 stripdebug since mid-2016*. + +### Motivation + +The main issue that led me to write this patch is the relatively high Lua memory consumption of its embedded debug information, as this typically results in a 60% memory increase for most Lua code. This information is generated when any Lua source is complied because the Lua parser uses this as meta information during the compilation process. It is then retained by default for use in generating debug information. The only standard method of removing this information is to use the “strip” option when precompiling source using a standard eLua **luac.cross** on the host, or (in the case of NodeMCU) using the `node.compile()` function on the target environment. + +Most application developers that are new to embedded development simply live with this overhead, because either they aren't familiar with these advanced techniques, or they want to keep the source line information in error messages for debugging. + +The standard Lua compiler generates fixed 4 byte instructions which are interpreted by the Lua VM during execution. The debug information consists of a map from instruction count to source line number (4 bytes per instruction) and two tables keyed by the names of each local and upvalue. These tables contain metadata on these variables used in the function. This information can be accessed to enable symbolic debugging of Lua source (which isn't supported on **NodeMCU** platforms anyway), and the line number information is also used to generate error messages. + +This overhead is sufficient large on limited RAM systems to replace this scheme by making two changes which optimize for space rather than time: + +- The encoding scheme used in this patch typically uses 1 byte per source line instead of 4 bytes per instruction, and this represents a 10 to 20-fold reduction in the size of this vector. The access time during compile is still **O(1)**, and **O(N)** during runtime error handling, where **N** is number of non-blank lines in the function. In practice this might add a few microseconds to the time take to generate the error message for typical embedded functions. + +- The line number, local and upvalue information is needed during the compilation of a given compilation unit (a source file or source string), but its only use after this is for debugging and so can be discarded. (This is what the `luac -s` option and `node.compile()` do). The line number information if available is used in error reporting. An extra API call has therefore been added to discarded this debug information on completion of the compilation. + +To minimise the impact within the C source code for the Lua system, an extra system define **LUA_OPTIMIZE_DEBUG** can be set in the `user_config.h` file to configure a given firmware build. This define sets the default value for all compiles and can take one of four values: + +1. (or not defined) use the default Lua scheme. +2. Use compact line encoding scheme and retain all debug information. +3. Use compact line encoding scheme and only retain line number debug information. +4. Discard all debug information on completion of compile. + +Building the firmware with the 0 option compiles to the pre-patch version. Options 1-3 generate the `strip_debug()` function, which allows this default value to be set at runtime. + +_Note that options 2 and 3 can also change the default behaviour of the `loadstring()` function in that any functions declared within the string cannot inherited any outer locals within the parent hierarchy as upvalues if these have been stripped of the locals and upvalues information._ + +### Details + +There are various API calls which compile and load Lua source code. During compilation each variable name is parsed, and is then resolved in the following order: + +- Against the list of local variables declared so far in the current scope is scanned for a match. + +- Against the local variable lists for each of the lexically parent functions are then scanned for a match, and if found the variable is tagged as an _upvalue_. + +- If unmatched against either of these local scopes then the variable defaults to being a global reference. + +The parser and code generator must therefore access the line mapping, upvalues, and locals information tables maintained in each function Prototype header during source compilation. This scoping scheme works because function compilation is recursive: if function A contains the definition of function B which contains the definition of function C, then the compilation of A is paused to compile B and this is in turn paused to compile C; then B complete and then A completes. + +The variable meta information is stored in standard Lua tables which are allocated using the standard Lua doubling algorithm and hence they can contain a lot of unused space. The parser therefore calls `close_func()` once compilation of a function has been completed to trim these vectors to the final sizes. + +The patch makes the following if `LUA_OPTIMIZE_DEBUG` > 0. (The existing functionality is preserved if this define is zero or undefined.) + +- It adds an extra API call: `stripdebug([level[, function]])` as discussed below. + +- It extends the trim logic in `close_func()` to replace this trim action by deleting the information according to the current default debug optimization level. + +- The `lineinfo` vector associated with each function is replaced by a `packedlineinfo` string using a run length encoding scheme that uses a repeat of an optional line number delta (this is omitted if the line offset is zero) and a count of the number of instruction generated for that source line. This scheme uses roughly an **M** byte vector where **M** is the number of non-blank source lines, as opposed to a **4N** byte vector where **N** is the number of VM instruction. This vector is built sequentially during code generation so it is this patch conditionally replaces the current map with an algorithm to generate the packed version on the fly. + +The `stripdebug([level[, function]])` call is processed as follows: + +- If both arguments are omitted then the function returns the current default strip level. + +- If the function parameter is omitted, then the level is used as the default setting for future compiles. The level must be 1-3 corresponding to the above debug optimization settings. Hence if `stripdebug(3)` is included in **init.lua**, then all debug information will be stripped out of subsequently compiled functions. + +- The function parameter if present is parsed in the same way as the function argument in `setfenv()` (except that the integer 0 level is not permitted, and this function tree corresponding to this scope is walked to implement this debug optimization level. + +The `packedlineinfo` encoding scheme is as follows: + +- It comprises a repeat of (optional) line delta + VM instruction count (IC) for that line starting from a base line number of zero. The line deltas are optional because line deltas of +1 are assumed as default and therefore not emitted. + +- ICs are stored as a single byte with the high bit set to zero. Sequences longer than 126 instructions for a single sequence are rare, but can be are encoded using a multi byte sequence using 0 line deltas, e.g. 126 (0) 24 for a line generating 150 VM instructions. The high bit is always unset, and note that this scheme reserves the code 0x7F as discussed below. + +- Line deltas are stored with the high bit set and are variable (little-endian) in length. Since deltas are always delimited by an IC which has the top bit unset, the length of each delta can be determined from these delimiters. Deltas are stored as signed ones-compliment with the sign bit in the second bit of low order byte, that is in the format (in binary) `1snnnnnnn [1nnnnnnn]*`, with `s` denoting the sign and `n…n` the value element using the following map. This means that a single byte is used encode line deltas in the range -63 … 65; two bytes used to encode line deltas in the range -8191 … 8193, etc.. +```C + value = (sign == 1) ? -delta : delta - 2 +``` + +- This approach has no arbitrary limits, in that it can accommodate any line delta or IC count. Though in practice, most deltas are omitted and multi-byte sequences are rarely generated. + +- The codes 0x00 and 0x7F are reserved in this scheme. This is because Lua allocates such growing vectors on a size-doubling basis. The line info vector is always null terminated so that the standard **strlen()** function can be used to determine its length. Any unused bytes between the last IC and the terminating null are filled with 0x7F. + +The current mapping scheme has **O(1)** access, but with a code-space overhead of some 140%. This alternative approach has been designed to be space optimized rather than time optimized. It requires the actual IC to line number map to be computed by linearly enumerating the string from the low instruction end during execution, resulting in an **O(N)** access cost, where **N** is the number of bytes in the encoded vector. However, code generation builds this information incrementally, and so only appends to it (or occasionally updates the last element's line number), and the patch adds a couple of fields to the parser `FuncState` record to enable efficient **O(1)** access during compilation. + +### Testing + +Essentially testing any eLua compiler or runtime changes are a total pain, because eLua is designed to be build against a **newlib**-based ELF. Newlib uses a stripped down set of headers and libraries that are intended for embedded use (rather than being ran over a standard operating system). Gdb support is effectively non-existent, so I found it just easier first to develop this code on a standard Lua build running under Linux (and therefore with full gdb support), and then port the patch to NodeMCU once tested and working. + +I tested my patch in standard Lua built with "make generic" and against the [Lua 5.1 suite](http://lua-users.org/lists/lua-l/2006-03/msg00723.html). The test suite was an excellent testing tool, and it revealed a number of cases that exposed logic flaws in my approach, resulting from Lua's approach of not carrying out inline status testing by instead implementing a throw / catch strategy. In fact I realised that I had to redesign the vector generation algorithm to handle this robustly. + +As with all eLua builds the patch assumes Lua will not be executing in a multithreaded environment with OS threads running different lua_States. (This is also the case for the NodeMCU firmware). It executes the full test suite cleanly as maximum test levels and I also added some specific tests to cover new **stripdebug** usecases. + +Once this testing was completed, I then ported the patch to the NodeMCU build. This was pretty straight forward as this code is essentially independent of the NodeMCU functional changes. The only real issue as to ensure that the NodeMCU `c_strlen()` calls replaced the standard `strlen()`, etc. + +I then built both `luac.cross` and firmware images with the patch disable to ensure binary compatibility with the non-patched version and then with the patch enabled at optimization level 3. + +In use there is little noticeable difference other than the code size during development are pretty much the same as when running with `node.compile()` stripped code. The new option 2 (retaining packed line info only) has such a minimal size impact that its worth using this all the time. I've also added a separate patch to NodeMCU (which this assumes) so that errors now generate a full traceback. + +### How to enable LCD + +Enabling LCD is simple: all you need is a patched version and define `LUA_OPTIMIZE_DEBUG` at the default level that you want in `app/include/user_config.h` and do a normal make. + +Without this define enabled, the unpatched version is generated. + +Note that since `node.compile()` strips all debug information, old **.lc** files generated by this command will still run under the patched firmware, but binary files which retain debug information will not work across patched and non-patched versions. + +Other than optionally including a `node.stripdebug(N)` or whatever in your **init.lua**, the patch is otherwise transparent at an application level. diff --git a/docs/en/lfs.md b/docs/en/lfs.md new file mode 100644 index 0000000000..91327d3cce --- /dev/null +++ b/docs/en/lfs.md @@ -0,0 +1,246 @@ +# Lua Flash Store (LFS) + +## Background + +An IoT device such as the ESP8266 has very different processor characteristics from the CPU in a typical PC: + +- Conventional CPUs have a lot of RAM, typically more than 1 Gb, that is used to store both code and data. IoT processors like the ESP variants use a [modified Harvard architecture](https://en.wikipedia.org/wiki/Modified_Harvard_architecture) where code can also be executed out of flash memory that is mapped into a address region separate from the limited RAM. + +- Conventional CPU motherboards include RAM and a lot of support chips. ESP modules are postage stamp-sized and typically comprise one ESP [SoC](https://en.wikipedia.org/wiki/System_on_a_chip) and a flash memory chip used to store firmware and a limited file system. + +Lua was originally designed as a general embeddable extension language for applications that would typically run on systems such as a PC, but its design goals of speed, portability, small kernel size, extensibility and ease-of-use also make Lua a good choice for embedded use on an IoT platform. Our NodeMCU firmware implementation was therefore constrained by the standard Lua core runtime system (**RTS**) that assumes a conventional CPU architecture with both Lua code and data in RAM; however ESP8266 modules only have approximately 48Kb RAM for application use, even though the firmware itself executes out of the larger flash-based program memory. + +This Lua Flash Store (**LFS**) patch modifies the NodeMCU Lua RTS to allow Lua code and its associated constant data to be executed directly out of flash-memory, just as the firmware itself is executed. This now enables NodeMCU Lua developers to create Lua applications with up to 256Kb Lua code and read-only (**RO**) constants executing out of flash, so that all of the RAM is available for read-write (**RW**) data. + +Though the ESP architecture does allow RW operations to flash, these are constrained by the write limitations of NAND flash architecture, as writing involves the block erasing of 4Kb pages and then overwriting each pach with new content. Whilst it is possible (as with SPIFFS) to develop R/W file systems working within this constraint, memory-mapped read access to flash is cached through a RAM cache in order to accelerate code execution, and this makes it practically impossible to modify executable code pages on the fly. Hence the LFS patch must work within a reflash-and-restart paradigm for reloading the LFS. + +The LFS patch does this by adding two API new calls to the `node` module: one to reflash the LFS and restart the processor, and one to access the LFS store once loaded. Under the hood, it also addresses all of the technical issues to make this magic happen. + +The remainder of this paper is split into two sections: + +- The first section provides an overview the issues that a Lua developer needs to understand at an application level to use LFS effectively. + +- The second gives more details on the technical issues that were addressed in order to implement the patch. This is a good overview for those that are interested, but many application programmers won't care how the magic happens, just that it does. + + +## Using LFS + +### Selecting the firmware + +Power developers might want to use Docker or their own build environment as per our [Building the firmware](https://nodemcu.readthedocs.io/en/master/en/build/) documentation, and so `app/include/user_config.h` has now been updated to include the necessary documentation on how to select the configuration options to make an LFS firmware build. + +However, most Lua developers seem to prefer the convenience of our [Cloud Build Service](https://nodemcu-build.com/), so we will add two extra menu options to facilitate building LFS images: + +Variable | Option +---------|------------ +LFS size | (none, 32Kb, 64Kb, 94Kb) The default is none. Selecting a numeric value builds in the corresponding LFS. +SPIFFS size | (default or a multiple of 64Kb) The cloud build will base the SPIFFS at 1Mb if an explicit size is specified. + +You must choose an explicit (non-default) LFS size to enable the use of LFS. Whilst you can use a default (maximal) SPIFFS configuration, most developers find it more useful to work with a fixed SPIFFS that has been sized to match their application reqirements. + +### Choosing your development lifecycle + + The build environment for generating the firmware images is Linux-based, but as you can use our cloud build service to generate these, you can develop NodeMCU applications on pretty much any platform including Windows and MacOS. Unfortunately LFS images must be built off-ESP on a host platform, so you must be able to run the `luac.cross` cross compiler on your development machine to build LFS images. + +- For Windows 10 developers, the easiest method of achieving this is to install the [Windows Subsystem for Linux](https://en.wikipedia.org/wiki/Windows_Subsystem_for_Linux). Most WSL users install the Ubuntu Bash shell as well; note that this is just a shell and some core GNU utilities (somewhat similar to Cygwin) rather than a full Ubuntu OS, as WSL extends the NT kernel to support the direct execution of Linux ELF images. WSL can directly run the `luac.cross` and `spiffsimg` that are build as part of the firmware. You will also need the `esptool.py` tool but `python.org` already provides Python releases for Windows. + +- Linux users can just use these tools natively. + +- In principle, only the build environment components needed to support `luac.cross` and `spiffsimg` are the `app/lua/lua_cross` and `tools/spifsimg` subdirectory makes. It should be straight forward to get these working under any environment which provides POSIX runtime support, including MacOS and Cygwin (for windows versions prior to Win10), but suitable developer effort is required to generate suitable executables; any volunteers? + +Most Lua developers seem to start with the [ESPlorer](https://github.com/4refr0nt/ESPlorer) tool, a 'simple to use' IDE that enables beginning Lua developers to get started. However, ESPlorer relies on a UART connection and this can be slow and cumbersome, and it doesn't scale well for larger ESP application. + +So many experienced Lua developers switch to a rapid development cycle where they use a development machine to maintain your master Lua source. Going this route will allow you use your favourite program editor and source control, with one of various techniques for compiling the lua on-host and downloading the compiled code to the ESP: + +- If you use a fixed SPIFFS image (I find 128Kb is enough for most of my applications), then you can script recompiling your LC files, rebuilding a SPIFFS image and loading it onto the ESP using `esptool.py` in less than 60 sec. You can either embed the LFS.img in the SPIFFS, or you can use the `luac.cross -a` option to directly load the new LFS image into the LFS region within the firmware. + +- I now have an LFS aware version of my LuaOTA provisioning system (see `lua_examples/luaOTA`). This handles all of the incremental compiling and LFS reloads transparently. This is typically integrated into the ESP application. + +- Another option would be to include the FTP and Telnet modules in the base LFS image and to use telnet and FTP to update your system. (Given that a 64Kb LFS can store thousands of lines of Lua, doing this isn't much of an issue.) + +My current practice is to use a small bootstrap `init.lua` file in SPIFFS to load the `_init` module from LFS, and this does all of the actual application initialisation. My `init.lua`: + +- Is really a Lua binary (`.lc`) file renamed to a `.lua` extension. Using a binary init file avoids loading the Lua compiler. This works because even though the firmware looks for `init.lua`, the file extension itself is a just a convention; it is treated as a Lua binary if it has the correct Lua binary header. + +- Includes a 1 sec delay before connecting to the Wifi. This is a "just in case" when I am developing. This is enough to allow me to paste a `file.remove'init.lua'` into the UART if I want to do different development paths. + +No doubt some standard usecase / templates will be developed by the community over the next six months. + +### Programming Techniques and approachs + +I have found that moving code into LFS has changed my coding style, as I tend to use larger modules and I don't worry about in-memory code size. This facilitates a more 'keep it simple stupid' coding style, so my ESP Lua code now looks more similar to host-based Lua code. I still prefer to keep the module that I am currently testing in SPIFFS, and only move modules into LFS once they are stable. However if you use `require` to load modules then this can all be handled by the require loader. + +Here is the code fragment that I use in my `_init` module to do this magic: + +```Lua +do + local index = node.flashindex + -- Setup the LFS object + local lfs_t = { + __index = function(_, name) + local fn_ut, ba, ma, size, modules = index(name) + if not ba then + return fn_ut + elseif name == '_time' then + return fn_ut + elseif name == '_config' then + local fs_ma, fs_size = file.fscfg() + return {ba, ma, fs_ma, size, fs_size} + elseif name == '_list' then + return modules + else + return nil + end + end, + __newindex = function(_, name, value) + error("LFS is readonly. Invalid write to LFS." .. name, 2) + end + } + rawset(getfenv(),'LFS', setmetatable(lfs_t,lfs_t)) + -- And add LFS to the require path list + local function loader_flash(module) + local fn, ba = index(module) + return ba and "Module not in LFS" or fn + end + package.loaders[3] = loader_flash + +end +``` + +Once this has been executed, if you have a function module `func1` in LFS, then `LFS.func1(x,y,z)` just works as you would expect. The LFS properties `_time`, `_config` and `_list` can be used to access the other LFS metadata that you need. + +Of course, if you use Lua modules to build your application then `require "some_module"` will automatically path in and load your modules from LFS. Note that SPIFFS is still ahead of LFS in this search list, so if you have a dev version in SPIFFS, say, then this will be loaded first. However, if you want to want to swap this search order so that the LFS is searched first, then set `package.loaders[1] = loader_flash` in your `_init` code. If you need to swap the search order temporarily for development or debugging, then do this after you've run the `_init` code: + +```Lua +do local pl = package.loaders; pl[1],pl[3] = pl[3],pl[1]; end +``` + +Whilst LFS is primarily used to store compiled modules, it also includes its own string table and any strings loaded into this can be used in your Lua application without taking any space in RAM. Hence, you might also want to preload any other frequently used strings into LFS as this will both save RAM use and reduced the Lua-custom Garbage Collector (**LGC**) overheads. + +The patch adds an extra debug function `getstrings()` function to help you determine what strings are worth adding to LFS. This takes an optional string argument `'RAM'` (the default) or `'ROM'`, and returns a list of the strings in the corresponding table. So the following example can be used to get a listing of the strings in RAM. You can enter the following Lua at the interactive prompt or call it as a debug function during a running application in order to generate this string list. + +```Lua +do + local a=debug.getstrings'RAM' + for i =1, #a do a[i] = ('%q'):format(a[i]) end + print ('local preload='..table.concat(a,',')) +end +``` + +If you then create a file, say `LFS_dummy_strings.lua`, and put these `local preload` lines in it, and include this file in your `luac.cross -f`, then the cross compiler will generate a ROM string table that includes all strings referenced in this dummy module. You never need to call this module; just it's inclusion in the LFS build is enough to add the strings to the ROM table. Once in the ROM table, then you can use them subsequently in your application without incurring any RAM or LGC overhead. The following example is a useful starting point, but if needed then you can add to this +for your application. + +```Lua +local preload = "?.lc;?.lua", "@init.lua", "_G", "_LOADED", "_LOADLIB", "__add", +"__call", "__concat", "__div", "__eq", "__gc", "__index", "__le", "__len", "__lt", +"__mod", "__mode", "__mul", "__newindex", "__pow", "__sub", "__tostring", "__unm", +"collectgarbage", "cpath", "debug", "file", "file.obj", "file.vol", "flash", +"getstrings", "index", "ipairs", "list", "loaded", "loader", "loaders", "loadlib", +"module", "net.tcpserver", "net.tcpsocket", "net.udpsocket", "newproxy", "package", +"pairs", "path", "preload", "reload", "require", "seeall", "wdclr" +``` + +## Technical Issues + +Whilst memory capacity isn't a material constraint on most conventional machines, the Lua RTS still embeds some features to minimise overall memory usage. In particular: + +- The more resource intensive data types are know as _collectable objects_, and the RTS includes a LGC which regularly scans these collectable resources to determine which are no longer in use, so that their associated memory can be reclaimed and reused. + +- The Lua RTS also treats strings and compiled function code as collectable objects, so that these can also be LGCed when no longer referenced + +The compiled code, as executed by Lua RTS, internally comprises one or more function prototypes (which use a `Proto` structure type) plus their associated vectors (constants, instructions and meta data for debug). Most of these compiled constant types are basic (e.g. numbers) and the only collectable constant data type are strings. The other collectable types such as arrays are actually created at runtime by executing Lua compiled instructions to build each resource dynamically. + +Currently, when any Lua file is loaded into an ESP application, the RTS loads the corresponding compiled version into RAM. Each compiled function has its own Proto structure hierarchy, but this hierarchy is not exposed directly to the running application; instead the compiler generate `CLOSURE` instruction which is executed at runtime to bind the Proto to a Lua function value thus creating a [closure](https://en.wikipedia.org/wiki/Closure_(computer_programming)). Since this occurs at runtime, any `Proto` can be bound to multiple closures. A Lua closure can have multiple RW [Upvalues](https://www.lua.org/pil/27.3.3.html) bound to it, and so function value is much like a Lua object in that it is refering to something that can contain RW state, even though the `Proto` hierarchy itself is intrinsically RO. + +Whilst advanced ESP Lua programmers can use overlay techniques to ensure that only active functions are loaded into RAM and thus increase the effective application size, this adds to runtime and program complexity. Moving Lua "program" resources into ESP Flash addressable memory typically doubles the effective RAM available, and largley removes the need to complicate applications code to facilitate overlaying. + +Any RO resources that are relocated to a flash address space: + +- Must not be collected. Also RW references to RO resources must be robustly handled by the LGC. +- Cannot reference to any volatile RW data elements (though RW resources can refer to RO resources). + +All strings in Lua are [interned](https://en.wikipedia.org/wiki/String_interning), so that only one copy of any string is kept in memory and most string manipulation uses the address of this single copy as a unique reference. This uniqueness and the LGC of strings is facilitated by using a global string table that is hooked into the Lua Global State. Under standard Lua, any new string is first resolved against RAM string table, with only the string-misses being added to the string table. The LFS patch adds a second RO string table in flash and which contains all strings used in LFS Protos. Maintaining integrity across the two RAM and RO string tables is simple and low-cost, with LFS resolution process extended across both the RAM and ROM string tables. Hence any strings already in the ROM string table will generate a unique string reference without the need to add an additional entry in the RAM table. This both significantly reduces the size of the RAM string table, and removes a lot of strings from the LCG scanning. + +Note that early development implementations of the LFS build process allowed on-target ESP builds. Unfortunately, we found in practice that the Lua compiler was so resource hungry that it was impractical to get this to scale to usable application sizes, and we therefore abandoned this approach, moving the LFS build process onto the development host machine by embedding this into `luac.cross`. This approach avoids all of the update integrity issues involved in building a new LFS which might require RO resources already referenced in the RW ones. + +Any LFS image can be loaded in the LFS store by one of two mechanisms: + +- The image can be build on the host and then copied into SPIFFS. Calling the `node.flashreload()` API with this filename will load the image, and then schedule a restart to leave the ESP in normal application mode, but with an updated flash block. This sequence is essentially atomic. Once called, the only exit is the reboot. + +- The second option is to build the LFS image using the `-a` option to base it at the correct absolute address of the LFS store for a given firmware image. The LFS can then be flashed to the ESP along with the firmware image. + +The LFS store is a fixed size for any given firmware build (configurable by the a pplication developer through `user_config.h`) and is at a build-specific base address within the `ICACHE_FLASH` address space. This is used to store the ROM string table and the set of `Proto` hierarchies corresponding to a list of Lua files in the loaded image. + +A separate `node.flashindex()` function creates a new Lua closure based on a module loaded into LFS and more specfically its flash-based prototype; whilst this access function is not transparent at a coding level, this is no different functionally than already having to handle `lua` and `lc` files and the existing range of load functions (`load`,`loadfile`, `loadstring`). Either way, creating a closure on flash-based prototype is _fast_ in terms of runtime. (It is basically a single instruction rather than a compile, and it has minimal RAM impact.) + +### Basic approach + +This **LFS** patch uses two string tables: the standard Lua RAM-based table (`RWstrt`) and a second RO flash-based one (`ROstrt`). The `RWstrt` is searched first when resolving new string requests, and then the `ROstrt`. Any string not already in either table is then added to the `RWstrt`, so this means that the RAM-based string table only contains application strings that are not already defined in the `ROstrt`. + +Any Lua file compiled into the LFS image includes its main function prototype and all the child resources that are linked in its `Proto` structure; so all of these resources are compiled into the LFS image with this entire hierarchy self-consistently within the flash memory. + +```C + TValue *k; Constants used by the function + Instruction *code The Lua VM instuction codes + struct Proto **p; Functions defined inside the function + int *lineinfo; Debug map from opcodes to source lines + struct LocVar *locvars; Debug information about local variables + TString **upvalues Debug information about upvalue names + TString *source String name associated with source file + ``` + +Such LFS images are created by `luac.cross` using the `-f` option, and this builds a flash image based on the list of modules provided but with a master "main" function of the form: + +```Lua +local n = ...,1518283691 -- The Unix Time of the compile +if n == "module1" then return module1 end +if n == "module2" then return module2 end +-- and so on +if n == "moduleN" then return module2 end +return 1518283691,"module1","module2", --[[ ... ]] ""moduleN" +``` + +You can't actually code this Lua because the modules are in separate compilation units, but the compiler being a compiler can just emit the compiled code directly. (See `app/lua/luac_cross/luac.c` for the details.) + +The deep cross-copy of the `Proto` hierarchy is also complicated because current hosts are typically 64bit whereas the ESPs are 32bit, so the structures need repacking. (See `app/lua/luac_cross/luac.c` for the details.) + +With this patch, the `luac.cross` build has been moved into the overall application hierarchy and is now simply a part of the NodeMCU make. The old Lua script has been removed from the `tools` directory, together with the need to have Lua preinstalled on the host. + +The LFS image is by default position independent, so is independent of the actual NodeMCU target image. You just have to copy it to the target file system and execute a `reload` to copy this to the correct location, relocating all address to the correct base. (See `app/lua/lflash.c` for the details.) This process is fast. However, -a `luac.cross -a` also allows absolute address images to be built for direct flashing into the LFS store during provisioning. + +### Impact of the Lua Garbage Collector + +The LGC applies to what the Lua VM classifies as collectable objects (strings, tables, functions, userdata, threads -- known collectively as `GCObjects`). A simple two "colour" LGC was used in previous Lua versions, but Lua 5.1 introduced the Dijkstra's 3-colour (*white*, *grey*, *black*) variant that enabled the LGC to operate in an incremental mode. This permits smaller LGC steps interspersed by LGC pause, and is very useful for larger scale Lua implementations. Whilst this is probably not really needed for IoT devices, NodeMCU follows this standard Lua 5.1 implementation, albeit with the `elua` EGC changes. + +In fact, two *white* flavours are used to support incremental working (so this 3-colour algorithm really uses 4). All newly allocated collectable objects are marked as the current *white*, and include a link in their header to enable scanning through all such Lua objects. They may also be referenced directly or indirectly via one of the Lua application's *roots*: the global environment, the Lua registry and the stack. The LGC operates two broad phases: **mark** and **sweep**. + +The LGC algorithm is quite complex and assumes that all GCObjects are RW so that a flag byte within each object can be updated during the mark and sweep processing. LFS introduces GCObjects that are actually stored in RO memory and are therefore truly RO. Any attempt to update their content during LGC will result in the firmware crashing with a memory exception, so the LFS patch must therefore modify the LGC processing to avoid such potential updates whilst maintaining its integrity, and the remainder of this +section provides further detail on how this was achieved. + +The **mark** phase walks collectable objects by a recursive walk starting at at the LGC roots. (This is referred to as _traverse_.) Any object that is visited in this walk has its colour flipped from *white* to *grey* to denote that it is in use, and it is relinked into a grey list. The grey list is iteratively processed, removing one grey object at a time. Such objects can reference other objects (e.g. a table has many keys and values which can also be collectable objects), so each one is then also traversed and all objects reachable from it are marked, as above. After an object has been traversed, it's turned from grey to black. The LGC will walks all RW collectable objects, traversing the dependents of each in turn. As RW objects can now refer to RO ones, the traverse routines has additinal tests to skip trying to mark any RO LFS references. + +The white flavour is flipped just before entering the **sweep** phase. This phase then loops over all collectable objects. Any objects found with previous white are no longer in user, and so can be freed. The 'current' white are kept; this prevents any new objected created during a paused sweep from being accidentally collected before being marked, but this means that it takes two sweeps to free all unused objects. There are other subtleties introduced in this 3-colour algorithm such as barriers and back-tracking to maintain integrity of the LGC, and these also needed extra rules to handle RO GCObjects correclty, but detailed explanation of these is really outside the scope of this paper. + +As well as standard collectable GCOobjets: + +- Standard Lua has the concept of **fixed** objects. (E.g. the main thread). These won't be collected by the LGC, but they may refer to objects that aren't fixed, so the LGC still has to walk through an fixed objects. + +- eLua added the the concept of **readonly** objects, which confusingly are a hybrid RW/RO implementation, where the underlying string resource is stored as a program constant in flash memory but the `TSstring` structure which points to this is still kept in RAM and can by GCed, except that in this case the LGC does not free the RO string constant itself. + +- LFS introduces a third variant **flash** object for `LUA_TPROTO` and `LUA_TSTRING` types. Flash objects can only refer to other flash objects and are entirely located in the LFS area in flash memory. + +The LGC already processed the _fixed_ and _readonly_ object, albeit as special cases. In the case of _flash_ GCObjects, the `mark` flag is in read-only memory and therefore the LGC clearly can't use this as a RW flag in its mark and sweep processing. So the LGC skips any marking operations for flash objects. Likewise, where all other GCObjects are linked into one of a number of sweeplists using the object's `gclist` field. In the case of flash objects, the compiler presets the `mark` and `gclist` fields with the fixed and readonly mark bits set, and the list pointer to `NULL` during the compile process. + +As far as the LGC algorithm is concerned, encountering any _flash_ object in a sweep is a dead end, so that branch of the walk of the GCObject hierarchy can be terminated on encountering a flash object. This in practice all _flash_ objects are entirely removed from the LGC process, without compromising collection of RW resources. + +### General comments + +- **Reboot implementation**. Whilst the application initiated LFS reload might seem an overhead, it typically only adds a few seconds per reboot. We may also consider the future enhancement of the `esptool.py` to enable the inclusion of an LFS image into the unified application flash image. + +- **LGC reduction**. Since the cost of LGC is directly related to the size of the LGC sweep lists, moving RO resources into LFS memory removes them from the LGC scope and therefore reduces LGC runtime accordingly. + +- **Typical Usecase**. The rebuilding of a store is an occasional step in the development cycle. (Say up to 10 times a day in a typical intensive development process). Modules and source files under development would typically be executed from SPIFFS in `.lua` format. The developer is free to reorder the `package.loaders` and load any SPIFFS files in preference to Flash ones. And if stable code is moved into Flash, then there is little to be gained in storing development Lua code in SPIFFS in `lc` compiled format. + +- **Flash caching coherency**. The ESP chipset employs hardware enabled caching of the `ICACHE_FLASH` address space, and writing to the flash does not flush this cache. However, in this restart model, the CPU is always restarted before any updates are read programmatically, so this (lack of) coherence isn't an issue. + +- **Failsafe reversion**. Since the entire image is precompiled, the chances of failure during reload are small. The loader uses the Flash NAND rules to write the flash header flag in two parts: one at start of the load and again at the end. If on reboot, the flag in on incostent state, then the LFS is cleared and disabled until the next reload. diff --git a/docs/en/modules/coap.md b/docs/en/modules/coap.md old mode 100755 new mode 100644 diff --git a/docs/en/modules/node.md b/docs/en/modules/node.md index 823e765017..76ef92cad6 100644 --- a/docs/en/modules/node.md +++ b/docs/en/modules/node.md @@ -172,6 +172,40 @@ none #### Returns flash ID (number) +## node.flashindex() + +Returns the function reference for a function in the LFS (Lua Flash Store). + +#### Syntax +`node.flashindex(modulename)` + +#### Parameters +`modulename` The name of the module to be loaded. If this is `nil` or invalid then an info list is returned + +#### Returns +- In the case where the LFS in not loaded, `node.flashindex` evaluates to `nil`, followed by the flash and mapped base addresss of the LFS +- If the LFS is loaded and the function is called with the name of a valid module in the LFS, then the function is returned in the same way the `load()` and the other Lua load functions do. +- Otherwise an extended info list is returned: the Unix time of the LFS build, the flash and mapped base addresses of the LFS and its current length, and an array of the valid module names in the LFS. + +#### Example + +The `node.flashindex()` is a low level API call that is normally wrapped using standard Lua code to present a simpler application API. See the module `_init.lua` in the `lua_examples/lfs` directory for an example of how to do this. + +## node.flashreload() + +Reload the LFS (Lua Flash Store) with the flash image provided. Flash images are generated on the host machine using the `luac.cross`commnad. + +#### Syntax +`node.flashreload(imageName)` + +#### Parameters +`imageName` The of name of a image file in the filesystem to be loaded into the LFS. + +#### Returns +If the LFS image has the incorrect signature or size, then `false` is returned. +In the case of the `imagename` being a valid LFS image, this is then loaded into flash. The ESP is then immediately rebooted so control is not returned to the calling application. + + ## node.flashsize() Returns the flash chip size in bytes. On 4MB modules like ESP-12 the return value is 4194304 = 4096KB. @@ -221,7 +255,7 @@ system heap size left in bytes (number) ## node.info() -Returns NodeMCU version, chipid, flashid, flash size, flash mode, flash speed. +Returns NodeMCU version, chipid, flashid, flash size, flash mode, flash speed, and Lua File Store (LFS) usage statics. #### Syntax `node.info()` diff --git a/ld/nodemcu.ld b/ld/nodemcu.ld index 9c0a0dc1fc..416a4e4160 100644 --- a/ld/nodemcu.ld +++ b/ld/nodemcu.ld @@ -233,22 +233,25 @@ SECTIONS /* Link-time arrays containing the defs for the included modules */ . = ALIGN(4); - lua_libs = ABSOLUTE(.); + lua_libs_base = ABSOLUTE(.); /* Allow either empty define or defined-to-1 to include the module */ KEEP(*(.lua_libs)) LONG(0) LONG(0) /* Null-terminate the array */ - lua_rotable = ABSOLUTE(.); + lua_rotable_base = ABSOLUTE(.); KEEP(*(.lua_rotable)) LONG(0) LONG(0) /* Null-terminate the array */ - /* SDK doesn't use libc functions, and are therefore safe to put in flash */ + /* SDK doesn't use libc functions, and are therefore safe to put in flash */ */libc.a:*.o(.text* .literal*) /* end libc functions */ + /* Reserved areas, flash page aligned and last */ + . = ALIGN(4096); + KEEP(*(.irom.reserved .irom.reserved.*)) + _irom0_text_end = ABSOLUTE(.); _flash_used_end = ABSOLUTE(.); } >irom0_0_seg :irom0_0_phdr =0xffffffff - } /* get ROM code address */ diff --git a/lua_examples/lfs/_init.lua b/lua_examples/lfs/_init.lua new file mode 100644 index 0000000000..3a6f406253 --- /dev/null +++ b/lua_examples/lfs/_init.lua @@ -0,0 +1,83 @@ +-- +-- File: _init.lua +--[[ + + This is a template for the LFS equivalent of the SPIFFS init.lua. + + It is a good idea to such an _init.lua module to your LFS and do most of the LFS + module related initialisaion in this. This example uses standard Lua features to + simplify the LFS API. + + The first section adds a 'LFS' table to _G and uses the __index metamethod to + resolve functions in the LFS, so you can execute the main function of module + 'fred' by executing LFS.fred(params), etc. It also implements some standard + readonly properties: + + LFS._time The Unix Timestamp when the luac.cross was executed. This can be + used as a version identifier. + + LFS._config This returns a table of useful configuration parameters, hence + print (("0x%6x"):format(LFS._config.lfs_base)) + gives you the parameter to use in the luac.cross -a option. + + LFS._list This returns a table of the LFS modules, hence + print(table.concat(LFS._list),'\n') + gives you a single column listing of all modules in the LFS. + +---------------------------------------------------------------------------------]] + +local index = node.flashindex + +local lfs_t = { + __index = function(_, name) + local fn_ut, ba, ma, size, modules = index(name) + if not ba then + return fn_ut + elseif name == '_time' then + return fn_ut + elseif name == '_config' then + local fs_ma, fs_size = file.fscfg() + return {lfs_base = ba, lfs_mapped = ma, lfs_size = size, + fs_mapped = fs_ma, fs_size = fs_size} + elseif name == '_list' then + return modules + else + return nil + end + end, + + __newindex = function(_, name, value) + error("LFS is readonly. Invalid write to LFS." .. name, 2) + end, + + } + +local G=getfenv() +G.LFS = setmetatable(lfs_t,lfs_t) + +--[[------------------------------------------------------------------------------- + The second section adds the LFS to the require searchlist, so that you can + require a Lua module 'jean' in the LFS by simply doing require "jean". However + note that this is at the search entry following the FS searcher, so if you also + have jean.lc or jean.lua in SPIFFS, then this SPIFFS version will get loaded into + RAM instead of using. (Useful, for development). + + Note that if you want LFS to take a higher priority than SPIFFS, the use the [2] + slot for loaders. If you want to reverse these in your init.lua or interactively + for debugging, then use + + do local pl = package.loaders; pl[2],pl[4] = pl[4],pl[2]; end +---------------------------------------------------------------------------------]] + +package.loaders[4] = function(module) -- loader_flash + local fn, ba = index(module) + return ba and "Module not in LFS" or fn +end + +--[[------------------------------------------------------------------------------- + You can add any other initialisation here, for example a couple of the globals + are never used, so setting them to nil saves a couple of global entries +---------------------------------------------------------------------------------]] + +G.module = nil -- disable Lua 5.0 style modules to save RAM +package.seeall = nil diff --git a/lua_examples/lfs/dummy_strings.lua b/lua_examples/lfs/dummy_strings.lua new file mode 100644 index 0000000000..344893cc5d --- /dev/null +++ b/lua_examples/lfs/dummy_strings.lua @@ -0,0 +1,37 @@ +-- +-- File: LFS_dummy_strings.lua +--[[ + luac.cross -f generates a ROM string table which is part of the compiled LFS + image. This table includes all strings referenced in the loaded modules. + + If you want to preload other string constants, then one way to achieve this is + to include a dummy module in the LFS that references the strings that you want + to load. You never need to call this module; it's inclusion in the LFS image is + enough to add the strings to the ROM table. Your application can use any strings + in the ROM table without incuring any RAM or Lua Garbage Collector (LGC) + overhead. + + The local preload example is a useful starting point. However, if you call the + following code in your application during testing, then this will provide a + listing of the current RAM string table. + +do + local a=debug.getstrings'RAM' + for i =1, #a do a[i] = ('%q'):format(a[i]) end + print ('local preload='..table.concat(a,',')) +end + + This will exclude any strings already in the ROM table, so the output is the list + of putative strings that you should consider adding to LFS ROM table. + +---------------------------------------------------------------------------------]] + +local preload = "?.lc;?.lua", "/\n;\n?\n!\n-", "@init.lua", "_G", "_LOADED", +"_LOADLIB", "__add", "__call", "__concat", "__div", "__eq", "__gc", "__index", +"__le", "__len", "__lt", "__mod", "__mode", "__mul", "__newindex", "__pow", +"__sub", "__tostring", "__unm", "collectgarbage", "cpath", "debug", "file", +"file.obj", "file.vol", "flash", "getstrings", "index", "ipairs", "list", "loaded", +"loader", "loaders", "loadlib", "module", "net.tcpserver", "net.tcpsocket", +"net.udpsocket", "newproxy", "package", "pairs", "path", "preload", "reload", +"require", "seeall", "wdclr", "not enough memory", "sjson.decoder","sjson.encoder", +"tmr.timer" diff --git a/lua_examples/lfs/lfs_fragments.lua b/lua_examples/lfs/lfs_fragments.lua new file mode 100644 index 0000000000..83f70f2a12 --- /dev/null +++ b/lua_examples/lfs/lfs_fragments.lua @@ -0,0 +1,63 @@ +-- First time image boot to discover the confuration +-- +-- If you want to use absolute address LFS load or SPIFFS imaging, then boot the +-- image for the first time bare, that is without either LFS or SPIFFS preloaded +-- then enter the following commands interactively through the UART: +-- +do + local _,ma,fa=node.flashindex() + for n,v in pairs{LFS_MAPPED=ma, LFS_BASE=fa, SPIFFS_BASE=sa} do + print(('export %s=""0x%x"'):format(n, v) + end +end +-- +-- This will print out 3 hex constants: the absolute address used in the +-- 'luac.cross -a' options and the flash adresses of the LFS and SPIFFS. +-- +--[[ So you would need these commands to image your ESP module: +USB=/dev/ttyUSB0 # or whatever the device of your USB is +NODEMCU=~/nodemcu # The root of your NodeMCU file hierarchy +SRC=$NODEMCU/local/lua # your source directory for your LFS Lua files. +BIN=$NODEMCU/bin +ESPTOOL=$NODEMCU/tools/esptool.py + +$ESPTOOL --port $USB erase_flash # Do this is you are having load funnies +$ESPTOOL --port $USB --baud 460800 write_flash -fm dio 0x00000 \ + $BIN/0x00000.bin 0x10000 $BIN/0x10000.bin +# +# Now restart your module and use whatever your intective tool is to do the above +# cmds, so if this outputs 0x4027b000, -0x7b000, 0x100000 then you can do +# +$NODEMCU/luac.cross -a 0x4027b000 -o $BIN/0x7b000-flash.img $SRC/*.lua +$ESPTOOL --port $USB --baud 460800 write_flash -fm dio 0x7b000 \ + $BIN/0x7b000-flash.img +# and if you've setup a SPIFFS then +$ESPTOOL --port $USB --baud 460800 write_flash -fm dio 0x100000 \ + $BIN/0x100000-0x10000.img +# and now you are good to go +]] + + +----------------------------------------------------------------------------------- +-- +-- File: init.lua +-- +-- With the previous example you still need an init.lua to bootstrap the _init +-- module in LFS. Here is an example. It's a good idea either to use a timer +-- delay or a GPIO pin during development, so that you as developer can break into +-- the boot sequence if there is a problem with the _init bootstrap that is causing +-- a panic loop. Here is one example of how you might do this. You have a second +-- to inject tmr.stop(0) into UART0. Extend this dealy if your reactions can't +-- meet this. +-- +-- You also want to do autoload the LFS, for example by adding the following: +-- +if node.flashindex() == nil then + node.flashreload('flash.img') +end + +tmr.alarm(0, 1000, tmr.ALARM_SINGLE, + function() + local fi=node.flashindex; return pcall(fi and fi'_init') + end) + diff --git a/mkdocs.yml b/mkdocs.yml index 840b43cd33..e28e6386e1 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -24,13 +24,16 @@ pages: - Home: 'en/index.md' - Building the firmware: 'en/build.md' - Flashing the firmware: 'en/flash.md' - - Internal filesystem notes: 'en/spiffs.md' - - Filesystem on SD card: 'en/sdcard.md' - Uploading code: 'en/upload.md' - FAQs: - Lua Developer FAQ: 'en/lua-developer-faq.md' - Extension Developer FAQ: 'en/extn-developer-faq.md' - Hardware FAQ: 'en/hardware-faq.md' + - Whitepapers: + - Filesystem on SD card: 'en/sdcard.md' + - Internal filesystem notes: 'en/spiffs.md' + - Lua Compact Debug(LCD): 'en/lfs.md' + - Lua Flash Store(LFS): 'en/lcd.md' - Support: 'en/support.md' - Modules: - 'adc': 'en/modules/adc.md' diff --git a/tools/Makefile b/tools/Makefile index ab36e7569a..b00f7a32bc 100644 --- a/tools/Makefile +++ b/tools/Makefile @@ -3,9 +3,10 @@ # FSSOURCE ?= ../local/fs/ +LUASOURCE ?= ../local/lua/ FLASHSIZE ?= 4mb 32mb 8mb +FLASH_SW = -S SUBDIRS = -HOSTCC ?= gcc OBJDUMP = $(or $(shell which objdump),xtensa-lx106-elf-objdump) @@ -16,11 +17,24 @@ OBJDUMP = $(or $(shell which objdump),xtensa-lx106-elf-objdump) SPIFFSFILES ?= $(patsubst $(FSSOURCE)%,%,$(shell find $(FSSOURCE) -name '*' '!' -name .gitignore )) ################################################################# -# Get the filesize of /bin/0x10000.bin +# Get the filesize of /bin/0x10000.bin and SPIFFS sizing # -FLASH_USED_END = $$((0x`$(OBJDUMP) -t ../app/.output/eagle/debug/image/eagle.app.v6.out |grep _flash_used_end |cut -f1 -d" "` - 0x40200000)) +FLASH_FS_SIZE := $(shell $(CC) -E -dM - <../app/include/user_config.h | grep SPIFFS_MAX_FILESYSTEM_SIZE| cut -d ' ' -f 3) +ifneq ($(strip $(FLASH_FS_SIZE)),) +FLASHSIZE = $(shell printf "0x%x" $(FLASH_FS_SIZE)) +FLASH_SW = -c +endif + +FLASH_FS_LOC := $(shell $(CC) -E -dM - <../app/include/user_config.h | grep SPIFFS_FIXED_LOCATION| cut -d ' ' -f 3) +ifeq ($(strip $(FLASH_FS_LOC)),) +FLASH_FS_LOC := $(shell printf "0x%x" $$((0x$(shell $(OBJDUMP) -t ../app/.output/eagle/debug/image/eagle.app.v6.out |grep " _flash_used_end" |cut -f1 -d" ") - 0x40200000))) +else +FLASH_FS_LOC := $(shell printf "0x%x" $(FLASH_FS_LOC)) +endif + +LFSSOURCES := $(wildcard $(LUASOURCE)/*.lua) ############################################################# # Rules base @@ -29,6 +43,16 @@ FLASH_USED_END = $$((0x`$(OBJDUMP) -t ../app/.output/eagle/debug/image/eagle.app all: spiffsscript +.PHONY: TEST + +TEST: + @echo $(FLASHSIZE) + @echo $(FLASH_FS_SIZE) + @echo $(FLASH_FS_LOC) + @echo $(FLASH_USED_END) + +spiffsimg/spiffsimg: + .PHONY: spiffsimg .PHONY: spiffsimg/spiffsimg @@ -37,15 +61,23 @@ spiffsimg: spiffsimg/spiffsimg @echo Built spiffsimg in spiffsimg/spiffsimg spiffsimg/spiffsimg: - @$(MAKE) -C spiffsimg CC=$(HOSTCC) + @$(MAKE) -C spiffsimg -spiffsscript: remove-image spiffsimg/spiffsimg +spiffsscript: remove-image LFSimage spiffsimg/spiffsimg rm -f ./spiffsimg/spiffs.lst - echo "" >> ./spiffsimg/spiffs.lst + @echo "" >> ./spiffsimg/spiffs.lst @$(foreach f, $(SPIFFSFILES), echo "import $(FSSOURCE)$(f) $(f)" >> ./spiffsimg/spiffs.lst ;) - @$(foreach sz, $(FLASHSIZE), spiffsimg/spiffsimg -U $(FLASH_USED_END) -o ../bin/spiffs-$(sz).dat -f ../bin/0x%x-$(sz).bin -S $(sz) -r ./spiffsimg/spiffs.lst -d; ) + $(foreach sz, $(FLASHSIZE), spiffsimg/spiffsimg -f ../bin/0x%x-$(sz).img $(FLASH_SW) $(sz) -U $(FLASH_FS_LOC) -r ./spiffsimg/spiffs.lst -d; ) @$(foreach sz, $(FLASHSIZE), if [ -r ../bin/spiffs-$(sz).dat ]; then echo Built $$(cat ../bin/spiffs-$(sz).dat)-$(sz).bin; fi; ) - + +ifneq ($(LFSSOURCES),) +LFSimage: $(LFSSOURCES) + ../luac.cross -f -o $(FSSOURCE)flash.img $(LFSSOURCES) +else +LFSimage: + rm -f $(FSSOURCE)flash.img +endif + remove-image: $(foreach sz, $(FLASHSIZE), if [ -r ../bin/spiffs-$(sz).dat ]; then rm -f ../bin/$$(cat ../bin/spiffs-$(sz).dat)-$(sz).bin; fi; ) rm -f ../bin/spiffs*.dat diff --git a/tools/build.lua b/tools/build.lua deleted file mode 100644 index 0d93dea2bb..0000000000 --- a/tools/build.lua +++ /dev/null @@ -1,848 +0,0 @@ --- eLua build system - -module( ..., package.seeall ) - -local lfs = require "lfs" -local sf = string.format -utils = require "tools.utils" - -------------------------------------------------------------------------------- --- Various helpers - --- Return the time of the last modification of the file -local function get_ftime( path ) - local t = lfs.attributes( path, 'modification' ) - return t or -1 -end - --- Check if a given target name is phony -local function is_phony( target ) - return target:find( "#phony" ) == 1 -end - --- Return a string with $(key) replaced with 'value' -local function expand_key( s, key, value ) - if not value then return s end - local fmt = sf( "%%$%%(%s%%)", key ) - return ( s:gsub( fmt, value ) ) -end - --- Return a target name considering phony targets -local function get_target_name( s ) - if not is_phony( s ) then return s end -end - --- 'Liniarize' a file name by replacing its path separators indicators with '_' -local function linearize_fname( s ) - return ( s:gsub( "[\\/]", "__" ) ) -end - --- Helper: transform a table into a string if needed -local function table_to_string( t ) - if not t then return nil end - if type( t ) == "table" then t = table.concat( t, " " ) end - return t -end - --- Helper: return the extended type of an object (takes into account __type) -local function exttype( o ) - local t = type( o ) - if t == "table" and o.__type then t = o:__type() end - return t -end - ---------------------------------------- --- Table utils --- (from http://lua-users.org/wiki/TableUtils) - -function table.val_to_str( v ) - if "string" == type( v ) then - v = string.gsub( v, "\n", "\\n" ) - if string.match( string.gsub(v,"[^'\"]",""), '^"+$' ) then - return "'" .. v .. "'" - end - return '"' .. string.gsub(v,'"', '\\"' ) .. '"' - else - return "table" == type( v ) and table.tostring( v ) or tostring( v ) - end -end - -function table.key_to_str ( k ) - if "string" == type( k ) and string.match( k, "^[_%a][_%a%d]*$" ) then - return k - else - return "[" .. table.val_to_str( k ) .. "]" - end -end - -function table.tostring( tbl ) - local result, done = {}, {} - for k, v in ipairs( tbl ) do - table.insert( result, table.val_to_str( v ) ) - done[ k ] = true - end - for k, v in pairs( tbl ) do - if not done[ k ] then - table.insert( result, - table.key_to_str( k ) .. "=" .. table.val_to_str( v ) ) - end - end - return "{" .. table.concat( result, "," ) .. "}" -end - -------------------------------------------------------------------------------- --- Dummy 'builder': simply checks the date of a file - -local _fbuilder = {} - -_fbuilder.new = function( target, dep ) - local self = {} - setmetatable( self, { __index = _fbuilder } ) - self.target = target - self.dep = dep - return self -end - -_fbuilder.build = function( self ) - -- Doesn't build anything but returns 'true' if the dependency is newer than - -- the target - if is_phony( self.target ) then - return true - else - return get_ftime( self.dep ) > get_ftime( self.target ) - end -end - -_fbuilder.target_name = function( self ) - return get_target_name( self.dep ) -end - --- Object type -_fbuilder.__type = function() - return "_fbuilder" -end - -------------------------------------------------------------------------------- --- Target object - -local _target = {} - -_target.new = function( target, dep, command, builder, ttype ) - local self = {} - setmetatable( self, { __index = _target } ) - self.target = target - self.command = command - self.builder = builder - builder:register_target( target, self ) - self:set_dependencies( dep ) - self.dep = self:_build_dependencies( self.origdep ) - self.dont_clean = false - self.can_substitute_cmdline = false - self._force_rebuild = #self.dep == 0 - builder.runlist[ target ] = false - self:set_type( ttype ) - return self -end - --- Set dependencies as a string; actual dependencies are computed by _build_dependencies --- (below) when 'build' is called -_target.set_dependencies = function( self, dep ) - self.origdep = dep -end - --- Set the target type --- This is only for displaying actions -_target.set_type = function( self, ttype ) - local atable = { comp = { "[COMPILE]", 'blue' } , dep = { "[DEPENDS]", 'magenta' }, link = { "[LINK]", 'yellow' }, asm = { "[ASM]", 'white' } } - local tdata = atable[ ttype ] - if not tdata then - self.dispstr = is_phony( self.target ) and "[PHONY]" or "[TARGET]" - self.dispcol = 'green' - else - self.dispstr = tdata[ 1 ] - self.dispcol = tdata[ 2 ] - end -end - --- Set dependencies --- This uses a proxy table and returns string deps dynamically according --- to the targets currently registered in the builder -_target._build_dependencies = function( self, dep ) - -- Step 1: start with an array - if type( dep ) == "string" then dep = utils.string_to_table( dep ) end - -- Step 2: linearize "dep" array keeping targets - local filter = function( e ) - local t = exttype( e ) - return t ~= "_ftarget" and t ~= "_target" - end - dep = utils.linearize_array( dep, filter ) - -- Step 3: strings are turned into _fbuilder objects if not found as targets; - -- otherwise the corresponding target object is used - for i = 1, #dep do - if type( dep[ i ] ) == 'string' then - local t = self.builder:get_registered_target( dep[ i ] ) - dep[ i ] = t or _fbuilder.new( self.target, dep[ i ] ) - end - end - return dep -end - --- Set pre-build function -_target.set_pre_build_function = function( self, f ) - self._pre_build_function = f -end - --- Set post-build function -_target.set_post_build_function = function( self, f ) - self._post_build_function = f -end - --- Force rebuild -_target.force_rebuild = function( self, flag ) - self._force_rebuild = flag -end - --- Set additional arguments to send to the builder function if it is a callable -_target.set_target_args = function( self, args ) - self._target_args = args -end - --- Function to execute in clean mode -_target._cleaner = function( target, deps, tobj, disp_mode ) - -- Clean the main target if it is not a phony target - local dprint = function( ... ) - if disp_mode ~= "minimal" then print( ... ) end - end - if not is_phony( target ) then - if tobj.dont_clean then - dprint( sf( "[builder] Target '%s' will not be deleted", target ) ) - return 0 - end - if disp_mode ~= "minimal" then io.write( sf( "[builder] Removing %s ... ", target ) ) end - if os.remove( target ) then dprint "done." else dprint "failed!" end - end - return 0 -end - --- Build the given target -_target.build = function( self ) - if self.builder.runlist[ self.target ] then return end - local docmd = self:target_name() and lfs.attributes( self:target_name(), "mode" ) ~= "file" - docmd = docmd or self.builder.global_force_rebuild - local initdocmd = docmd - self.dep = self:_build_dependencies( self.origdep ) - local depends, dep, previnit = '', self.dep, self.origdep - -- Iterate through all dependencies, execute each one in turn - local deprunner = function() - for i = 1, #dep do - local res = dep[ i ]:build() - docmd = docmd or res - local t = dep[ i ]:target_name() - if exttype( dep[ i ] ) == "_target" and t and not is_phony( self.target ) then - docmd = docmd or get_ftime( t ) > get_ftime( self.target ) - end - if t then depends = depends .. t .. " " end - end - end - deprunner() - -- Execute the preb-build function if needed - if self._pre_build_function then self._pre_build_function( self, docmd ) end - -- If the dependencies changed as a result of running the pre-build function - -- run through them again - if previnit ~= self.origdep then - self.dep = self:_build_dependencies( self.origdep ) - depends, dep, docmd = '', self.dep, initdocmd - deprunner() - end - -- If at least one dependency is new rebuild the target - docmd = docmd or self._force_rebuild or self.builder.clean_mode - local keep_flag = true - if docmd and self.command then - if self.builder.disp_mode ~= 'all' and self.builder.disp_mode ~= "minimal" and not self.builder.clean_mode then - io.write( utils.col_funcs[ self.dispcol ]( self.dispstr ) .. " " ) - end - local cmd, code = self.command - if self.builder.clean_mode then cmd = _target._cleaner end - if type( cmd ) == 'string' then - cmd = expand_key( cmd, "TARGET", self.target ) - cmd = expand_key( cmd, "DEPENDS", depends ) - cmd = expand_key( cmd, "FIRST", dep[ 1 ]:target_name() ) - if self.builder.disp_mode == 'all' then - print( cmd ) - elseif self.builder.disp_mode ~= "minimal" then - print( self.target ) - end - code = self:execute( cmd ) - else - if not self.builder.clean_mode and self.builder.disp_mode ~= "all" and self.builder.disp_mode ~= "minimal" then - print( self.target ) - end - code = cmd( self.target, self.dep, self.builder.clean_mode and self or self._target_args, self.builder.disp_mode ) - if code == 1 then -- this means "mark target as 'not executed'" - keep_flag = false - code = 0 - end - end - if code ~= 0 then - print( utils.col_red( "[builder] Error building target" ) ) - if self.builder.disp_mode ~= 'all' and type( cmd ) == "string" then - print( utils.col_red( "[builder] Last executed command was: " ) ) - print( cmd ) - end - os.exit( 1 ) - end - end - -- Execute the post-build function if needed - if self._post_build_function then self._post_build_function( self, docmd ) end - -- Marked target as "already ran" so it won't run again - self.builder.runlist[ self.target ] = true - return docmd and keep_flag -end - --- Return the actual target name (taking into account phony targets) -_target.target_name = function( self ) - return get_target_name( self.target ) -end - --- Restrict cleaning this target -_target.prevent_clean = function( self, flag ) - self.dont_clean = flag -end - --- Object type -_target.__type = function() - return "_target" -end - -_target.execute = function( self, cmd ) - local code - if utils.is_windows() and #cmd > 8190 and self.can_substitute_cmdline then - -- Avoid cmd's maximum command line length limitation - local t = cmd:find( " " ) - f = io.open( "tmpcmdline", "w" ) - local rest = cmd:sub( t + 1 ) - f:write( ( rest:gsub( "\\", "/" ) ) ) - f:close() - cmd = cmd:sub( 1, t - 1 ) .. " @tmpcmdline" - end - local code = os.execute( cmd ) - os.remove( "tmpcmdline" ) - return code -end - -_target.set_substitute_cmdline = function( self, flag ) - self.can_substitute_cmdline = flag -end - -------------------------------------------------------------------------------- --- Builder public interface - -builder = { KEEP_DIR = 0, BUILD_DIR_LINEARIZED = 1 } - ---------------------------------------- --- Initialization and option handling - --- Create a new builder object with the output in 'build_dir' and with the --- specified compile, dependencies and link command -builder.new = function( build_dir ) - self = {} - setmetatable( self, { __index = builder } ) - self.build_dir = build_dir or ".build" - self.exe_extension = utils.is_windows() and "exe" or "" - self.clean_mode = false - self.opts = utils.options_handler() - self.args = {} - self.user_args = {} - self.build_mode = self.KEEP_DIR - self.targets = {} - self.targetargs = {} - self._tlist = {} - self.runlist = {} - self.disp_mode = 'all' - self.cmdline_macros = {} - self.c_targets = {} - self.preprocess_mode = false - self.asm_mode = false - return self -end - --- Helper: create the build output directory -builder._create_build_dir = function( self ) - if self.build_dir_created then return end - if self.build_mode ~= self.KEEP_DIR then - -- Create builds directory if needed - local mode = lfs.attributes( self.build_dir, "mode" ) - if not mode or mode ~= "directory" then - if not utils.full_mkdir( self.build_dir ) then - print( "[builder] Unable to create directory " .. self.build_dir ) - os.exit( 1 ) - end - end - end - self.build_dir_created = true -end - --- Add an options to the builder -builder.add_option = function( self, name, help, default, data ) - self.opts:add_option( name, help, default, data ) -end - --- Initialize builder from the given command line -builder.init = function( self, args ) - -- Add the default options - local opts = self.opts - opts:add_option( "build_mode", 'choose location of the object files', self.KEEP_DIR, - { keep_dir = self.KEEP_DIR, build_dir_linearized = self.BUILD_DIR_LINEARIZED } ) - opts:add_option( "build_dir", 'choose build directory', self.build_dir ) - opts:add_option( "disp_mode", 'set builder display mode', 'summary', { 'all', 'summary', 'minimal' } ) - -- Apply default values to all options - for i = 1, opts:get_num_opts() do - local o = opts:get_option( i ) - self.args[ o.name:upper() ] = o.default - end - -- Read and interpret command line - for i = 1, #args do - local a = args[ i ] - if a:upper() == "-C" then -- clean option (-c) - self.clean_mode = true - elseif a:upper() == '-H' then -- help option (-h) - self:_show_help() - os.exit( 1 ) - elseif a:upper() == "-E" then -- preprocess - self.preprocess_mode = true - elseif a:upper() == "-S" then -- generate assembler - self.asm_mode = true - elseif a:find( '-D' ) == 1 and #a > 2 then -- this is a macro definition that will be auomatically added to the compiler flags - table.insert( self.cmdline_macros, a:sub( 3 ) ) - elseif a:find( '=' ) then -- builder argument (key=value) - local k, v = opts:handle_arg( a ) - if not k then - self:_show_help() - os.exit( 1 ) - end - self.args[ k:upper() ] = v - self.user_args[ k:upper() ] = true - else -- this must be the target name / target arguments - if self.targetname == nil then - self.targetname = a - else - table.insert( self.targetargs, a ) - end - end - end - -- Read back the default options - self.build_mode = self.args.BUILD_MODE - self.build_dir = self.args.BUILD_DIR - self.disp_mode = self.args.DISP_MODE -end - --- Return the value of the option with the given name -builder.get_option = function( self, optname ) - return self.args[ optname:upper() ] -end - --- Returns true if the given option was specified by the user on the command line, false otherwise -builder.is_user_option = function( self, optname ) - return self.user_args[ optname:upper() ] -end - --- Show builder help -builder._show_help = function( self ) - print( "[builder] Valid options:" ) - print( " -h: help (this text)" ) - print( " -c: clean target" ) - print( " -E: generate preprocessed output for single file targets" ) - print( " -S: generate assembler output for single file targets" ) - self.opts:show_help() -end - ---------------------------------------- --- Builder configuration - --- Set the compile command -builder.set_compile_cmd = function( self, cmd ) - self.comp_cmd = cmd -end - --- Set the link command -builder.set_link_cmd = function( self, cmd ) - self.link_cmd = cmd -end - --- Set the assembler command -builder.set_asm_cmd = function( self, cmd ) - self._asm_cmd = cmd -end - --- Set (actually force) the object file extension -builder.set_object_extension = function( self, ext ) - self.obj_extension = ext -end - --- Set (actually force) the executable file extension -builder.set_exe_extension = function( self, ext ) - self.exe_extension = ext -end - --- Set the clean mode -builder.set_clean_mode = function( self, isclean ) - self.clean_mode = isclean -end - --- Sets the build mode -builder.set_build_mode = function( self, mode ) - self.build_mode = mode -end - --- Set the build directory -builder.set_build_dir = function( self, dir ) - if self.build_dir_created then - print "[builder] Error: build directory already created" - os.exit( 1 ) - end - self.build_dir = dir - self:_create_build_dir() -end - --- Return the current build directory -builder.get_build_dir = function( self ) - return self.build_dir -end - --- Return the target arguments -builder.get_target_args = function( self ) - return self.targetargs -end - --- Set a specific dependency generation command for the assembler --- Pass 'false' to skip dependency generation for assembler files -builder.set_asm_dep_cmd = function( self, asm_dep_cmd ) - self.asm_dep_cmd = asm_dep_cmd -end - --- Set a specific dependency generation command for the compiler --- Pass 'false' to skip dependency generation for C files -builder.set_c_dep_cmd = function( self, c_dep_cmd ) - self.c_dep_cmd = c_dep_cmd -end - --- Save the builder configuration for a given component to a string -builder._config_to_string = function( self, what ) - local ctable = {} - local state_fields - if what == 'comp' then - state_fields = { 'comp_cmd', '_asm_cmd', 'c_dep_cmd', 'asm_dep_cmd', 'obj_extension' } - elseif what == 'link' then - state_fields = { 'link_cmd' } - else - print( sf( "Invalid argument '%s' to _config_to_string", what ) ) - os.exit( 1 ) - end - utils.foreach( state_fields, function( k, v ) ctable[ v ] = self[ v ] end ) - return table.tostring( ctable ) -end - --- Check the configuration of the given component against the previous one --- Return true if the configuration has changed -builder._compare_config = function( self, what ) - local res = false - local crtstate = self:_config_to_string( what ) - if not self.clean_mode then - local fconf = io.open( self.build_dir .. utils.dir_sep .. ".builddata." .. what, "rb" ) - if fconf then - local oldstate = fconf:read( "*a" ) - fconf:close() - if oldstate:lower() ~= crtstate:lower() then res = true end - end - end - -- Write state to build dir - fconf = io.open( self.build_dir .. utils.dir_sep .. ".builddata." .. what, "wb" ) - if fconf then - fconf:write( self:_config_to_string( what ) ) - fconf:close() - end - return res -end - --- Sets the way commands are displayed -builder.set_disp_mode = function( self, mode ) - mode = mode:lower() - if mode ~= 'all' and mode ~= 'summary' and mode ~= "minimal" then - print( sf( "[builder] Invalid display mode '%s'", mode ) ) - os.exit( 1 ) - end - self.disp_mode = mode -end - ---------------------------------------- --- Command line builders - --- Internal helper -builder._generic_cmd = function( self, args ) - local compcmd = args.compiler or "gcc" - compcmd = compcmd .. " " - local flags = type( args.flags ) == 'table' and table_to_string( utils.linearize_array( args.flags ) ) or args.flags - local defines = type( args.defines ) == 'table' and table_to_string( utils.linearize_array( args.defines ) ) or args.defines - local includes = type( args.includes ) == 'table' and table_to_string( utils.linearize_array( args.includes ) ) or args.includes - local comptype = table_to_string( args.comptype ) or "-c" - compcmd = compcmd .. utils.prepend_string( defines, "-D" ) - compcmd = compcmd .. utils.prepend_string( includes, "-I" ) - return compcmd .. flags .. " " .. comptype .. " -o $(TARGET) $(FIRST)" -end - --- Return a compile command based on the specified args -builder.compile_cmd = function( self, args ) - args.defines = { args.defines, self.cmdline_macros } - if self.preprocess_mode then - args.comptype = "-E" - elseif self.asm_mode then - args.comptype = "-S" - else - args.comptype = "-c" - end - return self:_generic_cmd( args ) -end - --- Return an assembler command based on the specified args -builder.asm_cmd = function( self, args ) - args.defines = { args.defines, self.cmdline_macros } - args.compiler = args.assembler - args.comptype = self.preprocess_mode and "-E" or "-c" - return self:_generic_cmd( args ) -end - --- Return a link command based on the specified args -builder.link_cmd = function( self, args ) - local flags = type( args.flags ) == 'table' and table_to_string( utils.linearize_array( args.flags ) ) or args.flags - local libraries = type( args.libraries ) == 'table' and table_to_string( utils.linearize_array( args.libraries ) ) or args.libraries - local linkcmd = args.linker or "gcc" - linkcmd = linkcmd .. " " .. flags .. " -o $(TARGET) $(DEPENDS)" - linkcmd = linkcmd .. " " .. utils.prepend_string( libraries, "-l" ) - return linkcmd -end - ---------------------------------------- --- Target handling - --- Create a return a new C to object target -builder.c_target = function( self, target, deps, comp_cmd ) - return _target.new( target, deps, comp_cmd or self.comp_cmd, self, 'comp' ) -end - --- Create a return a new ASM to object target -builder.asm_target = function( self, target, deps, asm_cmd ) - return _target.new( target, deps, asm_cmd or self._asm_cmd, self, 'asm' ) -end - --- Return the name of a dependency file name corresponding to a C source -builder.get_dep_filename = function( self, srcname ) - return utils.replace_extension( self.build_dir .. utils.dir_sep .. linearize_fname( srcname ), "d" ) -end - --- Create a return a new C dependency target -builder.dep_target = function( self, dep, depdeps, dep_cmd ) - local depname = self:get_dep_filename( dep ) - return _target.new( depname, depdeps, dep_cmd, self, 'dep' ) -end - --- Create and return a new link target -builder.link_target = function( self, out, dep, link_cmd ) - local path, ext = utils.split_ext( out ) - if not ext and self.exe_extension and #self.exe_extension > 0 then - out = out .. self.exe_extension - end - local t = _target.new( out, dep, link_cmd or self.link_cmd, self, 'link' ) - if self:_compare_config( 'link' ) then t:force_rebuild( true ) end - t:set_substitute_cmdline( true ) - return t -end - --- Create and return a new generic target -builder.target = function( self, dest_target, deps, cmd ) - return _target.new( dest_target, deps, cmd, self ) -end - --- Register a target (called from _target.new) -builder.register_target = function( self, name, obj ) - self._tlist[ name:gsub( "\\", "/" ) ] = obj -end - --- Returns a registered target (nil if not found) -builder.get_registered_target = function( self, name ) - return self._tlist[ name:gsub( "\\", "/" ) ] -end - ---------------------------------------- --- Actual building functions - --- Return the object name corresponding to a source file name -builder.obj_name = function( self, name, ext ) - local r = ext or self.obj_extension - if not r then - r = utils.is_windows() and "obj" or "o" - end - local objname = utils.replace_extension( name, r ) - -- KEEP_DIR: object file in the same directory as source file - -- BUILD_DIR_LINEARIZED: object file in the build directory, linearized filename - if self.build_mode == self.KEEP_DIR then - return objname - elseif self.build_mode == self.BUILD_DIR_LINEARIZED then - return self.build_dir .. utils.dir_sep .. linearize_fname( objname ) - end -end - --- Read and interpret dependencies for each file specified in "ftable" --- "ftable" is either a space-separated string with all the source files or an array -builder.read_depends = function( self, ftable ) - if type( ftable ) == 'string' then ftable = utils.string_to_table( ftable ) end - -- Read dependency data - local dtable = {} - for i = 1, #ftable do - local f = io.open( self:get_dep_filename( ftable[ i ] ), "rb" ) - local lines = ftable[ i ] - if f then - lines = f:read( "*a" ) - f:close() - lines = lines:gsub( "\n", " " ):gsub( "\\%s+", " " ):gsub( "%s+", " " ):gsub( "^.-: (.*)", "%1" ) - end - dtable[ ftable[ i ] ] = lines - end - return dtable -end - --- Create and return compile targets for the given sources -builder.create_compile_targets = function( self, ftable, res ) - if type( ftable ) == 'string' then ftable = utils.string_to_table( ftable ) end - res = res or {} - ccmd, oname = "-c", "o" - if self.preprocess_mode then - ccmd, oname = '-E', "pre" - elseif self.asm_mode then - ccmd, oname = '-S', 's' - end - -- Build dependencies for all targets - for i = 1, #ftable do - local isasm = ftable[ i ]:find( "%.c$" ) == nil - -- Skip assembler targets if 'asm_dep_cmd' is set to 'false' - -- Skip C targets if 'c_dep_cmd' is set to 'false' - local skip = isasm and self.asm_dep_cmd == false - skip = skip or ( not isasm and self.c_dep_cmd == false ) - local deps = self:get_dep_filename( ftable[ i ] ) - local target - if not isasm then - local depcmd = skip and self.comp_cmd or ( self.c_dep_cmd or self.comp_cmd:gsub( ccmd .. " ", sf( ccmd .. " -MD -MF %s ", deps ) ) ) - target = self:c_target( self:obj_name( ftable[ i ], oname ), { self:get_registered_target( deps ) or ftable[ i ] }, depcmd ) - else - local depcmd = skip and self._asm_cmd or ( self.asm_dep_cmd or self._asm_cmd:gsub( ccmd .. " ", sf( ccmd .. " -MD -MF %s ", deps ) ) ) - target = self:asm_target( self:obj_name( ftable[ i ], oname ), { self:get_registered_target( deps ) or ftable[ i ] }, depcmd ) - end - -- Pre build step: replace dependencies with the ones from the compiler generated dependency file - local dprint = function( ... ) if self.disp_mode ~= "minimal" then print( ... ) end end - if not skip then - target:set_pre_build_function( function( t, _ ) - if not self.clean_mode then - local fres = self:read_depends( ftable[ i ] ) - local fdeps = fres[ ftable[ i ] ] - if #fdeps:gsub( "%s+", "" ) == 0 then fdeps = ftable[ i ] end - t:set_dependencies( fdeps ) - else - if self.disp_mode ~= "minimal" then io.write( sf( "[builder] Removing %s ... ", deps ) ) end - if os.remove( deps ) then dprint "done." else dprint "failed!" end - end - end ) - end - target.srcname = ftable[ i ] - -- TODO: check clean mode? - if not isasm then self.c_targets[ #self.c_targets + 1 ] = target end - table.insert( res, target ) - end - return res -end - --- Add a target to the list of builder targets -builder.add_target = function( self, target, help, alias ) - self.targets[ target.target ] = { target = target, help = help } - alias = alias or {} - for _, v in ipairs( alias ) do - self.targets[ v ] = { target = target, help = help } - end - return target -end - --- Make a target the default one -builder.default = function( self, target ) - self.deftarget = target.target - self.targets.default = { target = target, help = "default target" } -end - --- Build everything -builder.build = function( self, target ) - local t = self.targetname or self.deftarget - if not t then - print( utils.col_red( "[builder] Error: build target not specified" ) ) - os.exit( 1 ) - end - local trg - -- Look for single targets (C source files) - for _, ct in pairs( self.c_targets ) do - if ct.srcname == t then - trg = ct - break - end - end - if not trg then - if not self.targets[ t ] then - print( sf( "[builder] Error: target '%s' not found", t ) ) - print( "Available targets: " ) - print( " all source files" ) - for k, v in pairs( self.targets ) do - if not is_phony( k ) then - print( sf( " %s - %s", k, v.help or "(no help available)" ) ) - end - end - if self.deftarget and not is_phony( self.deftarget ) then - print( sf( "Default target is '%s'", self.deftarget ) ) - end - os.exit( 1 ) - else - if self.preprocess_mode or self.asm_mode then - print( "[builder] Error: preprocess (-E) or asm (-S) works only with single file targets." ) - os.exit( 1 ) - end - trg = self.targets[ t ].target - end - end - self:_create_build_dir() - -- At this point check if we have a change in the state that would require a rebuild - if self:_compare_config( 'comp' ) then - print( utils.col_yellow( "[builder] Forcing rebuild due to configuration change." ) ) - self.global_force_rebuild = true - else - self.global_force_rebuild = false - end - -- Do the actual build - local res = trg:build() - if not res then print( utils.col_yellow( sf( '[builder] %s: up to date', t ) ) ) end - if self.clean_mode then - os.remove( self.build_dir .. utils.dir_sep .. ".builddata.comp" ) - os.remove( self.build_dir .. utils.dir_sep .. ".builddata.link" ) - end - print( utils.col_yellow( "[builder] Done building target." ) ) - return res -end - --- Create dependencies, create object files, link final object -builder.make_exe_target = function( self, target, file_list ) - local odeps = self:create_compile_targets( file_list ) - local exetarget = self:link_target( target, odeps ) - self:default( self:add_target( exetarget ) ) - return exetarget -end - -------------------------------------------------------------------------------- --- Other exported functions - -function new_builder( build_dir ) - return builder.new( build_dir ) -end - diff --git a/tools/cross-lua.lua b/tools/cross-lua.lua deleted file mode 100644 index a18bbeb4f0..0000000000 --- a/tools/cross-lua.lua +++ /dev/null @@ -1,42 +0,0 @@ -local args = { ... } -local b = require "tools.build" -local builder = b.new_builder( ".build/cross-lua" ) -local utils = b.utils -local sf = string.format - -if not (_VERSION == "Lua 5.1" and pcall(require,"lfs")) then - print [[ - -cross_lua.lua must be run within Lua 5.1 and it requires the Lua Filesystem to be installed. -On most *nix distrubitions youwill find a packages lua-5.1 and lua-filesystem, or -alternalively you can install lua-rocks and use the Rocks package manager to install lfs. -]] - os.exit(1) -end -builder:init( args ) -builder:set_build_mode( builder.BUILD_DIR_LINEARIZED ) -local output = 'luac.cross' -local cdefs = '-DLUA_CROSS_COMPILER -Ddbg_printf=printf' - --- Lua source files and include path -local lua_files = [[ - lapi.c lauxlib.c lbaselib.c lcode.c ldblib.c ldebug.c ldo.c ldump.c - lfunc.c lgc.c llex.c lmathlib.c lmem.c loadlib.c lobject.c lopcodes.c - lparser.c lrotable.c lstate.c lstring.c lstrlib.c ltable.c ltablib.c - ltm.c lundump.c lvm.c lzio.c - luac_cross/luac.c luac_cross/loslib.c luac_cross/print.c - ../modules/linit.c - ../libc/c_stdlib.c - ]] -lua_files = lua_files:gsub( "\n" , "" ) -local lua_full_files = utils.prepend_path( lua_files, "app/lua" ) -local local_include = "-Iapp/include -Iinclude -Iapp/lua" - --- Compiler/linker options -builder:set_compile_cmd( sf( "gcc -O2 %s -Wall %s -c $(FIRST) -o $(TARGET)", local_include, cdefs ) ) -builder:set_link_cmd( "gcc -o $(TARGET) $(DEPENDS) -lm" ) - --- Build everything -builder:make_exe_target( output, lua_full_files ) -builder:build() - diff --git a/tools/spiffsimg/Makefile b/tools/spiffsimg/Makefile index 8656d114f2..fbf73ea2d2 100644 --- a/tools/spiffsimg/Makefile +++ b/tools/spiffsimg/Makefile @@ -1,3 +1,5 @@ +CC =gcc + SRCS=\ main.c \ ../../app/spiffs/spiffs_cache.c ../../app/spiffs/spiffs_check.c ../../app/spiffs/spiffs_gc.c ../../app/spiffs/spiffs_hydrogen.c ../../app/spiffs/spiffs_nucleus.c diff --git a/tools/utils.lua b/tools/utils.lua deleted file mode 100644 index a952751115..0000000000 --- a/tools/utils.lua +++ /dev/null @@ -1,425 +0,0 @@ --- Generic utility functions - -module( ..., package.seeall ) - -local lfs = require "lfs" -local sf = string.format --- Taken from Lake -dir_sep = package.config:sub( 1, 1 ) -is_os_windows = dir_sep == '\\' - --- Converts a string with items separated by 'sep' into a table -string_to_table = function( s, sep ) - if type( s ) ~= "string" then return end - sep = sep or ' ' - if s:sub( -1, -1 ) ~= sep then s = s .. sep end - s = s:gsub( sf( "^%s*", sep ), "" ) - local t = {} - local fmt = sf( "(.-)%s+", sep ) - for w in s:gmatch( fmt ) do table.insert( t, w ) end - return t -end - --- Split a file name into 'path part' and 'extension part' -split_ext = function( s ) - local pos - for i = #s, 1, -1 do - if s:sub( i, i ) == "." then - pos = i - break - end - end - if not pos or s:find( dir_sep, pos + 1 ) then return s end - return s:sub( 1, pos - 1 ), s:sub( pos ) -end - --- Replace the extension of a given file name -replace_extension = function( s, newext ) - local p, e = split_ext( s ) - if e then - if newext and #newext > 0 then - s = p .. "." .. newext - else - s = p - end - end - return s -end - --- Return 'true' if building from Windows, false otherwise -is_windows = function() - return is_os_windows -end - --- Prepend each component of a 'pat'-separated string with 'prefix' -prepend_string = function( s, prefix, pat ) - if not s or #s == 0 then return "" end - pat = pat or ' ' - local res = '' - local st = string_to_table( s, pat ) - foreach( st, function( k, v ) res = res .. prefix .. v .. " " end ) - return res -end - --- Like above, but consider 'prefix' a path -prepend_path = function( s, prefix, pat ) - return prepend_string( s, prefix .. dir_sep, pat ) -end - --- full mkdir: create all the paths needed for a multipath -full_mkdir = function( path ) - local ptables = string_to_table( path, dir_sep ) - local p, res = '' - for i = 1, #ptables do - p = ( i ~= 1 and p .. dir_sep or p ) .. ptables[ i ] - res = lfs.mkdir( p ) - end - return res -end - --- Concatenate the given paths to form a complete path -concat_path = function( paths ) - return table.concat( paths, dir_sep ) -end - --- Return true if the given array contains the given element, false otherwise -array_element_index = function( arr, element ) - for i = 1, #arr do - if arr[ i ] == element then return i end - end -end - --- Linearize an array with (possibly) embedded arrays into a simple array -_linearize_array = function( arr, res, filter ) - if type( arr ) ~= "table" then return end - for i = 1, #arr do - local e = arr[ i ] - if type( e ) == 'table' and filter( e ) then - _linearize_array( e, res, filter ) - else - table.insert( res, e ) - end - end -end - -linearize_array = function( arr, filter ) - local res = {} - filter = filter or function( v ) return true end - _linearize_array( arr, res, filter ) - return res -end - --- Return an array with the keys of a table -table_keys = function( t ) - local keys = {} - foreach( t, function( k, v ) table.insert( keys, k ) end ) - return keys -end - --- Return an array with the values of a table -table_values = function( t ) - local vals = {} - foreach( t, function( k, v ) table.insert( vals, v ) end ) - return vals -end - --- Returns true if 'path' is a regular file, false otherwise -is_file = function( path ) - return lfs.attributes( path, "mode" ) == "file" -end - --- Returns true if 'path' is a directory, false otherwise -is_dir = function( path ) - return lfs.attributes( path, "mode" ) == "directory" -end - --- Return a list of files in the given directory matching a given mask -get_files = function( path, mask, norec, level ) - local t = '' - level = level or 0 - for f in lfs.dir( path ) do - local fname = path .. dir_sep .. f - if lfs.attributes( fname, "mode" ) == "file" then - local include - if type( mask ) == "string" then - include = fname:find( mask ) - else - include = mask( fname ) - end - if include then t = t .. ' ' .. fname end - elseif lfs.attributes( fname, "mode" ) == "directory" and not fname:find( "%.+$" ) and not norec then - t = t .. " " .. get_files( fname, mask, norec, level + 1 ) - end - end - return level > 0 and t or t:gsub( "^%s+", "" ) -end - --- Check if the given command can be executed properly -check_command = function( cmd ) - local res = os.execute( cmd .. " > .build.temp 2>&1" ) - os.remove( ".build.temp" ) - return res -end - --- Execute a command and capture output --- From: http://stackoverflow.com/a/326715/105950 -exec_capture = function( cmd, raw ) - local f = assert(io.popen(cmd, 'r')) - local s = assert(f:read('*a')) - f:close() - if raw then return s end - s = string.gsub(s, '^%s+', '') - s = string.gsub(s, '%s+$', '') - s = string.gsub(s, '[\n\r]+', ' ') - return s -end - --- Execute the given command for each value in a table -foreach = function ( t, cmd ) - if type( t ) ~= "table" then return end - for k, v in pairs( t ) do cmd( k, v ) end -end - --- Generate header with the given #defines, return result as string -gen_header_string = function( name, defines ) - local s = "// eLua " .. name:lower() .. " definition\n\n" - s = s .. "#ifndef __" .. name:upper() .. "_H__\n" - s = s .. "#define __" .. name:upper() .. "_H__\n\n" - - for key,value in pairs(defines) do - s = s .. string.format("#define %-25s%-19s\n",key:upper(),value) - end - - s = s .. "\n#endif\n" - return s -end - --- Generate header with the given #defines, save result to file -gen_header_file = function( name, defines ) - local hname = concat_path{ "inc", name:lower() .. ".h" } - local h = assert( io.open( hname, "w" ) ) - h:write( gen_header_string( name, defines ) ) - h:close() -end - --- Remove the given elements from an array -remove_array_elements = function( arr, del ) - del = istable( del ) and del or { del } - foreach( del, function( k, v ) - local pos = array_element_index( arr, v ) - if pos then table.remove( arr, pos ) end - end ) -end - --- Remove a directory recusively --- USE WITH CARE!! Doesn't do much checks :) -rmdir_rec = function ( dirname ) - if lfs.attributes( dirname, "mode" ) ~= "directory" then return end - for f in lfs.dir( dirname ) do - local ename = string.format( "%s/%s", dirname, f ) - local attrs = lfs.attributes( ename ) - if attrs.mode == 'directory' and f ~= '.' and f ~= '..' then - rmdir_rec( ename ) - elseif attrs.mode == 'file' or attrs.mode == 'named pipe' or attrs.mode == 'link' then - os.remove( ename ) - end - end - lfs.rmdir( dirname ) -end - --- Concatenates the second table into the first one -concat_tables = function( dst, src ) - foreach( src, function( k, v ) dst[ k ] = v end ) -end - -------------------------------------------------------------------------------- --- Color-related funtions --- Currently disabled when running in Windows --- (they can be enabled by setting WIN_ANSI_TERM) - -local dcoltable = { 'black', 'red', 'green', 'yellow', 'blue', 'magenta', 'cyan', 'white' } -local coltable = {} -foreach( dcoltable, function( k, v ) coltable[ v ] = k - 1 end ) - -local _col_builder = function( col ) - local _col_maker = function( s ) - if is_os_windows and not os.getenv( "WIN_ANSI_TERM" ) then - return s - else - return( sf( "\027[%d;1m%s\027[m", coltable[ col ] + 30, s ) ) - end - end - return _col_maker -end - -col_funcs = {} -foreach( coltable, function( k, v ) - local fname = "col_" .. k - _G[ fname ] = _col_builder( k ) - col_funcs[ k ] = _G[ fname ] -end ) - -------------------------------------------------------------------------------- --- Option handling - -local options = {} - -options.new = function() - local self = {} - self.options = {} - setmetatable( self, { __index = options } ) - return self -end - --- Argument validator: boolean value -options._bool_validator = function( v ) - if v == '0' or v:upper() == 'FALSE' then - return false - elseif v == '1' or v:upper() == 'TRUE' then - return true - end -end - --- Argument validator: choice value -options._choice_validator = function( v, allowed ) - for i = 1, #allowed do - if v:upper() == allowed[ i ]:upper() then return allowed[ i ] end - end -end - --- Argument validator: choice map (argument value maps to something) -options._choice_map_validator = function( v, allowed ) - for k, value in pairs( allowed ) do - if v:upper() == k:upper() then return value end - end -end - --- Argument validator: string value (no validation) -options._string_validator = function( v ) - return v -end - --- Argument printer: boolean value -options._bool_printer = function( o ) - return "true|false", o.default and "true" or "false" -end - --- Argument printer: choice value -options._choice_printer = function( o ) - local clist, opts = '', o.data - for i = 1, #opts do - clist = clist .. ( i ~= 1 and "|" or "" ) .. opts[ i ] - end - return clist, o.default -end - --- Argument printer: choice map printer -options._choice_map_printer = function( o ) - local clist, opts, def = '', o.data - local i = 1 - for k, v in pairs( opts ) do - clist = clist .. ( i ~= 1 and "|" or "" ) .. k - if o.default == v then def = k end - i = i + 1 - end - return clist, def -end - --- Argument printer: string printer -options._string_printer = function( o ) - return nil, o.default -end - --- Add an option of the specified type -options._add_option = function( self, optname, opttype, help, default, data ) - local validators = - { - string = options._string_validator, choice = options._choice_validator, - boolean = options._bool_validator, choice_map = options._choice_map_validator - } - local printers = - { - string = options._string_printer, choice = options._choice_printer, - boolean = options._bool_printer, choice_map = options._choice_map_printer - } - if not validators[ opttype ] then - print( sf( "[builder] Invalid option type '%s'", opttype ) ) - os.exit( 1 ) - end - table.insert( self.options, { name = optname, help = help, validator = validators[ opttype ], printer = printers[ opttype ], data = data, default = default } ) -end - --- Find an option with the given name -options._find_option = function( self, optname ) - for i = 1, #self.options do - local o = self.options[ i ] - if o.name:upper() == optname:upper() then return self.options[ i ] end - end -end - --- 'add option' helper (automatically detects option type) -options.add_option = function( self, name, help, default, data ) - local otype - if type( default ) == 'boolean' then - otype = 'boolean' - elseif data and type( data ) == 'table' and #data == 0 then - otype = 'choice_map' - elseif data and type( data ) == 'table' then - otype = 'choice' - data = linearize_array( data ) - elseif type( default ) == 'string' then - otype = 'string' - else - print( sf( "Error: cannot detect option type for '%s'", name ) ) - os.exit( 1 ) - end - self:_add_option( name, otype, help, default, data ) -end - -options.get_num_opts = function( self ) - return #self.options -end - -options.get_option = function( self, i ) - return self.options[ i ] -end - --- Handle an option of type 'key=value' --- Returns both the key and the value or nil for error -options.handle_arg = function( self, a ) - local si, ei, k, v = a:find( "([^=]+)=(.*)$" ) - if not k or not v then - print( sf( "Error: invalid syntax in '%s'", a ) ) - return - end - local opt = self:_find_option( k ) - if not opt then - print( sf( "Error: invalid option '%s'", k ) ) - return - end - local optv = opt.validator( v, opt.data ) - if optv == nil then - print( sf( "Error: invalid value '%s' for option '%s'", v, k ) ) - return - end - return k, optv -end - --- Show help for all the registered options -options.show_help = function( self ) - for i = 1, #self.options do - local o = self.options[ i ] - print( sf( "\n %s: %s", o.name, o.help ) ) - local values, default = o.printer( o ) - if values then - print( sf( " Possible values: %s", values ) ) - end - print( sf( " Default value: %s", default or "none (changes at runtime)" ) ) - end -end - --- Create a new option handler -function options_handler() - return options.new() -end -