mirror of
https://github.com/openwrt/openwrt.git
synced 2025-01-21 20:08:24 +00:00
121 lines
3.5 KiB
Diff
121 lines
3.5 KiB
Diff
|
From ffc137a8949cd7859bdf139fa7a56b9dbdb4b2ce Mon Sep 17 00:00:00 2001
|
||
|
From: Phil Elwell <phil@raspberrypi.com>
|
||
|
Date: Fri, 15 May 2020 16:28:32 +0100
|
||
|
Subject: [PATCH] rtc: rv3028: Write BSM and TCE/TCR to EEPROM
|
||
|
|
||
|
Periodically the RV3028 refreshes registers from the EEPROM. When this
|
||
|
happens, some settings that have only been committed to registers are
|
||
|
lost. Change the handling of backup-switchover-mode and
|
||
|
trickle-resistor-ohms to write the EEPROM instead (if something has
|
||
|
changed), on the understanding that registers will be refreshed
|
||
|
afterwards.
|
||
|
|
||
|
See: https://github.com/raspberrypi/linux/issues/2912
|
||
|
|
||
|
Signed-off-by: Phil Elwell <phil@raspberrypi.com>
|
||
|
---
|
||
|
drivers/rtc/rtc-rv3028.c | 60 ++++++++++++++++++++++++++++------------
|
||
|
1 file changed, 43 insertions(+), 17 deletions(-)
|
||
|
|
||
|
--- a/drivers/rtc/rtc-rv3028.c
|
||
|
+++ b/drivers/rtc/rtc-rv3028.c
|
||
|
@@ -18,6 +18,7 @@
|
||
|
#include <linux/of_device.h>
|
||
|
#include <linux/regmap.h>
|
||
|
#include <linux/rtc.h>
|
||
|
+//#include "rtc-core.h"
|
||
|
|
||
|
#define RV3028_SEC 0x00
|
||
|
#define RV3028_MIN 0x01
|
||
|
@@ -73,7 +74,7 @@
|
||
|
|
||
|
#define RV3028_BACKUP_TCE BIT(5)
|
||
|
#define RV3028_BACKUP_TCR_MASK GENMASK(1,0)
|
||
|
-#define RV3028_BACKUP_BSM_MASK 0x0C
|
||
|
+#define RV3028_BACKUP_BSM_MASK GENMASK(3,2)
|
||
|
|
||
|
#define OFFSET_STEP_PPT 953674
|
||
|
|
||
|
@@ -601,7 +602,8 @@ static int rv3028_probe(struct i2c_clien
|
||
|
struct rv3028_data *rv3028;
|
||
|
int ret, status;
|
||
|
u32 ohms;
|
||
|
- u8 bsm;
|
||
|
+ u32 bsm;
|
||
|
+ u8 backup, backup_bits, backup_mask;
|
||
|
struct nvmem_config nvmem_cfg = {
|
||
|
.name = "rv3028_nvram",
|
||
|
.word_size = 1,
|
||
|
@@ -673,16 +675,17 @@ static int rv3028_probe(struct i2c_clien
|
||
|
if (ret)
|
||
|
return ret;
|
||
|
|
||
|
+ backup_bits = 0;
|
||
|
+ backup_mask = 0;
|
||
|
+
|
||
|
/* setup backup switchover mode */
|
||
|
- if (!device_property_read_u8(&client->dev, "backup-switchover-mode",
|
||
|
- &bsm)) {
|
||
|
+ dev_dbg(&client->dev, "Checking RTC backup switchover-mode\n");
|
||
|
+ if (!device_property_read_u32(&client->dev,
|
||
|
+ "backup-switchover-mode",
|
||
|
+ &bsm)) {
|
||
|
if (bsm <= 3) {
|
||
|
- ret = regmap_update_bits(rv3028->regmap, RV3028_BACKUP,
|
||
|
- RV3028_BACKUP_BSM_MASK,
|
||
|
- (bsm & 0x03) << 2);
|
||
|
-
|
||
|
- if (ret)
|
||
|
- return ret;
|
||
|
+ backup_bits |= (u8)(bsm << 2);
|
||
|
+ backup_mask |= RV3028_BACKUP_BSM_MASK;
|
||
|
} else {
|
||
|
dev_warn(&client->dev, "invalid backup switchover mode value\n");
|
||
|
}
|
||
|
@@ -698,15 +701,38 @@ static int rv3028_probe(struct i2c_clien
|
||
|
break;
|
||
|
|
||
|
if (i < ARRAY_SIZE(rv3028_trickle_resistors)) {
|
||
|
- ret = regmap_update_bits(rv3028->regmap, RV3028_BACKUP,
|
||
|
- RV3028_BACKUP_TCE |
|
||
|
- RV3028_BACKUP_TCR_MASK,
|
||
|
- RV3028_BACKUP_TCE | i);
|
||
|
- if (ret)
|
||
|
- return ret;
|
||
|
+ backup_bits |= RV3028_BACKUP_TCE | i;
|
||
|
+ backup_mask |= RV3028_BACKUP_TCE |
|
||
|
+ RV3028_BACKUP_TCR_MASK;
|
||
|
} else {
|
||
|
- dev_warn(&client->dev, "invalid trickle resistor value\n");
|
||
|
+ dev_warn(&client->dev,
|
||
|
+ "invalid trickle resistor value\n");
|
||
|
+ }
|
||
|
+ }
|
||
|
+
|
||
|
+ if (backup_mask) {
|
||
|
+ ret = rv3028_eeprom_read((void *)(rv3028->regmap),
|
||
|
+ RV3028_BACKUP,
|
||
|
+ (void *)&backup, 1);
|
||
|
+ if (!ret) {
|
||
|
+ /* Write EEPROM only if needed */
|
||
|
+ if ((backup & backup_mask) != backup_bits) {
|
||
|
+ backup = (backup & ~backup_mask) | backup_bits;
|
||
|
+ dev_dbg(&client->dev,
|
||
|
+ "Backup register doesn't match: EEPROM write required\n");
|
||
|
+ ret = rv3028_eeprom_write(
|
||
|
+ (void *)(rv3028->regmap),
|
||
|
+ RV3028_BACKUP, (void *)&backup, 1);
|
||
|
+ }
|
||
|
}
|
||
|
+
|
||
|
+ /* In the event of an EEPROM failure, update the register
|
||
|
+ instead. */
|
||
|
+ if (ret)
|
||
|
+ ret = regmap_update_bits(rv3028->regmap, RV3028_BACKUP,
|
||
|
+ backup_mask, backup_bits);
|
||
|
+ if (ret)
|
||
|
+ return ret;
|
||
|
}
|
||
|
|
||
|
ret = rtc_add_group(rv3028->rtc, &rv3028_attr_group);
|