1 /*
2 * Copyright (c) 2023 Advanced Micro Devices, Inc.
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT xlnx_zynqmp_ipi_mailbox
8
9 #include "ipm_xlnx_ipi.h"
10
11 #include <errno.h>
12 #include <zephyr/device.h>
13 #include <zephyr/drivers/ipm.h>
14 #include <zephyr/irq.h>
15
16 #include <zephyr/logging/log.h>
17 LOG_MODULE_REGISTER(ipm_xlnx_ipi, CONFIG_IPM_LOG_LEVEL);
18
19 #define XLNX_IPI_MAX_BUF_SIZE_BYTES 32
20
21 struct xlnx_ipi_data {
22 size_t len;
23 void *user_data;
24 uint8_t data[];
25 };
26
27 struct xlnx_ipi_reg_info {
28 uint32_t ipi_ch_bit;
29 };
30
31 static const struct xlnx_ipi_reg_info xlnx_ipi_reg_info_zynqmp[] = {
32 {.ipi_ch_bit = IPI_CH0_BIT}, /* IPI CH ID 0 - Default APU */
33 {.ipi_ch_bit = IPI_CH1_BIT}, /* IPI CH ID 1 - Default RPU0 */
34 {.ipi_ch_bit = IPI_CH2_BIT}, /* IPI CH ID 2 - Default RPU1 */
35 {.ipi_ch_bit = IPI_CH3_BIT}, /* IPI CH ID 3 - Default PMU0 */
36 {.ipi_ch_bit = IPI_CH4_BIT}, /* IPI CH ID 4 - Default PMU1 */
37 {.ipi_ch_bit = IPI_CH5_BIT}, /* IPI CH ID 5 - Default PMU2 */
38 {.ipi_ch_bit = IPI_CH6_BIT}, /* IPI CH ID 6 - Default PMU3 */
39 {.ipi_ch_bit = IPI_CH7_BIT}, /* IPI CH ID 7 - Default PL0 */
40 {.ipi_ch_bit = IPI_CH8_BIT}, /* IPI CH ID 8 - Default PL1 */
41 {.ipi_ch_bit = IPI_CH9_BIT}, /* IPI CH ID 9 - Default PL2 */
42 {.ipi_ch_bit = IPI_CH10_BIT}, /* IPI CH ID 10 - Default PL3 */
43 };
44
45 struct xlnx_ipi_config {
46 uint32_t ipi_ch_bit;
47 uint32_t host_ipi_reg;
48 int (*xlnx_ipi_config_func)(const struct device *dev);
49 const struct device **cdev_list;
50 int num_cdev;
51 };
52
53 struct xlnx_ipi_child_data {
54 bool enabled;
55 ipm_callback_t ipm_callback;
56 void *user_data;
57 };
58
59 struct xlnx_ipi_child_config {
60 const char *node_id;
61 uint32_t local_request_region;
62 uint32_t local_response_region;
63 uint32_t remote_request_region;
64 uint32_t remote_response_region;
65 uint32_t host_ipi_reg;
66 uint32_t remote_ipi_id;
67 uint32_t remote_ipi_ch_bit;
68 };
69
xlnx_mailbox_rx_isr(const struct device * dev)70 static void xlnx_mailbox_rx_isr(const struct device *dev)
71 {
72 const struct xlnx_ipi_config *config;
73 const struct device **cdev_list;
74 const struct xlnx_ipi_child_config *cdev_conf;
75 const struct xlnx_ipi_child_data *cdev_data;
76 uint8_t ipi_buf[XLNX_IPI_MAX_BUF_SIZE_BYTES + sizeof(struct xlnx_ipi_data)];
77 int num_cdev;
78 struct xlnx_ipi_data *msg;
79 const struct device *cdev;
80 uint32_t remote_ipi_ch_bit;
81 int i, j;
82
83 config = dev->config;
84 cdev_list = config->cdev_list;
85 num_cdev = config->num_cdev;
86 msg = (struct xlnx_ipi_data *)ipi_buf;
87 for (i = 0; i < num_cdev; i++) {
88 cdev = cdev_list[i];
89 cdev_conf = cdev->config;
90 cdev_data = cdev->data;
91
92 if (!cdev_data->enabled) {
93 continue;
94 }
95
96 remote_ipi_ch_bit = cdev_conf->remote_ipi_ch_bit;
97 if (!sys_test_bit(config->host_ipi_reg + IPI_ISR, remote_ipi_ch_bit)) {
98 continue;
99 }
100
101 msg->len = XLNX_IPI_MAX_BUF_SIZE_BYTES;
102 msg->user_data = cdev_data->user_data;
103 for (j = 0; j < XLNX_IPI_MAX_BUF_SIZE_BYTES; j++) {
104 msg->data[j] = sys_read8(cdev_conf->remote_request_region + j);
105 }
106 if (cdev_data->ipm_callback) {
107 cdev_data->ipm_callback(cdev, cdev_data->user_data,
108 cdev_conf->remote_ipi_id, msg);
109 }
110 sys_set_bit(config->host_ipi_reg + IPI_ISR, remote_ipi_ch_bit);
111 }
112 }
113
xlnx_ipi_send(const struct device * ipmdev,int wait,uint32_t id,const void * data,int size)114 static int xlnx_ipi_send(const struct device *ipmdev, int wait, uint32_t id, const void *data,
115 int size)
116 {
117 const uint8_t *msg = (uint8_t *)data;
118 const struct xlnx_ipi_child_config *config = ipmdev->config;
119 unsigned int key;
120 int i, obs_bit;
121
122 ARG_UNUSED(id);
123
124 if (size > XLNX_IPI_MAX_BUF_SIZE_BYTES) {
125 return -EMSGSIZE;
126 }
127
128 key = irq_lock();
129 if (msg) {
130 /* Write buffer to send data */
131 for (i = 0; i < size; i++) {
132 sys_write8(msg[i], config->local_request_region + i);
133 }
134 }
135 irq_unlock(key);
136
137 sys_set_bit(config->host_ipi_reg + IPI_TRIG, config->remote_ipi_ch_bit);
138
139 obs_bit = 0;
140 do {
141 obs_bit = sys_test_bit(config->host_ipi_reg + IPI_OBS, config->remote_ipi_ch_bit);
142 } while (obs_bit && wait);
143
144 return 0;
145 }
146
xlnx_ipi_register_callback(const struct device * port,ipm_callback_t cb,void * user_data)147 static void xlnx_ipi_register_callback(const struct device *port, ipm_callback_t cb,
148 void *user_data)
149 {
150 struct xlnx_ipi_child_data *data = port->data;
151
152 data->ipm_callback = cb;
153 data->user_data = user_data;
154 }
155
xlnx_ipi_max_data_size_get(const struct device * ipmdev)156 static int xlnx_ipi_max_data_size_get(const struct device *ipmdev)
157 {
158 return XLNX_IPI_MAX_BUF_SIZE_BYTES;
159 }
160
xlnx_ipi_max_id_val_get(const struct device * ipmdev)161 static uint32_t xlnx_ipi_max_id_val_get(const struct device *ipmdev)
162 {
163 return UINT32_MAX;
164 }
165
xlnx_ipi_set_enabled(const struct device * ipmdev,int enable)166 static int xlnx_ipi_set_enabled(const struct device *ipmdev, int enable)
167 {
168 const struct xlnx_ipi_child_config *config = ipmdev->config;
169 struct xlnx_ipi_child_data *data = ipmdev->data;
170
171 if (enable) {
172 sys_set_bit(config->host_ipi_reg + IPI_IER, config->remote_ipi_ch_bit);
173 } else {
174 sys_set_bit(config->host_ipi_reg + IPI_IDR, config->remote_ipi_ch_bit);
175 }
176
177 /* If IPI channel bit in IPI Mask Register is not set, then interrupt is enabled */
178 if (!sys_test_bit(config->host_ipi_reg + IPI_IMR, config->remote_ipi_ch_bit)) {
179 data->enabled = enable;
180 return 0;
181 }
182
183 return -EINVAL;
184 }
185
xlnx_ipi_init(const struct device * dev)186 static int xlnx_ipi_init(const struct device *dev)
187 {
188 const struct xlnx_ipi_config *conf = dev->config;
189
190 /* disable all the interrupts */
191 sys_write32(0xFFFFFFFF, conf->host_ipi_reg + IPI_IDR);
192
193 /* clear status of any previous interrupts */
194 sys_write32(0xFFFFFFFF, conf->host_ipi_reg + IPI_ISR);
195
196 conf->xlnx_ipi_config_func(dev);
197
198 return 0;
199 }
200
201 static DEVICE_API(ipm, xlnx_ipi_api) = {
202 .send = xlnx_ipi_send,
203 .register_callback = xlnx_ipi_register_callback,
204 .max_data_size_get = xlnx_ipi_max_data_size_get,
205 .max_id_val_get = xlnx_ipi_max_id_val_get,
206 .set_enabled = xlnx_ipi_set_enabled,
207 };
208
209 #define GET_CHILD_DEV(node_id) DEVICE_DT_GET(node_id),
210
211 #define XLNX_IPI_CHILD(ch_node) \
212 struct xlnx_ipi_child_data xlnx_ipi_child_data##ch_node = { \
213 .enabled = false, \
214 .ipm_callback = NULL, \
215 }; \
216 struct xlnx_ipi_child_config xlnx_ipi_child_config##ch_node = { \
217 .local_request_region = DT_REG_ADDR_BY_NAME(ch_node, local_request_region), \
218 .local_response_region = DT_REG_ADDR_BY_NAME(ch_node, local_response_region), \
219 .remote_request_region = DT_REG_ADDR_BY_NAME(ch_node, remote_request_region), \
220 .remote_response_region = DT_REG_ADDR_BY_NAME(ch_node, remote_response_region), \
221 .remote_ipi_id = DT_PROP(ch_node, remote_ipi_id), \
222 .remote_ipi_ch_bit = \
223 xlnx_ipi_reg_info_zynqmp[DT_PROP(ch_node, remote_ipi_id)].ipi_ch_bit, \
224 .host_ipi_reg = DT_REG_ADDR_BY_NAME(DT_PARENT(ch_node), host_ipi_reg), \
225 }; \
226 DEVICE_DT_DEFINE(ch_node, NULL, NULL, &xlnx_ipi_child_data##ch_node, \
227 &xlnx_ipi_child_config##ch_node, POST_KERNEL, \
228 CONFIG_KERNEL_INIT_PRIORITY_DEFAULT, &xlnx_ipi_api);
229
230 #define XLNX_IPI(inst) \
231 DT_INST_FOREACH_CHILD_STATUS_OKAY(inst, XLNX_IPI_CHILD); \
232 static const struct device *cdev##inst[] = { \
233 DT_INST_FOREACH_CHILD_STATUS_OKAY(inst, GET_CHILD_DEV)}; \
234 static int xlnx_ipi_config_func##inst(const struct device *dev); \
235 struct xlnx_ipi_config xlnx_ipi_config##inst = { \
236 .host_ipi_reg = DT_INST_REG_ADDR_BY_NAME(inst, host_ipi_reg), \
237 .xlnx_ipi_config_func = xlnx_ipi_config_func##inst, \
238 .cdev_list = cdev##inst, \
239 .num_cdev = ARRAY_SIZE(cdev##inst), \
240 }; \
241 DEVICE_DT_INST_DEFINE(inst, &xlnx_ipi_init, NULL, NULL, /* data */ \
242 &xlnx_ipi_config##inst, /* conf */ \
243 POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT, NULL); \
244 static int xlnx_ipi_config_func##inst(const struct device *dev) \
245 { \
246 IRQ_CONNECT(DT_INST_IRQN(inst), DT_INST_IRQ(inst, priority), xlnx_mailbox_rx_isr, \
247 DEVICE_DT_INST_GET(inst), 0); \
248 irq_enable(DT_INST_IRQN(inst)); \
249 LOG_DBG("irq %d is enabled: %s\n", DT_INST_IRQN(inst), \
250 irq_is_enabled(DT_INST_IRQN(inst)) ? "true" : "false"); \
251 return 0; \
252 }
253
254 DT_INST_FOREACH_STATUS_OKAY(XLNX_IPI)
255