1 /*
2  * Copyright (c) 2019 Linaro Limited
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #define DT_DRV_COMPAT arm_mhu
8 
9 #include <errno.h>
10 #include <zephyr/device.h>
11 #include <soc.h>
12 #include <zephyr/irq.h>
13 #include "ipm_mhu.h"
14 
15 #define IPM_MHU_REGS(dev) \
16 	((volatile struct ipm_mhu_reg_map_t *) \
17 	 (((const struct ipm_mhu_device_config * const)(dev)->config)->base))
18 
ipm_mhu_get_cpu_id(const struct device * d)19 static enum ipm_mhu_cpu_id_t ipm_mhu_get_cpu_id(const struct device *d)
20 {
21 	volatile uint32_t *p_mhu_dev_base;
22 	volatile uint32_t *p_cpu_id;
23 
24 	p_mhu_dev_base = (volatile uint32_t *)IPM_MHU_REGS(d);
25 
26 	p_cpu_id = (volatile uint32_t *)(((uint32_t)p_mhu_dev_base &
27 						SSE_200_DEVICE_BASE_REG_MSK) +
28 						SSE_200_CPU_ID_UNIT_OFFSET);
29 
30 	return (enum ipm_mhu_cpu_id_t)*p_cpu_id;
31 }
32 
ipm_mhu_get_status(const struct device * d,enum ipm_mhu_cpu_id_t cpu_id,uint32_t * status)33 static uint32_t ipm_mhu_get_status(const struct device *d,
34 						enum ipm_mhu_cpu_id_t cpu_id,
35 						uint32_t *status)
36 {
37 	struct ipm_mhu_reg_map_t *p_mhu_dev;
38 
39 	if (status == NULL) {
40 		return IPM_MHU_ERR_INVALID_ARG;
41 	}
42 
43 	p_mhu_dev = (struct ipm_mhu_reg_map_t *)IPM_MHU_REGS(d);
44 
45 	switch (cpu_id) {
46 	case IPM_MHU_CPU1:
47 		*status = p_mhu_dev->cpu1intr_stat;
48 		break;
49 	case IPM_MHU_CPU0:
50 	default:
51 		*status = p_mhu_dev->cpu0intr_stat;
52 		break;
53 	}
54 
55 	return IPM_MHU_ERR_NONE;
56 }
57 
ipm_mhu_send(const struct device * d,int wait,uint32_t cpu_id,const void * data,int size)58 static int ipm_mhu_send(const struct device *d, int wait, uint32_t cpu_id,
59 			  const void *data, int size)
60 {
61 	ARG_UNUSED(wait);
62 	ARG_UNUSED(data);
63 	const uint32_t set_val = 0x01;
64 
65 	struct ipm_mhu_reg_map_t *p_mhu_dev;
66 
67 	if (cpu_id >= IPM_MHU_CPU_MAX) {
68 		return -EINVAL;
69 	}
70 
71 	if (size > IPM_MHU_MAX_DATA_SIZE) {
72 		return -EMSGSIZE;
73 	}
74 
75 	p_mhu_dev = (struct ipm_mhu_reg_map_t *)IPM_MHU_REGS(d);
76 
77 	switch (cpu_id) {
78 	case IPM_MHU_CPU1:
79 		p_mhu_dev->cpu1intr_set = set_val;
80 		break;
81 	case IPM_MHU_CPU0:
82 	default:
83 		p_mhu_dev->cpu0intr_set = set_val;
84 		break;
85 	}
86 
87 	return 0;
88 }
89 
ipm_mhu_clear_val(const struct device * d,enum ipm_mhu_cpu_id_t cpu_id,uint32_t clear_val)90 static void ipm_mhu_clear_val(const struct device *d,
91 						enum ipm_mhu_cpu_id_t cpu_id,
92 						uint32_t clear_val)
93 {
94 	struct ipm_mhu_reg_map_t *p_mhu_dev;
95 
96 	p_mhu_dev = (struct ipm_mhu_reg_map_t *)IPM_MHU_REGS(d);
97 
98 	switch (cpu_id) {
99 	case IPM_MHU_CPU1:
100 		p_mhu_dev->cpu1intr_clr = clear_val;
101 		break;
102 	case IPM_MHU_CPU0:
103 	default:
104 		p_mhu_dev->cpu0intr_clr = clear_val;
105 		break;
106 	}
107 }
108 
ipm_mhu_max_id_val_get(const struct device * d)109 static uint32_t ipm_mhu_max_id_val_get(const struct device *d)
110 {
111 	ARG_UNUSED(d);
112 
113 	return IPM_MHU_MAX_ID_VAL;
114 }
115 
ipm_mhu_init(const struct device * d)116 static int ipm_mhu_init(const struct device *d)
117 {
118 	const struct ipm_mhu_device_config *config = d->config;
119 
120 	config->irq_config_func(d);
121 
122 	return 0;
123 }
124 
ipm_mhu_isr(const struct device * d)125 static void ipm_mhu_isr(const struct device *d)
126 {
127 	struct ipm_mhu_data *driver_data = d->data;
128 	enum ipm_mhu_cpu_id_t cpu_id;
129 	uint32_t ipm_mhu_status;
130 
131 	cpu_id = ipm_mhu_get_cpu_id(d);
132 
133 	ipm_mhu_get_status(d, cpu_id, &ipm_mhu_status);
134 	ipm_mhu_clear_val(d, cpu_id, ipm_mhu_status);
135 
136 	if (driver_data->callback) {
137 		driver_data->callback(d, driver_data->user_data, cpu_id,
138 				      &ipm_mhu_status);
139 	}
140 }
141 
ipm_mhu_set_enabled(const struct device * d,int enable)142 static int ipm_mhu_set_enabled(const struct device *d, int enable)
143 {
144 	ARG_UNUSED(d);
145 	ARG_UNUSED(enable);
146 	return 0;
147 }
148 
ipm_mhu_max_data_size_get(const struct device * d)149 static int ipm_mhu_max_data_size_get(const struct device *d)
150 {
151 	ARG_UNUSED(d);
152 
153 	return IPM_MHU_MAX_DATA_SIZE;
154 }
155 
ipm_mhu_register_cb(const struct device * d,ipm_callback_t cb,void * user_data)156 static void ipm_mhu_register_cb(const struct device *d,
157 				ipm_callback_t cb,
158 				void *user_data)
159 {
160 	struct ipm_mhu_data *driver_data = d->data;
161 
162 	driver_data->callback = cb;
163 	driver_data->user_data = user_data;
164 }
165 
166 static DEVICE_API(ipm, ipm_mhu_driver_api) = {
167 	.send = ipm_mhu_send,
168 	.register_callback = ipm_mhu_register_cb,
169 	.max_data_size_get = ipm_mhu_max_data_size_get,
170 	.max_id_val_get = ipm_mhu_max_id_val_get,
171 	.set_enabled = ipm_mhu_set_enabled,
172 };
173 
174 static void ipm_mhu_irq_config_func_0(const struct device *d);
175 
176 static const struct ipm_mhu_device_config ipm_mhu_cfg_0 = {
177 	.base = (uint8_t *)DT_INST_REG_ADDR(0),
178 	.irq_config_func = ipm_mhu_irq_config_func_0,
179 };
180 
181 static struct ipm_mhu_data ipm_mhu_data_0 = {
182 	.callback = NULL,
183 	.user_data = NULL,
184 };
185 
186 DEVICE_DT_INST_DEFINE(0,
187 			&ipm_mhu_init,
188 			NULL,
189 			&ipm_mhu_data_0,
190 			&ipm_mhu_cfg_0, PRE_KERNEL_1,
191 			CONFIG_KERNEL_INIT_PRIORITY_DEVICE,
192 			&ipm_mhu_driver_api);
193 
ipm_mhu_irq_config_func_0(const struct device * d)194 static void ipm_mhu_irq_config_func_0(const struct device *d)
195 {
196 	ARG_UNUSED(d);
197 	IRQ_CONNECT(DT_INST_IRQN(0),
198 			DT_INST_IRQ(0, priority),
199 			ipm_mhu_isr,
200 			DEVICE_DT_INST_GET(0),
201 			0);
202 	irq_enable(DT_INST_IRQN(0));
203 }
204 
205 static void ipm_mhu_irq_config_func_1(const struct device *d);
206 
207 static const struct ipm_mhu_device_config ipm_mhu_cfg_1 = {
208 	.base = (uint8_t *)DT_INST_REG_ADDR(1),
209 	.irq_config_func = ipm_mhu_irq_config_func_1,
210 };
211 
212 static struct ipm_mhu_data ipm_mhu_data_1 = {
213 	.callback = NULL,
214 	.user_data = NULL,
215 };
216 
217 DEVICE_DT_INST_DEFINE(1,
218 			&ipm_mhu_init,
219 			NULL,
220 			&ipm_mhu_data_1,
221 			&ipm_mhu_cfg_1, PRE_KERNEL_1,
222 			CONFIG_KERNEL_INIT_PRIORITY_DEVICE,
223 			&ipm_mhu_driver_api);
224 
ipm_mhu_irq_config_func_1(const struct device * d)225 static void ipm_mhu_irq_config_func_1(const struct device *d)
226 {
227 	ARG_UNUSED(d);
228 	IRQ_CONNECT(DT_INST_IRQN(1),
229 			DT_INST_IRQ(1, priority),
230 			ipm_mhu_isr,
231 			DEVICE_DT_INST_GET(1),
232 			0);
233 	irq_enable(DT_INST_IRQN(1));
234 }
235