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