Revision 25db8dbb49a887fd523060338112637cb7ac1cd2 authored by Giovanni Condello on 27 March 2014, 10:32:59 UTC, committed by Giovanni Condello on 27 March 2014, 10:32:59 UTC
1 parent bd1d1b8
module.c
#include "dryos.h"
#include "menu.h"
#include "console.h"
#include "libtcc.h"
#include "module.h"
#include "config.h"
#include "string.h"
#include "property.h"
#include "beep.h"
#include "bmp.h"
#ifndef CONFIG_MODULES_MODEL_SYM
#error Not defined file name with symbols
#endif
#define MAGIC_SYMBOLS "ML/MODULES/"CONFIG_MODULES_MODEL_SYM
/* unloads TCC after linking the modules */
/* note: this breaks module_exec and ETTR */
#define CONFIG_TCC_UNLOAD
extern int sscanf(const char *str, const char *format, ...);
static module_entry_t module_list[MODULE_COUNT_MAX];
#ifdef CONFIG_TCC_UNLOAD
static void* module_code = NULL;
#else
static TCCState *module_state = NULL;
#endif
static struct menu_entry module_submenu[];
static struct menu_entry module_menu[];
CONFIG_INT("module.autoload", module_autoload_disabled, 0);
CONFIG_INT("module.console", module_console_enabled, 0);
CONFIG_INT("module.ignore_crashes", module_ignore_crashes, 0);
char *module_lockfile = MODULE_PATH"LOADING.LCK";
static struct msg_queue * module_mq = 0;
#define MSG_MODULE_LOAD_ALL 1
#define MSG_MODULE_UNLOAD_ALL 2
#define MSG_MODULE_LOAD_OFFLINE_STRINGS 3 /* argument: module index in high half (FFFF0000) */
#define MSG_MODULE_UNLOAD_OFFLINE_STRINGS 4 /* same argument */
void module_load_all(void)
{
msg_queue_post(module_mq, MSG_MODULE_LOAD_ALL);
}
void module_unload_all(void)
{
msg_queue_post(module_mq, MSG_MODULE_UNLOAD_ALL);
}
static void _module_load_all(uint32_t);
static void _module_unload_all(void);
static int module_load_symbols(TCCState *s, char *filename)
{
uint32_t size = 0;
FILE* file = NULL;
char *buf = NULL;
uint32_t count = 0;
uint32_t pos = 0;
if( FIO_GetFileSize( filename, &size ) != 0 )
{
console_printf("Error loading '%s': File does not exist\n", filename);
return -1;
}
buf = fio_malloc(size);
if(!buf)
{
console_printf("Error loading '%s': File too large\n", filename);
return -1;
}
file = FIO_OpenFile(filename, O_RDONLY | O_SYNC);
if(!file)
{
console_printf("Error loading '%s': File does not exist\n", filename);
fio_free(buf);
return -1;
}
FIO_ReadFile(file, buf, size);
FIO_CloseFile(file);
while(buf[pos])
{
char address_buf[16];
char symbol_buf[128];
uint32_t length = 0;
uint32_t address = 0;
while(buf[pos + length] && buf[pos + length] != ' ' && length < sizeof(address_buf))
{
address_buf[length] = buf[pos + length];
length++;
}
address_buf[length] = '\000';
pos += length + 1;
length = 0;
while(buf[pos + length] && buf[pos + length] != '\r' && buf[pos + length] != '\n' && length < sizeof(symbol_buf))
{
symbol_buf[length] = buf[pos + length];
length++;
}
symbol_buf[length] = '\000';
pos += length + 1;
length = 0;
while(buf[pos + length] && (buf[pos + length] == '\r' || buf[pos + length] == '\n'))
{
pos++;
}
//~ sscanf(address_buf, "%x", &address);
address = strtoul(address_buf, NULL, 16);
tcc_add_symbol(s, symbol_buf, (void*)address);
count++;
}
fio_free(buf);
return 0;
}
/* this is not perfect, as .Mo and .mO aren't detected. important? */
static int module_valid_filename(char* filename)
{
int len = strlen(filename);
if((len < 3) || (filename[0] == '.') || (filename[0] == '_'))
{
return 0;
}
if(!strcmp(&filename[len - 3], ".MO") || !strcmp(&filename[len - 3], ".mo") )
{
return 1;
}
return 0;
}
/* must be called before unloading TCC */
static void module_update_core_symbols(TCCState* state)
{
console_printf("Updating symbols...\n");
extern struct module_symbol_entry _module_symbols_start[];
extern struct module_symbol_entry _module_symbols_end[];
struct module_symbol_entry * module_symbol_entry = _module_symbols_start;
for( ; module_symbol_entry < _module_symbols_end ; module_symbol_entry++ )
{
void* old_address = *(module_symbol_entry->address);
void* new_address = (void*) tcc_get_symbol(state, (char*) module_symbol_entry->name);
if (new_address)
{
*(module_symbol_entry->address) = new_address;
console_printf(" [i] upd %s %x => %x\n", module_symbol_entry->name, old_address, new_address);
}
else
{
console_printf(" [i] 404: %s %x\n", module_symbol_entry->name, old_address);
}
}
}
static void _module_load_all(uint32_t list_only)
{
TCCState *state = NULL;
uint32_t module_cnt = 0;
struct fio_file file;
uint32_t update_properties = 0;
#ifdef CONFIG_MODULE_UNLOAD
/* ensure all modules are unloaded */
console_printf("Unloading modules...\n");
_module_unload_all();
#endif
#ifdef CONFIG_TCC_UNLOAD
if (module_code)
#else
if (module_state)
#endif
{
console_printf("Modules already loaded.\n");
beep();
return;
}
/* initialize linker */
state = tcc_new();
tcc_set_options(state, "-nostdlib");
if(module_load_symbols(state, MAGIC_SYMBOLS) < 0)
{
NotifyBox(2000, "Missing symbol file: " MAGIC_SYMBOLS );
tcc_delete(state); console_show();
return;
}
console_printf("Scanning modules...\n");
struct fio_dirent * dirent = FIO_FindFirstEx( MODULE_PATH, &file );
if( IS_ERROR(dirent) )
{
NotifyBox(2000, "Module dir missing" );
tcc_delete(state); console_show();
return;
}
do
{
if (file.mode & ATTR_DIRECTORY) continue; // is a directory
if (module_valid_filename(file.name))
{
char module_name[MODULE_FILENAME_LENGTH];
/* get filename, remove extension and append _init to get the init symbol */
//console_printf(" [i] found: %s\n", file.name);
/* ensure the buffer is null terminated */
memset(module_name, 0x00, sizeof(module_name));
strncpy(module_name, file.name, MODULE_NAME_LENGTH);
strncpy(module_list[module_cnt].filename, file.name, MODULE_FILENAME_LENGTH);
snprintf(module_list[module_cnt].long_filename, sizeof(module_list[module_cnt].long_filename), "%s%s", MODULE_PATH, module_list[module_cnt].filename);
uint32_t pos = 0;
while(module_name[pos])
{
/* extension starting? terminate string */
if(module_name[pos] == '.')
{
module_name[pos] = '\000';
break;
}
else if(module_name[pos] >= 'A' && module_name[pos] <= 'Z')
{
/* make lowercase */
module_name[pos] |= 0x20;
}
pos++;
}
strncpy(module_list[module_cnt].name, module_name, sizeof(module_list[module_cnt].name));
/* check for a .en file that tells the module is enabled */
char enable_file[MODULE_FILENAME_LENGTH];
snprintf(enable_file, sizeof(enable_file), "%s%s.en", get_config_dir(), module_list[module_cnt].name);
/* if enable-file is nonexistent, dont load module */
if(!config_flag_file_setting_load(enable_file))
{
module_list[module_cnt].enabled = 0;
snprintf(module_list[module_cnt].status, sizeof(module_list[module_cnt].status), "OFF");
snprintf(module_list[module_cnt].long_status, sizeof(module_list[module_cnt].long_status), "Module disabled");
//console_printf(" [i] %s\n", module_list[module_cnt].long_status);
}
else
{
module_list[module_cnt].enabled = 1;
if(list_only)
{
snprintf(module_list[module_cnt].status, sizeof(module_list[module_cnt].status), "");
snprintf(module_list[module_cnt].long_status, sizeof(module_list[module_cnt].long_status), "Module not loaded");
}
else
{
snprintf(module_list[module_cnt].status, sizeof(module_list[module_cnt].status), "???");
snprintf(module_list[module_cnt].long_status, sizeof(module_list[module_cnt].long_status), "Seems linking failed. Unknown symbols?");
}
}
module_cnt++;
if (module_cnt >= MODULE_COUNT_MAX)
{
NotifyBox(2000, "Too many modules" );
break;
}
}
} while( FIO_FindNextEx( dirent, &file ) == 0);
FIO_FindClose(dirent);
/* sort modules */
for (int i = 0; i < (int) module_cnt-1; i++)
{
for (int j = i+1; j < (int) module_cnt; j++)
{
if (
/* loaded modules first, then alphabetically */
(module_list[i].enabled == 0 && module_list[j].enabled) ||
(module_list[i].enabled == module_list[j].enabled && strcmp(module_list[i].name, module_list[j].name) > 0)
)
{
module_entry_t aux = module_list[i];
module_list[i] = module_list[j];
module_list[j] = aux;
}
}
}
/* dont load anything, just return */
if(list_only)
{
tcc_delete(state);
return;
}
/* load modules */
console_printf("Load modules...\n");
for (uint32_t mod = 0; mod < module_cnt; mod++)
{
if(module_list[mod].enabled)
{
console_printf(" [i] load: %s\n", module_list[mod].filename);
int32_t ret = tcc_add_file(state, module_list[mod].long_filename);
module_list[mod].valid = 1;
/* seems bad, disable it */
if(ret < 0)
{
module_list[mod].error = 1;
snprintf(module_list[mod].status, sizeof(module_list[mod].status), "FileErr");
snprintf(module_list[mod].long_status, sizeof(module_list[mod].long_status), "Load failed: %s, ret 0x%02X");
console_printf(" [E] %s\n", module_list[mod].long_status);
}
}
}
console_printf("Linking..\n");
#ifdef CONFIG_TCC_UNLOAD
int32_t size = tcc_relocate(state, NULL);
int32_t reloc_status = -1;
if (size > 0)
{
void* buf = (void*) malloc(size);
reloc_status = tcc_relocate(state, buf);
module_code = buf;
}
if(size < 0 || reloc_status < 0)
#else
int32_t ret = tcc_relocate(state, TCC_RELOCATE_AUTO);
if(ret < 0)
#endif
{
console_printf(" [E] failed to link modules\n");
for (uint32_t mod = 0; mod < module_cnt; mod++)
{
if(module_list[mod].enabled)
{
module_list[mod].error = 1;
snprintf(module_list[mod].status, sizeof(module_list[mod].status), "Err");
snprintf(module_list[mod].long_status, sizeof(module_list[mod].long_status), "Linking failed");
}
}
tcc_delete(state); console_show();
return;
}
/* load modules symbols */
console_printf("Register modules...\n");
for (uint32_t mod = 0; mod < module_cnt; mod++)
{
if(module_list[mod].valid && module_list[mod].enabled && !module_list[mod].error)
{
char module_info_name[32];
/* now check for info structure */
snprintf(module_info_name, sizeof(module_info_name), "%s%s", STR(MODULE_INFO_PREFIX), module_list[mod].name);
module_list[mod].info = tcc_get_symbol(state, module_info_name);
snprintf(module_info_name, sizeof(module_info_name), "%s%s", STR(MODULE_STRINGS_PREFIX), module_list[mod].name);
module_list[mod].strings = tcc_get_symbol(state, module_info_name);
snprintf(module_info_name, sizeof(module_info_name), "%s%s", STR(MODULE_PARAMS_PREFIX), module_list[mod].name);
module_list[mod].params = tcc_get_symbol(state, module_info_name);
snprintf(module_info_name, sizeof(module_info_name), "%s%s", STR(MODULE_PROPHANDLERS_PREFIX), module_list[mod].name);
module_list[mod].prop_handlers = tcc_get_symbol(state, module_info_name);
snprintf(module_info_name, sizeof(module_info_name), "%s%s", STR(MODULE_CBR_PREFIX), module_list[mod].name);
module_list[mod].cbr = tcc_get_symbol(state, module_info_name);
snprintf(module_info_name, sizeof(module_info_name), "%s%s", STR(MODULE_CONFIG_PREFIX), module_list[mod].name);
module_list[mod].config = tcc_get_symbol(state, module_info_name);
/* check if the module symbol is defined. simple check for valid memory address just in case. */
if((uint32_t)module_list[mod].info > 0x1000)
{
if(module_list[mod].info->api_magic == MODULE_MAGIC)
{
if((module_list[mod].info->api_major == MODULE_MAJOR) && (module_list[mod].info->api_minor <= MODULE_MINOR))
{
/* this module seems to be sane */
}
else
{
module_list[mod].error = 1;
snprintf(module_list[mod].status, sizeof(module_list[mod].status), "OldAPI");
snprintf(module_list[mod].long_status, sizeof(module_list[mod].long_status), "Wrong version (v%d.%d, expected v%d.%d)\n", module_list[mod].info->api_major, module_list[mod].info->api_minor, MODULE_MAJOR, MODULE_MINOR);
console_printf(" [E] %s\n", module_list[mod].long_status);
/* disable active stuff, since the results are unpredictable */
module_list[mod].cbr = 0;
module_list[mod].prop_handlers = 0;
}
}
else
{
module_list[mod].error = 1;
snprintf(module_list[mod].status, sizeof(module_list[mod].status), "Magic");
snprintf(module_list[mod].long_status, sizeof(module_list[mod].long_status), "Invalid Magic (0x%X)\n", module_list[mod].info->api_magic);
console_printf(" [E] %s\n", module_list[mod].long_status);
}
}
else
{
module_list[mod].error = 1;
snprintf(module_list[mod].status, sizeof(module_list[mod].status), "noInfo");
snprintf(module_list[mod].long_status, sizeof(module_list[mod].long_status), "No info structure found (addr 0x%X)\n", (uint32_t)module_list[mod].info);
console_printf(" [E] %s\n", module_list[mod].long_status);
}
}
}
console_printf("Load configs...\n");
for (uint32_t mod = 0; mod < module_cnt; mod++)
{
if(module_list[mod].enabled && module_list[mod].valid && !module_list[mod].error)
{
char filename[64];
snprintf(filename, sizeof(filename), "%s%s.cfg", get_config_dir(), module_list[mod].name);
module_config_load(filename, &module_list[mod]);
}
}
/* before we execute code, make sure a) data caches are drained and b) instruction caches are clean */
sync_caches();
/* go through all modules and initialize them */
console_printf("Init modules...\n");
for (uint32_t mod = 0; mod < module_cnt; mod++)
{
if(module_list[mod].valid && module_list[mod].enabled && !module_list[mod].error)
{
console_printf(" [i] Init: '%s'\n", module_list[mod].name);
if(0)
{
console_printf(" [i] info at: 0x%08X\n", (uint32_t)module_list[mod].info);
console_printf(" [i] strings at: 0x%08X\n", (uint32_t)module_list[mod].strings);
console_printf(" [i] params at: 0x%08X\n", (uint32_t)module_list[mod].params);
console_printf(" [i] props at: 0x%08X\n", (uint32_t)module_list[mod].prop_handlers);
console_printf(" [i] cbr at: 0x%08X\n", (uint32_t)module_list[mod].cbr);
console_printf(" [i] config at: 0x%08X\n", (uint32_t)module_list[mod].config);
console_printf("-----------------------------\n");
}
/* initialize module */
if(module_list[mod].info->init)
{
int err = module_list[mod].info->init();
if (err)
{
module_list[mod].error = err;
snprintf(module_list[mod].status, sizeof(module_list[mod].status), "Err");
snprintf(module_list[mod].long_status, sizeof(module_list[mod].long_status), "Module init failed");
/* disable active stuff, since the results are unpredictable */
module_list[mod].cbr = 0;
module_list[mod].prop_handlers = 0;
}
}
/* register property handlers */
if(module_list[mod].prop_handlers && !module_list[mod].error)
{
module_prophandler_t **props = module_list[mod].prop_handlers;
while(*props != NULL)
{
update_properties = 1;
console_printf(" [i] prop %s\n", (*props)->name);
prop_add_handler((*props)->property, (*props)->handler);
props++;
}
}
if(0)
{
console_printf("-----------------------------\n");
}
if (!module_list[mod].error)
{
snprintf(module_list[mod].status, sizeof(module_list[mod].status), "OK");
snprintf(module_list[mod].long_status, sizeof(module_list[mod].long_status), "Module loaded successfully");
}
}
}
if(update_properties)
{
prop_update_registration();
}
module_update_core_symbols(state);
#ifdef CONFIG_TCC_UNLOAD
tcc_delete(state);
#else
module_state = state;
#endif
console_printf("Modules loaded\n");
if(!module_console_enabled)
{
console_hide();
}
}
static void _module_unload_all(void)
{
/* unloading is not yet clean, we can end up with tasks running from freed memory or stuff like that */
#ifdef CONFIG_MODULE_UNLOAD
if(module_state)
{
TCCState *state = module_state;
module_state = NULL;
/* unregister all property handlers */
prop_reset_registration();
/* deinit and clean all modules */
for(int mod = 0; mod < MODULE_COUNT_MAX; mod++)
{
if(module_list[mod].valid && module_list[mod].enabled && !module_list[mod].error)
{
if(module_list[mod].info && module_list[mod].info->deinit)
{
module_list[mod].info->deinit();
}
}
module_list[mod].valid = 0;
module_list[mod].enabled = 0;
module_list[mod].error = 0;
module_list[mod].info = NULL;
module_list[mod].strings = NULL;
module_list[mod].params = NULL;
module_list[mod].prop_handlers = NULL;
module_list[mod].cbr = NULL;
strcpy(module_list[mod].name, "");
strcpy(module_list[mod].filename, "");
}
/* release the global module state */
tcc_delete(state);
}
#endif
}
void* module_load(char *filename)
{
int ret = -1;
TCCState *state = NULL;
state = tcc_new();
tcc_set_options(state, "-nostdlib");
if(module_load_symbols(state, MAGIC_SYMBOLS) < 0)
{
NotifyBox(2000, "Missing symbol file: " MAGIC_SYMBOLS );
tcc_delete(state);
return NULL;
}
ret = tcc_add_file(state, filename);
if(ret < 0)
{
tcc_delete(state);
return NULL;
}
ret = tcc_relocate(state, TCC_RELOCATE_AUTO);
if(ret < 0)
{
tcc_delete(state);
return NULL;
}
return (void*)state;
}
unsigned int module_get_symbol(void *module, char *symbol)
{
#ifndef CONFIG_TCC_UNLOAD
if (module == NULL) module = module_state;
#endif
if (module == NULL) return 0;
TCCState *state = (TCCState *)module;
return (int) tcc_get_symbol(state, symbol);
}
int module_exec(void *module, char *symbol, int count, ...)
{
int ret = -1;
#ifndef CONFIG_TCC_UNLOAD
if (module == NULL) module = module_state;
#endif
if (module == NULL) return ret;
TCCState *state = (TCCState *)module;
void *start_symbol = NULL;
va_list args;
va_start(args, count);
start_symbol = tcc_get_symbol(state, symbol);
/* check if the module symbol is defined. simple check for valid memory address just in case. */
if((uint32_t)start_symbol > 0x1000)
{
/* no parameters, special case */
if(count == 0)
{
uint32_t (*exec)() = start_symbol;
ret = exec();
}
else
{
uint32_t (*exec)(uint32_t parm1, ...) = start_symbol;
uint32_t parms[10];
for(int parm = 0; parm < count; parm++)
{
parms[parm] = va_arg(args,uint32_t);
}
switch(count)
{
case 1:
ret = exec(parms[0]);
break;
case 2:
ret = exec(parms[0], parms[1]);
break;
case 3:
ret = exec(parms[0], parms[1], parms[2]);
break;
case 4:
ret = exec(parms[0], parms[1], parms[2], parms[3]);
break;
case 5:
ret = exec(parms[0], parms[1], parms[2], parms[3], parms[4]);
break;
case 6:
ret = exec(parms[0], parms[1], parms[2], parms[3], parms[4], parms[5]);
break;
case 7:
ret = exec(parms[0], parms[1], parms[2], parms[3], parms[4], parms[5], parms[6]);
break;
case 8:
ret = exec(parms[0], parms[1], parms[2], parms[3], parms[4], parms[5], parms[6], parms[7]);
break;
default:
/* so many parameters?! */
NotifyBox(2000, "Passing too many parameters to '%s'", symbol );
break;
}
}
}
va_end(args);
return ret;
}
int module_unload(void *module)
{
TCCState *state = (TCCState *)module;
tcc_delete(state);
return 0;
}
/* execute all callback routines of given type. maybe it will get extended to support varargs */
int FAST module_exec_cbr(unsigned int type)
{
for(int mod = 0; mod < MODULE_COUNT_MAX; mod++)
{
module_cbr_t *cbr = module_list[mod].cbr;
if(module_list[mod].valid && cbr)
{
while(cbr->name)
{
if(cbr->type == type)
{
int ret = cbr->handler(cbr->ctx);
if (ret != CBR_RET_CONTINUE)
{
return ret;
}
}
cbr++;
}
}
}
return CBR_RET_CONTINUE;
}
/* translate camera specific key to portable module key */
#define MODULE_TRANSLATE_KEY(in,out,dest) if(dest == MODULE_KEY_PORTABLE) { if(in != -1){ if(key == in) { return out; } }} else {if(out != -1){ if(key == out) { return in; } }}
/* these are to ensure that the checked keys are defined. we have to ensure they're defined before using. are there better ways to ensure? */
#if !defined(BGMT_WHEEL_UP)
#define BGMT_WHEEL_UP -1
#endif
#if !defined(BGMT_WHEEL_DOWN)
#define BGMT_WHEEL_DOWN -1
#endif
#if !defined(BGMT_WHEEL_LEFT)
#define BGMT_WHEEL_LEFT -1
#endif
#if !defined(BGMT_WHEEL_RIGHT)
#define BGMT_WHEEL_RIGHT -1
#endif
#if !defined(BGMT_PRESS_SET)
#define BGMT_PRESS_SET -1
#endif
#if !defined(BGMT_UNPRESS_SET)
#define BGMT_UNPRESS_SET -1
#endif
#if !defined(BGMT_MENU)
#define BGMT_MENU -1
#endif
#if !defined(BGMT_INFO)
#define BGMT_INFO -1
#endif
#if !defined(BGMT_PLAY)
#define BGMT_PLAY -1
#endif
#if !defined(BGMT_TRASH)
#define BGMT_TRASH -1
#endif
#if !defined(BGMT_PRESS_DP)
#define BGMT_PRESS_DP -1
#endif
#if !defined(BGMT_UNPRESS_DP)
#define BGMT_UNPRESS_DP -1
#endif
#if !defined(BGMT_RATE)
#define BGMT_RATE -1
#endif
#if !defined(BGMT_REC)
#define BGMT_REC -1
#endif
#if !defined(BGMT_PRESS_ZOOMIN_MAYBE)
#define BGMT_PRESS_ZOOMIN_MAYBE -1
#endif
#if !defined(BGMT_LV)
#define BGMT_LV -1
#endif
#if !defined(BGMT_Q)
#define BGMT_Q -1
#endif
#if !defined(BGMT_PICSTYLE)
#define BGMT_PICSTYLE -1
#endif
#if !defined(BGMT_JOY_CENTER)
#define BGMT_JOY_CENTER -1
#endif
#if !defined(BGMT_PRESS_UP)
#define BGMT_PRESS_UP -1
#endif
#if !defined(BGMT_PRESS_UP_RIGHT)
#define BGMT_PRESS_UP_RIGHT -1
#endif
#if !defined(BGMT_PRESS_UP_LEFT)
#define BGMT_PRESS_UP_LEFT -1
#endif
#if !defined(BGMT_PRESS_RIGHT)
#define BGMT_PRESS_RIGHT -1
#endif
#if !defined(BGMT_PRESS_LEFT)
#define BGMT_PRESS_LEFT -1
#endif
#if !defined(BGMT_PRESS_DOWN_RIGHT)
#define BGMT_PRESS_DOWN_RIGHT -1
#endif
#if !defined(BGMT_PRESS_DOWN_LEFT)
#define BGMT_PRESS_DOWN_LEFT -1
#endif
#if !defined(BGMT_PRESS_DOWN)
#define BGMT_PRESS_DOWN -1
#endif
#if !defined(BGMT_UNPRESS_UDLR)
#define BGMT_UNPRESS_UDLR -1
#endif
#if !defined(BGMT_PRESS_HALFSHUTTER)
#define BGMT_PRESS_HALFSHUTTER -1
#endif
#if !defined(BGMT_UNPRESS_HALFSHUTTER)
#define BGMT_UNPRESS_HALFSHUTTER -1
#endif
#if !defined(BGMT_PRESS_FULLSHUTTER)
#define BGMT_PRESS_FULLSHUTTER -1
#endif
#if !defined(BGMT_UNPRESS_FULLSHUTTER)
#define BGMT_UNPRESS_FULLSHUTTER -1
#endif
#if !defined(BGMT_PRESS_FLASH_MOVIE)
#define BGMT_PRESS_FLASH_MOVIE -1
#endif
#if !defined(BGMT_UNPRESS_FLASH_MOVIE)
#define BGMT_UNPRESS_FLASH_MOVIE -1
#endif
#if !defined(BGMT_TOUCH_1_FINGER)
#define BGMT_TOUCH_1_FINGER -1
#endif
#if !defined(BGMT_UNTOUCH_1_FINGER)
#define BGMT_UNTOUCH_1_FINGER -1
#endif
#if !defined(BGMT_TOUCH_2_FINGER)
#define BGMT_TOUCH_2_FINGER -1
#endif
#if !defined(BGMT_UNTOUCH_2_FINGER)
#define BGMT_UNTOUCH_2_FINGER -1
#endif
int module_translate_key(int key, int dest)
{
MODULE_TRANSLATE_KEY(BGMT_WHEEL_UP , MODULE_KEY_WHEEL_UP , dest);
MODULE_TRANSLATE_KEY(BGMT_WHEEL_DOWN , MODULE_KEY_WHEEL_DOWN , dest);
MODULE_TRANSLATE_KEY(BGMT_WHEEL_LEFT , MODULE_KEY_WHEEL_LEFT , dest);
MODULE_TRANSLATE_KEY(BGMT_WHEEL_RIGHT , MODULE_KEY_WHEEL_RIGHT , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_SET , MODULE_KEY_PRESS_SET , dest);
MODULE_TRANSLATE_KEY(BGMT_UNPRESS_SET , MODULE_KEY_UNPRESS_SET , dest);
MODULE_TRANSLATE_KEY(BGMT_MENU , MODULE_KEY_MENU , dest);
MODULE_TRANSLATE_KEY(BGMT_INFO , MODULE_KEY_INFO , dest);
MODULE_TRANSLATE_KEY(BGMT_PLAY , MODULE_KEY_PLAY , dest);
MODULE_TRANSLATE_KEY(BGMT_TRASH , MODULE_KEY_TRASH , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_DP , MODULE_KEY_PRESS_DP , dest);
MODULE_TRANSLATE_KEY(BGMT_UNPRESS_DP , MODULE_KEY_UNPRESS_DP , dest);
MODULE_TRANSLATE_KEY(BGMT_RATE , MODULE_KEY_RATE , dest);
MODULE_TRANSLATE_KEY(BGMT_REC , MODULE_KEY_REC , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_ZOOMIN_MAYBE , MODULE_KEY_PRESS_ZOOMIN , dest);
MODULE_TRANSLATE_KEY(BGMT_LV , MODULE_KEY_LV , dest);
MODULE_TRANSLATE_KEY(BGMT_PICSTYLE , MODULE_KEY_PICSTYLE , dest);
MODULE_TRANSLATE_KEY(BGMT_JOY_CENTER , MODULE_KEY_JOY_CENTER , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_UP , MODULE_KEY_PRESS_UP , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_UP_RIGHT , MODULE_KEY_PRESS_UP_RIGHT , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_UP_LEFT , MODULE_KEY_PRESS_UP_LEFT , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_RIGHT , MODULE_KEY_PRESS_RIGHT , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_LEFT , MODULE_KEY_PRESS_LEFT , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_DOWN_RIGHT , MODULE_KEY_PRESS_DOWN_RIGHT , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_DOWN_LEFT , MODULE_KEY_PRESS_DOWN_LEFT , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_DOWN , MODULE_KEY_PRESS_DOWN , dest);
MODULE_TRANSLATE_KEY(BGMT_UNPRESS_UDLR , MODULE_KEY_UNPRESS_UDLR , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_HALFSHUTTER , MODULE_KEY_PRESS_HALFSHUTTER , dest);
MODULE_TRANSLATE_KEY(BGMT_UNPRESS_HALFSHUTTER , MODULE_KEY_UNPRESS_HALFSHUTTER , dest);
MODULE_TRANSLATE_KEY(BGMT_PRESS_FULLSHUTTER , MODULE_KEY_PRESS_FULLSHUTTER , dest);
MODULE_TRANSLATE_KEY(BGMT_UNPRESS_FULLSHUTTER , MODULE_KEY_UNPRESS_FULLSHUTTER , dest);
MODULE_TRANSLATE_KEY(BGMT_TOUCH_1_FINGER , MODULE_KEY_TOUCH_1_FINGER , dest);
MODULE_TRANSLATE_KEY(BGMT_UNTOUCH_1_FINGER , MODULE_KEY_UNTOUCH_1_FINGER , dest);
MODULE_TRANSLATE_KEY(BGMT_TOUCH_2_FINGER , MODULE_KEY_TOUCH_2_FINGER , dest);
MODULE_TRANSLATE_KEY(BGMT_UNTOUCH_2_FINGER , MODULE_KEY_UNTOUCH_2_FINGER , dest);
MODULE_TRANSLATE_KEY(BGMT_Q , MODULE_KEY_Q , dest);
/* these are not simple key codes, so they will not work with MODULE_TRANSLATE_KEY */
//~ MODULE_TRANSLATE_KEY(BGMT_PRESS_FLASH_MOVIE , MODULE_KEY_PRESS_FLASH_MOVIE , dest);
//~ MODULE_TRANSLATE_KEY(BGMT_UNPRESS_FLASH_MOVIE , MODULE_KEY_UNPRESS_FLASH_MOVIE , dest);
return 0;
}
#undef MODULE_TRANSLATE_KEY
int module_send_keypress(int module_key)
{
int key = module_translate_key(module_key, MODULE_KEY_CANON);
fake_simple_button(key);
return 0;
}
int handle_module_keys(struct event * event)
{
for(int mod = 0; mod < MODULE_COUNT_MAX; mod++)
{
module_cbr_t *cbr = module_list[mod].cbr;
if(module_list[mod].valid && cbr)
{
while(cbr->name)
{
if(cbr->type == CBR_KEYPRESS)
{
/* key got handled? */
if(!cbr->handler(module_translate_key(event->param, MODULE_KEY_PORTABLE)))
{
return 0;
}
}
if(cbr->type == CBR_KEYPRESS_RAW)
{
/* key got handled? */
if(!cbr->handler((int)event))
{
return 0;
}
}
cbr++;
}
}
}
/* noone handled */
return 1;
}
int module_display_filter_enabled()
{
#ifdef CONFIG_DISPLAY_FILTERS
for(int mod = 0; mod < MODULE_COUNT_MAX; mod++)
{
module_cbr_t *cbr = module_list[mod].cbr;
if(module_list[mod].valid && cbr)
{
while(cbr->name)
{
if(cbr->type == CBR_DISPLAY_FILTER)
{
/* arg=0: should this display filter run? */
cbr->ctx = cbr->handler(0);
if (cbr->ctx)
return 1;
}
cbr++;
}
}
}
#endif
return 0;
}
int module_display_filter_update()
{
#ifdef CONFIG_DISPLAY_FILTERS
for(int mod = 0; mod < MODULE_COUNT_MAX; mod++)
{
module_cbr_t *cbr = module_list[mod].cbr;
if(module_list[mod].valid && cbr)
{
while(cbr->name)
{
if(cbr->type == CBR_DISPLAY_FILTER && cbr->ctx)
{
/* arg!=0: draw the filtered image in these buffers */
struct display_filter_buffers buffers;
display_filter_get_buffers((uint32_t**)&(buffers.src_buf), (uint32_t**)&(buffers.dst_buf));
if (buffers.src_buf && buffers.dst_buf) /* do not call the CBR with invalid arguments */
{
cbr->handler((intptr_t) &buffers);
}
break;
}
cbr++;
}
}
}
#endif
return 0;
}
static MENU_SELECT_FUNC(module_menu_update_select)
{
char enable_file[MODULE_FILENAME_LENGTH];
int mod_number = (int) priv;
module_list[mod_number].enabled = !module_list[mod_number].enabled;
snprintf(enable_file, sizeof(enable_file), "%s%s.en", get_config_dir(), module_list[mod_number].name);
config_flag_file_setting_save(enable_file, module_list[mod_number].enabled);
}
static const char* module_get_string(int mod_number, const char* name);
static int startswith(const char* str, const char* prefix)
{
const char* s = str;
const char* p = prefix;
for (; *p; s++,p++)
if (*s != *p) return 0;
return 1;
}
/* used if we don't have any module strings */
static module_strpair_t module_default_strings [] = {
{0, 0}
};
static MENU_UPDATE_FUNC(module_menu_update_entry)
{
int mod_number = (int) entry->priv;
if(module_list[mod_number].valid)
{
if(module_list[mod_number].info && module_list[mod_number].info->long_name)
{
MENU_SET_NAME(module_list[mod_number].info->long_name);
}
else
{
MENU_SET_NAME(module_list[mod_number].name);
}
if (!module_list[mod_number].enabled)
{
MENU_SET_ICON(MNI_NEUTRAL, 0);
MENU_SET_ENABLED(0);
MENU_SET_VALUE("OFF, will not load");
MENU_SET_WARNING(MENU_WARN_NOT_WORKING, "This module will no longer be loaded at next reboot.");
}
else
{
MENU_SET_ICON(MNI_ON, 0);
MENU_SET_ENABLED(1);
MENU_SET_VALUE(module_list[mod_number].status);
if (module_list[mod_number].error)
{
MENU_SET_WARNING(MENU_WARN_NOT_WORKING, "%s.", module_list[mod_number].long_status);
}
else
{
MENU_SET_WARNING(MENU_WARN_INFO, "%s. Press %s for more info.", module_list[mod_number].long_status, Q_BTN_NAME);
}
}
}
else if(strlen(module_list[mod_number].filename))
{
MENU_SET_NAME(module_list[mod_number].filename);
str_make_lowercase(info->name);
if (module_list[mod_number].enabled)
{
MENU_SET_ICON(MNI_ON, 0);
MENU_SET_ENABLED(0);
MENU_SET_VALUE("ON, will load");
MENU_SET_WARNING(MENU_WARN_ADVICE, "This module will be loaded at next reboot.");
}
else
{
MENU_SET_ICON(MNI_OFF, 0);
MENU_SET_ENABLED(1);
MENU_SET_VALUE(module_list[mod_number].status);
if (module_list[mod_number].strings && module_list[mod_number].strings != module_default_strings)
{
MENU_SET_WARNING(MENU_WARN_NOT_WORKING, "%s. Press %s for more info.", module_list[mod_number].long_status, Q_BTN_NAME);
}
else
{
MENU_SET_WARNING(MENU_WARN_NOT_WORKING, "%s.", module_list[mod_number].long_status);
}
}
}
else
{
MENU_SET_NAME("");
MENU_SET_ICON(MNI_NONE, 0);
MENU_SET_ENABLED(1);
MENU_SET_VALUE("(nonexistent)");
MENU_SET_HELP("You should never see this");
}
static int last_menu_activity_time = 0;
static void* prev_selected = 0;
if (entry->selected && entry != prev_selected)
{
last_menu_activity_time = get_ms_clock_value();
prev_selected = entry;
}
if (entry->selected)
{
if (!module_list[mod_number].valid && !module_list[mod_number].strings)
{
char* fn = module_list[mod_number].long_filename;
if (fn)
{
msg_queue_post(module_mq, MSG_MODULE_LOAD_OFFLINE_STRINGS | (mod_number << 16));
}
}
}
/* clean up offline strings if the module menu is no longer used */
if (!entry->selected && get_ms_clock_value() > 3000 + last_menu_activity_time)
{
if (
!module_list[mod_number].valid &&
module_list[mod_number].strings &&
module_list[mod_number].strings != module_default_strings
)
{
/* module strings loaded from elf, module not loaded, clean them up */
msg_queue_post(module_mq, MSG_MODULE_UNLOAD_OFFLINE_STRINGS | (mod_number << 16));
}
}
/* show info based on module strings metadata */
if (module_list[mod_number].strings)
{
const char* summary = module_get_string(mod_number, "Summary");
if (summary)
{
int has_dot = summary[strlen(summary)-1] == '.';
MENU_SET_HELP("%s%s", summary, has_dot ? "" : ".");
}
if (module_list[mod_number].valid == module_list[mod_number].enabled)
{
const char* name = module_get_string(mod_number, "Name");
if (name)
{
int fg = COLOR_GRAY(40);
int bg = COLOR_BLACK;
int fnt = SHADOW_FONT(FONT(FONT_MED_LARGE, fg, bg));
bmp_printf(fnt | FONT_ALIGN_RIGHT | FONT_TEXT_WIDTH(320), 680, info->y+2, "%s", name);
}
}
}
}
static MENU_SELECT_FUNC(module_menu_select_empty)
{
}
/* check which modules are loaded and hide others */
static void module_menu_update()
{
int mod_number = 0;
struct menu_entry * entry = module_menu;
while (entry)
{
/* only update those which display module information */
if(entry->update == module_menu_update_entry)
{
if(module_list[mod_number].valid)
{
MENU_SET_SHIDDEN(0);
}
else if(strlen(module_list[mod_number].filename))
{
MENU_SET_SHIDDEN(0);
}
else
{
MENU_SET_SHIDDEN(1);
}
mod_number++;
}
entry = entry->next;
}
}
/* check which modules are loaded and hide others */
static void module_submenu_update(int mod_number)
{
/* set autoload menu's priv to module id */
module_submenu[0].priv = (void*)mod_number;
};
static int module_info_type = 0;
static MENU_SELECT_FUNC(module_info_toggle)
{
module_info_type++;
if (module_info_type > 1)
{
module_info_type = 0;
menu_close_submenu();
}
}
static const char* module_get_string(int mod_number, const char* name)
{
module_strpair_t *strings = module_list[mod_number].strings;
if (strings)
{
for ( ; strings->name != NULL; strings++)
{
if (streq(strings->name, name))
{
return strings->value;
}
}
}
return 0;
}
static int module_is_special_string(const char* name)
{
if (
streq(name, "Name") ||
streq(name, "Description") ||
streq(name, "Build user") ||
streq(name, "Build date") ||
streq(name, "Last update") ||
streq(name, "Summary") ||
streq(name, "Forum") ||
startswith(name, "Help page") ||
0)
return 1;
return 0;
}
static int module_show_about_page(int mod_number)
{
module_strpair_t *strings = module_list[mod_number].strings;
if (strings)
{
int max_width = 0;
int max_width_value = 0;
int num_extra_strings = 0;
for ( ; strings->name != NULL; strings++)
{
if (!module_is_special_string(strings->name))
{
max_width = MAX(max_width, strlen(strings->name) + strlen(strings->value) + 3);
max_width_value = MAX(max_width_value, strlen(strings->value) + 2);
num_extra_strings++;
}
}
const char* desc = module_get_string(mod_number, "Description");
const char* name = module_get_string(mod_number, "Name");
const char* module_build_user = module_get_string(mod_number, "Build user");
const char* module_build_date = module_get_string(mod_number, "Build date");
const char* module_last_update = module_get_string(mod_number, "Last update");
if (name && desc)
{
bmp_fill(COLOR_BLACK, 0, 0, 720, 480);
int fnt_special = FONT(FONT_MED, COLOR_CYAN, COLOR_BLACK);
bmp_printf(FONT_LARGE, 10, 10, "%s", name);
big_bmp_printf(FONT_MED | FONT_ALIGN_JUSTIFIED | FONT_TEXT_WIDTH(690), 10, 60, "%s", desc);
int xm = 710 - max_width_value * font_med.width;
int xl = 710 - max_width * font_med.width;
xm = xm - xl + 10;
xl = 10;
int lines_for_update_msg = strchr(module_last_update, '\n') ? 3 : 2;
int yr = 480 - (num_extra_strings + lines_for_update_msg) * font_med.height;
for (strings = module_list[mod_number].strings ; strings->name != NULL; strings++)
{
if (!module_is_special_string(strings->name))
{
bmp_printf(fnt_special, xl, yr, "%s", strings->name);
bmp_printf(fnt_special, xm, yr, ": %s", strings->value);
yr += font_med.height;
}
}
if (module_build_date && module_build_user)
{
bmp_printf(fnt_special, 10, 480-font_med.height, "Built on %s by %s.", module_build_date, module_build_user);
}
if (module_last_update)
{
bmp_printf(fnt_special, 10, 480-font_med.height * lines_for_update_msg, "Last update: %s", module_last_update);
}
return 1;
}
}
return 0;
}
static MENU_UPDATE_FUNC(module_menu_info_update)
{
int mod_number = (int) module_submenu[0].priv;
int x = info->x;
int y = info->y;
int x_val = info->x_val;
if (!x || !y)
return;
info->custom_drawing = CUSTOM_DRAW_THIS_MENU;
if (module_info_type == 0 && strlen(module_list[mod_number].long_filename))
{
/* try to show an About page for the module */
if (module_show_about_page(mod_number))
return;
}
/* make sure this module is being used */
if(module_list[mod_number].valid && !module_list[mod_number].error)
{
module_strpair_t *strings = module_list[mod_number].strings;
module_parminfo_t *parms = module_list[mod_number].params;
module_cbr_t *cbr = module_list[mod_number].cbr;
module_prophandler_t **props = module_list[mod_number].prop_handlers;
if (strings)
{
y += 10;
bmp_printf(FONT_MED, x - 32, y, "Information:");
y += font_med.height;
for ( ; strings->name != NULL; strings++)
{
if (strchr(strings->value, '\n'))
{
continue; /* don't display multiline strings here */
}
int is_short_string = strlen(strings->value) * font_med.width + x_val < 710;
if (module_is_special_string(strings->name) && !is_short_string)
{
continue; /* don't display long strings that are already shown on the info page */
}
bmp_printf(FONT_MED, x, y, "%s", strings->name);
if (is_short_string)
{
/* short string */
bmp_printf(FONT_MED, x_val, y, "%s", strings->value);
}
else
{
/* long string */
if ((strlen(strings->name) + strlen(strings->value)) * font_med.width > 710)
{
/* doesn't fit? print on the next line */
y += font_med.height;
}
/* right-align if possible */
int new_x = MAX(x, 710 - strlen(strings->value) * font_med.width);
bmp_printf(FONT_MED, new_x, y, "%s", strings->value);
}
y += font_med.height;
}
}
if (parms)
{
y += 10;
bmp_printf(FONT_MED, x - 32, y, "Parameters:");
y += font_med.height;
for (; parms->name != NULL; parms++)
{
bmp_printf(FONT_MED, x, y, "%s", parms->name);
bmp_printf(FONT_MED, x_val, y, "%s", parms->type);
y += font_med.height;
}
}
if (props && *props)
{
y += 10;
bmp_printf(FONT_MED, x - 32, y,
#if defined(CONFIG_UNREGISTER_PROP)
"Properties:"
#else
"Properties (no support):"
#endif
);
y += font_med.height;
for (; *props != NULL; props++)
{
bmp_printf(FONT_MED, x, y, "%s", (*props)->name);
y += font_med.height;
}
}
if (cbr)
{
y += 10;
bmp_printf(FONT_MED, x - 32, y, "Callbacks:");
y += font_med.height;
for ( ; cbr->name != NULL; cbr++)
{
bmp_printf(FONT_MED, x, y, "%s", cbr->name);
bmp_printf(FONT_MED, x_val, y, "%s", cbr->symbol);
y += font_med.height;
}
}
}
else
{
bmp_printf(FONT_MED, x - 32, y, "%s", module_list[mod_number].long_filename);
y += font_med.height;
bmp_printf(FONT_MED, x - 32, y, "More info after you load this module.");
}
}
static MENU_SELECT_FUNC(module_menu_load)
{
console_show();
module_load_all();
}
static MENU_SELECT_FUNC(module_menu_unload)
{
module_unload_all();
}
static MENU_SELECT_FUNC(module_open_submenu)
{
int mod_number = (int)priv;
module_submenu_update(mod_number);
menu_open_submenu();
}
static MENU_SELECT_FUNC(console_toggle)
{
module_console_enabled = !module_console_enabled;
if (module_console_enabled)
console_show();
else
console_hide();
}
static struct menu_entry module_submenu[] = {
{
.name = "Module info",
.update = module_menu_info_update,
.select = module_info_toggle,
.icon_type = IT_ACTION,
},
MENU_EOL
};
#define MODULE_ENTRY(i) \
{ \
.name = "Module", \
.priv = (void*)i, \
.select = module_menu_update_select, \
.select_Q = module_open_submenu, \
.update = module_menu_update_entry, \
.submenu_width = 700, \
.submenu_height = 400, \
.children = module_submenu, \
},
static struct menu_entry module_menu[] = {
MODULE_ENTRY(0)
MODULE_ENTRY(1)
MODULE_ENTRY(2)
MODULE_ENTRY(3)
MODULE_ENTRY(4)
MODULE_ENTRY(5)
MODULE_ENTRY(6)
MODULE_ENTRY(7)
MODULE_ENTRY(8)
MODULE_ENTRY(9)
MODULE_ENTRY(10)
MODULE_ENTRY(11)
MODULE_ENTRY(12)
MODULE_ENTRY(13)
MODULE_ENTRY(14)
MODULE_ENTRY(15)
MODULE_ENTRY(16)
MODULE_ENTRY(17)
MODULE_ENTRY(18)
MODULE_ENTRY(19)
MODULE_ENTRY(20)
MODULE_ENTRY(21)
MODULE_ENTRY(22)
MODULE_ENTRY(23)
MODULE_ENTRY(24)
MODULE_ENTRY(25)
MODULE_ENTRY(26)
MODULE_ENTRY(27)
MODULE_ENTRY(28)
MODULE_ENTRY(29)
MODULE_ENTRY(30)
MODULE_ENTRY(31)
};
static struct menu_entry module_debug_menu[] = {
{
.name = "Modules debug",
.select = menu_open_submenu,
.submenu_width = 710,
.help = "Diagnostic options for modules.",
.children = (struct menu_entry[]) {
#if 0
{
.name = "Load modules now",
.select = module_menu_load,
.help = "Loads modules in "MODULE_PATH,
},
#endif
#ifdef CONFIG_MODULE_UNLOAD
{
.name = "Unload modules now...",
.select = module_menu_unload,
.help = "Unload loaded modules",
},
#endif
{
.name = "Disable all modules",
.priv = &module_autoload_disabled,
.max = 1,
.help = "For troubleshooting.",
},
{
.name = "Load modules after crash",
.priv = &module_ignore_crashes,
.max = 1,
.help = "Load modules even after camera crashed and you took battery out.",
},
{
.name = "Show console",
.priv = &module_console_enabled,
.select = console_toggle,
.max = 1,
.help = "Keep console shown after modules were loaded",
},
MENU_EOL,
},
},
};
struct config_var * module_get_config_var(const char * name)
{
for(int mod = 0; mod < MODULE_COUNT_MAX; mod++)
{
module_config_t * config = module_list[mod].config;
if(module_list[mod].valid && config)
{
for (module_config_t * mconfig = config; mconfig && mconfig->name; mconfig++)
{
if (streq(mconfig->ref->name, name))
return (struct config_var *) mconfig->ref;
}
}
}
return 0;
}
struct config_var* module_config_var_lookup(int* ptr)
{
for(int mod = 0; mod < MODULE_COUNT_MAX; mod++)
{
module_config_t * config = module_list[mod].config;
if(module_list[mod].valid && config)
{
for (module_config_t * mconfig = config; mconfig && mconfig->name; mconfig++)
{
if (mconfig->ref->value == ptr)
return (struct config_var *) mconfig->ref;
}
}
}
return 0;
}
static void module_init()
{
module_mq = (struct msg_queue *) msg_queue_create("module_mq", 10);
menu_add("Modules", module_menu, COUNT(module_menu));
menu_add("Debug", module_debug_menu, COUNT(module_debug_menu));
module_menu_update();
}
static void module_load_offline_strings(int mod_number)
{
if (!module_list[mod_number].strings)
{
/* default value, if it won't work */
module_list[mod_number].strings = module_default_strings;
char* fn = module_list[mod_number].long_filename;
/* we should free this one after we are done with it */
extern void* tcc_load_offline_section(char* filename, char* section_name);
module_strpair_t * strings = tcc_load_offline_section(fn, ".module_strings");
if (strings)
{
int looks_ok = 1;
/* relocate strings from the unprocessed elf section */
/* question: is the string structure always at the very beginning of the section? */
for (module_strpair_t * str = strings; str->name != NULL; str++)
{
if ((intptr_t)str->name < 10000) /* does it look like a non-relocated pointer? */
{
str->name = (void*) str->name + (intptr_t) strings;
str->value = (void*) str->value + (intptr_t) strings;
}
else
{
looks_ok = 0;
break;
}
}
if (looks_ok)
{
/* use as module strings */
module_list[mod_number].strings = strings;
}
}
}
}
static void module_unload_offline_strings(int mod_number)
{
if (module_list[mod_number].strings)
{
free(module_list[mod_number].strings);
module_list[mod_number].strings = 0;
}
}
static void module_load_task(void* unused)
{
char *lockstr = "If you can read this, ML crashed last time. To save from faulty modules, autoload gets disabled.";
if(!module_autoload_disabled)
{
uint32_t size;
if(!module_ignore_crashes && FIO_GetFileSize( module_lockfile, &size ) == 0 )
{
/* uh, it seems the camera didnt shut down cleanly, skip module loading this time */
msleep(1000);
NotifyBox(10000, "Camera was not shut down cleanly.\r\nSkipping module loading." );
}
else
{
FILE *handle = FIO_CreateFile(module_lockfile);
FIO_WriteFile(handle, lockstr, strlen(lockstr));
FIO_CloseFile(handle);
/* now load modules */
_module_load_all(0);
module_menu_update();
}
}
else
{
/* only list modules */
_module_load_all(1);
module_menu_update();
}
/* main loop, also wait until clean shutdown */
TASK_LOOP
{
int msg;
int err = msg_queue_receive(module_mq, (struct event**)&msg, 200);
if (err) continue;
switch(msg & 0xFFFF)
{
case MSG_MODULE_LOAD_ALL:
_module_load_all(0);
module_menu_update();
break;
case MSG_MODULE_UNLOAD_ALL:
_module_unload_all();
_module_load_all(1);
module_menu_update();
beep();
break;
case MSG_MODULE_LOAD_OFFLINE_STRINGS:
{
int mod_number = msg >> 16;
module_load_offline_strings(mod_number);
break;
}
case MSG_MODULE_UNLOAD_OFFLINE_STRINGS:
{
int mod_number = msg >> 16;
module_unload_offline_strings(mod_number);
break;
}
default:
console_printf("invalid msg: %d\n", msg);
}
}
}
void module_save_configs()
{
/* save configuration */
console_printf("Save configs...\n");
for(int mod = 0; mod < MODULE_COUNT_MAX; mod++)
{
if(module_list[mod].valid && module_list[mod].enabled && !module_list[mod].error && module_list[mod].config)
{
/* save config */
char filename[64];
snprintf(filename, sizeof(filename), "%s%s.cfg", get_config_dir(), module_list[mod].name);
uint32_t ret = module_config_save(filename, &module_list[mod]);
if(ret)
{
console_printf(" [E] Error: %d\n", ret);
}
}
}
}
/* clean shutdown, unlink lockfile */
int module_shutdown()
{
_module_unload_all();
if(!module_autoload_disabled)
{
/* remove lockfile */
FIO_RemoveFile(module_lockfile);
}
return 0;
}
TASK_CREATE("module_task", module_load_task, 0, 0x1e, 0x4000 );
INIT_FUNC(__FILE__, module_init);
Computing file changes ...