 cfd3379bb1
			
		
	
	
		cfd3379bb1
		
	
	
	
	
		
			
			* Removed startup code patches in lib/csu regarding kernel to userland
   ABI.
 * Aligned stack layout on NetBSD stack layout.
 * Generate valid stack pointers instead of offsets by taking into account
   _minix_kerninfo->kinfo->user_sp.
 * Refactored stack generation, by moving part of execve in two
   functions {minix_stack_params(), minix_stack_fill()} and using them
   in execve(), rs and vm.
 * Changed load offset of rtld (ld.so) to:
      execi.args.stack_high - execi.args.stack_size - 0xa00000
   which is 10MB below the main executable stack.
Change-Id: I839daf3de43321cded44105634102d419cb36cec
		
	
			
		
			
				
	
	
		
			169 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			169 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "inc.h"
 | |
| #include <assert.h>
 | |
| #include <sys/exec.h>
 | |
| #include <libexec.h>
 | |
| #include <minix/param.h>
 | |
| #include <machine/vmparam.h>
 | |
| 
 | |
| static int do_exec(int proc_e, char *exec, size_t exec_len, char *progname,
 | |
| 	char *frame, int frame_len, vir_bytes ps_str);
 | |
| static int exec_restart(int proc_e, int result, vir_bytes pc, vir_bytes ps_str);
 | |
| static int read_seg(struct exec_info *execi, off_t off,
 | |
|         vir_bytes seg_addr, size_t seg_bytes);
 | |
| 
 | |
| /* Array of loaders for different object formats */
 | |
| static struct exec_loaders {
 | |
| 	libexec_exec_loadfunc_t load_object;
 | |
| } const exec_loaders[] = {
 | |
| 	{ libexec_load_elf },
 | |
| 	{ NULL }
 | |
| };
 | |
| 
 | |
| int srv_execve(int proc_e, char *exec, size_t exec_len, char **argv,
 | |
| 	char **envp)
 | |
| {
 | |
| 	size_t frame_size = 0;	/* Size of the new initial stack. */
 | |
| 	int argc = 0;		/* Argument count. */
 | |
| 	int envc = 0;		/* Environment count */
 | |
| 	char overflow = 0;	/* No overflow yet. */
 | |
| 	char *frame;
 | |
| 	struct ps_strings *psp;
 | |
| 	int vsp = 0;	/* (virtual) Stack pointer in new address space. */
 | |
| 
 | |
| 	char *progname;
 | |
| 	int r;
 | |
| 
 | |
| 	minix_stack_params(argv[0], argv, envp, &frame_size, &overflow,
 | |
| 		&argc, &envc);
 | |
| 
 | |
| 	/* The party is off if there is an overflow. */
 | |
| 	if (overflow) {
 | |
| 		errno = E2BIG;
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* Allocate space for the stack frame. */
 | |
| 	if ((frame = (char *) sbrk(frame_size)) == (char *) -1) {
 | |
| 		errno = E2BIG;
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	minix_stack_fill(argv[0], argc, argv, envc, envp, frame_size, frame,
 | |
| 		&vsp, &psp);
 | |
| 
 | |
| 	(progname=strrchr(argv[0], '/')) ? progname++ : (progname=argv[0]);
 | |
| 
 | |
| 	r = do_exec(proc_e, exec, exec_len, progname, frame, frame_size,
 | |
| 		vsp + ((char *)psp - frame));
 | |
| 
 | |
| 	/* Failure, return the memory used for the frame and exit. */
 | |
| 	(void) sbrk(-frame_size);
 | |
| 
 | |
| 	return r;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int do_exec(int proc_e, char *exec, size_t exec_len, char *progname,
 | |
| 	char *frame, int frame_len, vir_bytes ps_str)
 | |
| {
 | |
| 	int r;
 | |
| 	vir_bytes vsp;
 | |
| 	struct exec_info execi;
 | |
| 	int i;
 | |
| 
 | |
| 	memset(&execi, 0, sizeof(execi));
 | |
| 
 | |
| 	execi.stack_high = kinfo.user_sp;
 | |
| 	execi.stack_size = DEFAULT_STACK_LIMIT;
 | |
| 	execi.proc_e = proc_e;
 | |
| 	execi.hdr = exec;
 | |
| 	execi.filesize = execi.hdr_len = exec_len;
 | |
| 	strncpy(execi.progname, progname, PROC_NAME_LEN-1);
 | |
| 	execi.progname[PROC_NAME_LEN-1] = '\0';
 | |
| 	execi.frame_len = frame_len;
 | |
| 
 | |
| 	/* callback functions and data */
 | |
| 	execi.copymem = read_seg;
 | |
| 	execi.clearproc = libexec_clearproc_vm_procctl;
 | |
| 	execi.clearmem = libexec_clear_sys_memset;
 | |
| 	execi.allocmem_prealloc_cleared = libexec_alloc_mmap_prealloc_cleared;
 | |
| 	execi.allocmem_prealloc_junk = libexec_alloc_mmap_prealloc_junk;
 | |
| 	execi.allocmem_ondemand = libexec_alloc_mmap_ondemand;
 | |
| 
 | |
| 	for(i = 0; exec_loaders[i].load_object != NULL; i++) {
 | |
| 	    r = (*exec_loaders[i].load_object)(&execi);
 | |
| 	    /* Loaded successfully, so no need to try other loaders */
 | |
| 	    if (r == OK) break;
 | |
| 	}
 | |
| 
 | |
| 	/* No exec loader could load the object */
 | |
| 	if (r != OK) {
 | |
| 	    printf("RS: do_exec: loading error %d\n", r);
 | |
| 	    return r;
 | |
| 	}
 | |
| 
 | |
| 	/* Inform PM */
 | |
|         if((r = libexec_pm_newexec(execi.proc_e, &execi)) != OK)
 | |
| 		return r;
 | |
| 
 | |
| 	/* Patch up stack and copy it from RS to new core image. */
 | |
| 	vsp = execi.stack_high - frame_len;
 | |
| 	r = sys_datacopy(SELF, (vir_bytes) frame,
 | |
| 		proc_e, (vir_bytes) vsp, (phys_bytes)frame_len);
 | |
| 	if (r != OK) {
 | |
| 		printf("do_exec: copying out new stack failed: %d\n", r);
 | |
| 		exec_restart(proc_e, r, execi.pc, ps_str);
 | |
| 		return r;
 | |
| 	}
 | |
| 
 | |
| 	return exec_restart(proc_e, OK, execi.pc, ps_str);
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				exec_restart				     *
 | |
|  *===========================================================================*/
 | |
| static int exec_restart(int proc_e, int result, vir_bytes pc, vir_bytes ps_str)
 | |
| {
 | |
| 	int r;
 | |
| 	message m;
 | |
| 
 | |
| 	m.m_type = EXEC_RESTART;
 | |
| 	m.EXC_RS_PROC = proc_e;
 | |
| 	m.EXC_RS_RESULT = result;
 | |
| 	m.EXC_RS_PC = (void *)pc;
 | |
| 	m.EXC_RS_PS_STR = (void *)ps_str;
 | |
| 
 | |
| 	r = sendrec(PM_PROC_NR, &m);
 | |
| 	if (r != OK)
 | |
| 		return r;
 | |
| 
 | |
| 	return m.m_type;
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *                             read_seg                                     *
 | |
|  *===========================================================================*/
 | |
| static int read_seg(
 | |
| struct exec_info *execi,       /* various data needed for exec */
 | |
| off_t off,                     /* offset in file */
 | |
| vir_bytes seg_addr,            /* address to load segment */
 | |
| size_t seg_bytes           /* how much is to be transferred? */
 | |
| )
 | |
| {
 | |
| /*
 | |
|  * The byte count on read is usually smaller than the segment count, because
 | |
|  * a segment is padded out to a click multiple, and the data segment is only
 | |
|  * partially initialized.
 | |
|  */
 | |
| 
 | |
|   int r;
 | |
| 
 | |
|   if (off+seg_bytes > execi->hdr_len) return ENOEXEC;
 | |
|   if((r= sys_vircopy(SELF, ((vir_bytes)execi->hdr)+off,
 | |
|   	execi->proc_e, seg_addr, seg_bytes)) != OK) {
 | |
| 	printf("RS: exec read_seg: copy 0x%x bytes into %i at 0x%08lx failed: %i\n",
 | |
| 		seg_bytes, execi->proc_e, seg_addr, r);
 | |
|   }
 | |
|   return r;
 | |
| }
 |