xref: /minix3/minix/servers/vfs/exec.c (revision 7c48de6cc4c6d56f2277d378dba01dbac8a8c3b9)
1 /* This file handles the EXEC system call.  It performs the work as follows:
2  *    - see if the permissions allow the file to be executed
3  *    - read the header and extract the sizes
4  *    - fetch the initial args and environment from the user space
5  *    - allocate the memory for the new process
6  *    - copy the initial stack from PM to the process
7  *    - read in the text and data segments and copy to the process
8  *    - take care of setuid and setgid bits
9  *    - fix up 'mproc' table
10  *    - tell kernel about EXEC
11  *    - save offset to initial argc (for ps)
12  *
13  * The entry points into this file are:
14  *   pm_exec:	 perform the EXEC system call
15  */
16 
17 #include "fs.h"
18 #include <sys/stat.h>
19 #include <sys/mman.h>
20 #include <minix/callnr.h>
21 #include <minix/endpoint.h>
22 #include <minix/com.h>
23 #include <minix/u64.h>
24 #include <lib.h>
25 #include <signal.h>
26 #include <stdlib.h>
27 #include <string.h>
28 #include <sys/dirent.h>
29 #include <sys/exec.h>
30 #include <sys/param.h>
31 #include "path.h"
32 #include "vnode.h"
33 #include "file.h"
34 #include <minix/vfsif.h>
35 #include <machine/vmparam.h>
36 #include <assert.h>
37 #include <fcntl.h>
38 
39 #define _KERNEL	/* for ELF_AUX_ENTRIES */
40 #include <libexec.h>
41 
42 /* fields only used by elf and in VFS */
43 struct vfs_exec_info {
44     struct exec_info args;		/* libexec exec args */
45     struct vnode *vp;			/* Exec file's vnode */
46     struct vmnt *vmp;			/* Exec file's vmnt */
47     struct stat sb;			/* Exec file's stat structure */
48     int userflags;			/* exec() flags from userland */
49     int is_dyn;				/* Dynamically linked executable */
50     int elf_main_fd;			/* Dyn: FD of main program execuatble */
51     char execname[PATH_MAX];		/* Full executable invocation */
52     int vmfd;
53     int vmfd_used;
54 };
55 
56 static int patch_stack(struct vnode *vp, char stack[ARG_MAX],
57 	size_t *stk_bytes, char path[PATH_MAX], vir_bytes *vsp);
58 static int is_script(struct vfs_exec_info *execi);
59 static int insert_arg(char stack[ARG_MAX], size_t *stk_bytes, char *arg,
60 	vir_bytes *vsp, char replace);
61 static void clo_exec(struct fproc *rfp);
62 static int stack_prepare_elf(struct vfs_exec_info *execi,
63 	char *curstack, size_t *frame_len, vir_bytes *vsp);
64 static int map_header(struct vfs_exec_info *execi);
65 static int read_seg(struct exec_info *execi, off_t off, vir_bytes seg_addr, size_t seg_bytes);
66 
67 #define PTRSIZE	sizeof(char *) /* Size of pointers in argv[] and envp[]. */
68 
69 /* Array of loaders for different object file formats */
70 typedef int (*exechook_t)(struct vfs_exec_info *execpackage);
71 typedef int (*stackhook_t)(struct vfs_exec_info *execi, char *curstack,
72 	size_t *frame_len, vir_bytes *vsp);
73 struct exec_loaders {
74 	libexec_exec_loadfunc_t load_object;	 /* load executable into memory */
75 	stackhook_t setup_stack; /* prepare stack before argc and argv push */
76 };
77 
78 static const struct exec_loaders exec_loaders[] = {
79 	{ libexec_load_elf,  stack_prepare_elf },
80 	{ NULL, NULL }
81 };
82 
83 #define lock_exec() lock_proc(fproc_addr(VM_PROC_NR))
84 #define unlock_exec() unlock_proc(fproc_addr(VM_PROC_NR))
85 
86 /*===========================================================================*
87  *				get_read_vp				     *
88  *===========================================================================*/
89 static int get_read_vp(struct vfs_exec_info *execi,
90   char *fullpath, int copyprogname, int sugid, struct lookup *resolve, struct fproc *fp)
91 {
92 /* Make the executable that we want to exec() into the binary pointed
93  * to by 'fullpath.' This function fills in necessary details in the execi
94  * structure, such as opened vnode. It unlocks and releases the vnode if
95  * it was already there. This makes it easy to change the executable
96  * during the exec(), which is often necessary, by calling this function
97  * more than once. This is specifically necessary when we discover the
98  * executable is actually a script or a dynamically linked executable.
99  */
100 	int r;
101 
102 	/* Caller wants to switch vp to the file in 'fullpath.'
103 	 * unlock and put it first if there is any there.
104 	 */
105 	if(execi->vp) {
106 		unlock_vnode(execi->vp);
107 		put_vnode(execi->vp);
108 		execi->vp = NULL;
109 	}
110 
111 	/* Remember/overwrite the executable name if requested. */
112 	if(copyprogname) {
113 		char *cp = strrchr(fullpath, '/');
114 		if(cp) cp++;
115 		else cp = fullpath;
116 		strlcpy(execi->args.progname, cp, sizeof(execi->args.progname));
117 		execi->args.progname[sizeof(execi->args.progname)-1] = '\0';
118 	}
119 
120 	/* Open executable */
121 	if ((execi->vp = eat_path(resolve, fp)) == NULL)
122 		return err_code;
123 
124 	unlock_vmnt(execi->vmp);
125 
126 	if (!S_ISREG(execi->vp->v_mode))
127 		return ENOEXEC;
128 	else if ((r = forbidden(fp, execi->vp, X_BIT)) != OK)
129 		return r;
130 	else
131 		r = req_stat(execi->vp->v_fs_e, execi->vp->v_inode_nr,
132 			VFS_PROC_NR, (vir_bytes) &(execi->sb));
133 
134 	if (r != OK) return r;
135 
136 	/* If caller wants us to, honour suid/guid mode bits. */
137         if (sugid) {
138 		/* Deal with setuid/setgid executables */
139 		if (execi->vp->v_mode & I_SET_UID_BIT) {
140 			execi->args.new_uid = execi->vp->v_uid;
141 			execi->args.allow_setuid = 1;
142 		}
143 		if (execi->vp->v_mode & I_SET_GID_BIT) {
144 			execi->args.new_gid = execi->vp->v_gid;
145 			execi->args.allow_setuid = 1;
146 		}
147         }
148 
149 	/* Read in first chunk of file. */
150 	if((r=map_header(execi)) != OK)
151 		return r;
152 
153 	return OK;
154 }
155 
156 #define FAILCHECK(expr) if((r=(expr)) != OK) { goto pm_execfinal; } while(0)
157 #define Get_read_vp(e,f,p,s,rs,fp) do { \
158 	r=get_read_vp(&e,f,p,s,rs,fp); if(r != OK) { FAILCHECK(r); }	\
159 	} while(0)
160 
161 static int vfs_memmap(struct exec_info *execi,
162         vir_bytes vaddr, vir_bytes len, vir_bytes foffset, u16_t clearend,
163 	int protflags)
164 {
165 	struct vfs_exec_info *vi = (struct vfs_exec_info *) execi->opaque;
166 	struct vnode *vp = ((struct vfs_exec_info *) execi->opaque)->vp;
167 	int r;
168 	u16_t flags = 0;
169 
170 	if(protflags & PROT_WRITE)
171 		flags |= MVM_WRITABLE;
172 
173 	r = minix_vfs_mmap(execi->proc_e, foffset, len,
174 	        vp->v_dev, vp->v_inode_nr, vi->vmfd, vaddr, clearend, flags);
175 	if(r == OK) {
176 		vi->vmfd_used = 1;
177 	}
178 
179 	return r;
180 }
181 
182 /*===========================================================================*
183  *				pm_exec					     *
184  *===========================================================================*/
185 int pm_exec(vir_bytes path, size_t path_len, vir_bytes frame, size_t frame_len,
186 	vir_bytes *pc, vir_bytes *newsp, vir_bytes *UNUSED(ps_str))
187 {
188 /* Perform the execve(name, argv, envp) call.  The user library builds a
189  * complete stack image, including pointers, args, environ, etc.  The stack
190  * is copied to a buffer inside VFS, and then to the new core image.
191  *
192  * ps_str is not currently used, but may be if the ps_strings structure has to
193  * be moved to another location.
194  */
195   int r;
196   vir_bytes vsp;
197   static char mbuf[ARG_MAX];	/* buffer for stack and zeroes */
198   struct vfs_exec_info execi;
199   int i;
200   static char fullpath[PATH_MAX],
201   	elf_interpreter[PATH_MAX],
202 	firstexec[PATH_MAX],
203 	finalexec[PATH_MAX];
204   struct lookup resolve;
205   struct fproc *vmfp = fproc_addr(VM_PROC_NR);
206   stackhook_t makestack = NULL;
207   struct filp *newfilp = NULL;
208 
209   lock_exec();
210 
211   /* unset execi values are 0. */
212   memset(&execi, 0, sizeof(execi));
213   execi.vmfd = -1;
214 
215   /* passed from exec() libc code */
216   execi.userflags = 0;
217   execi.args.stack_high = minix_get_user_sp();
218   execi.args.stack_size = DEFAULT_STACK_LIMIT;
219 
220   fp->text_size = 0;
221   fp->data_size = 0;
222 
223   lookup_init(&resolve, fullpath, PATH_NOFLAGS, &execi.vmp, &execi.vp);
224 
225   resolve.l_vmnt_lock = VMNT_READ;
226   resolve.l_vnode_lock = VNODE_READ;
227 
228   /* Fetch the stack from the user before destroying the old core image. */
229   if (frame_len > ARG_MAX)
230 	FAILCHECK(ENOMEM); /* stack too big */
231 
232   r = sys_datacopy_wrapper(fp->fp_endpoint, (vir_bytes) frame, SELF, (vir_bytes) mbuf,
233 		   (size_t) frame_len);
234   if (r != OK) { /* can't fetch stack (e.g. bad virtual addr) */
235         printf("VFS: pm_exec: sys_datacopy failed\n");
236 	FAILCHECK(r);
237   }
238 
239   /* Compute the current virtual stack pointer, has to be done before calling
240    * patch_stack, which needs it, and will adapt as required. */
241   vsp = execi.args.stack_high - frame_len;
242 
243   /* The default is to keep the original user and group IDs */
244   execi.args.new_uid = fp->fp_effuid;
245   execi.args.new_gid = fp->fp_effgid;
246 
247   /* Get the exec file name. */
248   FAILCHECK(fetch_name(path, path_len, fullpath));
249   strlcpy(finalexec, fullpath, PATH_MAX);
250   strlcpy(firstexec, fullpath, PATH_MAX);
251 
252   /* Get_read_vp will return an opened vn in execi.
253    * if necessary it releases the existing vp so we can
254    * switch after we find out what's inside the file.
255    * It reads the start of the file.
256    */
257   Get_read_vp(execi, fullpath, 1, 1, &resolve, fp);
258 
259   /* If this is a script (i.e. has a #!/interpreter line),
260    * retrieve the name of the interpreter and open that
261    * executable instead.
262    */
263   if(is_script(&execi)) {
264   	/* patch_stack will add interpreter name and
265 	 * args to stack and retrieve the new binary
266 	 * name into fullpath.
267 	 */
268 	FAILCHECK(fetch_name(path, path_len, fullpath));
269 	FAILCHECK(patch_stack(execi.vp, mbuf, &frame_len, fullpath, &vsp));
270 
271 	strlcpy(finalexec, fullpath, PATH_MAX);
272   	strlcpy(firstexec, fullpath, PATH_MAX);
273 	Get_read_vp(execi, fullpath, 1, 0, &resolve, fp);
274   }
275 
276   /* If this is a dynamically linked executable, retrieve
277    * the name of that interpreter in elf_interpreter and open that
278    * executable instead. But open the current executable in an
279    * fd for the current process.
280    */
281   if(elf_has_interpreter(execi.args.hdr, execi.args.hdr_len,
282 	elf_interpreter, sizeof(elf_interpreter))) {
283 	/* Switch the executable vnode to the interpreter */
284 	execi.is_dyn = 1;
285 
286 	/* The interpreter (loader) needs an fd to the main program,
287 	 * which is currently in finalexec
288 	 */
289 	if ((r = execi.elf_main_fd =
290 	    common_open(finalexec, O_RDONLY, 0, TRUE /*for_exec*/)) < 0) {
291 		printf("VFS: exec: dynamic: open main exec failed %s (%d)\n",
292 			fullpath, r);
293 		FAILCHECK(r);
294 	}
295 
296 	/* ld.so is linked at 0, but it can relocate itself; we
297 	 * want it higher to trap NULL pointer dereferences.
298 	 * Let's put it below the stack, and reserve 10MB for ld.so.
299 	 */
300 	execi.args.load_offset =
301 		 execi.args.stack_high - execi.args.stack_size - 0xa00000;
302 
303 	/* Remember it */
304 	strlcpy(execi.execname, finalexec, PATH_MAX);
305 
306 	/* The executable we need to execute first (loader)
307 	 * is in elf_interpreter, and has to be in fullpath to
308 	 * be looked up
309 	 */
310 	strlcpy(fullpath, elf_interpreter, PATH_MAX);
311 	strlcpy(firstexec, elf_interpreter, PATH_MAX);
312 	Get_read_vp(execi, fullpath, 0, 0, &resolve, fp);
313   }
314 
315   /* We also want an FD for VM to mmap() the process in if possible. */
316   {
317 	struct vnode *vp = execi.vp;
318 	assert(vp);
319 	if ((vp->v_vmnt->m_fs_flags & RES_HASPEEK) &&
320 			major(vp->v_dev) != MEMORY_MAJOR) {
321 		int newfd = -1;
322 		if(get_fd(vmfp, 0, R_BIT, &newfd, &newfilp) == OK) {
323 			assert(newfd >= 0 && newfd < OPEN_MAX);
324 			assert(!vmfp->fp_filp[newfd]);
325 			newfilp->filp_count = 1;
326 			newfilp->filp_vno = vp;
327 			newfilp->filp_flags = O_RDONLY;
328 			vmfp->fp_filp[newfd] = newfilp;
329 			/* dup_vnode(vp); */
330 			execi.vmfd = newfd;
331 			execi.args.memmap = vfs_memmap;
332 		}
333 	}
334   }
335 
336   /* callback functions and data */
337   execi.args.copymem = read_seg;
338   execi.args.clearproc = libexec_clearproc_vm_procctl;
339   execi.args.clearmem = libexec_clear_sys_memset;
340   execi.args.allocmem_prealloc_cleared = libexec_alloc_mmap_prealloc_cleared;
341   execi.args.allocmem_prealloc_junk = libexec_alloc_mmap_prealloc_junk;
342   execi.args.allocmem_ondemand = libexec_alloc_mmap_ondemand;
343   execi.args.opaque = &execi;
344 
345   execi.args.proc_e = fp->fp_endpoint;
346   execi.args.frame_len = frame_len;
347   execi.args.filesize = execi.vp->v_size;
348 
349   for (i = 0; exec_loaders[i].load_object != NULL; i++) {
350       r = (*exec_loaders[i].load_object)(&execi.args);
351       /* Loaded successfully, so no need to try other loaders */
352       if (r == OK) { makestack = exec_loaders[i].setup_stack; break; }
353   }
354 
355   FAILCHECK(r);
356 
357   /* Inform PM */
358   FAILCHECK(libexec_pm_newexec(fp->fp_endpoint, &execi.args));
359 
360   /* Save off PC */
361   *pc = execi.args.pc;
362 
363   /* call a stack-setup function if this executable type wants it */
364   if(makestack) FAILCHECK(makestack(&execi, mbuf, &frame_len, &vsp));
365 
366   /* Copy the stack from VFS to new core image. */
367   FAILCHECK(sys_datacopy_wrapper(SELF, (vir_bytes) mbuf, fp->fp_endpoint,
368 	(vir_bytes) vsp, (phys_bytes)frame_len));
369 
370   /* Return new stack pointer to caller */
371   *newsp = vsp;
372 
373   clo_exec(fp);
374 
375   if (execi.args.allow_setuid) {
376 	/* If after loading the image we're still allowed to run with
377 	 * setuid or setgid, change credentials now */
378 	fp->fp_effuid = execi.args.new_uid;
379 	fp->fp_effgid = execi.args.new_gid;
380   }
381 
382   /* Remember the new name of the process */
383   strlcpy(fp->fp_name, execi.args.progname, PROC_NAME_LEN);
384   fp->text_size = execi.args.text_size;
385   fp->data_size = execi.args.data_size;
386 
387 pm_execfinal:
388   if(newfilp) unlock_filp(newfilp);
389   else if (execi.vp != NULL) {
390 	unlock_vnode(execi.vp);
391 	put_vnode(execi.vp);
392   }
393 
394   if(execi.vmfd >= 0 && !execi.vmfd_used) {
395   	if(OK != close_fd(vmfp, execi.vmfd)) {
396 		printf("VFS: unexpected close fail of vm fd\n");
397 	}
398   }
399 
400   unlock_exec();
401 
402   return(r);
403 }
404 
405 /* This is a copy-paste of the same macro in minix/lib/libc/sys/stack_utils.c.
406  * Keep it synchronized. */
407 #define STACK_MIN_SZ \
408 ( \
409        sizeof(int) + sizeof(void *) * 2 + \
410        sizeof(AuxInfo) * PMEF_AUXVECTORS + PMEF_EXECNAMELEN1 + \
411        sizeof(struct ps_strings) \
412 )
413 
414 static int stack_prepare_elf(struct vfs_exec_info *execi, char *frame, size_t *frame_size,
415 	vir_bytes *vsp)
416 {
417 	AuxInfo *aux_vec, *aux_vec_end;
418 	vir_bytes vap; /* Address in proc space of the first AuxVec. */
419 	Elf_Ehdr const * const elf_header = (Elf_Ehdr *) execi->args.hdr;
420 	struct ps_strings const * const psp = (struct ps_strings *)
421 		(frame + (*frame_size - sizeof(struct ps_strings)));
422 
423 	size_t const execname_len = strlen(execi->execname);
424 
425 	if (!execi->is_dyn)
426 		return OK;
427 
428 	if (execi->args.hdr_len < sizeof(*elf_header)) {
429 		printf("VFS: malformed ELF headers for exec\n");
430 		return ENOEXEC;
431 	}
432 
433 	if (*frame_size < STACK_MIN_SZ) {
434 		printf("VFS: malformed stack for exec(), smaller than minimum"
435 			" possible size.\n");
436 		return ENOEXEC;
437 	}
438 
439 	/* Find first Aux vector in the stack frame. */
440 	vap = (vir_bytes)(psp->ps_envstr + (psp->ps_nenvstr + 1));
441 	aux_vec = (AuxInfo *) (frame + (vap - *vsp));
442 	aux_vec_end = aux_vec + PMEF_AUXVECTORS;
443 
444 	if (((char *)aux_vec < frame) ||
445 		((char *)aux_vec > (frame + *frame_size))) {
446 		printf("VFS: malformed stack for exec(), first AuxVector is"
447 		       " not on the stack.\n");
448 		return ENOEXEC;
449 	}
450 
451 	if (((char *)aux_vec_end < frame) ||
452 		((char *)aux_vec_end > (frame + *frame_size))) {
453 		printf("VFS: malformed stack for exec(), last AuxVector is"
454 		       " not on the stack.\n");
455 		return ENOEXEC;
456 	}
457 
458 	/* Userland provides a fully filled stack frame, with argc, argv, envp
459 	 * and then all the argv and envp strings; consistent with ELF ABI,
460 	 * except for a list of Aux vectors that should be between envp points
461 	 * and the start of the strings.
462 	 *
463 	 * It would take some very unpleasant hackery to insert the aux vectors
464 	 * before the strings, and correct all the pointers, so the exec code
465 	 * in libc makes space for us.
466 	 */
467 
468 #define AUXINFO(a, type, value) \
469 	do { \
470 		if (a < aux_vec_end) { \
471 			a->a_type = type; \
472 			a->a_v = value; \
473 			a++; \
474 		} else { \
475 			printf("VFS: No more room for ELF AuxVec type %d, skipping it for %s\n", type, execi->execname); \
476 			(aux_vec_end - 1)->a_type = AT_NULL; \
477 			(aux_vec_end - 1)->a_v = 0; \
478 		} \
479 	} while(0)
480 
481 	AUXINFO(aux_vec, AT_BASE, execi->args.load_base);
482 	AUXINFO(aux_vec, AT_ENTRY, execi->args.pc);
483 	AUXINFO(aux_vec, AT_EXECFD, execi->elf_main_fd);
484 #if 0
485 	AUXINFO(aux_vec, AT_PHDR, XXX ); /* should be &phdr[0] */
486 	AUXINFO(aux_vec, AT_PHENT, elf_header->e_phentsize);
487 	AUXINFO(aux_vec, AT_PHNUM, elf_header->e_phnum);
488 
489 	AUXINFO(aux_vec, AT_RUID, XXX);
490 	AUXINFO(aux_vec, AT_RGID, XXX);
491 #endif
492 	AUXINFO(aux_vec, AT_EUID, execi->args.new_uid);
493 	AUXINFO(aux_vec, AT_EGID, execi->args.new_gid);
494 	AUXINFO(aux_vec, AT_PAGESZ, PAGE_SIZE);
495 
496 	if(execname_len < PMEF_EXECNAMELEN1) {
497 		char *spacestart;
498 		vir_bytes userp;
499 
500 		/* Empty space starts after aux_vec table; we can put the name
501 		 * here. */
502 		spacestart = (char *) aux_vec + 2 * sizeof(AuxInfo);
503 		strlcpy(spacestart, execi->execname, PMEF_EXECNAMELEN1);
504 		memset(spacestart + execname_len, '\0',
505 			PMEF_EXECNAMELEN1 - execname_len);
506 
507 		/* What will the address of the string for the user be */
508 		userp = *vsp + (spacestart - frame);
509 
510 		/* Move back to where the AT_NULL is */
511 		AUXINFO(aux_vec, AT_SUN_EXECNAME, userp);
512 	}
513 
514 	/* Always terminate with AT_NULL */
515 	AUXINFO(aux_vec, AT_NULL, 0);
516 
517 	return OK;
518 }
519 
520 /*===========================================================================*
521  *				is_script				     *
522  *===========================================================================*/
523 static int is_script(struct vfs_exec_info *execi)
524 {
525 /* Is Interpreted script? */
526   assert(execi->args.hdr != NULL);
527 
528   return(execi->args.hdr[0] == '#' && execi->args.hdr[1] == '!'
529   	&& execi->args.hdr_len >= 2);
530 }
531 
532 /*===========================================================================*
533  *				patch_stack				     *
534  *===========================================================================*/
535 static int patch_stack(vp, stack, stk_bytes, path, vsp)
536 struct vnode *vp;		/* pointer for open script file */
537 char stack[ARG_MAX];		/* pointer to stack image within VFS */
538 size_t *stk_bytes;		/* size of initial stack */
539 char path[PATH_MAX];		/* path to script file */
540 vir_bytes *vsp;
541 {
542 /* Patch the argument vector to include the path name of the script to be
543  * interpreted, and all strings on the #! line.  Returns the path name of
544  * the interpreter.
545  */
546   enum { INSERT=FALSE, REPLACE=TRUE };
547   int n, r;
548   off_t pos, new_pos;
549   char *sp, *interp = NULL;
550   size_t cum_io;
551   char buf[PAGE_SIZE];
552 
553   /* Make 'path' the new argv[0]. */
554   if (!insert_arg(stack, stk_bytes, path, vsp, REPLACE)) return(ENOMEM);
555 
556   pos = 0;	/* Read from the start of the file */
557 
558   /* Issue request */
559   r = req_readwrite(vp->v_fs_e, vp->v_inode_nr, pos, READING, VFS_PROC_NR,
560 			(vir_bytes) buf, sizeof(buf), &new_pos, &cum_io);
561 
562   if (r != OK) return(r);
563 
564   n = vp->v_size;
565   if (n > sizeof(buf))
566 	n = sizeof(buf);
567   if (n < 2) return ENOEXEC;
568 
569   sp = &(buf[2]);				/* just behind the #! */
570   n -= 2;
571   if (n > PATH_MAX) n = PATH_MAX;
572 
573   /* Use the 'path' variable for temporary storage */
574   memcpy(path, sp, n);
575 
576   if ((sp = memchr(path, '\n', n)) == NULL) /* must be a proper line */
577 	return(ENOEXEC);
578 
579   /* Move sp backwards through script[], prepending each string to stack. */
580   for (;;) {
581 	/* skip spaces behind argument. */
582 	while (sp > path && (*--sp == ' ' || *sp == '\t')) {}
583 	if (sp == path) break;
584 
585 	sp[1] = 0;
586 	/* Move to the start of the argument. */
587 	while (sp > path && sp[-1] != ' ' && sp[-1] != '\t') --sp;
588 
589 	interp = sp;
590 	if (!insert_arg(stack, stk_bytes, sp, vsp, INSERT)) {
591 		printf("VFS: patch_stack: insert_arg failed\n");
592 		return(ENOMEM);
593 	}
594   }
595 
596   if(!interp)
597   	return ENOEXEC;
598 
599   if (interp != path)
600 	memmove(path, interp, strlen(interp)+1);
601 
602   return(OK);
603 }
604 
605 /*===========================================================================*
606  *				insert_arg				     *
607  *===========================================================================*/
608 static int insert_arg(char stack[ARG_MAX], size_t *stk_bytes, char *arg,
609 	vir_bytes *vsp, char replace)
610 {
611 	/* Patch the stack so that arg will become argv[0]. Be careful, the
612 	 * stack may be filled with garbage, although it normally looks like
613 	 * this:
614 	 *	nargs argv[0] ... argv[nargs-1] NULL envp[0] ... NULL
615 	 * followed by the strings "pointed" to by the argv[i] and the envp[i].
616 	 * The * pointers are in the new process address space.
617 	 *
618 	 * Return true iff the operation succeeded.
619 	 */
620 	struct ps_strings *psp;
621 	int offset;
622 	size_t old_bytes = *stk_bytes;
623 
624 	int const arg_len = strlen(arg) + 1;
625 
626 	/* Offset to argv[0][0] in the stack frame. */
627 	int const a0 = (int)(((char **)stack)[1] - *vsp);
628 
629 	/* Check that argv[0] points within the stack frame. */
630 	if ((a0 < 0) || (a0 >= old_bytes)) {
631 		printf("vfs:: argv[0][] not within stack range!! %i\n", a0);
632 		return FALSE;
633 	}
634 
635 	if (!replace) {
636 		/* Prepending arg adds one pointer, one string and a zero byte. */
637 		offset = arg_len + PTRSIZE;
638 	} else {
639 		/* replacing argv[0] with arg adds the difference in length of
640 		 * the two strings. Make sure we don't go beyond the stack size
641 		 * when computing the length of the current argv[0]. */
642 		offset = arg_len - strnlen(stack + a0, ARG_MAX - a0 - 1);
643 	}
644 
645 	/* As ps_strings follows the strings, ensure the offset is word aligned. */
646 	offset = offset + (PTRSIZE - ((PTRSIZE + offset) % PTRSIZE));
647 
648 	/* The stack will grow (or shrink) by offset bytes. */
649 	if ((*stk_bytes += offset) > ARG_MAX) {
650 		printf("vfs:: offset too big!! %zu (max %d)\n", *stk_bytes,
651 			ARG_MAX);
652 		return FALSE;
653 	}
654 
655 	/* Reposition the strings by offset bytes */
656 	memmove(stack + a0 + offset, stack + a0, old_bytes - a0);
657 
658 	/* Put arg in the new space, leaving padding in front of it. */
659 	strlcpy(stack + a0 + offset - arg_len, arg, arg_len);
660 
661 	if (!replace) {
662 		/* Make space for a new argv[0]. */
663 		memmove(stack + 2 * PTRSIZE,
664 			stack + 1 * PTRSIZE, a0 - 2 * PTRSIZE);
665 
666 		((char **) stack)[0]++;	/* nargs++; */
667 	}
668 
669 	/* set argv[0] correctly */
670 	((char **) stack)[1] = (char *) a0 - arg_len + *vsp;
671 
672 	/* Update stack pointer in the process address space. */
673 	*vsp -= offset;
674 
675 	/* Update argv and envp in ps_strings */
676 	psp = (struct ps_strings *) (stack + *stk_bytes - sizeof(struct ps_strings));
677 	psp->ps_argvstr -= (offset / PTRSIZE);
678 	if (!replace) {
679 		psp->ps_nargvstr++;
680 	}
681 	psp->ps_envstr = psp->ps_argvstr + psp->ps_nargvstr + 1;
682 
683 	return TRUE;
684 }
685 
686 /*===========================================================================*
687  *				read_seg				     *
688  *===========================================================================*/
689 static int read_seg(struct exec_info *execi, off_t off, vir_bytes seg_addr, size_t seg_bytes)
690 {
691 /*
692  * The byte count on read is usually smaller than the segment count, because
693  * a segment is padded out to a click multiple, and the data segment is only
694  * partially initialized.
695  */
696   int r;
697   off_t new_pos;
698   size_t cum_io;
699   struct vnode *vp = ((struct vfs_exec_info *) execi->opaque)->vp;
700 
701   /* Make sure that the file is big enough */
702   if (off + seg_bytes > LONG_MAX) return(EIO);
703   if ((unsigned long) vp->v_size < off+seg_bytes) return(EIO);
704 
705   if ((r = req_readwrite(vp->v_fs_e, vp->v_inode_nr, off, READING,
706 		 execi->proc_e, (vir_bytes) seg_addr, seg_bytes,
707 		 &new_pos, &cum_io)) != OK) {
708     printf("VFS: read_seg: req_readwrite failed (data)\n");
709     return(r);
710   }
711 
712   if (r == OK && cum_io != seg_bytes)
713 	printf("VFS: read_seg segment has not been read properly\n");
714 
715 	return(r);
716 }
717 
718 
719 /*===========================================================================*
720  *				clo_exec				     *
721  *===========================================================================*/
722 static void clo_exec(struct fproc *rfp)
723 {
724 /* Files can be marked with the FD_CLOEXEC bit (in fp->fp_cloexec).
725  */
726   int i;
727 
728   /* Check the file desriptors one by one for presence of FD_CLOEXEC. */
729   for (i = 0; i < OPEN_MAX; i++)
730 	if ( FD_ISSET(i, &rfp->fp_cloexec_set))
731 		(void) close_fd(rfp, i);
732 }
733 
734 /*===========================================================================*
735  *				map_header				     *
736  *===========================================================================*/
737 static int map_header(struct vfs_exec_info *execi)
738 {
739   int r;
740   size_t cum_io;
741   off_t pos, new_pos;
742   /* Assume that header is not larger than a page. Align the buffer reasonably
743    * well, because libexec casts it to a structure directly and therefore
744    * expects it to be aligned appropriately. From here we can only guess the
745    * proper alignment, but 64 bits should work for all versions of ELF..
746    */
747   static char hdr[PAGE_SIZE] __aligned(8);
748 
749   pos = 0;	/* Read from the start of the file */
750 
751   /* How much is sensible to read */
752   execi->args.hdr_len = MIN(execi->vp->v_size, sizeof(hdr));
753   execi->args.hdr = hdr;
754 
755   r = req_readwrite(execi->vp->v_fs_e, execi->vp->v_inode_nr,
756   	pos, READING, VFS_PROC_NR, (vir_bytes) hdr,
757 	execi->args.hdr_len, &new_pos, &cum_io);
758   if (r != OK) {
759 	printf("VFS: exec: map_header: req_readwrite failed\n");
760 	return(r);
761   }
762 
763   return(OK);
764 }
765