diff options
Diffstat (limited to 'drivers/rtc')
| -rw-r--r-- | drivers/rtc/Kconfig | 19 | ||||
| -rw-r--r-- | drivers/rtc/Makefile | 2 | ||||
| -rw-r--r-- | drivers/rtc/rtc-ab8500.c | 363 | ||||
| -rw-r--r-- | drivers/rtc/rtc-cmos.c | 94 | ||||
| -rw-r--r-- | drivers/rtc/rtc-davinci.c | 673 | ||||
| -rw-r--r-- | drivers/rtc/rtc-ds1302.c | 85 | ||||
| -rw-r--r-- | drivers/rtc/rtc-ds1305.c | 6 | ||||
| -rw-r--r-- | drivers/rtc/rtc-ds1307.c | 6 | ||||
| -rw-r--r-- | drivers/rtc/rtc-ds1511.c | 10 | ||||
| -rw-r--r-- | drivers/rtc/rtc-ds1553.c | 4 | ||||
| -rw-r--r-- | drivers/rtc/rtc-ds1742.c | 4 | ||||
| -rw-r--r-- | drivers/rtc/rtc-isl1208.c | 45 | ||||
| -rw-r--r-- | drivers/rtc/rtc-m41t80.c | 22 | ||||
| -rw-r--r-- | drivers/rtc/rtc-m48t59.c | 4 | ||||
| -rw-r--r-- | drivers/rtc/rtc-mxc.c | 25 | ||||
| -rw-r--r-- | drivers/rtc/rtc-rx8581.c | 6 | ||||
| -rw-r--r-- | drivers/rtc/rtc-s3c.c | 107 | ||||
| -rw-r--r-- | drivers/rtc/rtc-stk17ta8.c | 8 | ||||
| -rw-r--r-- | drivers/rtc/rtc-tx4939.c | 4 | ||||
| -rw-r--r-- | drivers/rtc/rtc-wm831x.c | 16 | 
20 files changed, 1328 insertions, 175 deletions
| diff --git a/drivers/rtc/Kconfig b/drivers/rtc/Kconfig index 6a130375943..10ba12c8c5e 100644 --- a/drivers/rtc/Kconfig +++ b/drivers/rtc/Kconfig @@ -611,6 +611,13 @@ config RTC_DRV_AB3100  	  Select this to enable the ST-Ericsson AB3100 Mixed Signal IC RTC  	  support. This chip contains a battery- and capacitor-backed RTC. +config RTC_DRV_AB8500 +	tristate "ST-Ericsson AB8500 RTC" +	depends on AB8500_CORE +	help +	  Select this to enable the ST-Ericsson AB8500 power management IC RTC +	  support. This chip contains a battery- and capacitor-backed RTC. +  config RTC_DRV_NUC900  	tristate "NUC910/NUC920 RTC driver"  	depends on RTC_CLASS && ARCH_W90X900 @@ -620,6 +627,16 @@ config RTC_DRV_NUC900  comment "on-CPU RTC drivers" +config RTC_DRV_DAVINCI +	tristate "TI DaVinci RTC" +	depends on ARCH_DAVINCI_DM365 +	help +	  If you say yes here you get support for the RTC on the +	  DaVinci platforms (DM365). + +	  This driver can also be built as a module. If so, the module +	  will be called rtc-davinci. +  config RTC_DRV_OMAP  	tristate "TI OMAP1"  	depends on ARCH_OMAP15XX || ARCH_OMAP16XX || ARCH_OMAP730 || ARCH_DAVINCI_DA8XX @@ -630,7 +647,7 @@ config RTC_DRV_OMAP  config RTC_DRV_S3C  	tristate "Samsung S3C series SoC RTC" -	depends on ARCH_S3C2410 +	depends on ARCH_S3C2410 || ARCH_S3C64XX  	help  	  RTC (Realtime Clock) driver for the clock inbuilt into the  	  Samsung S3C24XX series of SoCs. This can provide periodic diff --git a/drivers/rtc/Makefile b/drivers/rtc/Makefile index 44ef194a957..5adbba7cf89 100644 --- a/drivers/rtc/Makefile +++ b/drivers/rtc/Makefile @@ -18,6 +18,7 @@ rtc-core-$(CONFIG_RTC_INTF_SYSFS) += rtc-sysfs.o  # Keep the list ordered.  obj-$(CONFIG_RTC_DRV_AB3100)	+= rtc-ab3100.o +obj-$(CONFIG_RTC_DRV_AB8500)	+= rtc-ab8500.o  obj-$(CONFIG_RTC_DRV_AT32AP700X)+= rtc-at32ap700x.o  obj-$(CONFIG_RTC_DRV_AT91RM9200)+= rtc-at91rm9200.o  obj-$(CONFIG_RTC_DRV_AT91SAM9)	+= rtc-at91sam9.o @@ -27,6 +28,7 @@ obj-$(CONFIG_RTC_DRV_BQ32K)	+= rtc-bq32k.o  obj-$(CONFIG_RTC_DRV_BQ4802)	+= rtc-bq4802.o  obj-$(CONFIG_RTC_DRV_CMOS)	+= rtc-cmos.o  obj-$(CONFIG_RTC_DRV_COH901331)	+= rtc-coh901331.o +obj-$(CONFIG_RTC_DRV_DAVINCI)	+= rtc-davinci.o  obj-$(CONFIG_RTC_DRV_DM355EVM)	+= rtc-dm355evm.o  obj-$(CONFIG_RTC_DRV_DS1216)	+= rtc-ds1216.o  obj-$(CONFIG_RTC_DRV_DS1286)	+= rtc-ds1286.o diff --git a/drivers/rtc/rtc-ab8500.c b/drivers/rtc/rtc-ab8500.c new file mode 100644 index 00000000000..2fda03125e5 --- /dev/null +++ b/drivers/rtc/rtc-ab8500.c @@ -0,0 +1,363 @@ +/* + * Copyright (C) ST-Ericsson SA 2010 + * + * License terms: GNU General Public License (GPL) version 2 + * Author: Virupax Sadashivpetimath <virupax.sadashivpetimath@stericsson.com> + * + * RTC clock driver for the RTC part of the AB8500 Power management chip. + * Based on RTC clock driver for the AB3100 Analog Baseband Chip by + * Linus Walleij <linus.walleij@stericsson.com> + */ + +#include <linux/module.h> +#include <linux/kernel.h> +#include <linux/init.h> +#include <linux/platform_device.h> +#include <linux/rtc.h> +#include <linux/mfd/ab8500.h> +#include <linux/delay.h> + +#define AB8500_RTC_SOFF_STAT_REG	0x0F00 +#define AB8500_RTC_CC_CONF_REG		0x0F01 +#define AB8500_RTC_READ_REQ_REG		0x0F02 +#define AB8500_RTC_WATCH_TSECMID_REG	0x0F03 +#define AB8500_RTC_WATCH_TSECHI_REG	0x0F04 +#define AB8500_RTC_WATCH_TMIN_LOW_REG	0x0F05 +#define AB8500_RTC_WATCH_TMIN_MID_REG	0x0F06 +#define AB8500_RTC_WATCH_TMIN_HI_REG	0x0F07 +#define AB8500_RTC_ALRM_MIN_LOW_REG	0x0F08 +#define AB8500_RTC_ALRM_MIN_MID_REG	0x0F09 +#define AB8500_RTC_ALRM_MIN_HI_REG	0x0F0A +#define AB8500_RTC_STAT_REG		0x0F0B +#define AB8500_RTC_BKUP_CHG_REG		0x0F0C +#define AB8500_RTC_FORCE_BKUP_REG	0x0F0D +#define AB8500_RTC_CALIB_REG		0x0F0E +#define AB8500_RTC_SWITCH_STAT_REG	0x0F0F +#define AB8500_REV_REG			0x1080 + +/* RtcReadRequest bits */ +#define RTC_READ_REQUEST		0x01 +#define RTC_WRITE_REQUEST		0x02 + +/* RtcCtrl bits */ +#define RTC_ALARM_ENA			0x04 +#define RTC_STATUS_DATA			0x01 + +#define COUNTS_PER_SEC			(0xF000 / 60) +#define AB8500_RTC_EPOCH		2000 + +static const unsigned long ab8500_rtc_time_regs[] = { +	AB8500_RTC_WATCH_TMIN_HI_REG, AB8500_RTC_WATCH_TMIN_MID_REG, +	AB8500_RTC_WATCH_TMIN_LOW_REG, AB8500_RTC_WATCH_TSECHI_REG, +	AB8500_RTC_WATCH_TSECMID_REG +}; + +static const unsigned long ab8500_rtc_alarm_regs[] = { +	AB8500_RTC_ALRM_MIN_HI_REG, AB8500_RTC_ALRM_MIN_MID_REG, +	AB8500_RTC_ALRM_MIN_LOW_REG +}; + +/* Calculate the seconds from 1970 to 01-01-2000 00:00:00 */ +static unsigned long get_elapsed_seconds(int year) +{ +	unsigned long secs; +	struct rtc_time tm = { +		.tm_year = year - 1900, +		.tm_mday = 1, +	}; + +	/* +	 * This function calculates secs from 1970 and not from +	 * 1900, even if we supply the offset from year 1900. +	 */ +	rtc_tm_to_time(&tm, &secs); +	return secs; +} + +static int ab8500_rtc_read_time(struct device *dev, struct rtc_time *tm) +{ +	struct ab8500 *ab8500 = dev_get_drvdata(dev->parent); +	unsigned long timeout = jiffies + HZ; +	int retval, i; +	unsigned long mins, secs; +	unsigned char buf[ARRAY_SIZE(ab8500_rtc_time_regs)]; + +	/* Request a data read */ +	retval = ab8500_write(ab8500, AB8500_RTC_READ_REQ_REG, +			      RTC_READ_REQUEST); +	if (retval < 0) +		return retval; + +	/* Early AB8500 chips will not clear the rtc read request bit */ +	if (ab8500->revision == 0) { +		msleep(1); +	} else { +		/* Wait for some cycles after enabling the rtc read in ab8500 */ +		while (time_before(jiffies, timeout)) { +			retval = ab8500_read(ab8500, AB8500_RTC_READ_REQ_REG); +			if (retval < 0) +				return retval; + +			if (!(retval & RTC_READ_REQUEST)) +				break; + +			msleep(1); +		} +	} + +	/* Read the Watchtime registers */ +	for (i = 0; i < ARRAY_SIZE(ab8500_rtc_time_regs); i++) { +		retval = ab8500_read(ab8500, ab8500_rtc_time_regs[i]); +		if (retval < 0) +			return retval; +		buf[i] = retval; +	} + +	mins = (buf[0] << 16) | (buf[1] << 8) | buf[2]; + +	secs =	(buf[3] << 8) | buf[4]; +	secs =	secs / COUNTS_PER_SEC; +	secs =	secs + (mins * 60); + +	/* Add back the initially subtracted number of seconds */ +	secs += get_elapsed_seconds(AB8500_RTC_EPOCH); + +	rtc_time_to_tm(secs, tm); +	return rtc_valid_tm(tm); +} + +static int ab8500_rtc_set_time(struct device *dev, struct rtc_time *tm) +{ +	struct ab8500 *ab8500 = dev_get_drvdata(dev->parent); +	int retval, i; +	unsigned char buf[ARRAY_SIZE(ab8500_rtc_time_regs)]; +	unsigned long no_secs, no_mins, secs = 0; + +	if (tm->tm_year < (AB8500_RTC_EPOCH - 1900)) { +		dev_dbg(dev, "year should be equal to or greater than %d\n", +				AB8500_RTC_EPOCH); +		return -EINVAL; +	} + +	/* Get the number of seconds since 1970 */ +	rtc_tm_to_time(tm, &secs); + +	/* +	 * Convert it to the number of seconds since 01-01-2000 00:00:00, since +	 * we only have a small counter in the RTC. +	 */ +	secs -= get_elapsed_seconds(AB8500_RTC_EPOCH); + +	no_mins = secs / 60; + +	no_secs = secs % 60; +	/* Make the seconds count as per the RTC resolution */ +	no_secs = no_secs * COUNTS_PER_SEC; + +	buf[4] = no_secs & 0xFF; +	buf[3] = (no_secs >> 8) & 0xFF; + +	buf[2] = no_mins & 0xFF; +	buf[1] = (no_mins >> 8) & 0xFF; +	buf[0] = (no_mins >> 16) & 0xFF; + +	for (i = 0; i < ARRAY_SIZE(ab8500_rtc_time_regs); i++) { +		retval = ab8500_write(ab8500, ab8500_rtc_time_regs[i], buf[i]); +		if (retval < 0) +			return retval; +	} + +	/* Request a data write */ +	return ab8500_write(ab8500, AB8500_RTC_READ_REQ_REG, RTC_WRITE_REQUEST); +} + +static int ab8500_rtc_read_alarm(struct device *dev, struct rtc_wkalrm *alarm) +{ +	struct ab8500 *ab8500 = dev_get_drvdata(dev->parent); +	int retval, i; +	int rtc_ctrl; +	unsigned char buf[ARRAY_SIZE(ab8500_rtc_alarm_regs)]; +	unsigned long secs, mins; + +	/* Check if the alarm is enabled or not */ +	rtc_ctrl = ab8500_read(ab8500, AB8500_RTC_STAT_REG); +	if (rtc_ctrl < 0) +		return rtc_ctrl; + +	if (rtc_ctrl & RTC_ALARM_ENA) +		alarm->enabled = 1; +	else +		alarm->enabled = 0; + +	alarm->pending = 0; + +	for (i = 0; i < ARRAY_SIZE(ab8500_rtc_alarm_regs); i++) { +		retval = ab8500_read(ab8500, ab8500_rtc_alarm_regs[i]); +		if (retval < 0) +			return retval; +		buf[i] = retval; +	} + +	mins = (buf[0] << 16) | (buf[1] << 8) | (buf[2]); +	secs = mins * 60; + +	/* Add back the initially subtracted number of seconds */ +	secs += get_elapsed_seconds(AB8500_RTC_EPOCH); + +	rtc_time_to_tm(secs, &alarm->time); + +	return rtc_valid_tm(&alarm->time); +} + +static int ab8500_rtc_irq_enable(struct device *dev, unsigned int enabled) +{ +	struct ab8500 *ab8500 = dev_get_drvdata(dev->parent); + +	return ab8500_set_bits(ab8500, AB8500_RTC_STAT_REG, RTC_ALARM_ENA, +			       enabled ? RTC_ALARM_ENA : 0); +} + +static int ab8500_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alarm) +{ +	struct ab8500 *ab8500 = dev_get_drvdata(dev->parent); +	int retval, i; +	unsigned char buf[ARRAY_SIZE(ab8500_rtc_alarm_regs)]; +	unsigned long mins, secs = 0; + +	if (alarm->time.tm_year < (AB8500_RTC_EPOCH - 1900)) { +		dev_dbg(dev, "year should be equal to or greater than %d\n", +				AB8500_RTC_EPOCH); +		return -EINVAL; +	} + +	/* Get the number of seconds since 1970 */ +	rtc_tm_to_time(&alarm->time, &secs); + +	/* +	 * Convert it to the number of seconds since 01-01-2000 00:00:00, since +	 * we only have a small counter in the RTC. +	 */ +	secs -= get_elapsed_seconds(AB8500_RTC_EPOCH); + +	mins = secs / 60; + +	buf[2] = mins & 0xFF; +	buf[1] = (mins >> 8) & 0xFF; +	buf[0] = (mins >> 16) & 0xFF; + +	/* Set the alarm time */ +	for (i = 0; i < ARRAY_SIZE(ab8500_rtc_alarm_regs); i++) { +		retval = ab8500_write(ab8500, ab8500_rtc_alarm_regs[i], buf[i]); +		if (retval < 0) +			return retval; +	} + +	return ab8500_rtc_irq_enable(dev, alarm->enabled); +} + +static irqreturn_t rtc_alarm_handler(int irq, void *data) +{ +	struct rtc_device *rtc = data; +	unsigned long events = RTC_IRQF | RTC_AF; + +	dev_dbg(&rtc->dev, "%s\n", __func__); +	rtc_update_irq(rtc, 1, events); + +	return IRQ_HANDLED; +} + +static const struct rtc_class_ops ab8500_rtc_ops = { +	.read_time		= ab8500_rtc_read_time, +	.set_time		= ab8500_rtc_set_time, +	.read_alarm		= ab8500_rtc_read_alarm, +	.set_alarm		= ab8500_rtc_set_alarm, +	.alarm_irq_enable	= ab8500_rtc_irq_enable, +}; + +static int __devinit ab8500_rtc_probe(struct platform_device *pdev) +{ +	struct ab8500 *ab8500 = dev_get_drvdata(pdev->dev.parent); +	int err; +	struct rtc_device *rtc; +	int rtc_ctrl; +	int irq; + +	irq = platform_get_irq_byname(pdev, "ALARM"); +	if (irq < 0) +		return irq; + +	/* For RTC supply test */ +	err = ab8500_set_bits(ab8500, AB8500_RTC_STAT_REG, RTC_STATUS_DATA, +			RTC_STATUS_DATA); +	if (err < 0) +		return err; + +	/* Wait for reset by the PorRtc */ +	msleep(1); + +	rtc_ctrl = ab8500_read(ab8500, AB8500_RTC_STAT_REG); +	if (rtc_ctrl < 0) +		return rtc_ctrl; + +	/* Check if the RTC Supply fails */ +	if (!(rtc_ctrl & RTC_STATUS_DATA)) { +		dev_err(&pdev->dev, "RTC supply failure\n"); +		return -ENODEV; +	} + +	rtc = rtc_device_register("ab8500-rtc", &pdev->dev, &ab8500_rtc_ops, +			THIS_MODULE); +	if (IS_ERR(rtc)) { +		dev_err(&pdev->dev, "Registration failed\n"); +		err = PTR_ERR(rtc); +		return err; +	} + +	err = request_threaded_irq(irq, NULL, rtc_alarm_handler, 0, +				   "ab8500-rtc", rtc); +	if (err < 0) { +		rtc_device_unregister(rtc); +		return err; +	} + +	platform_set_drvdata(pdev, rtc); + +	return 0; +} + +static int __devexit ab8500_rtc_remove(struct platform_device *pdev) +{ +	struct rtc_device *rtc = platform_get_drvdata(pdev); +	int irq = platform_get_irq_byname(pdev, "ALARM"); + +	free_irq(irq, rtc); +	rtc_device_unregister(rtc); +	platform_set_drvdata(pdev, NULL); + +	return 0; +} + +static struct platform_driver ab8500_rtc_driver = { +	.driver = { +		.name = "ab8500-rtc", +		.owner = THIS_MODULE, +	}, +	.probe	= ab8500_rtc_probe, +	.remove = __devexit_p(ab8500_rtc_remove), +}; + +static int __init ab8500_rtc_init(void) +{ +	return platform_driver_register(&ab8500_rtc_driver); +} + +static void __exit ab8500_rtc_exit(void) +{ +	platform_driver_unregister(&ab8500_rtc_driver); +} + +module_init(ab8500_rtc_init); +module_exit(ab8500_rtc_exit); +MODULE_AUTHOR("Virupax Sadashivpetimath <virupax.sadashivpetimath@stericsson.com>"); +MODULE_DESCRIPTION("AB8500 RTC Driver"); +MODULE_LICENSE("GPL v2"); diff --git a/drivers/rtc/rtc-cmos.c b/drivers/rtc/rtc-cmos.c index e9aa814ddd2..11b8ea29d2b 100644 --- a/drivers/rtc/rtc-cmos.c +++ b/drivers/rtc/rtc-cmos.c @@ -238,31 +238,32 @@ static int cmos_read_alarm(struct device *dev, struct rtc_wkalrm *t)  	rtc_control = CMOS_READ(RTC_CONTROL);  	spin_unlock_irq(&rtc_lock); -	/* REVISIT this assumes PC style usage:  always BCD */ - -	if (((unsigned)t->time.tm_sec) < 0x60) -		t->time.tm_sec = bcd2bin(t->time.tm_sec); -	else -		t->time.tm_sec = -1; -	if (((unsigned)t->time.tm_min) < 0x60) -		t->time.tm_min = bcd2bin(t->time.tm_min); -	else -		t->time.tm_min = -1; -	if (((unsigned)t->time.tm_hour) < 0x24) -		t->time.tm_hour = bcd2bin(t->time.tm_hour); -	else -		t->time.tm_hour = -1; - -	if (cmos->day_alrm) { -		if (((unsigned)t->time.tm_mday) <= 0x31) -			t->time.tm_mday = bcd2bin(t->time.tm_mday); +	if (!(rtc_control & RTC_DM_BINARY) || RTC_ALWAYS_BCD) { +		if (((unsigned)t->time.tm_sec) < 0x60) +			t->time.tm_sec = bcd2bin(t->time.tm_sec);  		else -			t->time.tm_mday = -1; -		if (cmos->mon_alrm) { -			if (((unsigned)t->time.tm_mon) <= 0x12) -				t->time.tm_mon = bcd2bin(t->time.tm_mon) - 1; +			t->time.tm_sec = -1; +		if (((unsigned)t->time.tm_min) < 0x60) +			t->time.tm_min = bcd2bin(t->time.tm_min); +		else +			t->time.tm_min = -1; +		if (((unsigned)t->time.tm_hour) < 0x24) +			t->time.tm_hour = bcd2bin(t->time.tm_hour); +		else +			t->time.tm_hour = -1; + +		if (cmos->day_alrm) { +			if (((unsigned)t->time.tm_mday) <= 0x31) +				t->time.tm_mday = bcd2bin(t->time.tm_mday);  			else -				t->time.tm_mon = -1; +				t->time.tm_mday = -1; + +			if (cmos->mon_alrm) { +				if (((unsigned)t->time.tm_mon) <= 0x12) +					t->time.tm_mon = bcd2bin(t->time.tm_mon)-1; +				else +					t->time.tm_mon = -1; +			}  		}  	}  	t->time.tm_year = -1; @@ -322,29 +323,26 @@ static void cmos_irq_disable(struct cmos_rtc *cmos, unsigned char mask)  static int cmos_set_alarm(struct device *dev, struct rtc_wkalrm *t)  {  	struct cmos_rtc	*cmos = dev_get_drvdata(dev); -	unsigned char	mon, mday, hrs, min, sec; +       unsigned char   mon, mday, hrs, min, sec, rtc_control;  	if (!is_valid_irq(cmos->irq))  		return -EIO; -	/* REVISIT this assumes PC style usage:  always BCD */ - -	/* Writing 0xff means "don't care" or "match all".  */ -  	mon = t->time.tm_mon + 1; -	mon = (mon <= 12) ? bin2bcd(mon) : 0xff; -  	mday = t->time.tm_mday; -	mday = (mday >= 1 && mday <= 31) ? bin2bcd(mday) : 0xff; -  	hrs = t->time.tm_hour; -	hrs = (hrs < 24) ? bin2bcd(hrs) : 0xff; -  	min = t->time.tm_min; -	min = (min < 60) ? bin2bcd(min) : 0xff; -  	sec = t->time.tm_sec; -	sec = (sec < 60) ? bin2bcd(sec) : 0xff; + +	rtc_control = CMOS_READ(RTC_CONTROL); +	if (!(rtc_control & RTC_DM_BINARY) || RTC_ALWAYS_BCD) { +		/* Writing 0xff means "don't care" or "match all".  */ +		mon = (mon <= 12) ? bin2bcd(mon) : 0xff; +		mday = (mday >= 1 && mday <= 31) ? bin2bcd(mday) : 0xff; +		hrs = (hrs < 24) ? bin2bcd(hrs) : 0xff; +		min = (min < 60) ? bin2bcd(min) : 0xff; +		sec = (sec < 60) ? bin2bcd(sec) : 0xff; +	}  	spin_lock_irq(&rtc_lock); @@ -478,7 +476,7 @@ static int cmos_procfs(struct device *dev, struct seq_file *seq)  			"update_IRQ\t: %s\n"  			"HPET_emulated\t: %s\n"  			// "square_wave\t: %s\n" -			// "BCD\t\t: %s\n" +			"BCD\t\t: %s\n"  			"DST_enable\t: %s\n"  			"periodic_freq\t: %d\n"  			"batt_status\t: %s\n", @@ -486,7 +484,7 @@ static int cmos_procfs(struct device *dev, struct seq_file *seq)  			(rtc_control & RTC_UIE) ? "yes" : "no",  			is_hpet_enabled() ? "yes" : "no",  			// (rtc_control & RTC_SQWE) ? "yes" : "no", -			// (rtc_control & RTC_DM_BINARY) ? "no" : "yes", +			(rtc_control & RTC_DM_BINARY) ? "no" : "yes",  			(rtc_control & RTC_DST_EN) ? "yes" : "no",  			cmos->rtc->irq_freq,  			(valid & RTC_VRT) ? "okay" : "dead"); @@ -519,7 +517,8 @@ static const struct rtc_class_ops cmos_rtc_ops = {  #define NVRAM_OFFSET	(RTC_REG_D + 1)  static ssize_t -cmos_nvram_read(struct kobject *kobj, struct bin_attribute *attr, +cmos_nvram_read(struct file *filp, struct kobject *kobj, +		struct bin_attribute *attr,  		char *buf, loff_t off, size_t count)  {  	int	retval; @@ -547,7 +546,8 @@ cmos_nvram_read(struct kobject *kobj, struct bin_attribute *attr,  }  static ssize_t -cmos_nvram_write(struct kobject *kobj, struct bin_attribute *attr, +cmos_nvram_write(struct file *filp, struct kobject *kobj, +		struct bin_attribute *attr,  		char *buf, loff_t off, size_t count)  {  	struct cmos_rtc	*cmos; @@ -719,6 +719,9 @@ cmos_do_probe(struct device *dev, struct resource *ports, int rtc_irq)  		}  	} +	cmos_rtc.dev = dev; +	dev_set_drvdata(dev, &cmos_rtc); +  	cmos_rtc.rtc = rtc_device_register(driver_name, dev,  				&cmos_rtc_ops, THIS_MODULE);  	if (IS_ERR(cmos_rtc.rtc)) { @@ -726,8 +729,6 @@ cmos_do_probe(struct device *dev, struct resource *ports, int rtc_irq)  		goto cleanup0;  	} -	cmos_rtc.dev = dev; -	dev_set_drvdata(dev, &cmos_rtc);  	rename_region(ports, dev_name(&cmos_rtc.rtc->dev));  	spin_lock_irq(&rtc_lock); @@ -749,12 +750,11 @@ cmos_do_probe(struct device *dev, struct resource *ports, int rtc_irq)  	spin_unlock_irq(&rtc_lock); -	/* FIXME teach the alarm code how to handle binary mode; +	/* FIXME:  	 * <asm-generic/rtc.h> doesn't know 12-hour mode either.  	 */ -	if (is_valid_irq(rtc_irq) && -	    (!(rtc_control & RTC_24H) || (rtc_control & (RTC_DM_BINARY)))) { -		dev_dbg(dev, "only 24-hr BCD mode supported\n"); +       if (is_valid_irq(rtc_irq) && !(rtc_control & RTC_24H)) { +		dev_warn(dev, "only 24-hr supported\n");  		retval = -ENXIO;  		goto cleanup1;  	} diff --git a/drivers/rtc/rtc-davinci.c b/drivers/rtc/rtc-davinci.c new file mode 100644 index 00000000000..92a8f6cacda --- /dev/null +++ b/drivers/rtc/rtc-davinci.c @@ -0,0 +1,673 @@ +/* + * DaVinci Power Management and Real Time Clock Driver for TI platforms + * + * Copyright (C) 2009 Texas Instruments, Inc + * + * Author: Miguel Aguilar <miguel.aguilar@ridgerun.com> + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include <linux/kernel.h> +#include <linux/init.h> +#include <linux/module.h> +#include <linux/ioport.h> +#include <linux/delay.h> +#include <linux/spinlock.h> +#include <linux/rtc.h> +#include <linux/bcd.h> +#include <linux/platform_device.h> +#include <linux/io.h> + +/* + * The DaVinci RTC is a simple RTC with the following + * Sec: 0 - 59 : BCD count + * Min: 0 - 59 : BCD count + * Hour: 0 - 23 : BCD count + * Day: 0 - 0x7FFF(32767) : Binary count ( Over 89 years ) + */ + +/* PRTC interface registers */ +#define DAVINCI_PRTCIF_PID		0x00 +#define PRTCIF_CTLR			0x04 +#define PRTCIF_LDATA			0x08 +#define PRTCIF_UDATA			0x0C +#define PRTCIF_INTEN			0x10 +#define PRTCIF_INTFLG			0x14 + +/* PRTCIF_CTLR bit fields */ +#define PRTCIF_CTLR_BUSY		BIT(31) +#define PRTCIF_CTLR_SIZE		BIT(25) +#define PRTCIF_CTLR_DIR			BIT(24) +#define PRTCIF_CTLR_BENU_MSB		BIT(23) +#define PRTCIF_CTLR_BENU_3RD_BYTE	BIT(22) +#define PRTCIF_CTLR_BENU_2ND_BYTE	BIT(21) +#define PRTCIF_CTLR_BENU_LSB		BIT(20) +#define PRTCIF_CTLR_BENU_MASK		(0x00F00000) +#define PRTCIF_CTLR_BENL_MSB		BIT(19) +#define PRTCIF_CTLR_BENL_3RD_BYTE	BIT(18) +#define PRTCIF_CTLR_BENL_2ND_BYTE	BIT(17) +#define PRTCIF_CTLR_BENL_LSB		BIT(16) +#define PRTCIF_CTLR_BENL_MASK		(0x000F0000) + +/* PRTCIF_INTEN bit fields */ +#define PRTCIF_INTEN_RTCSS		BIT(1) +#define PRTCIF_INTEN_RTCIF		BIT(0) +#define PRTCIF_INTEN_MASK		(PRTCIF_INTEN_RTCSS \ +					| PRTCIF_INTEN_RTCIF) + +/* PRTCIF_INTFLG bit fields */ +#define PRTCIF_INTFLG_RTCSS		BIT(1) +#define PRTCIF_INTFLG_RTCIF		BIT(0) +#define PRTCIF_INTFLG_MASK		(PRTCIF_INTFLG_RTCSS \ +					| PRTCIF_INTFLG_RTCIF) + +/* PRTC subsystem registers */ +#define PRTCSS_RTC_INTC_EXTENA1		(0x0C) +#define PRTCSS_RTC_CTRL			(0x10) +#define PRTCSS_RTC_WDT			(0x11) +#define PRTCSS_RTC_TMR0			(0x12) +#define PRTCSS_RTC_TMR1			(0x13) +#define PRTCSS_RTC_CCTRL		(0x14) +#define PRTCSS_RTC_SEC			(0x15) +#define PRTCSS_RTC_MIN			(0x16) +#define PRTCSS_RTC_HOUR			(0x17) +#define PRTCSS_RTC_DAY0			(0x18) +#define PRTCSS_RTC_DAY1			(0x19) +#define PRTCSS_RTC_AMIN			(0x1A) +#define PRTCSS_RTC_AHOUR		(0x1B) +#define PRTCSS_RTC_ADAY0		(0x1C) +#define PRTCSS_RTC_ADAY1		(0x1D) +#define PRTCSS_RTC_CLKC_CNT		(0x20) + +/* PRTCSS_RTC_INTC_EXTENA1 */ +#define PRTCSS_RTC_INTC_EXTENA1_MASK	(0x07) + +/* PRTCSS_RTC_CTRL bit fields */ +#define PRTCSS_RTC_CTRL_WDTBUS		BIT(7) +#define PRTCSS_RTC_CTRL_WEN		BIT(6) +#define PRTCSS_RTC_CTRL_WDRT		BIT(5) +#define PRTCSS_RTC_CTRL_WDTFLG		BIT(4) +#define PRTCSS_RTC_CTRL_TE		BIT(3) +#define PRTCSS_RTC_CTRL_TIEN		BIT(2) +#define PRTCSS_RTC_CTRL_TMRFLG		BIT(1) +#define PRTCSS_RTC_CTRL_TMMD		BIT(0) + +/* PRTCSS_RTC_CCTRL bit fields */ +#define PRTCSS_RTC_CCTRL_CALBUSY	BIT(7) +#define PRTCSS_RTC_CCTRL_DAEN		BIT(5) +#define PRTCSS_RTC_CCTRL_HAEN		BIT(4) +#define PRTCSS_RTC_CCTRL_MAEN		BIT(3) +#define PRTCSS_RTC_CCTRL_ALMFLG		BIT(2) +#define PRTCSS_RTC_CCTRL_AIEN		BIT(1) +#define PRTCSS_RTC_CCTRL_CAEN		BIT(0) + +static DEFINE_SPINLOCK(davinci_rtc_lock); + +struct davinci_rtc { +	struct rtc_device 		*rtc; +	void __iomem			*base; +	resource_size_t			pbase; +	size_t				base_size; +	int				irq; +}; + +static inline void rtcif_write(struct davinci_rtc *davinci_rtc, +			       u32 val, u32 addr) +{ +	writel(val, davinci_rtc->base + addr); +} + +static inline u32 rtcif_read(struct davinci_rtc *davinci_rtc, u32 addr) +{ +	return readl(davinci_rtc->base + addr); +} + +static inline void rtcif_wait(struct davinci_rtc *davinci_rtc) +{ +	while (rtcif_read(davinci_rtc, PRTCIF_CTLR) & PRTCIF_CTLR_BUSY) +		cpu_relax(); +} + +static inline void rtcss_write(struct davinci_rtc *davinci_rtc, +			       unsigned long val, u8 addr) +{ +	rtcif_wait(davinci_rtc); + +	rtcif_write(davinci_rtc, PRTCIF_CTLR_BENL_LSB | addr, PRTCIF_CTLR); +	rtcif_write(davinci_rtc, val, PRTCIF_LDATA); + +	rtcif_wait(davinci_rtc); +} + +static inline u8 rtcss_read(struct davinci_rtc *davinci_rtc, u8 addr) +{ +	rtcif_wait(davinci_rtc); + +	rtcif_write(davinci_rtc, PRTCIF_CTLR_DIR | PRTCIF_CTLR_BENL_LSB | addr, +		    PRTCIF_CTLR); + +	rtcif_wait(davinci_rtc); + +	return rtcif_read(davinci_rtc, PRTCIF_LDATA); +} + +static inline void davinci_rtcss_calendar_wait(struct davinci_rtc *davinci_rtc) +{ +	while (rtcss_read(davinci_rtc, PRTCSS_RTC_CCTRL) & +	       PRTCSS_RTC_CCTRL_CALBUSY) +		cpu_relax(); +} + +static irqreturn_t davinci_rtc_interrupt(int irq, void *class_dev) +{ +	struct davinci_rtc *davinci_rtc = class_dev; +	unsigned long events = 0; +	u32 irq_flg; +	u8 alm_irq, tmr_irq; +	u8 rtc_ctrl, rtc_cctrl; +	int ret = IRQ_NONE; + +	irq_flg = rtcif_read(davinci_rtc, PRTCIF_INTFLG) & +		  PRTCIF_INTFLG_RTCSS; + +	alm_irq = rtcss_read(davinci_rtc, PRTCSS_RTC_CCTRL) & +		  PRTCSS_RTC_CCTRL_ALMFLG; + +	tmr_irq = rtcss_read(davinci_rtc, PRTCSS_RTC_CTRL) & +		  PRTCSS_RTC_CTRL_TMRFLG; + +	if (irq_flg) { +		if (alm_irq) { +			events |= RTC_IRQF | RTC_AF; +			rtc_cctrl = rtcss_read(davinci_rtc, PRTCSS_RTC_CCTRL); +			rtc_cctrl |=  PRTCSS_RTC_CCTRL_ALMFLG; +			rtcss_write(davinci_rtc, rtc_cctrl, PRTCSS_RTC_CCTRL); +		} else if (tmr_irq) { +			events |= RTC_IRQF | RTC_PF; +			rtc_ctrl = rtcss_read(davinci_rtc, PRTCSS_RTC_CTRL); +			rtc_ctrl |=  PRTCSS_RTC_CTRL_TMRFLG; +			rtcss_write(davinci_rtc, rtc_ctrl, PRTCSS_RTC_CTRL); +		} + +		rtcif_write(davinci_rtc, PRTCIF_INTFLG_RTCSS, +				    PRTCIF_INTFLG); +		rtc_update_irq(davinci_rtc->rtc, 1, events); + +		ret = IRQ_HANDLED; +	} + +	return ret; +} + +static int +davinci_rtc_ioctl(struct device *dev, unsigned int cmd, unsigned long arg) +{ +	struct davinci_rtc *davinci_rtc = dev_get_drvdata(dev); +	u8 rtc_ctrl; +	unsigned long flags; +	int ret = 0; + +	spin_lock_irqsave(&davinci_rtc_lock, flags); + +	rtc_ctrl = rtcss_read(davinci_rtc, PRTCSS_RTC_CTRL); + +	switch (cmd) { +	case RTC_WIE_ON: +		rtc_ctrl |= PRTCSS_RTC_CTRL_WEN | PRTCSS_RTC_CTRL_WDTFLG; +		break; +	case RTC_WIE_OFF: +		rtc_ctrl &= ~PRTCSS_RTC_CTRL_WEN; +		break; +	case RTC_UIE_OFF: +	case RTC_UIE_ON: +		ret = -ENOTTY; +		break; +	default: +		ret = -ENOIOCTLCMD; +	} + +	rtcss_write(davinci_rtc, rtc_ctrl, PRTCSS_RTC_CTRL); + +	spin_unlock_irqrestore(&davinci_rtc_lock, flags); + +	return ret; +} + +static int convertfromdays(u16 days, struct rtc_time *tm) +{ +	int tmp_days, year, mon; + +	for (year = 2000;; year++) { +		tmp_days = rtc_year_days(1, 12, year); +		if (days >= tmp_days) +			days -= tmp_days; +		else { +			for (mon = 0;; mon++) { +				tmp_days = rtc_month_days(mon, year); +				if (days >= tmp_days) { +					days -= tmp_days; +				} else { +					tm->tm_year = year - 1900; +					tm->tm_mon = mon; +					tm->tm_mday = days + 1; +					break; +				} +			} +			break; +		} +	} +	return 0; +} + +static int convert2days(u16 *days, struct rtc_time *tm) +{ +	int i; +	*days = 0; + +	/* epoch == 1900 */ +	if (tm->tm_year < 100 || tm->tm_year > 199) +		return -EINVAL; + +	for (i = 2000; i < 1900 + tm->tm_year; i++) +		*days += rtc_year_days(1, 12, i); + +	*days += rtc_year_days(tm->tm_mday, tm->tm_mon, 1900 + tm->tm_year); + +	return 0; +} + +static int davinci_rtc_read_time(struct device *dev, struct rtc_time *tm) +{ +	struct davinci_rtc *davinci_rtc = dev_get_drvdata(dev); +	u16 days = 0; +	u8 day0, day1; +	unsigned long flags; + +	spin_lock_irqsave(&davinci_rtc_lock, flags); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	tm->tm_sec = bcd2bin(rtcss_read(davinci_rtc, PRTCSS_RTC_SEC)); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	tm->tm_min = bcd2bin(rtcss_read(davinci_rtc, PRTCSS_RTC_MIN)); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	tm->tm_hour = bcd2bin(rtcss_read(davinci_rtc, PRTCSS_RTC_HOUR)); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	day0 = rtcss_read(davinci_rtc, PRTCSS_RTC_DAY0); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	day1 = rtcss_read(davinci_rtc, PRTCSS_RTC_DAY1); + +	spin_unlock_irqrestore(&davinci_rtc_lock, flags); + +	days |= day1; +	days <<= 8; +	days |= day0; + +	if (convertfromdays(days, tm) < 0) +		return -EINVAL; + +	return 0; +} + +static int davinci_rtc_set_time(struct device *dev, struct rtc_time *tm) +{ +	struct davinci_rtc *davinci_rtc = dev_get_drvdata(dev); +	u16 days; +	u8 rtc_cctrl; +	unsigned long flags; + +	if (convert2days(&days, tm) < 0) +		return -EINVAL; + +	spin_lock_irqsave(&davinci_rtc_lock, flags); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	rtcss_write(davinci_rtc, bin2bcd(tm->tm_sec), PRTCSS_RTC_SEC); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	rtcss_write(davinci_rtc, bin2bcd(tm->tm_min), PRTCSS_RTC_MIN); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	rtcss_write(davinci_rtc, bin2bcd(tm->tm_hour), PRTCSS_RTC_HOUR); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	rtcss_write(davinci_rtc, days & 0xFF, PRTCSS_RTC_DAY0); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	rtcss_write(davinci_rtc, (days & 0xFF00) >> 8, PRTCSS_RTC_DAY1); + +	rtc_cctrl = rtcss_read(davinci_rtc, PRTCSS_RTC_CCTRL); +	rtc_cctrl |= PRTCSS_RTC_CCTRL_CAEN; +	rtcss_write(davinci_rtc, rtc_cctrl, PRTCSS_RTC_CCTRL); + +	spin_unlock_irqrestore(&davinci_rtc_lock, flags); + +	return 0; +} + +static int davinci_rtc_alarm_irq_enable(struct device *dev, +					unsigned int enabled) +{ +	struct davinci_rtc *davinci_rtc = dev_get_drvdata(dev); +	unsigned long flags; +	u8 rtc_cctrl = rtcss_read(davinci_rtc, PRTCSS_RTC_CCTRL); + +	spin_lock_irqsave(&davinci_rtc_lock, flags); + +	if (enabled) +		rtc_cctrl |= PRTCSS_RTC_CCTRL_DAEN | +			     PRTCSS_RTC_CCTRL_HAEN | +			     PRTCSS_RTC_CCTRL_MAEN | +			     PRTCSS_RTC_CCTRL_ALMFLG | +			     PRTCSS_RTC_CCTRL_AIEN; +	else +		rtc_cctrl &= ~PRTCSS_RTC_CCTRL_AIEN; + +	davinci_rtcss_calendar_wait(davinci_rtc); +	rtcss_write(davinci_rtc, rtc_cctrl, PRTCSS_RTC_CCTRL); + +	spin_unlock_irqrestore(&davinci_rtc_lock, flags); + +	return 0; +} + +static int davinci_rtc_read_alarm(struct device *dev, struct rtc_wkalrm *alm) +{ +	struct davinci_rtc *davinci_rtc = dev_get_drvdata(dev); +	u16 days = 0; +	u8 day0, day1; +	unsigned long flags; + +	spin_lock_irqsave(&davinci_rtc_lock, flags); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	alm->time.tm_min = bcd2bin(rtcss_read(davinci_rtc, PRTCSS_RTC_AMIN)); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	alm->time.tm_hour = bcd2bin(rtcss_read(davinci_rtc, PRTCSS_RTC_AHOUR)); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	day0 = rtcss_read(davinci_rtc, PRTCSS_RTC_ADAY0); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	day1 = rtcss_read(davinci_rtc, PRTCSS_RTC_ADAY1); + +	spin_unlock_irqrestore(&davinci_rtc_lock, flags); +	days |= day1; +	days <<= 8; +	days |= day0; + +	if (convertfromdays(days, &alm->time) < 0) +		return -EINVAL; + +	alm->pending = !!(rtcss_read(davinci_rtc, +			  PRTCSS_RTC_CCTRL) & +			PRTCSS_RTC_CCTRL_AIEN); +	alm->enabled = alm->pending && device_may_wakeup(dev); + +	return 0; +} + +static int davinci_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alm) +{ +	struct davinci_rtc *davinci_rtc = dev_get_drvdata(dev); +	unsigned long flags; +	u16 days; + +	if (alm->time.tm_mday <= 0 && alm->time.tm_mon < 0 +	    && alm->time.tm_year < 0) { +		struct rtc_time tm; +		unsigned long now, then; + +		davinci_rtc_read_time(dev, &tm); +		rtc_tm_to_time(&tm, &now); + +		alm->time.tm_mday = tm.tm_mday; +		alm->time.tm_mon = tm.tm_mon; +		alm->time.tm_year = tm.tm_year; +		rtc_tm_to_time(&alm->time, &then); + +		if (then < now) { +			rtc_time_to_tm(now + 24 * 60 * 60, &tm); +			alm->time.tm_mday = tm.tm_mday; +			alm->time.tm_mon = tm.tm_mon; +			alm->time.tm_year = tm.tm_year; +		} +	} + +	if (convert2days(&days, &alm->time) < 0) +		return -EINVAL; + +	spin_lock_irqsave(&davinci_rtc_lock, flags); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	rtcss_write(davinci_rtc, bin2bcd(alm->time.tm_min), PRTCSS_RTC_AMIN); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	rtcss_write(davinci_rtc, bin2bcd(alm->time.tm_hour), PRTCSS_RTC_AHOUR); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	rtcss_write(davinci_rtc, days & 0xFF, PRTCSS_RTC_ADAY0); + +	davinci_rtcss_calendar_wait(davinci_rtc); +	rtcss_write(davinci_rtc, (days & 0xFF00) >> 8, PRTCSS_RTC_ADAY1); + +	spin_unlock_irqrestore(&davinci_rtc_lock, flags); + +	return 0; +} + +static int davinci_rtc_irq_set_state(struct device *dev, int enabled) +{ +	struct davinci_rtc *davinci_rtc = dev_get_drvdata(dev); +	unsigned long flags; +	u8 rtc_ctrl; + +	spin_lock_irqsave(&davinci_rtc_lock, flags); + +	rtc_ctrl = rtcss_read(davinci_rtc, PRTCSS_RTC_CTRL); + +	if (enabled) { +		while (rtcss_read(davinci_rtc, PRTCSS_RTC_CTRL) +		       & PRTCSS_RTC_CTRL_WDTBUS) +			cpu_relax(); + +		rtc_ctrl |= PRTCSS_RTC_CTRL_TE; +		rtcss_write(davinci_rtc, rtc_ctrl, PRTCSS_RTC_CTRL); + +		rtcss_write(davinci_rtc, 0x0, PRTCSS_RTC_CLKC_CNT); + +		rtc_ctrl |= PRTCSS_RTC_CTRL_TIEN | +			    PRTCSS_RTC_CTRL_TMMD | +			    PRTCSS_RTC_CTRL_TMRFLG; +	} else +		rtc_ctrl &= ~PRTCSS_RTC_CTRL_TIEN; + +	rtcss_write(davinci_rtc, rtc_ctrl, PRTCSS_RTC_CTRL); + +	spin_unlock_irqrestore(&davinci_rtc_lock, flags); + +	return 0; +} + +static int davinci_rtc_irq_set_freq(struct device *dev, int freq) +{ +	struct davinci_rtc *davinci_rtc = dev_get_drvdata(dev); +	unsigned long flags; +	u16 tmr_counter = (0x8000 >> (ffs(freq) - 1)); + +	spin_lock_irqsave(&davinci_rtc_lock, flags); + +	rtcss_write(davinci_rtc, tmr_counter & 0xFF, PRTCSS_RTC_TMR0); +	rtcss_write(davinci_rtc, (tmr_counter & 0xFF00) >> 8, PRTCSS_RTC_TMR1); + +	spin_unlock_irqrestore(&davinci_rtc_lock, flags); + +	return 0; +} + +static struct rtc_class_ops davinci_rtc_ops = { +	.ioctl			= davinci_rtc_ioctl, +	.read_time		= davinci_rtc_read_time, +	.set_time		= davinci_rtc_set_time, +	.alarm_irq_enable	= davinci_rtc_alarm_irq_enable, +	.read_alarm		= davinci_rtc_read_alarm, +	.set_alarm		= davinci_rtc_set_alarm, +	.irq_set_state		= davinci_rtc_irq_set_state, +	.irq_set_freq		= davinci_rtc_irq_set_freq, +}; + +static int __init davinci_rtc_probe(struct platform_device *pdev) +{ +	struct device *dev = &pdev->dev; +	struct davinci_rtc *davinci_rtc; +	struct resource *res, *mem; +	int ret = 0; + +	davinci_rtc = kzalloc(sizeof(struct davinci_rtc), GFP_KERNEL); +	if (!davinci_rtc) { +		dev_dbg(dev, "could not allocate memory for private data\n"); +		return -ENOMEM; +	} + +	davinci_rtc->irq = platform_get_irq(pdev, 0); +	if (davinci_rtc->irq < 0) { +		dev_err(dev, "no RTC irq\n"); +		ret = davinci_rtc->irq; +		goto fail1; +	} + +	res = platform_get_resource(pdev, IORESOURCE_MEM, 0); +	if (!res) { +		dev_err(dev, "no mem resource\n"); +		ret = -EINVAL; +		goto fail1; +	} + +	davinci_rtc->pbase = res->start; +	davinci_rtc->base_size = resource_size(res); + +	mem = request_mem_region(davinci_rtc->pbase, davinci_rtc->base_size, +				 pdev->name); +	if (!mem) { +		dev_err(dev, "RTC registers at %08x are not free\n", +			davinci_rtc->pbase); +		ret = -EBUSY; +		goto fail1; +	} + +	davinci_rtc->base = ioremap(davinci_rtc->pbase, davinci_rtc->base_size); +	if (!davinci_rtc->base) { +		dev_err(dev, "unable to ioremap MEM resource\n"); +		ret = -ENOMEM; +		goto fail2; +	} + +	davinci_rtc->rtc = rtc_device_register(pdev->name, &pdev->dev, +				    &davinci_rtc_ops, THIS_MODULE); +	if (IS_ERR(davinci_rtc->rtc)) { +		dev_err(dev, "unable to register RTC device, err %ld\n", +				PTR_ERR(davinci_rtc->rtc)); +		goto fail3; +	} + +	rtcif_write(davinci_rtc, PRTCIF_INTFLG_RTCSS, PRTCIF_INTFLG); +	rtcif_write(davinci_rtc, 0, PRTCIF_INTEN); +	rtcss_write(davinci_rtc, 0, PRTCSS_RTC_INTC_EXTENA1); + +	rtcss_write(davinci_rtc, 0, PRTCSS_RTC_CTRL); +	rtcss_write(davinci_rtc, 0, PRTCSS_RTC_CCTRL); + +	ret = request_irq(davinci_rtc->irq, davinci_rtc_interrupt, +			  IRQF_DISABLED, "davinci_rtc", davinci_rtc); +	if (ret < 0) { +		dev_err(dev, "unable to register davinci RTC interrupt\n"); +		goto fail4; +	} + +	/* Enable interrupts */ +	rtcif_write(davinci_rtc, PRTCIF_INTEN_RTCSS, PRTCIF_INTEN); +	rtcss_write(davinci_rtc, PRTCSS_RTC_INTC_EXTENA1_MASK, +			    PRTCSS_RTC_INTC_EXTENA1); + +	rtcss_write(davinci_rtc, PRTCSS_RTC_CCTRL_CAEN, PRTCSS_RTC_CCTRL); + +	platform_set_drvdata(pdev, davinci_rtc); + +	device_init_wakeup(&pdev->dev, 0); + +	return 0; + +fail4: +	rtc_device_unregister(davinci_rtc->rtc); +fail3: +	iounmap(davinci_rtc->base); +fail2: +	release_mem_region(davinci_rtc->pbase, davinci_rtc->base_size); +fail1: +	kfree(davinci_rtc); + +	return ret; +} + +static int __devexit davinci_rtc_remove(struct platform_device *pdev) +{ +	struct davinci_rtc *davinci_rtc = platform_get_drvdata(pdev); + +	device_init_wakeup(&pdev->dev, 0); + +	rtcif_write(davinci_rtc, 0, PRTCIF_INTEN); + +	free_irq(davinci_rtc->irq, davinci_rtc); + +	rtc_device_unregister(davinci_rtc->rtc); + +	iounmap(davinci_rtc->base); +	release_mem_region(davinci_rtc->pbase, davinci_rtc->base_size); + +	platform_set_drvdata(pdev, NULL); + +	kfree(davinci_rtc); + +	return 0; +} + +static struct platform_driver davinci_rtc_driver = { +	.probe		= davinci_rtc_probe, +	.remove		= __devexit_p(davinci_rtc_remove), +	.driver		= { +		.name = "rtc_davinci", +		.owner = THIS_MODULE, +	}, +}; + +static int __init rtc_init(void) +{ +	return platform_driver_probe(&davinci_rtc_driver, davinci_rtc_probe); +} +module_init(rtc_init); + +static void __exit rtc_exit(void) +{ +	platform_driver_unregister(&davinci_rtc_driver); +} +module_exit(rtc_exit); + +MODULE_AUTHOR("Miguel Aguilar <miguel.aguilar@ridgerun.com>"); +MODULE_DESCRIPTION("Texas Instruments DaVinci PRTC Driver"); +MODULE_LICENSE("GPL"); diff --git a/drivers/rtc/rtc-ds1302.c b/drivers/rtc/rtc-ds1302.c index 532acf9b05d..359d1e04626 100644 --- a/drivers/rtc/rtc-ds1302.c +++ b/drivers/rtc/rtc-ds1302.c @@ -16,7 +16,6 @@  #include <linux/rtc.h>  #include <linux/io.h>  #include <linux/bcd.h> -#include <asm/rtc.h>  #define DRV_NAME	"rtc-ds1302"  #define DRV_VERSION	"0.1.1" @@ -34,14 +33,55 @@  #define	RTC_ADDR_MIN	0x01		/* Address of minute register */  #define	RTC_ADDR_SEC	0x00		/* Address of second register */ +#ifdef CONFIG_SH_SECUREEDGE5410 +#include <asm/rtc.h> +#include <mach/snapgear.h> +  #define	RTC_RESET	0x1000  #define	RTC_IODATA	0x0800  #define	RTC_SCLK	0x0400 -#ifdef CONFIG_SH_SECUREEDGE5410 -#include <mach/snapgear.h>  #define set_dp(x)	SECUREEDGE_WRITE_IOPORT(x, 0x1c00)  #define get_dp()	SECUREEDGE_READ_IOPORT() +#define ds1302_set_tx() +#define ds1302_set_rx() + +static inline int ds1302_hw_init(void) +{ +	return 0; +} + +static inline void ds1302_reset(void) +{ +	set_dp(get_dp() & ~(RTC_RESET | RTC_IODATA | RTC_SCLK)); +} + +static inline void ds1302_clock(void) +{ +	set_dp(get_dp() | RTC_SCLK);	/* clock high */ +	set_dp(get_dp() & ~RTC_SCLK);	/* clock low */ +} + +static inline void ds1302_start(void) +{ +	set_dp(get_dp() | RTC_RESET); +} + +static inline void ds1302_stop(void) +{ +	set_dp(get_dp() & ~RTC_RESET); +} + +static inline void ds1302_txbit(int bit) +{ +	set_dp((get_dp() & ~RTC_IODATA) | (bit ? RTC_IODATA : 0)); +} + +static inline int ds1302_rxbit(void) +{ +	return !!(get_dp() & RTC_IODATA); +} +  #else  #error "Add support for your platform"  #endif @@ -50,11 +90,11 @@ static void ds1302_sendbits(unsigned int val)  {  	int i; +	ds1302_set_tx(); +  	for (i = 8; (i); i--, val >>= 1) { -		set_dp((get_dp() & ~RTC_IODATA) | ((val & 0x1) ? -			RTC_IODATA : 0)); -		set_dp(get_dp() | RTC_SCLK);	/* clock high */ -		set_dp(get_dp() & ~RTC_SCLK);	/* clock low */ +		ds1302_txbit(val & 0x1); +		ds1302_clock();  	}  } @@ -63,10 +103,11 @@ static unsigned int ds1302_recvbits(void)  	unsigned int val;  	int i; +	ds1302_set_rx(); +  	for (i = 0, val = 0; (i < 8); i++) { -		val |= (((get_dp() & RTC_IODATA) ? 1 : 0) << i); -		set_dp(get_dp() | RTC_SCLK);	/* clock high */ -		set_dp(get_dp() & ~RTC_SCLK);	/* clock low */ +		val |= (ds1302_rxbit() << i); +		ds1302_clock();  	}  	return val; @@ -76,23 +117,24 @@ static unsigned int ds1302_readbyte(unsigned int addr)  {  	unsigned int val; -	set_dp(get_dp() & ~(RTC_RESET | RTC_IODATA | RTC_SCLK)); +	ds1302_reset(); -	set_dp(get_dp() | RTC_RESET); +	ds1302_start();  	ds1302_sendbits(((addr & 0x3f) << 1) | RTC_CMD_READ);  	val = ds1302_recvbits(); -	set_dp(get_dp() & ~RTC_RESET); +	ds1302_stop();  	return val;  }  static void ds1302_writebyte(unsigned int addr, unsigned int val)  { -	set_dp(get_dp() & ~(RTC_RESET | RTC_IODATA | RTC_SCLK)); -	set_dp(get_dp() | RTC_RESET); +	ds1302_reset(); + +	ds1302_start();  	ds1302_sendbits(((addr & 0x3f) << 1) | RTC_CMD_WRITE);  	ds1302_sendbits(val); -	set_dp(get_dp() & ~RTC_RESET); +	ds1302_stop();  }  static int ds1302_rtc_read_time(struct device *dev, struct rtc_time *tm) @@ -167,13 +209,20 @@ static int __init ds1302_rtc_probe(struct platform_device *pdev)  {  	struct rtc_device *rtc; +	if (ds1302_hw_init()) { +		dev_err(&pdev->dev, "Failed to init communication channel"); +		return -EINVAL; +	} +  	/* Reset */ -	set_dp(get_dp() & ~(RTC_RESET | RTC_IODATA | RTC_SCLK)); +	ds1302_reset();  	/* Write a magic value to the DS1302 RAM, and see if it sticks. */  	ds1302_writebyte(RTC_ADDR_RAM0, 0x42); -	if (ds1302_readbyte(RTC_ADDR_RAM0) != 0x42) +	if (ds1302_readbyte(RTC_ADDR_RAM0) != 0x42) { +		dev_err(&pdev->dev, "Failed to probe");  		return -ENODEV; +	}  	rtc = rtc_device_register("ds1302", &pdev->dev,  					   &ds1302_rtc_ops, THIS_MODULE); diff --git a/drivers/rtc/rtc-ds1305.c b/drivers/rtc/rtc-ds1305.c index 7836c9cec55..48da85e97ca 100644 --- a/drivers/rtc/rtc-ds1305.c +++ b/drivers/rtc/rtc-ds1305.c @@ -542,7 +542,8 @@ static void msg_init(struct spi_message *m, struct spi_transfer *x,  }  static ssize_t -ds1305_nvram_read(struct kobject *kobj, struct bin_attribute *attr, +ds1305_nvram_read(struct file *filp, struct kobject *kobj, +		struct bin_attribute *attr,  		char *buf, loff_t off, size_t count)  {  	struct spi_device	*spi; @@ -572,7 +573,8 @@ ds1305_nvram_read(struct kobject *kobj, struct bin_attribute *attr,  }  static ssize_t -ds1305_nvram_write(struct kobject *kobj, struct bin_attribute *attr, +ds1305_nvram_write(struct file *filp, struct kobject *kobj, +		struct bin_attribute *attr,  		char *buf, loff_t off, size_t count)  {  	struct spi_device	*spi; diff --git a/drivers/rtc/rtc-ds1307.c b/drivers/rtc/rtc-ds1307.c index c4ec5c158aa..de033b7ac21 100644 --- a/drivers/rtc/rtc-ds1307.c +++ b/drivers/rtc/rtc-ds1307.c @@ -556,7 +556,8 @@ static const struct rtc_class_ops ds13xx_rtc_ops = {  #define NVRAM_SIZE	56  static ssize_t -ds1307_nvram_read(struct kobject *kobj, struct bin_attribute *attr, +ds1307_nvram_read(struct file *filp, struct kobject *kobj, +		struct bin_attribute *attr,  		char *buf, loff_t off, size_t count)  {  	struct i2c_client	*client; @@ -580,7 +581,8 @@ ds1307_nvram_read(struct kobject *kobj, struct bin_attribute *attr,  }  static ssize_t -ds1307_nvram_write(struct kobject *kobj, struct bin_attribute *attr, +ds1307_nvram_write(struct file *filp, struct kobject *kobj, +		struct bin_attribute *attr,  		char *buf, loff_t off, size_t count)  {  	struct i2c_client	*client; diff --git a/drivers/rtc/rtc-ds1511.c b/drivers/rtc/rtc-ds1511.c index 06b8566c453..37268e97de4 100644 --- a/drivers/rtc/rtc-ds1511.c +++ b/drivers/rtc/rtc-ds1511.c @@ -423,8 +423,9 @@ static const struct rtc_class_ops ds1511_rtc_ops = {  };   static ssize_t -ds1511_nvram_read(struct kobject *kobj, struct bin_attribute *ba, -				char *buf, loff_t pos, size_t size) +ds1511_nvram_read(struct file *filp, struct kobject *kobj, +		  struct bin_attribute *ba, +		  char *buf, loff_t pos, size_t size)  {  	ssize_t count; @@ -452,8 +453,9 @@ ds1511_nvram_read(struct kobject *kobj, struct bin_attribute *ba,  }   static ssize_t -ds1511_nvram_write(struct kobject *kobj, struct bin_attribute *bin_attr, -				char *buf, loff_t pos, size_t size) +ds1511_nvram_write(struct file *filp, struct kobject *kobj, +		   struct bin_attribute *bin_attr, +		   char *buf, loff_t pos, size_t size)  {  	ssize_t count; diff --git a/drivers/rtc/rtc-ds1553.c b/drivers/rtc/rtc-ds1553.c index 244f9994bcb..ff432e2ca27 100644 --- a/drivers/rtc/rtc-ds1553.c +++ b/drivers/rtc/rtc-ds1553.c @@ -252,7 +252,7 @@ static const struct rtc_class_ops ds1553_rtc_ops = {  	.update_irq_enable	= ds1553_rtc_update_irq_enable,  }; -static ssize_t ds1553_nvram_read(struct kobject *kobj, +static ssize_t ds1553_nvram_read(struct file *filp, struct kobject *kobj,  				 struct bin_attribute *bin_attr,  				 char *buf, loff_t pos, size_t size)  { @@ -267,7 +267,7 @@ static ssize_t ds1553_nvram_read(struct kobject *kobj,  	return count;  } -static ssize_t ds1553_nvram_write(struct kobject *kobj, +static ssize_t ds1553_nvram_write(struct file *filp, struct kobject *kobj,  				  struct bin_attribute *bin_attr,  				  char *buf, loff_t pos, size_t size)  { diff --git a/drivers/rtc/rtc-ds1742.c b/drivers/rtc/rtc-ds1742.c index 2b4b0bc42d6..042630c90dd 100644 --- a/drivers/rtc/rtc-ds1742.c +++ b/drivers/rtc/rtc-ds1742.c @@ -128,7 +128,7 @@ static const struct rtc_class_ops ds1742_rtc_ops = {  	.set_time	= ds1742_rtc_set_time,  }; -static ssize_t ds1742_nvram_read(struct kobject *kobj, +static ssize_t ds1742_nvram_read(struct file *filp, struct kobject *kobj,  				 struct bin_attribute *bin_attr,  				 char *buf, loff_t pos, size_t size)  { @@ -143,7 +143,7 @@ static ssize_t ds1742_nvram_read(struct kobject *kobj,  	return count;  } -static ssize_t ds1742_nvram_write(struct kobject *kobj, +static ssize_t ds1742_nvram_write(struct file *filp, struct kobject *kobj,  				  struct bin_attribute *bin_attr,  				  char *buf, loff_t pos, size_t size)  { diff --git a/drivers/rtc/rtc-isl1208.c b/drivers/rtc/rtc-isl1208.c index 054e05294af..468200c38ec 100644 --- a/drivers/rtc/rtc-isl1208.c +++ b/drivers/rtc/rtc-isl1208.c @@ -462,39 +462,16 @@ isl1208_sysfs_store_usr(struct device *dev,  static DEVICE_ATTR(usr, S_IRUGO | S_IWUSR, isl1208_sysfs_show_usr,  		   isl1208_sysfs_store_usr); -static int -isl1208_sysfs_register(struct device *dev) -{ -	int err; - -	err = device_create_file(dev, &dev_attr_atrim); -	if (err) -		return err; - -	err = device_create_file(dev, &dev_attr_dtrim); -	if (err) { -		device_remove_file(dev, &dev_attr_atrim); -		return err; -	} - -	err = device_create_file(dev, &dev_attr_usr); -	if (err) { -		device_remove_file(dev, &dev_attr_atrim); -		device_remove_file(dev, &dev_attr_dtrim); -	} - -	return 0; -} - -static int -isl1208_sysfs_unregister(struct device *dev) -{ -	device_remove_file(dev, &dev_attr_dtrim); -	device_remove_file(dev, &dev_attr_atrim); -	device_remove_file(dev, &dev_attr_usr); +static struct attribute *isl1208_rtc_attrs[] = { +	&dev_attr_atrim.attr, +	&dev_attr_dtrim.attr, +	&dev_attr_usr.attr, +	NULL +}; -	return 0; -} +static const struct attribute_group isl1208_rtc_sysfs_files = { +	.attrs	= isl1208_rtc_attrs, +};  static int  isl1208_probe(struct i2c_client *client, const struct i2c_device_id *id) @@ -529,7 +506,7 @@ isl1208_probe(struct i2c_client *client, const struct i2c_device_id *id)  		dev_warn(&client->dev, "rtc power failure detected, "  			 "please set clock.\n"); -	rc = isl1208_sysfs_register(&client->dev); +	rc = sysfs_create_group(&client->dev.kobj, &isl1208_rtc_sysfs_files);  	if (rc)  		goto exit_unregister; @@ -546,7 +523,7 @@ isl1208_remove(struct i2c_client *client)  {  	struct rtc_device *rtc = i2c_get_clientdata(client); -	isl1208_sysfs_unregister(&client->dev); +	sysfs_remove_group(&client->dev.kobj, &isl1208_rtc_sysfs_files);  	rtc_device_unregister(rtc);  	return 0; diff --git a/drivers/rtc/rtc-m41t80.c b/drivers/rtc/rtc-m41t80.c index 60fe266f0f4..6dc4e624141 100644 --- a/drivers/rtc/rtc-m41t80.c +++ b/drivers/rtc/rtc-m41t80.c @@ -595,10 +595,6 @@ static void wdt_disable(void)  static ssize_t wdt_write(struct file *file, const char __user *buf,  			 size_t count, loff_t *ppos)  { -	/*  Can't seek (pwrite) on this device -	if (ppos != &file->f_pos) -	return -ESPIPE; -	*/  	if (count) {  		wdt_ping();  		return 1; @@ -623,7 +619,7 @@ static ssize_t wdt_read(struct file *file, char __user *buf,   *	according to their available features. We only actually usefully support   *	querying capabilities and current status.   */ -static int wdt_ioctl(struct inode *inode, struct file *file, unsigned int cmd, +static int wdt_ioctl(struct file *file, unsigned int cmd,  		     unsigned long arg)  {  	int new_margin, rv; @@ -676,6 +672,18 @@ static int wdt_ioctl(struct inode *inode, struct file *file, unsigned int cmd,  	return -ENOTTY;  } +static long wdt_unlocked_ioctl(struct file *file, unsigned int cmd, +			       unsigned long arg) +{ +	int ret; + +	lock_kernel(); +	ret = wdt_ioctl(file, cmd, arg); +	unlock_kernel(); + +	return ret; +} +  /**   *	wdt_open:   *	@inode: inode of device @@ -695,7 +703,7 @@ static int wdt_open(struct inode *inode, struct file *file)  		 */  		wdt_is_open = 1;  		unlock_kernel(); -		return 0; +		return nonseekable_open(inode, file);  	}  	return -ENODEV;  } @@ -736,7 +744,7 @@ static int wdt_notify_sys(struct notifier_block *this, unsigned long code,  static const struct file_operations wdt_fops = {  	.owner	= THIS_MODULE,  	.read	= wdt_read, -	.ioctl	= wdt_ioctl, +	.unlocked_ioctl = wdt_unlocked_ioctl,  	.write	= wdt_write,  	.open	= wdt_open,  	.release = wdt_release, diff --git a/drivers/rtc/rtc-m48t59.c b/drivers/rtc/rtc-m48t59.c index 365ff3ac234..be8359fdb65 100644 --- a/drivers/rtc/rtc-m48t59.c +++ b/drivers/rtc/rtc-m48t59.c @@ -343,7 +343,7 @@ static const struct rtc_class_ops m48t02_rtc_ops = {  	.set_time	= m48t59_rtc_set_time,  }; -static ssize_t m48t59_nvram_read(struct kobject *kobj, +static ssize_t m48t59_nvram_read(struct file *filp, struct kobject *kobj,  				struct bin_attribute *bin_attr,  				char *buf, loff_t pos, size_t size)  { @@ -363,7 +363,7 @@ static ssize_t m48t59_nvram_read(struct kobject *kobj,  	return cnt;  } -static ssize_t m48t59_nvram_write(struct kobject *kobj, +static ssize_t m48t59_nvram_write(struct file *filp, struct kobject *kobj,  				struct bin_attribute *bin_attr,  				char *buf, loff_t pos, size_t size)  { diff --git a/drivers/rtc/rtc-mxc.c b/drivers/rtc/rtc-mxc.c index d71fe61db1d..25ec921db07 100644 --- a/drivers/rtc/rtc-mxc.c +++ b/drivers/rtc/rtc-mxc.c @@ -379,7 +379,6 @@ static struct rtc_class_ops mxc_rtc_ops = {  static int __init mxc_rtc_probe(struct platform_device *pdev)  { -	struct clk *clk;  	struct resource *res;  	struct rtc_device *rtc;  	struct rtc_plat_data *pdata = NULL; @@ -402,14 +401,15 @@ static int __init mxc_rtc_probe(struct platform_device *pdev)  	pdata->ioaddr = devm_ioremap(&pdev->dev, res->start,  				     resource_size(res)); -	clk = clk_get(&pdev->dev, "ckil"); -	if (IS_ERR(clk)) { -		ret = PTR_ERR(clk); +	pdata->clk = clk_get(&pdev->dev, "rtc"); +	if (IS_ERR(pdata->clk)) { +		dev_err(&pdev->dev, "unable to get clock!\n"); +		ret = PTR_ERR(pdata->clk);  		goto exit_free_pdata;  	} -	rate = clk_get_rate(clk); -	clk_put(clk); +	clk_enable(pdata->clk); +	rate = clk_get_rate(pdata->clk);  	if (rate == 32768)  		reg = RTC_INPUT_CLK_32768HZ; @@ -420,7 +420,7 @@ static int __init mxc_rtc_probe(struct platform_device *pdev)  	else {  		dev_err(&pdev->dev, "rtc clock is not valid (%lu)\n", rate);  		ret = -EINVAL; -		goto exit_free_pdata; +		goto exit_put_clk;  	}  	reg |= RTC_ENABLE_BIT; @@ -428,18 +428,9 @@ static int __init mxc_rtc_probe(struct platform_device *pdev)  	if (((readw(pdata->ioaddr + RTC_RTCCTL)) & RTC_ENABLE_BIT) == 0) {  		dev_err(&pdev->dev, "hardware module can't be enabled!\n");  		ret = -EIO; -		goto exit_free_pdata; -	} - -	pdata->clk = clk_get(&pdev->dev, "rtc"); -	if (IS_ERR(pdata->clk)) { -		dev_err(&pdev->dev, "unable to get clock!\n"); -		ret = PTR_ERR(pdata->clk); -		goto exit_free_pdata; +		goto exit_put_clk;  	} -	clk_enable(pdata->clk); -  	rtc = rtc_device_register(pdev->name, &pdev->dev, &mxc_rtc_ops,  				  THIS_MODULE);  	if (IS_ERR(rtc)) { diff --git a/drivers/rtc/rtc-rx8581.c b/drivers/rtc/rtc-rx8581.c index c9522f3bc21..9718aaaa821 100644 --- a/drivers/rtc/rtc-rx8581.c +++ b/drivers/rtc/rtc-rx8581.c @@ -1,8 +1,8 @@  /*   * An I2C driver for the Epson RX8581 RTC   * - * Author: Martyn Welch <martyn.welch@gefanuc.com> - * Copyright 2008 GE Fanuc Intelligent Platforms Embedded Systems, Inc. + * Author: Martyn Welch <martyn.welch@ge.com> + * Copyright 2008 GE Intelligent Platforms Embedded Systems, Inc.   *   * This program is free software; you can redistribute it and/or modify   * it under the terms of the GNU General Public License version 2 as @@ -272,7 +272,7 @@ static void __exit rx8581_exit(void)  	i2c_del_driver(&rx8581_driver);  } -MODULE_AUTHOR("Martyn Welch <martyn.welch@gefanuc.com>"); +MODULE_AUTHOR("Martyn Welch <martyn.welch@ge.com>");  MODULE_DESCRIPTION("Epson RX-8581 RTC driver");  MODULE_LICENSE("GPL");  MODULE_VERSION(DRV_VERSION); diff --git a/drivers/rtc/rtc-s3c.c b/drivers/rtc/rtc-s3c.c index 4969b6059c8..e5972b2c17b 100644 --- a/drivers/rtc/rtc-s3c.c +++ b/drivers/rtc/rtc-s3c.c @@ -29,6 +29,11 @@  #include <asm/irq.h>  #include <plat/regs-rtc.h> +enum s3c_cpu_type { +	TYPE_S3C2410, +	TYPE_S3C64XX, +}; +  /* I have yet to find an S3C implementation with more than one   * of these rtc blocks in */ @@ -37,6 +42,7 @@ static struct resource *s3c_rtc_mem;  static void __iomem *s3c_rtc_base;  static int s3c_rtc_alarmno = NO_IRQ;  static int s3c_rtc_tickno  = NO_IRQ; +static enum s3c_cpu_type s3c_rtc_cpu_type;  static DEFINE_SPINLOCK(s3c_rtc_pie_lock); @@ -80,12 +86,25 @@ static int s3c_rtc_setpie(struct device *dev, int enabled)  	pr_debug("%s: pie=%d\n", __func__, enabled);  	spin_lock_irq(&s3c_rtc_pie_lock); -	tmp = readb(s3c_rtc_base + S3C2410_TICNT) & ~S3C2410_TICNT_ENABLE; -	if (enabled) -		tmp |= S3C2410_TICNT_ENABLE; +	if (s3c_rtc_cpu_type == TYPE_S3C64XX) { +		tmp = readb(s3c_rtc_base + S3C2410_RTCCON); +		tmp &= ~S3C64XX_RTCCON_TICEN; + +		if (enabled) +			tmp |= S3C64XX_RTCCON_TICEN; + +		writeb(tmp, s3c_rtc_base + S3C2410_RTCCON); +	} else { +		tmp = readb(s3c_rtc_base + S3C2410_TICNT); +		tmp &= ~S3C2410_TICNT_ENABLE; + +		if (enabled) +			tmp |= S3C2410_TICNT_ENABLE; + +		writeb(tmp, s3c_rtc_base + S3C2410_TICNT); +	} -	writeb(tmp, s3c_rtc_base + S3C2410_TICNT);  	spin_unlock_irq(&s3c_rtc_pie_lock);  	return 0; @@ -93,15 +112,21 @@ static int s3c_rtc_setpie(struct device *dev, int enabled)  static int s3c_rtc_setfreq(struct device *dev, int freq)  { -	unsigned int tmp; +	struct platform_device *pdev = to_platform_device(dev); +	struct rtc_device *rtc_dev = platform_get_drvdata(pdev); +	unsigned int tmp = 0;  	if (!is_power_of_2(freq))  		return -EINVAL;  	spin_lock_irq(&s3c_rtc_pie_lock); -	tmp = readb(s3c_rtc_base + S3C2410_TICNT) & S3C2410_TICNT_ENABLE; -	tmp |= (128 / freq)-1; +	if (s3c_rtc_cpu_type == TYPE_S3C2410) { +		tmp = readb(s3c_rtc_base + S3C2410_TICNT); +		tmp &= S3C2410_TICNT_ENABLE; +	} + +	tmp |= (rtc_dev->max_user_freq / freq)-1;  	writeb(tmp, s3c_rtc_base + S3C2410_TICNT);  	spin_unlock_irq(&s3c_rtc_pie_lock); @@ -283,10 +308,17 @@ static int s3c_rtc_setalarm(struct device *dev, struct rtc_wkalrm *alrm)  static int s3c_rtc_proc(struct device *dev, struct seq_file *seq)  { -	unsigned int ticnt = readb(s3c_rtc_base + S3C2410_TICNT); +	unsigned int ticnt; -	seq_printf(seq, "periodic_IRQ\t: %s\n", -		     (ticnt & S3C2410_TICNT_ENABLE) ? "yes" : "no" ); +	if (s3c_rtc_cpu_type == TYPE_S3C64XX) { +		ticnt = readb(s3c_rtc_base + S3C2410_RTCCON); +		ticnt &= S3C64XX_RTCCON_TICEN; +	} else { +		ticnt = readb(s3c_rtc_base + S3C2410_TICNT); +		ticnt &= S3C2410_TICNT_ENABLE; +	} + +	seq_printf(seq, "periodic_IRQ\t: %s\n", ticnt  ? "yes" : "no");  	return 0;  } @@ -353,10 +385,16 @@ static void s3c_rtc_enable(struct platform_device *pdev, int en)  	if (!en) {  		tmp = readb(base + S3C2410_RTCCON); -		writeb(tmp & ~S3C2410_RTCCON_RTCEN, base + S3C2410_RTCCON); - -		tmp = readb(base + S3C2410_TICNT); -		writeb(tmp & ~S3C2410_TICNT_ENABLE, base + S3C2410_TICNT); +		if (s3c_rtc_cpu_type == TYPE_S3C64XX) +			tmp &= ~S3C64XX_RTCCON_TICEN; +		tmp &= ~S3C2410_RTCCON_RTCEN; +		writeb(tmp, base + S3C2410_RTCCON); + +		if (s3c_rtc_cpu_type == TYPE_S3C2410) { +			tmp = readb(base + S3C2410_TICNT); +			tmp &= ~S3C2410_TICNT_ENABLE; +			writeb(tmp, base + S3C2410_TICNT); +		}  	} else {  		/* re-enable the device, and check it is ok */ @@ -472,7 +510,12 @@ static int __devinit s3c_rtc_probe(struct platform_device *pdev)  		goto err_nortc;  	} -	rtc->max_user_freq = 128; +	if (s3c_rtc_cpu_type == TYPE_S3C64XX) +		rtc->max_user_freq = 32768; +	else +		rtc->max_user_freq = 128; + +	s3c_rtc_cpu_type = platform_get_device_id(pdev)->driver_data;  	platform_set_drvdata(pdev, rtc);  	return 0; @@ -492,20 +535,30 @@ static int __devinit s3c_rtc_probe(struct platform_device *pdev)  /* RTC Power management control */ -static int ticnt_save; +static int ticnt_save, ticnt_en_save;  static int s3c_rtc_suspend(struct platform_device *pdev, pm_message_t state)  {  	/* save TICNT for anyone using periodic interrupts */  	ticnt_save = readb(s3c_rtc_base + S3C2410_TICNT); +	if (s3c_rtc_cpu_type == TYPE_S3C64XX) { +		ticnt_en_save = readb(s3c_rtc_base + S3C2410_RTCCON); +		ticnt_en_save &= S3C64XX_RTCCON_TICEN; +	}  	s3c_rtc_enable(pdev, 0);  	return 0;  }  static int s3c_rtc_resume(struct platform_device *pdev)  { +	unsigned int tmp; +  	s3c_rtc_enable(pdev, 1);  	writeb(ticnt_save, s3c_rtc_base + S3C2410_TICNT); +	if (s3c_rtc_cpu_type == TYPE_S3C64XX && ticnt_en_save) { +		tmp = readb(s3c_rtc_base + S3C2410_RTCCON); +		writeb(tmp | ticnt_en_save, s3c_rtc_base + S3C2410_RTCCON); +	}  	return 0;  }  #else @@ -513,13 +566,27 @@ static int s3c_rtc_resume(struct platform_device *pdev)  #define s3c_rtc_resume  NULL  #endif -static struct platform_driver s3c2410_rtc_driver = { +static struct platform_device_id s3c_rtc_driver_ids[] = { +	{ +		.name		= "s3c2410-rtc", +		.driver_data	= TYPE_S3C2410, +	}, { +		.name		= "s3c64xx-rtc", +		.driver_data	= TYPE_S3C64XX, +	}, +	{ } +}; + +MODULE_DEVICE_TABLE(platform, s3c_rtc_driver_ids); + +static struct platform_driver s3c_rtc_driver = {  	.probe		= s3c_rtc_probe,  	.remove		= __devexit_p(s3c_rtc_remove),  	.suspend	= s3c_rtc_suspend,  	.resume		= s3c_rtc_resume, +	.id_table	= s3c_rtc_driver_ids,  	.driver		= { -		.name	= "s3c2410-rtc", +		.name	= "s3c-rtc",  		.owner	= THIS_MODULE,  	},  }; @@ -529,12 +596,12 @@ static char __initdata banner[] = "S3C24XX RTC, (c) 2004,2006 Simtec Electronics  static int __init s3c_rtc_init(void)  {  	printk(banner); -	return platform_driver_register(&s3c2410_rtc_driver); +	return platform_driver_register(&s3c_rtc_driver);  }  static void __exit s3c_rtc_exit(void)  { -	platform_driver_unregister(&s3c2410_rtc_driver); +	platform_driver_unregister(&s3c_rtc_driver);  }  module_init(s3c_rtc_init); diff --git a/drivers/rtc/rtc-stk17ta8.c b/drivers/rtc/rtc-stk17ta8.c index 875ba099e7a..3b943673cd3 100644 --- a/drivers/rtc/rtc-stk17ta8.c +++ b/drivers/rtc/rtc-stk17ta8.c @@ -1,7 +1,7 @@  /*   * A RTC driver for the Simtek STK17TA8   * - * By Thomas Hommel <thomas.hommel@gefanuc.com> + * By Thomas Hommel <thomas.hommel@ge.com>   *   * Based on the DS1553 driver from   * Atsushi Nemoto <anemo@mba.ocn.ne.jp> @@ -244,7 +244,7 @@ static const struct rtc_class_ops stk17ta8_rtc_ops = {  	.alarm_irq_enable	= stk17ta8_rtc_alarm_irq_enable,  }; -static ssize_t stk17ta8_nvram_read(struct kobject *kobj, +static ssize_t stk17ta8_nvram_read(struct file *filp, struct kobject *kobj,  				 struct bin_attribute *attr, char *buf,  				 loff_t pos, size_t size)  { @@ -259,7 +259,7 @@ static ssize_t stk17ta8_nvram_read(struct kobject *kobj,  	return count;  } -static ssize_t stk17ta8_nvram_write(struct kobject *kobj, +static ssize_t stk17ta8_nvram_write(struct file *filp, struct kobject *kobj,  				  struct bin_attribute *attr, char *buf,  				  loff_t pos, size_t size)  { @@ -382,7 +382,7 @@ static __exit void stk17ta8_exit(void)  module_init(stk17ta8_init);  module_exit(stk17ta8_exit); -MODULE_AUTHOR("Thomas Hommel <thomas.hommel@gefanuc.com>"); +MODULE_AUTHOR("Thomas Hommel <thomas.hommel@ge.com>");  MODULE_DESCRIPTION("Simtek STK17TA8 RTC driver");  MODULE_LICENSE("GPL");  MODULE_VERSION(DRV_VERSION); diff --git a/drivers/rtc/rtc-tx4939.c b/drivers/rtc/rtc-tx4939.c index 20bfc64a15c..ec6313d1535 100644 --- a/drivers/rtc/rtc-tx4939.c +++ b/drivers/rtc/rtc-tx4939.c @@ -188,7 +188,7 @@ static const struct rtc_class_ops tx4939_rtc_ops = {  	.alarm_irq_enable	= tx4939_rtc_alarm_irq_enable,  }; -static ssize_t tx4939_rtc_nvram_read(struct kobject *kobj, +static ssize_t tx4939_rtc_nvram_read(struct file *filp, struct kobject *kobj,  				     struct bin_attribute *bin_attr,  				     char *buf, loff_t pos, size_t size)  { @@ -207,7 +207,7 @@ static ssize_t tx4939_rtc_nvram_read(struct kobject *kobj,  	return count;  } -static ssize_t tx4939_rtc_nvram_write(struct kobject *kobj, +static ssize_t tx4939_rtc_nvram_write(struct file *filp, struct kobject *kobj,  				      struct bin_attribute *bin_attr,  				      char *buf, loff_t pos, size_t size)  { diff --git a/drivers/rtc/rtc-wm831x.c b/drivers/rtc/rtc-wm831x.c index b16cfe57a48..82931dc65c0 100644 --- a/drivers/rtc/rtc-wm831x.c +++ b/drivers/rtc/rtc-wm831x.c @@ -449,17 +449,17 @@ static int wm831x_rtc_probe(struct platform_device *pdev)  		goto err;  	} -	ret = wm831x_request_irq(wm831x, per_irq, wm831x_per_irq, -				 IRQF_TRIGGER_RISING, "wm831x_rtc_per", -				 wm831x_rtc); +	ret = request_threaded_irq(per_irq, NULL, wm831x_per_irq, +				   IRQF_TRIGGER_RISING, "RTC period", +				   wm831x_rtc);  	if (ret != 0) {  		dev_err(&pdev->dev, "Failed to request periodic IRQ %d: %d\n",  			per_irq, ret);  	} -	ret = wm831x_request_irq(wm831x, alm_irq, wm831x_alm_irq, -				 IRQF_TRIGGER_RISING, "wm831x_rtc_alm", -				 wm831x_rtc); +	ret = request_threaded_irq(alm_irq, NULL, wm831x_alm_irq, +				   IRQF_TRIGGER_RISING, "RTC alarm", +				   wm831x_rtc);  	if (ret != 0) {  		dev_err(&pdev->dev, "Failed to request alarm IRQ %d: %d\n",  			alm_irq, ret); @@ -478,8 +478,8 @@ static int __devexit wm831x_rtc_remove(struct platform_device *pdev)  	int per_irq = platform_get_irq_byname(pdev, "PER");  	int alm_irq = platform_get_irq_byname(pdev, "ALM"); -	wm831x_free_irq(wm831x_rtc->wm831x, alm_irq, wm831x_rtc); -	wm831x_free_irq(wm831x_rtc->wm831x, per_irq, wm831x_rtc); +	free_irq(alm_irq, wm831x_rtc); +	free_irq(per_irq, wm831x_rtc);  	rtc_device_unregister(wm831x_rtc->rtc);  	kfree(wm831x_rtc); | 
