diff options
Diffstat (limited to 'drivers/rtc')
| -rw-r--r-- | drivers/rtc/rtc-stmp3xxx.c | 35 | ||||
| -rw-r--r-- | drivers/rtc/rtc-sysfs.c | 48 | 
2 files changed, 49 insertions, 34 deletions
| diff --git a/drivers/rtc/rtc-stmp3xxx.c b/drivers/rtc/rtc-stmp3xxx.c index 767fee2ab340..26019531db15 100644 --- a/drivers/rtc/rtc-stmp3xxx.c +++ b/drivers/rtc/rtc-stmp3xxx.c @@ -23,6 +23,7 @@  #include <linux/init.h>  #include <linux/platform_device.h>  #include <linux/interrupt.h> +#include <linux/delay.h>  #include <linux/rtc.h>  #include <linux/slab.h>  #include <linux/of_device.h> @@ -119,24 +120,39 @@ static void stmp3xxx_wdt_register(struct platform_device *rtc_pdev)  }  #endif /* CONFIG_STMP3XXX_RTC_WATCHDOG */ -static void stmp3xxx_wait_time(struct stmp3xxx_rtc_data *rtc_data) +static int stmp3xxx_wait_time(struct stmp3xxx_rtc_data *rtc_data)  { +	int timeout = 5000; /* 3ms according to i.MX28 Ref Manual */  	/* -	 * The datasheet doesn't say which way round the -	 * NEW_REGS/STALE_REGS bitfields go. In fact it's 0x1=P0, -	 * 0x2=P1, .., 0x20=P5, 0x40=ALARM, 0x80=SECONDS +	 * The i.MX28 Applications Processor Reference Manual, Rev. 1, 2010 +	 * states: +	 * | The order in which registers are updated is +	 * | Persistent 0, 1, 2, 3, 4, 5, Alarm, Seconds. +	 * | (This list is in bitfield order, from LSB to MSB, as they would +	 * | appear in the STALE_REGS and NEW_REGS bitfields of the HW_RTC_STAT +	 * | register. For example, the Seconds register corresponds to +	 * | STALE_REGS or NEW_REGS containing 0x80.)  	 */ -	while (readl(rtc_data->io + STMP3XXX_RTC_STAT) & -			(0x80 << STMP3XXX_RTC_STAT_STALE_SHIFT)) -		cpu_relax(); +	do { +		if (!(readl(rtc_data->io + STMP3XXX_RTC_STAT) & +				(0x80 << STMP3XXX_RTC_STAT_STALE_SHIFT))) +			return 0; +		udelay(1); +	} while (--timeout > 0); +	return (readl(rtc_data->io + STMP3XXX_RTC_STAT) & +		(0x80 << STMP3XXX_RTC_STAT_STALE_SHIFT)) ? -ETIME : 0;  }  /* Time read/write */  static int stmp3xxx_rtc_gettime(struct device *dev, struct rtc_time *rtc_tm)  { +	int ret;  	struct stmp3xxx_rtc_data *rtc_data = dev_get_drvdata(dev); -	stmp3xxx_wait_time(rtc_data); +	ret = stmp3xxx_wait_time(rtc_data); +	if (ret) +		return ret; +  	rtc_time_to_tm(readl(rtc_data->io + STMP3XXX_RTC_SECONDS), rtc_tm);  	return 0;  } @@ -146,8 +162,7 @@ static int stmp3xxx_rtc_set_mmss(struct device *dev, unsigned long t)  	struct stmp3xxx_rtc_data *rtc_data = dev_get_drvdata(dev);  	writel(t, rtc_data->io + STMP3XXX_RTC_SECONDS); -	stmp3xxx_wait_time(rtc_data); -	return 0; +	return stmp3xxx_wait_time(rtc_data);  }  /* interrupt(s) handler */ diff --git a/drivers/rtc/rtc-sysfs.c b/drivers/rtc/rtc-sysfs.c index 4b26f8672b2d..babd43bf3ddc 100644 --- a/drivers/rtc/rtc-sysfs.c +++ b/drivers/rtc/rtc-sysfs.c @@ -25,15 +25,14 @@   */  static ssize_t -rtc_sysfs_show_name(struct device *dev, struct device_attribute *attr, -		char *buf) +name_show(struct device *dev, struct device_attribute *attr, char *buf)  {  	return sprintf(buf, "%s\n", to_rtc_device(dev)->name);  } +static DEVICE_ATTR_RO(name);  static ssize_t -rtc_sysfs_show_date(struct device *dev, struct device_attribute *attr, -		char *buf) +date_show(struct device *dev, struct device_attribute *attr, char *buf)  {  	ssize_t retval;  	struct rtc_time tm; @@ -46,10 +45,10 @@ rtc_sysfs_show_date(struct device *dev, struct device_attribute *attr,  	return retval;  } +static DEVICE_ATTR_RO(date);  static ssize_t -rtc_sysfs_show_time(struct device *dev, struct device_attribute *attr, -		char *buf) +time_show(struct device *dev, struct device_attribute *attr, char *buf)  {  	ssize_t retval;  	struct rtc_time tm; @@ -62,10 +61,10 @@ rtc_sysfs_show_time(struct device *dev, struct device_attribute *attr,  	return retval;  } +static DEVICE_ATTR_RO(time);  static ssize_t -rtc_sysfs_show_since_epoch(struct device *dev, struct device_attribute *attr, -		char *buf) +since_epoch_show(struct device *dev, struct device_attribute *attr, char *buf)  {  	ssize_t retval;  	struct rtc_time tm; @@ -79,16 +78,16 @@ rtc_sysfs_show_since_epoch(struct device *dev, struct device_attribute *attr,  	return retval;  } +static DEVICE_ATTR_RO(since_epoch);  static ssize_t -rtc_sysfs_show_max_user_freq(struct device *dev, struct device_attribute *attr, -		char *buf) +max_user_freq_show(struct device *dev, struct device_attribute *attr, char *buf)  {  	return sprintf(buf, "%d\n", to_rtc_device(dev)->max_user_freq);  }  static ssize_t -rtc_sysfs_set_max_user_freq(struct device *dev, struct device_attribute *attr, +max_user_freq_store(struct device *dev, struct device_attribute *attr,  		const char *buf, size_t n)  {  	struct rtc_device *rtc = to_rtc_device(dev); @@ -101,6 +100,7 @@ rtc_sysfs_set_max_user_freq(struct device *dev, struct device_attribute *attr,  	return n;  } +static DEVICE_ATTR_RW(max_user_freq);  /**   * rtc_sysfs_show_hctosys - indicate if the given RTC set the system time @@ -109,8 +109,7 @@ rtc_sysfs_set_max_user_freq(struct device *dev, struct device_attribute *attr,   * boot or resume event.   */  static ssize_t -rtc_sysfs_show_hctosys(struct device *dev, struct device_attribute *attr, -		char *buf) +hctosys_show(struct device *dev, struct device_attribute *attr, char *buf)  {  #ifdef CONFIG_RTC_HCTOSYS_DEVICE  	if (rtc_hctosys_ret == 0 && @@ -121,17 +120,18 @@ rtc_sysfs_show_hctosys(struct device *dev, struct device_attribute *attr,  #endif  		return sprintf(buf, "0\n");  } - -static struct device_attribute rtc_attrs[] = { -	__ATTR(name, S_IRUGO, rtc_sysfs_show_name, NULL), -	__ATTR(date, S_IRUGO, rtc_sysfs_show_date, NULL), -	__ATTR(time, S_IRUGO, rtc_sysfs_show_time, NULL), -	__ATTR(since_epoch, S_IRUGO, rtc_sysfs_show_since_epoch, NULL), -	__ATTR(max_user_freq, S_IRUGO | S_IWUSR, rtc_sysfs_show_max_user_freq, -			rtc_sysfs_set_max_user_freq), -	__ATTR(hctosys, S_IRUGO, rtc_sysfs_show_hctosys, NULL), -	{ }, +static DEVICE_ATTR_RO(hctosys); + +static struct attribute *rtc_attrs[] = { +	&dev_attr_name.attr, +	&dev_attr_date.attr, +	&dev_attr_time.attr, +	&dev_attr_since_epoch.attr, +	&dev_attr_max_user_freq.attr, +	&dev_attr_hctosys.attr, +	NULL,  }; +ATTRIBUTE_GROUPS(rtc);  static ssize_t  rtc_sysfs_show_wakealarm(struct device *dev, struct device_attribute *attr, @@ -261,5 +261,5 @@ void rtc_sysfs_del_device(struct rtc_device *rtc)  void __init rtc_sysfs_init(struct class *rtc_class)  { -	rtc_class->dev_attrs = rtc_attrs; +	rtc_class->dev_groups = rtc_groups;  } | 
