~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

TOMOYO Linux Cross Reference
Linux/arch/sh/kernel/idle.c

Version: ~ [ linux-6.12-rc7 ] ~ [ linux-6.11.7 ] ~ [ linux-6.10.14 ] ~ [ linux-6.9.12 ] ~ [ linux-6.8.12 ] ~ [ linux-6.7.12 ] ~ [ linux-6.6.60 ] ~ [ linux-6.5.13 ] ~ [ linux-6.4.16 ] ~ [ linux-6.3.13 ] ~ [ linux-6.2.16 ] ~ [ linux-6.1.116 ] ~ [ linux-6.0.19 ] ~ [ linux-5.19.17 ] ~ [ linux-5.18.19 ] ~ [ linux-5.17.15 ] ~ [ linux-5.16.20 ] ~ [ linux-5.15.171 ] ~ [ linux-5.14.21 ] ~ [ linux-5.13.19 ] ~ [ linux-5.12.19 ] ~ [ linux-5.11.22 ] ~ [ linux-5.10.229 ] ~ [ linux-5.9.16 ] ~ [ linux-5.8.18 ] ~ [ linux-5.7.19 ] ~ [ linux-5.6.19 ] ~ [ linux-5.5.19 ] ~ [ linux-5.4.285 ] ~ [ linux-5.3.18 ] ~ [ linux-5.2.21 ] ~ [ linux-5.1.21 ] ~ [ linux-5.0.21 ] ~ [ linux-4.20.17 ] ~ [ linux-4.19.323 ] ~ [ linux-4.18.20 ] ~ [ linux-4.17.19 ] ~ [ linux-4.16.18 ] ~ [ linux-4.15.18 ] ~ [ linux-4.14.336 ] ~ [ linux-4.13.16 ] ~ [ linux-4.12.14 ] ~ [ linux-4.11.12 ] ~ [ linux-4.10.17 ] ~ [ linux-4.9.337 ] ~ [ linux-4.4.302 ] ~ [ linux-3.10.108 ] ~ [ linux-2.6.32.71 ] ~ [ linux-2.6.0 ] ~ [ linux-2.4.37.11 ] ~ [ unix-v6-master ] ~ [ ccs-tools-1.8.12 ] ~ [ policy-sample ] ~
Architecture: ~ [ i386 ] ~ [ alpha ] ~ [ m68k ] ~ [ mips ] ~ [ ppc ] ~ [ sparc ] ~ [ sparc64 ] ~

Diff markup

Differences between /arch/sh/kernel/idle.c (Version linux-6.12-rc7) and /arch/mips/kernel/idle.c (Version linux-4.12.14)


  1 // SPDX-License-Identifier: GPL-2.0            << 
  2 /*                                                  1 /*
  3  * The idle loop for all SuperH platforms.     !!   2  * MIPS idle loop and WAIT instruction support.
  4  *                                                  3  *
  5  *  Copyright (C) 2002 - 2009  Paul Mundt      !!   4  * Copyright (C) xxxx  the Anonymous
                                                   >>   5  * Copyright (C) 1994 - 2006 Ralf Baechle
                                                   >>   6  * Copyright (C) 2003, 2004  Maciej W. Rozycki
                                                   >>   7  * Copyright (C) 2001, 2004, 2011, 2012  MIPS Technologies, Inc.
                                                   >>   8  *
                                                   >>   9  * This program is free software; you can redistribute it and/or
                                                   >>  10  * modify it under the terms of the GNU General Public License
                                                   >>  11  * as published by the Free Software Foundation; either version
                                                   >>  12  * 2 of the License, or (at your option) any later version.
  6  */                                                13  */
  7 #include <linux/cpu.h>                         !!  14 #include <linux/export.h>
  8 #include <linux/module.h>                      << 
  9 #include <linux/init.h>                            15 #include <linux/init.h>
 10 #include <linux/mm.h>                          << 
 11 #include <linux/pm.h>                          << 
 12 #include <linux/tick.h>                        << 
 13 #include <linux/preempt.h>                     << 
 14 #include <linux/thread_info.h>                 << 
 15 #include <linux/irqflags.h>                        16 #include <linux/irqflags.h>
 16 #include <linux/smp.h>                         !!  17 #include <linux/printk.h>
 17 #include <linux/atomic.h>                      !!  18 #include <linux/sched.h>
 18 #include <asm/processor.h>                     !!  19 #include <asm/cpu.h>
 19 #include <asm/smp.h>                           !!  20 #include <asm/cpu-info.h>
 20 #include <asm/bl_bit.h>                        !!  21 #include <asm/cpu-type.h>
                                                   >>  22 #include <asm/idle.h>
                                                   >>  23 #include <asm/mipsregs.h>
 21                                                    24 
 22 static void (*sh_idle)(void);                  !!  25 /*
                                                   >>  26  * Not all of the MIPS CPUs have the "wait" instruction available. Moreover,
                                                   >>  27  * the implementation of the "wait" feature differs between CPU families. This
                                                   >>  28  * points to the function that implements CPU specific wait.
                                                   >>  29  * The wait instruction stops the pipeline and reduces the power consumption of
                                                   >>  30  * the CPU very much.
                                                   >>  31  */
                                                   >>  32 void (*cpu_wait)(void);
                                                   >>  33 EXPORT_SYMBOL(cpu_wait);
 23                                                    34 
 24 void default_idle(void)                        !!  35 static void r3081_wait(void)
 25 {                                                  36 {
 26         set_bl_bit();                          !!  37         unsigned long cfg = read_c0_conf();
 27         raw_local_irq_enable();                !!  38         write_c0_conf(cfg | R30XX_CONF_HALT);
 28         /* Isn't this racy ? */                !!  39         local_irq_enable();
 29         cpu_sleep();                           << 
 30         raw_local_irq_disable();               << 
 31         clear_bl_bit();                        << 
 32 }                                                  40 }
 33                                                    41 
 34 void __noreturn arch_cpu_idle_dead(void)       !!  42 static void r39xx_wait(void)
 35 {                                                  43 {
 36         play_dead();                           !!  44         if (!need_resched())
                                                   >>  45                 write_c0_conf(read_c0_conf() | TX39_CONF_HALT);
                                                   >>  46         local_irq_enable();
 37 }                                                  47 }
 38                                                    48 
 39 void arch_cpu_idle(void)                       !!  49 void r4k_wait(void)
                                                   >>  50 {
                                                   >>  51         local_irq_enable();
                                                   >>  52         __r4k_wait();
                                                   >>  53 }
                                                   >>  54 
                                                   >>  55 /*
                                                   >>  56  * This variant is preferable as it allows testing need_resched and going to
                                                   >>  57  * sleep depending on the outcome atomically.  Unfortunately the "It is
                                                   >>  58  * implementation-dependent whether the pipeline restarts when a non-enabled
                                                   >>  59  * interrupt is requested" restriction in the MIPS32/MIPS64 architecture makes
                                                   >>  60  * using this version a gamble.
                                                   >>  61  */
                                                   >>  62 void r4k_wait_irqoff(void)
                                                   >>  63 {
                                                   >>  64         if (!need_resched())
                                                   >>  65                 __asm__(
                                                   >>  66                 "       .set    push            \n"
                                                   >>  67                 "       .set    arch=r4000      \n"
                                                   >>  68                 "       wait                    \n"
                                                   >>  69                 "       .set    pop             \n");
                                                   >>  70         local_irq_enable();
                                                   >>  71 }
                                                   >>  72 
                                                   >>  73 /*
                                                   >>  74  * The RM7000 variant has to handle erratum 38.  The workaround is to not
                                                   >>  75  * have any pending stores when the WAIT instruction is executed.
                                                   >>  76  */
                                                   >>  77 static void rm7k_wait_irqoff(void)
 40 {                                                  78 {
 41         sh_idle();                             !!  79         if (!need_resched())
                                                   >>  80                 __asm__(
                                                   >>  81                 "       .set    push                                    \n"
                                                   >>  82                 "       .set    arch=r4000                              \n"
                                                   >>  83                 "       .set    noat                                    \n"
                                                   >>  84                 "       mfc0    $1, $12                                 \n"
                                                   >>  85                 "       sync                                            \n"
                                                   >>  86                 "       mtc0    $1, $12         # stalls until W stage  \n"
                                                   >>  87                 "       wait                                            \n"
                                                   >>  88                 "       mtc0    $1, $12         # stalls until W stage  \n"
                                                   >>  89                 "       .set    pop                                     \n");
                                                   >>  90         local_irq_enable();
 42 }                                                  91 }
 43                                                    92 
 44 void __init select_idle_routine(void)          !!  93 /*
                                                   >>  94  * Au1 'wait' is only useful when the 32kHz counter is used as timer,
                                                   >>  95  * since coreclock (and the cp0 counter) stops upon executing it. Only an
                                                   >>  96  * interrupt can wake it, so they must be enabled before entering idle modes.
                                                   >>  97  */
                                                   >>  98 static void au1k_wait(void)
 45 {                                                  99 {
                                                   >> 100         unsigned long c0status = read_c0_status() | 1;  /* irqs on */
                                                   >> 101 
                                                   >> 102         __asm__(
                                                   >> 103         "       .set    arch=r4000                      \n"
                                                   >> 104         "       cache   0x14, 0(%0)             \n"
                                                   >> 105         "       cache   0x14, 32(%0)            \n"
                                                   >> 106         "       sync                            \n"
                                                   >> 107         "       mtc0    %1, $12                 \n" /* wr c0status */
                                                   >> 108         "       wait                            \n"
                                                   >> 109         "       nop                             \n"
                                                   >> 110         "       nop                             \n"
                                                   >> 111         "       nop                             \n"
                                                   >> 112         "       nop                             \n"
                                                   >> 113         "       .set    mips0                   \n"
                                                   >> 114         : : "r" (au1k_wait), "r" (c0status));
                                                   >> 115 }
                                                   >> 116 
                                                   >> 117 static int __initdata nowait;
                                                   >> 118 
                                                   >> 119 static int __init wait_disable(char *s)
                                                   >> 120 {
                                                   >> 121         nowait = 1;
                                                   >> 122 
                                                   >> 123         return 1;
                                                   >> 124 }
                                                   >> 125 
                                                   >> 126 __setup("nowait", wait_disable);
                                                   >> 127 
                                                   >> 128 void __init check_wait(void)
                                                   >> 129 {
                                                   >> 130         struct cpuinfo_mips *c = &current_cpu_data;
                                                   >> 131 
                                                   >> 132         if (nowait) {
                                                   >> 133                 printk("Wait instruction disabled.\n");
                                                   >> 134                 return;
                                                   >> 135         }
                                                   >> 136 
 46         /*                                        137         /*
 47          * If a platform has set its own idle  !! 138          * MIPSr6 specifies that masked interrupts should unblock an executing
                                                   >> 139          * wait instruction, and thus that it is safe for us to use
                                                   >> 140          * r4k_wait_irqoff. Yippee!
 48          */                                       141          */
 49         if (!sh_idle)                          !! 142         if (cpu_has_mips_r6) {
 50                 sh_idle = default_idle;        !! 143                 cpu_wait = r4k_wait_irqoff;
                                                   >> 144                 return;
                                                   >> 145         }
                                                   >> 146 
                                                   >> 147         switch (current_cpu_type()) {
                                                   >> 148         case CPU_R3081:
                                                   >> 149         case CPU_R3081E:
                                                   >> 150                 cpu_wait = r3081_wait;
                                                   >> 151                 break;
                                                   >> 152         case CPU_TX3927:
                                                   >> 153                 cpu_wait = r39xx_wait;
                                                   >> 154                 break;
                                                   >> 155         case CPU_R4200:
                                                   >> 156 /*      case CPU_R4300: */
                                                   >> 157         case CPU_R4600:
                                                   >> 158         case CPU_R4640:
                                                   >> 159         case CPU_R4650:
                                                   >> 160         case CPU_R4700:
                                                   >> 161         case CPU_R5000:
                                                   >> 162         case CPU_R5500:
                                                   >> 163         case CPU_NEVADA:
                                                   >> 164         case CPU_4KC:
                                                   >> 165         case CPU_4KEC:
                                                   >> 166         case CPU_4KSC:
                                                   >> 167         case CPU_5KC:
                                                   >> 168         case CPU_5KE:
                                                   >> 169         case CPU_25KF:
                                                   >> 170         case CPU_PR4450:
                                                   >> 171         case CPU_BMIPS3300:
                                                   >> 172         case CPU_BMIPS4350:
                                                   >> 173         case CPU_BMIPS4380:
                                                   >> 174         case CPU_CAVIUM_OCTEON:
                                                   >> 175         case CPU_CAVIUM_OCTEON_PLUS:
                                                   >> 176         case CPU_CAVIUM_OCTEON2:
                                                   >> 177         case CPU_CAVIUM_OCTEON3:
                                                   >> 178         case CPU_JZRISC:
                                                   >> 179         case CPU_LOONGSON1:
                                                   >> 180         case CPU_XLR:
                                                   >> 181         case CPU_XLP:
                                                   >> 182                 cpu_wait = r4k_wait;
                                                   >> 183                 break;
                                                   >> 184         case CPU_LOONGSON3:
                                                   >> 185                 if ((c->processor_id & PRID_REV_MASK) >= PRID_REV_LOONGSON3A_R2)
                                                   >> 186                         cpu_wait = r4k_wait;
                                                   >> 187                 break;
                                                   >> 188 
                                                   >> 189         case CPU_BMIPS5000:
                                                   >> 190                 cpu_wait = r4k_wait_irqoff;
                                                   >> 191                 break;
                                                   >> 192         case CPU_RM7000:
                                                   >> 193                 cpu_wait = rm7k_wait_irqoff;
                                                   >> 194                 break;
                                                   >> 195 
                                                   >> 196         case CPU_PROAPTIV:
                                                   >> 197         case CPU_P5600:
                                                   >> 198                 /*
                                                   >> 199                  * Incoming Fast Debug Channel (FDC) data during a wait
                                                   >> 200                  * instruction causes the wait never to resume, even if an
                                                   >> 201                  * interrupt is received. Avoid using wait at all if FDC data is
                                                   >> 202                  * likely to be received.
                                                   >> 203                  */
                                                   >> 204                 if (IS_ENABLED(CONFIG_MIPS_EJTAG_FDC_TTY))
                                                   >> 205                         break;
                                                   >> 206                 /* fall through */
                                                   >> 207         case CPU_M14KC:
                                                   >> 208         case CPU_M14KEC:
                                                   >> 209         case CPU_24K:
                                                   >> 210         case CPU_34K:
                                                   >> 211         case CPU_1004K:
                                                   >> 212         case CPU_1074K:
                                                   >> 213         case CPU_INTERAPTIV:
                                                   >> 214         case CPU_M5150:
                                                   >> 215         case CPU_QEMU_GENERIC:
                                                   >> 216                 cpu_wait = r4k_wait;
                                                   >> 217                 if (read_c0_config7() & MIPS_CONF7_WII)
                                                   >> 218                         cpu_wait = r4k_wait_irqoff;
                                                   >> 219                 break;
                                                   >> 220 
                                                   >> 221         case CPU_74K:
                                                   >> 222                 cpu_wait = r4k_wait;
                                                   >> 223                 if ((c->processor_id & 0xff) >= PRID_REV_ENCODE_332(2, 1, 0))
                                                   >> 224                         cpu_wait = r4k_wait_irqoff;
                                                   >> 225                 break;
                                                   >> 226 
                                                   >> 227         case CPU_TX49XX:
                                                   >> 228                 cpu_wait = r4k_wait_irqoff;
                                                   >> 229                 break;
                                                   >> 230         case CPU_ALCHEMY:
                                                   >> 231                 cpu_wait = au1k_wait;
                                                   >> 232                 break;
                                                   >> 233         case CPU_20KC:
                                                   >> 234                 /*
                                                   >> 235                  * WAIT on Rev1.0 has E1, E2, E3 and E16.
                                                   >> 236                  * WAIT on Rev2.0 and Rev3.0 has E16.
                                                   >> 237                  * Rev3.1 WAIT is nop, why bother
                                                   >> 238                  */
                                                   >> 239                 if ((c->processor_id & 0xff) <= 0x64)
                                                   >> 240                         break;
                                                   >> 241 
                                                   >> 242                 /*
                                                   >> 243                  * Another rev is incremeting c0_count at a reduced clock
                                                   >> 244                  * rate while in WAIT mode.  So we basically have the choice
                                                   >> 245                  * between using the cp0 timer as clocksource or avoiding
                                                   >> 246                  * the WAIT instruction.  Until more details are known,
                                                   >> 247                  * disable the use of WAIT for 20Kc entirely.
                                                   >> 248                    cpu_wait = r4k_wait;
                                                   >> 249                  */
                                                   >> 250                 break;
                                                   >> 251         default:
                                                   >> 252                 break;
                                                   >> 253         }
 51 }                                                 254 }
 52                                                   255 
 53 void stop_this_cpu(void *unused)               !! 256 void arch_cpu_idle(void)
 54 {                                                 257 {
 55         local_irq_disable();                   !! 258         if (cpu_wait)
 56         set_cpu_online(smp_processor_id(), fal !! 259                 cpu_wait();
                                                   >> 260         else
                                                   >> 261                 local_irq_enable();
                                                   >> 262 }
                                                   >> 263 
                                                   >> 264 #ifdef CONFIG_CPU_IDLE
 57                                                   265 
 58         for (;;)                               !! 266 int mips_cpuidle_wait_enter(struct cpuidle_device *dev,
 59                 cpu_sleep();                   !! 267                             struct cpuidle_driver *drv, int index)
                                                   >> 268 {
                                                   >> 269         arch_cpu_idle();
                                                   >> 270         return index;
 60 }                                                 271 }
                                                   >> 272 
                                                   >> 273 #endif
 61                                                   274 

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

kernel.org | git.kernel.org | LWN.net | Project Home | SVN repository | Mail admin

Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.

sflogo.php