devices_SRC += devices/input.c # Serial and keyboard input.
devices_SRC += devices/intq.c # Interrupt queue.
devices_SRC += devices/rtc.c # Real-time clock.
+devices_SRC += devices/shutdown.c # Reboot and power off.
# Library code shared between kernel and user programs.
lib_SRC = lib/debug.c # Debug helpers.
#include <stdio.h>
#include <string.h>
#include "devices/input.h"
-#include "threads/init.h"
+#include "devices/shutdown.h"
#include "threads/interrupt.h"
#include "threads/io.h"
{
/* Reboot if Ctrl+Alt+Del pressed. */
if (c == 0177 && ctrl && alt)
- reboot ();
+ shutdown_reboot ();
/* Handle Ctrl, Shift.
Note that Ctrl overrides Shift. */
--- /dev/null
+#include "devices/shutdown.h"
+#include <console.h>
+#include <stdio.h>
+#include "devices/kbd.h"
+#include "devices/serial.h"
+#include "devices/timer.h"
+#include "threads/io.h"
+#include "threads/thread.h"
+#ifdef USERPROG
+#include "userprog/exception.h"
+#endif
+#ifdef FILESYS
+#include "devices/disk.h"
+#include "filesys/filesys.h"
+#endif
+
+/* Keyboard control register port. */
+#define CONTROL_REG 0x64
+
+static void print_stats (void);
+
+/* Reboots the machine via the keyboard controller. */
+void
+shutdown_reboot (void)
+{
+ int i;
+
+ printf ("Rebooting...\n");
+
+ /* See [kbd] for details on how to program the keyboard
+ * controller. */
+ for (i = 0; i < 100; i++)
+ {
+ int j;
+
+ /* Poll keyboard controller's status byte until
+ * 'input buffer empty' is reported. */
+ for (j = 0; j < 0x10000; j++)
+ {
+ if ((inb (CONTROL_REG) & 0x02) == 0)
+ break;
+ timer_udelay (2);
+ }
+
+ timer_udelay (50);
+
+ /* Pulse bit 0 of the output port P2 of the keyboard controller.
+ * This will reset the CPU. */
+ outb (CONTROL_REG, 0xfe);
+ timer_udelay (50);
+ }
+}
+
+/* Powers down the machine we're running on,
+ as long as we're running on Bochs or QEMU. */
+void
+shutdown_power_off (void)
+{
+ const char s[] = "Shutdown";
+ const char *p;
+
+#ifdef FILESYS
+ filesys_done ();
+#endif
+
+ print_stats ();
+
+ printf ("Powering off...\n");
+ serial_flush ();
+
+ /* This is a special power-off sequence supported by Bochs and
+ QEMU, but not by physical hardware. */
+ for (p = s; *p != '\0'; p++)
+ outb (0x8900, *p);
+
+ /* This will power off a VMware VM if "gui.exitOnCLIHLT = TRUE"
+ is set in its configuration file. (The "pintos" script does
+ that automatically.) */
+ asm volatile ("cli; hlt" : : : "memory");
+
+ /* None of those worked. */
+ printf ("still running...\n");
+ for (;;);
+}
+
+/* Print statistics about Pintos execution. */
+static void
+print_stats (void)
+{
+ timer_print_stats ();
+ thread_print_stats ();
+#ifdef FILESYS
+ disk_print_stats ();
+#endif
+ console_print_stats ();
+ kbd_print_stats ();
+#ifdef USERPROG
+ exception_print_stats ();
+#endif
+}
--- /dev/null
+#ifndef DEVICES_SHUTDOWN_H
+#define DEVICES_SHUTDOWN_H
+
+void shutdown_reboot (void);
+void shutdown_power_off (void);
+
+#endif /* devices/shutdown.h */
#include "threads/switch.h"
#include "threads/vaddr.h"
#include "devices/serial.h"
+#include "devices/shutdown.h"
/* Halts the OS, printing the source file name, line number, and
function name, plus a user-specific message. */
serial_flush ();
if (power_off_when_done)
- power_off ();
+ shutdown_power_off ();
for (;;);
}
#include "devices/kbd.h"
#include "devices/input.h"
#include "devices/serial.h"
+#include "devices/shutdown.h"
#include "devices/timer.h"
#include "devices/vga.h"
#include "devices/rtc.h"
static void run_actions (char **argv);
static void usage (void);
-static void print_stats (void);
-
-
int main (void) NO_RETURN;
/* Pintos main program. */
/* Finish up. */
if (reboot_when_done)
- reboot ();
+ shutdown_reboot ();
if (power_off_when_done)
- power_off ();
+ shutdown_power_off ();
thread_exit ();
}
\f
" -ul=COUNT Limit user memory to COUNT pages.\n"
#endif
);
- power_off ();
-}
-
-/* Keyboard control register port. */
-#define CONTROL_REG 0x64
-
-/* Reboots the machine via the keyboard controller. */
-void
-reboot (void)
-{
- int i;
-
- printf ("Rebooting...\n");
-
- /* See [kbd] for details on how to program the keyboard
- * controller. */
- for (i = 0; i < 100; i++)
- {
- int j;
-
- /* Poll keyboard controller's status byte until
- * 'input buffer empty' is reported. */
- for (j = 0; j < 0x10000; j++)
- {
- if ((inb (CONTROL_REG) & 0x02) == 0)
- break;
- timer_udelay (2);
- }
-
- timer_udelay (50);
-
- /* Pulse bit 0 of the output port P2 of the keyboard controller.
- * This will reset the CPU. */
- outb (CONTROL_REG, 0xfe);
- timer_udelay (50);
- }
-}
-
-/* Powers down the machine we're running on,
- as long as we're running on Bochs or QEMU. */
-void
-power_off (void)
-{
- const char s[] = "Shutdown";
- const char *p;
-
-#ifdef FILESYS
- filesys_done ();
-#endif
-
- print_stats ();
-
- printf ("Powering off...\n");
- serial_flush ();
-
- for (p = s; *p != '\0'; p++)
- outb (0x8900, *p);
- asm volatile ("cli; hlt" : : : "memory");
- printf ("still running...\n");
- for (;;);
-}
-
-/* Print statistics about Pintos execution. */
-static void
-print_stats (void)
-{
- timer_print_stats ();
- thread_print_stats ();
-#ifdef FILESYS
- disk_print_stats ();
-#endif
- console_print_stats ();
- kbd_print_stats ();
-#ifdef USERPROG
- exception_print_stats ();
-#endif
+ shutdown_power_off ();
}