162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * PS3 RTC Driver 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright 2009 Sony Corporation 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci#include <linux/kernel.h> 962306a36Sopenharmony_ci#include <linux/module.h> 1062306a36Sopenharmony_ci#include <linux/platform_device.h> 1162306a36Sopenharmony_ci#include <linux/rtc.h> 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_ci#include <asm/lv1call.h> 1462306a36Sopenharmony_ci#include <asm/ps3.h> 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_cistatic u64 read_rtc(void) 1862306a36Sopenharmony_ci{ 1962306a36Sopenharmony_ci int result; 2062306a36Sopenharmony_ci u64 rtc_val; 2162306a36Sopenharmony_ci u64 tb_val; 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci result = lv1_get_rtc(&rtc_val, &tb_val); 2462306a36Sopenharmony_ci BUG_ON(result); 2562306a36Sopenharmony_ci 2662306a36Sopenharmony_ci return rtc_val; 2762306a36Sopenharmony_ci} 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_cistatic int ps3_get_time(struct device *dev, struct rtc_time *tm) 3062306a36Sopenharmony_ci{ 3162306a36Sopenharmony_ci rtc_time64_to_tm(read_rtc() + ps3_os_area_get_rtc_diff(), tm); 3262306a36Sopenharmony_ci return 0; 3362306a36Sopenharmony_ci} 3462306a36Sopenharmony_ci 3562306a36Sopenharmony_cistatic int ps3_set_time(struct device *dev, struct rtc_time *tm) 3662306a36Sopenharmony_ci{ 3762306a36Sopenharmony_ci ps3_os_area_set_rtc_diff(rtc_tm_to_time64(tm) - read_rtc()); 3862306a36Sopenharmony_ci return 0; 3962306a36Sopenharmony_ci} 4062306a36Sopenharmony_ci 4162306a36Sopenharmony_cistatic const struct rtc_class_ops ps3_rtc_ops = { 4262306a36Sopenharmony_ci .read_time = ps3_get_time, 4362306a36Sopenharmony_ci .set_time = ps3_set_time, 4462306a36Sopenharmony_ci}; 4562306a36Sopenharmony_ci 4662306a36Sopenharmony_cistatic int __init ps3_rtc_probe(struct platform_device *dev) 4762306a36Sopenharmony_ci{ 4862306a36Sopenharmony_ci struct rtc_device *rtc; 4962306a36Sopenharmony_ci 5062306a36Sopenharmony_ci rtc = devm_rtc_allocate_device(&dev->dev); 5162306a36Sopenharmony_ci if (IS_ERR(rtc)) 5262306a36Sopenharmony_ci return PTR_ERR(rtc); 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_ci rtc->ops = &ps3_rtc_ops; 5562306a36Sopenharmony_ci rtc->range_max = U64_MAX; 5662306a36Sopenharmony_ci 5762306a36Sopenharmony_ci platform_set_drvdata(dev, rtc); 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_ci return devm_rtc_register_device(rtc); 6062306a36Sopenharmony_ci} 6162306a36Sopenharmony_ci 6262306a36Sopenharmony_cistatic struct platform_driver ps3_rtc_driver = { 6362306a36Sopenharmony_ci .driver = { 6462306a36Sopenharmony_ci .name = "rtc-ps3", 6562306a36Sopenharmony_ci }, 6662306a36Sopenharmony_ci}; 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_cimodule_platform_driver_probe(ps3_rtc_driver, ps3_rtc_probe); 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_ciMODULE_AUTHOR("Sony Corporation"); 7162306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 7262306a36Sopenharmony_ciMODULE_DESCRIPTION("ps3 RTC driver"); 7362306a36Sopenharmony_ciMODULE_ALIAS("platform:rtc-ps3"); 74