forked from len0rd/rockbox
lua -- temp loader
fix a few bugs, improve memory usage Change-Id: I6ceefd033000a6178eab0dd513679b5d72cec81c
This commit is contained in:
parent
15ad1c42db
commit
adff45ca21
7 changed files with 233 additions and 99 deletions
|
@ -1,82 +1,99 @@
|
||||||
|
--[[
|
||||||
|
/***************************************************************************
|
||||||
|
* __________ __ ___.
|
||||||
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
||||||
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
||||||
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
||||||
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
||||||
|
* \/ \/ \/ \/ \/
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
* Copyright (C) 2021 William Wilgus
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
||||||
|
* KIND, either express or implied.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
]]
|
||||||
--create keyboard layout
|
--create keyboard layout
|
||||||
--BILGUS 4/2021
|
--BILGUS 4/2021
|
||||||
-- kbdlayout = require("create_kbd_layout")
|
-- create_kbd_layout = require("create_kbd_layout")
|
||||||
-- local layout = kbdlayout.create_keyboard_layout("abcd")
|
-- local layout = create_kbd_layout("abcd")
|
||||||
local _kbdlayout = {} do
|
|
||||||
|
|
||||||
local function encode_short(n)
|
local function encode_short(n)
|
||||||
return string.char(bit.band(0x00FF, n), bit.rshift(bit.band(0xFF00, n), 8))
|
return string.char(bit.band(0x00FF, n), bit.rshift(bit.band(0xFF00, n), 8))
|
||||||
end
|
end
|
||||||
|
|
||||||
local function utf8decode(str)
|
local function utf8decode(str)
|
||||||
local INVALID = 0xfffd
|
local INVALID = 0xfffd
|
||||||
local t = {}
|
local t = {}
|
||||||
local function check_char(c)
|
local function check_char(c)
|
||||||
local tail = false
|
local tail = false
|
||||||
local code
|
local code
|
||||||
c = string.byte(c)
|
c = string.byte(c)
|
||||||
if (c <= 0x7f) or (c >= 0xc2) then
|
if (c <= 0x7f) or (c >= 0xc2) then
|
||||||
-- Start of new character
|
-- Start of new character
|
||||||
if (c < 0x80) then -- U-00000000 - U-0000007F, 1 string.byte
|
if (c < 0x80) then -- U-00000000 - U-0000007F, 1 string.byte
|
||||||
code = c;
|
code = c;
|
||||||
elseif (c < 0xe0) then -- U-00000080 - U-000007FF, 2 string.bytes
|
elseif (c < 0xe0) then -- U-00000080 - U-000007FF, 2 string.bytes
|
||||||
tail = 1;
|
tail = 1;
|
||||||
code = bit.band(c, 0x1f)
|
code = bit.band(c, 0x1f)
|
||||||
elseif (c < 0xf0) then -- U-00000800 - U-0000FFFF, 3 string.bytes
|
elseif (c < 0xf0) then -- U-00000800 - U-0000FFFF, 3 string.bytes
|
||||||
tail = 2;
|
tail = 2;
|
||||||
code = bit.band(c, 0x0f)
|
code = bit.band(c, 0x0f)
|
||||||
elseif (c < 0xf5) then -- U-00010000 - U-001FFFFF, 4 string.bytes
|
elseif (c < 0xf5) then -- U-00010000 - U-001FFFFF, 4 string.bytes
|
||||||
tail = 3;
|
tail = 3;
|
||||||
code = bit.band(c, 0x07)
|
code = bit.band(c, 0x07)
|
||||||
else
|
|
||||||
-- Invalid size
|
|
||||||
code = 0xfffd;
|
|
||||||
end
|
|
||||||
|
|
||||||
while tail and c ~= 0 do
|
|
||||||
tail = tail - 1
|
|
||||||
if bit.band(c, 0xc0) == 0x80 then
|
|
||||||
-- Valid continuation character
|
|
||||||
code = bit.bor(bit.lshift(code, 6),bit.band(c, 0x3f))
|
|
||||||
else
|
|
||||||
-- Invalid continuation char
|
|
||||||
code = INVALID;
|
|
||||||
break;
|
|
||||||
end
|
|
||||||
end
|
|
||||||
else
|
else
|
||||||
-- Invalid UTF-8 char
|
-- Invalid size
|
||||||
code = INVALID;
|
code = INVALID;
|
||||||
end
|
end
|
||||||
-- currently we don't support chars above U-FFFF
|
|
||||||
t[#t + 1 ] = encode_short((code < 0x10000) and code or 0xfffd)
|
while tail and c ~= 0 do
|
||||||
|
tail = tail - 1
|
||||||
|
if bit.band(c, 0xc0) == 0x80 then
|
||||||
|
-- Valid continuation character
|
||||||
|
code = bit.bor(bit.lshift(code, 6),bit.band(c, 0x3f))
|
||||||
|
else
|
||||||
|
-- Invalid continuation char
|
||||||
|
code = INVALID;
|
||||||
|
break;
|
||||||
|
end
|
||||||
|
end
|
||||||
|
else
|
||||||
|
-- Invalid UTF-8 char
|
||||||
|
code = INVALID;
|
||||||
end
|
end
|
||||||
str:gsub(".", check_char) -- run check function for every char
|
-- currently we don't support chars above U-FFFF
|
||||||
return table.concat(t)
|
t[#t + 1 ] = encode_short((code < 0x10000) and code or INVALID)
|
||||||
end
|
end
|
||||||
|
str:gsub(".", check_char) -- run check function for every char
|
||||||
|
return table.concat(t)
|
||||||
|
end
|
||||||
|
|
||||||
local function create_keyboard_layout(s_layout)
|
local function create_keyboard_layout(s_layout)
|
||||||
local insert = table.insert
|
local insert = table.insert
|
||||||
lines = {}
|
lines = {}
|
||||||
|
|
||||||
local t={}
|
for str in string.gmatch(s_layout, "([^\n]+)") do
|
||||||
for str in string.gmatch(s_layout, "([^\n]+)") do
|
local len = string.len(str)
|
||||||
local len = string.len(str)
|
lines[#lines + 1] =
|
||||||
lines[#lines + 1] =
|
table.concat({encode_short(len), utf8decode(str)})
|
||||||
table.concat({encode_short(len), utf8decode(str)})
|
|
||||||
end
|
|
||||||
lines[#lines + 1] = encode_short(0xFEFF)
|
|
||||||
|
|
||||||
return table.concat(lines)
|
|
||||||
end
|
end
|
||||||
_kbdlayout.create_keyboard_layout = create_keyboard_layout
|
lines[#lines + 1] = encode_short(0xFEFF)
|
||||||
_kbdlayout.utf8decode = utf8decode
|
|
||||||
|
return table.concat(lines)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
--[[
|
--[[
|
||||||
local name = "Test_KBD_LAYOUT_" .. tostring(1)
|
local name = "Test_KBD_LAYOUT_" .. tostring(1)
|
||||||
local test = _kbdlayout.create_keyboard_layout("ABCDEFGHIJKLM\nNOPQRSTUVWXYZ\n0123456789")
|
local test = create_keyboard_layout("ABCDEFGHIJKLM\nNOPQRSTUVWXYZ\n0123456789")
|
||||||
local file = io.open('/' .. name, "w+") -- overwrite, rb ignores the 'b' flag
|
local file = io.open('/' .. name, "w+") -- overwrite, rb ignores the 'b' flag
|
||||||
file:write(test)-- write the layout to the file now
|
file:write(test)-- write the layout to the file now
|
||||||
file:close()
|
file:close()
|
||||||
|
@ -87,4 +104,4 @@ if not file then
|
||||||
end
|
end
|
||||||
rb.kbd_input(name, test)
|
rb.kbd_input(name, test)
|
||||||
]]
|
]]
|
||||||
return _kbdlayout
|
return create_keyboard_layout
|
||||||
|
|
|
@ -1,6 +1,42 @@
|
||||||
|
--[[
|
||||||
|
/***************************************************************************
|
||||||
|
* __________ __ ___.
|
||||||
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
||||||
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
||||||
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
||||||
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
||||||
|
* \/ \/ \/ \/ \/
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
* Copyright (C) 2021 William Wilgus
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
||||||
|
* KIND, either express or implied.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
]]
|
||||||
|
-- Bilgus 4/2021
|
||||||
|
local oldrb = rb
|
||||||
|
local tmploader = require("temploader")
|
||||||
|
|
||||||
local rbac_is_loaded = (package.loaded.actions ~= nil)
|
local a_is_loaded = (package.loaded.actions ~= nil)
|
||||||
require("actions") -- Contains rb.actions & rb.contexts
|
local rbold = rb
|
||||||
|
|
||||||
|
if not a_is_loaded then
|
||||||
|
--replace the rb table so we can keep the defines out of the namespace
|
||||||
|
rb = {}
|
||||||
|
end
|
||||||
|
|
||||||
|
--require("actions") -- Contains rb.actions & rb.contexts
|
||||||
|
local actions, err = tmploader("actions")
|
||||||
|
if err then
|
||||||
|
error(err)
|
||||||
|
end
|
||||||
|
|
||||||
-- Menu Button definitions --
|
-- Menu Button definitions --
|
||||||
local button_t = {
|
local button_t = {
|
||||||
|
@ -19,10 +55,5 @@ local button_t = {
|
||||||
UPR = rb.actions.PLA_UP_REPEAT,
|
UPR = rb.actions.PLA_UP_REPEAT,
|
||||||
}
|
}
|
||||||
|
|
||||||
if not rbac_is_loaded then
|
rb = oldrb
|
||||||
rb.actions = nil
|
|
||||||
rb.contexts = nil
|
|
||||||
package.loaded.actionss = nil
|
|
||||||
end
|
|
||||||
|
|
||||||
return button_t
|
return button_t
|
||||||
|
|
|
@ -1,19 +1,51 @@
|
||||||
|
--[[
|
||||||
|
/***************************************************************************
|
||||||
|
* __________ __ ___.
|
||||||
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
||||||
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
||||||
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
||||||
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
||||||
|
* \/ \/ \/ \/ \/
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
* Copyright (C) 2021 William Wilgus
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
||||||
|
* KIND, either express or implied.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
]]
|
||||||
--menu core settings loaded from rockbox user settings
|
--menu core settings loaded from rockbox user settings
|
||||||
--Bilgus 4/2021
|
--Bilgus 4/2021
|
||||||
|
|
||||||
local function get_core_settings()
|
local function get_core_settings()
|
||||||
|
local tmploader = require("temploader")
|
||||||
|
-- rbsettings is a large module to have sitting in RAM
|
||||||
|
-- if user already has it in RAM then use that
|
||||||
local rbs_is_loaded = (package.loaded.rbsettings ~= nil)
|
local rbs_is_loaded = (package.loaded.rbsettings ~= nil)
|
||||||
local s_is_loaded = (package.loaded.settings ~= nil)
|
local s_is_loaded = (package.loaded.settings ~= nil)
|
||||||
|
local rbold = rb
|
||||||
|
|
||||||
require("rbsettings")
|
if not rbs_is_loaded then
|
||||||
require("settings")
|
--replace the rb table so we can keep the defines out of the namespace
|
||||||
rb.metadata = nil -- remove track metadata settings
|
rb = { global_settings = rb.global_settings,
|
||||||
|
global_status = rb.global_status}
|
||||||
|
end
|
||||||
|
|
||||||
|
tmploader("rbsettings")
|
||||||
|
tmploader("settings")
|
||||||
|
|
||||||
local rb_settings = rb.settings.dump('global_settings', "system")
|
local rb_settings = rb.settings.dump('global_settings', "system")
|
||||||
local color_table = {}
|
local color_table = {}
|
||||||
local talk_table = {}
|
local talk_table = {}
|
||||||
local list_settings_table = {}
|
local list_settings_table = {}
|
||||||
local list_settings = "cursor_style|show_icons|statusbar|scrollbar|scrollbar_width|list_separator_height|backdrop_file|"
|
local list_settings = "cursor_style|show_icons|statusbar|scrollbar|scrollbar_width|list_separator_height|backdrop_file|"
|
||||||
|
|
||||||
for key, value in pairs(rb_settings) do
|
for key, value in pairs(rb_settings) do
|
||||||
key = key or ""
|
key = key or ""
|
||||||
if (key:find("color")) then
|
if (key:find("color")) then
|
||||||
|
@ -27,15 +59,9 @@ local function get_core_settings()
|
||||||
|
|
||||||
if not s_is_loaded then
|
if not s_is_loaded then
|
||||||
rb.settings = nil
|
rb.settings = nil
|
||||||
package.loaded.settings = nil
|
|
||||||
end
|
|
||||||
|
|
||||||
if not rbs_is_loaded then
|
|
||||||
rb.system = nil
|
|
||||||
rb.metadata = nil
|
|
||||||
package.loaded.rbsettings = nil
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
rb = rbold
|
||||||
rb.core_color_table = color_table
|
rb.core_color_table = color_table
|
||||||
rb.core_talk_table = talk_table
|
rb.core_talk_table = talk_table
|
||||||
rb.core_list_settings_table = list_settings_table
|
rb.core_list_settings_table = list_settings_table
|
||||||
|
|
|
@ -337,7 +337,7 @@ function print_table(t, t_count, settings)
|
||||||
table_p = init_position(15, 5)
|
table_p = init_position(15, 5)
|
||||||
line, maxline = _print.opt.area(5, 1, rb.LCD_WIDTH - 10 - sb_width, rb.LCD_HEIGHT - 2)
|
line, maxline = _print.opt.area(5, 1, rb.LCD_WIDTH - 10 - sb_width, rb.LCD_HEIGHT - 2)
|
||||||
|
|
||||||
if curpos > maxline then
|
if (curpos or 0) > maxline then
|
||||||
local c = maxline / 2
|
local c = maxline / 2
|
||||||
start = (start or 1) + curpos - maxline
|
start = (start or 1) + curpos - maxline
|
||||||
curpos = maxline
|
curpos = maxline
|
||||||
|
|
|
@ -1,30 +1,63 @@
|
||||||
--[[
|
--[[
|
||||||
temp loader allows some lua requires to be loaded and later garbage collected
|
/***************************************************************************
|
||||||
unfortunately the module needs to be formatted in such a way to pass back a
|
* __________ __ ___.
|
||||||
call table in order to keep the functions within from being garbage collected
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
||||||
too early
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
||||||
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
||||||
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
||||||
|
* \/ \/ \/ \/ \/
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
* Copyright (C) 2021 William Wilgus
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
||||||
|
* KIND, either express or implied.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
--
|
||||||
|
temp_loader allows some (pure) lua requires to be loaded and later garbage collected
|
||||||
|
unfortunately the 'required' module needs to be formatted in such a way to
|
||||||
|
pass back a reference to a function or call table in order to keep the functions
|
||||||
|
within from being garbage collected too early
|
||||||
|
|
||||||
BE AWARE this bypasses the module loader which would allow code reuse
|
modules that add things to _G table are unaffected by using this function
|
||||||
so if you aren't careful this memory saving tool could spell disaster
|
except if later you use require or temp_loader those tables will again
|
||||||
for free RAM if you load the same code multiple times
|
be reloaded with fresh data since nothing was recorded about the module being loaded
|
||||||
|
|
||||||
|
modulename - same as require()
|
||||||
|
newinstance == true -- get a new copy (from disk) of the module
|
||||||
|
... other args for the module
|
||||||
|
|
||||||
|
BE AWARE this bypasses the module loader
|
||||||
|
which would allow code reuse so if you aren't careful this memory saving tool
|
||||||
|
could spell disaster for free RAM if you load the same code multiple times
|
||||||
--]]
|
--]]
|
||||||
|
|
||||||
|
local function tempload(modulename, newinstance, ...)
|
||||||
local function tempload(modulename)
|
|
||||||
--http://lua-users.org/wiki/LuaModulesLoader
|
--http://lua-users.org/wiki/LuaModulesLoader
|
||||||
local errmsg = ""
|
local errmsg = ""
|
||||||
|
-- Is there current a loaded module by this name?
|
||||||
|
if package.loaded[modulename] ~= nil and not newinstance then
|
||||||
|
return require(modulename)
|
||||||
|
end
|
||||||
-- Find source
|
-- Find source
|
||||||
local modulepath = string.gsub(modulename, "%.", "/")
|
local modulepath = string.gsub(modulename, "%.", "/")
|
||||||
for path in string.gmatch(package.path, "([^;]+)") do
|
for path in string.gmatch(package.path, "([^;]+)") do
|
||||||
local filename = string.gsub(path, "%?", modulepath)
|
local filename = string.gsub(path, "%?", modulepath)
|
||||||
local file = io.open(filename, "r")
|
--attempt to open and compile module
|
||||||
|
local file, err = loadfile(filename)
|
||||||
if file then
|
if file then
|
||||||
-- Compile and return the module
|
-- execute the compiled chunk
|
||||||
return assert(loadstring(assert(file:read("*a")), filename))()
|
return file(... or modulename)
|
||||||
end
|
end
|
||||||
errmsg = errmsg.."\n\tno file '"..filename.."' (temp loader)"
|
errmsg = table.concat({errmsg, "\n\tno file '", filename, "' (temp loader)"})
|
||||||
end
|
end
|
||||||
return errmsg
|
return nil, errmsg
|
||||||
end
|
end
|
||||||
|
|
||||||
return tempload
|
return tempload
|
||||||
|
|
|
@ -52,6 +52,8 @@ char *strerror(int errnum)
|
||||||
*/
|
*/
|
||||||
int splash_scroller(int timeout, const char* str)
|
int splash_scroller(int timeout, const char* str)
|
||||||
{
|
{
|
||||||
|
if (!str)
|
||||||
|
str = "[nil]";
|
||||||
int w, ch_w, ch_h;
|
int w, ch_w, ch_h;
|
||||||
rb->lcd_getstringsize("W", &ch_w, &ch_h);
|
rb->lcd_getstringsize("W", &ch_w, &ch_h);
|
||||||
|
|
||||||
|
|
|
@ -279,8 +279,9 @@ end
|
||||||
for i=1, #tWriteBuf do tWriteBuf[i] = _NIL end -- reuse table
|
for i=1, #tWriteBuf do tWriteBuf[i] = _NIL end -- reuse table
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
if ... then
|
||||||
tcBase= nil tkSortCbase= nil
|
tcBase= nil tkSortCbase= nil
|
||||||
|
end
|
||||||
tWriteBuf[#tWriteBuf + 1] = "\r\n"
|
tWriteBuf[#tWriteBuf + 1] = "\r\n"
|
||||||
tWriteBuf[#tWriteBuf + 1] = dtTag("?")
|
tWriteBuf[#tWriteBuf + 1] = dtTag("?")
|
||||||
tWriteBuf[#tWriteBuf + 1] = "\r\n\r\n"
|
tWriteBuf[#tWriteBuf + 1] = "\r\n\r\n"
|
||||||
|
@ -309,5 +310,29 @@ end
|
||||||
filehandle:write(table.concat(tWriteBuf))
|
filehandle:write(table.concat(tWriteBuf))
|
||||||
for i=1, #tWriteBuf do tWriteBuf[i] = _NIL end -- empty table
|
for i=1, #tWriteBuf do tWriteBuf[i] = _NIL end -- empty table
|
||||||
filehandle:close()
|
filehandle:close()
|
||||||
rb.splash(rb.HZ * 5, n .. " Items dumped to : " .. sDumpFile)
|
--rb.splash((rb.HZ or 100) * 5, n .. " Items dumped to : " .. sDumpFile)
|
||||||
--rb.splash(500, collectgarbage("count"))
|
--rb.splash(500, collectgarbage("count"))
|
||||||
|
if not ... then
|
||||||
|
local lu = collectgarbage("collect")
|
||||||
|
local used, allocd, free = rb.mem_stats()
|
||||||
|
local lu = collectgarbage("count")
|
||||||
|
local fmt = function(t, v) return string.format("%s: %d Kb\n", t, v /1024) end
|
||||||
|
local s_t = {}
|
||||||
|
s_t[1] = n
|
||||||
|
s_t[2] = " Items dumped to:\n"
|
||||||
|
s_t[3] = sDumpFile
|
||||||
|
s_t[4] = "\n\nLoaded Modules:\n"
|
||||||
|
n = 0
|
||||||
|
for k, v in pairsByPairs(tcBase, tkSortCbase ) do
|
||||||
|
n = n + 1
|
||||||
|
if n ~= 1 then
|
||||||
|
s_t[#s_t + 1] = ", "
|
||||||
|
end
|
||||||
|
s_t[#s_t + 1] = tostring(k)
|
||||||
|
if n >= 3 then -- split loaded modules to multiple lines
|
||||||
|
n = 0
|
||||||
|
s_t[#s_t + 1] = "\n"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
rb.splash_scroller(5 * (rb.HZ or 100), table.concat(s_t))
|
||||||
|
end
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue