URL
https://opencores.org/ocsvn/or1k/or1k/trunk
Subversion Repositories or1k
[/] [or1k/] [trunk/] [rc203soc/] [sw/] [uClinux/] [arch/] [sparc/] [kernel/] [time.c] - Rev 1765
Compare with Previous | Blame | View Log
/* $Id: time.c,v 1.1 2005-12-20 09:50:43 jcastillo Exp $ * linux/arch/sparc/kernel/time.c * * Copyright (C) 1995 David S. Miller (davem@caip.rutgers.edu) * * This file handles the Sparc specific time handling details. */ #include <linux/config.h> #include <linux/errno.h> #include <linux/sched.h> #include <linux/kernel.h> #include <linux/param.h> #include <linux/string.h> #include <linux/mm.h> #include <linux/timex.h> #include <asm/oplib.h> #include <asm/segment.h> #include <asm/timer.h> #include <asm/mostek.h> #include <asm/system.h> #include <asm/irq.h> #include <asm/io.h> enum sparc_clock_type sp_clock_typ; struct mostek48t02 *mstk48t02_regs = 0; struct mostek48t08 *mstk48t08_regs = 0; 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, void *dev_id, struct pt_regs * regs) { /* last time the cmos clock got updated */ static long last_rtc_update=0; clear_clock_irq(); do_timer(regs); /* XXX I don't know if this is right for the Sparc yet. XXX */ if (time_state != TIME_BAD && xtime.tv_sec > last_rtc_update + 660 && xtime.tv_usec > 500000 - (tick >> 1) && xtime.tv_usec < 500000 + (tick >> 1)) if (set_rtc_mmss(xtime.tv_sec) == 0) last_rtc_update = xtime.tv_sec; else last_rtc_update = xtime.tv_sec - 600; /* do it again in 60 s */ } /* Converts Gregorian date to seconds since 1970-01-01 00:00:00. * Assumes input in normal date format, i.e. 1980-12-31 23:59:59 * => year=1980, mon=12, day=31, hour=23, min=59, sec=59. * * [For the Julian calendar (which was used in Russia before 1917, * Britain & colonies before 1752, anywhere else before 1582, * and is still in use by some communities) leave out the * -year/100+year/400 terms, and add 10.] * * This algorithm was first published by Gauss (I think). * * WARNING: this function will overflow on 2106-02-07 06:28:16 on * machines were long is 32-bit! (However, as time_t is signed, we * will already get problems at other places on 2038-01-19 03:14:08) */ static inline unsigned long mktime(unsigned int year, unsigned int mon, unsigned int day, unsigned int hour, unsigned int min, unsigned int sec) { if (0 >= (int) (mon -= 2)) { /* 1..12 -> 11,12,1..10 */ mon += 12; /* Puts Feb last since it has leap day */ year -= 1; } return ((( (unsigned long)(year/4 - year/100 + year/400 + 367*mon/12 + day) + year*365 - 719499 )*24 + hour /* now have hours */ )*60 + min /* now have minutes */ )*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; } } } #ifndef BCD_TO_BIN #define BCD_TO_BIN(val) (((val)&15) + ((val)>>4)*10) #endif #ifndef BIN_TO_BCD #define BIN_TO_BCD(val) ((((val)/10)<<4) + (val)%10) #endif void time_init(void) { unsigned int year, mon, day, hour, min, sec; struct mostek48t02 *mregs; #if CONFIG_AP1000 init_timers(timer_interrupt); return; #endif clock_probe(); init_timers(timer_interrupt); mregs = mstk48t02_regs; if(!mregs) { prom_printf("Something wrong, clock regs not mapped yet.\n"); prom_halt(); } mregs->creg |= MSTK_CREG_READ; sec = BCD_TO_BIN(mregs->sec); min = BCD_TO_BIN(mregs->min); hour = BCD_TO_BIN(mregs->hour); day = BCD_TO_BIN(mregs->dom); mon = BCD_TO_BIN(mregs->mnth); year = (BCD_TO_BIN(mregs->yr) + MSTK_YR_ZERO); xtime.tv_sec = mktime(year, mon, day, hour, min, sec); xtime.tv_usec = 0; mregs->creg &= ~MSTK_CREG_READ; return; } /* Nothing fancy on the Sparc yet. */ void do_gettimeofday(struct timeval *tv) { unsigned long flags; save_flags(flags); cli(); #if CONFIG_AP1000 ap_gettimeofday(&xtime); #endif *tv = xtime; restore_flags(flags); } void do_settimeofday(struct timeval *tv) { cli(); xtime = *tv; time_state = TIME_BAD; time_maxerror = 0x70000000; time_esterror = 0x70000000; sti(); } static int set_rtc_mmss(unsigned long nowtime) { int retval = 0; int real_seconds, real_minutes, mostek_minutes; struct mostek48t02 *mregs = mstk48t02_regs; if(!mregs) retval = -1; else { mregs->creg |= MSTK_CREG_READ; mostek_minutes = BCD_TO_BIN(mregs->min); mregs->creg &= ~MSTK_CREG_READ; real_seconds = nowtime % 60; real_minutes = nowtime / 60; if (((abs(real_minutes - mostek_minutes) + 15)/30) & 1) real_minutes += 30; real_minutes %= 60; if (abs(real_minutes - mostek_minutes) < 30) { mregs->creg |= MSTK_CREG_WRITE; mregs->sec = real_seconds; mregs->min = real_minutes; mregs->creg &= ~MSTK_CREG_WRITE; } else retval = -1; } return retval; }