/*************************************************************************** * __________ __ ___. * Open \______ \ ____ ____ | | _\_ |__ _______ ___ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ * \/ \/ \/ \/ \/ * $Id$ * * Overlay loader * * Copyright (C) 2006 Jens Arnold * * 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. * ****************************************************************************/ #ifndef SIMULATOR #include "plugin.h" #include "overlay.h" /* load and run a plugin linked as an overlay. arguments: rb = pointer to plugin api, also passed on to the overlay parameter = plugin parameter, passed on to the overlay filename = overlay file name, absolute path as usual name = overlay display name result: return value from the overlay As long as a large plugin to be overlayed doesn't use the audiobuffer itself, no adjustments in the plugin source code are necessary to make it work as an overlay, it only needs an adapted linker script. If the overlayed plugin *does* use the audiobuffer itself, it needs to make sure not to overwrite itself. The linker script for the overlay should use a base address towards the end of the audiobuffer, just low enough to make the overlay fit. */ enum plugin_status run_overlay(const struct plugin_api* rb, const void* parameter, unsigned char *filename, unsigned char *name) { int fd, readsize; ssize_t audiobuf_size; unsigned char *audiobuf; static struct plugin_header header; fd = rb->open(filename, O_RDONLY); if (fd < 0) { rb->splashf(2*HZ, "Can't open %s", filename); return PLUGIN_ERROR; } readsize = rb->read(fd, &header, sizeof(header)); rb->close(fd); /* Close for now. Less code than doing it in all error checks. * Would need to seek back anyway. */ if (readsize != sizeof(header)) { rb->splashf(2*HZ, "Reading %s overlay failed.", name); return PLUGIN_ERROR; } if (header.magic != PLUGIN_MAGIC || header.target_id != TARGET_ID) { rb->splashf(2*HZ, "%s overlay: Incompatible model.", name); return PLUGIN_ERROR; } if (header.api_version != PLUGIN_API_VERSION) { rb->splashf(2*HZ, "%s overlay: Incompatible version.", name); return PLUGIN_ERROR; } audiobuf = rb->plugin_get_audio_buffer((size_t *)&audiobuf_size); if (header.load_addr < audiobuf || header.end_addr > audiobuf + audiobuf_size) { rb->splashf(2*HZ, "%s overlay doesn't fit into memory.", name); return PLUGIN_ERROR; } fd = rb->open(filename, O_RDONLY); if (fd < 0) { rb->splashf(2*HZ, "Can't open %s", filename); return PLUGIN_ERROR; } readsize = rb->read(fd, header.load_addr, header.end_addr - header.load_addr); rb->close(fd); if (readsize < 0) { rb->splashf(2*HZ, "Reading %s overlay failed.", name); return PLUGIN_ERROR; } /* Zero out bss area */ rb->memset(header.load_addr + readsize, 0, header.end_addr - (header.load_addr + readsize)); return header.entry_point(rb, parameter); } #endif