18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * PS3 RTC Driver 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright 2009 Sony Corporation 68c2ecf20Sopenharmony_ci */ 78c2ecf20Sopenharmony_ci 88c2ecf20Sopenharmony_ci#include <linux/kernel.h> 98c2ecf20Sopenharmony_ci#include <linux/module.h> 108c2ecf20Sopenharmony_ci#include <linux/platform_device.h> 118c2ecf20Sopenharmony_ci#include <linux/rtc.h> 128c2ecf20Sopenharmony_ci 138c2ecf20Sopenharmony_ci#include <asm/lv1call.h> 148c2ecf20Sopenharmony_ci#include <asm/ps3.h> 158c2ecf20Sopenharmony_ci 168c2ecf20Sopenharmony_ci 178c2ecf20Sopenharmony_cistatic u64 read_rtc(void) 188c2ecf20Sopenharmony_ci{ 198c2ecf20Sopenharmony_ci int result; 208c2ecf20Sopenharmony_ci u64 rtc_val; 218c2ecf20Sopenharmony_ci u64 tb_val; 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_ci result = lv1_get_rtc(&rtc_val, &tb_val); 248c2ecf20Sopenharmony_ci BUG_ON(result); 258c2ecf20Sopenharmony_ci 268c2ecf20Sopenharmony_ci return rtc_val; 278c2ecf20Sopenharmony_ci} 288c2ecf20Sopenharmony_ci 298c2ecf20Sopenharmony_cistatic int ps3_get_time(struct device *dev, struct rtc_time *tm) 308c2ecf20Sopenharmony_ci{ 318c2ecf20Sopenharmony_ci rtc_time64_to_tm(read_rtc() + ps3_os_area_get_rtc_diff(), tm); 328c2ecf20Sopenharmony_ci return 0; 338c2ecf20Sopenharmony_ci} 348c2ecf20Sopenharmony_ci 358c2ecf20Sopenharmony_cistatic int ps3_set_time(struct device *dev, struct rtc_time *tm) 368c2ecf20Sopenharmony_ci{ 378c2ecf20Sopenharmony_ci ps3_os_area_set_rtc_diff(rtc_tm_to_time64(tm) - read_rtc()); 388c2ecf20Sopenharmony_ci return 0; 398c2ecf20Sopenharmony_ci} 408c2ecf20Sopenharmony_ci 418c2ecf20Sopenharmony_cistatic const struct rtc_class_ops ps3_rtc_ops = { 428c2ecf20Sopenharmony_ci .read_time = ps3_get_time, 438c2ecf20Sopenharmony_ci .set_time = ps3_set_time, 448c2ecf20Sopenharmony_ci}; 458c2ecf20Sopenharmony_ci 468c2ecf20Sopenharmony_cistatic int __init ps3_rtc_probe(struct platform_device *dev) 478c2ecf20Sopenharmony_ci{ 488c2ecf20Sopenharmony_ci struct rtc_device *rtc; 498c2ecf20Sopenharmony_ci 508c2ecf20Sopenharmony_ci rtc = devm_rtc_allocate_device(&dev->dev); 518c2ecf20Sopenharmony_ci if (IS_ERR(rtc)) 528c2ecf20Sopenharmony_ci return PTR_ERR(rtc); 538c2ecf20Sopenharmony_ci 548c2ecf20Sopenharmony_ci rtc->ops = &ps3_rtc_ops; 558c2ecf20Sopenharmony_ci rtc->range_max = U64_MAX; 568c2ecf20Sopenharmony_ci 578c2ecf20Sopenharmony_ci platform_set_drvdata(dev, rtc); 588c2ecf20Sopenharmony_ci 598c2ecf20Sopenharmony_ci return rtc_register_device(rtc); 608c2ecf20Sopenharmony_ci} 618c2ecf20Sopenharmony_ci 628c2ecf20Sopenharmony_cistatic struct platform_driver ps3_rtc_driver = { 638c2ecf20Sopenharmony_ci .driver = { 648c2ecf20Sopenharmony_ci .name = "rtc-ps3", 658c2ecf20Sopenharmony_ci }, 668c2ecf20Sopenharmony_ci}; 678c2ecf20Sopenharmony_ci 688c2ecf20Sopenharmony_cimodule_platform_driver_probe(ps3_rtc_driver, ps3_rtc_probe); 698c2ecf20Sopenharmony_ci 708c2ecf20Sopenharmony_ciMODULE_AUTHOR("Sony Corporation"); 718c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 728c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("ps3 RTC driver"); 738c2ecf20Sopenharmony_ciMODULE_ALIAS("platform:rtc-ps3"); 74