Merge commit 'v2.6.28-rc2' into core/locking
[linux-2.6] / arch / blackfin / kernel / reboot.c
1 /*
2  * arch/blackfin/kernel/reboot.c - handle shutdown/reboot
3  *
4  * Copyright 2004-2007 Analog Devices Inc.
5  *
6  * Licensed under the GPL-2 or later.
7  */
8
9 #include <linux/interrupt.h>
10 #include <asm/bfin-global.h>
11 #include <asm/reboot.h>
12 #include <asm/system.h>
13 #include <asm/bfrom.h>
14
15 /* A system soft reset makes external memory unusable so force
16  * this function into L1.  We use the compiler ssync here rather
17  * than SSYNC() because it's safe (no interrupts and such) and
18  * we save some L1.  We do not need to force sanity in the SYSCR
19  * register as the BMODE selection bit is cleared by the soft
20  * reset while the Core B bit (on dual core parts) is cleared by
21  * the core reset.
22  */
23 __attribute__((l1_text))
24 static void bfin_reset(void)
25 {
26         /* Wait for completion of "system" events such as cache line
27          * line fills so that we avoid infinite stalls later on as
28          * much as possible.  This code is in L1, so it won't trigger
29          * any such event after this point in time.
30          */
31         __builtin_bfin_ssync();
32
33         while (1) {
34                 /* Initiate System software reset. */
35                 bfin_write_SWRST(0x7);
36
37                 /* Due to the way reset is handled in the hardware, we need
38                  * to delay for 10 SCLKS.  The only reliable way to do this is
39                  * to calculate the CCLK/SCLK ratio and multiply 10.  For now,
40                  * we'll assume worse case which is a 1:15 ratio.
41                  */
42                 asm(
43                         "LSETUP (1f, 1f) LC0 = %0\n"
44                         "1: nop;"
45                         :
46                         : "a" (15 * 10)
47                         : "LC0", "LB0", "LT0"
48                 );
49
50                 /* Clear System software reset */
51                 bfin_write_SWRST(0);
52
53                 /* Wait for the SWRST write to complete.  Cannot rely on SSYNC
54                  * though as the System state is all reset now.
55                  */
56                 asm(
57                         "LSETUP (1f, 1f) LC1 = %0\n"
58                         "1: nop;"
59                         :
60                         : "a" (15 * 1)
61                         : "LC1", "LB1", "LT1"
62                 );
63
64                 /* Issue core reset */
65                 asm("raise 1");
66         }
67 }
68
69 __attribute__((weak))
70 void native_machine_restart(char *cmd)
71 {
72 }
73
74 void machine_restart(char *cmd)
75 {
76         native_machine_restart(cmd);
77         local_irq_disable();
78         if (ANOMALY_05000353 || ANOMALY_05000386)
79                 bfin_reset();
80         else
81                 /* the bootrom checks to see how it was reset and will
82                  * automatically perform a software reset for us when
83                  * it starts executing boot
84                  */
85                 asm("raise 1;");
86 }
87
88 __attribute__((weak))
89 void native_machine_halt(void)
90 {
91         idle_with_irq_disabled();
92 }
93
94 void machine_halt(void)
95 {
96         native_machine_halt();
97 }
98
99 __attribute__((weak))
100 void native_machine_power_off(void)
101 {
102         idle_with_irq_disabled();
103 }
104
105 void machine_power_off(void)
106 {
107         native_machine_power_off();
108 }