mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2026-02-05 21:41:43 +00:00
312 lines
6.5 KiB
C
312 lines
6.5 KiB
C
/*
|
|
* Copyright (c) 2006-2023, RT-Thread Development Team
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Change Logs:
|
|
* Date Author Notes
|
|
* 2022-09-24 GuEe-GUI the first version
|
|
*/
|
|
|
|
#include <rtdevice.h>
|
|
|
|
#define DBG_TAG "rtdm.power"
|
|
#define DBG_LVL DBG_INFO
|
|
#include <rtdbg.h>
|
|
|
|
struct power_off_track
|
|
{
|
|
rt_slist_t list;
|
|
|
|
struct rt_device *dev;
|
|
rt_err_t (*callback)(struct rt_device *);
|
|
};
|
|
|
|
void (*rt_dm_machine_shutdown)(void) = RT_NULL;
|
|
void (*rt_dm_machine_reset)(void) = RT_NULL;
|
|
|
|
static RT_DEFINE_SPINLOCK(_power_off_lock);
|
|
static rt_slist_t _power_off_handler_nodes[RT_DM_POWER_OFF_MODE_NR][RT_DM_POWER_OFF_PRIO_NR] =
|
|
{
|
|
[0 ... RT_DM_POWER_OFF_MODE_NR - 1] =
|
|
{
|
|
[0 ... RT_DM_POWER_OFF_PRIO_NR - 1] =
|
|
{
|
|
RT_NULL,
|
|
}
|
|
}
|
|
};
|
|
|
|
static rt_used char * const _mode_name[] =
|
|
{
|
|
[RT_DM_POWER_OFF_MODE_SHUTDOWN] = "SHUTDOWN",
|
|
[RT_DM_POWER_OFF_MODE_RESET] = "RESET",
|
|
};
|
|
|
|
rt_err_t rt_dm_power_off_handler(struct rt_device *dev, int mode, int priority,
|
|
rt_err_t (*callback)(struct rt_device *dev))
|
|
{
|
|
struct power_off_track *track;
|
|
|
|
RT_ASSERT(mode < RT_DM_POWER_OFF_MODE_NR);
|
|
RT_ASSERT(priority < RT_DM_POWER_OFF_PRIO_NR);
|
|
|
|
track = rt_malloc(sizeof(*track));
|
|
|
|
if (!track)
|
|
{
|
|
return -RT_ENOMEM;
|
|
}
|
|
|
|
rt_slist_init(&track->list);
|
|
track->dev = dev;
|
|
track->callback = callback;
|
|
|
|
rt_hw_spin_lock(&_power_off_lock.lock);
|
|
|
|
rt_slist_insert(&_power_off_handler_nodes[mode][priority], &track->list);
|
|
|
|
rt_hw_spin_unlock(&_power_off_lock.lock);
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
static void dm_power_off_handler(int mode)
|
|
{
|
|
struct power_off_track *track;
|
|
|
|
rt_hw_spin_lock(&_power_off_lock.lock);
|
|
|
|
for (int i = 0; i < RT_DM_POWER_OFF_PRIO_NR; ++i)
|
|
{
|
|
rt_slist_t *nodes = &_power_off_handler_nodes[mode][i];
|
|
|
|
rt_slist_for_each_entry(track, nodes, list)
|
|
{
|
|
rt_err_t err;
|
|
struct rt_device *dev = track->dev;
|
|
|
|
if ((err = track->callback(dev)))
|
|
{
|
|
LOG_E("%s: %s fail error = %s", dev ? rt_dm_dev_get_name(dev) : RT_NULL,
|
|
_mode_name[mode], rt_strerror(err));
|
|
}
|
|
}
|
|
}
|
|
|
|
rt_hw_spin_unlock(&_power_off_lock.lock);
|
|
}
|
|
|
|
struct reboot_mode_track
|
|
{
|
|
rt_slist_t list;
|
|
|
|
struct rt_device *dev;
|
|
rt_err_t (*callback)(struct rt_device *, char *cmd);
|
|
};
|
|
|
|
static char *_reboot_mode_cmd = "normal";
|
|
static RT_DEFINE_SPINLOCK(_reboot_mode_lock);
|
|
static rt_slist_t _reboot_mode_handler_nodes = { RT_NULL };
|
|
|
|
rt_err_t rt_dm_reboot_mode_register(struct rt_device *dev,
|
|
rt_err_t (*callback)(struct rt_device *, char *cmd))
|
|
{
|
|
struct reboot_mode_track *track;
|
|
|
|
track = rt_malloc(sizeof(*track));
|
|
|
|
if (!track)
|
|
{
|
|
return -RT_ENOMEM;
|
|
}
|
|
|
|
rt_slist_init(&track->list);
|
|
track->dev = dev;
|
|
track->callback = callback;
|
|
|
|
rt_hw_spin_lock(&_reboot_mode_lock.lock);
|
|
|
|
rt_slist_insert(&_reboot_mode_handler_nodes, &track->list);
|
|
|
|
rt_hw_spin_unlock(&_reboot_mode_lock.lock);
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
rt_err_t rt_dm_reboot_mode_unregister(struct rt_device *dev)
|
|
{
|
|
struct reboot_mode_track *track, *target_track = RT_NULL;
|
|
|
|
rt_hw_spin_lock(&_reboot_mode_lock.lock);
|
|
|
|
rt_slist_for_each_entry(track, &_reboot_mode_handler_nodes, list)
|
|
{
|
|
if (track->dev == dev)
|
|
{
|
|
target_track = track;
|
|
rt_slist_remove(&_reboot_mode_handler_nodes, &track->list);
|
|
break;
|
|
}
|
|
}
|
|
|
|
rt_hw_spin_unlock(&_reboot_mode_lock.lock);
|
|
|
|
if (target_track)
|
|
{
|
|
rt_free(target_track);
|
|
}
|
|
|
|
return target_track ? RT_EOK : -RT_EEMPTY;
|
|
}
|
|
|
|
static rt_err_t dm_reboot_notifiy(struct rt_device *request_dev)
|
|
{
|
|
struct reboot_mode_track *track;
|
|
|
|
rt_hw_spin_lock(&_reboot_mode_lock.lock);
|
|
|
|
rt_slist_for_each_entry(track, &_reboot_mode_handler_nodes, list)
|
|
{
|
|
rt_err_t err;
|
|
struct rt_device *dev = track->dev;
|
|
|
|
if ((err = track->callback(dev, _reboot_mode_cmd)))
|
|
{
|
|
LOG_E("%s: %s fail error = %s", dev ? rt_dm_dev_get_name(dev) : RT_NULL,
|
|
"reboot mode apply", rt_strerror(err));
|
|
}
|
|
}
|
|
|
|
rt_hw_spin_unlock(&_reboot_mode_lock.lock);
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
static int reboot_mode_init(void)
|
|
{
|
|
return rt_dm_power_off_handler(RT_NULL, RT_DM_POWER_OFF_MODE_RESET,
|
|
RT_DM_POWER_OFF_PRIO_HIGH, &dm_reboot_notifiy);
|
|
}
|
|
INIT_CORE_EXPORT(reboot_mode_init);
|
|
|
|
void rt_hw_cpu_reset_mode(char *cmd)
|
|
{
|
|
static RT_DEFINE_SPINLOCK(pe_lock);
|
|
|
|
rt_hw_spin_lock(&pe_lock.lock);
|
|
|
|
_reboot_mode_cmd = cmd ? : _reboot_mode_cmd;
|
|
|
|
rt_hw_cpu_reset();
|
|
|
|
/* Unreachable */
|
|
rt_hw_spin_unlock(&pe_lock.lock);
|
|
}
|
|
|
|
static struct rt_thread power_task;
|
|
static void power_task_async(void (*fn)(void));
|
|
|
|
rt_inline rt_bool_t power_need_async(void)
|
|
{
|
|
struct rt_thread *tid = rt_thread_self();
|
|
|
|
return tid && tid != &power_task && rt_interrupt_get_nest();
|
|
}
|
|
|
|
void rt_hw_cpu_shutdown(void)
|
|
{
|
|
register rt_ubase_t level;
|
|
|
|
if (power_need_async())
|
|
{
|
|
power_task_async(&rt_hw_cpu_shutdown);
|
|
|
|
return;
|
|
}
|
|
|
|
dm_power_off_handler(RT_DM_POWER_OFF_MODE_SHUTDOWN);
|
|
|
|
LOG_I("Shutdown");
|
|
|
|
/* Machine shutdown */
|
|
if (rt_dm_machine_shutdown)
|
|
{
|
|
rt_dm_machine_shutdown();
|
|
}
|
|
|
|
level = rt_hw_interrupt_disable();
|
|
while (level)
|
|
{
|
|
RT_ASSERT(0);
|
|
}
|
|
}
|
|
MSH_CMD_EXPORT_ALIAS(rt_hw_cpu_shutdown, shutdown, shutdown machine);
|
|
|
|
void rt_hw_cpu_reset(void)
|
|
{
|
|
register rt_ubase_t level;
|
|
|
|
if (power_need_async())
|
|
{
|
|
power_task_async(&rt_hw_cpu_reset);
|
|
|
|
return;
|
|
}
|
|
|
|
dm_power_off_handler(RT_DM_POWER_OFF_MODE_RESET);
|
|
|
|
LOG_I("Reset");
|
|
|
|
/* Machine reset */
|
|
if (rt_dm_machine_reset)
|
|
{
|
|
rt_dm_machine_reset();
|
|
}
|
|
|
|
level = rt_hw_interrupt_disable();
|
|
while (level)
|
|
{
|
|
RT_ASSERT(0);
|
|
}
|
|
}
|
|
|
|
static int reset(int args, char**argv)
|
|
{
|
|
if (args > 1)
|
|
{
|
|
rt_hw_cpu_reset_mode(argv[1]);
|
|
}
|
|
else
|
|
{
|
|
rt_hw_cpu_reset();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
MSH_CMD_EXPORT(reset, reset machine);
|
|
|
|
static void power_task_entry(void *param)
|
|
{
|
|
void (*fn)(void) = rt_thread_self()->parameter;
|
|
|
|
fn();
|
|
}
|
|
|
|
static void power_task_async(void (*fn)(void))
|
|
{
|
|
power_task.parameter = fn;
|
|
|
|
rt_thread_startup(&power_task);
|
|
}
|
|
|
|
static int power_init(void)
|
|
{
|
|
static rt_uint8_t power_task_stack[DM_THREAD_STACK_SIZE];
|
|
|
|
return rt_thread_init(&power_task, "pwr", power_task_entry, RT_NULL,
|
|
&power_task_stack, sizeof(power_task_stack),
|
|
RT_THREAD_PRIORITY_MAX / 2, 32);
|
|
}
|
|
INIT_CORE_EXPORT(power_init);
|