forked from zephyrproject-rtos/zephyr
-
Notifications
You must be signed in to change notification settings - Fork 0
/
reset_npcx.c
78 lines (61 loc) · 2.11 KB
/
reset_npcx.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
/*
* Copyright (c) 2024 Nuvoton Technology Corporation.
*
* SPDX-License-Identifier: Apache-2.0
*/
#define DT_DRV_COMPAT nuvoton_npcx_rst
#include <zephyr/kernel.h>
#include <zephyr/drivers/reset.h>
#if defined(CONFIG_SOC_SERIES_NPCX7)
#include <zephyr/dt-bindings/reset/npcx7_reset.h>
#elif defined(CONFIG_SOC_SERIES_NPCX9)
#include <zephyr/dt-bindings/reset/npcx9_reset.h>
#elif defined(CONFIG_SOC_SERIES_NPCX4)
#include <zephyr/dt-bindings/reset/npcx4_reset.h>
#endif
#include <zephyr/logging/log.h>
LOG_MODULE_REGISTER(rst_npcx);
#define NPCX_RESET_CTL_REG_BYTE_SIZE 4
#define NPCX_RESET_CTL_REG_OFFSET(id) ((id) >> (NPCX_RESET_CTL_REG_BYTE_SIZE + 1))
#define NPCX_RESET_CTL_REG_BIT(id) (((id) & ((1 << (NPCX_RESET_CTL_REG_BYTE_SIZE + 1)) - 1)))
#define NPCX_SWRST_TRG_WORD_START 0xC183
#define NPCX_SWRST_TRG_WORD_CLEAR 0x0
#define NPCX_SWRST_TRG_WORD_DONE 0xFFFF
#define NPCX_SWRST_DONE_TIMEOUT_US 100
struct reset_npcx_dev_config {
struct swrst_reg *reg_base;
};
static int reset_npcx_line_toggle(const struct device *dev, uint32_t id)
{
const struct reset_npcx_dev_config *const config = dev->config;
struct swrst_reg *const reg = config->reg_base;
unsigned int key;
uint8_t reg_offset;
uint8_t reg_bit;
int ret = 0;
if (!IN_RANGE(id, NPCX_RESET_ID_START, NPCX_RESET_ID_END)) {
LOG_ERR("Invalid Reset ID");
return -EINVAL;
}
reg_offset = NPCX_RESET_CTL_REG_OFFSET(id);
reg_bit = NPCX_RESET_CTL_REG_BIT(id);
key = irq_lock();
reg->SWRST_CTL[reg_offset] |= BIT(reg_bit);
reg->SWRST_TRG = NPCX_SWRST_TRG_WORD_CLEAR;
reg->SWRST_TRG = NPCX_SWRST_TRG_WORD_START;
if (!WAIT_FOR((reg->SWRST_TRG == NPCX_SWRST_TRG_WORD_DONE), NPCX_SWRST_DONE_TIMEOUT_US,
NULL)) {
LOG_ERR("Reset trig timeout");
ret = -EBUSY;
}
irq_unlock(key);
return ret;
}
static DEVICE_API(reset, reset_npcx_driver_api) = {
.line_toggle = reset_npcx_line_toggle,
};
static const struct reset_npcx_dev_config reset_npcx_config = {
.reg_base = (struct swrst_reg *)DT_INST_REG_ADDR(0),
};
DEVICE_DT_INST_DEFINE(0, NULL, NULL, NULL, &reset_npcx_config, PRE_KERNEL_1,
CONFIG_RESET_INIT_PRIORITY, &reset_npcx_driver_api);