18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0+ 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * rtc-dm355evm.c - access battery-backed counter in MSP430 firmware 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright (c) 2008 by David Brownell 68c2ecf20Sopenharmony_ci */ 78c2ecf20Sopenharmony_ci#include <linux/kernel.h> 88c2ecf20Sopenharmony_ci#include <linux/init.h> 98c2ecf20Sopenharmony_ci#include <linux/rtc.h> 108c2ecf20Sopenharmony_ci#include <linux/platform_device.h> 118c2ecf20Sopenharmony_ci 128c2ecf20Sopenharmony_ci#include <linux/mfd/dm355evm_msp.h> 138c2ecf20Sopenharmony_ci#include <linux/module.h> 148c2ecf20Sopenharmony_ci 158c2ecf20Sopenharmony_ci 168c2ecf20Sopenharmony_ci/* 178c2ecf20Sopenharmony_ci * The MSP430 firmware on the DM355 EVM uses a watch crystal to feed 188c2ecf20Sopenharmony_ci * a 1 Hz counter. When a backup battery is supplied, that makes a 198c2ecf20Sopenharmony_ci * reasonable RTC for applications where alarms and non-NTP drift 208c2ecf20Sopenharmony_ci * compensation aren't important. 218c2ecf20Sopenharmony_ci * 228c2ecf20Sopenharmony_ci * The only real glitch is the inability to read or write all four 238c2ecf20Sopenharmony_ci * counter bytes atomically: the count may increment in the middle 248c2ecf20Sopenharmony_ci * of an operation, causing trouble when the LSB rolls over. 258c2ecf20Sopenharmony_ci * 268c2ecf20Sopenharmony_ci * This driver was tested with firmware revision A4. 278c2ecf20Sopenharmony_ci */ 288c2ecf20Sopenharmony_ciunion evm_time { 298c2ecf20Sopenharmony_ci u8 bytes[4]; 308c2ecf20Sopenharmony_ci u32 value; 318c2ecf20Sopenharmony_ci}; 328c2ecf20Sopenharmony_ci 338c2ecf20Sopenharmony_cistatic int dm355evm_rtc_read_time(struct device *dev, struct rtc_time *tm) 348c2ecf20Sopenharmony_ci{ 358c2ecf20Sopenharmony_ci union evm_time time; 368c2ecf20Sopenharmony_ci int status; 378c2ecf20Sopenharmony_ci int tries = 0; 388c2ecf20Sopenharmony_ci 398c2ecf20Sopenharmony_ci do { 408c2ecf20Sopenharmony_ci /* 418c2ecf20Sopenharmony_ci * Read LSB(0) to MSB(3) bytes. Defend against the counter 428c2ecf20Sopenharmony_ci * rolling over by re-reading until the value is stable, 438c2ecf20Sopenharmony_ci * and assuming the four reads take at most a few seconds. 448c2ecf20Sopenharmony_ci */ 458c2ecf20Sopenharmony_ci status = dm355evm_msp_read(DM355EVM_MSP_RTC_0); 468c2ecf20Sopenharmony_ci if (status < 0) 478c2ecf20Sopenharmony_ci return status; 488c2ecf20Sopenharmony_ci if (tries && time.bytes[0] == status) 498c2ecf20Sopenharmony_ci break; 508c2ecf20Sopenharmony_ci time.bytes[0] = status; 518c2ecf20Sopenharmony_ci 528c2ecf20Sopenharmony_ci status = dm355evm_msp_read(DM355EVM_MSP_RTC_1); 538c2ecf20Sopenharmony_ci if (status < 0) 548c2ecf20Sopenharmony_ci return status; 558c2ecf20Sopenharmony_ci if (tries && time.bytes[1] == status) 568c2ecf20Sopenharmony_ci break; 578c2ecf20Sopenharmony_ci time.bytes[1] = status; 588c2ecf20Sopenharmony_ci 598c2ecf20Sopenharmony_ci status = dm355evm_msp_read(DM355EVM_MSP_RTC_2); 608c2ecf20Sopenharmony_ci if (status < 0) 618c2ecf20Sopenharmony_ci return status; 628c2ecf20Sopenharmony_ci if (tries && time.bytes[2] == status) 638c2ecf20Sopenharmony_ci break; 648c2ecf20Sopenharmony_ci time.bytes[2] = status; 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_ci status = dm355evm_msp_read(DM355EVM_MSP_RTC_3); 678c2ecf20Sopenharmony_ci if (status < 0) 688c2ecf20Sopenharmony_ci return status; 698c2ecf20Sopenharmony_ci if (tries && time.bytes[3] == status) 708c2ecf20Sopenharmony_ci break; 718c2ecf20Sopenharmony_ci time.bytes[3] = status; 728c2ecf20Sopenharmony_ci 738c2ecf20Sopenharmony_ci } while (++tries < 5); 748c2ecf20Sopenharmony_ci 758c2ecf20Sopenharmony_ci dev_dbg(dev, "read timestamp %08x\n", time.value); 768c2ecf20Sopenharmony_ci 778c2ecf20Sopenharmony_ci rtc_time64_to_tm(le32_to_cpu(time.value), tm); 788c2ecf20Sopenharmony_ci return 0; 798c2ecf20Sopenharmony_ci} 808c2ecf20Sopenharmony_ci 818c2ecf20Sopenharmony_cistatic int dm355evm_rtc_set_time(struct device *dev, struct rtc_time *tm) 828c2ecf20Sopenharmony_ci{ 838c2ecf20Sopenharmony_ci union evm_time time; 848c2ecf20Sopenharmony_ci unsigned long value; 858c2ecf20Sopenharmony_ci int status; 868c2ecf20Sopenharmony_ci 878c2ecf20Sopenharmony_ci value = rtc_tm_to_time64(tm); 888c2ecf20Sopenharmony_ci time.value = cpu_to_le32(value); 898c2ecf20Sopenharmony_ci 908c2ecf20Sopenharmony_ci dev_dbg(dev, "write timestamp %08x\n", time.value); 918c2ecf20Sopenharmony_ci 928c2ecf20Sopenharmony_ci /* 938c2ecf20Sopenharmony_ci * REVISIT handle non-atomic writes ... maybe just retry until 948c2ecf20Sopenharmony_ci * byte[1] sticks (no rollover)? 958c2ecf20Sopenharmony_ci */ 968c2ecf20Sopenharmony_ci status = dm355evm_msp_write(time.bytes[0], DM355EVM_MSP_RTC_0); 978c2ecf20Sopenharmony_ci if (status < 0) 988c2ecf20Sopenharmony_ci return status; 998c2ecf20Sopenharmony_ci 1008c2ecf20Sopenharmony_ci status = dm355evm_msp_write(time.bytes[1], DM355EVM_MSP_RTC_1); 1018c2ecf20Sopenharmony_ci if (status < 0) 1028c2ecf20Sopenharmony_ci return status; 1038c2ecf20Sopenharmony_ci 1048c2ecf20Sopenharmony_ci status = dm355evm_msp_write(time.bytes[2], DM355EVM_MSP_RTC_2); 1058c2ecf20Sopenharmony_ci if (status < 0) 1068c2ecf20Sopenharmony_ci return status; 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci status = dm355evm_msp_write(time.bytes[3], DM355EVM_MSP_RTC_3); 1098c2ecf20Sopenharmony_ci if (status < 0) 1108c2ecf20Sopenharmony_ci return status; 1118c2ecf20Sopenharmony_ci 1128c2ecf20Sopenharmony_ci return 0; 1138c2ecf20Sopenharmony_ci} 1148c2ecf20Sopenharmony_ci 1158c2ecf20Sopenharmony_cistatic const struct rtc_class_ops dm355evm_rtc_ops = { 1168c2ecf20Sopenharmony_ci .read_time = dm355evm_rtc_read_time, 1178c2ecf20Sopenharmony_ci .set_time = dm355evm_rtc_set_time, 1188c2ecf20Sopenharmony_ci}; 1198c2ecf20Sopenharmony_ci 1208c2ecf20Sopenharmony_ci/*----------------------------------------------------------------------*/ 1218c2ecf20Sopenharmony_ci 1228c2ecf20Sopenharmony_cistatic int dm355evm_rtc_probe(struct platform_device *pdev) 1238c2ecf20Sopenharmony_ci{ 1248c2ecf20Sopenharmony_ci struct rtc_device *rtc; 1258c2ecf20Sopenharmony_ci 1268c2ecf20Sopenharmony_ci rtc = devm_rtc_allocate_device(&pdev->dev); 1278c2ecf20Sopenharmony_ci if (IS_ERR(rtc)) 1288c2ecf20Sopenharmony_ci return PTR_ERR(rtc); 1298c2ecf20Sopenharmony_ci 1308c2ecf20Sopenharmony_ci platform_set_drvdata(pdev, rtc); 1318c2ecf20Sopenharmony_ci 1328c2ecf20Sopenharmony_ci rtc->ops = &dm355evm_rtc_ops; 1338c2ecf20Sopenharmony_ci rtc->range_max = U32_MAX; 1348c2ecf20Sopenharmony_ci 1358c2ecf20Sopenharmony_ci return rtc_register_device(rtc); 1368c2ecf20Sopenharmony_ci} 1378c2ecf20Sopenharmony_ci 1388c2ecf20Sopenharmony_ci/* 1398c2ecf20Sopenharmony_ci * I2C is used to talk to the MSP430, but this platform device is 1408c2ecf20Sopenharmony_ci * exposed by an MFD driver that manages I2C communications. 1418c2ecf20Sopenharmony_ci */ 1428c2ecf20Sopenharmony_cistatic struct platform_driver rtc_dm355evm_driver = { 1438c2ecf20Sopenharmony_ci .probe = dm355evm_rtc_probe, 1448c2ecf20Sopenharmony_ci .driver = { 1458c2ecf20Sopenharmony_ci .name = "rtc-dm355evm", 1468c2ecf20Sopenharmony_ci }, 1478c2ecf20Sopenharmony_ci}; 1488c2ecf20Sopenharmony_ci 1498c2ecf20Sopenharmony_cimodule_platform_driver(rtc_dm355evm_driver); 1508c2ecf20Sopenharmony_ci 1518c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 152