162306a36Sopenharmony_ci/* 262306a36Sopenharmony_ci * HP i8042 SDC + MSM-58321 BBRTC driver. 362306a36Sopenharmony_ci * 462306a36Sopenharmony_ci * Copyright (c) 2001 Brian S. Julin 562306a36Sopenharmony_ci * All rights reserved. 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * Redistribution and use in source and binary forms, with or without 862306a36Sopenharmony_ci * modification, are permitted provided that the following conditions 962306a36Sopenharmony_ci * are met: 1062306a36Sopenharmony_ci * 1. Redistributions of source code must retain the above copyright 1162306a36Sopenharmony_ci * notice, this list of conditions, and the following disclaimer, 1262306a36Sopenharmony_ci * without modification. 1362306a36Sopenharmony_ci * 2. The name of the author may not be used to endorse or promote products 1462306a36Sopenharmony_ci * derived from this software without specific prior written permission. 1562306a36Sopenharmony_ci * 1662306a36Sopenharmony_ci * Alternatively, this software may be distributed under the terms of the 1762306a36Sopenharmony_ci * GNU General Public License ("GPL"). 1862306a36Sopenharmony_ci * 1962306a36Sopenharmony_ci * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 2062306a36Sopenharmony_ci * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 2162306a36Sopenharmony_ci * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 2262306a36Sopenharmony_ci * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR 2362306a36Sopenharmony_ci * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 2462306a36Sopenharmony_ci * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 2562306a36Sopenharmony_ci * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 2662306a36Sopenharmony_ci * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 2762306a36Sopenharmony_ci * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 2862306a36Sopenharmony_ci * 2962306a36Sopenharmony_ci * References: 3062306a36Sopenharmony_ci * System Device Controller Microprocessor Firmware Theory of Operation 3162306a36Sopenharmony_ci * for Part Number 1820-4784 Revision B. Dwg No. A-1820-4784-2 3262306a36Sopenharmony_ci * efirtc.c by Stephane Eranian/Hewlett Packard 3362306a36Sopenharmony_ci * 3462306a36Sopenharmony_ci */ 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_ci#include <linux/hp_sdc.h> 3762306a36Sopenharmony_ci#include <linux/errno.h> 3862306a36Sopenharmony_ci#include <linux/types.h> 3962306a36Sopenharmony_ci#include <linux/init.h> 4062306a36Sopenharmony_ci#include <linux/module.h> 4162306a36Sopenharmony_ci#include <linux/time.h> 4262306a36Sopenharmony_ci#include <linux/miscdevice.h> 4362306a36Sopenharmony_ci#include <linux/proc_fs.h> 4462306a36Sopenharmony_ci#include <linux/seq_file.h> 4562306a36Sopenharmony_ci#include <linux/poll.h> 4662306a36Sopenharmony_ci#include <linux/rtc.h> 4762306a36Sopenharmony_ci#include <linux/mutex.h> 4862306a36Sopenharmony_ci#include <linux/semaphore.h> 4962306a36Sopenharmony_ci 5062306a36Sopenharmony_ciMODULE_AUTHOR("Brian S. Julin <bri@calyx.com>"); 5162306a36Sopenharmony_ciMODULE_DESCRIPTION("HP i8042 SDC + MSM-58321 RTC Driver"); 5262306a36Sopenharmony_ciMODULE_LICENSE("Dual BSD/GPL"); 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_ci#define RTC_VERSION "1.10d" 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_cistatic unsigned long epoch = 2000; 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_cistatic struct semaphore i8042tregs; 5962306a36Sopenharmony_ci 6062306a36Sopenharmony_cistatic void hp_sdc_rtc_isr (int irq, void *dev_id, 6162306a36Sopenharmony_ci uint8_t status, uint8_t data) 6262306a36Sopenharmony_ci{ 6362306a36Sopenharmony_ci return; 6462306a36Sopenharmony_ci} 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_cistatic int hp_sdc_rtc_do_read_bbrtc (struct rtc_time *rtctm) 6762306a36Sopenharmony_ci{ 6862306a36Sopenharmony_ci struct semaphore tsem; 6962306a36Sopenharmony_ci hp_sdc_transaction t; 7062306a36Sopenharmony_ci uint8_t tseq[91]; 7162306a36Sopenharmony_ci int i; 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci i = 0; 7462306a36Sopenharmony_ci while (i < 91) { 7562306a36Sopenharmony_ci tseq[i++] = HP_SDC_ACT_DATAREG | 7662306a36Sopenharmony_ci HP_SDC_ACT_POSTCMD | HP_SDC_ACT_DATAIN; 7762306a36Sopenharmony_ci tseq[i++] = 0x01; /* write i8042[0x70] */ 7862306a36Sopenharmony_ci tseq[i] = i / 7; /* BBRTC reg address */ 7962306a36Sopenharmony_ci i++; 8062306a36Sopenharmony_ci tseq[i++] = HP_SDC_CMD_DO_RTCR; /* Trigger command */ 8162306a36Sopenharmony_ci tseq[i++] = 2; /* expect 1 stat/dat pair back. */ 8262306a36Sopenharmony_ci i++; i++; /* buffer for stat/dat pair */ 8362306a36Sopenharmony_ci } 8462306a36Sopenharmony_ci tseq[84] |= HP_SDC_ACT_SEMAPHORE; 8562306a36Sopenharmony_ci t.endidx = 91; 8662306a36Sopenharmony_ci t.seq = tseq; 8762306a36Sopenharmony_ci t.act.semaphore = &tsem; 8862306a36Sopenharmony_ci sema_init(&tsem, 0); 8962306a36Sopenharmony_ci 9062306a36Sopenharmony_ci if (hp_sdc_enqueue_transaction(&t)) return -1; 9162306a36Sopenharmony_ci 9262306a36Sopenharmony_ci /* Put ourselves to sleep for results. */ 9362306a36Sopenharmony_ci if (WARN_ON(down_interruptible(&tsem))) 9462306a36Sopenharmony_ci return -1; 9562306a36Sopenharmony_ci 9662306a36Sopenharmony_ci /* Check for nonpresence of BBRTC */ 9762306a36Sopenharmony_ci if (!((tseq[83] | tseq[90] | tseq[69] | tseq[76] | 9862306a36Sopenharmony_ci tseq[55] | tseq[62] | tseq[34] | tseq[41] | 9962306a36Sopenharmony_ci tseq[20] | tseq[27] | tseq[6] | tseq[13]) & 0x0f)) 10062306a36Sopenharmony_ci return -1; 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_ci memset(rtctm, 0, sizeof(struct rtc_time)); 10362306a36Sopenharmony_ci rtctm->tm_year = (tseq[83] & 0x0f) + (tseq[90] & 0x0f) * 10; 10462306a36Sopenharmony_ci rtctm->tm_mon = (tseq[69] & 0x0f) + (tseq[76] & 0x0f) * 10; 10562306a36Sopenharmony_ci rtctm->tm_mday = (tseq[55] & 0x0f) + (tseq[62] & 0x0f) * 10; 10662306a36Sopenharmony_ci rtctm->tm_wday = (tseq[48] & 0x0f); 10762306a36Sopenharmony_ci rtctm->tm_hour = (tseq[34] & 0x0f) + (tseq[41] & 0x0f) * 10; 10862306a36Sopenharmony_ci rtctm->tm_min = (tseq[20] & 0x0f) + (tseq[27] & 0x0f) * 10; 10962306a36Sopenharmony_ci rtctm->tm_sec = (tseq[6] & 0x0f) + (tseq[13] & 0x0f) * 10; 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_ci return 0; 11262306a36Sopenharmony_ci} 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_cistatic int hp_sdc_rtc_read_bbrtc (struct rtc_time *rtctm) 11562306a36Sopenharmony_ci{ 11662306a36Sopenharmony_ci struct rtc_time tm, tm_last; 11762306a36Sopenharmony_ci int i = 0; 11862306a36Sopenharmony_ci 11962306a36Sopenharmony_ci /* MSM-58321 has no read latch, so must read twice and compare. */ 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ci if (hp_sdc_rtc_do_read_bbrtc(&tm_last)) return -1; 12262306a36Sopenharmony_ci if (hp_sdc_rtc_do_read_bbrtc(&tm)) return -1; 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_ci while (memcmp(&tm, &tm_last, sizeof(struct rtc_time))) { 12562306a36Sopenharmony_ci if (i++ > 4) return -1; 12662306a36Sopenharmony_ci memcpy(&tm_last, &tm, sizeof(struct rtc_time)); 12762306a36Sopenharmony_ci if (hp_sdc_rtc_do_read_bbrtc(&tm)) return -1; 12862306a36Sopenharmony_ci } 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_ci memcpy(rtctm, &tm, sizeof(struct rtc_time)); 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci return 0; 13362306a36Sopenharmony_ci} 13462306a36Sopenharmony_ci 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_cistatic int64_t hp_sdc_rtc_read_i8042timer (uint8_t loadcmd, int numreg) 13762306a36Sopenharmony_ci{ 13862306a36Sopenharmony_ci hp_sdc_transaction t; 13962306a36Sopenharmony_ci uint8_t tseq[26] = { 14062306a36Sopenharmony_ci HP_SDC_ACT_PRECMD | HP_SDC_ACT_POSTCMD | HP_SDC_ACT_DATAIN, 14162306a36Sopenharmony_ci 0, 14262306a36Sopenharmony_ci HP_SDC_CMD_READ_T1, 2, 0, 0, 14362306a36Sopenharmony_ci HP_SDC_ACT_POSTCMD | HP_SDC_ACT_DATAIN, 14462306a36Sopenharmony_ci HP_SDC_CMD_READ_T2, 2, 0, 0, 14562306a36Sopenharmony_ci HP_SDC_ACT_POSTCMD | HP_SDC_ACT_DATAIN, 14662306a36Sopenharmony_ci HP_SDC_CMD_READ_T3, 2, 0, 0, 14762306a36Sopenharmony_ci HP_SDC_ACT_POSTCMD | HP_SDC_ACT_DATAIN, 14862306a36Sopenharmony_ci HP_SDC_CMD_READ_T4, 2, 0, 0, 14962306a36Sopenharmony_ci HP_SDC_ACT_POSTCMD | HP_SDC_ACT_DATAIN, 15062306a36Sopenharmony_ci HP_SDC_CMD_READ_T5, 2, 0, 0 15162306a36Sopenharmony_ci }; 15262306a36Sopenharmony_ci 15362306a36Sopenharmony_ci t.endidx = numreg * 5; 15462306a36Sopenharmony_ci 15562306a36Sopenharmony_ci tseq[1] = loadcmd; 15662306a36Sopenharmony_ci tseq[t.endidx - 4] |= HP_SDC_ACT_SEMAPHORE; /* numreg assumed > 1 */ 15762306a36Sopenharmony_ci 15862306a36Sopenharmony_ci t.seq = tseq; 15962306a36Sopenharmony_ci t.act.semaphore = &i8042tregs; 16062306a36Sopenharmony_ci 16162306a36Sopenharmony_ci /* Sleep if output regs in use. */ 16262306a36Sopenharmony_ci if (WARN_ON(down_interruptible(&i8042tregs))) 16362306a36Sopenharmony_ci return -1; 16462306a36Sopenharmony_ci 16562306a36Sopenharmony_ci if (hp_sdc_enqueue_transaction(&t)) { 16662306a36Sopenharmony_ci up(&i8042tregs); 16762306a36Sopenharmony_ci return -1; 16862306a36Sopenharmony_ci } 16962306a36Sopenharmony_ci 17062306a36Sopenharmony_ci /* Sleep until results come back. */ 17162306a36Sopenharmony_ci if (WARN_ON(down_interruptible(&i8042tregs))) 17262306a36Sopenharmony_ci return -1; 17362306a36Sopenharmony_ci 17462306a36Sopenharmony_ci up(&i8042tregs); 17562306a36Sopenharmony_ci 17662306a36Sopenharmony_ci return (tseq[5] | 17762306a36Sopenharmony_ci ((uint64_t)(tseq[10]) << 8) | ((uint64_t)(tseq[15]) << 16) | 17862306a36Sopenharmony_ci ((uint64_t)(tseq[20]) << 24) | ((uint64_t)(tseq[25]) << 32)); 17962306a36Sopenharmony_ci} 18062306a36Sopenharmony_ci 18162306a36Sopenharmony_ci 18262306a36Sopenharmony_ci/* Read the i8042 real-time clock */ 18362306a36Sopenharmony_cistatic inline int hp_sdc_rtc_read_rt(struct timespec64 *res) { 18462306a36Sopenharmony_ci int64_t raw; 18562306a36Sopenharmony_ci uint32_t tenms; 18662306a36Sopenharmony_ci unsigned int days; 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_ci raw = hp_sdc_rtc_read_i8042timer(HP_SDC_CMD_LOAD_RT, 5); 18962306a36Sopenharmony_ci if (raw < 0) return -1; 19062306a36Sopenharmony_ci 19162306a36Sopenharmony_ci tenms = (uint32_t)raw & 0xffffff; 19262306a36Sopenharmony_ci days = (unsigned int)(raw >> 24) & 0xffff; 19362306a36Sopenharmony_ci 19462306a36Sopenharmony_ci res->tv_nsec = (long)(tenms % 100) * 10000 * 1000; 19562306a36Sopenharmony_ci res->tv_sec = (tenms / 100) + (time64_t)days * 86400; 19662306a36Sopenharmony_ci 19762306a36Sopenharmony_ci return 0; 19862306a36Sopenharmony_ci} 19962306a36Sopenharmony_ci 20062306a36Sopenharmony_ci 20162306a36Sopenharmony_ci/* Read the i8042 fast handshake timer */ 20262306a36Sopenharmony_cistatic inline int hp_sdc_rtc_read_fhs(struct timespec64 *res) { 20362306a36Sopenharmony_ci int64_t raw; 20462306a36Sopenharmony_ci unsigned int tenms; 20562306a36Sopenharmony_ci 20662306a36Sopenharmony_ci raw = hp_sdc_rtc_read_i8042timer(HP_SDC_CMD_LOAD_FHS, 2); 20762306a36Sopenharmony_ci if (raw < 0) return -1; 20862306a36Sopenharmony_ci 20962306a36Sopenharmony_ci tenms = (unsigned int)raw & 0xffff; 21062306a36Sopenharmony_ci 21162306a36Sopenharmony_ci res->tv_nsec = (long)(tenms % 100) * 10000 * 1000; 21262306a36Sopenharmony_ci res->tv_sec = (time64_t)(tenms / 100); 21362306a36Sopenharmony_ci 21462306a36Sopenharmony_ci return 0; 21562306a36Sopenharmony_ci} 21662306a36Sopenharmony_ci 21762306a36Sopenharmony_ci 21862306a36Sopenharmony_ci/* Read the i8042 match timer (a.k.a. alarm) */ 21962306a36Sopenharmony_cistatic inline int hp_sdc_rtc_read_mt(struct timespec64 *res) { 22062306a36Sopenharmony_ci int64_t raw; 22162306a36Sopenharmony_ci uint32_t tenms; 22262306a36Sopenharmony_ci 22362306a36Sopenharmony_ci raw = hp_sdc_rtc_read_i8042timer(HP_SDC_CMD_LOAD_MT, 3); 22462306a36Sopenharmony_ci if (raw < 0) return -1; 22562306a36Sopenharmony_ci 22662306a36Sopenharmony_ci tenms = (uint32_t)raw & 0xffffff; 22762306a36Sopenharmony_ci 22862306a36Sopenharmony_ci res->tv_nsec = (long)(tenms % 100) * 10000 * 1000; 22962306a36Sopenharmony_ci res->tv_sec = (time64_t)(tenms / 100); 23062306a36Sopenharmony_ci 23162306a36Sopenharmony_ci return 0; 23262306a36Sopenharmony_ci} 23362306a36Sopenharmony_ci 23462306a36Sopenharmony_ci 23562306a36Sopenharmony_ci/* Read the i8042 delay timer */ 23662306a36Sopenharmony_cistatic inline int hp_sdc_rtc_read_dt(struct timespec64 *res) { 23762306a36Sopenharmony_ci int64_t raw; 23862306a36Sopenharmony_ci uint32_t tenms; 23962306a36Sopenharmony_ci 24062306a36Sopenharmony_ci raw = hp_sdc_rtc_read_i8042timer(HP_SDC_CMD_LOAD_DT, 3); 24162306a36Sopenharmony_ci if (raw < 0) return -1; 24262306a36Sopenharmony_ci 24362306a36Sopenharmony_ci tenms = (uint32_t)raw & 0xffffff; 24462306a36Sopenharmony_ci 24562306a36Sopenharmony_ci res->tv_nsec = (long)(tenms % 100) * 10000 * 1000; 24662306a36Sopenharmony_ci res->tv_sec = (time64_t)(tenms / 100); 24762306a36Sopenharmony_ci 24862306a36Sopenharmony_ci return 0; 24962306a36Sopenharmony_ci} 25062306a36Sopenharmony_ci 25162306a36Sopenharmony_ci 25262306a36Sopenharmony_ci/* Read the i8042 cycle timer (a.k.a. periodic) */ 25362306a36Sopenharmony_cistatic inline int hp_sdc_rtc_read_ct(struct timespec64 *res) { 25462306a36Sopenharmony_ci int64_t raw; 25562306a36Sopenharmony_ci uint32_t tenms; 25662306a36Sopenharmony_ci 25762306a36Sopenharmony_ci raw = hp_sdc_rtc_read_i8042timer(HP_SDC_CMD_LOAD_CT, 3); 25862306a36Sopenharmony_ci if (raw < 0) return -1; 25962306a36Sopenharmony_ci 26062306a36Sopenharmony_ci tenms = (uint32_t)raw & 0xffffff; 26162306a36Sopenharmony_ci 26262306a36Sopenharmony_ci res->tv_nsec = (long)(tenms % 100) * 10000 * 1000; 26362306a36Sopenharmony_ci res->tv_sec = (time64_t)(tenms / 100); 26462306a36Sopenharmony_ci 26562306a36Sopenharmony_ci return 0; 26662306a36Sopenharmony_ci} 26762306a36Sopenharmony_ci 26862306a36Sopenharmony_cistatic int __maybe_unused hp_sdc_rtc_proc_show(struct seq_file *m, void *v) 26962306a36Sopenharmony_ci{ 27062306a36Sopenharmony_ci#define YN(bit) ("no") 27162306a36Sopenharmony_ci#define NY(bit) ("yes") 27262306a36Sopenharmony_ci struct rtc_time tm; 27362306a36Sopenharmony_ci struct timespec64 tv; 27462306a36Sopenharmony_ci 27562306a36Sopenharmony_ci memset(&tm, 0, sizeof(struct rtc_time)); 27662306a36Sopenharmony_ci 27762306a36Sopenharmony_ci if (hp_sdc_rtc_read_bbrtc(&tm)) { 27862306a36Sopenharmony_ci seq_puts(m, "BBRTC\t\t: READ FAILED!\n"); 27962306a36Sopenharmony_ci } else { 28062306a36Sopenharmony_ci seq_printf(m, 28162306a36Sopenharmony_ci "rtc_time\t: %ptRt\n" 28262306a36Sopenharmony_ci "rtc_date\t: %ptRd\n" 28362306a36Sopenharmony_ci "rtc_epoch\t: %04lu\n", 28462306a36Sopenharmony_ci &tm, &tm, epoch); 28562306a36Sopenharmony_ci } 28662306a36Sopenharmony_ci 28762306a36Sopenharmony_ci if (hp_sdc_rtc_read_rt(&tv)) { 28862306a36Sopenharmony_ci seq_puts(m, "i8042 rtc\t: READ FAILED!\n"); 28962306a36Sopenharmony_ci } else { 29062306a36Sopenharmony_ci seq_printf(m, "i8042 rtc\t: %lld.%02ld seconds\n", 29162306a36Sopenharmony_ci (s64)tv.tv_sec, (long)tv.tv_nsec/1000000L); 29262306a36Sopenharmony_ci } 29362306a36Sopenharmony_ci 29462306a36Sopenharmony_ci if (hp_sdc_rtc_read_fhs(&tv)) { 29562306a36Sopenharmony_ci seq_puts(m, "handshake\t: READ FAILED!\n"); 29662306a36Sopenharmony_ci } else { 29762306a36Sopenharmony_ci seq_printf(m, "handshake\t: %lld.%02ld seconds\n", 29862306a36Sopenharmony_ci (s64)tv.tv_sec, (long)tv.tv_nsec/1000000L); 29962306a36Sopenharmony_ci } 30062306a36Sopenharmony_ci 30162306a36Sopenharmony_ci if (hp_sdc_rtc_read_mt(&tv)) { 30262306a36Sopenharmony_ci seq_puts(m, "alarm\t\t: READ FAILED!\n"); 30362306a36Sopenharmony_ci } else { 30462306a36Sopenharmony_ci seq_printf(m, "alarm\t\t: %lld.%02ld seconds\n", 30562306a36Sopenharmony_ci (s64)tv.tv_sec, (long)tv.tv_nsec/1000000L); 30662306a36Sopenharmony_ci } 30762306a36Sopenharmony_ci 30862306a36Sopenharmony_ci if (hp_sdc_rtc_read_dt(&tv)) { 30962306a36Sopenharmony_ci seq_puts(m, "delay\t\t: READ FAILED!\n"); 31062306a36Sopenharmony_ci } else { 31162306a36Sopenharmony_ci seq_printf(m, "delay\t\t: %lld.%02ld seconds\n", 31262306a36Sopenharmony_ci (s64)tv.tv_sec, (long)tv.tv_nsec/1000000L); 31362306a36Sopenharmony_ci } 31462306a36Sopenharmony_ci 31562306a36Sopenharmony_ci if (hp_sdc_rtc_read_ct(&tv)) { 31662306a36Sopenharmony_ci seq_puts(m, "periodic\t: READ FAILED!\n"); 31762306a36Sopenharmony_ci } else { 31862306a36Sopenharmony_ci seq_printf(m, "periodic\t: %lld.%02ld seconds\n", 31962306a36Sopenharmony_ci (s64)tv.tv_sec, (long)tv.tv_nsec/1000000L); 32062306a36Sopenharmony_ci } 32162306a36Sopenharmony_ci 32262306a36Sopenharmony_ci seq_printf(m, 32362306a36Sopenharmony_ci "DST_enable\t: %s\n" 32462306a36Sopenharmony_ci "BCD\t\t: %s\n" 32562306a36Sopenharmony_ci "24hr\t\t: %s\n" 32662306a36Sopenharmony_ci "square_wave\t: %s\n" 32762306a36Sopenharmony_ci "alarm_IRQ\t: %s\n" 32862306a36Sopenharmony_ci "update_IRQ\t: %s\n" 32962306a36Sopenharmony_ci "periodic_IRQ\t: %s\n" 33062306a36Sopenharmony_ci "periodic_freq\t: %ld\n" 33162306a36Sopenharmony_ci "batt_status\t: %s\n", 33262306a36Sopenharmony_ci YN(RTC_DST_EN), 33362306a36Sopenharmony_ci NY(RTC_DM_BINARY), 33462306a36Sopenharmony_ci YN(RTC_24H), 33562306a36Sopenharmony_ci YN(RTC_SQWE), 33662306a36Sopenharmony_ci YN(RTC_AIE), 33762306a36Sopenharmony_ci YN(RTC_UIE), 33862306a36Sopenharmony_ci YN(RTC_PIE), 33962306a36Sopenharmony_ci 1UL, 34062306a36Sopenharmony_ci 1 ? "okay" : "dead"); 34162306a36Sopenharmony_ci 34262306a36Sopenharmony_ci return 0; 34362306a36Sopenharmony_ci#undef YN 34462306a36Sopenharmony_ci#undef NY 34562306a36Sopenharmony_ci} 34662306a36Sopenharmony_ci 34762306a36Sopenharmony_cistatic int __init hp_sdc_rtc_init(void) 34862306a36Sopenharmony_ci{ 34962306a36Sopenharmony_ci int ret; 35062306a36Sopenharmony_ci 35162306a36Sopenharmony_ci#ifdef __mc68000__ 35262306a36Sopenharmony_ci if (!MACH_IS_HP300) 35362306a36Sopenharmony_ci return -ENODEV; 35462306a36Sopenharmony_ci#endif 35562306a36Sopenharmony_ci 35662306a36Sopenharmony_ci sema_init(&i8042tregs, 1); 35762306a36Sopenharmony_ci 35862306a36Sopenharmony_ci if ((ret = hp_sdc_request_timer_irq(&hp_sdc_rtc_isr))) 35962306a36Sopenharmony_ci return ret; 36062306a36Sopenharmony_ci 36162306a36Sopenharmony_ci proc_create_single("driver/rtc", 0, NULL, hp_sdc_rtc_proc_show); 36262306a36Sopenharmony_ci 36362306a36Sopenharmony_ci printk(KERN_INFO "HP i8042 SDC + MSM-58321 RTC support loaded " 36462306a36Sopenharmony_ci "(RTC v " RTC_VERSION ")\n"); 36562306a36Sopenharmony_ci 36662306a36Sopenharmony_ci return 0; 36762306a36Sopenharmony_ci} 36862306a36Sopenharmony_ci 36962306a36Sopenharmony_cistatic void __exit hp_sdc_rtc_exit(void) 37062306a36Sopenharmony_ci{ 37162306a36Sopenharmony_ci remove_proc_entry ("driver/rtc", NULL); 37262306a36Sopenharmony_ci hp_sdc_release_timer_irq(hp_sdc_rtc_isr); 37362306a36Sopenharmony_ci printk(KERN_INFO "HP i8042 SDC + MSM-58321 RTC support unloaded\n"); 37462306a36Sopenharmony_ci} 37562306a36Sopenharmony_ci 37662306a36Sopenharmony_cimodule_init(hp_sdc_rtc_init); 37762306a36Sopenharmony_cimodule_exit(hp_sdc_rtc_exit); 378