|
@@ -13,6 +13,8 @@
|
|
|
#include <addrspace.h>
|
|
|
#include <copyinout.h>
|
|
|
#include <synch.h>
|
|
|
+#include <kern/fcntl.h>
|
|
|
+
|
|
|
|
|
|
void sys__exit(int exitcode)
|
|
|
{
|
|
@@ -140,17 +142,19 @@ int sys_fork(struct trapframe * tf, int * retval)
|
|
|
// args is an array of null terminated strings, with the last element being a null pointer so we don't overflow if iterating
|
|
|
// since this is a userptr type, we need to use copyin and copyout to get data properly
|
|
|
// hopefully the path in program is actually a full filepath
|
|
|
-int execv(const char * program, userptr_t args)
|
|
|
+int sys_execv(const char * program, userptr_t args)
|
|
|
{
|
|
|
- userptr_t ** temp = args;
|
|
|
+ userptr_t temp = args;
|
|
|
int argcount = 0;
|
|
|
char ** argv;
|
|
|
+ int errcode; // why don't we have errno!!! my beautful one-liners!!!
|
|
|
|
|
|
// see how long args is
|
|
|
while (1)
|
|
|
{
|
|
|
char * temp2;
|
|
|
- if (copyin(temp, temp2, sizeof(char *))) return errno;
|
|
|
+ errcode = copyin((const userptr_t)temp, temp2, sizeof(char *));
|
|
|
+ if (errcode) return errcode;
|
|
|
if (!(temp2)) break;
|
|
|
temp += sizeof(char *);
|
|
|
++argcount;
|
|
@@ -164,17 +168,18 @@ int execv(const char * program, userptr_t args)
|
|
|
temp = args;
|
|
|
for (int i = 0; i < argcount; ++i)
|
|
|
{
|
|
|
- if (copyin(args, argv[i], sizeof(char *)))
|
|
|
+ errcode = copyin(temp, argv[i], sizeof(char *));
|
|
|
+ if (errcode)
|
|
|
{
|
|
|
kfree(argv);
|
|
|
- return errno;
|
|
|
+ return errcode;
|
|
|
}
|
|
|
|
|
|
temp += sizeof(char *);
|
|
|
}
|
|
|
|
|
|
// allocate space for filepath
|
|
|
- const char * filepath = kmalloc(sizeof(program));
|
|
|
+ char * filepath = kmalloc(sizeof(program));
|
|
|
if (!(filepath))
|
|
|
{
|
|
|
kfree(argv);
|
|
@@ -183,63 +188,75 @@ int execv(const char * program, userptr_t args)
|
|
|
|
|
|
// get kernel copy of filepath
|
|
|
size_t useless;
|
|
|
- if (copyinstr((const_userptr_t)program, filepath, strlen(program) + 1, &useless))
|
|
|
+ errcode = copyinstr((const_userptr_t)program, filepath, strlen(program) + 1, &useless);
|
|
|
+ if (errcode)
|
|
|
{
|
|
|
kfree(argv);
|
|
|
kfree(filepath);
|
|
|
- return errno;
|
|
|
+ return errcode;
|
|
|
}
|
|
|
|
|
|
// open program file
|
|
|
struct vnode * v;
|
|
|
- if (vfs_open(filepath, O_RDONLY, 0, &v))
|
|
|
+ errcode = vfs_open(filepath, O_RDONLY, 0, &v);
|
|
|
+ if (errcode)
|
|
|
{
|
|
|
kfree(argv);
|
|
|
kfree(filepath);
|
|
|
- return errno;
|
|
|
+ return errcode;
|
|
|
}
|
|
|
|
|
|
// create new address space
|
|
|
- struct addrspace * as = as_create();
|
|
|
- if (!(as))
|
|
|
- {
|
|
|
- vfs_close(v);
|
|
|
- kfree(argv);
|
|
|
- kfree(filepath);
|
|
|
- return ENOMEM;
|
|
|
- }
|
|
|
+ struct addrspace * as = as_create();
|
|
|
+ if (!(as))
|
|
|
+ {
|
|
|
+ vfs_close(v);
|
|
|
+ kfree(argv);
|
|
|
+ kfree(filepath);
|
|
|
+ return ENOMEM;
|
|
|
+ }
|
|
|
|
|
|
- // Switch to the new as and activate. keep old in case we fail later
|
|
|
- struct addrspace * oldas = curproc_getas();
|
|
|
- curproc_setas(as);
|
|
|
- as_activate();
|
|
|
+ // Switch to the new as and activate. keep old in case we fail later
|
|
|
+ struct addrspace * oldas = curproc_getas();
|
|
|
+ curproc_setas(as);
|
|
|
+ as_activate();
|
|
|
|
|
|
- // Load santa's favourite slave
|
|
|
- if (load_elf(v, &entrypoint))
|
|
|
- {
|
|
|
- vfs_close(v);
|
|
|
- kfree(argv);
|
|
|
- kfree(filepath);
|
|
|
- curproc_setas(oldas);
|
|
|
- as_destroy(as);
|
|
|
- return errno;
|
|
|
- }
|
|
|
+ vaddr_t entrypoint;
|
|
|
+ // Load santa's favourite slave
|
|
|
+ errcode = load_elf(v, &entrypoint);
|
|
|
+ if (errcode)
|
|
|
+ {
|
|
|
+ vfs_close(v);
|
|
|
+ kfree(argv);
|
|
|
+ kfree(filepath);
|
|
|
+ curproc_setas(oldas);
|
|
|
+ as_destroy(as);
|
|
|
+ return errcode;
|
|
|
+ }
|
|
|
|
|
|
- // close the file now that we're done
|
|
|
- vfs_close(v);
|
|
|
+ // close the file now that we're done
|
|
|
+ vfs_close(v);
|
|
|
|
|
|
// create the new stack
|
|
|
vaddr_t newstack;
|
|
|
- if (as_define_stack(as, &newstack))
|
|
|
+ errcode = as_define_stack(as, &newstack);
|
|
|
+ if (errcode)
|
|
|
{
|
|
|
kfree(argv);
|
|
|
kfree(filepath);
|
|
|
curproc_setas(oldas);
|
|
|
as_destroy(as);
|
|
|
- return errno;
|
|
|
+ return errcode;
|
|
|
}
|
|
|
|
|
|
userptr_t userArgv;
|
|
|
|
|
|
// need to copy data over into the new address space, as rn it is all in kernel heap
|
|
|
+
|
|
|
+ // delete old addrspace, enter new process
|
|
|
+ as_destroy(oldas);
|
|
|
+ enter_new_process(argcount, userArgv, newstack, entrypoint);
|
|
|
+ panic("Enter new process returned!\n");
|
|
|
+ return EINVAL;
|
|
|
}
|
|
|
+
|