1 /*
2 * Copyright (c) 2021-2022 Antmicro <www.antmicro.com>
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT xlnx_fpga
8
9 #include <zephyr/device.h>
10 #include <zephyr/drivers/fpga.h>
11 #include "fpga_zynqmp.h"
12 #include <errno.h>
13 #include <string.h>
14 #include <zephyr/sys/byteorder.h>
15 #include <stdlib.h>
16 #include <stdio.h>
17
18 #include <zephyr/logging/log.h>
19 LOG_MODULE_REGISTER(fpga_zynqmp, CONFIG_FPGA_LOG_LEVEL);
20
power_up_fpga(void)21 static void power_up_fpga(void)
22 {
23 PMU_GLOBAL_PWRUP_EN = PWR_PL_MASK;
24 PMU_REQ_PWRUP_TRIG = PWR_PL_MASK;
25
26 while (PWR_STATUS & PWR_PL_MASK) {
27 };
28 }
29
30 struct zynqmp_fpga_data {
31 char FPGA_info[16];
32 };
33
update_part_name(const struct device * dev)34 static void update_part_name(const struct device *dev)
35 {
36 struct zynqmp_fpga_data *data = dev->data;
37
38 int zu_number = 0;
39
40 switch (IDCODE & IDCODE_MASK) {
41 case ZU2_IDCODE:
42 zu_number = 2;
43 break;
44 case ZU3_IDCODE:
45 zu_number = 3;
46 break;
47 case ZU4_IDCODE:
48 zu_number = 4;
49 break;
50 case ZU5_IDCODE:
51 zu_number = 5;
52 break;
53 case ZU6_IDCODE:
54 zu_number = 6;
55 break;
56 case ZU7_IDCODE:
57 zu_number = 7;
58 break;
59 case ZU9_IDCODE:
60 zu_number = 9;
61 break;
62 case ZU11_IDCODE:
63 zu_number = 11;
64 break;
65 case ZU15_IDCODE:
66 zu_number = 15;
67 break;
68 case ZU17_IDCODE:
69 zu_number = 17;
70 break;
71 case ZU19_IDCODE:
72 zu_number = 19;
73 break;
74 case ZU21_IDCODE:
75 zu_number = 21;
76 break;
77 case ZU25_IDCODE:
78 zu_number = 25;
79 break;
80 case ZU27_IDCODE:
81 zu_number = 27;
82 break;
83 case ZU28_IDCODE:
84 zu_number = 28;
85 break;
86 case ZU29_IDCODE:
87 zu_number = 29;
88 break;
89 case ZU39_IDCODE:
90 zu_number = 39;
91 break;
92 case ZU43_IDCODE:
93 zu_number = 43;
94 break;
95 case ZU46_IDCODE:
96 zu_number = 46;
97 break;
98 case ZU47_IDCODE:
99 zu_number = 47;
100 break;
101 case ZU48_IDCODE:
102 zu_number = 48;
103 break;
104 case ZU49_IDCODE:
105 zu_number = 49;
106 break;
107 }
108
109 if (zu_number == 0) {
110 snprintf(data->FPGA_info, sizeof(data->FPGA_info), "unknown");
111 } else {
112 snprintf(data->FPGA_info, sizeof(data->FPGA_info), "Part name: ZU%d", zu_number);
113 }
114
115 }
116
117 /*
118 * This function is responsible for shifting the bitstream
119 * by its header and extracting information from this header.
120 * The bitstream header has 5 sections starting with the letters a,b,c...
121 * Each section has the following structure:
122 * [key][length of data][data]
123 */
parse_header(const struct device * dev,uint32_t * image_ptr,uint32_t * img_size)124 static uint32_t *parse_header(const struct device *dev, uint32_t *image_ptr,
125 uint32_t *img_size)
126 {
127 unsigned char *header = (unsigned char *)image_ptr;
128
129 uint32_t length = XLNX_BITSTREAM_SECTION_LENGTH(header);
130
131 /* shift to the next section*/
132 header += 0x4U + length;
133
134 if (*header++ != 'a') {
135 LOG_ERR("Incorrect bitstream format");
136 return NULL;
137 }
138
139 length = XLNX_BITSTREAM_SECTION_LENGTH(header);
140 /* shift to the data section*/
141 header += 0x2U;
142
143 LOG_DBG("Design name = %s", header);
144
145 header += length;
146
147 if (*header++ != 'b') {
148 LOG_ERR("Incorrect bitstream format");
149 return NULL;
150 }
151
152 length = XLNX_BITSTREAM_SECTION_LENGTH(header);
153 /* shift to the data section*/
154 header += 0x2U;
155 LOG_DBG("Part name = %s", header);
156
157 header += length;
158
159 if (*header++ != 'c') {
160 LOG_ERR("Incorrect bitstream format");
161 return NULL;
162 }
163
164 length = XLNX_BITSTREAM_SECTION_LENGTH(header);
165 /* shift to the data section*/
166 header += 0x2U;
167
168 LOG_DBG("Date = %s", header);
169
170 header += length;
171
172 if (*header++ != 'd') {
173 LOG_ERR("Incorrect bitstream format");
174 return NULL;
175 }
176
177 length = XLNX_BITSTREAM_SECTION_LENGTH(header);
178 /* shift to the data section*/
179 header += 0x2U;
180
181 LOG_DBG("Time = %s", header);
182
183 header += length;
184
185 if (*header++ != 'e') {
186 LOG_ERR("Incorrect bitstream format");
187 return NULL;
188 }
189
190 /*
191 * The last section is the raw bitstream.
192 * It is preceded by its size, which is needed for DMA transfer.
193 */
194 *img_size =
195 ((uint32_t)*header << 24) | ((uint32_t) *(header + 1) << 16) |
196 ((uint32_t) *(header + 2) << 8) | ((uint32_t) *(header + 3));
197
198 return (uint32_t *)header;
199 }
200
csudma_transfer(uint32_t size)201 static int csudma_transfer(uint32_t size)
202 {
203 /* setup the source DMA channel */
204 CSUDMA_SRC_ADDR = (uint32_t)BITSTREAM & CSUDMA_SRC_ADDR_MASK;
205 CSUDMA_SRC_ADDR_MSB = 0;
206 CSUDMA_SRC_SIZE = size << CSUDMA_SRC_SIZE_SHIFT;
207
208 /* wait for the SRC_DMA to complete */
209 while ((CSUDMA_SRC_I_STS & CSUDMA_I_STS_DONE_MASK) != CSUDMA_I_STS_DONE_MASK) {
210 };
211
212 /* acknowledge the transfer has completed */
213 CSUDMA_SRC_I_STS = CSUDMA_I_STS_DONE_MASK;
214
215 return 0;
216 }
217
wait_for_done(void)218 static int wait_for_done(void)
219 {
220 /* wait for PCAP PL_DONE */
221 while ((PCAP_STATUS & PCAP_PL_DONE_MASK) != PCAP_PL_DONE_MASK) {
222 };
223
224 PCAP_RESET = PCAP_RESET_MASK;
225 power_up_fpga();
226
227 return 0;
228 }
229
zynqmp_fpga_get_status(const struct device * dev)230 static enum FPGA_status zynqmp_fpga_get_status(const struct device *dev)
231 {
232 ARG_UNUSED(dev);
233
234 if ((PCAP_STATUS & PCAP_PL_INIT_MASK) && (PCAP_STATUS & PCAP_PL_DONE_MASK)) {
235 return FPGA_STATUS_ACTIVE;
236 } else {
237 return FPGA_STATUS_INACTIVE;
238 }
239 }
240
zynqmp_fpga_get_info(const struct device * dev)241 static const char *zynqmp_fpga_get_info(const struct device *dev)
242 {
243 struct zynqmp_fpga_data *data = dev->data;
244
245 return data->FPGA_info;
246 }
247
zynqmp_fpga_reset(const struct device * dev)248 static int zynqmp_fpga_reset(const struct device *dev)
249 {
250 ARG_UNUSED(dev);
251
252 /* Reset PL */
253 PCAP_PROG = PCAP_PROG_RESET_MASK;
254 PCAP_PROG = ~PCAP_PROG_RESET_MASK;
255
256 while ((PCAP_STATUS & PCAP_CFG_RESET) != PCAP_CFG_RESET) {
257 };
258
259 return 0;
260 }
261
init_pcap(const struct device * dev)262 static int init_pcap(const struct device *dev)
263 {
264 /* take PCAP out of Reset */
265 PCAP_RESET = ~PCAP_RESET_MASK;
266
267 /* select PCAP mode and change PCAP to write mode */
268 PCAP_CTRL = PCAP_PR_MASK;
269 PCAP_RDWR = PCAP_WRITE_MASK;
270
271 power_up_fpga();
272
273 /* setup the SSS */
274 CSU_SSS_CFG = PCAP_PCAP_SSS_MASK;
275
276 zynqmp_fpga_reset(dev);
277
278 /* wait for pl init */
279 while ((PCAP_STATUS & PCAP_PL_INIT_MASK) != PCAP_PL_INIT_MASK) {
280 };
281
282 return 0;
283 }
284
zynqmp_fpga_load(const struct device * dev,uint32_t * image_ptr,uint32_t img_size)285 static int zynqmp_fpga_load(const struct device *dev, uint32_t *image_ptr,
286 uint32_t img_size)
287 {
288 uint32_t *addr = parse_header(dev, image_ptr, &img_size);
289
290 if (addr == NULL) {
291 LOG_ERR("Failed to read bitstream");
292 return -EINVAL;
293 }
294
295 for (int i = 0; i < (img_size / 4); i++) {
296 *(BITSTREAM + i) = BSWAP_32(*(addr + i));
297 }
298
299 init_pcap(dev);
300 csudma_transfer(img_size);
301 wait_for_done();
302
303 return 0;
304 }
305
zynqmp_fpga_init(const struct device * dev)306 static int zynqmp_fpga_init(const struct device *dev)
307 {
308 /* turn on PCAP CLK */
309 PCAP_CLK_CTRL = PCAP_CLK_CTRL | PCAP_CLKACT_MASK;
310
311 update_part_name(dev);
312
313 return 0;
314 }
315
316 static struct zynqmp_fpga_data fpga_data;
317
318 static DEVICE_API(fpga, zynqmp_api) = {
319 .reset = zynqmp_fpga_reset,
320 .load = zynqmp_fpga_load,
321 .get_status = zynqmp_fpga_get_status,
322 .get_info = zynqmp_fpga_get_info
323 };
324
325 DEVICE_DT_INST_DEFINE(0, &zynqmp_fpga_init, NULL, &fpga_data, NULL,
326 POST_KERNEL, CONFIG_FPGA_INIT_PRIORITY, &zynqmp_api);
327