352 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			352 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
/* $XFree86: xc/programs/Xserver/hw/xfree86/loader/hash.c,v 1.25 2003/11/23 00:57:56 dawes Exp $ */
 | 
						|
 | 
						|
/*
 | 
						|
 *
 | 
						|
 * Copyright 1995-1998 by Metro Link, Inc.
 | 
						|
 *
 | 
						|
 * Permission to use, copy, modify, distribute, and sell this software and its
 | 
						|
 * documentation for any purpose is hereby granted without fee, provided that
 | 
						|
 * the above copyright notice appear in all copies and that both that
 | 
						|
 * copyright notice and this permission notice appear in supporting
 | 
						|
 * documentation, and that the name of Metro Link, Inc. not be used in
 | 
						|
 * advertising or publicity pertaining to distribution of the software without
 | 
						|
 * specific, written prior permission.  Metro Link, Inc. makes no
 | 
						|
 * representations about the suitability of this software for any purpose.
 | 
						|
 *  It is provided "as is" without express or implied warranty.
 | 
						|
 *
 | 
						|
 * METRO LINK, INC. DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
 | 
						|
 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
 | 
						|
 * EVENT SHALL METRO LINK, INC. BE LIABLE FOR ANY SPECIAL, INDIRECT OR
 | 
						|
 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
 | 
						|
 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
 | 
						|
 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
 | 
						|
 * PERFORMANCE OF THIS SOFTWARE.
 | 
						|
 */
 | 
						|
 | 
						|
#include "os.h"
 | 
						|
#include "Xos.h"
 | 
						|
#undef abs
 | 
						|
#include <stdlib.h>
 | 
						|
#include "sym.h"
 | 
						|
#include "loader.h"
 | 
						|
#include "hash.h"
 | 
						|
 | 
						|
#if defined(Lynx)
 | 
						|
#define MAXINT	32000
 | 
						|
#else
 | 
						|
#include <limits.h>
 | 
						|
#undef MAXINT
 | 
						|
#define MAXINT INT_MAX
 | 
						|
#endif
 | 
						|
 | 
						|
/* Prototypes for static functions. */
 | 
						|
static unsigned int hashFunc(const char *);
 | 
						|
static itemPtr LoaderHashFindNearest(unsigned long);
 | 
						|
 | 
						|
static itemPtr LoaderhashTable[HASHSIZE];
 | 
						|
 | 
						|
#ifdef DEBUG
 | 
						|
static int hashhits[HASHSIZE];
 | 
						|
 | 
						|
void
 | 
						|
DumpHashHits(void)
 | 
						|
{
 | 
						|
    int i;
 | 
						|
    int depth = 0;
 | 
						|
    int dev = 0;
 | 
						|
 | 
						|
    for (i = 0; i < HASHSIZE; i++) {
 | 
						|
	ErrorF("hashhits[%d]=%d\n", i, hashhits[i]);
 | 
						|
	depth += hashhits[i];
 | 
						|
    }
 | 
						|
 | 
						|
    depth /= HASHSIZE;
 | 
						|
    ErrorF("Average hash depth=%d\n", depth);
 | 
						|
 | 
						|
    for (i = 0; i < HASHSIZE; i++) {
 | 
						|
	if (hashhits[i] < depth)
 | 
						|
	    dev += depth - hashhits[i];
 | 
						|
	else
 | 
						|
	    dev += hashhits[i] - depth;
 | 
						|
    }
 | 
						|
 | 
						|
    dev /= HASHSIZE;
 | 
						|
    ErrorF("Average hash deviation=%d\n", dev);
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
static unsigned int
 | 
						|
hashFunc(const char *string)
 | 
						|
{
 | 
						|
    int i = 0;
 | 
						|
 | 
						|
    while (i < 10 && string[i])
 | 
						|
	i++;
 | 
						|
 | 
						|
    if (i < 5) {
 | 
						|
#ifdef DEBUG
 | 
						|
	hashhits[i]++;
 | 
						|
#endif
 | 
						|
	return i;
 | 
						|
    }
 | 
						|
 | 
						|
/*
 | 
						|
 * Original has function
 | 
						|
#define HASH ((string[ i-4 ] * string[i-3] + string[i-2] ) & (HASHSIZE-1))
 | 
						|
 */
 | 
						|
 | 
						|
#define HASH ((string[i-5] * string[ i-4 ] + string[i-3] * string[i-2] ) & (HASHSIZE-1))
 | 
						|
 | 
						|
#ifdef DEBUG
 | 
						|
    hashhits[HASH]++;
 | 
						|
#endif
 | 
						|
 | 
						|
    return HASH;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
LoaderHashAdd(itemPtr entry)
 | 
						|
{
 | 
						|
    int bucket = hashFunc(entry->name);
 | 
						|
    itemPtr oentry;
 | 
						|
 | 
						|
    if ((oentry = LoaderHashFind(entry->name)) != NULL)
 | 
						|
	LoaderDuplicateSymbol(entry->name, oentry->handle);
 | 
						|
 | 
						|
    entry->next = LoaderhashTable[bucket];
 | 
						|
    LoaderhashTable[bucket] = entry;
 | 
						|
    return;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
LoaderAddSymbols(int handle, int module, LOOKUP *list)
 | 
						|
{
 | 
						|
    LOOKUP *l = list, *exports = NULL;
 | 
						|
    itemPtr i, exportsItem = NULL;
 | 
						|
    char *modname;
 | 
						|
 | 
						|
    if (!list)
 | 
						|
	return;
 | 
						|
 | 
						|
    /*
 | 
						|
     * First look for a symbol called <name>ExportedSymbols.  If it exists,
 | 
						|
     * only export the symbols that are listed in that array.  Otherwise
 | 
						|
     * export all of the external symbols.
 | 
						|
     */
 | 
						|
    modname = _LoaderHandleToCanonicalName(handle);
 | 
						|
    if (modname) {
 | 
						|
	char *exportname;
 | 
						|
 | 
						|
	exportname = xf86loadermalloc(strlen("ExportedSymbols") +
 | 
						|
				      strlen(modname) + 1);
 | 
						|
	if (exportname) {
 | 
						|
	    sprintf(exportname, "%sExportedSymbols", modname);
 | 
						|
	    while (l->symName) {
 | 
						|
		if (strcmp(l->symName, exportname) == 0) {
 | 
						|
		    exports = l;
 | 
						|
		    ErrorF("LoaderAddSymbols: %s: %s found\n", modname,
 | 
						|
			   exportname);
 | 
						|
		    break;
 | 
						|
		}
 | 
						|
		l++;
 | 
						|
	    }
 | 
						|
	    xf86loaderfree(exportname);
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
    /*
 | 
						|
     * Allocate the exports list item first.
 | 
						|
     */
 | 
						|
    if (exports) {
 | 
						|
	exportsItem = xf86loadermalloc(sizeof(itemRec));
 | 
						|
	exportsItem->name = exports->symName;
 | 
						|
	exportsItem->address = (char *)exports->offset;
 | 
						|
	exportsItem->handle = handle;
 | 
						|
	exportsItem->module = module;
 | 
						|
	exportsItem->exports = NULL;
 | 
						|
	LoaderHashAdd(exportsItem);
 | 
						|
    }
 | 
						|
 | 
						|
    /*
 | 
						|
     * Visit every symbol in the lookup table, tagging it with the
 | 
						|
     * reference to the export list, if present.
 | 
						|
     */
 | 
						|
    l = list;
 | 
						|
    while (l->symName) {
 | 
						|
	if (l != exports) {
 | 
						|
	    i = xf86loadermalloc(sizeof(itemRec));
 | 
						|
	    i->name = l->symName;
 | 
						|
	    i->address = (char *)l->offset;
 | 
						|
	    i->handle = handle;
 | 
						|
	    i->module = module;
 | 
						|
	    i->exports = exportsItem;
 | 
						|
	    LoaderHashAdd(i);
 | 
						|
	}
 | 
						|
	l++;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
itemPtr
 | 
						|
LoaderHashDelete(const char *string)
 | 
						|
{
 | 
						|
    int bucket = hashFunc(string);
 | 
						|
    itemPtr entry;
 | 
						|
    itemPtr *entry2;
 | 
						|
 | 
						|
    entry = LoaderhashTable[bucket];
 | 
						|
    entry2 = &(LoaderhashTable[bucket]);
 | 
						|
    while (entry) {
 | 
						|
	if (!strcmp(entry->name, string)) {
 | 
						|
	    *entry2 = entry->next;
 | 
						|
	    xf86loaderfree(entry->name);
 | 
						|
	    xf86loaderfree(entry);
 | 
						|
	    return 0;
 | 
						|
	}
 | 
						|
	entry2 = &(entry->next);
 | 
						|
	entry = entry->next;
 | 
						|
    }
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
itemPtr
 | 
						|
LoaderHashFind(const char *string)
 | 
						|
{
 | 
						|
    int bucket = hashFunc(string);
 | 
						|
    itemPtr entry;
 | 
						|
 | 
						|
    entry = LoaderhashTable[bucket];
 | 
						|
    while (entry) {
 | 
						|
	if (!strcmp(entry->name, string)) {
 | 
						|
	    return entry;
 | 
						|
	}
 | 
						|
	entry = entry->next;
 | 
						|
    }
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
static itemPtr
 | 
						|
LoaderHashFindNearest(unsigned long address)
 | 
						|
{
 | 
						|
    int i;
 | 
						|
    itemPtr entry, best_entry = 0;
 | 
						|
    long best_difference = MAXINT;
 | 
						|
 | 
						|
    for (i = 0; i < HASHSIZE; i++) {
 | 
						|
	entry = LoaderhashTable[i];
 | 
						|
	while (entry) {
 | 
						|
	    long difference = (long)address - (long)entry->address;
 | 
						|
 | 
						|
	    if (difference >= 0) {
 | 
						|
		if (best_entry) {
 | 
						|
		    if (difference < best_difference) {
 | 
						|
			best_entry = entry;
 | 
						|
			best_difference = difference;
 | 
						|
		    }
 | 
						|
		} else {
 | 
						|
		    best_entry = entry;
 | 
						|
		    best_difference = difference;
 | 
						|
		}
 | 
						|
	    }
 | 
						|
	    entry = entry->next;
 | 
						|
	}
 | 
						|
    }
 | 
						|
    return best_entry;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
LoaderPrintSymbol(unsigned long address)
 | 
						|
{
 | 
						|
    itemPtr entry;
 | 
						|
 | 
						|
    entry = LoaderHashFindNearest(address);
 | 
						|
    if (entry) {
 | 
						|
	const char *module, *section;
 | 
						|
 | 
						|
#if defined(__alpha__) || defined(__ia64__)
 | 
						|
	ErrorF("0x%016lx %s+%lx\n", (unsigned long)entry->address,
 | 
						|
	       entry->name, address - (unsigned long)entry->address);
 | 
						|
#else
 | 
						|
	ErrorF("0x%lx %s+%lx\n", (unsigned long)entry->address, entry->name,
 | 
						|
	       address - (unsigned long)entry->address);
 | 
						|
#endif
 | 
						|
 | 
						|
	if (_LoaderAddressToSection(address, &module, §ion))
 | 
						|
	    ErrorF("\tModule \"%s\"\n\tSection \"%s\"\n", module, section);
 | 
						|
    } else {
 | 
						|
	ErrorF("(null)\n");
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
LoaderPrintItem(itemPtr pItem)
 | 
						|
{
 | 
						|
    if (pItem) {
 | 
						|
	const char *module, *section;
 | 
						|
 | 
						|
#if defined(__alpha__) || defined(__ia64__)
 | 
						|
	ErrorF("0x%016lx %s\n", (unsigned long)pItem->address, pItem->name);
 | 
						|
#else
 | 
						|
	ErrorF("0x%lx %s\n", (unsigned long)pItem->address, pItem->name);
 | 
						|
#endif
 | 
						|
	if (_LoaderAddressToSection((unsigned long)pItem->address,
 | 
						|
				    &module, §ion))
 | 
						|
	    ErrorF("\tModule \"%s\"\n\tSection \"%s\"\n", module, section);
 | 
						|
    } else
 | 
						|
	ErrorF("(null)\n");
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
LoaderPrintAddress(const char *symbol)
 | 
						|
{
 | 
						|
    itemPtr entry;
 | 
						|
 | 
						|
    entry = LoaderHashFind(symbol);
 | 
						|
    LoaderPrintItem(entry);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
LoaderHashTraverse(void *card, int (*fnp)(void *, itemPtr))
 | 
						|
{
 | 
						|
    int i;
 | 
						|
    itemPtr entry, last_entry = 0;
 | 
						|
 | 
						|
    for (i = 0; i < HASHSIZE; i++) {
 | 
						|
	last_entry = 0;
 | 
						|
	entry = LoaderhashTable[i];
 | 
						|
	while (entry) {
 | 
						|
	    if ((*fnp) (card, entry)) {
 | 
						|
		if (last_entry) {
 | 
						|
		    last_entry->next = entry->next;
 | 
						|
		    xf86loaderfree(entry->name);
 | 
						|
		    xf86loaderfree(entry);
 | 
						|
		    entry = last_entry->next;
 | 
						|
		} else {
 | 
						|
		    LoaderhashTable[i] = entry->next;
 | 
						|
		    xf86loaderfree(entry->name);
 | 
						|
		    xf86loaderfree(entry);
 | 
						|
		    entry = LoaderhashTable[i];
 | 
						|
		}
 | 
						|
	    } else {
 | 
						|
		last_entry = entry;
 | 
						|
		entry = entry->next;
 | 
						|
	    }
 | 
						|
	}
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
LoaderDumpSymbols()
 | 
						|
{
 | 
						|
    itemPtr entry;
 | 
						|
    int j;
 | 
						|
 | 
						|
    for (j = 0; j < HASHSIZE; j++) {
 | 
						|
	entry = LoaderhashTable[j];
 | 
						|
	while (entry) {
 | 
						|
	    LoaderPrintItem(entry);
 | 
						|
	    entry = entry->next;
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
}
 |