xref: /openbmc/linux/drivers/acpi/acpi_cmos_rtc.c (revision 596ca52a)
1d2912cb1SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
22fa97febSLan Tianyu /*
32fa97febSLan Tianyu  * ACPI support for CMOS RTC Address Space access
42fa97febSLan Tianyu  *
52fa97febSLan Tianyu  * Copyright (C) 2013, Intel Corporation
62fa97febSLan Tianyu  * Authors: Lan Tianyu <tianyu.lan@intel.com>
72fa97febSLan Tianyu  */
82fa97febSLan Tianyu 
9606e56c6SHanjun Guo #define pr_fmt(fmt) "ACPI: " fmt
10606e56c6SHanjun Guo 
112fa97febSLan Tianyu #include <linux/acpi.h>
122fa97febSLan Tianyu #include <linux/device.h>
132fa97febSLan Tianyu #include <linux/err.h>
142fa97febSLan Tianyu #include <linux/kernel.h>
152fa97febSLan Tianyu #include <linux/module.h>
16463a8630SArnd Bergmann #include <linux/mc146818rtc.h>
172fa97febSLan Tianyu 
182fa97febSLan Tianyu #include "internal.h"
192fa97febSLan Tianyu 
202fa97febSLan Tianyu static const struct acpi_device_id acpi_cmos_rtc_ids[] = {
212fa97febSLan Tianyu 	{ "PNP0B00" },
222fa97febSLan Tianyu 	{ "PNP0B01" },
232fa97febSLan Tianyu 	{ "PNP0B02" },
242fa97febSLan Tianyu 	{}
252fa97febSLan Tianyu };
262fa97febSLan Tianyu 
272fa97febSLan Tianyu static acpi_status
acpi_cmos_rtc_space_handler(u32 function,acpi_physical_address address,u32 bits,u64 * value64,void * handler_context,void * region_context)282fa97febSLan Tianyu acpi_cmos_rtc_space_handler(u32 function, acpi_physical_address address,
292fa97febSLan Tianyu 		      u32 bits, u64 *value64,
302fa97febSLan Tianyu 		      void *handler_context, void *region_context)
312fa97febSLan Tianyu {
322fa97febSLan Tianyu 	int i;
339389f46eSLee, Chun-Yi 	u8 *value = (u8 *)value64;
342fa97febSLan Tianyu 
352fa97febSLan Tianyu 	if (address > 0xff || !value64)
362fa97febSLan Tianyu 		return AE_BAD_PARAMETER;
372fa97febSLan Tianyu 
382fa97febSLan Tianyu 	if (function != ACPI_WRITE && function != ACPI_READ)
392fa97febSLan Tianyu 		return AE_BAD_PARAMETER;
402fa97febSLan Tianyu 
412fa97febSLan Tianyu 	spin_lock_irq(&rtc_lock);
422fa97febSLan Tianyu 
432fa97febSLan Tianyu 	for (i = 0; i < DIV_ROUND_UP(bits, 8); ++i, ++address, ++value)
442fa97febSLan Tianyu 		if (function == ACPI_READ)
452fa97febSLan Tianyu 			*value = CMOS_READ(address);
462fa97febSLan Tianyu 		else
472fa97febSLan Tianyu 			CMOS_WRITE(*value, address);
482fa97febSLan Tianyu 
492fa97febSLan Tianyu 	spin_unlock_irq(&rtc_lock);
502fa97febSLan Tianyu 
512fa97febSLan Tianyu 	return AE_OK;
522fa97febSLan Tianyu }
532fa97febSLan Tianyu 
acpi_install_cmos_rtc_space_handler(acpi_handle handle)54*596ca52aSZhang Rui int acpi_install_cmos_rtc_space_handler(acpi_handle handle)
552fa97febSLan Tianyu {
562fa97febSLan Tianyu 	acpi_status status;
572fa97febSLan Tianyu 
58*596ca52aSZhang Rui 	status = acpi_install_address_space_handler(handle,
592fa97febSLan Tianyu 			ACPI_ADR_SPACE_CMOS,
602fa97febSLan Tianyu 			&acpi_cmos_rtc_space_handler,
612fa97febSLan Tianyu 			NULL, NULL);
622fa97febSLan Tianyu 	if (ACPI_FAILURE(status)) {
63606e56c6SHanjun Guo 		pr_err("Error installing CMOS-RTC region handler\n");
642fa97febSLan Tianyu 		return -ENODEV;
652fa97febSLan Tianyu 	}
662fa97febSLan Tianyu 
67eec15edbSZhang Rui 	return 1;
682fa97febSLan Tianyu }
69*596ca52aSZhang Rui EXPORT_SYMBOL_GPL(acpi_install_cmos_rtc_space_handler);
702fa97febSLan Tianyu 
acpi_remove_cmos_rtc_space_handler(acpi_handle handle)71*596ca52aSZhang Rui void acpi_remove_cmos_rtc_space_handler(acpi_handle handle)
722fa97febSLan Tianyu {
73*596ca52aSZhang Rui 	if (ACPI_FAILURE(acpi_remove_address_space_handler(handle,
742fa97febSLan Tianyu 			ACPI_ADR_SPACE_CMOS, &acpi_cmos_rtc_space_handler)))
75606e56c6SHanjun Guo 		pr_err("Error removing CMOS-RTC region handler\n");
762fa97febSLan Tianyu }
77*596ca52aSZhang Rui EXPORT_SYMBOL_GPL(acpi_remove_cmos_rtc_space_handler);
78*596ca52aSZhang Rui 
acpi_cmos_rtc_attach_handler(struct acpi_device * adev,const struct acpi_device_id * id)79*596ca52aSZhang Rui static int acpi_cmos_rtc_attach_handler(struct acpi_device *adev, const struct acpi_device_id *id)
80*596ca52aSZhang Rui {
81*596ca52aSZhang Rui 	return acpi_install_cmos_rtc_space_handler(adev->handle);
82*596ca52aSZhang Rui }
83*596ca52aSZhang Rui 
acpi_cmos_rtc_detach_handler(struct acpi_device * adev)84*596ca52aSZhang Rui static void acpi_cmos_rtc_detach_handler(struct acpi_device *adev)
85*596ca52aSZhang Rui {
86*596ca52aSZhang Rui 	acpi_remove_cmos_rtc_space_handler(adev->handle);
87*596ca52aSZhang Rui }
882fa97febSLan Tianyu 
892fa97febSLan Tianyu static struct acpi_scan_handler cmos_rtc_handler = {
902fa97febSLan Tianyu 	.ids = acpi_cmos_rtc_ids,
91*596ca52aSZhang Rui 	.attach = acpi_cmos_rtc_attach_handler,
92*596ca52aSZhang Rui 	.detach = acpi_cmos_rtc_detach_handler,
932fa97febSLan Tianyu };
942fa97febSLan Tianyu 
acpi_cmos_rtc_init(void)952fa97febSLan Tianyu void __init acpi_cmos_rtc_init(void)
962fa97febSLan Tianyu {
972fa97febSLan Tianyu 	acpi_scan_add_handler(&cmos_rtc_handler);
982fa97febSLan Tianyu }
99