head64.c 2.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
/*
 *  linux/arch/x86_64/kernel/head64.c -- prepare to run common code
 *
 *  Copyright (C) 2000 Andrea Arcangeli <andrea@suse.de> SuSE
 */

#include <linux/init.h>
#include <linux/linkage.h>
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/string.h>
#include <linux/percpu.h>

#include <asm/processor.h>
#include <asm/proto.h>
#include <asm/smp.h>
#include <asm/bootsetup.h>
#include <asm/setup.h>
#include <asm/desc.h>
20
#include <asm/pgtable.h>
21
#include <asm/tlbflush.h>
22
#include <asm/sections.h>
L
Linus Torvalds 已提交
23

24 25 26 27 28 29 30
static void __init zap_identity_mappings(void)
{
	pgd_t *pgd = pgd_offset_k(0UL);
	pgd_clear(pgd);
	__flush_tlb();
}

L
Linus Torvalds 已提交
31 32 33 34 35
/* Don't add a printk in there. printk relies on the PDA which is not initialized 
   yet. */
static void __init clear_bss(void)
{
	memset(__bss_start, 0,
36
	       (unsigned long) __bss_stop - (unsigned long) __bss_start);
L
Linus Torvalds 已提交
37 38 39
}

#define NEW_CL_POINTER		0x228	/* Relative to real mode data */
40
#define OLD_CL_MAGIC_ADDR	0x20
L
Linus Torvalds 已提交
41
#define OLD_CL_MAGIC            0xA33F
42
#define OLD_CL_OFFSET           0x22
L
Linus Torvalds 已提交
43 44 45

static void __init copy_bootdata(char *real_mode_data)
{
46
	unsigned long new_data;
L
Linus Torvalds 已提交
47 48
	char * command_line;

49
	memcpy(x86_boot_params, real_mode_data, BOOT_PARAM_SIZE);
50
	new_data = *(u32 *) (x86_boot_params + NEW_CL_POINTER);
L
Linus Torvalds 已提交
51
	if (!new_data) {
52
		if (OLD_CL_MAGIC != *(u16 *)(real_mode_data + OLD_CL_MAGIC_ADDR)) {
L
Linus Torvalds 已提交
53 54
			return;
		}
55
		new_data = __pa(real_mode_data) + *(u16 *)(real_mode_data + OLD_CL_OFFSET);
L
Linus Torvalds 已提交
56
	}
57
	command_line = __va(new_data);
58
	memcpy(boot_command_line, command_line, COMMAND_LINE_SIZE);
L
Linus Torvalds 已提交
59 60 61 62 63 64
}

void __init x86_64_start_kernel(char * real_mode_data)
{
	int i;

65 66 67
	/* clear bss before set_intr_gate with early_idt_handler */
	clear_bss();

68 69 70
	/* Make NULL pointers segfault */
	zap_identity_mappings();

71
	for (i = 0; i < IDT_ENTRIES; i++)
L
Linus Torvalds 已提交
72 73
		set_intr_gate(i, early_idt_handler);
	asm volatile("lidt %0" :: "m" (idt_descr));
74

75
	early_printk("Kernel alive\n");
I
Ingo Molnar 已提交
76

77 78 79
 	for (i = 0; i < NR_CPUS; i++)
 		cpu_pda(i) = &boot_cpu_pda[i];

L
Linus Torvalds 已提交
80
	pda_init(0);
81
	copy_bootdata(__va(real_mode_data));
L
Linus Torvalds 已提交
82 83 84 85 86
#ifdef CONFIG_SMP
	cpu_set(0, cpu_online_map);
#endif
	start_kernel();
}