#include "dlfcn.h" #include "string.h" #include "stdio.h" #define INCL_BASE #include static ULONG retcode; static char fail[300]; char *os2error(int rc); void * dlopen(const char *path, int mode) { HMODULE handle; char tmp[260], *beg, *dot; ULONG rc; fail[0] = 0; if ((rc = DosLoadModule(fail, sizeof fail, (char*)path, &handle)) == 0) return (void *)handle; retcode = rc; if (strlen(path) >= sizeof(tmp)) return NULL; /* Not found. Check for non-FAT name and try truncated name. */ /* Don't know if this helps though... */ for (beg = dot = path + strlen(path); beg > path && !strchr(":/\\", *(beg-1)); beg--) if (*beg == '.') dot = beg; if (dot - beg > 8) { int n = beg+8-path; memmove(tmp, path, n); memmove(tmp+n, dot, strlen(dot)+1); if (DosLoadModule(fail, sizeof fail, tmp, &handle) == 0) return (void *)handle; } return NULL; } void * dlsym(void *handle, const char *symbol) { ULONG rc, type; PFN addr; fail[0] = 0; rc = DosQueryProcAddr((HMODULE)handle, 0, symbol, &addr); if (rc == 0) { rc = DosQueryProcType((HMODULE)handle, 0, symbol, &type); if (rc == 0 && type == PT_32BIT) return (void *)addr; rc = ERROR_CALL_NOT_IMPLEMENTED; } retcode = rc; return NULL; } char * dlerror(void) { static char buf[700]; ULONG len; char *err; if (retcode == 0) return NULL; err = os2error(retcode); len = strlen(err); if (len > sizeof(buf) - 1) len = sizeof(buf) - 1; strncpy(buf, err, len+1); if (fail[0] && len < 300) sprintf(buf + len, ", possible problematic module: '%s'", fail); retcode = 0; return buf; } int dlclose(void *handle) { ULONG rc; if ((rc = DosFreeModule((HMODULE)handle)) == 0) return 0; retcode = rc; return 2; }