2 * linux/arch/m32r/kernel/smp.c
4 * M32R SMP support routines.
6 * Copyright (c) 2001, 2002 Hitoshi Yamamoto
8 * Taken from i386 version.
9 * (c) 1995 Alan Cox, Building #3 <alan@redhat.com>
10 * (c) 1998-99, 2000 Ingo Molnar <mingo@redhat.com>
12 * This code is released under the GNU General Public License version 2 or
18 #include <linux/irq.h>
19 #include <linux/interrupt.h>
20 #include <linux/sched.h>
21 #include <linux/spinlock.h>
23 #include <linux/smp.h>
24 #include <linux/profile.h>
25 #include <linux/cpu.h>
27 #include <asm/cacheflush.h>
28 #include <asm/pgalloc.h>
29 #include <linux/atomic.h>
31 #include <asm/mmu_context.h>
33 #include <asm/tlbflush.h>
35 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
36 /* Data structures and variables */
37 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
40 * For flush_cache_all()
42 static DEFINE_SPINLOCK(flushcache_lock
);
43 static volatile unsigned long flushcache_cpumask
= 0;
46 * For flush_tlb_others()
48 static cpumask_t flush_cpumask
;
49 static struct mm_struct
*flush_mm
;
50 static struct vm_area_struct
*flush_vma
;
51 static volatile unsigned long flush_va
;
52 static DEFINE_SPINLOCK(tlbstate_lock
);
53 #define FLUSH_ALL 0xffffffff
55 DECLARE_PER_CPU(int, prof_multiplier
);
56 DECLARE_PER_CPU(int, prof_old_multiplier
);
57 DECLARE_PER_CPU(int, prof_counter
);
59 extern spinlock_t ipi_lock
[];
61 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
62 /* Function Prototypes */
63 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
65 void smp_reschedule_interrupt(void);
66 void smp_flush_cache_all_interrupt(void);
68 static void flush_tlb_all_ipi(void *);
69 static void flush_tlb_others(cpumask_t
, struct mm_struct
*,
70 struct vm_area_struct
*, unsigned long);
72 void smp_invalidate_interrupt(void);
74 static void stop_this_cpu(void *);
76 void smp_ipi_timer_interrupt(struct pt_regs
*);
77 void smp_local_timer_interrupt(void);
79 static void send_IPI_allbutself(int, int);
80 static void send_IPI_mask(const struct cpumask
*, int, int);
82 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
83 /* Rescheduling request Routines */
84 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
86 /*==========================================================================*
87 * Name: smp_send_reschedule
89 * Description: This routine requests other CPU to execute rescheduling.
90 * 1.Send 'RESCHEDULE_IPI' to other CPU.
91 * Request other CPU to execute 'smp_reschedule_interrupt()'.
93 * Born on Date: 2002.02.05
95 * Arguments: cpu_id - Target CPU ID
97 * Returns: void (cannot fail)
100 * Date Who Description
101 * ---------- --- --------------------------------------------------------
103 *==========================================================================*/
104 void smp_send_reschedule(int cpu_id
)
106 WARN_ON(cpu_is_offline(cpu_id
));
107 send_IPI_mask(cpumask_of(cpu_id
), RESCHEDULE_IPI
, 1);
110 /*==========================================================================*
111 * Name: smp_reschedule_interrupt
113 * Description: This routine executes on CPU which received
116 * Born on Date: 2002.02.05
120 * Returns: void (cannot fail)
123 * Date Who Description
124 * ---------- --- --------------------------------------------------------
126 *==========================================================================*/
127 void smp_reschedule_interrupt(void)
132 /*==========================================================================*
133 * Name: smp_flush_cache_all
135 * Description: This routine sends a 'INVALIDATE_CACHE_IPI' to all other
136 * CPUs in the system.
138 * Born on Date: 2003-05-28
142 * Returns: void (cannot fail)
145 * Date Who Description
146 * ---------- --- --------------------------------------------------------
148 *==========================================================================*/
149 void smp_flush_cache_all(void)
155 cpumask_copy(&cpumask
, cpu_online_mask
);
156 cpumask_clear_cpu(smp_processor_id(), &cpumask
);
157 spin_lock(&flushcache_lock
);
158 mask
=cpumask_bits(&cpumask
);
159 atomic_set_mask(*mask
, (atomic_t
*)&flushcache_cpumask
);
160 send_IPI_mask(&cpumask
, INVALIDATE_CACHE_IPI
, 0);
161 _flush_cache_copyback_all();
162 while (flushcache_cpumask
)
164 spin_unlock(&flushcache_lock
);
168 void smp_flush_cache_all_interrupt(void)
170 _flush_cache_copyback_all();
171 clear_bit(smp_processor_id(), &flushcache_cpumask
);
174 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
175 /* TLB flush request Routines */
176 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
178 /*==========================================================================*
179 * Name: smp_flush_tlb_all
181 * Description: This routine flushes all processes TLBs.
182 * 1.Request other CPU to execute 'flush_tlb_all_ipi()'.
183 * 2.Execute 'do_flush_tlb_all_local()'.
185 * Born on Date: 2002.02.05
189 * Returns: void (cannot fail)
192 * Date Who Description
193 * ---------- --- --------------------------------------------------------
195 *==========================================================================*/
196 void smp_flush_tlb_all(void)
201 local_irq_save(flags
);
203 local_irq_restore(flags
);
204 smp_call_function(flush_tlb_all_ipi
, NULL
, 1);
208 /*==========================================================================*
209 * Name: flush_tlb_all_ipi
211 * Description: This routine flushes all local TLBs.
212 * 1.Execute 'do_flush_tlb_all_local()'.
214 * Born on Date: 2002.02.05
216 * Arguments: *info - not used
218 * Returns: void (cannot fail)
221 * Date Who Description
222 * ---------- --- --------------------------------------------------------
224 *==========================================================================*/
225 static void flush_tlb_all_ipi(void *info
)
230 /*==========================================================================*
231 * Name: smp_flush_tlb_mm
233 * Description: This routine flushes the specified mm context TLB's.
235 * Born on Date: 2002.02.05
237 * Arguments: *mm - a pointer to the mm struct for flush TLB
239 * Returns: void (cannot fail)
242 * Date Who Description
243 * ---------- --- --------------------------------------------------------
245 *==========================================================================*/
246 void smp_flush_tlb_mm(struct mm_struct
*mm
)
254 cpu_id
= smp_processor_id();
255 mmc
= &mm
->context
[cpu_id
];
256 cpumask_copy(&cpu_mask
, mm_cpumask(mm
));
257 cpumask_clear_cpu(cpu_id
, &cpu_mask
);
259 if (*mmc
!= NO_CONTEXT
) {
260 local_irq_save(flags
);
262 if (mm
== current
->mm
)
263 activate_context(mm
);
265 cpumask_clear_cpu(cpu_id
, mm_cpumask(mm
));
266 local_irq_restore(flags
);
268 if (!cpumask_empty(&cpu_mask
))
269 flush_tlb_others(cpu_mask
, mm
, NULL
, FLUSH_ALL
);
274 /*==========================================================================*
275 * Name: smp_flush_tlb_range
277 * Description: This routine flushes a range of pages.
279 * Born on Date: 2002.02.05
281 * Arguments: *mm - a pointer to the mm struct for flush TLB
285 * Returns: void (cannot fail)
288 * Date Who Description
289 * ---------- --- --------------------------------------------------------
291 *==========================================================================*/
292 void smp_flush_tlb_range(struct vm_area_struct
*vma
, unsigned long start
,
295 smp_flush_tlb_mm(vma
->vm_mm
);
298 /*==========================================================================*
299 * Name: smp_flush_tlb_page
301 * Description: This routine flushes one page.
303 * Born on Date: 2002.02.05
305 * Arguments: *vma - a pointer to the vma struct include va
306 * va - virtual address for flush TLB
308 * Returns: void (cannot fail)
311 * Date Who Description
312 * ---------- --- --------------------------------------------------------
314 *==========================================================================*/
315 void smp_flush_tlb_page(struct vm_area_struct
*vma
, unsigned long va
)
317 struct mm_struct
*mm
= vma
->vm_mm
;
324 cpu_id
= smp_processor_id();
325 mmc
= &mm
->context
[cpu_id
];
326 cpumask_copy(&cpu_mask
, mm_cpumask(mm
));
327 cpumask_clear_cpu(cpu_id
, &cpu_mask
);
334 if (*mmc
!= NO_CONTEXT
) {
335 local_irq_save(flags
);
337 va
|= (*mmc
& MMU_CONTEXT_ASID_MASK
);
338 __flush_tlb_page(va
);
339 local_irq_restore(flags
);
341 if (!cpumask_empty(&cpu_mask
))
342 flush_tlb_others(cpu_mask
, mm
, vma
, va
);
347 /*==========================================================================*
348 * Name: flush_tlb_others
350 * Description: This routine requests other CPU to execute flush TLB.
351 * 1.Setup parameters.
352 * 2.Send 'INVALIDATE_TLB_IPI' to other CPU.
353 * Request other CPU to execute 'smp_invalidate_interrupt()'.
354 * 3.Wait for other CPUs operation finished.
356 * Born on Date: 2002.02.05
358 * Arguments: cpumask - bitmap of target CPUs
359 * *mm - a pointer to the mm struct for flush TLB
360 * *vma - a pointer to the vma struct include va
361 * va - virtual address for flush TLB
363 * Returns: void (cannot fail)
366 * Date Who Description
367 * ---------- --- --------------------------------------------------------
369 *==========================================================================*/
370 static void flush_tlb_others(cpumask_t cpumask
, struct mm_struct
*mm
,
371 struct vm_area_struct
*vma
, unsigned long va
)
377 if (!(flags
& 0x0040)) /* Interrupt Disable NONONO */
379 #endif /* DEBUG_SMP */
382 * A couple of (to be removed) sanity checks:
384 * - we do not send IPIs to not-yet booted CPUs.
385 * - current CPU must not be in mask
386 * - mask must exist :)
388 BUG_ON(cpumask_empty(&cpumask
));
390 BUG_ON(cpumask_test_cpu(smp_processor_id(), &cpumask
));
393 /* If a CPU which we ran on has gone down, OK. */
394 cpumask_and(&cpumask
, &cpumask
, cpu_online_mask
);
395 if (cpumask_empty(&cpumask
))
399 * i'm not happy about this global shared spinlock in the
400 * MM hot path, but we'll see how contended it is.
401 * Temporarily this turns IRQs off, so that lockups are
402 * detected by the NMI watchdog.
404 spin_lock(&tlbstate_lock
);
409 mask
=cpumask_bits(&cpumask
);
410 atomic_set_mask(*mask
, (atomic_t
*)&flush_cpumask
);
413 * We have to send the IPI only to
416 send_IPI_mask(&cpumask
, INVALIDATE_TLB_IPI
, 0);
418 while (!cpumask_empty(&flush_cpumask
)) {
419 /* nothing. lockup detection does not belong here */
426 spin_unlock(&tlbstate_lock
);
429 /*==========================================================================*
430 * Name: smp_invalidate_interrupt
432 * Description: This routine executes on CPU which received
433 * 'INVALIDATE_TLB_IPI'.
435 * 2.Report flush TLB process was finished.
437 * Born on Date: 2002.02.05
441 * Returns: void (cannot fail)
444 * Date Who Description
445 * ---------- --- --------------------------------------------------------
447 *==========================================================================*/
448 void smp_invalidate_interrupt(void)
450 int cpu_id
= smp_processor_id();
451 unsigned long *mmc
= &flush_mm
->context
[cpu_id
];
453 if (!cpumask_test_cpu(cpu_id
, &flush_cpumask
))
456 if (flush_va
== FLUSH_ALL
) {
458 if (flush_mm
== current
->active_mm
)
459 activate_context(flush_mm
);
461 cpumask_clear_cpu(cpu_id
, mm_cpumask(flush_mm
));
463 unsigned long va
= flush_va
;
465 if (*mmc
!= NO_CONTEXT
) {
467 va
|= (*mmc
& MMU_CONTEXT_ASID_MASK
);
468 __flush_tlb_page(va
);
471 cpumask_clear_cpu(cpu_id
, &flush_cpumask
);
474 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
475 /* Stop CPU request Routines */
476 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
478 /*==========================================================================*
479 * Name: smp_send_stop
481 * Description: This routine requests stop all CPUs.
482 * 1.Request other CPU to execute 'stop_this_cpu()'.
484 * Born on Date: 2002.02.05
488 * Returns: void (cannot fail)
491 * Date Who Description
492 * ---------- --- --------------------------------------------------------
494 *==========================================================================*/
495 void smp_send_stop(void)
497 smp_call_function(stop_this_cpu
, NULL
, 0);
500 /*==========================================================================*
501 * Name: stop_this_cpu
503 * Description: This routine halt CPU.
505 * Born on Date: 2002.02.05
509 * Returns: void (cannot fail)
512 * Date Who Description
513 * ---------- --- --------------------------------------------------------
515 *==========================================================================*/
516 static void stop_this_cpu(void *dummy
)
518 int cpu_id
= smp_processor_id();
523 set_cpu_online(cpu_id
, false);
531 outl(0, M32R_ICU_IMASK_PORTL
);
532 inl(M32R_ICU_IMASK_PORTL
); /* dummy read */
538 void arch_send_call_function_ipi_mask(const struct cpumask
*mask
)
540 send_IPI_mask(mask
, CALL_FUNCTION_IPI
, 0);
543 void arch_send_call_function_single_ipi(int cpu
)
545 send_IPI_mask(cpumask_of(cpu
), CALL_FUNC_SINGLE_IPI
, 0);
548 /*==========================================================================*
549 * Name: smp_call_function_interrupt
551 * Description: This routine executes on CPU which received
552 * 'CALL_FUNCTION_IPI'.
554 * Born on Date: 2002.02.05
558 * Returns: void (cannot fail)
561 * Date Who Description
562 * ---------- --- --------------------------------------------------------
564 *==========================================================================*/
565 void smp_call_function_interrupt(void)
568 generic_smp_call_function_interrupt();
572 void smp_call_function_single_interrupt(void)
575 generic_smp_call_function_single_interrupt();
579 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
581 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
583 /*==========================================================================*
584 * Name: smp_send_timer
586 * Description: This routine sends a 'LOCAL_TIMER_IPI' to all other CPUs
589 * Born on Date: 2002.02.05
593 * Returns: void (cannot fail)
596 * Date Who Description
597 * ---------- --- --------------------------------------------------------
599 *==========================================================================*/
600 void smp_send_timer(void)
602 send_IPI_allbutself(LOCAL_TIMER_IPI
, 1);
605 /*==========================================================================*
606 * Name: smp_send_timer
608 * Description: This routine executes on CPU which received
611 * Born on Date: 2002.02.05
613 * Arguments: *regs - a pointer to the saved regster info
615 * Returns: void (cannot fail)
618 * Date Who Description
619 * ---------- --- --------------------------------------------------------
621 *==========================================================================*/
622 void smp_ipi_timer_interrupt(struct pt_regs
*regs
)
624 struct pt_regs
*old_regs
;
625 old_regs
= set_irq_regs(regs
);
627 smp_local_timer_interrupt();
629 set_irq_regs(old_regs
);
632 /*==========================================================================*
633 * Name: smp_local_timer_interrupt
635 * Description: Local timer interrupt handler. It does both profiling and
636 * process statistics/rescheduling.
637 * We do profiling in every local tick, statistics/rescheduling
638 * happen only every 'profiling multiplier' ticks. The default
639 * multiplier is 1 and it can be changed by writing the new
640 * multiplier value into /proc/profile.
642 * Born on Date: 2002.02.05
644 * Arguments: *regs - a pointer to the saved regster info
646 * Returns: void (cannot fail)
648 * Original: arch/i386/kernel/apic.c
651 * Date Who Description
652 * ---------- --- --------------------------------------------------------
653 * 2003-06-24 hy use per_cpu structure.
654 *==========================================================================*/
655 void smp_local_timer_interrupt(void)
657 int user
= user_mode(get_irq_regs());
658 int cpu_id
= smp_processor_id();
661 * The profiling function is SMP safe. (nothing can mess
662 * around with "current", and the profiling counters are
663 * updated with atomic operations). This is especially
664 * useful with a profiling multiplier != 1
667 profile_tick(CPU_PROFILING
);
669 if (--per_cpu(prof_counter
, cpu_id
) <= 0) {
671 * The multiplier may have changed since the last time we got
672 * to this point as a result of the user writing to
673 * /proc/profile. In this case we need to adjust the APIC
676 * Interrupts are already masked off at this point.
678 per_cpu(prof_counter
, cpu_id
)
679 = per_cpu(prof_multiplier
, cpu_id
);
680 if (per_cpu(prof_counter
, cpu_id
)
681 != per_cpu(prof_old_multiplier
, cpu_id
))
683 per_cpu(prof_old_multiplier
, cpu_id
)
684 = per_cpu(prof_counter
, cpu_id
);
687 update_process_times(user
);
691 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
692 /* Send IPI Routines */
693 /*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
695 /*==========================================================================*
696 * Name: send_IPI_allbutself
698 * Description: This routine sends a IPI to all other CPUs in the system.
700 * Born on Date: 2002.02.05
702 * Arguments: ipi_num - Number of IPI
703 * try - 0 : Send IPI certainly.
704 * !0 : The following IPI is not sent when Target CPU
705 * has not received the before IPI.
707 * Returns: void (cannot fail)
710 * Date Who Description
711 * ---------- --- --------------------------------------------------------
713 *==========================================================================*/
714 static void send_IPI_allbutself(int ipi_num
, int try)
718 cpumask_copy(&cpumask
, cpu_online_mask
);
719 cpumask_clear_cpu(smp_processor_id(), &cpumask
);
721 send_IPI_mask(&cpumask
, ipi_num
, try);
724 /*==========================================================================*
725 * Name: send_IPI_mask
727 * Description: This routine sends a IPI to CPUs in the system.
729 * Born on Date: 2002.02.05
731 * Arguments: cpu_mask - Bitmap of target CPUs logical ID
732 * ipi_num - Number of IPI
733 * try - 0 : Send IPI certainly.
734 * !0 : The following IPI is not sent when Target CPU
735 * has not received the before IPI.
737 * Returns: void (cannot fail)
740 * Date Who Description
741 * ---------- --- --------------------------------------------------------
743 *==========================================================================*/
744 static void send_IPI_mask(const struct cpumask
*cpumask
, int ipi_num
, int try)
746 cpumask_t physid_mask
, tmp
;
748 int num_cpus
= num_online_cpus();
750 if (num_cpus
<= 1) /* NO MP */
753 cpumask_and(&tmp
, cpumask
, cpu_online_mask
);
754 BUG_ON(!cpumask_equal(cpumask
, &tmp
));
756 cpumask_clear(&physid_mask
);
757 for_each_cpu(cpu_id
, cpumask
) {
758 if ((phys_id
= cpu_to_physid(cpu_id
)) != -1)
759 cpumask_set_cpu(phys_id
, &physid_mask
);
762 send_IPI_mask_phys(&physid_mask
, ipi_num
, try);
765 /*==========================================================================*
766 * Name: send_IPI_mask_phys
768 * Description: This routine sends a IPI to other CPUs in the system.
770 * Born on Date: 2002.02.05
772 * Arguments: cpu_mask - Bitmap of target CPUs physical ID
773 * ipi_num - Number of IPI
774 * try - 0 : Send IPI certainly.
775 * !0 : The following IPI is not sent when Target CPU
776 * has not received the before IPI.
778 * Returns: IPICRi regster value.
781 * Date Who Description
782 * ---------- --- --------------------------------------------------------
784 *==========================================================================*/
785 unsigned long send_IPI_mask_phys(const cpumask_t
*physid_mask
, int ipi_num
,
789 volatile unsigned long *ipicr_addr
;
790 unsigned long ipicr_val
;
791 unsigned long my_physid_mask
;
792 unsigned long mask
= cpumask_bits(physid_mask
)[0];
795 if (mask
& ~physids_coerce(phys_cpu_present_map
))
797 if (ipi_num
>= NR_IPIS
|| ipi_num
< 0)
801 ipilock
= &ipi_lock
[ipi_num
];
802 ipicr_addr
= (volatile unsigned long *)(M32R_ICU_IPICR_ADDR
804 my_physid_mask
= ~(1 << smp_processor_id());
809 * write IPICRi (send IPIi)
813 __asm__
__volatile__ (
814 ";; CHECK IPICRi == 0 \n\t"
822 ";; WRITE IPICRi (send IPIi) \n\t"
829 : "r"(ipicr_addr
), "r"(mask
), "r"(try), "r"(my_physid_mask
)
832 spin_unlock(ipilock
);