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