How to automatically close the execution of the `qemu` after end of process?
UPDATED:
The new solution
Here is another better solution that will work for both pintos run ...
and make grade
add this line to devices/shutdown.c :: shutdown_power_off(void) before the loop.
outw( 0x604, 0x0 | 0x2000 );
The old Solution
For the newer versions of qemu you need to run it with the option
-device isa-debug-exit
Which exits on any write to an IO port, by default it's 0x501
i.e in your pintos project under the src/utils directory you will need to add one line to the pintos file in the run_qemu subroutine
sub run_qemu { print "warning: qemu doesn't support --terminal\n" if $vga eq 'terminal'; print "warning: qemu doesn't support jitter\n" if defined $jitter; my (@cmd) = ('qemu-system-i386'); push (@cmd, '-device', 'isa-debug-exit'); # <====== add this line .. .. push (@cmd, '-monitor', 'null') if $vga eq 'none' && $debug eq 'none'; run_command (@cmd);}
and in shutdown.c file under the devices directoryadd this line in the shutdown_power_off function after the for loop
for (p = s; *p != '\0'; p++) outb (0x8900, *p);outb (0x501, 0x31); // <====== add this line
Qemu's exit code is double the value plus one, so there is no way to exit cleanly. Use 0x31 which should result in a qemu exit code of 0x63
finally run pintos with -q option
pintos -q run alarm-multiple
- Note: this solution will not work for
make grade
see the comment below by @pranav3688 for a solution.
[I recognize the question refers specifically to pintos, but I found myself learning from the answers here, enough to do this under Linux. I thought I would leave this here in case others came to this page for similar reasons...]
I found myself doing this for testing purposes. Here is the code I used (run as root from within the QEMU session):
#include <stdio.h>#include <stdlib.h>#include <sys/io.h>#include <unistd.h>#define SHUTDOWN_PORT 0x604#define EXIT_PORT 0x501static void clean_exit(void) { ioperm(SHUTDOWN_PORT, 16, 1); outw(0x2000, SHUTDOWN_PORT);}int main(int argc, char **argv) { int status; if (argc != 2) { clean_exit(); } status = atoi(argv[1]); printf("exiting with status %d (in three seconds)\n", status); sleep(3); if (!status) { clean_exit(); } ioperm(EXIT_PORT, 8, 1); /* * status returned is 1+(2*orig_status) */ outb(status-1, EXIT_PORT); printf("didn't exit.. did you include '-device isa-debug-exit'" " in qemu command?\n"); exit(1);}
My QEMU environment image is pretty sparse, so I compile it statically as follows:
$ gcc -O2 exit.c -o exit --static
Usage:
Just exit with status 0 # ./exit Same as above # ./exit 0 Exit with status 1 # ./exit 1 Exit with status 1+2*(n-1) # ./exit n
For my purposes I only really use exit
, exit 0
and exit 1
.