patch-1.3.71 linux/arch/sparc/kernel/time.c

Next file: linux/arch/sparc/kernel/traps.c
Previous file: linux/arch/sparc/kernel/systbls.S
Back to the patch index
Back to the overall index

diff -u --recursive --new-file v1.3.70/linux/arch/sparc/kernel/time.c linux/arch/sparc/kernel/time.c
@@ -1,4 +1,4 @@
-/* $Id: time.c,v 1.4 1995/11/25 03:29:31 davem Exp $
+/* $Id: time.c,v 1.7 1996/03/01 07:16:05 davem Exp $
  * linux/arch/sparc/kernel/time.c
  *
  * Copyright (C) 1995 David S. Miller (davem@caip.rutgers.edu)
@@ -17,16 +17,26 @@
 #include <asm/segment.h>
 #include <asm/timer.h>
 #include <asm/mostek.h>
+#include <asm/system.h>
+#include <asm/irq.h>
+#include <asm/io.h>
 
 #define TIMER_IRQ  10    /* Also at level 14, but we ignore that one. */
 
+enum sparc_clock_type sp_clock_typ;
+struct mostek48t02 *mstk48t02_regs = 0;
+struct mostek48t08 *mstk48t08_regs = 0;
+volatile unsigned int *master_l10_limit = 0;
+volatile unsigned int *master_l10_counter = 0;
+struct sun4m_timer_regs *sun4m_timers;
+
 static int set_rtc_mmss(unsigned long);
 
 /*
  * timer_interrupt() needs to keep up the real-time clock,
  * as well as call the "do_timer()" routine every clocktick
  */
-void timer_interrupt(int irq, struct pt_regs * regs)
+void timer_interrupt(int irq, void *dev_id, struct pt_regs * regs)
 {
 	/* last time the cmos clock got updated */
 	static long last_rtc_update=0;
@@ -78,6 +88,141 @@
 	  )*60 + sec; /* finally seconds */
 }
 
+/* Clock probing, we probe the timers here also. */
+volatile unsigned int foo_limit;
+
+static void clock_probe(void)
+{
+	char node_str[128];
+	register int node, type;
+	struct linux_prom_registers clk_reg[2];
+
+	/* This will basically traverse the node-tree of the prom to see
+	 * which timer chip is on this machine.
+	 */
+
+	node = 0;
+	if(sparc_cpu_model == sun4) {
+		printk("clock_probe: No SUN4 Clock/Timer support yet...\n");
+		return;
+	}
+	if(sparc_cpu_model == sun4c) node=prom_getchild(prom_root_node);
+	else
+		if(sparc_cpu_model == sun4m)
+			node=prom_getchild(prom_searchsiblings(prom_getchild(prom_root_node), "obio"));
+	type = 0;
+	sp_clock_typ = MSTK_INVALID;
+	for(;;) {
+		prom_getstring(node, "model", node_str, sizeof(node_str));
+		if(strcmp(node_str, "mk48t02") == 0) {
+			sp_clock_typ = MSTK48T02;
+			if(prom_getproperty(node, "reg", (char *) clk_reg, sizeof(clk_reg)) == -1) {
+				printk("clock_probe: FAILED!\n");
+				halt();
+			}
+			prom_apply_obio_ranges(clk_reg, 1);
+			/* Map the clock register io area read-only */
+			mstk48t02_regs = (struct mostek48t02 *) 
+				sparc_alloc_io((void *) clk_reg[0].phys_addr,
+					       (void *) 0, sizeof(*mstk48t02_regs),
+					       "clock", clk_reg[0].which_io, 0x0);
+			mstk48t08_regs = 0;  /* To catch weirdness */
+			break;
+		}
+
+		if(strcmp(node_str, "mk48t08") == 0) {
+			sp_clock_typ = MSTK48T08;
+			if(prom_getproperty(node, "reg", (char *) clk_reg,
+					    sizeof(clk_reg)) == -1) {
+				printk("clock_probe: FAILED!\n");
+				halt();
+			}
+			prom_apply_obio_ranges(clk_reg, 1);
+			/* Map the clock register io area read-only */
+			mstk48t08_regs = (struct mostek48t08 *)
+				sparc_alloc_io((void *) clk_reg[0].phys_addr,
+					       (void *) 0, sizeof(*mstk48t08_regs),
+					       "clock", clk_reg[0].which_io, 0x0);
+
+			mstk48t02_regs = &mstk48t08_regs->regs;
+			break;
+		}
+
+		node = prom_getsibling(node);
+		if(node == 0) {
+			printk("Aieee, could not find timer chip type\n");
+			return;
+		}
+	}
+
+	if(sparc_cpu_model == sun4c) {
+		/* Map the Timer chip, this is implemented in hardware inside
+		 * the cache chip on the sun4c.
+		 */
+		sun4c_timers = sparc_alloc_io ((void *) SUN4C_TIMER_PHYSADDR, 0,
+					       sizeof(struct sun4c_timer_info),
+					       "timer", 0x0, 0x0);
+
+		/* Have the level 10 timer tick at 100HZ.  We don't touch the
+		 * level 14 timer limit since we are letting the prom handle
+		 * them until we have a real console driver so L1-A works.
+		 */
+		sun4c_timers->timer_limit10 = (((1000000/HZ) + 1) << 10);
+		master_l10_limit = &(sun4c_timers->timer_limit10);
+		master_l10_counter = &(sun4c_timers->cur_count10);
+	} else {
+		/* XXX FIx this SHIT... UP and MP sun4m configurations
+		 * XXX have completely different layouts for the counter
+		 * XXX registers. AIEEE!!!
+		 */
+
+		int reg_count;
+		struct linux_prom_registers cnt_regs[PROMREG_MAX];
+		volatile unsigned long *real_limit;
+		int obio_node, cnt_node;
+
+		cnt_node = 0;
+		if((obio_node =
+		    prom_searchsiblings (prom_getchild(prom_root_node), "obio")) == 0 ||
+		   (obio_node = prom_getchild (obio_node)) == 0 ||
+		   (cnt_node = prom_searchsiblings (obio_node, "counter")) == 0) {
+			prom_printf("Cannot find /obio/counter node\n");
+			prom_halt();
+		}
+		reg_count = prom_getproperty(cnt_node, "reg",
+					     (void *) cnt_regs, sizeof(cnt_regs));
+		reg_count = (reg_count/sizeof(struct linux_prom_registers));
+
+		/* Apply the obio ranges to the timer registers. */
+		prom_apply_obio_ranges(cnt_regs, reg_count);
+
+		/* Map the per-cpu Counter registers. */
+		sparc_alloc_io(cnt_regs[0].phys_addr, 0,
+			       PAGE_SIZE*NCPUS, "counters_percpu",
+			       cnt_regs[0].which_io, 0x0);
+
+		/* Map the system Counter register. */
+		sun4m_timers = sparc_alloc_io(cnt_regs[reg_count-1].phys_addr, 0,
+					      cnt_regs[reg_count-1].reg_size,
+					      "counters_system",
+					      cnt_regs[reg_count-1].which_io, 0x0);
+
+		real_limit = &sun4m_timers->l10_timer_limit;
+		if(reg_count < 4) {
+			/* Uniprocessor timers, ugh. */
+			real_limit = (volatile unsigned long *) sun4m_timers;
+		}
+
+		/* Avoid interrupt bombs... */
+		foo_limit = (volatile) *real_limit;
+
+		/* Must set the master pointer first or we will lose badly. */
+		master_l10_limit = real_limit;
+		master_l10_counter = real_limit + 1;
+		*master_l10_limit =  (((1000000/HZ) + 1) << 10);
+	}
+}
+
 #ifndef BCD_TO_BIN
 #define BCD_TO_BIN(val) (((val)&15) + ((val)>>4)*10)
 #endif
@@ -91,7 +236,9 @@
 	unsigned int year, mon, day, hour, min, sec;
 	struct mostek48t02 *mregs;
 
-	request_irq(TIMER_IRQ, timer_interrupt, SA_INTERRUPT, "timer");
+	clock_probe();
+	/*	request_irq(TIMER_IRQ, timer_interrupt, SA_INTERRUPT, "timer", NULL); */
+	enable_irq(TIMER_IRQ);
 	mregs = mstk48t02_regs;
 	if(!mregs) {
 		prom_printf("Something wrong, clock regs not mapped yet.\n");

FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen, slshen@lbl.gov with Sam's (original) version
of this