forked from len0rd/rockbox
		
	git-svn-id: svn://svn.rockbox.org/rockbox/trunk@25884 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			112 lines
		
	
	
	
		
			3.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			112 lines
		
	
	
	
		
			3.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   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:
 | |
|    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 void* parameter,
 | |
|                                unsigned char *filename, unsigned char *name)
 | |
| {
 | |
|     int fd, readsize;
 | |
|     size_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(&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));
 | |
| 
 | |
|     *(header.api) = rb;
 | |
|     return header.entry_point(parameter);
 | |
| }
 | |
| #endif
 |