00001 /* 00002 * Copyright (c) 2007, Swedish Institute of Computer Science 00003 * All rights reserved. 00004 * 00005 * Redistribution and use in source and binary forms, with or without 00006 * modification, are permitted provided that the following conditions 00007 * are met: 00008 * 1. Redistributions of source code must retain the above copyright 00009 * notice, this list of conditions and the following disclaimer. 00010 * 2. Redistributions in binary form must reproduce the above copyright 00011 * notice, this list of conditions and the following disclaimer in the 00012 * documentation and/or other materials provided with the distribution. 00013 * 3. Neither the name of the Institute nor the names of its contributors 00014 * may be used to endorse or promote products derived from this software 00015 * without specific prior written permission. 00016 * 00017 * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND 00018 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 00019 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 00020 * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE 00021 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 00022 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 00023 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 00024 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 00025 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 00026 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 00027 * SUCH DAMAGE. 00028 * 00029 * @(#)$Id: sym.c,v 1.1 2007/04/25 15:33:54 bg- Exp $ 00030 */ 00031 00032 #ifdef __AVR__ 00033 #include <avr/pgmspace.h> 00034 #endif 00035 00036 #include "loader/sym.h" 00037 00038 #include <string.h> 00039 00040 static union sym_value 00041 sym_lookup(const char *name, const struct sym_bol *symbols, int nelts) 00042 { 00043 union sym_value ret; 00044 int start, middle, end; 00045 int r; 00046 00047 start = 0; 00048 end = nelts - 1; 00049 00050 while(start <= end) { 00051 /* Check middle, divide */ 00052 middle = (start + end) / 2; 00053 #ifdef __AVR__ 00054 PGM_P addr = (PGM_P)pgm_read_word(&symbols[middle].name); 00055 r = strcmp_P(name, addr); 00056 #else 00057 r = strcmp(name, symbols[middle].name); 00058 #endif 00059 if(r < 0) { 00060 end = middle - 1; 00061 } else if(r > 0) { 00062 start = middle + 1; 00063 } else { 00064 #ifdef __AVR__ 00065 ret.func = (sym_func_t)pgm_read_word(&symbols[middle].value); 00066 return ret; 00067 #else 00068 return symbols[middle].value; 00069 #endif 00070 } 00071 } 00072 ret.obj = NULL; 00073 ret.func = NULL; 00074 return ret; 00075 } 00076 00077 /* Lookup a pointer to an ANSI C object. */ 00078 void * 00079 sym_object(const char *name) 00080 { 00081 union sym_value ret; 00082 ret = sym_lookup(name, sym_obj, sym_obj_nelts); 00083 if(ret.obj != NULL) 00084 return ret.obj; 00085 00086 /* 00087 * If the implementation puts constants into the text segment, this 00088 * is where to find them! 00089 */ 00090 ret = sym_lookup(name, sym_func, sym_func_nelts); 00091 return ret.obj; 00092 } 00093 00094 /* Lookup a pointer to an ANSI C function. */ 00095 sym_func_t 00096 sym_function(const char *name) 00097 { 00098 return sym_lookup(name, sym_func, sym_func_nelts).func; 00099 }