xref: /xv6-public/exec.c (revision 2ea6c764)
1 #include "types.h"
2 #include "param.h"
3 #include "mmu.h"
4 #include "proc.h"
5 #include "defs.h"
6 #include "x86.h"
7 #include "elf.h"
8 
9 int
10 exec(char *path, char **argv)
11 {
12   char *s, *last;
13   int i, off;
14   uint sz = 0;
15   struct elfhdr elf;
16   struct inode *ip = 0;
17   struct proghdr ph;
18   pde_t *pgdir = 0, *oldpgdir;
19 
20   if((ip = namei(path)) == 0)
21     return -1;
22   ilock(ip);
23 
24   // Check ELF header
25   if(readi(ip, (char*)&elf, 0, sizeof(elf)) < sizeof(elf))
26     goto bad;
27   if(elf.magic != ELF_MAGIC)
28     goto bad;
29 
30   if(!(pgdir = setupkvm()))
31     goto bad;
32 
33   // Load program into memory.
34   for(i=0, off=elf.phoff; i<elf.phnum; i++, off+=sizeof(ph)){
35     if(readi(ip, (char*)&ph, off, sizeof(ph)) != sizeof(ph))
36       goto bad;
37     if(ph.type != ELF_PROG_LOAD)
38       continue;
39     if(ph.memsz < ph.filesz)
40       goto bad;
41     if(!(sz = allocuvm(pgdir, sz, ph.va + ph.memsz)))
42       goto bad;
43     if(!loaduvm(pgdir, (char *)ph.va, ip, ph.offset, ph.filesz))
44       goto bad;
45   }
46   iunlockput(ip);
47   ip = 0;
48 
49   // Allocate a one-page stack at the next page boundary
50   sz = PGROUNDUP(sz);
51   if(!(sz = allocuvm(pgdir, sz, sz + PGSIZE)))
52     goto bad;
53 
54   // initialize stack content:
55 
56   // "argumentN"                      -- nul-terminated string
57   // ...
58   // "argument0"
59   // 0                                -- argv[argc]
60   // address of argumentN
61   // ...
62   // address of argument0             -- argv[0]
63   // address of address of argument0  -- argv argument to main()
64   // argc                             -- argc argument to main()
65   // ffffffff                         -- return PC for main() call
66 
67   uint sp = sz;
68 
69   // count arguments
70   int argc;
71   for(argc = 0; argv[argc]; argc++)
72     ;
73   if(argc >= MAXARG)
74     goto bad;
75 
76   // push strings and remember where they are
77   uint strings[MAXARG];
78   for(i = argc - 1; i >= 0; --i){
79     sp -= strlen(argv[i]) + 1;
80     strings[i] = sp;
81     copyout(pgdir, sp, argv[i], strlen(argv[i]) + 1);
82   }
83 
84 #define PUSH(x) { int xx = (int)(x); sp -= 4; copyout(pgdir, sp, &xx, 4); }
85 
86   PUSH(0); // argv[argc] is zero
87 
88   // push argv[] elements
89   for(i = argc - 1; i >= 0; --i)
90     PUSH(strings[i]);
91 
92   PUSH(sp); // argv
93 
94   PUSH(argc);
95 
96   PUSH(0xffffffff); // in case main tries to return
97 
98   if(sp < sz - PGSIZE)
99     goto bad;
100 
101   // Save program name for debugging.
102   for(last=s=path; *s; s++)
103     if(*s == '/')
104       last = s+1;
105   safestrcpy(proc->name, last, sizeof(proc->name));
106 
107   // Commit to the user image.
108   oldpgdir = proc->pgdir;
109   proc->pgdir = pgdir;
110   proc->sz = sz;
111   proc->tf->eip = elf.entry;  // main
112   proc->tf->esp = sp;
113 
114   switchuvm(proc);
115 
116   freevm(oldpgdir);
117 
118   return 0;
119 
120  bad:
121   if(pgdir)
122     freevm(pgdir);
123   if(ip)
124     iunlockput(ip);
125   return -1;
126 }
127