|
1
|
1 |
|
from plugin.core.configuration import Configuration |
|
2
|
1 |
|
from plugin.core.libraries.helpers.arm import ArmHelper |
|
3
|
|
|
|
|
4
|
1 |
|
from elftools.elf.attributes import AttributesSection |
|
5
|
1 |
|
from elftools.elf.elffile import ELFFile |
|
6
|
1 |
|
import logging |
|
7
|
1 |
|
import os |
|
8
|
1 |
|
import platform |
|
9
|
1 |
|
import sys |
|
10
|
|
|
|
|
11
|
1 |
|
log = logging.getLogger(__name__) |
|
12
|
|
|
|
|
13
|
1 |
|
BITS_MAP = { |
|
14
|
|
|
'32bit': 'i386', |
|
15
|
|
|
'64bit': 'x86_64' |
|
16
|
|
|
} |
|
17
|
|
|
|
|
18
|
1 |
|
MACHINE_MAP = { |
|
19
|
|
|
('32bit', 'i686'): 'i686', |
|
20
|
|
|
('32bit', 'ppc' ): 'PowerPC' |
|
21
|
|
|
} |
|
22
|
|
|
|
|
23
|
1 |
|
MSVCR_MAP = { |
|
24
|
|
|
'msvcr120.dll': 'vc12', |
|
25
|
|
|
'msvcr130.dll': 'vc14' |
|
26
|
|
|
} |
|
27
|
|
|
|
|
28
|
1 |
|
NAME_MAP = { |
|
29
|
|
|
'Darwin': 'MacOSX' |
|
30
|
|
|
} |
|
31
|
|
|
|
|
32
|
1 |
|
FALLBACK_EXECUTABLE = '/bin/ls' |
|
33
|
|
|
|
|
34
|
|
|
|
|
35
|
1 |
|
class SystemHelper(object): |
|
36
|
1 |
|
@classmethod |
|
37
|
|
|
def architecture(cls): |
|
38
|
|
|
"""Retrieve system architecture (i386, i686, x86_64)""" |
|
39
|
|
|
|
|
40
|
|
|
# Use `cpu_architecture` value from advanced configuration (if defined) |
|
41
|
1 |
|
cpu_architecture = Configuration.advanced['libraries'].get('cpu_architecture') |
|
42
|
|
|
|
|
43
|
1 |
|
if cpu_architecture: |
|
44
|
|
|
log.info('Using CPU Architecture from advanced configuration: %r', cpu_architecture) |
|
45
|
|
|
return cpu_architecture |
|
46
|
|
|
|
|
47
|
|
|
# Determine architecture from platform attributes |
|
48
|
1 |
|
bits, _ = platform.architecture() |
|
49
|
1 |
|
machine = platform.machine() |
|
50
|
|
|
|
|
51
|
|
|
# Check for ARM machine |
|
52
|
1 |
|
if bits == '32bit' and machine.startswith('armv'): |
|
53
|
|
|
return cls.arm(machine) |
|
54
|
|
|
|
|
55
|
|
|
# Check (bits, machine) map |
|
56
|
1 |
|
machine_key = (bits, machine) |
|
57
|
|
|
|
|
58
|
1 |
|
if machine_key in MACHINE_MAP: |
|
59
|
|
|
return MACHINE_MAP[machine_key] |
|
60
|
|
|
|
|
61
|
|
|
# Check (bits) map |
|
62
|
1 |
|
if bits in BITS_MAP: |
|
63
|
1 |
|
return BITS_MAP[bits] |
|
64
|
|
|
|
|
65
|
|
|
log.error('Unable to determine system architecture - bits: %r, machine: %r', bits, machine) |
|
66
|
|
|
return None |
|
67
|
|
|
|
|
68
|
1 |
|
@classmethod |
|
69
|
|
|
def name(cls): |
|
70
|
|
|
"""Retrieve system name (Windows, Linux, FreeBSD, MacOSX)""" |
|
71
|
|
|
|
|
72
|
1 |
|
system = platform.system() |
|
73
|
|
|
|
|
74
|
|
|
# Apply system map |
|
75
|
1 |
|
if system in NAME_MAP: |
|
76
|
|
|
system = NAME_MAP[system] |
|
77
|
|
|
|
|
78
|
1 |
|
return system |
|
79
|
|
|
|
|
80
|
1 |
|
@classmethod |
|
81
|
|
|
def arm(cls, machine): |
|
82
|
|
|
# Determine floating-point type |
|
83
|
|
|
float_type = cls.arm_float_type() |
|
84
|
|
|
|
|
85
|
|
|
if float_type is None: |
|
86
|
|
|
log.warn('Unable to use ARM libraries, unsupported floating-point type?') |
|
87
|
|
|
return None |
|
88
|
|
|
|
|
89
|
|
|
# Determine ARM version |
|
90
|
|
|
version = cls.arm_version() |
|
91
|
|
|
|
|
92
|
|
|
if version is None: |
|
93
|
|
|
log.warn('Unable to use ARM libraries, unsupported ARM version (%r)?' % machine) |
|
94
|
|
|
return None |
|
95
|
|
|
|
|
96
|
|
|
return '%s_%s' % (version, float_type) |
|
97
|
|
|
|
|
98
|
1 |
|
@classmethod |
|
99
|
1 |
|
def arm_version(cls, machine=None): |
|
100
|
|
|
# Read `machine` name if not provided |
|
101
|
|
|
if machine is None: |
|
102
|
|
|
machine = platform.machine() |
|
103
|
|
|
|
|
104
|
|
|
# Ensure `machine` is valid |
|
105
|
|
|
if not machine: |
|
106
|
|
|
return None |
|
107
|
|
|
|
|
108
|
|
|
# ARMv5 |
|
109
|
|
|
if machine.startswith('armv5'): |
|
110
|
|
|
return 'armv5' |
|
111
|
|
|
|
|
112
|
|
|
# ARMv6 |
|
113
|
|
|
if machine.startswith('armv6'): |
|
114
|
|
|
return 'armv6' |
|
115
|
|
|
|
|
116
|
|
|
# ARMv7 |
|
117
|
|
|
if machine.startswith('armv7'): |
|
118
|
|
|
return 'armv7' |
|
119
|
|
|
|
|
120
|
|
|
return None |
|
121
|
|
|
|
|
122
|
1 |
|
@classmethod |
|
123
|
1 |
|
def arm_float_type(cls, executable_path=sys.executable): |
|
124
|
|
|
# Use `arm_float_type` value from advanced configuration (if defined) |
|
125
|
|
|
arm_float_type = Configuration.advanced['libraries'].get('arm_float_type') |
|
126
|
|
|
|
|
127
|
|
|
if arm_float_type: |
|
128
|
|
|
log.info('Using ARM Float Type from advanced configuration: %r', arm_float_type) |
|
129
|
|
|
return arm_float_type |
|
130
|
|
|
|
|
131
|
|
|
# Try determine float-type from "/lib" directories |
|
132
|
|
|
if os.path.exists('/lib/arm-linux-gnueabihf'): |
|
133
|
|
|
return 'hf' |
|
134
|
|
|
|
|
135
|
|
|
if os.path.exists('/lib/arm-linux-gnueabi'): |
|
136
|
|
|
return 'sf' |
|
137
|
|
|
|
|
138
|
|
|
# Determine system float-type from python executable |
|
139
|
|
|
section, attributes = cls.elf_attributes(executable_path) |
|
140
|
|
|
|
|
141
|
|
|
if not section or not attributes: |
|
142
|
|
|
return None |
|
143
|
|
|
|
|
144
|
|
|
if section.name != 'aeabi': |
|
145
|
|
|
log.warn('Unknown attributes section name: %r', section.name) |
|
146
|
|
|
return None |
|
147
|
|
|
|
|
148
|
|
|
# Assume hard-float if "tag_abi_vfp_args" is present |
|
149
|
|
|
if attributes.get('abi_vfp_args'): |
|
150
|
|
|
return 'hf' |
|
151
|
|
|
|
|
152
|
|
|
return 'sf' |
|
153
|
|
|
|
|
154
|
1 |
|
@classmethod |
|
155
|
1 |
|
def cpu_name(cls, executable_path=sys.executable): |
|
156
|
|
|
# Retrieve CPU name from ELF |
|
157
|
1 |
|
section, attributes = cls.elf_attributes(executable_path) |
|
158
|
|
|
|
|
159
|
1 |
|
if not section or not attributes: |
|
160
|
1 |
|
return None |
|
161
|
|
|
|
|
162
|
|
|
name = attributes.get('cpu_name') |
|
163
|
|
|
|
|
164
|
|
|
if not name: |
|
165
|
|
|
return None |
|
166
|
|
|
|
|
167
|
|
|
return name.lower() |
|
168
|
|
|
|
|
169
|
1 |
|
@classmethod |
|
170
|
1 |
|
def cpu_type(cls, executable_path=sys.executable): |
|
171
|
|
|
# Use `cpu_type` value from advanced configuration (if defined) |
|
172
|
1 |
|
cpu_type = Configuration.advanced['libraries'].get('cpu_type') |
|
173
|
|
|
|
|
174
|
1 |
|
if cpu_type: |
|
175
|
|
|
log.info('Using CPU Type from advanced configuration: %r', cpu_type) |
|
176
|
|
|
return cpu_type |
|
177
|
|
|
|
|
178
|
|
|
# Try retrieve cpu type via "/proc/cpuinfo" |
|
179
|
1 |
|
try: |
|
180
|
1 |
|
_, _, cpu_type = ArmHelper.identifier() |
|
181
|
|
|
|
|
182
|
1 |
|
if cpu_type: |
|
183
|
|
|
return cpu_type |
|
184
|
|
|
except Exception, ex: |
|
185
|
|
|
log.warn('Unable to retrieve cpu type from "/proc/cpuinfo": %s', ex, exc_info=True) |
|
186
|
|
|
|
|
187
|
|
|
# Fallback to using the ELF cpu name |
|
188
|
1 |
|
return cls.cpu_name(executable_path) |
|
189
|
|
|
|
|
190
|
1 |
|
@classmethod |
|
191
|
1 |
|
def elf_attributes(cls, executable_path=sys.executable): |
|
192
|
1 |
|
if cls.name() == 'MacOSX': |
|
193
|
|
|
log.info('Unable to retrieve ELF attributes on Mac OSX (not supported)') |
|
194
|
|
|
return None, None |
|
195
|
|
|
|
|
196
|
|
|
# Read attributes from "/bin/ls" if `executable_path` doesn't exist |
|
197
|
1 |
|
if not executable_path or not os.path.exists(executable_path): |
|
198
|
|
|
log.info('Executable at %r doesn\'t exist, using %r instead', executable_path, FALLBACK_EXECUTABLE) |
|
199
|
|
|
executable_path = FALLBACK_EXECUTABLE |
|
200
|
|
|
|
|
201
|
1 |
|
try: |
|
202
|
|
|
# Open executable stream |
|
203
|
1 |
|
stream = open(executable_path, 'rb') |
|
204
|
|
|
|
|
205
|
|
|
# Retrieve magic number (header) |
|
206
|
1 |
|
magic = stream.read(4) |
|
207
|
|
|
|
|
208
|
1 |
|
if magic != b'\x7fELF': |
|
209
|
|
|
log.warn('Unknown ELF format for %r (magic: %r)', executable_path, magic) |
|
210
|
|
|
return None, None |
|
211
|
|
|
|
|
212
|
1 |
|
stream.seek(0) |
|
213
|
|
|
|
|
214
|
|
|
# Parse ELF |
|
215
|
1 |
|
elf = ELFFile(stream) |
|
216
|
|
|
|
|
217
|
|
|
# Find attributes section |
|
218
|
1 |
|
section = cls._find_elf_section(elf, AttributesSection) |
|
219
|
|
|
|
|
220
|
1 |
|
if section is None: |
|
221
|
1 |
|
log.info('Unable to find attributes section in ELF: %r', executable_path) |
|
222
|
1 |
|
return None, None |
|
223
|
|
|
|
|
224
|
|
|
# Build dictionary of attributes |
|
225
|
|
|
attributes = dict([ |
|
226
|
|
|
(attr.tag.lower(), attr.value) |
|
227
|
|
|
for attr in section.iter_attributes() |
|
228
|
|
|
]) |
|
229
|
|
|
|
|
230
|
|
|
return section, attributes |
|
231
|
|
|
except Exception, ex: |
|
232
|
|
|
log.warn('Unable to retrieve attributes from ELF %r: %s', executable_path, ex, exc_info=True) |
|
233
|
|
|
|
|
234
|
|
|
return None, None |
|
235
|
|
|
|
|
236
|
1 |
|
@classmethod |
|
237
|
|
|
def page_size(cls): |
|
238
|
1 |
|
try: |
|
239
|
1 |
|
import resource |
|
240
|
1 |
|
page_size = resource.getpagesize() |
|
241
|
|
|
|
|
242
|
1 |
|
if not page_size: |
|
243
|
|
|
return None |
|
244
|
|
|
|
|
245
|
1 |
|
return '%dk' % (page_size / 1024) |
|
246
|
|
|
except Exception, ex: |
|
247
|
|
|
log.warn('Unable to retrieve memory page size: %s', ex, exc_info=True) |
|
248
|
|
|
return None |
|
249
|
|
|
|
|
250
|
1 |
|
@staticmethod |
|
251
|
|
|
def _find_elf_section(elf, cls): |
|
252
|
1 |
|
for section in elf.iter_sections(): |
|
253
|
1 |
|
if isinstance(section, cls): |
|
254
|
|
|
return section |
|
255
|
|
|
|
|
256
|
1 |
|
return None |
|
257
|
|
|
|
|
258
|
1 |
|
@classmethod |
|
259
|
|
|
def vcr_version(cls): |
|
260
|
|
|
try: |
|
261
|
|
|
import ctypes.util |
|
262
|
|
|
|
|
263
|
|
|
# Retrieve linked msvcr dll |
|
264
|
|
|
name = ctypes.util.find_msvcrt() |
|
265
|
|
|
|
|
266
|
|
|
# Return VC++ version from map |
|
267
|
|
|
if name not in MSVCR_MAP: |
|
268
|
|
|
log.error('Unknown VC++ runtime: %r', name) |
|
269
|
|
|
return None |
|
270
|
|
|
|
|
271
|
|
|
return MSVCR_MAP[name] |
|
272
|
|
|
except Exception: |
|
273
|
|
|
log.error('Unable to retrieve VC++ runtime version', exc_info=True) |
|
274
|
|
|
return None |
|
275
|
|
|
|