1*4418919fSjohnjiang /* SPDX-License-Identifier: BSD-3-Clause
2*4418919fSjohnjiang * Copyright(c) 2010-2014 Intel Corporation
3*4418919fSjohnjiang */
4*4418919fSjohnjiang
5*4418919fSjohnjiang #include <stdio.h>
6*4418919fSjohnjiang #include <stdint.h>
7*4418919fSjohnjiang
8*4418919fSjohnjiang #include <rte_eal.h>
9*4418919fSjohnjiang #include <rte_memory.h>
10*4418919fSjohnjiang #include <rte_common.h>
11*4418919fSjohnjiang #include <rte_memzone.h>
12*4418919fSjohnjiang
13*4418919fSjohnjiang #include "test.h"
14*4418919fSjohnjiang
15*4418919fSjohnjiang /*
16*4418919fSjohnjiang * Memory
17*4418919fSjohnjiang * ======
18*4418919fSjohnjiang *
19*4418919fSjohnjiang * - Dump the mapped memory. The python-expect script checks that at
20*4418919fSjohnjiang * least one line is dumped.
21*4418919fSjohnjiang *
22*4418919fSjohnjiang * - Check that memory size is different than 0.
23*4418919fSjohnjiang *
24*4418919fSjohnjiang * - Try to read all memory; it should not segfault.
25*4418919fSjohnjiang */
26*4418919fSjohnjiang
27*4418919fSjohnjiang static int
check_mem(const struct rte_memseg_list * msl __rte_unused,const struct rte_memseg * ms,void * arg __rte_unused)28*4418919fSjohnjiang check_mem(const struct rte_memseg_list *msl __rte_unused,
29*4418919fSjohnjiang const struct rte_memseg *ms, void *arg __rte_unused)
30*4418919fSjohnjiang {
31*4418919fSjohnjiang volatile uint8_t *mem = (volatile uint8_t *) ms->addr;
32*4418919fSjohnjiang size_t i, max = ms->len;
33*4418919fSjohnjiang
34*4418919fSjohnjiang for (i = 0; i < max; i++, mem++)
35*4418919fSjohnjiang *mem;
36*4418919fSjohnjiang return 0;
37*4418919fSjohnjiang }
38*4418919fSjohnjiang
39*4418919fSjohnjiang static int
check_seg_fds(const struct rte_memseg_list * msl,const struct rte_memseg * ms,void * arg __rte_unused)40*4418919fSjohnjiang check_seg_fds(const struct rte_memseg_list *msl, const struct rte_memseg *ms,
41*4418919fSjohnjiang void *arg __rte_unused)
42*4418919fSjohnjiang {
43*4418919fSjohnjiang size_t offset;
44*4418919fSjohnjiang int ret;
45*4418919fSjohnjiang
46*4418919fSjohnjiang /* skip external segments */
47*4418919fSjohnjiang if (msl->external)
48*4418919fSjohnjiang return 0;
49*4418919fSjohnjiang
50*4418919fSjohnjiang /* try segment fd first. we're in a callback, so thread-unsafe */
51*4418919fSjohnjiang ret = rte_memseg_get_fd_thread_unsafe(ms);
52*4418919fSjohnjiang if (ret < 0) {
53*4418919fSjohnjiang /* ENOTSUP means segment is valid, but there is not support for
54*4418919fSjohnjiang * segment fd API (e.g. on FreeBSD).
55*4418919fSjohnjiang */
56*4418919fSjohnjiang if (errno == ENOTSUP)
57*4418919fSjohnjiang return 1;
58*4418919fSjohnjiang /* all other errors are treated as failures */
59*4418919fSjohnjiang return -1;
60*4418919fSjohnjiang }
61*4418919fSjohnjiang
62*4418919fSjohnjiang /* we're able to get memseg fd - try getting its offset */
63*4418919fSjohnjiang ret = rte_memseg_get_fd_offset_thread_unsafe(ms, &offset);
64*4418919fSjohnjiang if (ret < 0) {
65*4418919fSjohnjiang if (errno == ENOTSUP)
66*4418919fSjohnjiang return 1;
67*4418919fSjohnjiang return -1;
68*4418919fSjohnjiang }
69*4418919fSjohnjiang return 0;
70*4418919fSjohnjiang }
71*4418919fSjohnjiang
72*4418919fSjohnjiang static int
test_memory(void)73*4418919fSjohnjiang test_memory(void)
74*4418919fSjohnjiang {
75*4418919fSjohnjiang uint64_t s;
76*4418919fSjohnjiang int ret;
77*4418919fSjohnjiang
78*4418919fSjohnjiang /*
79*4418919fSjohnjiang * dump the mapped memory: the python-expect script checks
80*4418919fSjohnjiang * that at least one line is dumped
81*4418919fSjohnjiang */
82*4418919fSjohnjiang printf("Dump memory layout\n");
83*4418919fSjohnjiang rte_dump_physmem_layout(stdout);
84*4418919fSjohnjiang
85*4418919fSjohnjiang /* check that memory size is != 0 */
86*4418919fSjohnjiang s = rte_eal_get_physmem_size();
87*4418919fSjohnjiang if (s == 0) {
88*4418919fSjohnjiang printf("No memory detected\n");
89*4418919fSjohnjiang return -1;
90*4418919fSjohnjiang }
91*4418919fSjohnjiang
92*4418919fSjohnjiang /* try to read memory (should not segfault) */
93*4418919fSjohnjiang rte_memseg_walk(check_mem, NULL);
94*4418919fSjohnjiang
95*4418919fSjohnjiang /* check segment fd support */
96*4418919fSjohnjiang ret = rte_memseg_walk(check_seg_fds, NULL);
97*4418919fSjohnjiang if (ret == 1) {
98*4418919fSjohnjiang printf("Segment fd API is unsupported\n");
99*4418919fSjohnjiang } else if (ret == -1) {
100*4418919fSjohnjiang printf("Error getting segment fd's\n");
101*4418919fSjohnjiang return -1;
102*4418919fSjohnjiang }
103*4418919fSjohnjiang
104*4418919fSjohnjiang return 0;
105*4418919fSjohnjiang }
106*4418919fSjohnjiang
107*4418919fSjohnjiang REGISTER_TEST_COMMAND(memory_autotest, test_memory);
108