1
2
3
4
5
6
7
8
9
10
11
12
13#include <linux/errno.h>
14#include <linux/kernel.h>
15#include <linux/kernel_stat.h>
16#include <linux/module.h>
17#include <linux/sched.h>
18#include <linux/mm.h>
19#include <linux/err.h>
20#include <linux/threads.h>
21#include <linux/smp.h>
22#include <linux/interrupt.h>
23#include <linux/init.h>
24#include <linux/delay.h>
25#include <linux/spinlock.h>
26#include <linux/irq.h>
27#include <linux/cache.h>
28#include <linux/profile.h>
29#include <linux/bitops.h>
30#include <linux/cpu.h>
31
32#include <asm/hwrpb.h>
33#include <asm/ptrace.h>
34#include <linux/atomic.h>
35
36#include <asm/io.h>
37#include <asm/irq.h>
38#include <asm/pgtable.h>
39#include <asm/pgalloc.h>
40#include <asm/mmu_context.h>
41#include <asm/tlbflush.h>
42
43#include "proto.h"
44#include "irq_impl.h"
45
46
47#define DEBUG_SMP 0
48#if DEBUG_SMP
49#define DBGS(args) printk args
50#else
51#define DBGS(args)
52#endif
53
54
55struct cpuinfo_alpha cpu_data[NR_CPUS];
56EXPORT_SYMBOL(cpu_data);
57
58
59static struct {
60 unsigned long bits ____cacheline_aligned;
61} ipi_data[NR_CPUS] __cacheline_aligned;
62
63enum ipi_message_type {
64 IPI_RESCHEDULE,
65 IPI_CALL_FUNC,
66 IPI_CPU_STOP,
67};
68
69
70static int smp_secondary_alive = 0;
71
72int smp_num_probed;
73int smp_num_cpus = 1;
74EXPORT_SYMBOL(smp_num_cpus);
75
76
77
78
79
80static inline void __init
81smp_store_cpu_info(int cpuid)
82{
83 cpu_data[cpuid].loops_per_jiffy = loops_per_jiffy;
84 cpu_data[cpuid].last_asn = ASN_FIRST_VERSION;
85 cpu_data[cpuid].need_new_asn = 0;
86 cpu_data[cpuid].asn_lock = 0;
87}
88
89
90
91
92static inline void __init
93smp_setup_percpu_timer(int cpuid)
94{
95 cpu_data[cpuid].prof_counter = 1;
96 cpu_data[cpuid].prof_multiplier = 1;
97}
98
99static void __init
100wait_boot_cpu_to_stop(int cpuid)
101{
102 unsigned long stop = jiffies + 10*HZ;
103
104 while (time_before(jiffies, stop)) {
105 if (!smp_secondary_alive)
106 return;
107 barrier();
108 }
109
110 printk("wait_boot_cpu_to_stop: FAILED on CPU %d, hanging now\n", cpuid);
111 for (;;)
112 barrier();
113}
114
115
116
117
118void
119smp_callin(void)
120{
121 int cpuid = hard_smp_processor_id();
122
123 if (cpu_online(cpuid)) {
124 printk("??, cpu 0x%x already present??\n", cpuid);
125 BUG();
126 }
127 set_cpu_online(cpuid, true);
128
129
130 wrmces(7);
131
132
133 trap_init();
134
135
136 wrent(entInt, 0);
137
138
139 smp_setup_percpu_timer(cpuid);
140 init_clockevent();
141
142
143 if (alpha_mv.smp_callin)
144 alpha_mv.smp_callin();
145
146
147 atomic_inc(&init_mm.mm_count);
148 current->active_mm = &init_mm;
149
150
151 notify_cpu_starting(cpuid);
152
153
154 local_irq_enable();
155
156
157
158 wait_boot_cpu_to_stop(cpuid);
159 mb();
160 calibrate_delay();
161
162 smp_store_cpu_info(cpuid);
163
164 wmb();
165 smp_secondary_alive = 1;
166
167 DBGS(("smp_callin: commencing CPU %d current %p active_mm %p\n",
168 cpuid, current, current->active_mm));
169
170 preempt_disable();
171 cpu_startup_entry(CPUHP_AP_ONLINE_IDLE);
172}
173
174
175static int
176wait_for_txrdy (unsigned long cpumask)
177{
178 unsigned long timeout;
179
180 if (!(hwrpb->txrdy & cpumask))
181 return 0;
182
183 timeout = jiffies + 10*HZ;
184 while (time_before(jiffies, timeout)) {
185 if (!(hwrpb->txrdy & cpumask))
186 return 0;
187 udelay(10);
188 barrier();
189 }
190
191 return -1;
192}
193
194
195
196
197
198static void
199send_secondary_console_msg(char *str, int cpuid)
200{
201 struct percpu_struct *cpu;
202 register char *cp1, *cp2;
203 unsigned long cpumask;
204 size_t len;
205
206 cpu = (struct percpu_struct *)
207 ((char*)hwrpb
208 + hwrpb->processor_offset
209 + cpuid * hwrpb->processor_size);
210
211 cpumask = (1UL << cpuid);
212 if (wait_for_txrdy(cpumask))
213 goto timeout;
214
215 cp2 = str;
216 len = strlen(cp2);
217 *(unsigned int *)&cpu->ipc_buffer[0] = len;
218 cp1 = (char *) &cpu->ipc_buffer[1];
219 memcpy(cp1, cp2, len);
220
221
222 wmb();
223 set_bit(cpuid, &hwrpb->rxrdy);
224
225 if (wait_for_txrdy(cpumask))
226 goto timeout;
227 return;
228
229 timeout:
230 printk("Processor %x not ready\n", cpuid);
231}
232
233
234
235
236static void
237recv_secondary_console_msg(void)
238{
239 int mycpu, i, cnt;
240 unsigned long txrdy = hwrpb->txrdy;
241 char *cp1, *cp2, buf[80];
242 struct percpu_struct *cpu;
243
244 DBGS(("recv_secondary_console_msg: TXRDY 0x%lx.\n", txrdy));
245
246 mycpu = hard_smp_processor_id();
247
248 for (i = 0; i < NR_CPUS; i++) {
249 if (!(txrdy & (1UL << i)))
250 continue;
251
252 DBGS(("recv_secondary_console_msg: "
253 "TXRDY contains CPU %d.\n", i));
254
255 cpu = (struct percpu_struct *)
256 ((char*)hwrpb
257 + hwrpb->processor_offset
258 + i * hwrpb->processor_size);
259
260 DBGS(("recv_secondary_console_msg: on %d from %d"
261 " HALT_REASON 0x%lx FLAGS 0x%lx\n",
262 mycpu, i, cpu->halt_reason, cpu->flags));
263
264 cnt = cpu->ipc_buffer[0] >> 32;
265 if (cnt <= 0 || cnt >= 80)
266 strcpy(buf, "<<< BOGUS MSG >>>");
267 else {
268 cp1 = (char *) &cpu->ipc_buffer[1];
269 cp2 = buf;
270 memcpy(cp2, cp1, cnt);
271 cp2[cnt] = '\0';
272
273 while ((cp2 = strchr(cp2, '\r')) != 0) {
274 *cp2 = ' ';
275 if (cp2[1] == '\n')
276 cp2[1] = ' ';
277 }
278 }
279
280 DBGS((KERN_INFO "recv_secondary_console_msg: on %d "
281 "message is '%s'\n", mycpu, buf));
282 }
283
284 hwrpb->txrdy = 0;
285}
286
287
288
289
290static int
291secondary_cpu_start(int cpuid, struct task_struct *idle)
292{
293 struct percpu_struct *cpu;
294 struct pcb_struct *hwpcb, *ipcb;
295 unsigned long timeout;
296
297 cpu = (struct percpu_struct *)
298 ((char*)hwrpb
299 + hwrpb->processor_offset
300 + cpuid * hwrpb->processor_size);
301 hwpcb = (struct pcb_struct *) cpu->hwpcb;
302 ipcb = &task_thread_info(idle)->pcb;
303
304
305
306
307
308 hwpcb->ksp = (unsigned long)ipcb + sizeof(union thread_union) - 16;
309 hwpcb->usp = 0;
310 hwpcb->ptbr = ipcb->ptbr;
311 hwpcb->pcc = 0;
312 hwpcb->asn = 0;
313 hwpcb->unique = virt_to_phys(ipcb);
314 hwpcb->flags = ipcb->flags;
315 hwpcb->res1 = hwpcb->res2 = 0;
316
317#if 0
318 DBGS(("KSP 0x%lx PTBR 0x%lx VPTBR 0x%lx UNIQUE 0x%lx\n",
319 hwpcb->ksp, hwpcb->ptbr, hwrpb->vptb, hwpcb->unique));
320#endif
321 DBGS(("Starting secondary cpu %d: state 0x%lx pal_flags 0x%lx\n",
322 cpuid, idle->state, ipcb->flags));
323
324
325 hwrpb->CPU_restart = __smp_callin;
326 hwrpb->CPU_restart_data = (unsigned long) __smp_callin;
327
328
329 hwrpb_update_checksum(hwrpb);
330
331
332
333
334
335
336 cpu->flags |= 0x22;
337 cpu->flags &= ~1;
338 wmb();
339
340 send_secondary_console_msg("START\r\n", cpuid);
341
342
343 timeout = jiffies + 10*HZ;
344 while (time_before(jiffies, timeout)) {
345 if (cpu->flags & 1)
346 goto started;
347 udelay(10);
348 barrier();
349 }
350 printk(KERN_ERR "SMP: Processor %d failed to start.\n", cpuid);
351 return -1;
352
353 started:
354 DBGS(("secondary_cpu_start: SUCCESS for CPU %d!!!\n", cpuid));
355 return 0;
356}
357
358
359
360
361static int
362smp_boot_one_cpu(int cpuid, struct task_struct *idle)
363{
364 unsigned long timeout;
365
366
367 smp_secondary_alive = -1;
368
369
370 if (secondary_cpu_start(cpuid, idle))
371 return -1;
372
373
374 mb();
375 smp_secondary_alive = 0;
376
377
378
379 timeout = jiffies + 1*HZ;
380 while (time_before(jiffies, timeout)) {
381 if (smp_secondary_alive == 1)
382 goto alive;
383 udelay(10);
384 barrier();
385 }
386
387
388
389 printk(KERN_ERR "SMP: Processor %d is stuck.\n", cpuid);
390 return -1;
391
392 alive:
393
394 return 0;
395}
396
397
398
399
400
401void __init
402setup_smp(void)
403{
404 struct percpu_struct *cpubase, *cpu;
405 unsigned long i;
406
407 if (boot_cpuid != 0) {
408 printk(KERN_WARNING "SMP: Booting off cpu %d instead of 0?\n",
409 boot_cpuid);
410 }
411
412 if (hwrpb->nr_processors > 1) {
413 int boot_cpu_palrev;
414
415 DBGS(("setup_smp: nr_processors %ld\n",
416 hwrpb->nr_processors));
417
418 cpubase = (struct percpu_struct *)
419 ((char*)hwrpb + hwrpb->processor_offset);
420 boot_cpu_palrev = cpubase->pal_revision;
421
422 for (i = 0; i < hwrpb->nr_processors; i++) {
423 cpu = (struct percpu_struct *)
424 ((char *)cpubase + i*hwrpb->processor_size);
425 if ((cpu->flags & 0x1cc) == 0x1cc) {
426 smp_num_probed++;
427 set_cpu_possible(i, true);
428 set_cpu_present(i, true);
429 cpu->pal_revision = boot_cpu_palrev;
430 }
431
432 DBGS(("setup_smp: CPU %d: flags 0x%lx type 0x%lx\n",
433 i, cpu->flags, cpu->type));
434 DBGS(("setup_smp: CPU %d: PAL rev 0x%lx\n",
435 i, cpu->pal_revision));
436 }
437 } else {
438 smp_num_probed = 1;
439 }
440
441 printk(KERN_INFO "SMP: %d CPUs probed -- cpu_present_mask = %lx\n",
442 smp_num_probed, cpumask_bits(cpu_present_mask)[0]);
443}
444
445
446
447
448void __init
449smp_prepare_cpus(unsigned int max_cpus)
450{
451
452 memset(ipi_data, 0, sizeof(ipi_data));
453
454 current_thread_info()->cpu = boot_cpuid;
455
456 smp_store_cpu_info(boot_cpuid);
457 smp_setup_percpu_timer(boot_cpuid);
458
459
460 if (smp_num_probed == 1 || max_cpus == 0) {
461 init_cpu_possible(cpumask_of(boot_cpuid));
462 init_cpu_present(cpumask_of(boot_cpuid));
463 printk(KERN_INFO "SMP mode deactivated.\n");
464 return;
465 }
466
467 printk(KERN_INFO "SMP starting up secondaries.\n");
468
469 smp_num_cpus = smp_num_probed;
470}
471
472void
473smp_prepare_boot_cpu(void)
474{
475}
476
477int
478__cpu_up(unsigned int cpu, struct task_struct *tidle)
479{
480 smp_boot_one_cpu(cpu, tidle);
481
482 return cpu_online(cpu) ? 0 : -ENOSYS;
483}
484
485void __init
486smp_cpus_done(unsigned int max_cpus)
487{
488 int cpu;
489 unsigned long bogosum = 0;
490
491 for(cpu = 0; cpu < NR_CPUS; cpu++)
492 if (cpu_online(cpu))
493 bogosum += cpu_data[cpu].loops_per_jiffy;
494
495 printk(KERN_INFO "SMP: Total of %d processors activated "
496 "(%lu.%02lu BogoMIPS).\n",
497 num_online_cpus(),
498 (bogosum + 2500) / (500000/HZ),
499 ((bogosum + 2500) / (5000/HZ)) % 100);
500}
501
502int
503setup_profiling_timer(unsigned int multiplier)
504{
505 return -EINVAL;
506}
507
508static void
509send_ipi_message(const struct cpumask *to_whom, enum ipi_message_type operation)
510{
511 int i;
512
513 mb();
514 for_each_cpu(i, to_whom)
515 set_bit(operation, &ipi_data[i].bits);
516
517 mb();
518 for_each_cpu(i, to_whom)
519 wripir(i);
520}
521
522void
523handle_ipi(struct pt_regs *regs)
524{
525 int this_cpu = smp_processor_id();
526 unsigned long *pending_ipis = &ipi_data[this_cpu].bits;
527 unsigned long ops;
528
529#if 0
530 DBGS(("handle_ipi: on CPU %d ops 0x%lx PC 0x%lx\n",
531 this_cpu, *pending_ipis, regs->pc));
532#endif
533
534 mb();
535 while ((ops = xchg(pending_ipis, 0)) != 0) {
536 mb();
537 do {
538 unsigned long which;
539
540 which = ops & -ops;
541 ops &= ~which;
542 which = __ffs(which);
543
544 switch (which) {
545 case IPI_RESCHEDULE:
546 scheduler_ipi();
547 break;
548
549 case IPI_CALL_FUNC:
550 generic_smp_call_function_interrupt();
551 break;
552
553 case IPI_CPU_STOP:
554 halt();
555
556 default:
557 printk(KERN_CRIT "Unknown IPI on CPU %d: %lu\n",
558 this_cpu, which);
559 break;
560 }
561 } while (ops);
562
563 mb();
564 }
565
566 cpu_data[this_cpu].ipi_count++;
567
568 if (hwrpb->txrdy)
569 recv_secondary_console_msg();
570}
571
572void
573smp_send_reschedule(int cpu)
574{
575#ifdef DEBUG_IPI_MSG
576 if (cpu == hard_smp_processor_id())
577 printk(KERN_WARNING
578 "smp_send_reschedule: Sending IPI to self.\n");
579#endif
580 send_ipi_message(cpumask_of(cpu), IPI_RESCHEDULE);
581}
582
583void
584smp_send_stop(void)
585{
586 cpumask_t to_whom;
587 cpumask_copy(&to_whom, cpu_possible_mask);
588 cpumask_clear_cpu(smp_processor_id(), &to_whom);
589#ifdef DEBUG_IPI_MSG
590 if (hard_smp_processor_id() != boot_cpu_id)
591 printk(KERN_WARNING "smp_send_stop: Not on boot cpu.\n");
592#endif
593 send_ipi_message(&to_whom, IPI_CPU_STOP);
594}
595
596void arch_send_call_function_ipi_mask(const struct cpumask *mask)
597{
598 send_ipi_message(mask, IPI_CALL_FUNC);
599}
600
601void arch_send_call_function_single_ipi(int cpu)
602{
603 send_ipi_message(cpumask_of(cpu), IPI_CALL_FUNC);
604}
605
606static void
607ipi_imb(void *ignored)
608{
609 imb();
610}
611
612void
613smp_imb(void)
614{
615
616 if (on_each_cpu(ipi_imb, NULL, 1))
617 printk(KERN_CRIT "smp_imb: timed out\n");
618}
619EXPORT_SYMBOL(smp_imb);
620
621static void
622ipi_flush_tlb_all(void *ignored)
623{
624 tbia();
625}
626
627void
628flush_tlb_all(void)
629{
630
631
632 if (on_each_cpu(ipi_flush_tlb_all, NULL, 1)) {
633 printk(KERN_CRIT "flush_tlb_all: timed out\n");
634 }
635}
636
637#define asn_locked() (cpu_data[smp_processor_id()].asn_lock)
638
639static void
640ipi_flush_tlb_mm(void *x)
641{
642 struct mm_struct *mm = (struct mm_struct *) x;
643 if (mm == current->active_mm && !asn_locked())
644 flush_tlb_current(mm);
645 else
646 flush_tlb_other(mm);
647}
648
649void
650flush_tlb_mm(struct mm_struct *mm)
651{
652 preempt_disable();
653
654 if (mm == current->active_mm) {
655 flush_tlb_current(mm);
656 if (atomic_read(&mm->mm_users) <= 1) {
657 int cpu, this_cpu = smp_processor_id();
658 for (cpu = 0; cpu < NR_CPUS; cpu++) {
659 if (!cpu_online(cpu) || cpu == this_cpu)
660 continue;
661 if (mm->context[cpu])
662 mm->context[cpu] = 0;
663 }
664 preempt_enable();
665 return;
666 }
667 }
668
669 if (smp_call_function(ipi_flush_tlb_mm, mm, 1)) {
670 printk(KERN_CRIT "flush_tlb_mm: timed out\n");
671 }
672
673 preempt_enable();
674}
675EXPORT_SYMBOL(flush_tlb_mm);
676
677struct flush_tlb_page_struct {
678 struct vm_area_struct *vma;
679 struct mm_struct *mm;
680 unsigned long addr;
681};
682
683static void
684ipi_flush_tlb_page(void *x)
685{
686 struct flush_tlb_page_struct *data = (struct flush_tlb_page_struct *)x;
687 struct mm_struct * mm = data->mm;
688
689 if (mm == current->active_mm && !asn_locked())
690 flush_tlb_current_page(mm, data->vma, data->addr);
691 else
692 flush_tlb_other(mm);
693}
694
695void
696flush_tlb_page(struct vm_area_struct *vma, unsigned long addr)
697{
698 struct flush_tlb_page_struct data;
699 struct mm_struct *mm = vma->vm_mm;
700
701 preempt_disable();
702
703 if (mm == current->active_mm) {
704 flush_tlb_current_page(mm, vma, addr);
705 if (atomic_read(&mm->mm_users) <= 1) {
706 int cpu, this_cpu = smp_processor_id();
707 for (cpu = 0; cpu < NR_CPUS; cpu++) {
708 if (!cpu_online(cpu) || cpu == this_cpu)
709 continue;
710 if (mm->context[cpu])
711 mm->context[cpu] = 0;
712 }
713 preempt_enable();
714 return;
715 }
716 }
717
718 data.vma = vma;
719 data.mm = mm;
720 data.addr = addr;
721
722 if (smp_call_function(ipi_flush_tlb_page, &data, 1)) {
723 printk(KERN_CRIT "flush_tlb_page: timed out\n");
724 }
725
726 preempt_enable();
727}
728EXPORT_SYMBOL(flush_tlb_page);
729
730void
731flush_tlb_range(struct vm_area_struct *vma, unsigned long start, unsigned long end)
732{
733
734 flush_tlb_mm(vma->vm_mm);
735}
736EXPORT_SYMBOL(flush_tlb_range);
737
738static void
739ipi_flush_icache_page(void *x)
740{
741 struct mm_struct *mm = (struct mm_struct *) x;
742 if (mm == current->active_mm && !asn_locked())
743 __load_new_mm_context(mm);
744 else
745 flush_tlb_other(mm);
746}
747
748void
749flush_icache_user_range(struct vm_area_struct *vma, struct page *page,
750 unsigned long addr, int len)
751{
752 struct mm_struct *mm = vma->vm_mm;
753
754 if ((vma->vm_flags & VM_EXEC) == 0)
755 return;
756
757 preempt_disable();
758
759 if (mm == current->active_mm) {
760 __load_new_mm_context(mm);
761 if (atomic_read(&mm->mm_users) <= 1) {
762 int cpu, this_cpu = smp_processor_id();
763 for (cpu = 0; cpu < NR_CPUS; cpu++) {
764 if (!cpu_online(cpu) || cpu == this_cpu)
765 continue;
766 if (mm->context[cpu])
767 mm->context[cpu] = 0;
768 }
769 preempt_enable();
770 return;
771 }
772 }
773
774 if (smp_call_function(ipi_flush_icache_page, mm, 1)) {
775 printk(KERN_CRIT "flush_icache_page: timed out\n");
776 }
777
778 preempt_enable();
779}
780