os161-1.99
 All Data Structures
runprogram.c
00001 /*
00002  * Copyright (c) 2000, 2001, 2002, 2003, 2004, 2005, 2008, 2009
00003  *      The President and Fellows of Harvard College.
00004  *
00005  * Redistribution and use in source and binary forms, with or without
00006  * modification, are permitted provided that the following conditions
00007  * are met:
00008  * 1. Redistributions of source code must retain the above copyright
00009  *    notice, this list of conditions and the following disclaimer.
00010  * 2. Redistributions in binary form must reproduce the above copyright
00011  *    notice, this list of conditions and the following disclaimer in the
00012  *    documentation and/or other materials provided with the distribution.
00013  * 3. Neither the name of the University nor the names of its contributors
00014  *    may be used to endorse or promote products derived from this software
00015  *    without specific prior written permission.
00016  *
00017  * THIS SOFTWARE IS PROVIDED BY THE UNIVERSITY AND CONTRIBUTORS ``AS IS'' AND
00018  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
00019  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
00020  * ARE DISCLAIMED.  IN NO EVENT SHALL THE UNIVERSITY OR CONTRIBUTORS BE LIABLE
00021  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
00022  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
00023  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
00024  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
00025  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
00026  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
00027  * SUCH DAMAGE.
00028  */
00029 
00030 /*
00031  * Sample/test code for running a user program.  You can use this for
00032  * reference when implementing the execv() system call. Remember though
00033  * that execv() needs to do more than this function does.
00034  */
00035 
00036 #include <types.h>
00037 #include <kern/errno.h>
00038 #include <kern/fcntl.h>
00039 #include <lib.h>
00040 #include <proc.h>
00041 #include <current.h>
00042 #include <addrspace.h>
00043 #include <vm.h>
00044 #include <vfs.h>
00045 #include <syscall.h>
00046 #include <test.h>
00047 
00048 /*
00049  * Load program "progname" and start running it in usermode.
00050  * Does not return except on error.
00051  *
00052  * Calls vfs_open on progname and thus may destroy it.
00053  */
00054 int
00055 runprogram(char *progname)
00056 {
00057         struct addrspace *as;
00058         struct vnode *v;
00059         vaddr_t entrypoint, stackptr;
00060         int result;
00061 
00062         /* Open the file. */
00063         result = vfs_open(progname, O_RDONLY, 0, &v);
00064         if (result) {
00065                 return result;
00066         }
00067 
00068         /* We should be a new process. */
00069         KASSERT(curproc_getas() == NULL);
00070 
00071         /* Create a new address space. */
00072         as = as_create();
00073         if (as ==NULL) {
00074                 vfs_close(v);
00075                 return ENOMEM;
00076         }
00077 
00078         /* Switch to it and activate it. */
00079         curproc_setas(as);
00080         as_activate();
00081 
00082         /* Load the executable. */
00083         result = load_elf(v, &entrypoint);
00084         if (result) {
00085                 /* p_addrspace will go away when curproc is destroyed */
00086                 vfs_close(v);
00087                 return result;
00088         }
00089 
00090         /* Done with the file now. */
00091         vfs_close(v);
00092 
00093         /* Define the user stack in the address space */
00094         result = as_define_stack(as, &stackptr);
00095         if (result) {
00096                 /* p_addrspace will go away when curproc is destroyed */
00097                 return result;
00098         }
00099 
00100         /* Warp to user mode. */
00101         enter_new_process(0 /*argc*/, NULL /*userspace addr of argv*/,
00102                           stackptr, entrypoint);
00103         
00104         /* enter_new_process does not return. */
00105         panic("enter_new_process returned\n");
00106         return EINVAL;
00107 }
00108 
 All Data Structures