| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104 |
- /*
- * Copyright (c) 2006-2023, RT-Thread Development Team
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- * 2023-02-25 GuEe-GUI the first version
- */
- #include <rtthread.h>
- #include <rtdevice.h>
- #define DBG_TAG "reset.syscon.poweroff"
- #define DBG_LVL DBG_INFO
- #include <rtdbg.h>
- static struct rt_syscon *syscon;
- static rt_uint32_t offset, value, mask;
- static void syscon_poweroff(void)
- {
- /* Issue the poweroff */
- rt_syscon_update_bits(syscon, offset, mask, value);
- rt_thread_mdelay(1000);
- LOG_E("Unable to poweroff system");
- }
- static rt_err_t syscon_poweroff_probe(struct rt_platform_device *pdev)
- {
- rt_err_t mask_err, value_err;
- struct rt_ofw_node *np = pdev->parent.ofw_node;
- syscon = rt_syscon_find_by_ofw_phandle(np, "regmap");
- if (!syscon)
- {
- return -RT_ERROR;
- }
- if (rt_ofw_prop_read_u32(np, "offset", &offset))
- {
- LOG_E("read '%s' fail", "offset");
- return -RT_EINVAL;
- }
- value_err = rt_ofw_prop_read_u32(np, "value", &value);
- mask_err = rt_ofw_prop_read_u32(np, "mask", &mask);
- if (value_err && mask_err)
- {
- LOG_E("read '%s' and '%s' fail", "value", "mask");
- return -RT_EINVAL;
- }
- if (value_err)
- {
- /* support old binding */
- value = mask;
- mask = 0xffffffff;
- }
- else if (mask_err)
- {
- /* support value without mask */
- mask = 0xffffffff;
- }
- if (rt_dm_machine_shutdown)
- {
- LOG_E("rt_dm_machine_shutdown have hook %p", rt_dm_machine_shutdown);
- return -RT_EBUSY;
- }
- rt_dm_machine_shutdown = syscon_poweroff;
- return RT_EOK;
- }
- static const struct rt_ofw_node_id syscon_poweroff_ofw_ids[] =
- {
- { .compatible = "syscon-poweroff" },
- { /* sentinel */ }
- };
- static struct rt_platform_driver syscon_poweroff_driver =
- {
- .name = "reset-syscon-poweroff",
- .ids = syscon_poweroff_ofw_ids,
- .probe = syscon_poweroff_probe,
- };
- static int syscon_poweroff_driver_register(void)
- {
- rt_platform_driver_register(&syscon_poweroff_driver);
- return 0;
- }
- INIT_SUBSYS_EXPORT(syscon_poweroff_driver_register);
|