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