| # x86 bootblock used in migration test |
| # repeatedly increments the first byte of each page in a 100MB |
| # range. |
| # Outputs an initial 'A' on serial followed by repeated 'B's |
| # |
| # Copyright (c) 2016 Red Hat, Inc. and/or its affiliates |
| # This work is licensed under the terms of the GNU GPL, version 2 or later. |
| # See the COPYING file in the top-level directory. |
| # |
| # Author: dgilbert@redhat.com |
| |
| #include "migration-test.h" |
| |
| #define ACPI_ENABLE 0xf1 |
| #define ACPI_PORT_SMI_CMD 0xb2 |
| #define ACPI_PM_BASE 0x600 |
| #define PM1A_CNT_OFFSET 4 |
| |
| #define ACPI_SCI_ENABLE 0x0001 |
| #define ACPI_SLEEP_TYPE 0x0400 |
| #define ACPI_SLEEP_ENABLE 0x2000 |
| #define SLEEP (ACPI_SCI_ENABLE + ACPI_SLEEP_TYPE + ACPI_SLEEP_ENABLE) |
| |
| #define LOW_ADDR X86_TEST_MEM_START |
| #define HIGH_ADDR X86_TEST_MEM_END |
| |
| /* Save the suspended status at an address that is not written in the loop. */ |
| #define suspended (X86_TEST_MEM_START + 4) |
| |
| .code16 |
| .org 0x7c00 |
| .file "fill.s" |
| .text |
| .globl start |
| .type start, @function |
| start: # at 0x7c00 ? |
| cli |
| lgdt gdtdesc |
| mov $1,%eax |
| mov %eax,%cr0 # Protected mode enable |
| data32 ljmp $8,$0x7c20 |
| |
| .org 0x7c20 |
| .code32 |
| # A20 enable - not sure I actually need this |
| inb $0x92,%al |
| or $2,%al |
| outb %al, $0x92 |
| |
| # set up DS for the whole of RAM (needed on KVM) |
| mov $16,%eax |
| mov %eax,%ds |
| |
| # Start from 1MB |
| .set TEST_MEM_START, X86_TEST_MEM_START |
| .set TEST_MEM_END, X86_TEST_MEM_END |
| |
| mov $65,%ax |
| mov $0x3f8,%dx |
| outb %al,%dx |
| |
| # bl keeps a counter so we limit the output speed |
| mov $0, %bl |
| |
| pre_zero: |
| mov $TEST_MEM_START,%eax |
| do_zero: |
| movb $0, (%eax) |
| add $4096,%eax |
| cmp $TEST_MEM_END,%eax |
| jl do_zero |
| |
| mainloop: |
| mov $TEST_MEM_START,%eax |
| innerloop: |
| incb (%eax) |
| add $4096,%eax |
| cmp $TEST_MEM_END,%eax |
| jl innerloop |
| |
| inc %bl |
| andb $0x3f,%bl |
| jnz mainloop |
| |
| mov $66,%ax |
| mov $0x3f8,%dx |
| outb %al,%dx |
| |
| # should this test suspend? |
| mov (suspend_me),%eax |
| cmp $0,%eax |
| je mainloop |
| |
| # are we waking after suspend? do not suspend again. |
| mov $suspended,%eax |
| mov (%eax),%eax |
| cmp $1,%eax |
| je mainloop |
| |
| # enable acpi |
| mov $ACPI_ENABLE,%al |
| outb %al,$ACPI_PORT_SMI_CMD |
| |
| # suspend to ram |
| mov $suspended,%eax |
| movl $1,(%eax) |
| mov $SLEEP,%ax |
| mov $(ACPI_PM_BASE + PM1A_CNT_OFFSET),%dx |
| outw %ax,%dx |
| # not reached. The wakeup causes reset and restart at 0x7c00, and we |
| # do not save and restore registers as a real kernel would do. |
| |
| |
| # GDT magic from old (GPLv2) Grub startup.S |
| .p2align 2 /* force 4-byte alignment */ |
| gdt: |
| .word 0, 0 |
| .byte 0, 0, 0, 0 |
| |
| /* -- code segment -- |
| * base = 0x00000000, limit = 0xFFFFF (4 KiB Granularity), present |
| * type = 32bit code execute/read, DPL = 0 |
| */ |
| .word 0xFFFF, 0 |
| .byte 0, 0x9A, 0xCF, 0 |
| |
| /* -- data segment -- |
| * base = 0x00000000, limit 0xFFFFF (4 KiB Granularity), present |
| * type = 32 bit data read/write, DPL = 0 |
| */ |
| .word 0xFFFF, 0 |
| .byte 0, 0x92, 0xCF, 0 |
| |
| gdtdesc: |
| .word 0x27 /* limit */ |
| .long gdt /* addr */ |
| |
| /* test launcher can poke a 1 here to exercise suspend */ |
| suspend_me: |
| .int 0 |
| |
| /* I'm a bootable disk */ |
| .org 0x7dfe |
| .byte 0x55 |
| .byte 0xAA |