Aquí está mi código para user/sleep.c:
#include "kernel/types.h" #include "user/user.h" int main(int argc, char *argv[]) { if (argc < 2) { printf("You have forgotten to pass an argument."); exit(1); } int arg = atoi(argv[1]); sleep(arg); exit(0); // return 0; }
Todo está bien, pero con el retorno 0 en lugar de salir (0), aparece el siguiente error:
usertrap(): unexpected scause 0x000000000000000d pid=4 sepc=0x00000000000000f6 stval=0x0000000000003008
¿Porqué es eso?
Es porque xv6
no es estándar en este punto:
consultehttps://tnallen.people.clemson.edu/2019/03/04/intro-to-xv6.html
Regresar después de main es un caso especial que no se admite en XV6 , así que recuerde siempre salir() al final de main en lugar de regresar. De lo contrario, obtendrá una "trampa 14", que en este caso es XV6-speak para "falla de segmentación". Finalmente, observe que nuestros encabezados son diferentes. No hay #include <stdio.h> como podría estar acostumbrado. Nuevamente, esto se debe a que la biblioteca estándar es diferente. Consulte estos encabezados para ver qué tipo de utilidades de usuario tiene disponibles.
Entonces, ¿por qué necesitábamos exit()
en xv6?
Porque, al crear un programa con gcc
para Linux, por ejemplo, la cadena de herramientas agrega la llamada de exit
requerida: consulte https://en.wikipedia.org/wiki/Crt0
En resumen, en Linux, su programa no comienza en main
sino en start
:
void start(void) { /* get argc, argv, env) */ int r = main(argc, argv, envp); /* << start calls the actual main */ exit(r); }
En xv6, por el contrario, el punto de partida real es main
cuando el sistema operativo intenta regresar desde main
, no tiene dirección para mostrar lo que causa una falla de segmento