131 lines
2.9 KiB
C
131 lines
2.9 KiB
C
/*
|
|
* Copyright © 2005 Novell, 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
|
|
* Novell, Inc. not be used in advertising or publicity pertaining to
|
|
* distribution of the software without specific, written prior permission.
|
|
* Novell, Inc. makes no representations about the suitability of this
|
|
* software for any purpose. It is provided "as is" without express or
|
|
* implied warranty.
|
|
*
|
|
* NOVELL, INC. DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
|
|
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN
|
|
* NO EVENT SHALL NOVELL, 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.
|
|
*
|
|
* Author: David Reveman <davidr@novell.com>
|
|
*/
|
|
|
|
#include "xgl.h"
|
|
#include "xglmodule.h"
|
|
|
|
#ifdef XGL_MODULAR
|
|
|
|
#include <dlfcn.h>
|
|
|
|
#define SYM(ptr, name) { (void **) &(ptr), (name) }
|
|
|
|
void *
|
|
xglLoadModule (const char *name,
|
|
int flag)
|
|
{
|
|
ModuleVersionProcPtr moduleVersion;
|
|
ModuleInitProcPtr moduleInit;
|
|
void *handle = 0;
|
|
char *module;
|
|
xglSymbolRec mSym[] = {
|
|
SYM (moduleVersion, "moduleVersion"),
|
|
SYM (moduleInit, "moduleInit")
|
|
};
|
|
|
|
module = malloc (strlen (XGL_MODULE_PATH "/lib.so") + strlen (name) + 1);
|
|
if (!module)
|
|
return 0;
|
|
|
|
sprintf (module, XGL_MODULE_PATH "/lib%s.so", name);
|
|
|
|
handle = dlopen (module, flag);
|
|
if (handle)
|
|
{
|
|
if (xglLookupSymbols (handle, mSym, sizeof (mSym) / sizeof (mSym[0])))
|
|
{
|
|
const char *version;
|
|
|
|
version = (*moduleVersion) ();
|
|
if (strcmp (VERSION, version) == 0)
|
|
{
|
|
if (!(*moduleInit) (module))
|
|
{
|
|
dlclose (handle);
|
|
handle = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ErrorF ("Module version mismatch. "
|
|
"%s is %s Xserver is" VERSION "\n",
|
|
module, version);
|
|
|
|
dlclose (handle);
|
|
handle = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
dlclose (handle);
|
|
handle = 0;
|
|
}
|
|
}
|
|
else
|
|
ErrorF ("dlopen: %s\n", dlerror ());
|
|
|
|
free (module);
|
|
|
|
return handle;
|
|
}
|
|
|
|
void
|
|
xglUnloadModule (void *handle)
|
|
{
|
|
dlclose (handle);
|
|
}
|
|
|
|
Bool
|
|
xglLookupSymbols (void *handle,
|
|
xglSymbolPtr sym,
|
|
int nSym)
|
|
{
|
|
void *symbol;
|
|
char *error;
|
|
int i;
|
|
|
|
/* avoid previous error */
|
|
dlerror ();
|
|
|
|
for (i = 0; i < nSym; i++)
|
|
{
|
|
symbol = dlsym (handle, sym[i].name);
|
|
if (!symbol)
|
|
{
|
|
error = dlerror ();
|
|
if (error != 0)
|
|
ErrorF ("dlsym: %s: %s\n", sym[i].name, error);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
*(sym[i].ptr) = symbol;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#endif
|