18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * ACPI support for CMOS RTC Address Space access 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright (C) 2013, Intel Corporation 68c2ecf20Sopenharmony_ci * Authors: Lan Tianyu <tianyu.lan@intel.com> 78c2ecf20Sopenharmony_ci */ 88c2ecf20Sopenharmony_ci 98c2ecf20Sopenharmony_ci#include <linux/acpi.h> 108c2ecf20Sopenharmony_ci#include <linux/device.h> 118c2ecf20Sopenharmony_ci#include <linux/err.h> 128c2ecf20Sopenharmony_ci#include <linux/kernel.h> 138c2ecf20Sopenharmony_ci#include <linux/module.h> 148c2ecf20Sopenharmony_ci#include <linux/mc146818rtc.h> 158c2ecf20Sopenharmony_ci 168c2ecf20Sopenharmony_ci#include "internal.h" 178c2ecf20Sopenharmony_ci 188c2ecf20Sopenharmony_cistatic const struct acpi_device_id acpi_cmos_rtc_ids[] = { 198c2ecf20Sopenharmony_ci { "PNP0B00" }, 208c2ecf20Sopenharmony_ci { "PNP0B01" }, 218c2ecf20Sopenharmony_ci { "PNP0B02" }, 228c2ecf20Sopenharmony_ci {} 238c2ecf20Sopenharmony_ci}; 248c2ecf20Sopenharmony_ci 258c2ecf20Sopenharmony_cistatic acpi_status 268c2ecf20Sopenharmony_ciacpi_cmos_rtc_space_handler(u32 function, acpi_physical_address address, 278c2ecf20Sopenharmony_ci u32 bits, u64 *value64, 288c2ecf20Sopenharmony_ci void *handler_context, void *region_context) 298c2ecf20Sopenharmony_ci{ 308c2ecf20Sopenharmony_ci int i; 318c2ecf20Sopenharmony_ci u8 *value = (u8 *)value64; 328c2ecf20Sopenharmony_ci 338c2ecf20Sopenharmony_ci if (address > 0xff || !value64) 348c2ecf20Sopenharmony_ci return AE_BAD_PARAMETER; 358c2ecf20Sopenharmony_ci 368c2ecf20Sopenharmony_ci if (function != ACPI_WRITE && function != ACPI_READ) 378c2ecf20Sopenharmony_ci return AE_BAD_PARAMETER; 388c2ecf20Sopenharmony_ci 398c2ecf20Sopenharmony_ci spin_lock_irq(&rtc_lock); 408c2ecf20Sopenharmony_ci 418c2ecf20Sopenharmony_ci for (i = 0; i < DIV_ROUND_UP(bits, 8); ++i, ++address, ++value) 428c2ecf20Sopenharmony_ci if (function == ACPI_READ) 438c2ecf20Sopenharmony_ci *value = CMOS_READ(address); 448c2ecf20Sopenharmony_ci else 458c2ecf20Sopenharmony_ci CMOS_WRITE(*value, address); 468c2ecf20Sopenharmony_ci 478c2ecf20Sopenharmony_ci spin_unlock_irq(&rtc_lock); 488c2ecf20Sopenharmony_ci 498c2ecf20Sopenharmony_ci return AE_OK; 508c2ecf20Sopenharmony_ci} 518c2ecf20Sopenharmony_ci 528c2ecf20Sopenharmony_cistatic int acpi_install_cmos_rtc_space_handler(struct acpi_device *adev, 538c2ecf20Sopenharmony_ci const struct acpi_device_id *id) 548c2ecf20Sopenharmony_ci{ 558c2ecf20Sopenharmony_ci acpi_status status; 568c2ecf20Sopenharmony_ci 578c2ecf20Sopenharmony_ci status = acpi_install_address_space_handler(adev->handle, 588c2ecf20Sopenharmony_ci ACPI_ADR_SPACE_CMOS, 598c2ecf20Sopenharmony_ci &acpi_cmos_rtc_space_handler, 608c2ecf20Sopenharmony_ci NULL, NULL); 618c2ecf20Sopenharmony_ci if (ACPI_FAILURE(status)) { 628c2ecf20Sopenharmony_ci pr_err(PREFIX "Error installing CMOS-RTC region handler\n"); 638c2ecf20Sopenharmony_ci return -ENODEV; 648c2ecf20Sopenharmony_ci } 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_ci return 1; 678c2ecf20Sopenharmony_ci} 688c2ecf20Sopenharmony_ci 698c2ecf20Sopenharmony_cistatic void acpi_remove_cmos_rtc_space_handler(struct acpi_device *adev) 708c2ecf20Sopenharmony_ci{ 718c2ecf20Sopenharmony_ci if (ACPI_FAILURE(acpi_remove_address_space_handler(adev->handle, 728c2ecf20Sopenharmony_ci ACPI_ADR_SPACE_CMOS, &acpi_cmos_rtc_space_handler))) 738c2ecf20Sopenharmony_ci pr_err(PREFIX "Error removing CMOS-RTC region handler\n"); 748c2ecf20Sopenharmony_ci} 758c2ecf20Sopenharmony_ci 768c2ecf20Sopenharmony_cistatic struct acpi_scan_handler cmos_rtc_handler = { 778c2ecf20Sopenharmony_ci .ids = acpi_cmos_rtc_ids, 788c2ecf20Sopenharmony_ci .attach = acpi_install_cmos_rtc_space_handler, 798c2ecf20Sopenharmony_ci .detach = acpi_remove_cmos_rtc_space_handler, 808c2ecf20Sopenharmony_ci}; 818c2ecf20Sopenharmony_ci 828c2ecf20Sopenharmony_civoid __init acpi_cmos_rtc_init(void) 838c2ecf20Sopenharmony_ci{ 848c2ecf20Sopenharmony_ci acpi_scan_add_handler(&cmos_rtc_handler); 858c2ecf20Sopenharmony_ci} 86