blob: d38296e1e5b9a52f58aafe4775faff573c35f74b (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
|
/* SPDX-License-Identifier: BSD-3-Clause
* Copyright 2018 Red Hat, Inc.
*/
#include <elf.h>
#include <fcntl.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#if defined(__GLIBC__) && defined(__GLIBC_PREREQ)
#if __GLIBC_PREREQ(2, 16)
#include <sys/auxv.h>
#define HAS_AUXV 1
#endif
#endif
#include <rte_cpuflags.h>
#ifndef HAS_AUXV
static unsigned long
getauxval(unsigned long type __rte_unused)
{
errno = ENOTSUP;
return 0;
}
#endif
#ifdef RTE_ARCH_64
typedef Elf64_auxv_t Internal_Elfx_auxv_t;
#else
typedef Elf32_auxv_t Internal_Elfx_auxv_t;
#endif
/**
* Provides a method for retrieving values from the auxiliary vector and
* possibly running a string comparison.
*
* @return Always returns a result. When the result is 0, check errno
* to see if an error occurred during processing.
*/
static unsigned long
_rte_cpu_getauxval(unsigned long type, const char *str)
{
unsigned long val;
errno = 0;
val = getauxval(type);
if (!val && (errno == ENOTSUP || errno == ENOENT)) {
int auxv_fd = open("/proc/self/auxv", O_RDONLY);
Internal_Elfx_auxv_t auxv;
if (auxv_fd == -1)
return 0;
errno = ENOENT;
while (read(auxv_fd, &auxv, sizeof(auxv)) == sizeof(auxv)) {
if (auxv.a_type == type) {
errno = 0;
val = auxv.a_un.a_val;
if (str)
val = strcmp((const char *)val, str);
break;
}
}
close(auxv_fd);
}
return val;
}
unsigned long
rte_cpu_getauxval(unsigned long type)
{
return _rte_cpu_getauxval(type, NULL);
}
int
rte_cpu_strcmp_auxval(unsigned long type, const char *str)
{
return _rte_cpu_getauxval(type, str);
}
|