1 /*
2 * Copyright (c) 2024 Nordic Semiconductor ASA
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <errno.h>
8 #include <string.h>
9 #include <zephyr/kernel.h>
10 #include <zephyr/drivers/spi.h>
11 #include <zephyr/linker/devicetree_regions.h>
12 #include <zephyr/ztest.h>
13
14 #if CONFIG_TESTED_SPI_MODE == 0
15 #define SPI_MODE (SPI_WORD_SET(8) | SPI_LINES_SINGLE | SPI_TRANSFER_LSB)
16 #elif CONFIG_TESTED_SPI_MODE == 1
17 #define SPI_MODE (SPI_WORD_SET(8) | SPI_LINES_SINGLE | SPI_TRANSFER_MSB | SPI_MODE_CPHA)
18 #elif CONFIG_TESTED_SPI_MODE == 2
19 #define SPI_MODE (SPI_WORD_SET(8) | SPI_LINES_SINGLE | SPI_TRANSFER_LSB | SPI_MODE_CPOL)
20 #elif CONFIG_TESTED_SPI_MODE == 3
21 #define SPI_MODE (SPI_WORD_SET(8) | SPI_LINES_SINGLE | SPI_TRANSFER_MSB | SPI_MODE_CPHA \
22 | SPI_MODE_CPOL)
23 #endif
24
25 #define SPIM_OP (SPI_OP_MODE_MASTER | SPI_MODE)
26 #define SPIS_OP (SPI_OP_MODE_SLAVE | SPI_MODE)
27
28 static struct spi_dt_spec spim = SPI_DT_SPEC_GET(DT_NODELABEL(dut_spi_dt), SPIM_OP, 0);
29 static const struct device *spis_dev = DEVICE_DT_GET(DT_NODELABEL(dut_spis));
30 static const struct spi_config spis_config = {
31 .operation = SPIS_OP
32 };
33
34 static struct k_poll_signal async_sig = K_POLL_SIGNAL_INITIALIZER(async_sig);
35 static struct k_poll_event async_evt =
36 K_POLL_EVENT_INITIALIZER(K_POLL_TYPE_SIGNAL, K_POLL_MODE_NOTIFY_ONLY, &async_sig);
37
38 static struct k_poll_signal async_sig_spim = K_POLL_SIGNAL_INITIALIZER(async_sig_spim);
39 static struct k_poll_event async_evt_spim =
40 K_POLL_EVENT_INITIALIZER(K_POLL_TYPE_SIGNAL, K_POLL_MODE_NOTIFY_ONLY, &async_sig_spim);
41
42 #define MEMORY_SECTION(node) \
43 COND_CODE_1(DT_NODE_HAS_PROP(node, memory_regions), \
44 (__attribute__((__section__( \
45 LINKER_DT_NODE_REGION_NAME(DT_PHANDLE(node, memory_regions)))))), \
46 ())
47
48 static uint8_t spim_buffer[32] MEMORY_SECTION(DT_BUS(DT_NODELABEL(dut_spi_dt)));
49 static uint8_t spis_buffer[32] MEMORY_SECTION(DT_NODELABEL(dut_spis));
50
51 struct test_data {
52 struct k_work_delayable test_work;
53 struct k_sem sem;
54 int spim_alloc_idx;
55 int spis_alloc_idx;
56 struct spi_buf_set sets[4];
57 struct spi_buf_set *mtx_set;
58 struct spi_buf_set *mrx_set;
59 struct spi_buf_set *stx_set;
60 struct spi_buf_set *srx_set;
61 struct spi_buf bufs[8];
62 bool async;
63 };
64
65 static struct test_data tdata;
66
67 /* Allocate buffer from spim or spis space. */
buf_alloc(size_t len,bool spim)68 static uint8_t *buf_alloc(size_t len, bool spim)
69 {
70 int *idx = spim ? &tdata.spim_alloc_idx : &tdata.spis_alloc_idx;
71 uint8_t *buf = spim ? spim_buffer : spis_buffer;
72 size_t total = spim ? sizeof(spim_buffer) : sizeof(spis_buffer);
73 uint8_t *rv;
74
75 if (*idx + len > total) {
76 zassert_false(true);
77
78 return NULL;
79 }
80
81 rv = &buf[*idx];
82 *idx += len;
83
84 return rv;
85 }
86
work_handler(struct k_work * work)87 static void work_handler(struct k_work *work)
88 {
89 struct k_work_delayable *dwork = k_work_delayable_from_work(work);
90 struct test_data *td = CONTAINER_OF(dwork, struct test_data, test_work);
91 int rv;
92
93 if (!td->async) {
94 rv = spi_transceive_dt(&spim, td->mtx_set, td->mrx_set);
95 if (rv == 0) {
96 k_sem_give(&td->sem);
97 }
98 } else {
99 rv = spi_transceive_signal(spim.bus, &spim.config, td->mtx_set, td->mrx_set,
100 &async_sig_spim);
101 zassert_equal(rv, 0);
102
103 rv = k_poll(&async_evt_spim, 1, K_MSEC(200));
104 zassert_false(rv, "one or more events are not ready");
105
106 rv = async_evt_spim.signal->result;
107 zassert_equal(rv, 0);
108
109 /* Reinitializing for next call */
110 async_evt_spim.signal->signaled = 0U;
111 async_evt_spim.state = K_POLL_STATE_NOT_READY;
112
113 k_sem_give(&td->sem);
114 }
115 }
116
117 /** Copies data from buffers in the set to a single buffer which makes it easier
118 * to compare transmitted and received data.
119 *
120 * @param buf Output buffer.
121 * @param len Buffer length.
122 * @param set Set of buffers.
123 *
124 * @return Number of bytes copied.
125 */
cpy_data(uint8_t * buf,size_t len,struct spi_buf_set * set)126 static int cpy_data(uint8_t *buf, size_t len, struct spi_buf_set *set)
127 {
128 int idx = 0;
129
130 for (size_t i = 0; i < set->count; i++) {
131 size_t l = set->buffers[i].len;
132
133 if (len - idx >= l) {
134 memcpy(&buf[idx], set->buffers[i].buf, l);
135 idx += l;
136 } else {
137 return -1;
138 }
139 }
140
141 return idx;
142 }
143
144 /** Compare two sets.
145 *
146 * @param tx_set TX set.
147 * @param rx_set RX set.
148 * @param same_size True if it is expected to have the same amount of data in both sets.
149 *
150 * @return 0 if data is the same and other value indicate that check failed.
151 */
check_buffers(struct spi_buf_set * tx_set,struct spi_buf_set * rx_set,bool same_size)152 static int check_buffers(struct spi_buf_set *tx_set, struct spi_buf_set *rx_set, bool same_size)
153 {
154 static uint8_t tx_data[256];
155 static uint8_t rx_data[256];
156 int rx_len;
157 int tx_len;
158
159 if (!tx_set || !rx_set) {
160 return 0;
161 }
162
163 rx_len = cpy_data(rx_data, sizeof(rx_data), rx_set);
164 tx_len = cpy_data(tx_data, sizeof(tx_data), tx_set);
165 if (same_size && (rx_len != tx_len)) {
166 return -1;
167 }
168
169 return memcmp(tx_data, rx_data, rx_len);
170 }
171
172 /** Calculate expected number of received bytes by the SPI peripheral.
173 *
174 * It is used to check if SPI API call for peripheral SPI device returns correct value.
175 * @param tx_set TX set.
176 * @param rx_set RX set.
177 *
178 * @return Expected amount of received bytes.
179 */
peripheral_rx_len(struct spi_buf_set * tx_set,struct spi_buf_set * rx_set)180 static int peripheral_rx_len(struct spi_buf_set *tx_set, struct spi_buf_set *rx_set)
181 {
182 size_t tx_len = 0;
183 size_t rx_len = 0;
184
185 if (!tx_set || !rx_set) {
186 return 0;
187 }
188
189 for (size_t i = 0; i < tx_set->count; i++) {
190 tx_len += tx_set->buffers[i].len;
191 }
192
193 for (size_t i = 0; i < rx_set->count; i++) {
194 rx_len += rx_set->buffers[i].len;
195 }
196
197 return MIN(rx_len, tx_len);
198 }
199
200 /** Generic function which runs the test with sets prepared in the test data structure. */
run_test(bool m_same_size,bool s_same_size,bool async)201 static void run_test(bool m_same_size, bool s_same_size, bool async)
202 {
203 int rv;
204 int periph_rv;
205 int srx_len;
206
207 tdata.async = async;
208 rv = k_work_schedule(&tdata.test_work, K_MSEC(10));
209 zassert_equal(rv, 1);
210
211 if (!async) {
212 periph_rv = spi_transceive(spis_dev, &spis_config, tdata.stx_set, tdata.srx_set);
213 if (periph_rv == -ENOTSUP) {
214 ztest_test_skip();
215 }
216 } else {
217 rv = spi_transceive_signal(spis_dev, &spis_config, tdata.stx_set, tdata.srx_set,
218 &async_sig);
219 if (rv == -ENOTSUP) {
220 ztest_test_skip();
221 }
222 zassert_equal(rv, 0);
223
224 /* Transfer not finished yet */
225 rv = k_sem_take(&tdata.sem, K_NO_WAIT);
226 zassert_equal(rv, -EBUSY);
227
228 rv = k_poll(&async_evt, 1, K_MSEC(200));
229 zassert_false(rv, "one or more events are not ready");
230
231 periph_rv = async_evt.signal->result;
232
233 /* Reinitializing for next call */
234 async_evt.signal->signaled = 0U;
235 async_evt.state = K_POLL_STATE_NOT_READY;
236 }
237
238 rv = k_sem_take(&tdata.sem, K_MSEC(100));
239 zassert_equal(rv, 0);
240
241 srx_len = peripheral_rx_len(tdata.mtx_set, tdata.srx_set);
242
243 zassert_equal(periph_rv, srx_len, "Got: %d but expected:%d", periph_rv, srx_len);
244
245 rv = check_buffers(tdata.mtx_set, tdata.srx_set, m_same_size);
246 zassert_equal(rv, 0);
247
248 rv = check_buffers(tdata.stx_set, tdata.mrx_set, s_same_size);
249 zassert_equal(rv, 0);
250 }
251
252 /** Basic test where SPI controller and SPI peripheral have RX and TX sets which contains only one
253 * same size buffer.
254 */
test_basic(bool async)255 static void test_basic(bool async)
256 {
257 size_t len = 16;
258
259 for (int i = 0; i < 4; i++) {
260 tdata.bufs[i].buf = buf_alloc(len, i < 2);
261 tdata.bufs[i].len = len;
262 tdata.sets[i].buffers = &tdata.bufs[i];
263 tdata.sets[i].count = 1;
264 }
265
266 tdata.mtx_set = &tdata.sets[0];
267 tdata.mrx_set = &tdata.sets[1];
268 tdata.stx_set = &tdata.sets[2];
269 tdata.srx_set = &tdata.sets[3];
270
271 run_test(true, true, async);
272 }
273
ZTEST(spi_controller_peripheral,test_basic)274 ZTEST(spi_controller_peripheral, test_basic)
275 {
276 test_basic(false);
277 }
278
ZTEST(spi_controller_peripheral,test_basic_async)279 ZTEST(spi_controller_peripheral, test_basic_async)
280 {
281 test_basic(true);
282 }
283
284 /** Basic test with zero length buffers.
285 */
test_basic_zero_len(bool async)286 void test_basic_zero_len(bool async)
287 {
288 size_t len = 8;
289
290 /* SPIM */
291 tdata.bufs[0].buf = buf_alloc(len, true);
292 tdata.bufs[0].len = len;
293 tdata.bufs[1].buf = buf_alloc(len, true);
294 /* Intentionally len was set to 0 - second buffer "is empty". */
295 tdata.bufs[1].len = 0;
296 tdata.sets[0].buffers = &tdata.bufs[0];
297 tdata.sets[0].count = 2;
298 tdata.mtx_set = &tdata.sets[0];
299
300 tdata.bufs[2].buf = buf_alloc(len, true);
301 tdata.bufs[2].len = len;
302 tdata.bufs[3].buf = buf_alloc(len, true);
303 /* Intentionally len was set to 0 - second buffer "is empty". */
304 tdata.bufs[3].len = 0;
305 tdata.sets[1].buffers = &tdata.bufs[2];
306 tdata.sets[1].count = 2;
307 tdata.mrx_set = &tdata.sets[1];
308
309 /* SPIS */
310 tdata.bufs[4].buf = buf_alloc(len, false);
311 tdata.bufs[4].len = len;
312 tdata.sets[2].buffers = &tdata.bufs[4];
313 tdata.sets[2].count = 1;
314 tdata.stx_set = &tdata.sets[2];
315
316 tdata.bufs[6].buf = buf_alloc(len, false);
317 tdata.bufs[6].len = len;
318 tdata.sets[3].buffers = &tdata.bufs[6];
319 tdata.sets[3].count = 1;
320 tdata.srx_set = &tdata.sets[3];
321
322 run_test(true, true, async);
323 }
324
ZTEST(spi_controller_peripheral,test_basic_zero_len)325 ZTEST(spi_controller_peripheral, test_basic_zero_len)
326 {
327 test_basic_zero_len(false);
328 }
329
ZTEST(spi_controller_peripheral,test_basic_zero_len_async)330 ZTEST(spi_controller_peripheral, test_basic_zero_len_async)
331 {
332 test_basic_zero_len(true);
333 }
334
335 /** Setup a transfer where RX buffer on SPI controller and SPI peripheral are
336 * shorter than TX buffers. RX buffers shall contain beginning of TX data
337 * and last TX bytes that did not fit in the RX buffers shall be lost.
338 */
test_short_rx(bool async)339 static void test_short_rx(bool async)
340 {
341 size_t len = 16;
342
343 tdata.bufs[0].buf = buf_alloc(len, true);
344 tdata.bufs[0].len = len;
345 tdata.bufs[1].buf = buf_alloc(len, true);
346 tdata.bufs[1].len = len - 3; /* RX buffer */
347 tdata.bufs[2].buf = buf_alloc(len, false);
348 tdata.bufs[2].len = len;
349 tdata.bufs[3].buf = buf_alloc(len, false);
350 tdata.bufs[3].len = len - 4; /* RX buffer */
351
352 for (int i = 0; i < 4; i++) {
353 tdata.sets[i].buffers = &tdata.bufs[i];
354 tdata.sets[i].count = 1;
355 }
356
357 tdata.mtx_set = &tdata.sets[0];
358 tdata.mrx_set = &tdata.sets[1];
359 tdata.stx_set = &tdata.sets[2];
360 tdata.srx_set = &tdata.sets[3];
361
362 run_test(false, false, async);
363 }
364
ZTEST(spi_controller_peripheral,test_short_rx)365 ZTEST(spi_controller_peripheral, test_short_rx)
366 {
367 test_short_rx(false);
368 }
369
ZTEST(spi_controller_peripheral,test_short_rx_async)370 ZTEST(spi_controller_peripheral, test_short_rx_async)
371 {
372 test_short_rx(true);
373 }
374
375 /** Test where only master transmits. */
test_only_tx(bool async)376 static void test_only_tx(bool async)
377 {
378 size_t len = 16;
379
380 /* MTX buffer */
381 tdata.bufs[0].buf = buf_alloc(len, true);
382 tdata.bufs[0].len = len;
383 tdata.sets[0].buffers = &tdata.bufs[0];
384 tdata.sets[0].count = 1;
385 tdata.mtx_set = &tdata.sets[0];
386 tdata.mrx_set = NULL;
387
388 /* STX buffer */
389 tdata.bufs[1].buf = buf_alloc(len, false);
390 tdata.bufs[1].len = len;
391 tdata.sets[1].buffers = &tdata.bufs[1];
392 tdata.sets[1].count = 1;
393 tdata.srx_set = &tdata.sets[1];
394 tdata.stx_set = NULL;
395
396 run_test(true, true, async);
397 }
398
ZTEST(spi_controller_peripheral,test_only_tx)399 ZTEST(spi_controller_peripheral, test_only_tx)
400 {
401 test_only_tx(false);
402 }
403
ZTEST(spi_controller_peripheral,test_only_tx_async)404 ZTEST(spi_controller_peripheral, test_only_tx_async)
405 {
406 test_only_tx(true);
407 }
408
409 /** Test where only SPI controller transmits and SPI peripheral receives in chunks. */
test_only_tx_in_chunks(bool async)410 static void test_only_tx_in_chunks(bool async)
411 {
412 size_t len1 = 7;
413 size_t len2 = 8;
414
415 /* MTX buffer */
416 tdata.bufs[0].buf = buf_alloc(len1 + len2, true);
417 tdata.bufs[0].len = len1 + len2;
418 tdata.sets[0].buffers = &tdata.bufs[0];
419 tdata.sets[0].count = 1;
420 tdata.mtx_set = &tdata.sets[0];
421 tdata.mrx_set = NULL;
422
423 /* STX buffer */
424 tdata.bufs[1].buf = buf_alloc(len1, false);
425 tdata.bufs[1].len = len1;
426 tdata.bufs[2].buf = buf_alloc(len2, false);
427 tdata.bufs[2].len = len2;
428 tdata.sets[1].buffers = &tdata.bufs[1];
429 tdata.sets[1].count = 2;
430 tdata.srx_set = &tdata.sets[1];
431 tdata.stx_set = NULL;
432
433 run_test(true, true, async);
434 }
435
ZTEST(spi_controller_peripheral,test_only_tx_in_chunks)436 ZTEST(spi_controller_peripheral, test_only_tx_in_chunks)
437 {
438 test_only_tx_in_chunks(false);
439 }
440
ZTEST(spi_controller_peripheral,test_only_tx_in_chunks_async)441 ZTEST(spi_controller_peripheral, test_only_tx_in_chunks_async)
442 {
443 test_only_tx_in_chunks(true);
444 }
445
446 /** Test where only SPI peripheral transmits. */
test_only_rx(bool async)447 static void test_only_rx(bool async)
448 {
449 size_t len = 16;
450
451 /* MTX buffer */
452 tdata.bufs[0].buf = buf_alloc(len, true);
453 tdata.bufs[0].len = len;
454 tdata.sets[0].buffers = &tdata.bufs[0];
455 tdata.sets[0].count = 1;
456 tdata.mrx_set = &tdata.sets[0];
457 tdata.mtx_set = NULL;
458
459 /* STX buffer */
460 tdata.bufs[1].buf = buf_alloc(len, false);
461 tdata.bufs[1].len = len;
462 tdata.sets[1].buffers = &tdata.bufs[1];
463 tdata.sets[1].count = 1;
464 tdata.stx_set = &tdata.sets[1];
465 tdata.srx_set = NULL;
466
467 run_test(true, true, async);
468 }
469
ZTEST(spi_controller_peripheral,test_only_rx)470 ZTEST(spi_controller_peripheral, test_only_rx)
471 {
472 test_only_rx(false);
473 }
474
ZTEST(spi_controller_peripheral,test_only_rx_async)475 ZTEST(spi_controller_peripheral, test_only_rx_async)
476 {
477 test_only_rx(true);
478 }
479
480 /** Test where only SPI peripheral transmits in chunks. */
test_only_rx_in_chunks(bool async)481 static void test_only_rx_in_chunks(bool async)
482 {
483 size_t len1 = 7;
484 size_t len2 = 9;
485
486 /* MTX buffer */
487 tdata.bufs[0].buf = buf_alloc(len1 + len2, true);
488 tdata.bufs[0].len = len1 + len2;
489 tdata.sets[0].buffers = &tdata.bufs[0];
490 tdata.sets[0].count = 1;
491 tdata.mrx_set = &tdata.sets[0];
492 tdata.mtx_set = NULL;
493
494 /* STX buffer */
495 tdata.bufs[1].buf = buf_alloc(len1, false);
496 tdata.bufs[1].len = len1;
497 tdata.bufs[2].buf = buf_alloc(len2, false);
498 tdata.bufs[2].len = len2;
499 tdata.sets[1].buffers = &tdata.bufs[1];
500 tdata.sets[1].count = 2;
501 tdata.stx_set = &tdata.sets[1];
502 tdata.srx_set = NULL;
503
504 run_test(true, true, async);
505 }
506
ZTEST(spi_controller_peripheral,test_only_rx_in_chunks)507 ZTEST(spi_controller_peripheral, test_only_rx_in_chunks)
508 {
509 test_only_rx_in_chunks(false);
510 }
511
ZTEST(spi_controller_peripheral,test_only_rx_in_chunks_async)512 ZTEST(spi_controller_peripheral, test_only_rx_in_chunks_async)
513 {
514 test_only_rx_in_chunks(true);
515 }
516
before(void * not_used)517 static void before(void *not_used)
518 {
519 ARG_UNUSED(not_used);
520
521 memset(&tdata, 0, sizeof(tdata));
522 for (size_t i = 0; i < sizeof(spim_buffer); i++) {
523 spim_buffer[i] = (uint8_t)i;
524 }
525 for (size_t i = 0; i < sizeof(spis_buffer); i++) {
526 spis_buffer[i] = (uint8_t)(i + 0x80);
527 }
528
529 k_work_init_delayable(&tdata.test_work, work_handler);
530 k_sem_init(&tdata.sem, 0, 1);
531 }
532
after(void * not_used)533 static void after(void *not_used)
534 {
535 ARG_UNUSED(not_used);
536
537 k_work_cancel_delayable(&tdata.test_work);
538 }
539
suite_setup(void)540 static void *suite_setup(void)
541 {
542 return NULL;
543 }
544
545 ZTEST_SUITE(spi_controller_peripheral, NULL, suite_setup, before, after, NULL);
546