Re: [RFC PATCH 1/4] rtc: Introduce one interface to save the RTC hardware time range

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi Baolin,

Could you have a look at
https://git.kernel.org/pub/scm/linux/kernel/git/abelloni/linux.git/commit/?h=rtc-ranges

My approach has multiple advantages as it works for 64-bit counters and
the range can be updated at runtime.

On 02/01/2018 at 13:10:05 +0800, Baolin Wang wrote:
> In order to the setting time values are not beyond the limitation
> supported by RTC hardware, we introduce one interface to tell the
> hardware range to the RTC core, which are used to valid if the
> setting time values are in the RTC hardware range.
> 
> Moreover we also need the RTC hardware range to expand the RTC
> range in next patches by adding one offset.
> 
> Signed-off-by: Baolin Wang <baolin.wang@xxxxxxxxxx>
> ---
>  drivers/rtc/class.c     |   13 ++++++++++
>  drivers/rtc/interface.c |   62 +++++++++++++++++++++++++++++++++++++++++++++++
>  include/linux/rtc.h     |    9 +++++++
>  3 files changed, 84 insertions(+)
> 
> diff --git a/drivers/rtc/class.c b/drivers/rtc/class.c
> index 722d683..31fc0f1 100644
> --- a/drivers/rtc/class.c
> +++ b/drivers/rtc/class.c
> @@ -247,6 +247,12 @@ struct rtc_device *rtc_device_register(const char *name, struct device *dev,
>  
>  	dev_set_name(&rtc->dev, "rtc%d", id);
>  
> +	err = rtc_read_range(rtc, &rtc->max_hw_secs, &rtc->min_hw_secs);
> +	if (err) {
> +		dev_err(&rtc->dev, "%s: failed to get RTC range\n", name);
> +		goto exit_ida;
> +	}
> +
>  	/* Check to see if there is an ALARM already set in hw */
>  	err = __rtc_read_alarm(rtc, &alrm);
>  
> @@ -436,6 +442,13 @@ int __rtc_register_device(struct module *owner, struct rtc_device *rtc)
>  
>  	rtc->owner = owner;
>  
> +	err = rtc_read_range(rtc, &rtc->max_hw_secs, &rtc->min_hw_secs);
> +	if (err) {
> +		dev_err(&rtc->dev, "%s: failed to get RTC range\n",
> +			dev_name(&rtc->dev));
> +		return err;
> +	}
> +
>  	/* Check to see if there is an ALARM already set in hw */
>  	err = __rtc_read_alarm(rtc, &alrm);
>  	if (!err && !rtc_valid_tm(&alrm.time))
> diff --git a/drivers/rtc/interface.c b/drivers/rtc/interface.c
> index 672b192..c8090e3 100644
> --- a/drivers/rtc/interface.c
> +++ b/drivers/rtc/interface.c
> @@ -65,6 +65,10 @@ int rtc_set_time(struct rtc_device *rtc, struct rtc_time *tm)
>  	if (err != 0)
>  		return err;
>  
> +	err = rtc_valid_range(rtc, tm);
> +	if (err)
> +		return err;
> +
>  	err = mutex_lock_interruptible(&rtc->ops_lock);
>  	if (err)
>  		return err;
> @@ -329,6 +333,11 @@ static int __rtc_set_alarm(struct rtc_device *rtc, struct rtc_wkalrm *alarm)
>  	err = rtc_valid_tm(&alarm->time);
>  	if (err)
>  		return err;
> +
> +	err = rtc_valid_range(rtc, &alarm->time);
> +	if (err)
> +		return err;
> +
>  	scheduled = rtc_tm_to_time64(&alarm->time);
>  
>  	/* Make sure we're not setting alarms in the past */
> @@ -1027,3 +1036,56 @@ int rtc_set_offset(struct rtc_device *rtc, long offset)
>  	mutex_unlock(&rtc->ops_lock);
>  	return ret;
>  }
> +
> +/* rtc_read_range - Read the max and min hardware count supported by RTC device
> + * @ rtc: rtc device to be used.
> + * @ max_hw_secs: maximum hardware count in seconds, which can represent
> + * the maximum time values in RTC hardware.
> + * @ min_hw_secs: minimum hardware count in seconds, which can represent
> + * the minimum time values in RTC hardware.
> + *
> + * The max_hw_secs and min_hw_secs implemented by user must represent the
> + * correct hardware start time and maximum time, which means the count
> + * will wrap around to min_hw_secs after the maximum count.
> + *
> + * If user did not implement the read_range() interface, we can set max_hw_secs
> + * and min_hw_secs to 0, which avoids validing the range.
> + */
> +int rtc_read_range(struct rtc_device *rtc, time64_t *max_hw_secs,
> +		   time64_t *min_hw_secs)
> +{
> +	int ret;
> +
> +	if (!rtc->ops || !rtc->ops->read_range) {
> +		*max_hw_secs = 0;
> +		*min_hw_secs = 0;
> +		return 0;
> +	}
> +
> +	mutex_lock(&rtc->ops_lock);
> +	ret = rtc->ops->read_range(rtc->dev.parent, max_hw_secs, min_hw_secs);
> +	mutex_unlock(&rtc->ops_lock);
> +
> +	return ret;
> +}
> +
> +/* rtc_valid_range - Valid if the setting time in the RTC range
> + * @ rtc: rtc device to be used.
> + * @ tm: time values need to valid.
> + *
> + * Only the rtc->max_hw_secs was set, then we can valid if the setting time
> + * values are beyond the RTC range.
> + */
> +int rtc_valid_range(struct rtc_device *rtc, struct rtc_time *tm)
> +{
> +	time64_t secs;
> +
> +	if (!rtc->max_hw_secs)
> +		return 0;
> +
> +	secs = rtc_tm_to_time64(tm);
> +	if (secs < rtc->min_hw_secs || secs > rtc->max_hw_secs)
> +		return -EINVAL;
> +
> +	return 0;
> +}
> diff --git a/include/linux/rtc.h b/include/linux/rtc.h
> index 41319a2..19a8989 100644
> --- a/include/linux/rtc.h
> +++ b/include/linux/rtc.h
> @@ -85,6 +85,8 @@ struct rtc_class_ops {
>  	int (*alarm_irq_enable)(struct device *, unsigned int enabled);
>  	int (*read_offset)(struct device *, long *offset);
>  	int (*set_offset)(struct device *, long offset);
> +	int (*read_range)(struct device *, time64_t *max_hw_secs,
> +			  time64_t *min_hw_secs);
>  };
>  
>  #define RTC_DEVICE_NAME_SIZE 20
> @@ -152,6 +154,9 @@ struct rtc_device {
>  	bool nvram_old_abi;
>  	struct bin_attribute *nvram;
>  
> +	time64_t max_hw_secs;
> +	time64_t min_hw_secs;
> +
>  #ifdef CONFIG_RTC_INTF_DEV_UIE_EMUL
>  	struct work_struct uie_task;
>  	struct timer_list uie_timer;
> @@ -225,6 +230,10 @@ int rtc_timer_start(struct rtc_device *rtc, struct rtc_timer *timer,
>  int rtc_set_offset(struct rtc_device *rtc, long offset);
>  void rtc_timer_do_work(struct work_struct *work);
>  
> +int rtc_read_range(struct rtc_device *rtc, time64_t *max_hw_secs,
> +		   time64_t *min_hw_secs);
> +int rtc_valid_range(struct rtc_device *rtc, struct rtc_time *tm);
> +
>  static inline bool is_leap_year(unsigned int year)
>  {
>  	return (!(year % 4) && (year % 100)) || !(year % 400);
> -- 
> 1.7.9.5
> 

-- 
Alexandre Belloni, Free Electrons
Embedded Linux and Kernel engineering
http://free-electrons.com



[Index of Archives]     [Linux Sound]     [ALSA Users]     [ALSA Devel]     [Linux Audio Users]     [Linux Media]     [Kernel]     [Gimp]     [Yosemite News]     [Linux Media]

  Powered by Linux