123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279 |
- #include "dso_local.h"
- #ifdef DSO_DL
- # include <dl.h>
- # define DSO_MAX_TRANSLATED_SIZE 256
- static int dl_load(DSO *dso);
- static int dl_unload(DSO *dso);
- static DSO_FUNC_TYPE dl_bind_func(DSO *dso, const char *symname);
- static char *dl_name_converter(DSO *dso, const char *filename);
- static char *dl_merger(DSO *dso, const char *filespec1,
- const char *filespec2);
- static int dl_pathbyaddr(void *addr, char *path, int sz);
- static void *dl_globallookup(const char *name);
- static DSO_METHOD dso_meth_dl = {
- "OpenSSL 'dl' shared library method",
- dl_load,
- dl_unload,
- dl_bind_func,
- NULL,
- dl_name_converter,
- dl_merger,
- NULL,
- NULL,
- dl_pathbyaddr,
- dl_globallookup
- };
- DSO_METHOD *DSO_METHOD_openssl(void)
- {
- return &dso_meth_dl;
- }
- static int dl_load(DSO *dso)
- {
- shl_t ptr = NULL;
-
- char *filename = DSO_convert_filename(dso, NULL);
- if (filename == NULL) {
- DSOerr(DSO_F_DL_LOAD, DSO_R_NO_FILENAME);
- goto err;
- }
- ptr = shl_load(filename, BIND_IMMEDIATE |
- (dso->flags & DSO_FLAG_NO_NAME_TRANSLATION ? 0 :
- DYNAMIC_PATH), 0L);
- if (ptr == NULL) {
- char errbuf[160];
- DSOerr(DSO_F_DL_LOAD, DSO_R_LOAD_FAILED);
- if (openssl_strerror_r(errno, errbuf, sizeof(errbuf)))
- ERR_add_error_data(4, "filename(", filename, "): ", errbuf);
- goto err;
- }
- if (!sk_push(dso->meth_data, (char *)ptr)) {
- DSOerr(DSO_F_DL_LOAD, DSO_R_STACK_ERROR);
- goto err;
- }
-
- dso->loaded_filename = filename;
- return 1;
- err:
-
- OPENSSL_free(filename);
- if (ptr != NULL)
- shl_unload(ptr);
- return 0;
- }
- static int dl_unload(DSO *dso)
- {
- shl_t ptr;
- if (dso == NULL) {
- DSOerr(DSO_F_DL_UNLOAD, ERR_R_PASSED_NULL_PARAMETER);
- return 0;
- }
- if (sk_num(dso->meth_data) < 1)
- return 1;
-
- ptr = (shl_t) sk_pop(dso->meth_data);
- if (ptr == NULL) {
- DSOerr(DSO_F_DL_UNLOAD, DSO_R_NULL_HANDLE);
-
- sk_push(dso->meth_data, (char *)ptr);
- return 0;
- }
- shl_unload(ptr);
- return 1;
- }
- static DSO_FUNC_TYPE dl_bind_func(DSO *dso, const char *symname)
- {
- shl_t ptr;
- void *sym;
- if ((dso == NULL) || (symname == NULL)) {
- DSOerr(DSO_F_DL_BIND_FUNC, ERR_R_PASSED_NULL_PARAMETER);
- return NULL;
- }
- if (sk_num(dso->meth_data) < 1) {
- DSOerr(DSO_F_DL_BIND_FUNC, DSO_R_STACK_ERROR);
- return NULL;
- }
- ptr = (shl_t) sk_value(dso->meth_data, sk_num(dso->meth_data) - 1);
- if (ptr == NULL) {
- DSOerr(DSO_F_DL_BIND_FUNC, DSO_R_NULL_HANDLE);
- return NULL;
- }
- if (shl_findsym(&ptr, symname, TYPE_UNDEFINED, &sym) < 0) {
- char errbuf[160];
- DSOerr(DSO_F_DL_BIND_FUNC, DSO_R_SYM_FAILURE);
- if (openssl_strerror_r(errno, errbuf, sizeof(errbuf)))
- ERR_add_error_data(4, "symname(", symname, "): ", errbuf);
- return NULL;
- }
- return (DSO_FUNC_TYPE)sym;
- }
- static char *dl_merger(DSO *dso, const char *filespec1, const char *filespec2)
- {
- char *merged;
- if (!filespec1 && !filespec2) {
- DSOerr(DSO_F_DL_MERGER, ERR_R_PASSED_NULL_PARAMETER);
- return NULL;
- }
-
- if (!filespec2 || filespec1[0] == '/') {
- merged = OPENSSL_strdup(filespec1);
- if (merged == NULL) {
- DSOerr(DSO_F_DL_MERGER, ERR_R_MALLOC_FAILURE);
- return NULL;
- }
- }
-
- else if (!filespec1) {
- merged = OPENSSL_strdup(filespec2);
- if (merged == NULL) {
- DSOerr(DSO_F_DL_MERGER, ERR_R_MALLOC_FAILURE);
- return NULL;
- }
- } else
-
- {
- int spec2len, len;
- spec2len = (filespec2 ? strlen(filespec2) : 0);
- len = spec2len + (filespec1 ? strlen(filespec1) : 0);
- if (spec2len && filespec2[spec2len - 1] == '/') {
- spec2len--;
- len--;
- }
- merged = OPENSSL_malloc(len + 2);
- if (merged == NULL) {
- DSOerr(DSO_F_DL_MERGER, ERR_R_MALLOC_FAILURE);
- return NULL;
- }
- strcpy(merged, filespec2);
- merged[spec2len] = '/';
- strcpy(&merged[spec2len + 1], filespec1);
- }
- return merged;
- }
- static char *dl_name_converter(DSO *dso, const char *filename)
- {
- char *translated;
- int len, rsize, transform;
- len = strlen(filename);
- rsize = len + 1;
- transform = (strstr(filename, "/") == NULL);
- {
-
- rsize += strlen(DSO_EXTENSION);
- if ((DSO_flags(dso) & DSO_FLAG_NAME_TRANSLATION_EXT_ONLY) == 0)
- rsize += 3;
- }
- translated = OPENSSL_malloc(rsize);
- if (translated == NULL) {
- DSOerr(DSO_F_DL_NAME_CONVERTER, DSO_R_NAME_TRANSLATION_FAILED);
- return NULL;
- }
- if (transform) {
- if ((DSO_flags(dso) & DSO_FLAG_NAME_TRANSLATION_EXT_ONLY) == 0)
- sprintf(translated, "lib%s%s", filename, DSO_EXTENSION);
- else
- sprintf(translated, "%s%s", filename, DSO_EXTENSION);
- } else
- sprintf(translated, "%s", filename);
- return translated;
- }
- static int dl_pathbyaddr(void *addr, char *path, int sz)
- {
- struct shl_descriptor inf;
- int i, len;
- if (addr == NULL) {
- union {
- int (*f) (void *, char *, int);
- void *p;
- } t = {
- dl_pathbyaddr
- };
- addr = t.p;
- }
- for (i = -1; shl_get_r(i, &inf) == 0; i++) {
- if (((size_t)addr >= inf.tstart && (size_t)addr < inf.tend) ||
- ((size_t)addr >= inf.dstart && (size_t)addr < inf.dend)) {
- len = (int)strlen(inf.filename);
- if (sz <= 0)
- return len + 1;
- if (len >= sz)
- len = sz - 1;
- memcpy(path, inf.filename, len);
- path[len++] = 0;
- return len;
- }
- }
- return -1;
- }
- static void *dl_globallookup(const char *name)
- {
- void *ret;
- shl_t h = NULL;
- return shl_findsym(&h, name, TYPE_UNDEFINED, &ret) ? NULL : ret;
- }
- #endif
|