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 #include <string.h>
8*4418919fSjohnjiang #include <inttypes.h>
9*4418919fSjohnjiang #include <sys/queue.h>
10*4418919fSjohnjiang
11*4418919fSjohnjiang #include <rte_random.h>
12*4418919fSjohnjiang #include <rte_cycles.h>
13*4418919fSjohnjiang #include <rte_memory.h>
14*4418919fSjohnjiang #include <rte_memzone.h>
15*4418919fSjohnjiang #include <rte_eal.h>
16*4418919fSjohnjiang #include <rte_lcore.h>
17*4418919fSjohnjiang #include <rte_common.h>
18*4418919fSjohnjiang #include <rte_string_fns.h>
19*4418919fSjohnjiang #include <rte_errno.h>
20*4418919fSjohnjiang #include <rte_malloc.h>
21*4418919fSjohnjiang #include "../../lib/librte_eal/common/malloc_elem.h"
22*4418919fSjohnjiang
23*4418919fSjohnjiang #include "test.h"
24*4418919fSjohnjiang
25*4418919fSjohnjiang /*
26*4418919fSjohnjiang * Memzone
27*4418919fSjohnjiang * =======
28*4418919fSjohnjiang *
29*4418919fSjohnjiang * - Search for three reserved zones or reserve them if they do not exist:
30*4418919fSjohnjiang *
31*4418919fSjohnjiang * - One is on any socket id.
32*4418919fSjohnjiang * - The second is on socket 0.
33*4418919fSjohnjiang * - The last one is on socket 1 (if socket 1 exists).
34*4418919fSjohnjiang *
35*4418919fSjohnjiang * - Check that the zones exist.
36*4418919fSjohnjiang *
37*4418919fSjohnjiang * - Check that the zones are cache-aligned.
38*4418919fSjohnjiang *
39*4418919fSjohnjiang * - Check that zones do not overlap.
40*4418919fSjohnjiang *
41*4418919fSjohnjiang * - Check that the zones are on the correct socket id.
42*4418919fSjohnjiang *
43*4418919fSjohnjiang * - Check that a lookup of the first zone returns the same pointer.
44*4418919fSjohnjiang *
45*4418919fSjohnjiang * - Check that it is not possible to create another zone with the
46*4418919fSjohnjiang * same name as an existing zone.
47*4418919fSjohnjiang *
48*4418919fSjohnjiang * - Check flags for specific huge page size reservation
49*4418919fSjohnjiang */
50*4418919fSjohnjiang
51*4418919fSjohnjiang #define TEST_MEMZONE_NAME(suffix) "MZ_TEST_" suffix
52*4418919fSjohnjiang
53*4418919fSjohnjiang /* Test if memory overlaps: return 1 if true, or 0 if false. */
54*4418919fSjohnjiang static int
is_memory_overlap(rte_iova_t ptr1,size_t len1,rte_iova_t ptr2,size_t len2)55*4418919fSjohnjiang is_memory_overlap(rte_iova_t ptr1, size_t len1, rte_iova_t ptr2, size_t len2)
56*4418919fSjohnjiang {
57*4418919fSjohnjiang if (ptr2 >= ptr1 && (ptr2 - ptr1) < len1)
58*4418919fSjohnjiang return 1;
59*4418919fSjohnjiang else if (ptr2 < ptr1 && (ptr1 - ptr2) < len2)
60*4418919fSjohnjiang return 1;
61*4418919fSjohnjiang return 0;
62*4418919fSjohnjiang }
63*4418919fSjohnjiang
64*4418919fSjohnjiang static int
test_memzone_invalid_alignment(void)65*4418919fSjohnjiang test_memzone_invalid_alignment(void)
66*4418919fSjohnjiang {
67*4418919fSjohnjiang const struct rte_memzone * mz;
68*4418919fSjohnjiang
69*4418919fSjohnjiang mz = rte_memzone_lookup(TEST_MEMZONE_NAME("invalid_alignment"));
70*4418919fSjohnjiang if (mz != NULL) {
71*4418919fSjohnjiang printf("Zone with invalid alignment has been reserved\n");
72*4418919fSjohnjiang return -1;
73*4418919fSjohnjiang }
74*4418919fSjohnjiang
75*4418919fSjohnjiang mz = rte_memzone_reserve_aligned(TEST_MEMZONE_NAME("invalid_alignment"),
76*4418919fSjohnjiang 100, SOCKET_ID_ANY, 0, 100);
77*4418919fSjohnjiang if (mz != NULL) {
78*4418919fSjohnjiang printf("Zone with invalid alignment has been reserved\n");
79*4418919fSjohnjiang return -1;
80*4418919fSjohnjiang }
81*4418919fSjohnjiang return 0;
82*4418919fSjohnjiang }
83*4418919fSjohnjiang
84*4418919fSjohnjiang static int
test_memzone_reserving_zone_size_bigger_than_the_maximum(void)85*4418919fSjohnjiang test_memzone_reserving_zone_size_bigger_than_the_maximum(void)
86*4418919fSjohnjiang {
87*4418919fSjohnjiang const struct rte_memzone * mz;
88*4418919fSjohnjiang
89*4418919fSjohnjiang mz = rte_memzone_lookup(
90*4418919fSjohnjiang TEST_MEMZONE_NAME("zone_size_bigger_than_the_maximum"));
91*4418919fSjohnjiang if (mz != NULL) {
92*4418919fSjohnjiang printf("zone_size_bigger_than_the_maximum has been reserved\n");
93*4418919fSjohnjiang return -1;
94*4418919fSjohnjiang }
95*4418919fSjohnjiang
96*4418919fSjohnjiang mz = rte_memzone_reserve(
97*4418919fSjohnjiang TEST_MEMZONE_NAME("zone_size_bigger_than_the_maximum"),
98*4418919fSjohnjiang (size_t)-1, SOCKET_ID_ANY, 0);
99*4418919fSjohnjiang if (mz != NULL) {
100*4418919fSjohnjiang printf("It is impossible to reserve such big a memzone\n");
101*4418919fSjohnjiang return -1;
102*4418919fSjohnjiang }
103*4418919fSjohnjiang
104*4418919fSjohnjiang return 0;
105*4418919fSjohnjiang }
106*4418919fSjohnjiang
107*4418919fSjohnjiang struct walk_arg {
108*4418919fSjohnjiang int hugepage_2MB_avail;
109*4418919fSjohnjiang int hugepage_1GB_avail;
110*4418919fSjohnjiang int hugepage_16MB_avail;
111*4418919fSjohnjiang int hugepage_16GB_avail;
112*4418919fSjohnjiang };
113*4418919fSjohnjiang static int
find_available_pagesz(const struct rte_memseg_list * msl,void * arg)114*4418919fSjohnjiang find_available_pagesz(const struct rte_memseg_list *msl, void *arg)
115*4418919fSjohnjiang {
116*4418919fSjohnjiang struct walk_arg *wa = arg;
117*4418919fSjohnjiang
118*4418919fSjohnjiang if (msl->external)
119*4418919fSjohnjiang return 0;
120*4418919fSjohnjiang
121*4418919fSjohnjiang if (msl->page_sz == RTE_PGSIZE_2M)
122*4418919fSjohnjiang wa->hugepage_2MB_avail = 1;
123*4418919fSjohnjiang if (msl->page_sz == RTE_PGSIZE_1G)
124*4418919fSjohnjiang wa->hugepage_1GB_avail = 1;
125*4418919fSjohnjiang if (msl->page_sz == RTE_PGSIZE_16M)
126*4418919fSjohnjiang wa->hugepage_16MB_avail = 1;
127*4418919fSjohnjiang if (msl->page_sz == RTE_PGSIZE_16G)
128*4418919fSjohnjiang wa->hugepage_16GB_avail = 1;
129*4418919fSjohnjiang
130*4418919fSjohnjiang return 0;
131*4418919fSjohnjiang }
132*4418919fSjohnjiang
133*4418919fSjohnjiang static int
test_memzone_reserve_flags(void)134*4418919fSjohnjiang test_memzone_reserve_flags(void)
135*4418919fSjohnjiang {
136*4418919fSjohnjiang const struct rte_memzone *mz;
137*4418919fSjohnjiang struct walk_arg wa;
138*4418919fSjohnjiang int hugepage_2MB_avail, hugepage_1GB_avail;
139*4418919fSjohnjiang int hugepage_16MB_avail, hugepage_16GB_avail;
140*4418919fSjohnjiang const size_t size = 100;
141*4418919fSjohnjiang
142*4418919fSjohnjiang memset(&wa, 0, sizeof(wa));
143*4418919fSjohnjiang
144*4418919fSjohnjiang rte_memseg_list_walk(find_available_pagesz, &wa);
145*4418919fSjohnjiang
146*4418919fSjohnjiang hugepage_2MB_avail = wa.hugepage_2MB_avail;
147*4418919fSjohnjiang hugepage_1GB_avail = wa.hugepage_1GB_avail;
148*4418919fSjohnjiang hugepage_16MB_avail = wa.hugepage_16MB_avail;
149*4418919fSjohnjiang hugepage_16GB_avail = wa.hugepage_16GB_avail;
150*4418919fSjohnjiang
151*4418919fSjohnjiang /* Display the availability of 2MB ,1GB, 16MB, 16GB pages */
152*4418919fSjohnjiang if (hugepage_2MB_avail)
153*4418919fSjohnjiang printf("2MB Huge pages available\n");
154*4418919fSjohnjiang if (hugepage_1GB_avail)
155*4418919fSjohnjiang printf("1GB Huge pages available\n");
156*4418919fSjohnjiang if (hugepage_16MB_avail)
157*4418919fSjohnjiang printf("16MB Huge pages available\n");
158*4418919fSjohnjiang if (hugepage_16GB_avail)
159*4418919fSjohnjiang printf("16GB Huge pages available\n");
160*4418919fSjohnjiang /*
161*4418919fSjohnjiang * If 2MB pages available, check that a small memzone is correctly
162*4418919fSjohnjiang * reserved from 2MB huge pages when requested by the RTE_MEMZONE_2MB flag.
163*4418919fSjohnjiang * Also check that RTE_MEMZONE_SIZE_HINT_ONLY flag only defaults to an
164*4418919fSjohnjiang * available page size (i.e 1GB ) when 2MB pages are unavailable.
165*4418919fSjohnjiang */
166*4418919fSjohnjiang if (hugepage_2MB_avail) {
167*4418919fSjohnjiang mz = rte_memzone_reserve(TEST_MEMZONE_NAME("flag_zone_2M"),
168*4418919fSjohnjiang size, SOCKET_ID_ANY, RTE_MEMZONE_2MB);
169*4418919fSjohnjiang if (mz == NULL) {
170*4418919fSjohnjiang printf("MEMZONE FLAG 2MB\n");
171*4418919fSjohnjiang return -1;
172*4418919fSjohnjiang }
173*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_2M) {
174*4418919fSjohnjiang printf("hugepage_sz not equal 2M\n");
175*4418919fSjohnjiang return -1;
176*4418919fSjohnjiang }
177*4418919fSjohnjiang if (rte_memzone_free(mz)) {
178*4418919fSjohnjiang printf("Fail memzone free\n");
179*4418919fSjohnjiang return -1;
180*4418919fSjohnjiang }
181*4418919fSjohnjiang
182*4418919fSjohnjiang mz = rte_memzone_reserve(TEST_MEMZONE_NAME("flag_zone_2M_HINT"),
183*4418919fSjohnjiang size, SOCKET_ID_ANY,
184*4418919fSjohnjiang RTE_MEMZONE_2MB|RTE_MEMZONE_SIZE_HINT_ONLY);
185*4418919fSjohnjiang if (mz == NULL) {
186*4418919fSjohnjiang printf("MEMZONE FLAG 2MB\n");
187*4418919fSjohnjiang return -1;
188*4418919fSjohnjiang }
189*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_2M) {
190*4418919fSjohnjiang printf("hugepage_sz not equal 2M\n");
191*4418919fSjohnjiang return -1;
192*4418919fSjohnjiang }
193*4418919fSjohnjiang if (rte_memzone_free(mz)) {
194*4418919fSjohnjiang printf("Fail memzone free\n");
195*4418919fSjohnjiang return -1;
196*4418919fSjohnjiang }
197*4418919fSjohnjiang
198*4418919fSjohnjiang /* Check if 1GB huge pages are unavailable, that function fails unless
199*4418919fSjohnjiang * HINT flag is indicated
200*4418919fSjohnjiang */
201*4418919fSjohnjiang if (!hugepage_1GB_avail) {
202*4418919fSjohnjiang mz = rte_memzone_reserve(
203*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_1G_HINT"),
204*4418919fSjohnjiang size, SOCKET_ID_ANY,
205*4418919fSjohnjiang RTE_MEMZONE_1GB|RTE_MEMZONE_SIZE_HINT_ONLY);
206*4418919fSjohnjiang if (mz == NULL) {
207*4418919fSjohnjiang printf("MEMZONE FLAG 1GB & HINT\n");
208*4418919fSjohnjiang return -1;
209*4418919fSjohnjiang }
210*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_2M) {
211*4418919fSjohnjiang printf("hugepage_sz not equal 2M\n");
212*4418919fSjohnjiang return -1;
213*4418919fSjohnjiang }
214*4418919fSjohnjiang if (rte_memzone_free(mz)) {
215*4418919fSjohnjiang printf("Fail memzone free\n");
216*4418919fSjohnjiang return -1;
217*4418919fSjohnjiang }
218*4418919fSjohnjiang
219*4418919fSjohnjiang mz = rte_memzone_reserve(
220*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_1G"), size,
221*4418919fSjohnjiang SOCKET_ID_ANY, RTE_MEMZONE_1GB);
222*4418919fSjohnjiang if (mz != NULL) {
223*4418919fSjohnjiang printf("MEMZONE FLAG 1GB\n");
224*4418919fSjohnjiang return -1;
225*4418919fSjohnjiang }
226*4418919fSjohnjiang }
227*4418919fSjohnjiang }
228*4418919fSjohnjiang
229*4418919fSjohnjiang /*As with 2MB tests above for 1GB huge page requests*/
230*4418919fSjohnjiang if (hugepage_1GB_avail) {
231*4418919fSjohnjiang mz = rte_memzone_reserve(TEST_MEMZONE_NAME("flag_zone_1G"),
232*4418919fSjohnjiang size, SOCKET_ID_ANY, RTE_MEMZONE_1GB);
233*4418919fSjohnjiang if (mz == NULL) {
234*4418919fSjohnjiang printf("MEMZONE FLAG 1GB\n");
235*4418919fSjohnjiang return -1;
236*4418919fSjohnjiang }
237*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_1G) {
238*4418919fSjohnjiang printf("hugepage_sz not equal 1G\n");
239*4418919fSjohnjiang return -1;
240*4418919fSjohnjiang }
241*4418919fSjohnjiang if (rte_memzone_free(mz)) {
242*4418919fSjohnjiang printf("Fail memzone free\n");
243*4418919fSjohnjiang return -1;
244*4418919fSjohnjiang }
245*4418919fSjohnjiang
246*4418919fSjohnjiang mz = rte_memzone_reserve(TEST_MEMZONE_NAME("flag_zone_1G_HINT"),
247*4418919fSjohnjiang size, SOCKET_ID_ANY,
248*4418919fSjohnjiang RTE_MEMZONE_1GB|RTE_MEMZONE_SIZE_HINT_ONLY);
249*4418919fSjohnjiang if (mz == NULL) {
250*4418919fSjohnjiang printf("MEMZONE FLAG 1GB\n");
251*4418919fSjohnjiang return -1;
252*4418919fSjohnjiang }
253*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_1G) {
254*4418919fSjohnjiang printf("hugepage_sz not equal 1G\n");
255*4418919fSjohnjiang return -1;
256*4418919fSjohnjiang }
257*4418919fSjohnjiang if (rte_memzone_free(mz)) {
258*4418919fSjohnjiang printf("Fail memzone free\n");
259*4418919fSjohnjiang return -1;
260*4418919fSjohnjiang }
261*4418919fSjohnjiang
262*4418919fSjohnjiang /* Check if 1GB huge pages are unavailable, that function fails unless
263*4418919fSjohnjiang * HINT flag is indicated
264*4418919fSjohnjiang */
265*4418919fSjohnjiang if (!hugepage_2MB_avail) {
266*4418919fSjohnjiang mz = rte_memzone_reserve(
267*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_2M_HINT"),
268*4418919fSjohnjiang size, SOCKET_ID_ANY,
269*4418919fSjohnjiang RTE_MEMZONE_2MB|RTE_MEMZONE_SIZE_HINT_ONLY);
270*4418919fSjohnjiang if (mz == NULL){
271*4418919fSjohnjiang printf("MEMZONE FLAG 2MB & HINT\n");
272*4418919fSjohnjiang return -1;
273*4418919fSjohnjiang }
274*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_1G) {
275*4418919fSjohnjiang printf("hugepage_sz not equal 1G\n");
276*4418919fSjohnjiang return -1;
277*4418919fSjohnjiang }
278*4418919fSjohnjiang if (rte_memzone_free(mz)) {
279*4418919fSjohnjiang printf("Fail memzone free\n");
280*4418919fSjohnjiang return -1;
281*4418919fSjohnjiang }
282*4418919fSjohnjiang mz = rte_memzone_reserve(
283*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_2M"), size,
284*4418919fSjohnjiang SOCKET_ID_ANY, RTE_MEMZONE_2MB);
285*4418919fSjohnjiang if (mz != NULL) {
286*4418919fSjohnjiang printf("MEMZONE FLAG 2MB\n");
287*4418919fSjohnjiang return -1;
288*4418919fSjohnjiang }
289*4418919fSjohnjiang }
290*4418919fSjohnjiang
291*4418919fSjohnjiang if (hugepage_2MB_avail && hugepage_1GB_avail) {
292*4418919fSjohnjiang mz = rte_memzone_reserve(
293*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_2M_HINT"),
294*4418919fSjohnjiang size, SOCKET_ID_ANY,
295*4418919fSjohnjiang RTE_MEMZONE_2MB|RTE_MEMZONE_1GB);
296*4418919fSjohnjiang if (mz == NULL) {
297*4418919fSjohnjiang printf("BOTH SIZES SET\n");
298*4418919fSjohnjiang return -1;
299*4418919fSjohnjiang }
300*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_1G &&
301*4418919fSjohnjiang mz->hugepage_sz != RTE_PGSIZE_2M) {
302*4418919fSjohnjiang printf("Wrong size when both sizes set\n");
303*4418919fSjohnjiang return -1;
304*4418919fSjohnjiang }
305*4418919fSjohnjiang if (rte_memzone_free(mz)) {
306*4418919fSjohnjiang printf("Fail memzone free\n");
307*4418919fSjohnjiang return -1;
308*4418919fSjohnjiang }
309*4418919fSjohnjiang }
310*4418919fSjohnjiang }
311*4418919fSjohnjiang /*
312*4418919fSjohnjiang * This option is for IBM Power. If 16MB pages available, check
313*4418919fSjohnjiang * that a small memzone is correctly reserved from 16MB huge pages
314*4418919fSjohnjiang * when requested by the RTE_MEMZONE_16MB flag. Also check that
315*4418919fSjohnjiang * RTE_MEMZONE_SIZE_HINT_ONLY flag only defaults to an available
316*4418919fSjohnjiang * page size (i.e 16GB ) when 16MB pages are unavailable.
317*4418919fSjohnjiang */
318*4418919fSjohnjiang if (hugepage_16MB_avail) {
319*4418919fSjohnjiang mz = rte_memzone_reserve(TEST_MEMZONE_NAME("flag_zone_16M"),
320*4418919fSjohnjiang size, SOCKET_ID_ANY, RTE_MEMZONE_16MB);
321*4418919fSjohnjiang if (mz == NULL) {
322*4418919fSjohnjiang printf("MEMZONE FLAG 16MB\n");
323*4418919fSjohnjiang return -1;
324*4418919fSjohnjiang }
325*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_16M) {
326*4418919fSjohnjiang printf("hugepage_sz not equal 16M\n");
327*4418919fSjohnjiang return -1;
328*4418919fSjohnjiang }
329*4418919fSjohnjiang if (rte_memzone_free(mz)) {
330*4418919fSjohnjiang printf("Fail memzone free\n");
331*4418919fSjohnjiang return -1;
332*4418919fSjohnjiang }
333*4418919fSjohnjiang
334*4418919fSjohnjiang mz = rte_memzone_reserve(
335*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_16M_HINT"), size,
336*4418919fSjohnjiang SOCKET_ID_ANY,
337*4418919fSjohnjiang RTE_MEMZONE_16MB|RTE_MEMZONE_SIZE_HINT_ONLY);
338*4418919fSjohnjiang if (mz == NULL) {
339*4418919fSjohnjiang printf("MEMZONE FLAG 16MB\n");
340*4418919fSjohnjiang return -1;
341*4418919fSjohnjiang }
342*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_16M) {
343*4418919fSjohnjiang printf("hugepage_sz not equal 16M\n");
344*4418919fSjohnjiang return -1;
345*4418919fSjohnjiang }
346*4418919fSjohnjiang if (rte_memzone_free(mz)) {
347*4418919fSjohnjiang printf("Fail memzone free\n");
348*4418919fSjohnjiang return -1;
349*4418919fSjohnjiang }
350*4418919fSjohnjiang
351*4418919fSjohnjiang /* Check if 1GB huge pages are unavailable, that function fails
352*4418919fSjohnjiang * unless HINT flag is indicated
353*4418919fSjohnjiang */
354*4418919fSjohnjiang if (!hugepage_16GB_avail) {
355*4418919fSjohnjiang mz = rte_memzone_reserve(
356*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_16G_HINT"),
357*4418919fSjohnjiang size, SOCKET_ID_ANY,
358*4418919fSjohnjiang RTE_MEMZONE_16GB |
359*4418919fSjohnjiang RTE_MEMZONE_SIZE_HINT_ONLY);
360*4418919fSjohnjiang if (mz == NULL) {
361*4418919fSjohnjiang printf("MEMZONE FLAG 16GB & HINT\n");
362*4418919fSjohnjiang return -1;
363*4418919fSjohnjiang }
364*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_16M) {
365*4418919fSjohnjiang printf("hugepage_sz not equal 16M\n");
366*4418919fSjohnjiang return -1;
367*4418919fSjohnjiang }
368*4418919fSjohnjiang if (rte_memzone_free(mz)) {
369*4418919fSjohnjiang printf("Fail memzone free\n");
370*4418919fSjohnjiang return -1;
371*4418919fSjohnjiang }
372*4418919fSjohnjiang
373*4418919fSjohnjiang mz = rte_memzone_reserve(
374*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_16G"),
375*4418919fSjohnjiang size,
376*4418919fSjohnjiang SOCKET_ID_ANY, RTE_MEMZONE_16GB);
377*4418919fSjohnjiang if (mz != NULL) {
378*4418919fSjohnjiang printf("MEMZONE FLAG 16GB\n");
379*4418919fSjohnjiang return -1;
380*4418919fSjohnjiang }
381*4418919fSjohnjiang }
382*4418919fSjohnjiang }
383*4418919fSjohnjiang /*As with 16MB tests above for 16GB huge page requests*/
384*4418919fSjohnjiang if (hugepage_16GB_avail) {
385*4418919fSjohnjiang mz = rte_memzone_reserve(TEST_MEMZONE_NAME("flag_zone_16G"),
386*4418919fSjohnjiang size, SOCKET_ID_ANY, RTE_MEMZONE_16GB);
387*4418919fSjohnjiang if (mz == NULL) {
388*4418919fSjohnjiang printf("MEMZONE FLAG 16GB\n");
389*4418919fSjohnjiang return -1;
390*4418919fSjohnjiang }
391*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_16G) {
392*4418919fSjohnjiang printf("hugepage_sz not equal 16G\n");
393*4418919fSjohnjiang return -1;
394*4418919fSjohnjiang }
395*4418919fSjohnjiang if (rte_memzone_free(mz)) {
396*4418919fSjohnjiang printf("Fail memzone free\n");
397*4418919fSjohnjiang return -1;
398*4418919fSjohnjiang }
399*4418919fSjohnjiang
400*4418919fSjohnjiang mz = rte_memzone_reserve(
401*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_16G_HINT"), size,
402*4418919fSjohnjiang SOCKET_ID_ANY,
403*4418919fSjohnjiang RTE_MEMZONE_16GB|RTE_MEMZONE_SIZE_HINT_ONLY);
404*4418919fSjohnjiang if (mz == NULL) {
405*4418919fSjohnjiang printf("MEMZONE FLAG 16GB\n");
406*4418919fSjohnjiang return -1;
407*4418919fSjohnjiang }
408*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_16G) {
409*4418919fSjohnjiang printf("hugepage_sz not equal 16G\n");
410*4418919fSjohnjiang return -1;
411*4418919fSjohnjiang }
412*4418919fSjohnjiang if (rte_memzone_free(mz)) {
413*4418919fSjohnjiang printf("Fail memzone free\n");
414*4418919fSjohnjiang return -1;
415*4418919fSjohnjiang }
416*4418919fSjohnjiang
417*4418919fSjohnjiang /* Check if 1GB huge pages are unavailable, that function fails
418*4418919fSjohnjiang * unless HINT flag is indicated
419*4418919fSjohnjiang */
420*4418919fSjohnjiang if (!hugepage_16MB_avail) {
421*4418919fSjohnjiang mz = rte_memzone_reserve(
422*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_16M_HINT"),
423*4418919fSjohnjiang size, SOCKET_ID_ANY,
424*4418919fSjohnjiang RTE_MEMZONE_16MB |
425*4418919fSjohnjiang RTE_MEMZONE_SIZE_HINT_ONLY);
426*4418919fSjohnjiang if (mz == NULL) {
427*4418919fSjohnjiang printf("MEMZONE FLAG 16MB & HINT\n");
428*4418919fSjohnjiang return -1;
429*4418919fSjohnjiang }
430*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_16G) {
431*4418919fSjohnjiang printf("hugepage_sz not equal 16G\n");
432*4418919fSjohnjiang return -1;
433*4418919fSjohnjiang }
434*4418919fSjohnjiang if (rte_memzone_free(mz)) {
435*4418919fSjohnjiang printf("Fail memzone free\n");
436*4418919fSjohnjiang return -1;
437*4418919fSjohnjiang }
438*4418919fSjohnjiang mz = rte_memzone_reserve(
439*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_16M"),
440*4418919fSjohnjiang size, SOCKET_ID_ANY, RTE_MEMZONE_16MB);
441*4418919fSjohnjiang if (mz != NULL) {
442*4418919fSjohnjiang printf("MEMZONE FLAG 16MB\n");
443*4418919fSjohnjiang return -1;
444*4418919fSjohnjiang }
445*4418919fSjohnjiang }
446*4418919fSjohnjiang
447*4418919fSjohnjiang if (hugepage_16MB_avail && hugepage_16GB_avail) {
448*4418919fSjohnjiang mz = rte_memzone_reserve(
449*4418919fSjohnjiang TEST_MEMZONE_NAME("flag_zone_16M_HINT"),
450*4418919fSjohnjiang size, SOCKET_ID_ANY,
451*4418919fSjohnjiang RTE_MEMZONE_16MB|RTE_MEMZONE_16GB);
452*4418919fSjohnjiang if (mz == NULL) {
453*4418919fSjohnjiang printf("BOTH SIZES SET\n");
454*4418919fSjohnjiang return -1;
455*4418919fSjohnjiang }
456*4418919fSjohnjiang if (mz->hugepage_sz != RTE_PGSIZE_16G &&
457*4418919fSjohnjiang mz->hugepage_sz != RTE_PGSIZE_16M) {
458*4418919fSjohnjiang printf("Wrong size when both sizes set\n");
459*4418919fSjohnjiang return -1;
460*4418919fSjohnjiang }
461*4418919fSjohnjiang if (rte_memzone_free(mz)) {
462*4418919fSjohnjiang printf("Fail memzone free\n");
463*4418919fSjohnjiang return -1;
464*4418919fSjohnjiang }
465*4418919fSjohnjiang }
466*4418919fSjohnjiang }
467*4418919fSjohnjiang return 0;
468*4418919fSjohnjiang }
469*4418919fSjohnjiang
470*4418919fSjohnjiang
471*4418919fSjohnjiang /* Find the heap with the greatest free block size */
472*4418919fSjohnjiang static size_t
find_max_block_free_size(unsigned int align,unsigned int socket_id)473*4418919fSjohnjiang find_max_block_free_size(unsigned int align, unsigned int socket_id)
474*4418919fSjohnjiang {
475*4418919fSjohnjiang struct rte_malloc_socket_stats stats;
476*4418919fSjohnjiang size_t len, overhead;
477*4418919fSjohnjiang
478*4418919fSjohnjiang if (rte_malloc_get_socket_stats(socket_id, &stats) < 0)
479*4418919fSjohnjiang return 0;
480*4418919fSjohnjiang
481*4418919fSjohnjiang len = stats.greatest_free_size;
482*4418919fSjohnjiang overhead = MALLOC_ELEM_OVERHEAD;
483*4418919fSjohnjiang
484*4418919fSjohnjiang if (len == 0)
485*4418919fSjohnjiang return 0;
486*4418919fSjohnjiang
487*4418919fSjohnjiang align = RTE_CACHE_LINE_ROUNDUP(align);
488*4418919fSjohnjiang overhead += align;
489*4418919fSjohnjiang
490*4418919fSjohnjiang if (len < overhead)
491*4418919fSjohnjiang return 0;
492*4418919fSjohnjiang
493*4418919fSjohnjiang return len - overhead;
494*4418919fSjohnjiang }
495*4418919fSjohnjiang
496*4418919fSjohnjiang static int
test_memzone_reserve_max(void)497*4418919fSjohnjiang test_memzone_reserve_max(void)
498*4418919fSjohnjiang {
499*4418919fSjohnjiang unsigned int i;
500*4418919fSjohnjiang
501*4418919fSjohnjiang for (i = 0; i < rte_socket_count(); i++) {
502*4418919fSjohnjiang const struct rte_memzone *mz;
503*4418919fSjohnjiang size_t maxlen;
504*4418919fSjohnjiang int socket;
505*4418919fSjohnjiang
506*4418919fSjohnjiang socket = rte_socket_id_by_idx(i);
507*4418919fSjohnjiang maxlen = find_max_block_free_size(0, socket);
508*4418919fSjohnjiang
509*4418919fSjohnjiang if (maxlen == 0) {
510*4418919fSjohnjiang printf("There is no space left!\n");
511*4418919fSjohnjiang return 0;
512*4418919fSjohnjiang }
513*4418919fSjohnjiang
514*4418919fSjohnjiang mz = rte_memzone_reserve(TEST_MEMZONE_NAME("max_zone"), 0,
515*4418919fSjohnjiang socket, 0);
516*4418919fSjohnjiang if (mz == NULL) {
517*4418919fSjohnjiang printf("Failed to reserve a big chunk of memory - %s\n",
518*4418919fSjohnjiang rte_strerror(rte_errno));
519*4418919fSjohnjiang rte_dump_physmem_layout(stdout);
520*4418919fSjohnjiang rte_memzone_dump(stdout);
521*4418919fSjohnjiang return -1;
522*4418919fSjohnjiang }
523*4418919fSjohnjiang
524*4418919fSjohnjiang if (mz->len != maxlen) {
525*4418919fSjohnjiang printf("Memzone reserve with 0 size did not return bigest block\n");
526*4418919fSjohnjiang printf("Expected size = %zu, actual size = %zu\n",
527*4418919fSjohnjiang maxlen, mz->len);
528*4418919fSjohnjiang rte_dump_physmem_layout(stdout);
529*4418919fSjohnjiang rte_memzone_dump(stdout);
530*4418919fSjohnjiang return -1;
531*4418919fSjohnjiang }
532*4418919fSjohnjiang
533*4418919fSjohnjiang if (rte_memzone_free(mz)) {
534*4418919fSjohnjiang printf("Fail memzone free\n");
535*4418919fSjohnjiang return -1;
536*4418919fSjohnjiang }
537*4418919fSjohnjiang }
538*4418919fSjohnjiang
539*4418919fSjohnjiang return 0;
540*4418919fSjohnjiang }
541*4418919fSjohnjiang
542*4418919fSjohnjiang static int
test_memzone_reserve_max_aligned(void)543*4418919fSjohnjiang test_memzone_reserve_max_aligned(void)
544*4418919fSjohnjiang {
545*4418919fSjohnjiang unsigned int i;
546*4418919fSjohnjiang
547*4418919fSjohnjiang for (i = 0; i < rte_socket_count(); i++) {
548*4418919fSjohnjiang const struct rte_memzone *mz;
549*4418919fSjohnjiang size_t maxlen, minlen = 0;
550*4418919fSjohnjiang int socket;
551*4418919fSjohnjiang
552*4418919fSjohnjiang socket = rte_socket_id_by_idx(i);
553*4418919fSjohnjiang
554*4418919fSjohnjiang /* random alignment */
555*4418919fSjohnjiang rte_srand((unsigned int)rte_rdtsc());
556*4418919fSjohnjiang const unsigned int align = 1 << ((rte_rand() % 8) + 5); /* from 128 up to 4k alignment */
557*4418919fSjohnjiang
558*4418919fSjohnjiang /* memzone size may be between size and size - align */
559*4418919fSjohnjiang minlen = find_max_block_free_size(align, socket);
560*4418919fSjohnjiang maxlen = find_max_block_free_size(0, socket);
561*4418919fSjohnjiang
562*4418919fSjohnjiang if (minlen == 0 || maxlen == 0) {
563*4418919fSjohnjiang printf("There is no space left for biggest %u-aligned memzone!\n",
564*4418919fSjohnjiang align);
565*4418919fSjohnjiang return 0;
566*4418919fSjohnjiang }
567*4418919fSjohnjiang
568*4418919fSjohnjiang mz = rte_memzone_reserve_aligned(
569*4418919fSjohnjiang TEST_MEMZONE_NAME("max_zone_aligned"),
570*4418919fSjohnjiang 0, socket, 0, align);
571*4418919fSjohnjiang if (mz == NULL) {
572*4418919fSjohnjiang printf("Failed to reserve a big chunk of memory - %s\n",
573*4418919fSjohnjiang rte_strerror(rte_errno));
574*4418919fSjohnjiang rte_dump_physmem_layout(stdout);
575*4418919fSjohnjiang rte_memzone_dump(stdout);
576*4418919fSjohnjiang return -1;
577*4418919fSjohnjiang }
578*4418919fSjohnjiang if (mz->addr != RTE_PTR_ALIGN(mz->addr, align)) {
579*4418919fSjohnjiang printf("Memzone reserve with 0 size and alignment %u did not return aligned block\n",
580*4418919fSjohnjiang align);
581*4418919fSjohnjiang rte_dump_physmem_layout(stdout);
582*4418919fSjohnjiang rte_memzone_dump(stdout);
583*4418919fSjohnjiang return -1;
584*4418919fSjohnjiang }
585*4418919fSjohnjiang
586*4418919fSjohnjiang if (mz->len < minlen || mz->len > maxlen) {
587*4418919fSjohnjiang printf("Memzone reserve with 0 size and alignment %u did not return"
588*4418919fSjohnjiang " bigest block\n", align);
589*4418919fSjohnjiang printf("Expected size = %zu-%zu, actual size = %zu\n",
590*4418919fSjohnjiang minlen, maxlen, mz->len);
591*4418919fSjohnjiang rte_dump_physmem_layout(stdout);
592*4418919fSjohnjiang rte_memzone_dump(stdout);
593*4418919fSjohnjiang return -1;
594*4418919fSjohnjiang }
595*4418919fSjohnjiang
596*4418919fSjohnjiang if (rte_memzone_free(mz)) {
597*4418919fSjohnjiang printf("Fail memzone free\n");
598*4418919fSjohnjiang return -1;
599*4418919fSjohnjiang }
600*4418919fSjohnjiang }
601*4418919fSjohnjiang return 0;
602*4418919fSjohnjiang }
603*4418919fSjohnjiang
604*4418919fSjohnjiang static int
test_memzone_aligned(void)605*4418919fSjohnjiang test_memzone_aligned(void)
606*4418919fSjohnjiang {
607*4418919fSjohnjiang const struct rte_memzone *memzone_aligned_32;
608*4418919fSjohnjiang const struct rte_memzone *memzone_aligned_128;
609*4418919fSjohnjiang const struct rte_memzone *memzone_aligned_256;
610*4418919fSjohnjiang const struct rte_memzone *memzone_aligned_512;
611*4418919fSjohnjiang const struct rte_memzone *memzone_aligned_1024;
612*4418919fSjohnjiang
613*4418919fSjohnjiang /* memzone that should automatically be adjusted to align on 64 bytes */
614*4418919fSjohnjiang memzone_aligned_32 = rte_memzone_reserve_aligned(
615*4418919fSjohnjiang TEST_MEMZONE_NAME("aligned_32"), 100, SOCKET_ID_ANY, 0,
616*4418919fSjohnjiang 32);
617*4418919fSjohnjiang
618*4418919fSjohnjiang /* memzone that is supposed to be aligned on a 128 byte boundary */
619*4418919fSjohnjiang memzone_aligned_128 = rte_memzone_reserve_aligned(
620*4418919fSjohnjiang TEST_MEMZONE_NAME("aligned_128"), 100, SOCKET_ID_ANY, 0,
621*4418919fSjohnjiang 128);
622*4418919fSjohnjiang
623*4418919fSjohnjiang /* memzone that is supposed to be aligned on a 256 byte boundary */
624*4418919fSjohnjiang memzone_aligned_256 = rte_memzone_reserve_aligned(
625*4418919fSjohnjiang TEST_MEMZONE_NAME("aligned_256"), 100, SOCKET_ID_ANY, 0,
626*4418919fSjohnjiang 256);
627*4418919fSjohnjiang
628*4418919fSjohnjiang /* memzone that is supposed to be aligned on a 512 byte boundary */
629*4418919fSjohnjiang memzone_aligned_512 = rte_memzone_reserve_aligned(
630*4418919fSjohnjiang TEST_MEMZONE_NAME("aligned_512"), 100, SOCKET_ID_ANY, 0,
631*4418919fSjohnjiang 512);
632*4418919fSjohnjiang
633*4418919fSjohnjiang /* memzone that is supposed to be aligned on a 1024 byte boundary */
634*4418919fSjohnjiang memzone_aligned_1024 = rte_memzone_reserve_aligned(
635*4418919fSjohnjiang TEST_MEMZONE_NAME("aligned_1024"), 100, SOCKET_ID_ANY,
636*4418919fSjohnjiang 0, 1024);
637*4418919fSjohnjiang
638*4418919fSjohnjiang printf("check alignments and lengths\n");
639*4418919fSjohnjiang if (memzone_aligned_32 == NULL) {
640*4418919fSjohnjiang printf("Unable to reserve 64-byte aligned memzone!\n");
641*4418919fSjohnjiang return -1;
642*4418919fSjohnjiang }
643*4418919fSjohnjiang if ((memzone_aligned_32->iova & RTE_CACHE_LINE_MASK) != 0)
644*4418919fSjohnjiang return -1;
645*4418919fSjohnjiang if (((uintptr_t) memzone_aligned_32->addr & RTE_CACHE_LINE_MASK) != 0)
646*4418919fSjohnjiang return -1;
647*4418919fSjohnjiang if ((memzone_aligned_32->len & RTE_CACHE_LINE_MASK) != 0)
648*4418919fSjohnjiang return -1;
649*4418919fSjohnjiang
650*4418919fSjohnjiang if (memzone_aligned_128 == NULL) {
651*4418919fSjohnjiang printf("Unable to reserve 128-byte aligned memzone!\n");
652*4418919fSjohnjiang return -1;
653*4418919fSjohnjiang }
654*4418919fSjohnjiang if ((memzone_aligned_128->iova & 127) != 0)
655*4418919fSjohnjiang return -1;
656*4418919fSjohnjiang if (((uintptr_t) memzone_aligned_128->addr & 127) != 0)
657*4418919fSjohnjiang return -1;
658*4418919fSjohnjiang if ((memzone_aligned_128->len & RTE_CACHE_LINE_MASK) != 0)
659*4418919fSjohnjiang return -1;
660*4418919fSjohnjiang
661*4418919fSjohnjiang if (memzone_aligned_256 == NULL) {
662*4418919fSjohnjiang printf("Unable to reserve 256-byte aligned memzone!\n");
663*4418919fSjohnjiang return -1;
664*4418919fSjohnjiang }
665*4418919fSjohnjiang if ((memzone_aligned_256->iova & 255) != 0)
666*4418919fSjohnjiang return -1;
667*4418919fSjohnjiang if (((uintptr_t) memzone_aligned_256->addr & 255) != 0)
668*4418919fSjohnjiang return -1;
669*4418919fSjohnjiang if ((memzone_aligned_256->len & RTE_CACHE_LINE_MASK) != 0)
670*4418919fSjohnjiang return -1;
671*4418919fSjohnjiang
672*4418919fSjohnjiang if (memzone_aligned_512 == NULL) {
673*4418919fSjohnjiang printf("Unable to reserve 512-byte aligned memzone!\n");
674*4418919fSjohnjiang return -1;
675*4418919fSjohnjiang }
676*4418919fSjohnjiang if ((memzone_aligned_512->iova & 511) != 0)
677*4418919fSjohnjiang return -1;
678*4418919fSjohnjiang if (((uintptr_t) memzone_aligned_512->addr & 511) != 0)
679*4418919fSjohnjiang return -1;
680*4418919fSjohnjiang if ((memzone_aligned_512->len & RTE_CACHE_LINE_MASK) != 0)
681*4418919fSjohnjiang return -1;
682*4418919fSjohnjiang
683*4418919fSjohnjiang if (memzone_aligned_1024 == NULL) {
684*4418919fSjohnjiang printf("Unable to reserve 1024-byte aligned memzone!\n");
685*4418919fSjohnjiang return -1;
686*4418919fSjohnjiang }
687*4418919fSjohnjiang if ((memzone_aligned_1024->iova & 1023) != 0)
688*4418919fSjohnjiang return -1;
689*4418919fSjohnjiang if (((uintptr_t) memzone_aligned_1024->addr & 1023) != 0)
690*4418919fSjohnjiang return -1;
691*4418919fSjohnjiang if ((memzone_aligned_1024->len & RTE_CACHE_LINE_MASK) != 0)
692*4418919fSjohnjiang return -1;
693*4418919fSjohnjiang
694*4418919fSjohnjiang /* check that zones don't overlap */
695*4418919fSjohnjiang printf("check overlapping\n");
696*4418919fSjohnjiang if (is_memory_overlap(memzone_aligned_32->iova, memzone_aligned_32->len,
697*4418919fSjohnjiang memzone_aligned_128->iova, memzone_aligned_128->len))
698*4418919fSjohnjiang return -1;
699*4418919fSjohnjiang if (is_memory_overlap(memzone_aligned_32->iova, memzone_aligned_32->len,
700*4418919fSjohnjiang memzone_aligned_256->iova, memzone_aligned_256->len))
701*4418919fSjohnjiang return -1;
702*4418919fSjohnjiang if (is_memory_overlap(memzone_aligned_32->iova, memzone_aligned_32->len,
703*4418919fSjohnjiang memzone_aligned_512->iova, memzone_aligned_512->len))
704*4418919fSjohnjiang return -1;
705*4418919fSjohnjiang if (is_memory_overlap(memzone_aligned_32->iova, memzone_aligned_32->len,
706*4418919fSjohnjiang memzone_aligned_1024->iova, memzone_aligned_1024->len))
707*4418919fSjohnjiang return -1;
708*4418919fSjohnjiang if (is_memory_overlap(memzone_aligned_128->iova, memzone_aligned_128->len,
709*4418919fSjohnjiang memzone_aligned_256->iova, memzone_aligned_256->len))
710*4418919fSjohnjiang return -1;
711*4418919fSjohnjiang if (is_memory_overlap(memzone_aligned_128->iova, memzone_aligned_128->len,
712*4418919fSjohnjiang memzone_aligned_512->iova, memzone_aligned_512->len))
713*4418919fSjohnjiang return -1;
714*4418919fSjohnjiang if (is_memory_overlap(memzone_aligned_128->iova, memzone_aligned_128->len,
715*4418919fSjohnjiang memzone_aligned_1024->iova, memzone_aligned_1024->len))
716*4418919fSjohnjiang return -1;
717*4418919fSjohnjiang if (is_memory_overlap(memzone_aligned_256->iova, memzone_aligned_256->len,
718*4418919fSjohnjiang memzone_aligned_512->iova, memzone_aligned_512->len))
719*4418919fSjohnjiang return -1;
720*4418919fSjohnjiang if (is_memory_overlap(memzone_aligned_256->iova, memzone_aligned_256->len,
721*4418919fSjohnjiang memzone_aligned_1024->iova, memzone_aligned_1024->len))
722*4418919fSjohnjiang return -1;
723*4418919fSjohnjiang if (is_memory_overlap(memzone_aligned_512->iova, memzone_aligned_512->len,
724*4418919fSjohnjiang memzone_aligned_1024->iova, memzone_aligned_1024->len))
725*4418919fSjohnjiang return -1;
726*4418919fSjohnjiang
727*4418919fSjohnjiang /* free all used zones */
728*4418919fSjohnjiang if (rte_memzone_free(memzone_aligned_32)) {
729*4418919fSjohnjiang printf("Fail memzone free\n");
730*4418919fSjohnjiang return -1;
731*4418919fSjohnjiang }
732*4418919fSjohnjiang if (rte_memzone_free(memzone_aligned_128)) {
733*4418919fSjohnjiang printf("Fail memzone free\n");
734*4418919fSjohnjiang return -1;
735*4418919fSjohnjiang }
736*4418919fSjohnjiang if (rte_memzone_free(memzone_aligned_256)) {
737*4418919fSjohnjiang printf("Fail memzone free\n");
738*4418919fSjohnjiang return -1;
739*4418919fSjohnjiang }
740*4418919fSjohnjiang if (rte_memzone_free(memzone_aligned_512)) {
741*4418919fSjohnjiang printf("Fail memzone free\n");
742*4418919fSjohnjiang return -1;
743*4418919fSjohnjiang }
744*4418919fSjohnjiang if (rte_memzone_free(memzone_aligned_1024)) {
745*4418919fSjohnjiang printf("Fail memzone free\n");
746*4418919fSjohnjiang return -1;
747*4418919fSjohnjiang }
748*4418919fSjohnjiang return 0;
749*4418919fSjohnjiang }
750*4418919fSjohnjiang
751*4418919fSjohnjiang static int
check_memzone_bounded(const char * name,uint32_t len,uint32_t align,uint32_t bound)752*4418919fSjohnjiang check_memzone_bounded(const char *name, uint32_t len, uint32_t align,
753*4418919fSjohnjiang uint32_t bound)
754*4418919fSjohnjiang {
755*4418919fSjohnjiang const struct rte_memzone *mz;
756*4418919fSjohnjiang rte_iova_t bmask;
757*4418919fSjohnjiang
758*4418919fSjohnjiang bmask = ~((rte_iova_t)bound - 1);
759*4418919fSjohnjiang
760*4418919fSjohnjiang if ((mz = rte_memzone_reserve_bounded(name, len, SOCKET_ID_ANY, 0,
761*4418919fSjohnjiang align, bound)) == NULL) {
762*4418919fSjohnjiang printf("%s(%s): memzone creation failed\n",
763*4418919fSjohnjiang __func__, name);
764*4418919fSjohnjiang return -1;
765*4418919fSjohnjiang }
766*4418919fSjohnjiang
767*4418919fSjohnjiang if ((mz->iova & ((rte_iova_t)align - 1)) != 0) {
768*4418919fSjohnjiang printf("%s(%s): invalid phys addr alignment\n",
769*4418919fSjohnjiang __func__, mz->name);
770*4418919fSjohnjiang return -1;
771*4418919fSjohnjiang }
772*4418919fSjohnjiang
773*4418919fSjohnjiang if (((uintptr_t) mz->addr & ((uintptr_t)align - 1)) != 0) {
774*4418919fSjohnjiang printf("%s(%s): invalid virtual addr alignment\n",
775*4418919fSjohnjiang __func__, mz->name);
776*4418919fSjohnjiang return -1;
777*4418919fSjohnjiang }
778*4418919fSjohnjiang
779*4418919fSjohnjiang if ((mz->len & RTE_CACHE_LINE_MASK) != 0 || mz->len < len ||
780*4418919fSjohnjiang mz->len < RTE_CACHE_LINE_SIZE) {
781*4418919fSjohnjiang printf("%s(%s): invalid length\n",
782*4418919fSjohnjiang __func__, mz->name);
783*4418919fSjohnjiang return -1;
784*4418919fSjohnjiang }
785*4418919fSjohnjiang
786*4418919fSjohnjiang if ((mz->iova & bmask) !=
787*4418919fSjohnjiang ((mz->iova + mz->len - 1) & bmask)) {
788*4418919fSjohnjiang printf("%s(%s): invalid memzone boundary %u crossed\n",
789*4418919fSjohnjiang __func__, mz->name, bound);
790*4418919fSjohnjiang return -1;
791*4418919fSjohnjiang }
792*4418919fSjohnjiang
793*4418919fSjohnjiang if (rte_memzone_free(mz)) {
794*4418919fSjohnjiang printf("Fail memzone free\n");
795*4418919fSjohnjiang return -1;
796*4418919fSjohnjiang }
797*4418919fSjohnjiang
798*4418919fSjohnjiang return 0;
799*4418919fSjohnjiang }
800*4418919fSjohnjiang
801*4418919fSjohnjiang static int
test_memzone_bounded(void)802*4418919fSjohnjiang test_memzone_bounded(void)
803*4418919fSjohnjiang {
804*4418919fSjohnjiang const struct rte_memzone *memzone_err;
805*4418919fSjohnjiang int rc;
806*4418919fSjohnjiang
807*4418919fSjohnjiang /* should fail as boundary is not power of two */
808*4418919fSjohnjiang memzone_err = rte_memzone_reserve_bounded(
809*4418919fSjohnjiang TEST_MEMZONE_NAME("bounded_error_31"), 100,
810*4418919fSjohnjiang SOCKET_ID_ANY, 0, 32, UINT32_MAX);
811*4418919fSjohnjiang if (memzone_err != NULL) {
812*4418919fSjohnjiang printf("%s(%s)created a memzone with invalid boundary "
813*4418919fSjohnjiang "conditions\n", __func__, memzone_err->name);
814*4418919fSjohnjiang return -1;
815*4418919fSjohnjiang }
816*4418919fSjohnjiang
817*4418919fSjohnjiang /* should fail as len is greater then boundary */
818*4418919fSjohnjiang memzone_err = rte_memzone_reserve_bounded(
819*4418919fSjohnjiang TEST_MEMZONE_NAME("bounded_error_32"), 100,
820*4418919fSjohnjiang SOCKET_ID_ANY, 0, 32, 32);
821*4418919fSjohnjiang if (memzone_err != NULL) {
822*4418919fSjohnjiang printf("%s(%s)created a memzone with invalid boundary "
823*4418919fSjohnjiang "conditions\n", __func__, memzone_err->name);
824*4418919fSjohnjiang return -1;
825*4418919fSjohnjiang }
826*4418919fSjohnjiang
827*4418919fSjohnjiang rc = check_memzone_bounded(TEST_MEMZONE_NAME("bounded_128"), 100, 128,
828*4418919fSjohnjiang 128);
829*4418919fSjohnjiang if (rc != 0)
830*4418919fSjohnjiang return rc;
831*4418919fSjohnjiang
832*4418919fSjohnjiang rc = check_memzone_bounded(TEST_MEMZONE_NAME("bounded_256"), 100, 256,
833*4418919fSjohnjiang 128);
834*4418919fSjohnjiang if (rc != 0)
835*4418919fSjohnjiang return rc;
836*4418919fSjohnjiang
837*4418919fSjohnjiang rc = check_memzone_bounded(TEST_MEMZONE_NAME("bounded_1K"), 100, 64,
838*4418919fSjohnjiang 1024);
839*4418919fSjohnjiang if (rc != 0)
840*4418919fSjohnjiang return rc;
841*4418919fSjohnjiang
842*4418919fSjohnjiang rc = check_memzone_bounded(TEST_MEMZONE_NAME("bounded_1K_MAX"), 0, 64,
843*4418919fSjohnjiang 1024);
844*4418919fSjohnjiang if (rc != 0)
845*4418919fSjohnjiang return rc;
846*4418919fSjohnjiang
847*4418919fSjohnjiang return 0;
848*4418919fSjohnjiang }
849*4418919fSjohnjiang
850*4418919fSjohnjiang static int
test_memzone_free(void)851*4418919fSjohnjiang test_memzone_free(void)
852*4418919fSjohnjiang {
853*4418919fSjohnjiang const struct rte_memzone *mz[RTE_MAX_MEMZONE + 1];
854*4418919fSjohnjiang int i;
855*4418919fSjohnjiang char name[20];
856*4418919fSjohnjiang
857*4418919fSjohnjiang mz[0] = rte_memzone_reserve(TEST_MEMZONE_NAME("tempzone0"), 2000,
858*4418919fSjohnjiang SOCKET_ID_ANY, 0);
859*4418919fSjohnjiang mz[1] = rte_memzone_reserve(TEST_MEMZONE_NAME("tempzone1"), 4000,
860*4418919fSjohnjiang SOCKET_ID_ANY, 0);
861*4418919fSjohnjiang
862*4418919fSjohnjiang if (mz[0] > mz[1])
863*4418919fSjohnjiang return -1;
864*4418919fSjohnjiang if (!rte_memzone_lookup(TEST_MEMZONE_NAME("tempzone0")))
865*4418919fSjohnjiang return -1;
866*4418919fSjohnjiang if (!rte_memzone_lookup(TEST_MEMZONE_NAME("tempzone1")))
867*4418919fSjohnjiang return -1;
868*4418919fSjohnjiang
869*4418919fSjohnjiang if (rte_memzone_free(mz[0])) {
870*4418919fSjohnjiang printf("Fail memzone free - tempzone0\n");
871*4418919fSjohnjiang return -1;
872*4418919fSjohnjiang }
873*4418919fSjohnjiang if (rte_memzone_lookup(TEST_MEMZONE_NAME("tempzone0"))) {
874*4418919fSjohnjiang printf("Found previously free memzone - tempzone0\n");
875*4418919fSjohnjiang return -1;
876*4418919fSjohnjiang }
877*4418919fSjohnjiang mz[2] = rte_memzone_reserve(TEST_MEMZONE_NAME("tempzone2"), 2000,
878*4418919fSjohnjiang SOCKET_ID_ANY, 0);
879*4418919fSjohnjiang
880*4418919fSjohnjiang if (mz[2] > mz[1]) {
881*4418919fSjohnjiang printf("tempzone2 should have gotten the free entry from tempzone0\n");
882*4418919fSjohnjiang return -1;
883*4418919fSjohnjiang }
884*4418919fSjohnjiang if (rte_memzone_free(mz[2])) {
885*4418919fSjohnjiang printf("Fail memzone free - tempzone2\n");
886*4418919fSjohnjiang return -1;
887*4418919fSjohnjiang }
888*4418919fSjohnjiang if (rte_memzone_lookup(TEST_MEMZONE_NAME("tempzone2"))) {
889*4418919fSjohnjiang printf("Found previously free memzone - tempzone2\n");
890*4418919fSjohnjiang return -1;
891*4418919fSjohnjiang }
892*4418919fSjohnjiang if (rte_memzone_free(mz[1])) {
893*4418919fSjohnjiang printf("Fail memzone free - tempzone1\n");
894*4418919fSjohnjiang return -1;
895*4418919fSjohnjiang }
896*4418919fSjohnjiang if (rte_memzone_lookup(TEST_MEMZONE_NAME("tempzone1"))) {
897*4418919fSjohnjiang printf("Found previously free memzone - tempzone1\n");
898*4418919fSjohnjiang return -1;
899*4418919fSjohnjiang }
900*4418919fSjohnjiang
901*4418919fSjohnjiang i = 0;
902*4418919fSjohnjiang do {
903*4418919fSjohnjiang snprintf(name, sizeof(name), TEST_MEMZONE_NAME("tempzone%u"),
904*4418919fSjohnjiang i);
905*4418919fSjohnjiang mz[i] = rte_memzone_reserve(name, 1, SOCKET_ID_ANY, 0);
906*4418919fSjohnjiang } while (mz[i++] != NULL);
907*4418919fSjohnjiang
908*4418919fSjohnjiang if (rte_memzone_free(mz[0])) {
909*4418919fSjohnjiang printf("Fail memzone free - tempzone0\n");
910*4418919fSjohnjiang return -1;
911*4418919fSjohnjiang }
912*4418919fSjohnjiang mz[0] = rte_memzone_reserve(TEST_MEMZONE_NAME("tempzone0new"), 0,
913*4418919fSjohnjiang SOCKET_ID_ANY, 0);
914*4418919fSjohnjiang
915*4418919fSjohnjiang if (mz[0] == NULL) {
916*4418919fSjohnjiang printf("Fail to create memzone - tempzone0new - when MAX memzones were "
917*4418919fSjohnjiang "created and one was free\n");
918*4418919fSjohnjiang return -1;
919*4418919fSjohnjiang }
920*4418919fSjohnjiang
921*4418919fSjohnjiang for (i = i - 2; i >= 0; i--) {
922*4418919fSjohnjiang if (rte_memzone_free(mz[i])) {
923*4418919fSjohnjiang printf("Fail memzone free - tempzone%d\n", i);
924*4418919fSjohnjiang return -1;
925*4418919fSjohnjiang }
926*4418919fSjohnjiang }
927*4418919fSjohnjiang
928*4418919fSjohnjiang return 0;
929*4418919fSjohnjiang }
930*4418919fSjohnjiang
931*4418919fSjohnjiang static int test_memzones_left;
932*4418919fSjohnjiang static int memzone_walk_cnt;
memzone_walk_clb(const struct rte_memzone * mz,void * arg __rte_unused)933*4418919fSjohnjiang static void memzone_walk_clb(const struct rte_memzone *mz,
934*4418919fSjohnjiang void *arg __rte_unused)
935*4418919fSjohnjiang {
936*4418919fSjohnjiang memzone_walk_cnt++;
937*4418919fSjohnjiang if (!strncmp(TEST_MEMZONE_NAME(""), mz->name, RTE_MEMZONE_NAMESIZE))
938*4418919fSjohnjiang test_memzones_left++;
939*4418919fSjohnjiang }
940*4418919fSjohnjiang
941*4418919fSjohnjiang static int
test_memzone_basic(void)942*4418919fSjohnjiang test_memzone_basic(void)
943*4418919fSjohnjiang {
944*4418919fSjohnjiang const struct rte_memzone *memzone1;
945*4418919fSjohnjiang const struct rte_memzone *memzone2;
946*4418919fSjohnjiang const struct rte_memzone *memzone3;
947*4418919fSjohnjiang const struct rte_memzone *memzone4;
948*4418919fSjohnjiang const struct rte_memzone *mz;
949*4418919fSjohnjiang int memzone_cnt_after, memzone_cnt_expected;
950*4418919fSjohnjiang int memzone_cnt_before;
951*4418919fSjohnjiang
952*4418919fSjohnjiang memzone_walk_cnt = 0;
953*4418919fSjohnjiang test_memzones_left = 0;
954*4418919fSjohnjiang rte_memzone_walk(memzone_walk_clb, NULL);
955*4418919fSjohnjiang memzone_cnt_before = memzone_walk_cnt;
956*4418919fSjohnjiang
957*4418919fSjohnjiang memzone1 = rte_memzone_reserve(TEST_MEMZONE_NAME("testzone1"), 100,
958*4418919fSjohnjiang SOCKET_ID_ANY, 0);
959*4418919fSjohnjiang
960*4418919fSjohnjiang memzone2 = rte_memzone_reserve(TEST_MEMZONE_NAME("testzone2"), 1000,
961*4418919fSjohnjiang 0, 0);
962*4418919fSjohnjiang
963*4418919fSjohnjiang memzone3 = rte_memzone_reserve(TEST_MEMZONE_NAME("testzone3"), 1000,
964*4418919fSjohnjiang 1, 0);
965*4418919fSjohnjiang
966*4418919fSjohnjiang memzone4 = rte_memzone_reserve(TEST_MEMZONE_NAME("testzone4"), 1024,
967*4418919fSjohnjiang SOCKET_ID_ANY, 0);
968*4418919fSjohnjiang
969*4418919fSjohnjiang /* memzone3 may be NULL if we don't have NUMA */
970*4418919fSjohnjiang if (memzone1 == NULL || memzone2 == NULL || memzone4 == NULL)
971*4418919fSjohnjiang return -1;
972*4418919fSjohnjiang
973*4418919fSjohnjiang /* check how many memzones we are expecting */
974*4418919fSjohnjiang memzone_cnt_expected = memzone_cnt_before +
975*4418919fSjohnjiang (memzone1 != NULL) + (memzone2 != NULL) +
976*4418919fSjohnjiang (memzone3 != NULL) + (memzone4 != NULL);
977*4418919fSjohnjiang
978*4418919fSjohnjiang memzone_walk_cnt = 0;
979*4418919fSjohnjiang test_memzones_left = 0;
980*4418919fSjohnjiang rte_memzone_walk(memzone_walk_clb, NULL);
981*4418919fSjohnjiang memzone_cnt_after = memzone_walk_cnt;
982*4418919fSjohnjiang
983*4418919fSjohnjiang if (memzone_cnt_after != memzone_cnt_expected)
984*4418919fSjohnjiang return -1;
985*4418919fSjohnjiang
986*4418919fSjohnjiang
987*4418919fSjohnjiang rte_memzone_dump(stdout);
988*4418919fSjohnjiang
989*4418919fSjohnjiang /* check cache-line alignments */
990*4418919fSjohnjiang printf("check alignments and lengths\n");
991*4418919fSjohnjiang
992*4418919fSjohnjiang if ((memzone1->iova & RTE_CACHE_LINE_MASK) != 0)
993*4418919fSjohnjiang return -1;
994*4418919fSjohnjiang if ((memzone2->iova & RTE_CACHE_LINE_MASK) != 0)
995*4418919fSjohnjiang return -1;
996*4418919fSjohnjiang if (memzone3 != NULL && (memzone3->iova & RTE_CACHE_LINE_MASK) != 0)
997*4418919fSjohnjiang return -1;
998*4418919fSjohnjiang if ((memzone1->len & RTE_CACHE_LINE_MASK) != 0 || memzone1->len == 0)
999*4418919fSjohnjiang return -1;
1000*4418919fSjohnjiang if ((memzone2->len & RTE_CACHE_LINE_MASK) != 0 || memzone2->len == 0)
1001*4418919fSjohnjiang return -1;
1002*4418919fSjohnjiang if (memzone3 != NULL && ((memzone3->len & RTE_CACHE_LINE_MASK) != 0 ||
1003*4418919fSjohnjiang memzone3->len == 0))
1004*4418919fSjohnjiang return -1;
1005*4418919fSjohnjiang if (memzone4->len != 1024)
1006*4418919fSjohnjiang return -1;
1007*4418919fSjohnjiang
1008*4418919fSjohnjiang /* check that zones don't overlap */
1009*4418919fSjohnjiang printf("check overlapping\n");
1010*4418919fSjohnjiang
1011*4418919fSjohnjiang if (is_memory_overlap(memzone1->iova, memzone1->len,
1012*4418919fSjohnjiang memzone2->iova, memzone2->len))
1013*4418919fSjohnjiang return -1;
1014*4418919fSjohnjiang if (memzone3 != NULL &&
1015*4418919fSjohnjiang is_memory_overlap(memzone1->iova, memzone1->len,
1016*4418919fSjohnjiang memzone3->iova, memzone3->len))
1017*4418919fSjohnjiang return -1;
1018*4418919fSjohnjiang if (memzone3 != NULL &&
1019*4418919fSjohnjiang is_memory_overlap(memzone2->iova, memzone2->len,
1020*4418919fSjohnjiang memzone3->iova, memzone3->len))
1021*4418919fSjohnjiang return -1;
1022*4418919fSjohnjiang
1023*4418919fSjohnjiang printf("check socket ID\n");
1024*4418919fSjohnjiang
1025*4418919fSjohnjiang /* memzone2 must be on socket id 0 and memzone3 on socket 1 */
1026*4418919fSjohnjiang if (memzone2->socket_id != 0)
1027*4418919fSjohnjiang return -1;
1028*4418919fSjohnjiang if (memzone3 != NULL && memzone3->socket_id != 1)
1029*4418919fSjohnjiang return -1;
1030*4418919fSjohnjiang
1031*4418919fSjohnjiang printf("test zone lookup\n");
1032*4418919fSjohnjiang mz = rte_memzone_lookup(TEST_MEMZONE_NAME("testzone1"));
1033*4418919fSjohnjiang if (mz != memzone1)
1034*4418919fSjohnjiang return -1;
1035*4418919fSjohnjiang
1036*4418919fSjohnjiang printf("test duplcate zone name\n");
1037*4418919fSjohnjiang mz = rte_memzone_reserve(TEST_MEMZONE_NAME("testzone1"), 100,
1038*4418919fSjohnjiang SOCKET_ID_ANY, 0);
1039*4418919fSjohnjiang if (mz != NULL)
1040*4418919fSjohnjiang return -1;
1041*4418919fSjohnjiang
1042*4418919fSjohnjiang if (rte_memzone_free(memzone1)) {
1043*4418919fSjohnjiang printf("Fail memzone free - memzone1\n");
1044*4418919fSjohnjiang return -1;
1045*4418919fSjohnjiang }
1046*4418919fSjohnjiang if (rte_memzone_free(memzone2)) {
1047*4418919fSjohnjiang printf("Fail memzone free - memzone2\n");
1048*4418919fSjohnjiang return -1;
1049*4418919fSjohnjiang }
1050*4418919fSjohnjiang if (memzone3 && rte_memzone_free(memzone3)) {
1051*4418919fSjohnjiang printf("Fail memzone free - memzone3\n");
1052*4418919fSjohnjiang return -1;
1053*4418919fSjohnjiang }
1054*4418919fSjohnjiang if (rte_memzone_free(memzone4)) {
1055*4418919fSjohnjiang printf("Fail memzone free - memzone4\n");
1056*4418919fSjohnjiang return -1;
1057*4418919fSjohnjiang }
1058*4418919fSjohnjiang
1059*4418919fSjohnjiang memzone_walk_cnt = 0;
1060*4418919fSjohnjiang test_memzones_left = 0;
1061*4418919fSjohnjiang rte_memzone_walk(memzone_walk_clb, NULL);
1062*4418919fSjohnjiang memzone_cnt_after = memzone_walk_cnt;
1063*4418919fSjohnjiang if (memzone_cnt_after != memzone_cnt_before)
1064*4418919fSjohnjiang return -1;
1065*4418919fSjohnjiang
1066*4418919fSjohnjiang return 0;
1067*4418919fSjohnjiang }
1068*4418919fSjohnjiang
1069*4418919fSjohnjiang static int
test_memzone(void)1070*4418919fSjohnjiang test_memzone(void)
1071*4418919fSjohnjiang {
1072*4418919fSjohnjiang /* take note of how many memzones were allocated before running */
1073*4418919fSjohnjiang int memzone_cnt;
1074*4418919fSjohnjiang
1075*4418919fSjohnjiang memzone_walk_cnt = 0;
1076*4418919fSjohnjiang test_memzones_left = 0;
1077*4418919fSjohnjiang rte_memzone_walk(memzone_walk_clb, NULL);
1078*4418919fSjohnjiang memzone_cnt = memzone_walk_cnt;
1079*4418919fSjohnjiang
1080*4418919fSjohnjiang printf("test basic memzone API\n");
1081*4418919fSjohnjiang if (test_memzone_basic() < 0)
1082*4418919fSjohnjiang return -1;
1083*4418919fSjohnjiang
1084*4418919fSjohnjiang printf("test free memzone\n");
1085*4418919fSjohnjiang if (test_memzone_free() < 0)
1086*4418919fSjohnjiang return -1;
1087*4418919fSjohnjiang
1088*4418919fSjohnjiang printf("test reserving memzone with bigger size than the maximum\n");
1089*4418919fSjohnjiang if (test_memzone_reserving_zone_size_bigger_than_the_maximum() < 0)
1090*4418919fSjohnjiang return -1;
1091*4418919fSjohnjiang
1092*4418919fSjohnjiang printf("test memzone_reserve flags\n");
1093*4418919fSjohnjiang if (test_memzone_reserve_flags() < 0)
1094*4418919fSjohnjiang return -1;
1095*4418919fSjohnjiang
1096*4418919fSjohnjiang printf("test alignment for memzone_reserve\n");
1097*4418919fSjohnjiang if (test_memzone_aligned() < 0)
1098*4418919fSjohnjiang return -1;
1099*4418919fSjohnjiang
1100*4418919fSjohnjiang printf("test boundary alignment for memzone_reserve\n");
1101*4418919fSjohnjiang if (test_memzone_bounded() < 0)
1102*4418919fSjohnjiang return -1;
1103*4418919fSjohnjiang
1104*4418919fSjohnjiang printf("test invalid alignment for memzone_reserve\n");
1105*4418919fSjohnjiang if (test_memzone_invalid_alignment() < 0)
1106*4418919fSjohnjiang return -1;
1107*4418919fSjohnjiang
1108*4418919fSjohnjiang printf("test reserving the largest size memzone possible\n");
1109*4418919fSjohnjiang if (test_memzone_reserve_max() < 0)
1110*4418919fSjohnjiang return -1;
1111*4418919fSjohnjiang
1112*4418919fSjohnjiang printf("test reserving the largest size aligned memzone possible\n");
1113*4418919fSjohnjiang if (test_memzone_reserve_max_aligned() < 0)
1114*4418919fSjohnjiang return -1;
1115*4418919fSjohnjiang
1116*4418919fSjohnjiang printf("check memzone cleanup\n");
1117*4418919fSjohnjiang memzone_walk_cnt = 0;
1118*4418919fSjohnjiang test_memzones_left = 0;
1119*4418919fSjohnjiang rte_memzone_walk(memzone_walk_clb, NULL);
1120*4418919fSjohnjiang if (memzone_walk_cnt != memzone_cnt || test_memzones_left > 0) {
1121*4418919fSjohnjiang printf("there are some memzones left after test\n");
1122*4418919fSjohnjiang rte_memzone_dump(stdout);
1123*4418919fSjohnjiang return -1;
1124*4418919fSjohnjiang }
1125*4418919fSjohnjiang
1126*4418919fSjohnjiang return 0;
1127*4418919fSjohnjiang }
1128*4418919fSjohnjiang
1129*4418919fSjohnjiang REGISTER_TEST_COMMAND(memzone_autotest, test_memzone);
1130