484 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			484 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
| 
 | |
| /* Dynamic loading.  Copyright unclear. */
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include "sysdep.h"
 | |
| #include "scheme48.h"
 | |
| 
 | |
| /*
 | |
|    Hey folks, please help us out with conditions under which the
 | |
|    ANCIENT_DYNLOAD and/or HAVE_DLOPEN code might work.  About all we
 | |
|    know is that ANCIENT_DYNLOAD worked once upon a time on the DEC
 | |
|    MIPS (how to conditionalize for that I don't know -- #ifdef ultrix
 | |
|    perhaps?), is very similar to something that worked under BSD 4.2,
 | |
|    and doesn't work under HPUX, NeXT, or SGI.
 | |
|  */
 | |
| #if defined(ultrix)
 | |
| #define ANCIENT_DYNLOAD
 | |
| #endif
 | |
| 
 | |
| 
 | |
| #if defined(HAVE_DLOPEN)
 | |
| 
 | |
| /*-------------------------------------------------------------------
 | |
|    Following is the preferred modern method, supposedly.  dlopen() is
 | |
|    some kind of newfangled SYSV thing.  This code was contributed by
 | |
|    Basile Starynkevitch <basile@soleil.serma.cea.fr> in October 1993 --
 | |
|    thanks!
 | |
|  */
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <unistd.h>
 | |
| #if defined(__NetBSD__) || defined(__FreeBSD__)
 | |
| #include <nlist.h>
 | |
| #include <link.h>
 | |
| #else
 | |
| #include <dlfcn.h>
 | |
| #endif
 | |
| 
 | |
| #if defined(HAVE_LIBGEN) && defined(HAVE_LIBGEN_H)
 | |
| #include <libgen.h>
 | |
| /* if we have pathfind, get the file name with $LD_LIBRARY_PATH or $S48_EXTERN_PATH */
 | |
| static char *shared_object_name(char *name)
 | |
| {
 | |
|   char *res=0;
 | |
|   res = pathfind(getenv("LD_LIBRARY_PATH"), name, "r");
 | |
|   if (res)
 | |
|     return res;
 | |
|   res = pathfind(getenv("S48_EXTERN_PATH"), name, "r");
 | |
|   if (res)
 | |
|     return res;
 | |
|   return name;
 | |
| } /* end of shared_object_name */
 | |
| #define SHARED_OBJECT_NAME(Name) shared_object_name(Name)
 | |
| #else /* no HAVE_LIBGEN_H */
 | |
| #define SHARED_OBJECT_NAME(Name) (Name)
 | |
| #endif /*HAVE_LIBGEN_H*/
 | |
| 
 | |
| 
 | |
| #ifndef MAXNB_DLOPEN
 | |
| #define MAXNB_DLOPEN 40
 | |
| #endif /* MAXNB_DLOPEN */
 | |
| 
 | |
| #ifndef S48_DLOPEN_MODE
 | |
| /* SunoS5 & SVR4 define RTLD_NOW */
 | |
| #ifdef RTLD_NOW
 | |
| #define S48_DLOPEN_MODE RTLD_NOW
 | |
| #else
 | |
| /* SunOS4 just says that mode should be 1 ie RTLD_LAZY */
 | |
| #define S48_DLOPEN_MODE 1
 | |
| #endif /*RTLD_NOW*/
 | |
| #endif /*!S48_DLOPEN_MODE*/
 | |
| 
 | |
| static void* dlopened_handle[MAXNB_DLOPEN];
 | |
| 
 | |
| /* with dlopen the only user argument is the full shared object name */
 | |
| int
 | |
| dynamic_load(char*sharedobjname)
 | |
| {
 | |
|   char pathname[520];
 | |
|   int  rank= -1;
 | |
|   int cnt;
 | |
|   void *newhandle;
 | |
|   pathname[0]='\0';
 | |
|   strncpy(pathname, SHARED_OBJECT_NAME(sharedobjname), sizeof(pathname)-1);
 | |
|   pathname[sizeof(pathname)-1]='\0';
 | |
|   /* find an unused slot */
 | |
|   for (cnt=0; cnt<MAXNB_DLOPEN; cnt++)
 | |
|     if (!dlopened_handle[cnt]) {
 | |
|       rank=cnt;
 | |
|       break;
 | |
|     }
 | |
|   if (rank<0) {
 | |
|     fprintf(stderr,
 | |
| 	    " dynamic_load - table of dlopened handles is full (MAXNB_DLOPEN=%d) ",
 | |
| 	    MAXNB_DLOPEN);
 | |
|     return -1;
 | |
|   };
 | |
|   newhandle=dlopen(pathname, S48_DLOPEN_MODE);
 | |
| #if defined(__NetBSD__) || defined(__FreeBSD__)
 | |
|   if (newhandle == -1) {
 | |
|     fprintf(stderr, " dynamic_load of %s can't dlopen %s",
 | |
| 	    sharedobjname, pathname);
 | |
|     return -1;
 | |
|   };
 | |
| #else
 | |
|   if (!newhandle) {
 | |
|     fprintf(stderr, " dynamic_load of %s can't dlopen %s: %s ",
 | |
| 	    sharedobjname, pathname, dlerror());
 | |
|     return -1;
 | |
|   };
 | |
| #endif
 | |
|   dlopened_handle[rank] = newhandle;
 | |
| #ifdef DLDEBUG
 | |
|   printf(" %s:%d %s sharedobjname='%s' pathname='%s' handle %#x rank=%d \n",
 | |
| 	 __FILE__, __LINE__, __FUNCTION__,
 | |
| 	 sharedobjname, pathname, newhandle, rank);
 | |
| #endif /*DLDEBUG*/
 | |
|   return 0;
 | |
| } /*------end of dlopening dynamic load -----*/
 | |
| 
 | |
| /* this function implements lookup_external_name with dlsym */
 | |
| long
 | |
| lookup_dlsym(char *name, long *location)
 | |
| {
 | |
|   void *adr;
 | |
|   static void *selfhandle;
 | |
|   int rank;
 | |
| 
 | |
| #if defined(USCORE) && defined(DLSYM_ADDS_USCORE)
 | |
|   name++;
 | |
| #endif
 | |
| 
 | |
|   /* find the name in the self process image - ie original scheme48
 | |
|      executable file */
 | |
|   if (!selfhandle)
 | |
|     selfhandle=dlopen((char*)0, S48_DLOPEN_MODE);
 | |
|   if (adr=dlsym(selfhandle, name)) {
 | |
|     *location = (long) adr;
 | |
| #ifdef DLDEBUG
 | |
|   printf(" %s:%d %s name='%s' in self at adr=%#x\n",
 | |
| 	 __FILE__, __LINE__, __FUNCTION__,
 | |
| 	 name, (long) adr);
 | |
| #endif /*DLDEBUG*/
 | |
|     return 1;
 | |
|   };
 | |
| 
 | |
| /* perhaps i should scan the dlopened_handle from last to first,
 | |
|    to find newest incarnation of symbol? in practice it should be faster
 | |
|    to go from first (oldest) to last,
 | |
|    and i hope that every external symbol is defined exactly once most of
 | |
|    the time!*/
 | |
|   for (rank=0; rank<MAXNB_DLOPEN; rank++)
 | |
|     if (dlopened_handle[rank] && (adr=dlsym(dlopened_handle[rank], name))) {
 | |
|       *location = (long) adr;
 | |
| #ifdef DLDEBUG
 | |
|   printf(" %s:%d %s name='%s' in rank=%d at adr=%#x\n",
 | |
| 	 __FILE__, __LINE__, __FUNCTION__,
 | |
| 	 name, rank, (long) adr);
 | |
| #endif /*DLDEBUG*/
 | |
|       return 1;
 | |
|     };
 | |
|   /* can't find name so return 0 */
 | |
| #ifdef DLDEBUG
 | |
|   printf(" %s:%d %s name='%s' not found\n",
 | |
| 	 __FILE__, __LINE__, __FUNCTION__,
 | |
| 	 name);
 | |
| #endif /*DLDEBUG*/
 | |
|   return 0;
 | |
| } /*----end of lookup_dlsym ---*/
 | |
| 
 | |
| #elif defined(ANCIENT_DYNLOAD)
 | |
| 
 | |
| /*-------------------------------------------------------------------
 | |
|    This is the ancient Berkeley method for dynamic loading, using the
 | |
|    disgusting -A flag to "ld".
 | |
|  */
 | |
| 
 | |
| #include <a.out.h>
 | |
| #include <sys/types.h>      /* sbrk   */
 | |
| #include <strings.h>        /* strlen */
 | |
| 
 | |
| #ifdef mips /* or should this be ultrix? */
 | |
| struct exec {               /* an exec-like structure for the MIPS */
 | |
|   struct filehdr ex_f;
 | |
|   struct aouthdr ex_o;
 | |
| };
 | |
| 
 | |
| #define a_magic ex_o.magic
 | |
| #define a_text ex_o.tsize
 | |
| #define a_data ex_o.dsize
 | |
| #define a_bss  ex_o.bsize
 | |
| 
 | |
| #define BAD_MAGIC(output_file_header) N_BADMAG(output_file_header.ex_o)
 | |
| #define TEXT_OFFSET(output_file_header) \
 | |
| 	    (long) N_TXTOFF(output_file_header.ex_f, output_file_header.ex_o)
 | |
| 
 | |
| #else /* not mips */
 | |
| 
 | |
| #define BAD_MAGIC(output_file_header) N_BADMAG(output_file_header)
 | |
| #define TEXT_OFFSET(output_file_header) \
 | |
| 	    (long) N_TXTOFF(output_file_header)
 | |
| 
 | |
| #endif
 | |
| 
 | |
| 
 | |
| extern char *object_file;   /* specified via a command line argument */
 | |
| extern char *reloc_file;
 | |
| 
 | |
| extern char *get_reloc_file();
 | |
| 
 | |
| /* declarations so that I can put the functions in a reasonable order */
 | |
| int really_dynamic_load( char *, char *, char * );
 | |
| void abort_load( char *, FILE *, char *);
 | |
| 
 | |
| int
 | |
| dynamic_load( char *user_args )
 | |
| {
 | |
|   char *old_reloc_file, *new_reloc_file;
 | |
| 
 | |
|   old_reloc_file = get_reloc_file();
 | |
| 
 | |
|   if (old_reloc_file == NULL) {
 | |
|     fprintf(stderr, "aborting dynamic load\n");
 | |
|     return(0);
 | |
|   }
 | |
| 
 | |
|   new_reloc_file = (char *) malloc(L_tmpnam);
 | |
| 
 | |
|   tmpnam(new_reloc_file);
 | |
| 
 | |
|   if (-1 == really_dynamic_load(old_reloc_file, new_reloc_file,
 | |
| 				user_args)) {
 | |
|     free(new_reloc_file);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   reloc_file = new_reloc_file;
 | |
| 
 | |
|   if (old_reloc_file != object_file)
 | |
|     if (0 != unlink(old_reloc_file))
 | |
|       fprintf(stderr, "unable to delete file %s\n", old_reloc_file);
 | |
|     free(old_reloc_file);
 | |
| 
 | |
|   return(0);
 | |
| }
 | |
| 
 | |
| /*
 | |
| 
 | |
|   really_dynamic_load() executes an `ld' command that links
 | |
|   `user_args' into `output_file', including relocation information
 | |
|   from `reloc_info_file'.  `output_file' is then loaded into the
 | |
|   current process.
 | |
| 
 | |
|   `output_file' can then be used as `reloc_info_file' for subsequent
 | |
|   dynamic loads.
 | |
| 
 | |
|   0 is returned if all goes well, -1 if something goes wrong.
 | |
| 
 | |
|   This is just N consecutive system calls with error checking.
 | |
|   The basic sequence is:
 | |
|     1) allocate storage for the command string
 | |
|     2) use sbrk() to align the storage pointer on a page boundary
 | |
|     3) fill in the command string
 | |
|     4) execute the command
 | |
|     5) open the resulting file and read its header
 | |
|     6) allocate storage for reading in the file
 | |
|     7) read in the text and data segments
 | |
|   It is important that the storage pointer does not change between
 | |
|   steps 2 and 6.  The problem is that ld needs to know the eventual
 | |
|   starting address of the text, but the amount of text cannot be
 | |
|   determined until after the ld command.
 | |
| 
 | |
| */
 | |
| 
 | |
| int
 | |
| really_dynamic_load( char *reloc_info_file, char *output_file, char *user_args )
 | |
| {
 | |
|   int command_length;
 | |
|   char *command;                /* the ld command */
 | |
|   int status;
 | |
| 
 | |
|   long load_point, actual_load_point, end_of_memory;
 | |
| 
 | |
|   FILE *output_file_desc;
 | |
|   char output_file_buffer[BUFSIZ];
 | |
|   struct exec output_file_header;
 | |
|   long loaded_size, required_size;
 | |
| 
 | |
|   extern int getpagesize();
 | |
|   extern char *sbrk(int);
 | |
| 
 | |
|   int page_size = getpagesize();
 | |
| 
 | |
|   char *template = "/bin/ld -N -x -A %s -T %lx -o %s %s -lc";
 | |
| 
 | |
|   /* calculate how long the ld command will be */
 | |
|   command_length = strlen(template) + strlen(reloc_info_file) +
 | |
|     strlen(user_args) + 30;  /* additional space for load_point etc. */
 | |
| 
 | |
|   command = (char *) malloc(command_length);
 | |
| 
 | |
|   if (command == NULL) {
 | |
|     fprintf(stderr, "malloc failed to allocate %d characters\n",
 | |
| 	    command_length);
 | |
|     abort_load(NULL, NULL, NULL);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   end_of_memory = (long) sbrk(0);
 | |
| 
 | |
|   if (end_of_memory < 0) {
 | |
|     fprintf(stderr, "sbrk(0) failed\n");
 | |
|     abort_load(command, NULL, NULL);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   /* move the storage pointer to the next page boundary */
 | |
|   end_of_memory = (long) sbrk(page_size - end_of_memory % page_size);
 | |
| 
 | |
|   if (end_of_memory < 0) {
 | |
|     fprintf(stderr, "sbrk(...) failed\n");
 | |
|     abort_load(command, NULL, NULL);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   load_point = (long) sbrk(0);    /* no malloc or printf after this */
 | |
| 
 | |
|   if (load_point < 0 || 0 != load_point % page_size) {
 | |
|     fprintf(stderr, "couldn't align sbrk on page boundary\n");
 | |
|     abort_load(command, NULL, NULL);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   /* finish making the load command */
 | |
| #ifdef sun  /* Sun's sprintf has a nonstandard return value */
 | |
|   sprintf(command, template, reloc_info_file, load_point, output_file,
 | |
| 	  user_args);
 | |
| #else
 | |
|   status = sprintf(command, template, reloc_info_file, load_point, output_file,
 | |
| 		   user_args);
 | |
| 
 | |
|   if (status < 0) {
 | |
|     fprintf(stderr, "sprintf error %d\n", status);
 | |
|     abort_load(command, NULL, NULL);
 | |
|     return(-1);
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   if (strlen(command) >= command_length) {
 | |
|     fprintf(stderr, "load command overflowed buffer by %d chars\n",
 | |
| 	    strlen(command) - command_length);
 | |
|     abort_load(command, NULL, NULL);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   /* run the ld comand */
 | |
|   if (system(command) != 0 ) {
 | |
|     fprintf(stderr, "ld command failed\n");
 | |
|     abort_load(command, NULL, output_file);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   free(command);
 | |
| 
 | |
|   /* open the output file */
 | |
|   output_file_desc = fopen(output_file, "rb");
 | |
|   if (output_file_desc == NULL ) {
 | |
|     fprintf(stderr, "unable to open output file\n");
 | |
|     abort_load(NULL, NULL, output_file);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   /* use our own buffer to prevent calls to malloc etc. */
 | |
|   setbuf(output_file_desc, output_file_buffer);
 | |
| 
 | |
|   /* read the output file header */
 | |
|   status = fread((char *)&output_file_header, sizeof(struct exec), 1,
 | |
| 		 output_file_desc);
 | |
|   if (status != 1) {
 | |
|     fprintf(stderr, "couldn't read output file header\n");
 | |
|     abort_load(NULL, output_file_desc, output_file);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   if (BAD_MAGIC(output_file_header)) {
 | |
|     fprintf(stderr, "output file has bad magic number %d\n",
 | |
| 	    output_file_header.a_magic);
 | |
|     abort_load(NULL, output_file_desc, output_file);
 | |
|     return(-1);
 | |
|   }
 | |
|   loaded_size = output_file_header.a_text + output_file_header.a_data;
 | |
|   required_size = loaded_size + output_file_header.a_bss;
 | |
| 
 | |
|   /* get required memory */
 | |
|   actual_load_point = (long) sbrk(required_size);
 | |
|   if (actual_load_point < 0L) {
 | |
|     fprintf(stderr, "sbrk(%ld) failed\n", required_size);
 | |
|     abort_load(NULL, output_file_desc, output_file);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   /* make sure nothing went wrong */
 | |
|   if (actual_load_point != load_point) {
 | |
|     fprintf(stderr, "storage pointer changed before file could be loaded\n");
 | |
|     abort_load(NULL, output_file_desc, output_file);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   /* go to beginning of text */
 | |
|   if (fseek(output_file_desc, TEXT_OFFSET(output_file_header), 0)
 | |
|       < 0) {
 | |
|     fprintf(stderr, "unable to seek to beginning of linked file text\n");
 | |
|     abort_load(NULL, output_file_desc, output_file);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   /* read the text and data segments */
 | |
|   if (fread((char *)load_point, 1, loaded_size, output_file_desc)
 | |
|       !=
 | |
|       loaded_size) {
 | |
|     fprintf(stderr, "unable to load linked file\n");
 | |
|     abort_load(NULL, output_file_desc, output_file);
 | |
|     return(-1);
 | |
|   }
 | |
| 
 | |
|   if (0 != fclose(output_file_desc))
 | |
|     fprintf(stderr, "unable to close file %s\n", output_file);
 | |
| 
 | |
|   return(0);  /* We made it! */
 | |
| }
 | |
| 
 | |
| /* Print a message and then clean up. */
 | |
| 
 | |
| void
 | |
| abort_load( char *command, FILE *file_desc, char *filename )
 | |
| {
 | |
|   fprintf(stderr, "aborting dynamic load\n");
 | |
|   if (command != NULL)
 | |
|     free(command);
 | |
|   if (file_desc != NULL)
 | |
|     if (0 != fclose(file_desc))
 | |
|       fprintf(stderr, "unable to close file %s\n", filename);
 | |
|   if (filename != NULL)
 | |
|     if (0 != unlink(filename))
 | |
|       fprintf(stderr, "unable to delete file %s\n", filename);
 | |
| }
 | |
| 
 | |
| 
 | |
| #else /* !ANCIENT_DYNLOAD */
 | |
| 
 | |
| /*-------------------------------------------------------------------
 | |
|    Dummy definition in case we don't have a clue.
 | |
|  */
 | |
| 
 | |
| int
 | |
| dynamic_load( char *user_args )
 | |
| {
 | |
|   fprintf(stderr, "Dynamic .o loading not implemented\n");
 | |
|   return -1;
 | |
| }
 | |
| 
 | |
| #endif
 | |
| 
 | |
| 
 | |
| /* Stub for dynamic_load() that can be called from Scheme using
 | |
|    external call interface */
 | |
| 
 | |
| scheme_value
 | |
| s48_dynamic_load( long nargs, scheme_value *argv )
 | |
| {
 | |
|   scheme_value arg;
 | |
| 
 | |
|   if (nargs != 1) return(SCHFALSE);
 | |
| 
 | |
|   arg = argv[0];
 | |
| 
 | |
|   if (!STRINGP(arg)) return(SCHFALSE);
 | |
| 
 | |
|   if (0 == dynamic_load(&STRING_REF(arg, 0)))
 | |
|     return(SCHTRUE);
 | |
|   else
 | |
|     return(SCHFALSE);
 | |
| }
 |