/* * linux/arch/sh/kernel/rtc.c -- SH3 / SH4 on-chip RTC support * * Copyright (C) 2000 Philipp Rumpf * Copyright (C) 1999 Tetsuya Okada & Niibe Yutaka */ #include #include #include #include #include #include #ifndef BCD_TO_BIN #define BCD_TO_BIN(val) ((val)=((val)&15) + ((val)>>4)*10) #endif #ifndef BIN_TO_BCD #define BIN_TO_BCD(val) ((val)=(((val)/10)<<4) + (val)%10) #endif void sh_rtc_gettimeofday(struct timespec *ts) { unsigned int sec128, sec, sec2, min, hr, wk, day, mon, yr, yr100, cf_bit; unsigned long flags; again: do { local_irq_save(flags); ctrl_outb(0, RCR1); /* Clear CF-bit */ sec128 = ctrl_inb(R64CNT); sec = ctrl_inb(RSECCNT); min = ctrl_inb(RMINCNT); hr = ctrl_inb(RHRCNT); wk = ctrl_inb(RWKCNT); day = ctrl_inb(RDAYCNT); mon = ctrl_inb(RMONCNT); #if defined(CONFIG_CPU_SH4) yr = ctrl_inw(RYRCNT); yr100 = (yr >> 8); yr &= 0xff; #else yr = ctrl_inb(RYRCNT); yr100 = (yr == 0x99) ? 0x19 : 0x20; #endif sec2 = ctrl_inb(R64CNT); cf_bit = ctrl_inb(RCR1) & RCR1_CF; local_irq_restore(flags); } while (cf_bit != 0 || ((sec128 ^ sec2) & RTC_BIT_INVERTED) != 0); BCD_TO_BIN(yr100); BCD_TO_BIN(yr); BCD_TO_BIN(mon); BCD_TO_BIN(day); BCD_TO_BIN(hr); BCD_TO_BIN(min); BCD_TO_BIN(sec); if (yr > 99 || mon < 1 || mon > 12 || day > 31 || day < 1 || hr > 23 || min > 59 || sec > 59) { printk(KERN_ERR "SH RTC: invalid value, resetting to 1 Jan 2000\n"); local_irq_save(flags); ctrl_outb(RCR2_RESET, RCR2); /* Reset & Stop */ ctrl_outb(0, RSECCNT); ctrl_outb(0, RMINCNT); ctrl_outb(0, RHRCNT); ctrl_outb(6, RWKCNT); ctrl_outb(1, RDAYCNT); ctrl_outb(1, RMONCNT); #if defined(CONFIG_CPU_SH4) ctrl_outw(0x2000, RYRCNT); #else ctrl_outb(0, RYRCNT); #endif ctrl_outb(RCR2_RTCEN|RCR2_START, RCR2); /* Start */ goto again; } #if RTC_BIT_INVERTED != 0 if ((sec128 & RTC_BIT_INVERTED)) sec--; #endif ts->tv_sec = mktime(yr100 * 100 + yr, mon, day, hr, min, sec); ts->tv_nsec = ((sec128 * 1000000) / 128) * 1000; } /* * Changed to only care about tv_sec, and not the full timespec struct * (i.e. tv_nsec). It can easily be switched to timespec for future cpus * that support setting usec or nsec RTC values. */ int sh_rtc_settimeofday(const time_t secs) { int retval = 0; int real_seconds, real_minutes, cmos_minutes; unsigned long flags; local_irq_save(flags); ctrl_outb(RCR2_RESET, RCR2); /* Reset pre-scaler & stop RTC */ cmos_minutes = ctrl_inb(RMINCNT); BCD_TO_BIN(cmos_minutes); /* * since we're only adjusting minutes and seconds, * don't interfere with hour overflow. This avoids * messing with unknown time zones but requires your * RTC not to be off by more than 15 minutes */ real_seconds = secs % 60; real_minutes = secs / 60; if (((abs(real_minutes - cmos_minutes) + 15)/30) & 1) real_minutes += 30; /* correct for half hour time zone */ real_minutes %= 60; if (abs(real_minutes - cmos_minutes) < 30) { BIN_TO_BCD(real_seconds); BIN_TO_BCD(real_minutes); ctrl_outb(real_seconds, RSECCNT); ctrl_outb(real_minutes, RMINCNT); } else { printk(KERN_WARNING "set_rtc_time: can't update from %d to %d\n", cmos_minutes, real_minutes); retval = -1; } ctrl_outb(RCR2_RTCEN|RCR2_START, RCR2); /* Start RTC */ local_irq_restore(flags); return retval; }