#include #include #include #include #include void Init_prv_ext_loader(void); static void *_dln_load(const char *file); static VALUE _load_shared_lib(VALUE self, volatile VALUE fname) { (void) self; // check that path is not tainted SafeStringValue(fname); FilePathValue(fname); VALUE path = rb_str_encode_ospath(fname); char *loc = StringValueCStr(path); void *handle = _dln_load(loc); return handle ? Qtrue : Qfalse; } #define INIT_FUNC_PREFIX ((char[]) {'I', 'n', 'i', 't', '_'}) #define INIT_FUNCNAME(buf, file) do { \ const char *base = (file); \ const size_t flen = _init_funcname(&base); \ const size_t plen = sizeof(INIT_FUNC_PREFIX); \ char *const tmp = ALLOCA_N(char, plen + flen + 1); \ memcpy(tmp, INIT_FUNC_PREFIX, plen); \ memcpy(tmp+plen, base, flen); \ tmp[plen+flen] = '\0'; \ *(buf) = tmp; \ } while(0) static size_t _init_funcname(const char **file) { const char *p = *file, *base, *dot = NULL; for (base = p; *p; p++) { /* Find position of last '/' */ if (*p == '.' && !dot) { dot = p; } if (*p == '/') { base = p + 1; dot = NULL; } } *file = base; return (uintptr_t) ((dot ? dot : p) - base); } static void *_dln_load(const char *file) { char *buf; const char *error; #define DLN_ERROR() (error = dlerror(), strcpy(ALLOCA_N(char, strlen(error) + 1), error)) void *handle; void (*init_fct)(void); INIT_FUNCNAME(&buf, file); /* Load file */ if ((handle = dlopen(file, RTLD_LAZY|RTLD_LOCAL|RTLD_DEEPBIND)) == NULL) { DLN_ERROR(); goto failed; } #if defined(RUBY_EXPORT) { static const char incompatible[] = "incompatible library version"; void *ex = dlsym(handle, "ruby_xmalloc"); if (ex && ex != (void *) &ruby_xmalloc) { # if defined __APPLE__ /* dlclose() segfaults */ rb_fatal("%s - %s", incompatible, file); # else dlclose(handle); error = incompatible; goto failed; #endif } } # endif init_fct = (void (*)(void)) dlsym(handle, buf); if (init_fct == NULL) { error = DLN_ERROR(); dlclose(handle); goto failed; } /* Call the init code */ (*init_fct)(); return handle; failed: rb_raise(rb_eLoadError, "%s", error); } void Init_prv_ext_loader() { VALUE mSqreen = rb_define_module("Sqreen"); VALUE mPrvExtLoader = rb_define_module_under(mSqreen, "PrvExtLoader"); rb_define_singleton_method(mPrvExtLoader, "load", _load_shared_lib, 1); }