| /* tadpole.c: Probing for the tadpole clock stopping h/w at boot time. |
| * |
| * Copyright (C) 1996 David Redman (djhr@tadpole.co.uk) |
| */ |
| |
| #include <linux/string.h> |
| #include <linux/kernel.h> |
| #include <linux/sched.h> |
| #include <linux/init.h> |
| |
| #include <asm/asi.h> |
| #include <asm/oplib.h> |
| #include <asm/io.h> |
| |
| #define MACIO_SCSI_CSR_ADDR 0x78400000 |
| #define MACIO_EN_DMA 0x00000200 |
| #define CLOCK_INIT_DONE 1 |
| |
| static int clk_state; |
| static volatile unsigned char *clk_ctrl; |
| void (*cpu_pwr_save)(void); |
| |
| static inline unsigned int ldphys(unsigned int addr) |
| { |
| unsigned long data; |
| |
| __asm__ __volatile__("\n\tlda [%1] %2, %0\n\t" : |
| "=r" (data) : |
| "r" (addr), "i" (ASI_M_BYPASS)); |
| return data; |
| } |
| |
| static void clk_init(void) |
| { |
| __asm__ __volatile__("mov 0x6c, %%g1\n\t" |
| "mov 0x4c, %%g2\n\t" |
| "mov 0xdf, %%g3\n\t" |
| "stb %%g1, [%0+3]\n\t" |
| "stb %%g2, [%0+3]\n\t" |
| "stb %%g3, [%0+3]\n\t" : : |
| "r" (clk_ctrl) : |
| "g1", "g2", "g3"); |
| } |
| |
| static void clk_slow(void) |
| { |
| __asm__ __volatile__("mov 0xcc, %%g2\n\t" |
| "mov 0x4c, %%g3\n\t" |
| "mov 0xcf, %%g4\n\t" |
| "mov 0xdf, %%g5\n\t" |
| "stb %%g2, [%0+3]\n\t" |
| "stb %%g3, [%0+3]\n\t" |
| "stb %%g4, [%0+3]\n\t" |
| "stb %%g5, [%0+3]\n\t" : : |
| "r" (clk_ctrl) : |
| "g2", "g3", "g4", "g5"); |
| } |
| |
| /* |
| * Tadpole is guaranteed to be UP, using local_irq_save. |
| */ |
| static void tsu_clockstop(void) |
| { |
| unsigned int mcsr; |
| unsigned long flags; |
| |
| if (!clk_ctrl) |
| return; |
| if (!(clk_state & CLOCK_INIT_DONE)) { |
| local_irq_save(flags); |
| clk_init(); |
| clk_state |= CLOCK_INIT_DONE; /* all done */ |
| local_irq_restore(flags); |
| return; |
| } |
| if (!(clk_ctrl[2] & 1)) |
| return; /* no speed up yet */ |
| |
| local_irq_save(flags); |
| |
| /* if SCSI DMA in progress, don't slow clock */ |
| mcsr = ldphys(MACIO_SCSI_CSR_ADDR); |
| if ((mcsr&MACIO_EN_DMA) != 0) { |
| local_irq_restore(flags); |
| return; |
| } |
| /* TODO... the minimum clock setting ought to increase the |
| * memory refresh interval.. |
| */ |
| clk_slow(); |
| local_irq_restore(flags); |
| } |
| |
| static void swift_clockstop(void) |
| { |
| if (!clk_ctrl) |
| return; |
| clk_ctrl[0] = 0; |
| } |
| |
| void __init clock_stop_probe(void) |
| { |
| unsigned int node, clk_nd; |
| char name[20]; |
| |
| prom_getstring(prom_root_node, "name", name, sizeof(name)); |
| if (strncmp(name, "Tadpole", 7)) |
| return; |
| node = prom_getchild(prom_root_node); |
| node = prom_searchsiblings(node, "obio"); |
| node = prom_getchild(node); |
| clk_nd = prom_searchsiblings(node, "clk-ctrl"); |
| if (!clk_nd) |
| return; |
| printk("Clock Stopping h/w detected... "); |
| clk_ctrl = (char *) prom_getint(clk_nd, "address"); |
| clk_state = 0; |
| if (name[10] == '\0') { |
| cpu_pwr_save = tsu_clockstop; |
| printk("enabled (S3)\n"); |
| } else if ((name[10] == 'X') || (name[10] == 'G')) { |
| cpu_pwr_save = swift_clockstop; |
| printk("enabled (%s)\n",name+7); |
| } else |
| printk("disabled %s\n",name+7); |
| } |