2011-05-13 08:22:28 +00:00
|
|
|
/* radare - LGPL - Copyright 2008-2011 pancake<nopcode.org> */
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
|
|
#include "r_types.h"
|
2010-06-03 08:57:34 +00:00
|
|
|
#include "r_util.h"
|
2009-02-05 21:08:46 +00:00
|
|
|
#include "r_lib.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <dirent.h>
|
|
|
|
|
|
|
|
/* TODO: support for nested plugins ?? here */
|
|
|
|
|
|
|
|
#if __UNIX__
|
|
|
|
#include <dlfcn.h>
|
2009-03-08 20:41:02 +00:00
|
|
|
#define DLOPEN(x) dlopen(x, RTLD_GLOBAL | RTLD_NOW)
|
2009-02-05 21:08:46 +00:00
|
|
|
#define DLSYM(x,y) dlsym(x,y)
|
|
|
|
#define DLCLOSE(x) dlclose(x)
|
2010-04-12 00:22:52 +00:00
|
|
|
#elif __WINDOWS__
|
2009-02-05 21:08:46 +00:00
|
|
|
#include <windows.h>
|
|
|
|
#define DLOPEN(x) LoadLibrary(x)
|
|
|
|
#define DLSYM(x,y) GetProcAddress(x,y)
|
2010-01-15 12:02:54 +00:00
|
|
|
#define DLCLOSE(x) 0//(x)
|
|
|
|
//CloseLibrary(x)
|
2009-02-05 21:08:46 +00:00
|
|
|
#else
|
|
|
|
#define DLOPEN(x) NULL
|
|
|
|
#define DLSYM(x,y) NULL
|
|
|
|
#define DLCLOSE(x) NULL
|
|
|
|
#endif
|
|
|
|
|
2009-09-17 09:48:36 +00:00
|
|
|
/* XXX : this must be registered in runtime */
|
|
|
|
static const char *r_lib_types[] = {
|
2010-02-21 19:21:36 +00:00
|
|
|
"io", "dbg", "lang", "asm", "anal", "parse", "bin", //"bininfo",
|
2010-05-20 23:46:26 +00:00
|
|
|
"bp", "syscall", "fastcall", "crypto", "cmd", NULL
|
2009-04-01 00:28:13 +00:00
|
|
|
};
|
|
|
|
|
2010-06-03 08:57:34 +00:00
|
|
|
static int r_lib_debug_enabled = 0;
|
|
|
|
|
2009-09-17 09:48:36 +00:00
|
|
|
/* XXX: Rename this helper function */
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API const char *r_lib_types_get(int idx) {
|
2009-04-01 01:40:04 +00:00
|
|
|
if (idx<0||idx>R_LIB_TYPE_LAST)
|
2009-04-01 01:46:37 +00:00
|
|
|
return "unk";
|
|
|
|
return r_lib_types[idx];
|
2009-04-01 01:40:04 +00:00
|
|
|
}
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API void *r_lib_dl_open(const char *libname) {
|
2009-03-31 00:50:02 +00:00
|
|
|
void *ret;
|
2010-04-12 00:22:52 +00:00
|
|
|
ret = DLOPEN (libname);
|
2010-06-03 08:57:34 +00:00
|
|
|
if (r_lib_debug_enabled && ret == NULL)
|
2010-01-15 12:02:54 +00:00
|
|
|
#if __UNIX__
|
2010-06-03 08:57:34 +00:00
|
|
|
eprintf ("dlerror(%s): %s\n", libname, dlerror ());
|
2010-01-15 12:02:54 +00:00
|
|
|
#else
|
|
|
|
eprintf ("r_lib_dl_open: Cannot open '%s'\n", libname);
|
|
|
|
#endif
|
2009-02-05 21:08:46 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-05-26 16:25:35 +00:00
|
|
|
R_API void *r_lib_dl_sym(void *handler, const char *name) {
|
|
|
|
return DLSYM (handler, name);
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-05-26 16:25:35 +00:00
|
|
|
R_API int r_lib_dl_close(void *handler) {
|
|
|
|
return DLCLOSE (handler);
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* ---- */
|
|
|
|
|
2010-05-20 15:40:58 +00:00
|
|
|
R_API RLib *r_lib_new(const char *symname) {
|
2011-05-13 08:22:28 +00:00
|
|
|
RLib *lib = R_NEW (RLib);
|
2009-09-17 09:48:36 +00:00
|
|
|
if (lib) {
|
2010-06-03 08:57:34 +00:00
|
|
|
r_lib_debug_enabled = r_sys_getenv ("R_DEBUG")?R_TRUE:R_FALSE;
|
2011-05-13 08:22:28 +00:00
|
|
|
lib->handlers = r_list_new ();
|
|
|
|
lib->plugins = r_list_new ();
|
2010-04-12 00:22:52 +00:00
|
|
|
strncpy (lib->symname, symname, sizeof (lib->symname)-1);
|
2009-09-17 09:48:36 +00:00
|
|
|
}
|
|
|
|
return lib;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API RLib *r_lib_free(RLib *lib) {
|
2009-02-05 21:08:46 +00:00
|
|
|
/* TODO: iterate over libraries and free them all */
|
|
|
|
/* TODO: iterate over handlers and free them all */
|
2010-04-12 00:22:52 +00:00
|
|
|
r_lib_close (lib, NULL);
|
2009-02-05 21:08:46 +00:00
|
|
|
free (lib);
|
2009-09-17 09:48:36 +00:00
|
|
|
return NULL;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* THIS IS WRONG */
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API int r_lib_dl_check_filename(const char *file) {
|
|
|
|
if (strstr (file, "."R_LIB_EXT))
|
2009-02-05 21:08:46 +00:00
|
|
|
return R_TRUE;
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* high level api */
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API int r_lib_run_handler(RLib *lib, RLibPlugin *plugin, RLibStruct *symbol) {
|
|
|
|
RLibHandler *h = plugin->handler;
|
2009-02-18 15:20:14 +00:00
|
|
|
if (h && h->constructor != NULL)
|
2009-02-05 21:08:46 +00:00
|
|
|
return h->constructor(plugin, h->user, symbol->data);
|
2009-09-17 09:48:36 +00:00
|
|
|
return R_FAIL;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API RLibHandler *r_lib_get_handler(RLib *lib, int type) {
|
2011-05-13 08:22:28 +00:00
|
|
|
RLibHandler *h;
|
|
|
|
RListIter *iter;
|
|
|
|
r_list_foreach (lib->handlers, iter, h) {
|
2009-09-17 09:48:36 +00:00
|
|
|
if (h->type == type)
|
2009-02-05 21:08:46 +00:00
|
|
|
return h;
|
|
|
|
}
|
2009-09-17 09:48:36 +00:00
|
|
|
return NULL;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API R_API int r_lib_close(RLib *lib, const char *file) {
|
2011-05-13 08:22:28 +00:00
|
|
|
RLibPlugin *p;
|
|
|
|
RListIter *iter;
|
|
|
|
r_list_foreach (lib->plugins, iter, p) {
|
|
|
|
if ((file==NULL || (!strcmp(file, p->file))) && p->handler->destructor != NULL) {
|
|
|
|
int ret = p->handler->destructor (p, p->handler->user, p->data);
|
|
|
|
free (p->file);
|
|
|
|
r_list_delete (lib->plugins, iter);
|
|
|
|
free (p);
|
2009-02-05 21:08:46 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-04-01 00:28:13 +00:00
|
|
|
// XXX ugly hack ?
|
2010-04-12 00:22:52 +00:00
|
|
|
static int samefile(const char *a, const char *b) {
|
2009-04-01 00:28:13 +00:00
|
|
|
char *sa = strdup(a);
|
|
|
|
char *sb = strdup(b);
|
|
|
|
char *ptr;
|
2010-11-20 17:35:40 +00:00
|
|
|
int ret, len;
|
2009-04-01 00:28:13 +00:00
|
|
|
|
2009-04-02 00:07:58 +00:00
|
|
|
if (sa != NULL && sb != NULL) {
|
|
|
|
do {
|
|
|
|
ptr = strstr(sa, "//");
|
2010-11-20 17:35:40 +00:00
|
|
|
if (ptr) {
|
|
|
|
len = strlen (ptr+1) + 1;
|
|
|
|
memmove (ptr, ptr+1, len);
|
|
|
|
}
|
2009-04-02 00:07:58 +00:00
|
|
|
} while(ptr);
|
|
|
|
do {
|
|
|
|
ptr = strstr(sb, "//");
|
2010-11-20 17:35:40 +00:00
|
|
|
if (ptr) {
|
|
|
|
len = strlen (ptr+1) + 1;
|
|
|
|
memmove (ptr, ptr+1, len);
|
|
|
|
}
|
2011-05-13 08:22:28 +00:00
|
|
|
} while (ptr);
|
|
|
|
ret = strcmp (sa,sb)? R_FALSE: R_TRUE;
|
2009-04-02 00:07:58 +00:00
|
|
|
}
|
2009-04-01 00:28:13 +00:00
|
|
|
|
2011-05-13 08:22:28 +00:00
|
|
|
free (sa);
|
|
|
|
free (sb);
|
2009-04-01 00:28:13 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API int r_lib_open(RLib *lib, const char *file) {
|
|
|
|
RLibPlugin *p;
|
2011-05-13 08:22:28 +00:00
|
|
|
RListIter *iter;
|
2010-04-12 00:22:52 +00:00
|
|
|
RLibStruct *stru;
|
2009-02-05 21:08:46 +00:00
|
|
|
void * handler;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* ignored by filename */
|
2010-04-12 00:22:52 +00:00
|
|
|
if (!r_lib_dl_check_filename (file)) {
|
|
|
|
eprintf ("Invalid library extension: %s\n", file);
|
2009-09-17 09:48:36 +00:00
|
|
|
return R_FAIL;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
handler = r_lib_dl_open (file);
|
2009-02-05 21:08:46 +00:00
|
|
|
if (handler == NULL) {
|
2010-04-12 00:22:52 +00:00
|
|
|
IFDBG eprintf ("Cannot open library: '%s'\n", file);
|
2009-09-17 09:48:36 +00:00
|
|
|
return R_FAIL;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2009-04-01 00:28:13 +00:00
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
stru = (RLibStruct *) r_lib_dl_sym (handler, lib->symname);
|
2009-02-05 21:08:46 +00:00
|
|
|
if (stru == NULL) {
|
2010-04-12 00:22:52 +00:00
|
|
|
IFDBG eprintf ("No root symbol '%s' found in library '%s'\n", lib->symname, file);
|
2009-09-17 09:48:36 +00:00
|
|
|
return R_FAIL;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2009-09-17 09:48:36 +00:00
|
|
|
|
2011-05-13 08:22:28 +00:00
|
|
|
r_list_foreach (lib->plugins, iter, p) {
|
|
|
|
if (samefile (file, p->file)) {
|
|
|
|
r_lib_dl_close (handler);
|
2009-09-17 09:48:36 +00:00
|
|
|
return R_FAIL;
|
2009-04-01 00:28:13 +00:00
|
|
|
}
|
|
|
|
}
|
2009-09-17 09:48:36 +00:00
|
|
|
|
2011-05-13 08:22:28 +00:00
|
|
|
p = R_NEW (RLibPlugin);
|
2009-02-05 21:08:46 +00:00
|
|
|
p->type = stru->type;
|
|
|
|
p->data = stru->data;
|
2011-05-13 08:22:28 +00:00
|
|
|
p->file = strdup (file);
|
2009-02-05 21:08:46 +00:00
|
|
|
p->dl_handler = handler;
|
2011-05-13 08:22:28 +00:00
|
|
|
p->handler = r_lib_get_handler (lib, p->type);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
2011-05-13 08:22:28 +00:00
|
|
|
ret = r_lib_run_handler (lib, p, stru);
|
2009-09-17 09:48:36 +00:00
|
|
|
if (ret == R_FAIL) {
|
2010-06-03 08:57:34 +00:00
|
|
|
IFDBG eprintf ("Library handler has failed for '%s'\n", file);
|
|
|
|
free (p->file);
|
|
|
|
free (p);
|
|
|
|
r_lib_dl_close (handler);
|
2011-05-13 08:22:28 +00:00
|
|
|
} else r_list_append (lib->plugins, p);
|
2009-09-17 09:48:36 +00:00
|
|
|
|
2009-02-05 21:08:46 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API int r_lib_opendir(RLib *lib, const char *path) {
|
2009-02-05 21:08:46 +00:00
|
|
|
char file[1024];
|
|
|
|
struct dirent *de;
|
2009-03-10 01:49:24 +00:00
|
|
|
DIR *dh;
|
|
|
|
|
2009-03-10 21:58:00 +00:00
|
|
|
#ifdef LIBR_PLUGINS
|
|
|
|
if (path == NULL)
|
|
|
|
path = LIBR_PLUGINS;
|
|
|
|
#endif
|
2009-03-10 01:49:24 +00:00
|
|
|
if (path == NULL)
|
|
|
|
return R_FALSE;
|
2009-04-01 00:28:13 +00:00
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
dh = opendir (path);
|
2009-02-05 21:08:46 +00:00
|
|
|
if (dh == NULL) {
|
2010-04-12 00:22:52 +00:00
|
|
|
IFDBG eprintf ("Cannot open directory '%s'\n", path);
|
2009-03-10 01:49:24 +00:00
|
|
|
return R_FALSE;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2010-06-03 08:57:34 +00:00
|
|
|
while ((de = (struct dirent *)readdir (dh))) {
|
2010-04-12 00:22:52 +00:00
|
|
|
snprintf (file, sizeof (file), "%s/%s", path, de->d_name);
|
|
|
|
if (r_lib_dl_check_filename (file))
|
|
|
|
r_lib_open (lib, file);
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2010-04-12 00:22:52 +00:00
|
|
|
closedir (dh);
|
2009-03-10 01:49:24 +00:00
|
|
|
return R_TRUE;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API int r_lib_add_handler(RLib *lib,
|
2009-02-05 21:08:46 +00:00
|
|
|
int type, const char *desc,
|
2010-04-12 00:22:52 +00:00
|
|
|
int (*cb)(RLibPlugin *, void *, void *), /* constructor */
|
|
|
|
int (*dt)(RLibPlugin *, void *, void *), /* destructor */
|
2009-02-05 21:08:46 +00:00
|
|
|
void *user)
|
|
|
|
{
|
2011-05-13 08:22:28 +00:00
|
|
|
RLibHandler *h;
|
|
|
|
RListIter *iter;
|
2010-04-12 00:22:52 +00:00
|
|
|
RLibHandler *handler = NULL;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
2011-05-13 08:22:28 +00:00
|
|
|
r_list_foreach (lib->handlers, iter, h) {
|
2009-02-05 21:08:46 +00:00
|
|
|
if (type == h->type) {
|
2010-06-03 08:57:34 +00:00
|
|
|
IFDBG eprintf ("Redefining library handler constructor for %d\n", type);
|
2009-02-05 21:08:46 +00:00
|
|
|
handler = h;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (handler == NULL) {
|
2011-05-13 08:22:28 +00:00
|
|
|
handler = R_NEW (RLibHandler);
|
2009-09-17 09:48:36 +00:00
|
|
|
if (handler == NULL)
|
|
|
|
return R_FALSE;
|
2009-02-05 21:08:46 +00:00
|
|
|
handler->type = type;
|
2011-05-13 08:22:28 +00:00
|
|
|
r_list_append (lib->handlers, handler);
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2011-05-13 08:22:28 +00:00
|
|
|
strncpy (handler->desc, desc, sizeof (handler->desc));
|
2009-02-05 21:08:46 +00:00
|
|
|
handler->user = user;
|
|
|
|
handler->constructor = cb;
|
|
|
|
handler->destructor = dt;
|
|
|
|
|
2009-09-17 09:48:36 +00:00
|
|
|
return R_TRUE;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API int r_lib_del_handler(RLib *lib, int type) {
|
2011-05-13 08:22:28 +00:00
|
|
|
RLibHandler *h;
|
|
|
|
RListIter *iter;
|
2009-09-17 09:48:36 +00:00
|
|
|
// TODO: remove all handlers for that type? or only one?
|
2011-05-13 08:22:28 +00:00
|
|
|
r_list_foreach (lib->handlers, iter, h) {
|
2009-09-17 09:48:36 +00:00
|
|
|
if (type == h->type) {
|
2011-05-13 08:22:28 +00:00
|
|
|
r_list_delete (lib->handlers, iter);
|
2009-09-17 09:48:36 +00:00
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return R_FALSE;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2009-09-17 09:48:36 +00:00
|
|
|
/* XXX _list methods must be deprecated before r2-1.0 */
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API void r_lib_list(RLib *lib) {
|
2011-05-13 08:22:28 +00:00
|
|
|
RListIter *iter;
|
|
|
|
RLibPlugin *p;
|
2009-09-17 09:48:36 +00:00
|
|
|
#if 0
|
2010-05-25 23:42:22 +00:00
|
|
|
printf("Plugin Plugins:\n");
|
2009-09-17 09:48:36 +00:00
|
|
|
list_for_each_prev(pos, &lib->handlers) {
|
2010-04-12 00:22:52 +00:00
|
|
|
RLibHandler *h = list_entry(pos, RLibHandler, list);
|
2009-09-17 09:48:36 +00:00
|
|
|
printf(" - %d: %s\n", h->type, h->desc);
|
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
#endif
|
2009-09-17 09:48:36 +00:00
|
|
|
//printf("Loaded plugins:\n");
|
2011-05-13 08:22:28 +00:00
|
|
|
r_list_foreach (lib->plugins, iter, p) {
|
2009-09-17 09:48:36 +00:00
|
|
|
printf(" %5s %p %s \n", r_lib_types_get(p->type), p->handler->destructor, p->file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|