diff options
Diffstat (limited to 'drivers/rtc')
| -rw-r--r-- | drivers/rtc/Kconfig | 10 | ||||
| -rw-r--r-- | drivers/rtc/Makefile | 1 | ||||
| -rw-r--r-- | drivers/rtc/class.c | 1 | ||||
| -rw-r--r-- | drivers/rtc/rtc-at91sam9.c | 2 | ||||
| -rw-r--r-- | drivers/rtc/rtc-coh901331.c | 5 | ||||
| -rw-r--r-- | drivers/rtc/rtc-ep93xx.c | 71 | ||||
| -rw-r--r-- | drivers/rtc/rtc-max8925.c | 314 | ||||
| -rw-r--r-- | drivers/rtc/rtc-mc13783.c | 214 | ||||
| -rw-r--r-- | drivers/rtc/rtc-mxc.c | 7 | ||||
| -rw-r--r-- | drivers/rtc/rtc-pcf2123.c | 2 | ||||
| -rw-r--r-- | drivers/rtc/rtc-twl.c | 4 | ||||
| -rw-r--r-- | drivers/rtc/rtc-wm8350.c | 11 |
12 files changed, 563 insertions, 79 deletions
diff --git a/drivers/rtc/Kconfig b/drivers/rtc/Kconfig index 2bb8a8b7ffa..6a130375943 100644 --- a/drivers/rtc/Kconfig +++ b/drivers/rtc/Kconfig | |||
| @@ -175,6 +175,16 @@ config RTC_DRV_MAX6900 | |||
| 175 | This driver can also be built as a module. If so, the module | 175 | This driver can also be built as a module. If so, the module |
| 176 | will be called rtc-max6900. | 176 | will be called rtc-max6900. |
| 177 | 177 | ||
| 178 | config RTC_DRV_MAX8925 | ||
| 179 | tristate "Maxim MAX8925" | ||
| 180 | depends on MFD_MAX8925 | ||
| 181 | help | ||
| 182 | If you say yes here you will get support for the | ||
| 183 | RTC of Maxim MAX8925 PMIC. | ||
| 184 | |||
| 185 | This driver can also be built as a module. If so, the module | ||
| 186 | will be called rtc-max8925. | ||
| 187 | |||
| 178 | config RTC_DRV_RS5C372 | 188 | config RTC_DRV_RS5C372 |
| 179 | tristate "Ricoh R2025S/D, RS5C372A/B, RV5C386, RV5C387A" | 189 | tristate "Ricoh R2025S/D, RS5C372A/B, RV5C386, RV5C387A" |
| 180 | help | 190 | help |
diff --git a/drivers/rtc/Makefile b/drivers/rtc/Makefile index b7148afb8f5..44ef194a957 100644 --- a/drivers/rtc/Makefile +++ b/drivers/rtc/Makefile | |||
| @@ -52,6 +52,7 @@ obj-$(CONFIG_RTC_DRV_M48T59) += rtc-m48t59.o | |||
| 52 | obj-$(CONFIG_RTC_DRV_M48T86) += rtc-m48t86.o | 52 | obj-$(CONFIG_RTC_DRV_M48T86) += rtc-m48t86.o |
| 53 | obj-$(CONFIG_RTC_MXC) += rtc-mxc.o | 53 | obj-$(CONFIG_RTC_MXC) += rtc-mxc.o |
| 54 | obj-$(CONFIG_RTC_DRV_MAX6900) += rtc-max6900.o | 54 | obj-$(CONFIG_RTC_DRV_MAX6900) += rtc-max6900.o |
| 55 | obj-$(CONFIG_RTC_DRV_MAX8925) += rtc-max8925.o | ||
| 55 | obj-$(CONFIG_RTC_DRV_MAX6902) += rtc-max6902.o | 56 | obj-$(CONFIG_RTC_DRV_MAX6902) += rtc-max6902.o |
| 56 | obj-$(CONFIG_RTC_DRV_MC13783) += rtc-mc13783.o | 57 | obj-$(CONFIG_RTC_DRV_MC13783) += rtc-mc13783.o |
| 57 | obj-$(CONFIG_RTC_DRV_MSM6242) += rtc-msm6242.o | 58 | obj-$(CONFIG_RTC_DRV_MSM6242) += rtc-msm6242.o |
diff --git a/drivers/rtc/class.c b/drivers/rtc/class.c index be5a6b73e60..40845c7e932 100644 --- a/drivers/rtc/class.c +++ b/drivers/rtc/class.c | |||
| @@ -226,6 +226,7 @@ static void __exit rtc_exit(void) | |||
| 226 | { | 226 | { |
| 227 | rtc_dev_exit(); | 227 | rtc_dev_exit(); |
| 228 | class_destroy(rtc_class); | 228 | class_destroy(rtc_class); |
| 229 | idr_destroy(&rtc_idr); | ||
| 229 | } | 230 | } |
| 230 | 231 | ||
| 231 | subsys_initcall(rtc_init); | 232 | subsys_initcall(rtc_init); |
diff --git a/drivers/rtc/rtc-at91sam9.c b/drivers/rtc/rtc-at91sam9.c index 86c61f14351..78a018b5c94 100644 --- a/drivers/rtc/rtc-at91sam9.c +++ b/drivers/rtc/rtc-at91sam9.c | |||
| @@ -161,7 +161,7 @@ static int at91_rtc_readalarm(struct device *dev, struct rtc_wkalrm *alrm) | |||
| 161 | if (offset == 0) | 161 | if (offset == 0) |
| 162 | return -EILSEQ; | 162 | return -EILSEQ; |
| 163 | 163 | ||
| 164 | memset(alrm, 0, sizeof(alrm)); | 164 | memset(alrm, 0, sizeof(*alrm)); |
| 165 | if (alarm != ALARM_DISABLED && offset != 0) { | 165 | if (alarm != ALARM_DISABLED && offset != 0) { |
| 166 | rtc_time_to_tm(offset + alarm, tm); | 166 | rtc_time_to_tm(offset + alarm, tm); |
| 167 | 167 | ||
diff --git a/drivers/rtc/rtc-coh901331.c b/drivers/rtc/rtc-coh901331.c index 03ea530981d..44c4399ee71 100644 --- a/drivers/rtc/rtc-coh901331.c +++ b/drivers/rtc/rtc-coh901331.c | |||
| @@ -271,12 +271,13 @@ static int coh901331_resume(struct platform_device *pdev) | |||
| 271 | { | 271 | { |
| 272 | struct coh901331_port *rtap = dev_get_drvdata(&pdev->dev); | 272 | struct coh901331_port *rtap = dev_get_drvdata(&pdev->dev); |
| 273 | 273 | ||
| 274 | if (device_may_wakeup(&pdev->dev)) | 274 | if (device_may_wakeup(&pdev->dev)) { |
| 275 | disable_irq_wake(rtap->irq); | 275 | disable_irq_wake(rtap->irq); |
| 276 | else | 276 | } else { |
| 277 | clk_enable(rtap->clk); | 277 | clk_enable(rtap->clk); |
| 278 | writel(rtap->irqmaskstore, rtap->virtbase + COH901331_IRQ_MASK); | 278 | writel(rtap->irqmaskstore, rtap->virtbase + COH901331_IRQ_MASK); |
| 279 | clk_disable(rtap->clk); | 279 | clk_disable(rtap->clk); |
| 280 | } | ||
| 280 | return 0; | 281 | return 0; |
| 281 | } | 282 | } |
| 282 | #else | 283 | #else |
diff --git a/drivers/rtc/rtc-ep93xx.c b/drivers/rtc/rtc-ep93xx.c index 9da02d108b7..91bde976bc0 100644 --- a/drivers/rtc/rtc-ep93xx.c +++ b/drivers/rtc/rtc-ep93xx.c | |||
| @@ -115,6 +115,15 @@ static ssize_t ep93xx_rtc_show_comp_delete(struct device *dev, | |||
| 115 | } | 115 | } |
| 116 | static DEVICE_ATTR(comp_delete, S_IRUGO, ep93xx_rtc_show_comp_delete, NULL); | 116 | static DEVICE_ATTR(comp_delete, S_IRUGO, ep93xx_rtc_show_comp_delete, NULL); |
| 117 | 117 | ||
| 118 | static struct attribute *ep93xx_rtc_attrs[] = { | ||
| 119 | &dev_attr_comp_preload.attr, | ||
| 120 | &dev_attr_comp_delete.attr, | ||
| 121 | NULL | ||
| 122 | }; | ||
| 123 | |||
| 124 | static const struct attribute_group ep93xx_rtc_sysfs_files = { | ||
| 125 | .attrs = ep93xx_rtc_attrs, | ||
| 126 | }; | ||
| 118 | 127 | ||
| 119 | static int __init ep93xx_rtc_probe(struct platform_device *pdev) | 128 | static int __init ep93xx_rtc_probe(struct platform_device *pdev) |
| 120 | { | 129 | { |
| @@ -123,27 +132,22 @@ static int __init ep93xx_rtc_probe(struct platform_device *pdev) | |||
| 123 | struct rtc_device *rtc; | 132 | struct rtc_device *rtc; |
| 124 | int err; | 133 | int err; |
| 125 | 134 | ||
| 126 | ep93xx_rtc = kzalloc(sizeof(struct ep93xx_rtc), GFP_KERNEL); | 135 | ep93xx_rtc = devm_kzalloc(&pdev->dev, sizeof(*ep93xx_rtc), GFP_KERNEL); |
| 127 | if (ep93xx_rtc == NULL) | 136 | if (!ep93xx_rtc) |
| 128 | return -ENOMEM; | 137 | return -ENOMEM; |
| 129 | 138 | ||
| 130 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 139 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
| 131 | if (res == NULL) { | 140 | if (!res) |
| 132 | err = -ENXIO; | 141 | return -ENXIO; |
| 133 | goto fail_free; | ||
| 134 | } | ||
| 135 | 142 | ||
| 136 | res = request_mem_region(res->start, resource_size(res), pdev->name); | 143 | if (!devm_request_mem_region(&pdev->dev, res->start, |
| 137 | if (res == NULL) { | 144 | resource_size(res), pdev->name)) |
| 138 | err = -EBUSY; | 145 | return -EBUSY; |
| 139 | goto fail_free; | ||
| 140 | } | ||
| 141 | 146 | ||
| 142 | ep93xx_rtc->mmio_base = ioremap(res->start, resource_size(res)); | 147 | ep93xx_rtc->mmio_base = devm_ioremap(&pdev->dev, res->start, |
| 143 | if (ep93xx_rtc->mmio_base == NULL) { | 148 | resource_size(res)); |
| 144 | err = -ENXIO; | 149 | if (!ep93xx_rtc->mmio_base) |
| 145 | goto fail; | 150 | return -ENXIO; |
| 146 | } | ||
| 147 | 151 | ||
| 148 | pdev->dev.platform_data = ep93xx_rtc; | 152 | pdev->dev.platform_data = ep93xx_rtc; |
| 149 | 153 | ||
| @@ -151,53 +155,34 @@ static int __init ep93xx_rtc_probe(struct platform_device *pdev) | |||
| 151 | &pdev->dev, &ep93xx_rtc_ops, THIS_MODULE); | 155 | &pdev->dev, &ep93xx_rtc_ops, THIS_MODULE); |
| 152 | if (IS_ERR(rtc)) { | 156 | if (IS_ERR(rtc)) { |
| 153 | err = PTR_ERR(rtc); | 157 | err = PTR_ERR(rtc); |
| 154 | goto fail; | 158 | goto exit; |
| 155 | } | 159 | } |
| 156 | 160 | ||
| 157 | platform_set_drvdata(pdev, rtc); | 161 | platform_set_drvdata(pdev, rtc); |
| 158 | 162 | ||
| 159 | err = device_create_file(&pdev->dev, &dev_attr_comp_preload); | 163 | err = sysfs_create_group(&pdev->dev.kobj, &ep93xx_rtc_sysfs_files); |
| 160 | if (err) | 164 | if (err) |
| 161 | goto fail; | 165 | goto fail; |
| 162 | err = device_create_file(&pdev->dev, &dev_attr_comp_delete); | ||
| 163 | if (err) { | ||
| 164 | device_remove_file(&pdev->dev, &dev_attr_comp_preload); | ||
| 165 | goto fail; | ||
| 166 | } | ||
| 167 | 166 | ||
| 168 | return 0; | 167 | return 0; |
| 169 | 168 | ||
| 170 | fail: | 169 | fail: |
| 171 | if (ep93xx_rtc->mmio_base) { | 170 | platform_set_drvdata(pdev, NULL); |
| 172 | iounmap(ep93xx_rtc->mmio_base); | 171 | rtc_device_unregister(rtc); |
| 173 | pdev->dev.platform_data = NULL; | 172 | exit: |
| 174 | } | 173 | pdev->dev.platform_data = NULL; |
| 175 | release_mem_region(res->start, resource_size(res)); | ||
| 176 | fail_free: | ||
| 177 | kfree(ep93xx_rtc); | ||
| 178 | return err; | 174 | return err; |
| 179 | } | 175 | } |
| 180 | 176 | ||
| 181 | static int __exit ep93xx_rtc_remove(struct platform_device *pdev) | 177 | static int __exit ep93xx_rtc_remove(struct platform_device *pdev) |
| 182 | { | 178 | { |
| 183 | struct rtc_device *rtc = platform_get_drvdata(pdev); | 179 | struct rtc_device *rtc = platform_get_drvdata(pdev); |
| 184 | struct ep93xx_rtc *ep93xx_rtc = pdev->dev.platform_data; | ||
| 185 | struct resource *res; | ||
| 186 | |||
| 187 | /* cleanup sysfs */ | ||
| 188 | device_remove_file(&pdev->dev, &dev_attr_comp_delete); | ||
| 189 | device_remove_file(&pdev->dev, &dev_attr_comp_preload); | ||
| 190 | 180 | ||
| 181 | sysfs_remove_group(&pdev->dev.kobj, &ep93xx_rtc_sysfs_files); | ||
| 182 | platform_set_drvdata(pdev, NULL); | ||
| 191 | rtc_device_unregister(rtc); | 183 | rtc_device_unregister(rtc); |
| 192 | |||
| 193 | iounmap(ep93xx_rtc->mmio_base); | ||
| 194 | pdev->dev.platform_data = NULL; | 184 | pdev->dev.platform_data = NULL; |
| 195 | 185 | ||
| 196 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | ||
| 197 | release_mem_region(res->start, resource_size(res)); | ||
| 198 | |||
| 199 | platform_set_drvdata(pdev, NULL); | ||
| 200 | |||
| 201 | return 0; | 186 | return 0; |
| 202 | } | 187 | } |
| 203 | 188 | ||
diff --git a/drivers/rtc/rtc-max8925.c b/drivers/rtc/rtc-max8925.c new file mode 100644 index 00000000000..acdbb176018 --- /dev/null +++ b/drivers/rtc/rtc-max8925.c | |||
| @@ -0,0 +1,314 @@ | |||
| 1 | /* | ||
| 2 | * RTC driver for Maxim MAX8925 | ||
| 3 | * | ||
| 4 | * Copyright (C) 2009-2010 Marvell International Ltd. | ||
| 5 | * Haojian Zhuang <haojian.zhuang@marvell.com> | ||
| 6 | * | ||
| 7 | * This program is free software; you can redistribute it and/or modify | ||
| 8 | * it under the terms of the GNU General Public License version 2 as | ||
| 9 | * published by the Free Software Foundation. | ||
| 10 | */ | ||
| 11 | |||
| 12 | #include <linux/module.h> | ||
| 13 | #include <linux/i2c.h> | ||
| 14 | #include <linux/rtc.h> | ||
| 15 | #include <linux/platform_device.h> | ||
| 16 | #include <linux/mfd/max8925.h> | ||
| 17 | |||
| 18 | enum { | ||
| 19 | RTC_SEC = 0, | ||
| 20 | RTC_MIN, | ||
| 21 | RTC_HOUR, | ||
| 22 | RTC_WEEKDAY, | ||
| 23 | RTC_DATE, | ||
| 24 | RTC_MONTH, | ||
| 25 | RTC_YEAR1, | ||
| 26 | RTC_YEAR2, | ||
| 27 | }; | ||
| 28 | |||
| 29 | #define MAX8925_RTC_SEC 0x00 | ||
| 30 | #define MAX8925_RTC_MIN 0x01 | ||
| 31 | #define MAX8925_RTC_HOUR 0x02 | ||
| 32 | #define MAX8925_RTC_WEEKDAY 0x03 | ||
| 33 | #define MAX8925_RTC_DATE 0x04 | ||
| 34 | #define MAX8925_RTC_MONTH 0x05 | ||
| 35 | #define MAX8925_RTC_YEAR1 0x06 | ||
| 36 | #define MAX8925_RTC_YEAR2 0x07 | ||
| 37 | #define MAX8925_ALARM0_SEC 0x08 | ||
| 38 | #define MAX8925_ALARM0_MIN 0x09 | ||
| 39 | #define MAX8925_ALARM0_HOUR 0x0a | ||
| 40 | #define MAX8925_ALARM0_WEEKDAY 0x0b | ||
| 41 | #define MAX8925_ALARM0_DATE 0x0c | ||
| 42 | #define MAX8925_ALARM0_MON 0x0d | ||
| 43 | #define MAX8925_ALARM0_YEAR1 0x0e | ||
| 44 | #define MAX8925_ALARM0_YEAR2 0x0f | ||
| 45 | #define MAX8925_ALARM1_SEC 0x10 | ||
| 46 | #define MAX8925_ALARM1_MIN 0x11 | ||
| 47 | #define MAX8925_ALARM1_HOUR 0x12 | ||
| 48 | #define MAX8925_ALARM1_WEEKDAY 0x13 | ||
| 49 | #define MAX8925_ALARM1_DATE 0x14 | ||
| 50 | #define MAX8925_ALARM1_MON 0x15 | ||
| 51 | #define MAX8925_ALARM1_YEAR1 0x16 | ||
| 52 | #define MAX8925_ALARM1_YEAR2 0x17 | ||
| 53 | #define MAX8925_RTC_CNTL 0x1b | ||
| 54 | #define MAX8925_RTC_STATUS 0x20 | ||
| 55 | |||
| 56 | #define TIME_NUM 8 | ||
| 57 | #define ALARM_1SEC (1 << 7) | ||
| 58 | #define HOUR_12 (1 << 7) | ||
| 59 | #define HOUR_AM_PM (1 << 5) | ||
| 60 | #define ALARM0_IRQ (1 << 3) | ||
| 61 | #define ALARM1_IRQ (1 << 2) | ||
| 62 | #define ALARM0_STATUS (1 << 2) | ||
| 63 | #define ALARM1_STATUS (1 << 1) | ||
| 64 | |||
| 65 | |||
| 66 | struct max8925_rtc_info { | ||
| 67 | struct rtc_device *rtc_dev; | ||
| 68 | struct max8925_chip *chip; | ||
| 69 | struct i2c_client *rtc; | ||
| 70 | struct device *dev; | ||
| 71 | }; | ||
| 72 | |||
| 73 | static irqreturn_t rtc_update_handler(int irq, void *data) | ||
| 74 | { | ||
| 75 | struct max8925_rtc_info *info = (struct max8925_rtc_info *)data; | ||
| 76 | |||
| 77 | /* disable ALARM0 except for 1SEC alarm */ | ||
| 78 | max8925_set_bits(info->rtc, MAX8925_ALARM0_CNTL, 0x7f, 0); | ||
| 79 | rtc_update_irq(info->rtc_dev, 1, RTC_IRQF | RTC_AF); | ||
| 80 | return IRQ_HANDLED; | ||
| 81 | } | ||
| 82 | |||
| 83 | static int tm_calc(struct rtc_time *tm, unsigned char *buf, int len) | ||
| 84 | { | ||
| 85 | if (len < TIME_NUM) | ||
| 86 | return -EINVAL; | ||
| 87 | tm->tm_year = (buf[RTC_YEAR2] >> 4) * 1000 | ||
| 88 | + (buf[RTC_YEAR2] & 0xf) * 100 | ||
| 89 | + (buf[RTC_YEAR1] >> 4) * 10 | ||
| 90 | + (buf[RTC_YEAR1] & 0xf); | ||
| 91 | tm->tm_year -= 1900; | ||
| 92 | tm->tm_mon = ((buf[RTC_MONTH] >> 4) & 0x01) * 10 | ||
| 93 | + (buf[RTC_MONTH] & 0x0f); | ||
| 94 | tm->tm_mday = ((buf[RTC_DATE] >> 4) & 0x03) * 10 | ||
| 95 | + (buf[RTC_DATE] & 0x0f); | ||
| 96 | tm->tm_wday = buf[RTC_WEEKDAY] & 0x07; | ||
| 97 | if (buf[RTC_HOUR] & HOUR_12) { | ||
| 98 | tm->tm_hour = ((buf[RTC_HOUR] >> 4) & 0x1) * 10 | ||
| 99 | + (buf[RTC_HOUR] & 0x0f); | ||
| 100 | if (buf[RTC_HOUR] & HOUR_AM_PM) | ||
| 101 | tm->tm_hour += 12; | ||
| 102 | } else | ||
| 103 | tm->tm_hour = ((buf[RTC_HOUR] >> 4) & 0x03) * 10 | ||
| 104 | + (buf[RTC_HOUR] & 0x0f); | ||
| 105 | tm->tm_min = ((buf[RTC_MIN] >> 4) & 0x7) * 10 | ||
| 106 | + (buf[RTC_MIN] & 0x0f); | ||
| 107 | tm->tm_sec = ((buf[RTC_SEC] >> 4) & 0x7) * 10 | ||
| 108 | + (buf[RTC_SEC] & 0x0f); | ||
| 109 | return 0; | ||
| 110 | } | ||
| 111 | |||
| 112 | static int data_calc(unsigned char *buf, struct rtc_time *tm, int len) | ||
| 113 | { | ||
| 114 | unsigned char high, low; | ||
| 115 | |||
| 116 | if (len < TIME_NUM) | ||
| 117 | return -EINVAL; | ||
| 118 | |||
| 119 | high = (tm->tm_year + 1900) / 1000; | ||
| 120 | low = (tm->tm_year + 1900) / 100; | ||
| 121 | low = low - high * 10; | ||
| 122 | buf[RTC_YEAR2] = (high << 4) + low; | ||
| 123 | high = (tm->tm_year + 1900) / 10; | ||
| 124 | low = tm->tm_year + 1900; | ||
| 125 | low = low - high * 10; | ||
| 126 | high = high - (high / 10) * 10; | ||
| 127 | buf[RTC_YEAR1] = (high << 4) + low; | ||
| 128 | high = tm->tm_mon / 10; | ||
| 129 | low = tm->tm_mon; | ||
| 130 | low = low - high * 10; | ||
| 131 | buf[RTC_MONTH] = (high << 4) + low; | ||
| 132 | high = tm->tm_mday / 10; | ||
| 133 | low = tm->tm_mday; | ||
| 134 | low = low - high * 10; | ||
| 135 | buf[RTC_DATE] = (high << 4) + low; | ||
| 136 | buf[RTC_WEEKDAY] = tm->tm_wday; | ||
| 137 | high = tm->tm_hour / 10; | ||
| 138 | low = tm->tm_hour; | ||
| 139 | low = low - high * 10; | ||
| 140 | buf[RTC_HOUR] = (high << 4) + low; | ||
| 141 | high = tm->tm_min / 10; | ||
| 142 | low = tm->tm_min; | ||
| 143 | low = low - high * 10; | ||
| 144 | buf[RTC_MIN] = (high << 4) + low; | ||
| 145 | high = tm->tm_sec / 10; | ||
| 146 | low = tm->tm_sec; | ||
| 147 | low = low - high * 10; | ||
| 148 | buf[RTC_SEC] = (high << 4) + low; | ||
| 149 | return 0; | ||
| 150 | } | ||
| 151 | |||
| 152 | static int max8925_rtc_read_time(struct device *dev, struct rtc_time *tm) | ||
| 153 | { | ||
| 154 | struct max8925_rtc_info *info = dev_get_drvdata(dev); | ||
| 155 | unsigned char buf[TIME_NUM]; | ||
| 156 | int ret; | ||
| 157 | |||
| 158 | ret = max8925_bulk_read(info->rtc, MAX8925_RTC_SEC, TIME_NUM, buf); | ||
| 159 | if (ret < 0) | ||
| 160 | goto out; | ||
| 161 | ret = tm_calc(tm, buf, TIME_NUM); | ||
| 162 | out: | ||
| 163 | return ret; | ||
| 164 | } | ||
| 165 | |||
| 166 | static int max8925_rtc_set_time(struct device *dev, struct rtc_time *tm) | ||
| 167 | { | ||
| 168 | struct max8925_rtc_info *info = dev_get_drvdata(dev); | ||
| 169 | unsigned char buf[TIME_NUM]; | ||
| 170 | int ret; | ||
| 171 | |||
| 172 | ret = data_calc(buf, tm, TIME_NUM); | ||
| 173 | if (ret < 0) | ||
| 174 | goto out; | ||
| 175 | ret = max8925_bulk_write(info->rtc, MAX8925_RTC_SEC, TIME_NUM, buf); | ||
| 176 | out: | ||
| 177 | return ret; | ||
| 178 | } | ||
| 179 | |||
| 180 | static int max8925_rtc_read_alarm(struct device *dev, struct rtc_wkalrm *alrm) | ||
| 181 | { | ||
| 182 | struct max8925_rtc_info *info = dev_get_drvdata(dev); | ||
| 183 | unsigned char buf[TIME_NUM]; | ||
| 184 | int ret; | ||
| 185 | |||
| 186 | ret = max8925_bulk_read(info->rtc, MAX8925_ALARM0_SEC, TIME_NUM, buf); | ||
| 187 | if (ret < 0) | ||
| 188 | goto out; | ||
| 189 | ret = tm_calc(&alrm->time, buf, TIME_NUM); | ||
| 190 | if (ret < 0) | ||
| 191 | goto out; | ||
| 192 | ret = max8925_reg_read(info->rtc, MAX8925_RTC_IRQ_MASK); | ||
| 193 | if (ret < 0) | ||
| 194 | goto out; | ||
| 195 | if ((ret & ALARM0_IRQ) == 0) | ||
| 196 | alrm->enabled = 1; | ||
| 197 | else | ||
| 198 | alrm->enabled = 0; | ||
| 199 | ret = max8925_reg_read(info->rtc, MAX8925_RTC_STATUS); | ||
| 200 | if (ret < 0) | ||
| 201 | goto out; | ||
| 202 | if (ret & ALARM0_STATUS) | ||
| 203 | alrm->pending = 1; | ||
| 204 | else | ||
| 205 | alrm->pending = 0; | ||
| 206 | out: | ||
| 207 | return ret; | ||
| 208 | } | ||
| 209 | |||
| 210 | static int max8925_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alrm) | ||
| 211 | { | ||
| 212 | struct max8925_rtc_info *info = dev_get_drvdata(dev); | ||
| 213 | unsigned char buf[TIME_NUM]; | ||
| 214 | int ret; | ||
| 215 | |||
| 216 | ret = data_calc(buf, &alrm->time, TIME_NUM); | ||
| 217 | if (ret < 0) | ||
| 218 | goto out; | ||
| 219 | ret = max8925_bulk_write(info->rtc, MAX8925_ALARM0_SEC, TIME_NUM, buf); | ||
| 220 | if (ret < 0) | ||
| 221 | goto out; | ||
| 222 | /* only enable alarm on year/month/day/hour/min/sec */ | ||
| 223 | ret = max8925_reg_write(info->rtc, MAX8925_ALARM0_CNTL, 0x77); | ||
| 224 | if (ret < 0) | ||
| 225 | goto out; | ||
| 226 | out: | ||
| 227 | return ret; | ||
| 228 | } | ||
| 229 | |||
| 230 | static const struct rtc_class_ops max8925_rtc_ops = { | ||
| 231 | .read_time = max8925_rtc_read_time, | ||
| 232 | .set_time = max8925_rtc_set_time, | ||
| 233 | .read_alarm = max8925_rtc_read_alarm, | ||
| 234 | .set_alarm = max8925_rtc_set_alarm, | ||
| 235 | }; | ||
| 236 | |||
| 237 | static int __devinit max8925_rtc_probe(struct platform_device *pdev) | ||
| 238 | { | ||
| 239 | struct max8925_chip *chip = dev_get_drvdata(pdev->dev.parent); | ||
| 240 | struct max8925_rtc_info *info; | ||
| 241 | int irq, ret; | ||
| 242 | |||
| 243 | info = kzalloc(sizeof(struct max8925_rtc_info), GFP_KERNEL); | ||
| 244 | if (!info) | ||
| 245 | return -ENOMEM; | ||
| 246 | info->chip = chip; | ||
| 247 | info->rtc = chip->rtc; | ||
| 248 | info->dev = &pdev->dev; | ||
| 249 | irq = chip->irq_base + MAX8925_IRQ_RTC_ALARM0; | ||
| 250 | |||
| 251 | ret = request_threaded_irq(irq, NULL, rtc_update_handler, | ||
| 252 | IRQF_ONESHOT, "rtc-alarm0", info); | ||
| 253 | if (ret < 0) { | ||
| 254 | dev_err(chip->dev, "Failed to request IRQ: #%d: %d\n", | ||
| 255 | irq, ret); | ||
| 256 | goto out_irq; | ||
| 257 | } | ||
| 258 | |||
| 259 | info->rtc_dev = rtc_device_register("max8925-rtc", &pdev->dev, | ||
| 260 | &max8925_rtc_ops, THIS_MODULE); | ||
| 261 | ret = PTR_ERR(info->rtc_dev); | ||
| 262 | if (IS_ERR(info->rtc_dev)) { | ||
| 263 | dev_err(&pdev->dev, "Failed to register RTC device: %d\n", ret); | ||
| 264 | goto out_rtc; | ||
| 265 | } | ||
| 266 | |||
| 267 | dev_set_drvdata(&pdev->dev, info); | ||
| 268 | platform_set_drvdata(pdev, info); | ||
| 269 | |||
| 270 | return 0; | ||
| 271 | out_rtc: | ||
| 272 | free_irq(chip->irq_base + MAX8925_IRQ_RTC_ALARM0, info); | ||
| 273 | out_irq: | ||
| 274 | kfree(info); | ||
| 275 | return ret; | ||
| 276 | } | ||
| 277 | |||
| 278 | static int __devexit max8925_rtc_remove(struct platform_device *pdev) | ||
| 279 | { | ||
| 280 | struct max8925_rtc_info *info = platform_get_drvdata(pdev); | ||
| 281 | |||
| 282 | if (info) { | ||
| 283 | free_irq(info->chip->irq_base + MAX8925_IRQ_RTC_ALARM0, info); | ||
| 284 | rtc_device_unregister(info->rtc_dev); | ||
| 285 | kfree(info); | ||
| 286 | } | ||
| 287 | return 0; | ||
| 288 | } | ||
| 289 | |||
| 290 | static struct platform_driver max8925_rtc_driver = { | ||
| 291 | .driver = { | ||
| 292 | .name = "max8925-rtc", | ||
| 293 | .owner = THIS_MODULE, | ||
| 294 | }, | ||
| 295 | .probe = max8925_rtc_probe, | ||
| 296 | .remove = __devexit_p(max8925_rtc_remove), | ||
| 297 | }; | ||
| 298 | |||
| 299 | static int __init max8925_rtc_init(void) | ||
| 300 | { | ||
| 301 | return platform_driver_register(&max8925_rtc_driver); | ||
| 302 | } | ||
| 303 | module_init(max8925_rtc_init); | ||
| 304 | |||
| 305 | static void __exit max8925_rtc_exit(void) | ||
| 306 | { | ||
| 307 | platform_driver_unregister(&max8925_rtc_driver); | ||
| 308 | } | ||
| 309 | module_exit(max8925_rtc_exit); | ||
| 310 | |||
| 311 | MODULE_DESCRIPTION("Maxim MAX8925 RTC driver"); | ||
| 312 | MODULE_AUTHOR("Haojian Zhuang <haojian.zhuang@marvell.com>"); | ||
| 313 | MODULE_LICENSE("GPL"); | ||
| 314 | |||
diff --git a/drivers/rtc/rtc-mc13783.c b/drivers/rtc/rtc-mc13783.c index 850f983c039..d60c81b7b69 100644 --- a/drivers/rtc/rtc-mc13783.c +++ b/drivers/rtc/rtc-mc13783.c | |||
| @@ -28,6 +28,34 @@ struct mc13783_rtc { | |||
| 28 | int valid; | 28 | int valid; |
| 29 | }; | 29 | }; |
| 30 | 30 | ||
| 31 | static int mc13783_rtc_irq_enable_unlocked(struct device *dev, | ||
| 32 | unsigned int enabled, int irq) | ||
| 33 | { | ||
| 34 | struct mc13783_rtc *priv = dev_get_drvdata(dev); | ||
| 35 | int (*func)(struct mc13783 *mc13783, int irq); | ||
| 36 | |||
| 37 | if (!priv->valid) | ||
| 38 | return -ENODATA; | ||
| 39 | |||
| 40 | func = enabled ? mc13783_irq_unmask : mc13783_irq_mask; | ||
| 41 | return func(priv->mc13783, irq); | ||
| 42 | } | ||
| 43 | |||
| 44 | static int mc13783_rtc_irq_enable(struct device *dev, | ||
| 45 | unsigned int enabled, int irq) | ||
| 46 | { | ||
| 47 | struct mc13783_rtc *priv = dev_get_drvdata(dev); | ||
| 48 | int ret; | ||
| 49 | |||
| 50 | mc13783_lock(priv->mc13783); | ||
| 51 | |||
| 52 | ret = mc13783_rtc_irq_enable_unlocked(dev, enabled, irq); | ||
| 53 | |||
| 54 | mc13783_unlock(priv->mc13783); | ||
| 55 | |||
| 56 | return ret; | ||
| 57 | } | ||
| 58 | |||
| 31 | static int mc13783_rtc_read_time(struct device *dev, struct rtc_time *tm) | 59 | static int mc13783_rtc_read_time(struct device *dev, struct rtc_time *tm) |
| 32 | { | 60 | { |
| 33 | struct mc13783_rtc *priv = dev_get_drvdata(dev); | 61 | struct mc13783_rtc *priv = dev_get_drvdata(dev); |
| @@ -78,6 +106,7 @@ static int mc13783_rtc_set_mmss(struct device *dev, unsigned long secs) | |||
| 78 | { | 106 | { |
| 79 | struct mc13783_rtc *priv = dev_get_drvdata(dev); | 107 | struct mc13783_rtc *priv = dev_get_drvdata(dev); |
| 80 | unsigned int seconds, days; | 108 | unsigned int seconds, days; |
| 109 | unsigned int alarmseconds; | ||
| 81 | int ret; | 110 | int ret; |
| 82 | 111 | ||
| 83 | seconds = secs % 86400; | 112 | seconds = secs % 86400; |
| @@ -86,7 +115,22 @@ static int mc13783_rtc_set_mmss(struct device *dev, unsigned long secs) | |||
| 86 | mc13783_lock(priv->mc13783); | 115 | mc13783_lock(priv->mc13783); |
| 87 | 116 | ||
| 88 | /* | 117 | /* |
| 89 | * first write seconds=0 to prevent a day switch between writing days | 118 | * temporarily invalidate alarm to prevent triggering it when the day is |
| 119 | * already updated while the time isn't yet. | ||
| 120 | */ | ||
| 121 | ret = mc13783_reg_read(priv->mc13783, MC13783_RTCTODA, &alarmseconds); | ||
| 122 | if (unlikely(ret)) | ||
| 123 | goto out; | ||
| 124 | |||
| 125 | if (alarmseconds < 86400) { | ||
| 126 | ret = mc13783_reg_write(priv->mc13783, | ||
| 127 | MC13783_RTCTODA, 0x1ffff); | ||
| 128 | if (unlikely(ret)) | ||
| 129 | goto out; | ||
| 130 | } | ||
| 131 | |||
| 132 | /* | ||
| 133 | * write seconds=0 to prevent a day switch between writing days | ||
| 90 | * and seconds below | 134 | * and seconds below |
| 91 | */ | 135 | */ |
| 92 | ret = mc13783_reg_write(priv->mc13783, MC13783_RTCTOD, 0); | 136 | ret = mc13783_reg_write(priv->mc13783, MC13783_RTCTOD, 0); |
| @@ -101,11 +145,19 @@ static int mc13783_rtc_set_mmss(struct device *dev, unsigned long secs) | |||
| 101 | if (unlikely(ret)) | 145 | if (unlikely(ret)) |
| 102 | goto out; | 146 | goto out; |
| 103 | 147 | ||
| 104 | ret = mc13783_ackirq(priv->mc13783, MC13783_IRQ_RTCRST); | 148 | /* restore alarm */ |
| 149 | if (alarmseconds < 86400) { | ||
| 150 | ret = mc13783_reg_write(priv->mc13783, | ||
| 151 | MC13783_RTCTODA, alarmseconds); | ||
| 152 | if (unlikely(ret)) | ||
| 153 | goto out; | ||
| 154 | } | ||
| 155 | |||
| 156 | ret = mc13783_irq_ack(priv->mc13783, MC13783_IRQ_RTCRST); | ||
| 105 | if (unlikely(ret)) | 157 | if (unlikely(ret)) |
| 106 | goto out; | 158 | goto out; |
| 107 | 159 | ||
| 108 | ret = mc13783_unmask(priv->mc13783, MC13783_IRQ_RTCRST); | 160 | ret = mc13783_irq_unmask(priv->mc13783, MC13783_IRQ_RTCRST); |
| 109 | out: | 161 | out: |
| 110 | priv->valid = !ret; | 162 | priv->valid = !ret; |
| 111 | 163 | ||
| @@ -114,41 +166,139 @@ out: | |||
| 114 | return ret; | 166 | return ret; |
| 115 | } | 167 | } |
| 116 | 168 | ||
| 117 | static irqreturn_t mc13783_rtc_update_handler(int irq, void *dev) | 169 | static int mc13783_rtc_read_alarm(struct device *dev, struct rtc_wkalrm *alarm) |
| 118 | { | 170 | { |
| 119 | struct mc13783_rtc *priv = dev; | 171 | struct mc13783_rtc *priv = dev_get_drvdata(dev); |
| 120 | struct mc13783 *mc13783 = priv->mc13783; | 172 | unsigned seconds, days; |
| 173 | unsigned long s1970; | ||
| 174 | int enabled, pending; | ||
| 175 | int ret; | ||
| 121 | 176 | ||
| 122 | dev_dbg(&priv->rtc->dev, "1HZ\n"); | 177 | mc13783_lock(priv->mc13783); |
| 123 | 178 | ||
| 124 | rtc_update_irq(priv->rtc, 1, RTC_IRQF | RTC_UF); | 179 | ret = mc13783_reg_read(priv->mc13783, MC13783_RTCTODA, &seconds); |
| 180 | if (unlikely(ret)) | ||
| 181 | goto out; | ||
| 182 | if (seconds >= 86400) { | ||
| 183 | ret = -ENODATA; | ||
| 184 | goto out; | ||
| 185 | } | ||
| 186 | |||
| 187 | ret = mc13783_reg_read(priv->mc13783, MC13783_RTCDAY, &days); | ||
| 188 | if (unlikely(ret)) | ||
| 189 | goto out; | ||
| 125 | 190 | ||
| 126 | mc13783_ackirq(mc13783, irq); | 191 | ret = mc13783_irq_status(priv->mc13783, MC13783_IRQ_TODA, |
| 192 | &enabled, &pending); | ||
| 127 | 193 | ||
| 128 | return IRQ_HANDLED; | 194 | out: |
| 195 | mc13783_unlock(priv->mc13783); | ||
| 196 | |||
| 197 | if (ret) | ||
| 198 | return ret; | ||
| 199 | |||
| 200 | alarm->enabled = enabled; | ||
| 201 | alarm->pending = pending; | ||
| 202 | |||
| 203 | s1970 = days * 86400 + seconds; | ||
| 204 | |||
| 205 | rtc_time_to_tm(s1970, &alarm->time); | ||
| 206 | dev_dbg(dev, "%s: %lu\n", __func__, s1970); | ||
| 207 | |||
| 208 | return 0; | ||
| 129 | } | 209 | } |
| 130 | 210 | ||
| 131 | static int mc13783_rtc_update_irq_enable(struct device *dev, | 211 | static int mc13783_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alarm) |
| 132 | unsigned int enabled) | ||
| 133 | { | 212 | { |
| 134 | struct mc13783_rtc *priv = dev_get_drvdata(dev); | 213 | struct mc13783_rtc *priv = dev_get_drvdata(dev); |
| 135 | int ret = -ENODATA; | 214 | unsigned long s1970; |
| 215 | unsigned seconds, days; | ||
| 216 | int ret; | ||
| 136 | 217 | ||
| 137 | mc13783_lock(priv->mc13783); | 218 | mc13783_lock(priv->mc13783); |
| 138 | if (!priv->valid) | 219 | |
| 220 | /* disable alarm to prevent false triggering */ | ||
| 221 | ret = mc13783_reg_write(priv->mc13783, MC13783_RTCTODA, 0x1ffff); | ||
| 222 | if (unlikely(ret)) | ||
| 139 | goto out; | 223 | goto out; |
| 140 | 224 | ||
| 141 | ret = (enabled ? mc13783_unmask : mc13783_mask)(priv->mc13783, | 225 | ret = mc13783_irq_ack(priv->mc13783, MC13783_IRQ_TODA); |
| 142 | MC13783_IRQ_1HZ); | 226 | if (unlikely(ret)) |
| 227 | goto out; | ||
| 228 | |||
| 229 | ret = rtc_tm_to_time(&alarm->time, &s1970); | ||
| 230 | if (unlikely(ret)) | ||
| 231 | goto out; | ||
| 232 | |||
| 233 | dev_dbg(dev, "%s: o%2.s %lu\n", __func__, alarm->enabled ? "n" : "ff", | ||
| 234 | s1970); | ||
| 235 | |||
| 236 | ret = mc13783_rtc_irq_enable_unlocked(dev, alarm->enabled, | ||
| 237 | MC13783_IRQ_TODA); | ||
| 238 | if (unlikely(ret)) | ||
| 239 | goto out; | ||
| 240 | |||
| 241 | seconds = s1970 % 86400; | ||
| 242 | days = s1970 / 86400; | ||
| 243 | |||
| 244 | ret = mc13783_reg_write(priv->mc13783, MC13783_RTCDAYA, days); | ||
| 245 | if (unlikely(ret)) | ||
| 246 | goto out; | ||
| 247 | |||
| 248 | ret = mc13783_reg_write(priv->mc13783, MC13783_RTCTODA, seconds); | ||
| 249 | |||
| 143 | out: | 250 | out: |
| 144 | mc13783_unlock(priv->mc13783); | 251 | mc13783_unlock(priv->mc13783); |
| 145 | 252 | ||
| 146 | return ret; | 253 | return ret; |
| 147 | } | 254 | } |
| 148 | 255 | ||
| 256 | static irqreturn_t mc13783_rtc_alarm_handler(int irq, void *dev) | ||
| 257 | { | ||
| 258 | struct mc13783_rtc *priv = dev; | ||
| 259 | struct mc13783 *mc13783 = priv->mc13783; | ||
| 260 | |||
| 261 | dev_dbg(&priv->rtc->dev, "Alarm\n"); | ||
| 262 | |||
| 263 | rtc_update_irq(priv->rtc, 1, RTC_IRQF | RTC_AF); | ||
| 264 | |||
| 265 | mc13783_irq_ack(mc13783, irq); | ||
| 266 | |||
| 267 | return IRQ_HANDLED; | ||
| 268 | } | ||
| 269 | |||
| 270 | static irqreturn_t mc13783_rtc_update_handler(int irq, void *dev) | ||
| 271 | { | ||
| 272 | struct mc13783_rtc *priv = dev; | ||
| 273 | struct mc13783 *mc13783 = priv->mc13783; | ||
| 274 | |||
| 275 | dev_dbg(&priv->rtc->dev, "1HZ\n"); | ||
| 276 | |||
| 277 | rtc_update_irq(priv->rtc, 1, RTC_IRQF | RTC_UF); | ||
| 278 | |||
| 279 | mc13783_irq_ack(mc13783, irq); | ||
| 280 | |||
| 281 | return IRQ_HANDLED; | ||
| 282 | } | ||
| 283 | |||
| 284 | static int mc13783_rtc_update_irq_enable(struct device *dev, | ||
| 285 | unsigned int enabled) | ||
| 286 | { | ||
| 287 | return mc13783_rtc_irq_enable(dev, enabled, MC13783_IRQ_1HZ); | ||
| 288 | } | ||
| 289 | |||
| 290 | static int mc13783_rtc_alarm_irq_enable(struct device *dev, | ||
| 291 | unsigned int enabled) | ||
| 292 | { | ||
| 293 | return mc13783_rtc_irq_enable(dev, enabled, MC13783_IRQ_TODA); | ||
| 294 | } | ||
| 295 | |||
| 149 | static const struct rtc_class_ops mc13783_rtc_ops = { | 296 | static const struct rtc_class_ops mc13783_rtc_ops = { |
| 150 | .read_time = mc13783_rtc_read_time, | 297 | .read_time = mc13783_rtc_read_time, |
| 151 | .set_mmss = mc13783_rtc_set_mmss, | 298 | .set_mmss = mc13783_rtc_set_mmss, |
| 299 | .read_alarm = mc13783_rtc_read_alarm, | ||
| 300 | .set_alarm = mc13783_rtc_set_alarm, | ||
| 301 | .alarm_irq_enable = mc13783_rtc_alarm_irq_enable, | ||
| 152 | .update_irq_enable = mc13783_rtc_update_irq_enable, | 302 | .update_irq_enable = mc13783_rtc_update_irq_enable, |
| 153 | }; | 303 | }; |
| 154 | 304 | ||
| @@ -160,7 +310,7 @@ static irqreturn_t mc13783_rtc_reset_handler(int irq, void *dev) | |||
| 160 | dev_dbg(&priv->rtc->dev, "RTCRST\n"); | 310 | dev_dbg(&priv->rtc->dev, "RTCRST\n"); |
| 161 | priv->valid = 0; | 311 | priv->valid = 0; |
| 162 | 312 | ||
| 163 | mc13783_mask(mc13783, irq); | 313 | mc13783_irq_mask(mc13783, irq); |
| 164 | 314 | ||
| 165 | return IRQ_HANDLED; | 315 | return IRQ_HANDLED; |
| 166 | } | 316 | } |
| @@ -169,6 +319,7 @@ static int __devinit mc13783_rtc_probe(struct platform_device *pdev) | |||
| 169 | { | 319 | { |
| 170 | int ret; | 320 | int ret; |
| 171 | struct mc13783_rtc *priv; | 321 | struct mc13783_rtc *priv; |
| 322 | int rtcrst_pending; | ||
| 172 | 323 | ||
| 173 | priv = kzalloc(sizeof(*priv), GFP_KERNEL); | 324 | priv = kzalloc(sizeof(*priv), GFP_KERNEL); |
| 174 | if (!priv) | 325 | if (!priv) |
| @@ -177,8 +328,6 @@ static int __devinit mc13783_rtc_probe(struct platform_device *pdev) | |||
| 177 | priv->mc13783 = dev_get_drvdata(pdev->dev.parent); | 328 | priv->mc13783 = dev_get_drvdata(pdev->dev.parent); |
| 178 | platform_set_drvdata(pdev, priv); | 329 | platform_set_drvdata(pdev, priv); |
| 179 | 330 | ||
| 180 | priv->valid = 1; | ||
| 181 | |||
| 182 | mc13783_lock(priv->mc13783); | 331 | mc13783_lock(priv->mc13783); |
| 183 | 332 | ||
| 184 | ret = mc13783_irq_request(priv->mc13783, MC13783_IRQ_RTCRST, | 333 | ret = mc13783_irq_request(priv->mc13783, MC13783_IRQ_RTCRST, |
| @@ -186,33 +335,45 @@ static int __devinit mc13783_rtc_probe(struct platform_device *pdev) | |||
| 186 | if (ret) | 335 | if (ret) |
| 187 | goto err_reset_irq_request; | 336 | goto err_reset_irq_request; |
| 188 | 337 | ||
| 338 | ret = mc13783_irq_status(priv->mc13783, MC13783_IRQ_RTCRST, | ||
| 339 | NULL, &rtcrst_pending); | ||
| 340 | if (ret) | ||
| 341 | goto err_reset_irq_status; | ||
| 342 | |||
| 343 | priv->valid = !rtcrst_pending; | ||
| 344 | |||
| 189 | ret = mc13783_irq_request_nounmask(priv->mc13783, MC13783_IRQ_1HZ, | 345 | ret = mc13783_irq_request_nounmask(priv->mc13783, MC13783_IRQ_1HZ, |
| 190 | mc13783_rtc_update_handler, DRIVER_NAME, priv); | 346 | mc13783_rtc_update_handler, DRIVER_NAME, priv); |
| 191 | if (ret) | 347 | if (ret) |
| 192 | goto err_update_irq_request; | 348 | goto err_update_irq_request; |
| 193 | 349 | ||
| 194 | mc13783_unlock(priv->mc13783); | 350 | ret = mc13783_irq_request_nounmask(priv->mc13783, MC13783_IRQ_TODA, |
| 351 | mc13783_rtc_alarm_handler, DRIVER_NAME, priv); | ||
| 352 | if (ret) | ||
| 353 | goto err_alarm_irq_request; | ||
| 195 | 354 | ||
| 196 | priv->rtc = rtc_device_register(pdev->name, | 355 | priv->rtc = rtc_device_register(pdev->name, |
| 197 | &pdev->dev, &mc13783_rtc_ops, THIS_MODULE); | 356 | &pdev->dev, &mc13783_rtc_ops, THIS_MODULE); |
| 198 | |||
| 199 | if (IS_ERR(priv->rtc)) { | 357 | if (IS_ERR(priv->rtc)) { |
| 200 | ret = PTR_ERR(priv->rtc); | 358 | ret = PTR_ERR(priv->rtc); |
| 201 | 359 | ||
| 202 | mc13783_lock(priv->mc13783); | 360 | mc13783_irq_free(priv->mc13783, MC13783_IRQ_TODA, priv); |
| 361 | err_alarm_irq_request: | ||
| 203 | 362 | ||
| 204 | mc13783_irq_free(priv->mc13783, MC13783_IRQ_1HZ, priv); | 363 | mc13783_irq_free(priv->mc13783, MC13783_IRQ_1HZ, priv); |
| 205 | err_update_irq_request: | 364 | err_update_irq_request: |
| 206 | 365 | ||
| 366 | err_reset_irq_status: | ||
| 367 | |||
| 207 | mc13783_irq_free(priv->mc13783, MC13783_IRQ_RTCRST, priv); | 368 | mc13783_irq_free(priv->mc13783, MC13783_IRQ_RTCRST, priv); |
| 208 | err_reset_irq_request: | 369 | err_reset_irq_request: |
| 209 | 370 | ||
| 210 | mc13783_unlock(priv->mc13783); | ||
| 211 | |||
| 212 | platform_set_drvdata(pdev, NULL); | 371 | platform_set_drvdata(pdev, NULL); |
| 213 | kfree(priv); | 372 | kfree(priv); |
| 214 | } | 373 | } |
| 215 | 374 | ||
| 375 | mc13783_unlock(priv->mc13783); | ||
| 376 | |||
| 216 | return ret; | 377 | return ret; |
| 217 | } | 378 | } |
| 218 | 379 | ||
| @@ -220,10 +381,11 @@ static int __devexit mc13783_rtc_remove(struct platform_device *pdev) | |||
| 220 | { | 381 | { |
| 221 | struct mc13783_rtc *priv = platform_get_drvdata(pdev); | 382 | struct mc13783_rtc *priv = platform_get_drvdata(pdev); |
| 222 | 383 | ||
| 223 | rtc_device_unregister(priv->rtc); | ||
| 224 | |||
| 225 | mc13783_lock(priv->mc13783); | 384 | mc13783_lock(priv->mc13783); |
| 226 | 385 | ||
| 386 | rtc_device_unregister(priv->rtc); | ||
| 387 | |||
| 388 | mc13783_irq_free(priv->mc13783, MC13783_IRQ_TODA, priv); | ||
| 227 | mc13783_irq_free(priv->mc13783, MC13783_IRQ_1HZ, priv); | 389 | mc13783_irq_free(priv->mc13783, MC13783_IRQ_1HZ, priv); |
| 228 | mc13783_irq_free(priv->mc13783, MC13783_IRQ_RTCRST, priv); | 390 | mc13783_irq_free(priv->mc13783, MC13783_IRQ_RTCRST, priv); |
| 229 | 391 | ||
diff --git a/drivers/rtc/rtc-mxc.c b/drivers/rtc/rtc-mxc.c index 6bd5072d4eb..8710f9415d9 100644 --- a/drivers/rtc/rtc-mxc.c +++ b/drivers/rtc/rtc-mxc.c | |||
| @@ -396,8 +396,11 @@ static int __init mxc_rtc_probe(struct platform_device *pdev) | |||
| 396 | pdata->ioaddr = ioremap(res->start, resource_size(res)); | 396 | pdata->ioaddr = ioremap(res->start, resource_size(res)); |
| 397 | 397 | ||
| 398 | clk = clk_get(&pdev->dev, "ckil"); | 398 | clk = clk_get(&pdev->dev, "ckil"); |
| 399 | if (IS_ERR(clk)) | 399 | if (IS_ERR(clk)) { |
| 400 | return PTR_ERR(clk); | 400 | iounmap(pdata->ioaddr); |
| 401 | ret = PTR_ERR(clk); | ||
| 402 | goto exit_free_pdata; | ||
| 403 | } | ||
| 401 | 404 | ||
| 402 | rate = clk_get_rate(clk); | 405 | rate = clk_get_rate(clk); |
| 403 | clk_put(clk); | 406 | clk_put(clk); |
diff --git a/drivers/rtc/rtc-pcf2123.c b/drivers/rtc/rtc-pcf2123.c index e75df9d50e2..2ceb365533b 100644 --- a/drivers/rtc/rtc-pcf2123.c +++ b/drivers/rtc/rtc-pcf2123.c | |||
| @@ -315,7 +315,7 @@ kfree_exit: | |||
| 315 | return ret; | 315 | return ret; |
| 316 | } | 316 | } |
| 317 | 317 | ||
| 318 | static int pcf2123_remove(struct spi_device *spi) | 318 | static int __devexit pcf2123_remove(struct spi_device *spi) |
| 319 | { | 319 | { |
| 320 | struct pcf2123_plat_data *pdata = spi->dev.platform_data; | 320 | struct pcf2123_plat_data *pdata = spi->dev.platform_data; |
| 321 | int i; | 321 | int i; |
diff --git a/drivers/rtc/rtc-twl.c b/drivers/rtc/rtc-twl.c index c6a83a2a722..ed1b8682812 100644 --- a/drivers/rtc/rtc-twl.c +++ b/drivers/rtc/rtc-twl.c | |||
| @@ -57,7 +57,7 @@ enum { | |||
| 57 | REG_RTC_COMP_LSB_REG, | 57 | REG_RTC_COMP_LSB_REG, |
| 58 | REG_RTC_COMP_MSB_REG, | 58 | REG_RTC_COMP_MSB_REG, |
| 59 | }; | 59 | }; |
| 60 | const static u8 twl4030_rtc_reg_map[] = { | 60 | static const u8 twl4030_rtc_reg_map[] = { |
| 61 | [REG_SECONDS_REG] = 0x00, | 61 | [REG_SECONDS_REG] = 0x00, |
| 62 | [REG_MINUTES_REG] = 0x01, | 62 | [REG_MINUTES_REG] = 0x01, |
| 63 | [REG_HOURS_REG] = 0x02, | 63 | [REG_HOURS_REG] = 0x02, |
| @@ -80,7 +80,7 @@ const static u8 twl4030_rtc_reg_map[] = { | |||
| 80 | [REG_RTC_COMP_LSB_REG] = 0x10, | 80 | [REG_RTC_COMP_LSB_REG] = 0x10, |
| 81 | [REG_RTC_COMP_MSB_REG] = 0x11, | 81 | [REG_RTC_COMP_MSB_REG] = 0x11, |
| 82 | }; | 82 | }; |
| 83 | const static u8 twl6030_rtc_reg_map[] = { | 83 | static const u8 twl6030_rtc_reg_map[] = { |
| 84 | [REG_SECONDS_REG] = 0x00, | 84 | [REG_SECONDS_REG] = 0x00, |
| 85 | [REG_MINUTES_REG] = 0x01, | 85 | [REG_MINUTES_REG] = 0x01, |
| 86 | [REG_HOURS_REG] = 0x02, | 86 | [REG_HOURS_REG] = 0x02, |
diff --git a/drivers/rtc/rtc-wm8350.c b/drivers/rtc/rtc-wm8350.c index f1e440521c5..3d0dc76b38a 100644 --- a/drivers/rtc/rtc-wm8350.c +++ b/drivers/rtc/rtc-wm8350.c | |||
| @@ -307,11 +307,18 @@ static int wm8350_rtc_update_irq_enable(struct device *dev, | |||
| 307 | { | 307 | { |
| 308 | struct wm8350 *wm8350 = dev_get_drvdata(dev); | 308 | struct wm8350 *wm8350 = dev_get_drvdata(dev); |
| 309 | 309 | ||
| 310 | /* Suppress duplicate changes since genirq nests enable and | ||
| 311 | * disable calls. */ | ||
| 312 | if (enabled == wm8350->rtc.update_enabled) | ||
| 313 | return 0; | ||
| 314 | |||
| 310 | if (enabled) | 315 | if (enabled) |
| 311 | wm8350_unmask_irq(wm8350, WM8350_IRQ_RTC_SEC); | 316 | wm8350_unmask_irq(wm8350, WM8350_IRQ_RTC_SEC); |
| 312 | else | 317 | else |
| 313 | wm8350_mask_irq(wm8350, WM8350_IRQ_RTC_SEC); | 318 | wm8350_mask_irq(wm8350, WM8350_IRQ_RTC_SEC); |
| 314 | 319 | ||
| 320 | wm8350->rtc.update_enabled = enabled; | ||
| 321 | |||
| 315 | return 0; | 322 | return 0; |
| 316 | } | 323 | } |
| 317 | 324 | ||
| @@ -478,8 +485,8 @@ static int __devexit wm8350_rtc_remove(struct platform_device *pdev) | |||
| 478 | struct wm8350 *wm8350 = platform_get_drvdata(pdev); | 485 | struct wm8350 *wm8350 = platform_get_drvdata(pdev); |
| 479 | struct wm8350_rtc *wm_rtc = &wm8350->rtc; | 486 | struct wm8350_rtc *wm_rtc = &wm8350->rtc; |
| 480 | 487 | ||
| 481 | wm8350_free_irq(wm8350, WM8350_IRQ_RTC_SEC); | 488 | wm8350_free_irq(wm8350, WM8350_IRQ_RTC_SEC, wm8350); |
| 482 | wm8350_free_irq(wm8350, WM8350_IRQ_RTC_ALM); | 489 | wm8350_free_irq(wm8350, WM8350_IRQ_RTC_ALM, wm8350); |
| 483 | 490 | ||
| 484 | rtc_device_unregister(wm_rtc->rtc); | 491 | rtc_device_unregister(wm_rtc->rtc); |
| 485 | 492 | ||
