diff options
Diffstat (limited to 'arch/i386/mach-voyager')
-rw-r--r-- | arch/i386/mach-voyager/setup.c | 8 | ||||
-rw-r--r-- | arch/i386/mach-voyager/voyager_cat.c | 4 | ||||
-rw-r--r-- | arch/i386/mach-voyager/voyager_smp.c | 97 | ||||
-rw-r--r-- | arch/i386/mach-voyager/voyager_thread.c | 69 |
4 files changed, 82 insertions, 96 deletions
diff --git a/arch/i386/mach-voyager/setup.c b/arch/i386/mach-voyager/setup.c index cfa16c151c8..447bb105cf5 100644 --- a/arch/i386/mach-voyager/setup.c +++ b/arch/i386/mach-voyager/setup.c | |||
@@ -40,10 +40,16 @@ void __init trap_init_hook(void) | |||
40 | { | 40 | { |
41 | } | 41 | } |
42 | 42 | ||
43 | static struct irqaction irq0 = { timer_interrupt, IRQF_DISABLED, CPU_MASK_NONE, "timer", NULL, NULL}; | 43 | static struct irqaction irq0 = { |
44 | .handler = timer_interrupt, | ||
45 | .flags = IRQF_DISABLED | IRQF_NOBALANCING, | ||
46 | .mask = CPU_MASK_NONE, | ||
47 | .name = "timer" | ||
48 | }; | ||
44 | 49 | ||
45 | void __init time_init_hook(void) | 50 | void __init time_init_hook(void) |
46 | { | 51 | { |
52 | irq0.mask = cpumask_of_cpu(safe_smp_processor_id()); | ||
47 | setup_irq(0, &irq0); | 53 | setup_irq(0, &irq0); |
48 | } | 54 | } |
49 | 55 | ||
diff --git a/arch/i386/mach-voyager/voyager_cat.c b/arch/i386/mach-voyager/voyager_cat.c index 943a9473b13..26a2d4c54b6 100644 --- a/arch/i386/mach-voyager/voyager_cat.c +++ b/arch/i386/mach-voyager/voyager_cat.c | |||
@@ -1111,7 +1111,7 @@ voyager_cat_do_common_interrupt(void) | |||
1111 | printk(KERN_ERR "Voyager front panel switch turned off\n"); | 1111 | printk(KERN_ERR "Voyager front panel switch turned off\n"); |
1112 | voyager_status.switch_off = 1; | 1112 | voyager_status.switch_off = 1; |
1113 | voyager_status.request_from_kernel = 1; | 1113 | voyager_status.request_from_kernel = 1; |
1114 | up(&kvoyagerd_sem); | 1114 | wake_up_process(voyager_thread); |
1115 | } | 1115 | } |
1116 | /* Tell the hardware we're taking care of the | 1116 | /* Tell the hardware we're taking care of the |
1117 | * shutdown, otherwise it will power the box off | 1117 | * shutdown, otherwise it will power the box off |
@@ -1157,7 +1157,7 @@ voyager_cat_do_common_interrupt(void) | |||
1157 | outb(VOYAGER_CAT_END, CAT_CMD); | 1157 | outb(VOYAGER_CAT_END, CAT_CMD); |
1158 | voyager_status.power_fail = 1; | 1158 | voyager_status.power_fail = 1; |
1159 | voyager_status.request_from_kernel = 1; | 1159 | voyager_status.request_from_kernel = 1; |
1160 | up(&kvoyagerd_sem); | 1160 | wake_up_process(voyager_thread); |
1161 | } | 1161 | } |
1162 | 1162 | ||
1163 | 1163 | ||
diff --git a/arch/i386/mach-voyager/voyager_smp.c b/arch/i386/mach-voyager/voyager_smp.c index b9ce33c0c20..1a5e448a29c 100644 --- a/arch/i386/mach-voyager/voyager_smp.c +++ b/arch/i386/mach-voyager/voyager_smp.c | |||
@@ -536,15 +536,6 @@ do_boot_cpu(__u8 cpu) | |||
536 | & ~( voyager_extended_vic_processors | 536 | & ~( voyager_extended_vic_processors |
537 | & voyager_allowed_boot_processors); | 537 | & voyager_allowed_boot_processors); |
538 | 538 | ||
539 | /* For the 486, we can't use the 4Mb page table trick, so | ||
540 | * must map a region of memory */ | ||
541 | #ifdef CONFIG_M486 | ||
542 | int i; | ||
543 | unsigned long *page_table_copies = (unsigned long *) | ||
544 | __get_free_page(GFP_KERNEL); | ||
545 | #endif | ||
546 | pgd_t orig_swapper_pg_dir0; | ||
547 | |||
548 | /* This is an area in head.S which was used to set up the | 539 | /* This is an area in head.S which was used to set up the |
549 | * initial kernel stack. We need to alter this to give the | 540 | * initial kernel stack. We need to alter this to give the |
550 | * booting CPU a new stack (taken from its idle process) */ | 541 | * booting CPU a new stack (taken from its idle process) */ |
@@ -573,6 +564,8 @@ do_boot_cpu(__u8 cpu) | |||
573 | hijack_source.idt.Segment = (start_phys_address >> 4) & 0xFFFF; | 564 | hijack_source.idt.Segment = (start_phys_address >> 4) & 0xFFFF; |
574 | 565 | ||
575 | cpucount++; | 566 | cpucount++; |
567 | alternatives_smp_switch(1); | ||
568 | |||
576 | idle = fork_idle(cpu); | 569 | idle = fork_idle(cpu); |
577 | if(IS_ERR(idle)) | 570 | if(IS_ERR(idle)) |
578 | panic("failed fork for CPU%d", cpu); | 571 | panic("failed fork for CPU%d", cpu); |
@@ -587,24 +580,11 @@ do_boot_cpu(__u8 cpu) | |||
587 | VDEBUG(("VOYAGER SMP: Booting CPU%d at 0x%lx[%x:%x], stack %p\n", cpu, | 580 | VDEBUG(("VOYAGER SMP: Booting CPU%d at 0x%lx[%x:%x], stack %p\n", cpu, |
588 | (unsigned long)hijack_source.val, hijack_source.idt.Segment, | 581 | (unsigned long)hijack_source.val, hijack_source.idt.Segment, |
589 | hijack_source.idt.Offset, stack_start.esp)); | 582 | hijack_source.idt.Offset, stack_start.esp)); |
590 | /* set the original swapper_pg_dir[0] to map 0 to 4Mb transparently | 583 | |
591 | * (so that the booting CPU can find start_32 */ | 584 | /* init lowmem identity mapping */ |
592 | orig_swapper_pg_dir0 = swapper_pg_dir[0]; | 585 | clone_pgd_range(swapper_pg_dir, swapper_pg_dir + USER_PGD_PTRS, |
593 | #ifdef CONFIG_M486 | 586 | min_t(unsigned long, KERNEL_PGD_PTRS, USER_PGD_PTRS)); |
594 | if(page_table_copies == NULL) | 587 | flush_tlb_all(); |
595 | panic("No free memory for 486 page tables\n"); | ||
596 | for(i = 0; i < PAGE_SIZE/sizeof(unsigned long); i++) | ||
597 | page_table_copies[i] = (i * PAGE_SIZE) | ||
598 | | _PAGE_RW | _PAGE_USER | _PAGE_PRESENT; | ||
599 | |||
600 | ((unsigned long *)swapper_pg_dir)[0] = | ||
601 | ((virt_to_phys(page_table_copies)) & PAGE_MASK) | ||
602 | | _PAGE_RW | _PAGE_USER | _PAGE_PRESENT; | ||
603 | #else | ||
604 | ((unsigned long *)swapper_pg_dir)[0] = | ||
605 | (virt_to_phys(pg0) & PAGE_MASK) | ||
606 | | _PAGE_RW | _PAGE_USER | _PAGE_PRESENT; | ||
607 | #endif | ||
608 | 588 | ||
609 | if(quad_boot) { | 589 | if(quad_boot) { |
610 | printk("CPU %d: non extended Quad boot\n", cpu); | 590 | printk("CPU %d: non extended Quad boot\n", cpu); |
@@ -647,11 +627,7 @@ do_boot_cpu(__u8 cpu) | |||
647 | udelay(100); | 627 | udelay(100); |
648 | } | 628 | } |
649 | /* reset the page table */ | 629 | /* reset the page table */ |
650 | swapper_pg_dir[0] = orig_swapper_pg_dir0; | 630 | zap_low_mappings(); |
651 | local_flush_tlb(); | ||
652 | #ifdef CONFIG_M486 | ||
653 | free_page((unsigned long)page_table_copies); | ||
654 | #endif | ||
655 | 631 | ||
656 | if (cpu_booted_map) { | 632 | if (cpu_booted_map) { |
657 | VDEBUG(("CPU%d: Booted successfully, back in CPU %d\n", | 633 | VDEBUG(("CPU%d: Booted successfully, back in CPU %d\n", |
@@ -1068,20 +1044,11 @@ smp_call_function_interrupt(void) | |||
1068 | } | 1044 | } |
1069 | } | 1045 | } |
1070 | 1046 | ||
1071 | /* Call this function on all CPUs using the function_interrupt above | 1047 | static int |
1072 | <func> The function to run. This must be fast and non-blocking. | 1048 | __smp_call_function_mask (void (*func) (void *info), void *info, int retry, |
1073 | <info> An arbitrary pointer to pass to the function. | 1049 | int wait, __u32 mask) |
1074 | <retry> If true, keep retrying until ready. | ||
1075 | <wait> If true, wait until function has completed on other CPUs. | ||
1076 | [RETURNS] 0 on success, else a negative status code. Does not return until | ||
1077 | remote CPUs are nearly ready to execute <<func>> or are or have executed. | ||
1078 | */ | ||
1079 | int | ||
1080 | smp_call_function (void (*func) (void *info), void *info, int retry, | ||
1081 | int wait) | ||
1082 | { | 1050 | { |
1083 | struct call_data_struct data; | 1051 | struct call_data_struct data; |
1084 | __u32 mask = cpus_addr(cpu_online_map)[0]; | ||
1085 | 1052 | ||
1086 | mask &= ~(1<<smp_processor_id()); | 1053 | mask &= ~(1<<smp_processor_id()); |
1087 | 1054 | ||
@@ -1102,7 +1069,7 @@ smp_call_function (void (*func) (void *info), void *info, int retry, | |||
1102 | call_data = &data; | 1069 | call_data = &data; |
1103 | wmb(); | 1070 | wmb(); |
1104 | /* Send a message to all other CPUs and wait for them to respond */ | 1071 | /* Send a message to all other CPUs and wait for them to respond */ |
1105 | send_CPI_allbutself(VIC_CALL_FUNCTION_CPI); | 1072 | send_CPI(mask, VIC_CALL_FUNCTION_CPI); |
1106 | 1073 | ||
1107 | /* Wait for response */ | 1074 | /* Wait for response */ |
1108 | while (data.started) | 1075 | while (data.started) |
@@ -1116,8 +1083,48 @@ smp_call_function (void (*func) (void *info), void *info, int retry, | |||
1116 | 1083 | ||
1117 | return 0; | 1084 | return 0; |
1118 | } | 1085 | } |
1086 | |||
1087 | /* Call this function on all CPUs using the function_interrupt above | ||
1088 | <func> The function to run. This must be fast and non-blocking. | ||
1089 | <info> An arbitrary pointer to pass to the function. | ||
1090 | <retry> If true, keep retrying until ready. | ||
1091 | <wait> If true, wait until function has completed on other CPUs. | ||
1092 | [RETURNS] 0 on success, else a negative status code. Does not return until | ||
1093 | remote CPUs are nearly ready to execute <<func>> or are or have executed. | ||
1094 | */ | ||
1095 | int | ||
1096 | smp_call_function(void (*func) (void *info), void *info, int retry, | ||
1097 | int wait) | ||
1098 | { | ||
1099 | __u32 mask = cpus_addr(cpu_online_map)[0]; | ||
1100 | |||
1101 | return __smp_call_function_mask(func, info, retry, wait, mask); | ||
1102 | } | ||
1119 | EXPORT_SYMBOL(smp_call_function); | 1103 | EXPORT_SYMBOL(smp_call_function); |
1120 | 1104 | ||
1105 | /* | ||
1106 | * smp_call_function_single - Run a function on another CPU | ||
1107 | * @func: The function to run. This must be fast and non-blocking. | ||
1108 | * @info: An arbitrary pointer to pass to the function. | ||
1109 | * @nonatomic: Currently unused. | ||
1110 | * @wait: If true, wait until function has completed on other CPUs. | ||
1111 | * | ||
1112 | * Retrurns 0 on success, else a negative status code. | ||
1113 | * | ||
1114 | * Does not return until the remote CPU is nearly ready to execute <func> | ||
1115 | * or is or has executed. | ||
1116 | */ | ||
1117 | |||
1118 | int | ||
1119 | smp_call_function_single(int cpu, void (*func) (void *info), void *info, | ||
1120 | int nonatomic, int wait) | ||
1121 | { | ||
1122 | __u32 mask = 1 << cpu; | ||
1123 | |||
1124 | return __smp_call_function_mask(func, info, nonatomic, wait, mask); | ||
1125 | } | ||
1126 | EXPORT_SYMBOL(smp_call_function_single); | ||
1127 | |||
1121 | /* Sorry about the name. In an APIC based system, the APICs | 1128 | /* Sorry about the name. In an APIC based system, the APICs |
1122 | * themselves are programmed to send a timer interrupt. This is used | 1129 | * themselves are programmed to send a timer interrupt. This is used |
1123 | * by linux to reschedule the processor. Voyager doesn't have this, | 1130 | * by linux to reschedule the processor. Voyager doesn't have this, |
diff --git a/arch/i386/mach-voyager/voyager_thread.c b/arch/i386/mach-voyager/voyager_thread.c index f39887359e8..fdc1d926fb2 100644 --- a/arch/i386/mach-voyager/voyager_thread.c +++ b/arch/i386/mach-voyager/voyager_thread.c | |||
@@ -24,33 +24,16 @@ | |||
24 | #include <linux/kmod.h> | 24 | #include <linux/kmod.h> |
25 | #include <linux/completion.h> | 25 | #include <linux/completion.h> |
26 | #include <linux/sched.h> | 26 | #include <linux/sched.h> |
27 | #include <linux/kthread.h> | ||
27 | #include <asm/desc.h> | 28 | #include <asm/desc.h> |
28 | #include <asm/voyager.h> | 29 | #include <asm/voyager.h> |
29 | #include <asm/vic.h> | 30 | #include <asm/vic.h> |
30 | #include <asm/mtrr.h> | 31 | #include <asm/mtrr.h> |
31 | #include <asm/msr.h> | 32 | #include <asm/msr.h> |
32 | 33 | ||
33 | #define THREAD_NAME "kvoyagerd" | ||
34 | 34 | ||
35 | /* external variables */ | 35 | struct task_struct *voyager_thread; |
36 | int kvoyagerd_running = 0; | 36 | static __u8 set_timeout; |
37 | DECLARE_MUTEX_LOCKED(kvoyagerd_sem); | ||
38 | |||
39 | static int thread(void *); | ||
40 | |||
41 | static __u8 set_timeout = 0; | ||
42 | |||
43 | /* Start the machine monitor thread. Return 1 if OK, 0 if fail */ | ||
44 | static int __init | ||
45 | voyager_thread_start(void) | ||
46 | { | ||
47 | if(kernel_thread(thread, NULL, CLONE_KERNEL) < 0) { | ||
48 | /* This is serious, but not fatal */ | ||
49 | printk(KERN_ERR "Voyager: Failed to create system monitor thread!!!\n"); | ||
50 | return 1; | ||
51 | } | ||
52 | return 0; | ||
53 | } | ||
54 | 37 | ||
55 | static int | 38 | static int |
56 | execute(const char *string) | 39 | execute(const char *string) |
@@ -110,31 +93,15 @@ check_continuing_condition(void) | |||
110 | } | 93 | } |
111 | } | 94 | } |
112 | 95 | ||
113 | static void | ||
114 | wakeup(unsigned long unused) | ||
115 | { | ||
116 | up(&kvoyagerd_sem); | ||
117 | } | ||
118 | |||
119 | static int | 96 | static int |
120 | thread(void *unused) | 97 | thread(void *unused) |
121 | { | 98 | { |
122 | struct timer_list wakeup_timer; | ||
123 | |||
124 | kvoyagerd_running = 1; | ||
125 | |||
126 | daemonize(THREAD_NAME); | ||
127 | |||
128 | set_timeout = 0; | ||
129 | |||
130 | init_timer(&wakeup_timer); | ||
131 | |||
132 | sigfillset(¤t->blocked); | ||
133 | |||
134 | printk(KERN_NOTICE "Voyager starting monitor thread\n"); | 99 | printk(KERN_NOTICE "Voyager starting monitor thread\n"); |
135 | 100 | ||
136 | for(;;) { | 101 | for (;;) { |
137 | down_interruptible(&kvoyagerd_sem); | 102 | set_current_state(TASK_INTERRUPTIBLE); |
103 | schedule_timeout(set_timeout ? HZ : MAX_SCHEDULE_TIMEOUT); | ||
104 | |||
138 | VDEBUG(("Voyager Daemon awoken\n")); | 105 | VDEBUG(("Voyager Daemon awoken\n")); |
139 | if(voyager_status.request_from_kernel == 0) { | 106 | if(voyager_status.request_from_kernel == 0) { |
140 | /* probably awoken from timeout */ | 107 | /* probably awoken from timeout */ |
@@ -143,20 +110,26 @@ thread(void *unused) | |||
143 | check_from_kernel(); | 110 | check_from_kernel(); |
144 | voyager_status.request_from_kernel = 0; | 111 | voyager_status.request_from_kernel = 0; |
145 | } | 112 | } |
146 | if(set_timeout) { | ||
147 | del_timer(&wakeup_timer); | ||
148 | wakeup_timer.expires = HZ + jiffies; | ||
149 | wakeup_timer.function = wakeup; | ||
150 | add_timer(&wakeup_timer); | ||
151 | } | ||
152 | } | 113 | } |
153 | } | 114 | } |
154 | 115 | ||
116 | static int __init | ||
117 | voyager_thread_start(void) | ||
118 | { | ||
119 | voyager_thread = kthread_run(thread, NULL, "kvoyagerd"); | ||
120 | if (IS_ERR(voyager_thread)) { | ||
121 | printk(KERN_ERR "Voyager: Failed to create system monitor thread.\n"); | ||
122 | return PTR_ERR(voyager_thread); | ||
123 | } | ||
124 | return 0; | ||
125 | } | ||
126 | |||
127 | |||
155 | static void __exit | 128 | static void __exit |
156 | voyager_thread_stop(void) | 129 | voyager_thread_stop(void) |
157 | { | 130 | { |
158 | /* FIXME: do nothing at the moment */ | 131 | kthread_stop(voyager_thread); |
159 | } | 132 | } |
160 | 133 | ||
161 | module_init(voyager_thread_start); | 134 | module_init(voyager_thread_start); |
162 | //module_exit(voyager_thread_stop); | 135 | module_exit(voyager_thread_stop); |