966 lines
		
	
	
	
		
			27 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			966 lines
		
	
	
	
		
			27 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								 *  linux/arch/m32r/kernel/smp.c
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *  M32R SMP support routines.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *  Copyright (c) 2001, 2002  Hitoshi Yamamoto
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *  Taken from i386 version.
							 | 
						||
| 
								 | 
							
								 *    (c) 1995 Alan Cox, Building #3 <alan@redhat.com>
							 | 
						||
| 
								 | 
							
								 *    (c) 1998-99, 2000 Ingo Molnar <mingo@redhat.com>
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *  This code is released under the GNU General Public License version 2 or
							 | 
						||
| 
								 | 
							
								 *  later.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#undef DEBUG_SMP
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <linux/irq.h>
							 | 
						||
| 
								 | 
							
								#include <linux/interrupt.h>
							 | 
						||
| 
								 | 
							
								#include <linux/spinlock.h>
							 | 
						||
| 
								 | 
							
								#include <linux/mm.h>
							 | 
						||
| 
								 | 
							
								#include <linux/smp.h>
							 | 
						||
| 
								 | 
							
								#include <linux/profile.h>
							 | 
						||
| 
								 | 
							
								#include <linux/cpu.h>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <asm/cacheflush.h>
							 | 
						||
| 
								 | 
							
								#include <asm/pgalloc.h>
							 | 
						||
| 
								 | 
							
								#include <asm/atomic.h>
							 | 
						||
| 
								 | 
							
								#include <asm/io.h>
							 | 
						||
| 
								 | 
							
								#include <asm/mmu_context.h>
							 | 
						||
| 
								 | 
							
								#include <asm/m32r.h>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								/* Data structures and variables                                             */
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								 * Structure and data for smp_call_function(). This is designed to minimise
							 | 
						||
| 
								 | 
							
								 * static memory requirements. It also looks cleaner.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								static DEFINE_SPINLOCK(call_lock);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								struct call_data_struct {
							 | 
						||
| 
								 | 
							
									void (*func) (void *info);
							 | 
						||
| 
								 | 
							
									void *info;
							 | 
						||
| 
								 | 
							
									atomic_t started;
							 | 
						||
| 
								 | 
							
									atomic_t finished;
							 | 
						||
| 
								 | 
							
									int wait;
							 | 
						||
| 
								 | 
							
								} __attribute__ ((__aligned__(SMP_CACHE_BYTES)));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static struct call_data_struct *call_data;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								 * For flush_cache_all()
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								static DEFINE_SPINLOCK(flushcache_lock);
							 | 
						||
| 
								 | 
							
								static volatile unsigned long flushcache_cpumask = 0;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								 * For flush_tlb_others()
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								static volatile cpumask_t flush_cpumask;
							 | 
						||
| 
								 | 
							
								static struct mm_struct *flush_mm;
							 | 
						||
| 
								 | 
							
								static struct vm_area_struct *flush_vma;
							 | 
						||
| 
								 | 
							
								static volatile unsigned long flush_va;
							 | 
						||
| 
								 | 
							
								static DEFINE_SPINLOCK(tlbstate_lock);
							 | 
						||
| 
								 | 
							
								#define FLUSH_ALL 0xffffffff
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								DECLARE_PER_CPU(int, prof_multiplier);
							 | 
						||
| 
								 | 
							
								DECLARE_PER_CPU(int, prof_old_multiplier);
							 | 
						||
| 
								 | 
							
								DECLARE_PER_CPU(int, prof_counter);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								extern spinlock_t ipi_lock[];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								/* Function Prototypes                                                       */
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void smp_send_reschedule(int);
							 | 
						||
| 
								 | 
							
								void smp_reschedule_interrupt(void);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void smp_flush_cache_all(void);
							 | 
						||
| 
								 | 
							
								void smp_flush_cache_all_interrupt(void);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void smp_flush_tlb_all(void);
							 | 
						||
| 
								 | 
							
								static void flush_tlb_all_ipi(void *);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void smp_flush_tlb_mm(struct mm_struct *);
							 | 
						||
| 
								 | 
							
								void smp_flush_tlb_range(struct vm_area_struct *, unsigned long, \
							 | 
						||
| 
								 | 
							
									unsigned long);
							 | 
						||
| 
								 | 
							
								void smp_flush_tlb_page(struct vm_area_struct *, unsigned long);
							 | 
						||
| 
								 | 
							
								static void flush_tlb_others(cpumask_t, struct mm_struct *,
							 | 
						||
| 
								 | 
							
									struct vm_area_struct *, unsigned long);
							 | 
						||
| 
								 | 
							
								void smp_invalidate_interrupt(void);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void smp_send_stop(void);
							 | 
						||
| 
								 | 
							
								static void stop_this_cpu(void *);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								int smp_call_function(void (*) (void *), void *, int, int);
							 | 
						||
| 
								 | 
							
								void smp_call_function_interrupt(void);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void smp_send_timer(void);
							 | 
						||
| 
								 | 
							
								void smp_ipi_timer_interrupt(struct pt_regs *);
							 | 
						||
| 
								 | 
							
								void smp_local_timer_interrupt(struct pt_regs *);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void send_IPI_allbutself(int, int);
							 | 
						||
| 
								 | 
							
								static void send_IPI_mask(cpumask_t, int, int);
							 | 
						||
| 
								 | 
							
								unsigned long send_IPI_mask_phys(cpumask_t, int, int);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								/* Rescheduling request Routines                                             */
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_send_reschedule
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine requests other CPU to execute rescheduling.
							 | 
						||
| 
								 | 
							
								 *               1.Send 'RESCHEDULE_IPI' to other CPU.
							 | 
						||
| 
								 | 
							
								 *                 Request other CPU to execute 'smp_reschedule_interrupt()'.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    cpu_id - Target CPU ID
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_send_reschedule(int cpu_id)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									WARN_ON(cpu_is_offline(cpu_id));
							 | 
						||
| 
								 | 
							
									send_IPI_mask(cpumask_of_cpu(cpu_id), RESCHEDULE_IPI, 1);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_reschedule_interrupt
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine executes on CPU which received
							 | 
						||
| 
								 | 
							
								 *               'RESCHEDULE_IPI'.
							 | 
						||
| 
								 | 
							
								 *               Rescheduling is processed at the exit of interrupt
							 | 
						||
| 
								 | 
							
								 *               operation.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    NONE
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_reschedule_interrupt(void)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									/* nothing to do */
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_flush_cache_all
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine sends a 'INVALIDATE_CACHE_IPI' to all other
							 | 
						||
| 
								 | 
							
								 *               CPUs in the system.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2003-05-28
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    NONE
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_flush_cache_all(void)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									cpumask_t cpumask;
							 | 
						||
| 
								 | 
							
									unsigned long *mask;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									preempt_disable();
							 | 
						||
| 
								 | 
							
									cpumask = cpu_online_map;
							 | 
						||
| 
								 | 
							
									cpu_clear(smp_processor_id(), cpumask);
							 | 
						||
| 
								 | 
							
									spin_lock(&flushcache_lock);
							 | 
						||
| 
								 | 
							
									mask=cpus_addr(cpumask);
							 | 
						||
| 
								 | 
							
									atomic_set_mask(*mask, (atomic_t *)&flushcache_cpumask);
							 | 
						||
| 
								 | 
							
									send_IPI_mask(cpumask, INVALIDATE_CACHE_IPI, 0);
							 | 
						||
| 
								 | 
							
									_flush_cache_copyback_all();
							 | 
						||
| 
								 | 
							
									while (flushcache_cpumask)
							 | 
						||
| 
								 | 
							
										mb();
							 | 
						||
| 
								 | 
							
									spin_unlock(&flushcache_lock);
							 | 
						||
| 
								 | 
							
									preempt_enable();
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void smp_flush_cache_all_interrupt(void)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									_flush_cache_copyback_all();
							 | 
						||
| 
								 | 
							
									clear_bit(smp_processor_id(), &flushcache_cpumask);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								/* TLB flush request Routins                                                 */
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_flush_tlb_all
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine flushes all processes TLBs.
							 | 
						||
| 
								 | 
							
								 *               1.Request other CPU to execute 'flush_tlb_all_ipi()'.
							 | 
						||
| 
								 | 
							
								 *               2.Execute 'do_flush_tlb_all_local()'.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    NONE
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_flush_tlb_all(void)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									unsigned long flags;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									preempt_disable();
							 | 
						||
| 
								 | 
							
									local_irq_save(flags);
							 | 
						||
| 
								 | 
							
									__flush_tlb_all();
							 | 
						||
| 
								 | 
							
									local_irq_restore(flags);
							 | 
						||
| 
								 | 
							
									smp_call_function(flush_tlb_all_ipi, 0, 1, 1);
							 | 
						||
| 
								 | 
							
									preempt_enable();
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         flush_tlb_all_ipi
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine flushes all local TLBs.
							 | 
						||
| 
								 | 
							
								 *               1.Execute 'do_flush_tlb_all_local()'.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    *info - not used
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								static void flush_tlb_all_ipi(void *info)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									__flush_tlb_all();
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_flush_tlb_mm
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine flushes the specified mm context TLB's.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    *mm - a pointer to the mm struct for flush TLB
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_flush_tlb_mm(struct mm_struct *mm)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									int cpu_id = smp_processor_id();
							 | 
						||
| 
								 | 
							
									cpumask_t cpu_mask;
							 | 
						||
| 
								 | 
							
									unsigned long *mmc = &mm->context[cpu_id];
							 | 
						||
| 
								 | 
							
									unsigned long flags;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									preempt_disable();
							 | 
						||
| 
								 | 
							
									cpu_mask = mm->cpu_vm_mask;
							 | 
						||
| 
								 | 
							
									cpu_clear(cpu_id, cpu_mask);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (*mmc != NO_CONTEXT) {
							 | 
						||
| 
								 | 
							
										local_irq_save(flags);
							 | 
						||
| 
								 | 
							
										*mmc = NO_CONTEXT;
							 | 
						||
| 
								 | 
							
										if (mm == current->mm)
							 | 
						||
| 
								 | 
							
											activate_context(mm);
							 | 
						||
| 
								 | 
							
										else
							 | 
						||
| 
								 | 
							
											cpu_clear(cpu_id, mm->cpu_vm_mask);
							 | 
						||
| 
								 | 
							
										local_irq_restore(flags);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									if (!cpus_empty(cpu_mask))
							 | 
						||
| 
								 | 
							
										flush_tlb_others(cpu_mask, mm, NULL, FLUSH_ALL);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									preempt_enable();
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_flush_tlb_range
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine flushes a range of pages.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    *mm - a pointer to the mm struct for flush TLB
							 | 
						||
| 
								 | 
							
								 *               start - not used
							 | 
						||
| 
								 | 
							
								 *               end - not used
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_flush_tlb_range(struct vm_area_struct *vma, unsigned long start,
							 | 
						||
| 
								 | 
							
									unsigned long end)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									smp_flush_tlb_mm(vma->vm_mm);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_flush_tlb_page
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine flushes one page.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    *vma - a pointer to the vma struct include va
							 | 
						||
| 
								 | 
							
								 *               va - virtual address for flush TLB
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_flush_tlb_page(struct vm_area_struct *vma, unsigned long va)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									struct mm_struct *mm = vma->vm_mm;
							 | 
						||
| 
								 | 
							
									int cpu_id = smp_processor_id();
							 | 
						||
| 
								 | 
							
									cpumask_t cpu_mask;
							 | 
						||
| 
								 | 
							
									unsigned long *mmc = &mm->context[cpu_id];
							 | 
						||
| 
								 | 
							
									unsigned long flags;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									preempt_disable();
							 | 
						||
| 
								 | 
							
									cpu_mask = mm->cpu_vm_mask;
							 | 
						||
| 
								 | 
							
									cpu_clear(cpu_id, cpu_mask);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifdef DEBUG_SMP
							 | 
						||
| 
								 | 
							
									if (!mm)
							 | 
						||
| 
								 | 
							
										BUG();
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (*mmc != NO_CONTEXT) {
							 | 
						||
| 
								 | 
							
										local_irq_save(flags);
							 | 
						||
| 
								 | 
							
										va &= PAGE_MASK;
							 | 
						||
| 
								 | 
							
										va |= (*mmc & MMU_CONTEXT_ASID_MASK);
							 | 
						||
| 
								 | 
							
										__flush_tlb_page(va);
							 | 
						||
| 
								 | 
							
										local_irq_restore(flags);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									if (!cpus_empty(cpu_mask))
							 | 
						||
| 
								 | 
							
										flush_tlb_others(cpu_mask, mm, vma, va);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									preempt_enable();
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         flush_tlb_others
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine requests other CPU to execute flush TLB.
							 | 
						||
| 
								 | 
							
								 *               1.Setup parmeters.
							 | 
						||
| 
								 | 
							
								 *               2.Send 'INVALIDATE_TLB_IPI' to other CPU.
							 | 
						||
| 
								 | 
							
								 *                 Request other CPU to execute 'smp_invalidate_interrupt()'.
							 | 
						||
| 
								 | 
							
								 *               3.Wait for other CPUs operation finished.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    cpumask - bitmap of target CPUs
							 | 
						||
| 
								 | 
							
								 *               *mm -  a pointer to the mm struct for flush TLB
							 | 
						||
| 
								 | 
							
								 *               *vma -  a pointer to the vma struct include va
							 | 
						||
| 
								 | 
							
								 *               va - virtual address for flush TLB
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								static void flush_tlb_others(cpumask_t cpumask, struct mm_struct *mm,
							 | 
						||
| 
								 | 
							
									struct vm_area_struct *vma, unsigned long va)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									unsigned long *mask;
							 | 
						||
| 
								 | 
							
								#ifdef DEBUG_SMP
							 | 
						||
| 
								 | 
							
									unsigned long flags;
							 | 
						||
| 
								 | 
							
									__save_flags(flags);
							 | 
						||
| 
								 | 
							
									if (!(flags & 0x0040))	/* Interrupt Disable NONONO */
							 | 
						||
| 
								 | 
							
										BUG();
							 | 
						||
| 
								 | 
							
								#endif /* DEBUG_SMP */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/*
							 | 
						||
| 
								 | 
							
									 * A couple of (to be removed) sanity checks:
							 | 
						||
| 
								 | 
							
									 *
							 | 
						||
| 
								 | 
							
									 * - we do not send IPIs to not-yet booted CPUs.
							 | 
						||
| 
								 | 
							
									 * - current CPU must not be in mask
							 | 
						||
| 
								 | 
							
									 * - mask must exist :)
							 | 
						||
| 
								 | 
							
									 */
							 | 
						||
| 
								 | 
							
									BUG_ON(cpus_empty(cpumask));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									BUG_ON(cpu_isset(smp_processor_id(), cpumask));
							 | 
						||
| 
								 | 
							
									BUG_ON(!mm);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/* If a CPU which we ran on has gone down, OK. */
							 | 
						||
| 
								 | 
							
									cpus_and(cpumask, cpumask, cpu_online_map);
							 | 
						||
| 
								 | 
							
									if (cpus_empty(cpumask))
							 | 
						||
| 
								 | 
							
										return;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/*
							 | 
						||
| 
								 | 
							
									 * i'm not happy about this global shared spinlock in the
							 | 
						||
| 
								 | 
							
									 * MM hot path, but we'll see how contended it is.
							 | 
						||
| 
								 | 
							
									 * Temporarily this turns IRQs off, so that lockups are
							 | 
						||
| 
								 | 
							
									 * detected by the NMI watchdog.
							 | 
						||
| 
								 | 
							
									 */
							 | 
						||
| 
								 | 
							
									spin_lock(&tlbstate_lock);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									flush_mm = mm;
							 | 
						||
| 
								 | 
							
									flush_vma = vma;
							 | 
						||
| 
								 | 
							
									flush_va = va;
							 | 
						||
| 
								 | 
							
									mask=cpus_addr(cpumask);
							 | 
						||
| 
								 | 
							
									atomic_set_mask(*mask, (atomic_t *)&flush_cpumask);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/*
							 | 
						||
| 
								 | 
							
									 * We have to send the IPI only to
							 | 
						||
| 
								 | 
							
									 * CPUs affected.
							 | 
						||
| 
								 | 
							
									 */
							 | 
						||
| 
								 | 
							
									send_IPI_mask(cpumask, INVALIDATE_TLB_IPI, 0);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									while (!cpus_empty(flush_cpumask)) {
							 | 
						||
| 
								 | 
							
										/* nothing. lockup detection does not belong here */
							 | 
						||
| 
								 | 
							
										mb();
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									flush_mm = NULL;
							 | 
						||
| 
								 | 
							
									flush_vma = NULL;
							 | 
						||
| 
								 | 
							
									flush_va = 0;
							 | 
						||
| 
								 | 
							
									spin_unlock(&tlbstate_lock);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_invalidate_interrupt
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine executes on CPU which received
							 | 
						||
| 
								 | 
							
								 *               'INVALIDATE_TLB_IPI'.
							 | 
						||
| 
								 | 
							
								 *               1.Flush local TLB.
							 | 
						||
| 
								 | 
							
								 *               2.Report flush TLB process was finished.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    NONE
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_invalidate_interrupt(void)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									int cpu_id = smp_processor_id();
							 | 
						||
| 
								 | 
							
									unsigned long *mmc = &flush_mm->context[cpu_id];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (!cpu_isset(cpu_id, flush_cpumask))
							 | 
						||
| 
								 | 
							
										return;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (flush_va == FLUSH_ALL) {
							 | 
						||
| 
								 | 
							
										*mmc = NO_CONTEXT;
							 | 
						||
| 
								 | 
							
										if (flush_mm == current->active_mm)
							 | 
						||
| 
								 | 
							
											activate_context(flush_mm);
							 | 
						||
| 
								 | 
							
										else
							 | 
						||
| 
								 | 
							
											cpu_clear(cpu_id, flush_mm->cpu_vm_mask);
							 | 
						||
| 
								 | 
							
									} else {
							 | 
						||
| 
								 | 
							
										unsigned long va = flush_va;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										if (*mmc != NO_CONTEXT) {
							 | 
						||
| 
								 | 
							
											va &= PAGE_MASK;
							 | 
						||
| 
								 | 
							
											va |= (*mmc & MMU_CONTEXT_ASID_MASK);
							 | 
						||
| 
								 | 
							
											__flush_tlb_page(va);
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									cpu_clear(cpu_id, flush_cpumask);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								/* Stop CPU request Routins                                                 */
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_send_stop
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine requests stop all CPUs.
							 | 
						||
| 
								 | 
							
								 *               1.Request other CPU to execute 'stop_this_cpu()'.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    NONE
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_send_stop(void)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									smp_call_function(stop_this_cpu, NULL, 1, 0);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         stop_this_cpu
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine halt CPU.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    NONE
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								static void stop_this_cpu(void *dummy)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									int cpu_id = smp_processor_id();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/*
							 | 
						||
| 
								 | 
							
									 * Remove this CPU:
							 | 
						||
| 
								 | 
							
									 */
							 | 
						||
| 
								 | 
							
									cpu_clear(cpu_id, cpu_online_map);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/*
							 | 
						||
| 
								 | 
							
									 * PSW IE = 1;
							 | 
						||
| 
								 | 
							
									 * IMASK = 0;
							 | 
						||
| 
								 | 
							
									 * goto SLEEP
							 | 
						||
| 
								 | 
							
									 */
							 | 
						||
| 
								 | 
							
									local_irq_disable();
							 | 
						||
| 
								 | 
							
									outl(0, M32R_ICU_IMASK_PORTL);
							 | 
						||
| 
								 | 
							
									inl(M32R_ICU_IMASK_PORTL);	/* dummy read */
							 | 
						||
| 
								 | 
							
									local_irq_enable();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									for ( ; ; );
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								/* Call function Routins                                                     */
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_call_function
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine sends a 'CALL_FUNCTION_IPI' to all other CPUs
							 | 
						||
| 
								 | 
							
								 *               in the system.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    *func - The function to run. This must be fast and
							 | 
						||
| 
								 | 
							
								 *                       non-blocking.
							 | 
						||
| 
								 | 
							
								 *               *info - An arbitrary pointer to pass to the function.
							 | 
						||
| 
								 | 
							
								 *               nonatomic - currently unused.
							 | 
						||
| 
								 | 
							
								 *               wait - If true, wait (atomically) until function has
							 | 
						||
| 
								 | 
							
								 *                      completed on other CPUs.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      0 on success, else a negative status code. Does not return
							 | 
						||
| 
								 | 
							
								 *               until remote CPUs are nearly ready to execute <<func>> or
							 | 
						||
| 
								 | 
							
								 *               are or have executed.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Cautions:     You must not call this function with disabled interrupts or
							 | 
						||
| 
								 | 
							
								 *               from a hardware interrupt handler, you may call it from a
							 | 
						||
| 
								 | 
							
								 *               bottom half handler.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								int smp_call_function(void (*func) (void *info), void *info, int nonatomic,
							 | 
						||
| 
								 | 
							
									int wait)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									struct call_data_struct data;
							 | 
						||
| 
								 | 
							
									int cpus;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifdef DEBUG_SMP
							 | 
						||
| 
								 | 
							
									unsigned long flags;
							 | 
						||
| 
								 | 
							
									__save_flags(flags);
							 | 
						||
| 
								 | 
							
									if (!(flags & 0x0040))	/* Interrupt Disable NONONO */
							 | 
						||
| 
								 | 
							
										BUG();
							 | 
						||
| 
								 | 
							
								#endif /* DEBUG_SMP */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/* Holding any lock stops cpus from going down. */
							 | 
						||
| 
								 | 
							
									spin_lock(&call_lock);
							 | 
						||
| 
								 | 
							
									cpus = num_online_cpus() - 1;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (!cpus) {
							 | 
						||
| 
								 | 
							
										spin_unlock(&call_lock);
							 | 
						||
| 
								 | 
							
										return 0;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/* Can deadlock when called with interrupts disabled */
							 | 
						||
| 
								 | 
							
									WARN_ON(irqs_disabled());
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									data.func = func;
							 | 
						||
| 
								 | 
							
									data.info = info;
							 | 
						||
| 
								 | 
							
									atomic_set(&data.started, 0);
							 | 
						||
| 
								 | 
							
									data.wait = wait;
							 | 
						||
| 
								 | 
							
									if (wait)
							 | 
						||
| 
								 | 
							
										atomic_set(&data.finished, 0);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									call_data = &data;
							 | 
						||
| 
								 | 
							
									mb();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/* Send a message to all other CPUs and wait for them to respond */
							 | 
						||
| 
								 | 
							
									send_IPI_allbutself(CALL_FUNCTION_IPI, 0);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/* Wait for response */
							 | 
						||
| 
								 | 
							
									while (atomic_read(&data.started) != cpus)
							 | 
						||
| 
								 | 
							
										barrier();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (wait)
							 | 
						||
| 
								 | 
							
										while (atomic_read(&data.finished) != cpus)
							 | 
						||
| 
								 | 
							
											barrier();
							 | 
						||
| 
								 | 
							
									spin_unlock(&call_lock);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									return 0;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_call_function_interrupt
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine executes on CPU which received
							 | 
						||
| 
								 | 
							
								 *               'CALL_FUNCTION_IPI'.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    NONE
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_call_function_interrupt(void)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									void (*func) (void *info) = call_data->func;
							 | 
						||
| 
								 | 
							
									void *info = call_data->info;
							 | 
						||
| 
								 | 
							
									int wait = call_data->wait;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/*
							 | 
						||
| 
								 | 
							
									 * Notify initiating CPU that I've grabbed the data and am
							 | 
						||
| 
								 | 
							
									 * about to execute the function
							 | 
						||
| 
								 | 
							
									 */
							 | 
						||
| 
								 | 
							
									mb();
							 | 
						||
| 
								 | 
							
									atomic_inc(&call_data->started);
							 | 
						||
| 
								 | 
							
									/*
							 | 
						||
| 
								 | 
							
									 * At this point the info structure may be out of scope unless wait==1
							 | 
						||
| 
								 | 
							
									 */
							 | 
						||
| 
								 | 
							
									irq_enter();
							 | 
						||
| 
								 | 
							
									(*func)(info);
							 | 
						||
| 
								 | 
							
									irq_exit();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (wait) {
							 | 
						||
| 
								 | 
							
										mb();
							 | 
						||
| 
								 | 
							
										atomic_inc(&call_data->finished);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								/* Timer Routins                                                             */
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_send_timer
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine sends a 'LOCAL_TIMER_IPI' to all other CPUs
							 | 
						||
| 
								 | 
							
								 *               in the system.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    NONE
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_send_timer(void)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									send_IPI_allbutself(LOCAL_TIMER_IPI, 1);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_send_timer
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine executes on CPU which received
							 | 
						||
| 
								 | 
							
								 *               'LOCAL_TIMER_IPI'.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    *regs - a pointer to the saved regster info
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_ipi_timer_interrupt(struct pt_regs *regs)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									irq_enter();
							 | 
						||
| 
								 | 
							
									smp_local_timer_interrupt(regs);
							 | 
						||
| 
								 | 
							
									irq_exit();
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         smp_local_timer_interrupt
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  Local timer interrupt handler. It does both profiling and
							 | 
						||
| 
								 | 
							
								 *               process statistics/rescheduling.
							 | 
						||
| 
								 | 
							
								 *               We do profiling in every local tick, statistics/rescheduling
							 | 
						||
| 
								 | 
							
								 *               happen only every 'profiling multiplier' ticks. The default
							 | 
						||
| 
								 | 
							
								 *               multiplier is 1 and it can be changed by writing the new
							 | 
						||
| 
								 | 
							
								 *               multiplier value into /proc/profile.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    *regs - a pointer to the saved regster info
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Original:     arch/i386/kernel/apic.c
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 * 2003-06-24 hy  use per_cpu structure.
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void smp_local_timer_interrupt(struct pt_regs *regs)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									int user = user_mode(regs);
							 | 
						||
| 
								 | 
							
									int cpu_id = smp_processor_id();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/*
							 | 
						||
| 
								 | 
							
									 * The profiling function is SMP safe. (nothing can mess
							 | 
						||
| 
								 | 
							
									 * around with "current", and the profiling counters are
							 | 
						||
| 
								 | 
							
									 * updated with atomic operations). This is especially
							 | 
						||
| 
								 | 
							
									 * useful with a profiling multiplier != 1
							 | 
						||
| 
								 | 
							
									 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									profile_tick(CPU_PROFILING, regs);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (--per_cpu(prof_counter, cpu_id) <= 0) {
							 | 
						||
| 
								 | 
							
										/*
							 | 
						||
| 
								 | 
							
										 * The multiplier may have changed since the last time we got
							 | 
						||
| 
								 | 
							
										 * to this point as a result of the user writing to
							 | 
						||
| 
								 | 
							
										 * /proc/profile. In this case we need to adjust the APIC
							 | 
						||
| 
								 | 
							
										 * timer accordingly.
							 | 
						||
| 
								 | 
							
										 *
							 | 
						||
| 
								 | 
							
										 * Interrupts are already masked off at this point.
							 | 
						||
| 
								 | 
							
										 */
							 | 
						||
| 
								 | 
							
										per_cpu(prof_counter, cpu_id)
							 | 
						||
| 
								 | 
							
											= per_cpu(prof_multiplier, cpu_id);
							 | 
						||
| 
								 | 
							
										if (per_cpu(prof_counter, cpu_id)
							 | 
						||
| 
								 | 
							
											!= per_cpu(prof_old_multiplier, cpu_id))
							 | 
						||
| 
								 | 
							
										{
							 | 
						||
| 
								 | 
							
											per_cpu(prof_old_multiplier, cpu_id)
							 | 
						||
| 
								 | 
							
												= per_cpu(prof_counter, cpu_id);
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										update_process_times(user);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								/* Send IPI Routins                                                          */
							 | 
						||
| 
								 | 
							
								/*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         send_IPI_allbutself
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine sends a IPI to all other CPUs in the system.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    ipi_num - Number of IPI
							 | 
						||
| 
								 | 
							
								 *               try -  0 : Send IPI certainly.
							 | 
						||
| 
								 | 
							
								 *                     !0 : The following IPI is not sended when Target CPU
							 | 
						||
| 
								 | 
							
								 *                          has not received the before IPI.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								void send_IPI_allbutself(int ipi_num, int try)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									cpumask_t cpumask;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									cpumask = cpu_online_map;
							 | 
						||
| 
								 | 
							
									cpu_clear(smp_processor_id(), cpumask);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									send_IPI_mask(cpumask, ipi_num, try);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         send_IPI_mask
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine sends a IPI to CPUs in the system.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    cpu_mask - Bitmap of target CPUs logical ID
							 | 
						||
| 
								 | 
							
								 *               ipi_num - Number of IPI
							 | 
						||
| 
								 | 
							
								 *               try -  0 : Send IPI certainly.
							 | 
						||
| 
								 | 
							
								 *                     !0 : The following IPI is not sended when Target CPU
							 | 
						||
| 
								 | 
							
								 *                          has not received the before IPI.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      void (cannot fail)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								static void send_IPI_mask(cpumask_t cpumask, int ipi_num, int try)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									cpumask_t physid_mask, tmp;
							 | 
						||
| 
								 | 
							
									int cpu_id, phys_id;
							 | 
						||
| 
								 | 
							
									int num_cpus = num_online_cpus();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (num_cpus <= 1)	/* NO MP */
							 | 
						||
| 
								 | 
							
										return;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									cpus_and(tmp, cpumask, cpu_online_map);
							 | 
						||
| 
								 | 
							
									BUG_ON(!cpus_equal(cpumask, tmp));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									physid_mask = CPU_MASK_NONE;
							 | 
						||
| 
								 | 
							
									for_each_cpu_mask(cpu_id, cpumask){
							 | 
						||
| 
								 | 
							
										if ((phys_id = cpu_to_physid(cpu_id)) != -1)
							 | 
						||
| 
								 | 
							
											cpu_set(phys_id, physid_mask);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									send_IPI_mask_phys(physid_mask, ipi_num, try);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*==========================================================================*
							 | 
						||
| 
								 | 
							
								 * Name:         send_IPI_mask_phys
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Description:  This routine sends a IPI to other CPUs in the system.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Born on Date: 2002.02.05
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Arguments:    cpu_mask - Bitmap of target CPUs physical ID
							 | 
						||
| 
								 | 
							
								 *               ipi_num - Number of IPI
							 | 
						||
| 
								 | 
							
								 *               try -  0 : Send IPI certainly.
							 | 
						||
| 
								 | 
							
								 *                     !0 : The following IPI is not sended when Target CPU
							 | 
						||
| 
								 | 
							
								 *                          has not received the before IPI.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Returns:      IPICRi regster value.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Modification log:
							 | 
						||
| 
								 | 
							
								 * Date       Who Description
							 | 
						||
| 
								 | 
							
								 * ---------- --- --------------------------------------------------------
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *==========================================================================*/
							 | 
						||
| 
								 | 
							
								unsigned long send_IPI_mask_phys(cpumask_t physid_mask, int ipi_num,
							 | 
						||
| 
								 | 
							
									int try)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									spinlock_t *ipilock;
							 | 
						||
| 
								 | 
							
									unsigned long flags = 0;
							 | 
						||
| 
								 | 
							
									volatile unsigned long *ipicr_addr;
							 | 
						||
| 
								 | 
							
									unsigned long ipicr_val;
							 | 
						||
| 
								 | 
							
									unsigned long my_physid_mask;
							 | 
						||
| 
								 | 
							
									unsigned long mask = cpus_addr(physid_mask)[0];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (mask & ~physids_coerce(phys_cpu_present_map))
							 | 
						||
| 
								 | 
							
										BUG();
							 | 
						||
| 
								 | 
							
									if (ipi_num >= NR_IPIS)
							 | 
						||
| 
								 | 
							
										BUG();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									mask <<= IPI_SHIFT;
							 | 
						||
| 
								 | 
							
									ipilock = &ipi_lock[ipi_num];
							 | 
						||
| 
								 | 
							
									ipicr_addr = (volatile unsigned long *)(M32R_ICU_IPICR_ADDR
							 | 
						||
| 
								 | 
							
										+ (ipi_num << 2));
							 | 
						||
| 
								 | 
							
									my_physid_mask = ~(1 << smp_processor_id());
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/*
							 | 
						||
| 
								 | 
							
									 * lock ipi_lock[i]
							 | 
						||
| 
								 | 
							
									 * check IPICRi == 0
							 | 
						||
| 
								 | 
							
									 * write IPICRi (send IPIi)
							 | 
						||
| 
								 | 
							
									 * unlock ipi_lock[i]
							 | 
						||
| 
								 | 
							
									 */
							 | 
						||
| 
								 | 
							
									__asm__ __volatile__ (
							 | 
						||
| 
								 | 
							
										";; LOCK ipi_lock[i]		\n\t"
							 | 
						||
| 
								 | 
							
										".fillinsn			\n"
							 | 
						||
| 
								 | 
							
										"1:				\n\t"
							 | 
						||
| 
								 | 
							
										"mvfc	%1, psw 		\n\t"
							 | 
						||
| 
								 | 
							
										"clrpsw	#0x40 -> nop		\n\t"
							 | 
						||
| 
								 | 
							
										DCACHE_CLEAR("r4", "r5", "%2")
							 | 
						||
| 
								 | 
							
										"lock	r4, @%2			\n\t"
							 | 
						||
| 
								 | 
							
										"addi	r4, #-1			\n\t"
							 | 
						||
| 
								 | 
							
										"unlock	r4, @%2			\n\t"
							 | 
						||
| 
								 | 
							
										"mvtc	%1, psw			\n\t"
							 | 
						||
| 
								 | 
							
										"bnez	r4, 2f			\n\t"
							 | 
						||
| 
								 | 
							
										LOCK_SECTION_START(".balign 4 \n\t")
							 | 
						||
| 
								 | 
							
										".fillinsn			\n"
							 | 
						||
| 
								 | 
							
										"2:				\n\t"
							 | 
						||
| 
								 | 
							
										"ld	r4, @%2			\n\t"
							 | 
						||
| 
								 | 
							
										"blez	r4, 2b			\n\t"
							 | 
						||
| 
								 | 
							
										"bra	1b			\n\t"
							 | 
						||
| 
								 | 
							
										LOCK_SECTION_END
							 | 
						||
| 
								 | 
							
										";; CHECK IPICRi == 0		\n\t"
							 | 
						||
| 
								 | 
							
										".fillinsn			\n"
							 | 
						||
| 
								 | 
							
										"3:				\n\t"
							 | 
						||
| 
								 | 
							
										"ld	%0, @%3			\n\t"
							 | 
						||
| 
								 | 
							
										"and	%0, %6			\n\t"
							 | 
						||
| 
								 | 
							
										"beqz	%0, 4f			\n\t"
							 | 
						||
| 
								 | 
							
										"bnez	%5, 5f			\n\t"
							 | 
						||
| 
								 | 
							
										"bra	3b			\n\t"
							 | 
						||
| 
								 | 
							
										";; WRITE IPICRi (send IPIi)	\n\t"
							 | 
						||
| 
								 | 
							
										".fillinsn			\n"
							 | 
						||
| 
								 | 
							
										"4:				\n\t"
							 | 
						||
| 
								 | 
							
										"st	%4, @%3			\n\t"
							 | 
						||
| 
								 | 
							
										";; UNLOCK ipi_lock[i]		\n\t"
							 | 
						||
| 
								 | 
							
										".fillinsn			\n"
							 | 
						||
| 
								 | 
							
										"5:				\n\t"
							 | 
						||
| 
								 | 
							
										"ldi	r4, #1			\n\t"
							 | 
						||
| 
								 | 
							
										"st	r4, @%2			\n\t"
							 | 
						||
| 
								 | 
							
										: "=&r"(ipicr_val)
							 | 
						||
| 
								 | 
							
										: "r"(flags), "r"(&ipilock->slock), "r"(ipicr_addr),
							 | 
						||
| 
								 | 
							
										  "r"(mask), "r"(try), "r"(my_physid_mask)
							 | 
						||
| 
								 | 
							
										: "memory", "r4"
							 | 
						||
| 
								 | 
							
								#ifdef CONFIG_CHIP_M32700_TS1
							 | 
						||
| 
								 | 
							
										, "r5"
							 | 
						||
| 
								 | 
							
								#endif	/* CONFIG_CHIP_M32700_TS1 */
							 | 
						||
| 
								 | 
							
									);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									return ipicr_val;
							 | 
						||
| 
								 | 
							
								}
							 |