forked from ThomasKaiser/sbc-bench
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathsbc-bench.sh
executable file
·664 lines (611 loc) · 27.6 KB
/
sbc-bench.sh
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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
#!/bin/bash
Version=0.5.4
InstallLocation=/usr/local/src # change to /tmp if you want tools to be deleted after reboot
Main() {
export PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin
# check whether we're running in monitoring or benchmark mode
if [ "X$1" = "Xm" ]; then
interval=$2
MonitorBoard
else
# Check if we're outputting to a terminal. If yes try to use bold and colors for messages
if test -t 1; then
ncolors=$(tput colors)
if test -n "$ncolors" && test $ncolors -ge 8; then
BOLD="$(tput bold)"
NC='\033[0m' # No Color
LGREEN='\033[1;32m'
LRED='\e[0;91m'
fi
fi
if [ "X$1" = "Xneon" ]; then
TestNEON=yes
fi
PathToMe="$( cd "$(dirname "$0")" ; pwd -P )/${0##*/}"
CheckRelease
CheckLoad
BasicSetup >/dev/null 2>&1
InstallPrerequisits
InitialMonitoring
CheckClockspeeds
CheckTimeInState before
RunTinyMemBench
RunOpenSSLBenchmark
Run7ZipBenchmark
if [ "${TestNEON}" = "yes" -a -x "${InstallLocation}"/cpuminer-multi/cpuminer ]; then
RunCpuminerBenchmark
fi
CheckTimeInState after
CheckClockspeeds # again after heating the SoC to the max
DisplayResults
fi
} # Main
MonitorBoard() {
# Background monitoring
# In Armbian we can rely on /etc/armbianmonitor/datasources/soctemp
if [ -f /etc/armbianmonitor/datasources/soctemp ]; then
TempSource=/etc/armbianmonitor/datasources/soctemp
else
TempSource="$(mktemp /tmp/soctemp.XXXXXX)"
trap "rm -f \"${TempSource}\" ; exit 0" 0 1 2 3 15
if [[ -d "/sys/devices/platform/a20-tp-hwmon" ]]; then
# Allwinner A20 with old 3.4 kernel
ln -fs /sys/devices/platform/a20-tp-hwmon/temp1_input ${TempSource}
elif [[ -f /sys/class/hwmon/hwmon0/temp1_input ]]; then
# usual convention with modern kernels
ln -fs /sys/class/hwmon/hwmon0/temp1_input ${TempSource}
else
# all other boards/kernels use the same sysfs node except of Actions Semi S500
# so on LeMaker Guitar, Roseapple Pi or Allo Sparky it must read "thermal_zone1"
ln -fs /sys/devices/virtual/thermal/thermal_zone0/temp ${TempSource}
fi
fi
# Try to renice to 19 to not interfere with benchmark behaviour
renice 19 $BASHPID >/dev/null 2>&1
# check platform
case $(lscpu | awk -F" " '/^Architecture/ {print $2}') in
x86*|i686)
IsIntel="yes"
;;
esac
LastUserStat=0
LastNiceStat=0
LastSystemStat=0
LastIdleStat=0
LastIOWaitStat=0
LastIrqStat=0
LastSoftIrqStat=0
LastCpuStatCheck=0
LastTotal=0
SleepInterval=${interval:-5}
if [ -f /usr/bin/vcgencmd ]; then
DisplayHeader="Time fake/real load %cpu %sys %usr %nice %io %irq Temp VCore"
CPUs=raspberrypi
elif [ "${IsIntel}" = "yes" ] ; then
DisplayHeader="Time CPU load %cpu %sys %usr %nice %io %irq Temp"
CPUs=normal
elif [ -f /sys/devices/system/cpu/cpu4/cpufreq/cpuinfo_cur_freq ]; then
DisplayHeader="Time big.LITTLE load %cpu %sys %usr %nice %io %irq Temp"
read MaxSpeed0 </sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_max_freq
read MaxSpeed4 </sys/devices/system/cpu/cpu4/cpufreq/cpuinfo_max_freq
if [ ${MaxSpeed4} -ge ${MaxSpeed0} ]; then
CPUs=biglittle
else
CPUs=littlebig
fi
elif [ -f /sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_cur_freq ]; then
DisplayHeader="Time CPU load %cpu %sys %usr %nice %io %irq Temp"
CPUs=normal
else
DisplayHeader="Time CPU n/a load %cpu %sys %usr %nice %io %irq Temp"
CPUs=notavailable
fi
[ -f "${TempSource}" ] || SocTemp='n/a'
echo -e "${DisplayHeader}"
while true ; do
LoadAvg=$(cut -f1 -d" " </proc/loadavg)
if [ "X${SocTemp}" != "Xn/a" ]; then
read SocTemp <"${TempSource}"
if [ ${SocTemp} -ge 1000 ]; then
SocTemp=$(awk '{printf ("%0.1f",$1/1000); }' <<<${SocTemp})
fi
fi
ProcessStats
case ${CPUs} in
raspberrypi)
FakeFreq=$(awk '{printf ("%0.0f",$1/1000); }' </sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_cur_freq) 2>/dev/null
RealFreq=$(/usr/bin/vcgencmd measure_clock arm | awk -F"=" '{printf ("%0.0f",$2/1000000); }' )
CoreVoltage=$(/usr/bin/vcgencmd measure_volts | cut -f2 -d= | sed 's/000//')
echo -e "$(date "+%H:%M:%S"): $(printf "%4s" ${FakeFreq})/$(printf "%4s" ${RealFreq})MHz $(printf "%5s" ${LoadAvg}) ${procStats} $(printf "%4s" ${SocTemp})°C ${CoreVoltage}"
;;
biglittle)
BigFreq=$(awk '{printf ("%0.0f",$1/1000); }' </sys/devices/system/cpu/cpu4/cpufreq/cpuinfo_cur_freq) 2>/dev/null
LittleFreq=$(awk '{printf ("%0.0f",$1/1000); }' </sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_cur_freq) 2>/dev/null
echo -e "$(date "+%H:%M:%S"): $(printf "%4s" ${BigFreq})/$(printf "%4s" ${LittleFreq})MHz $(printf "%5s" ${LoadAvg}) ${procStats} $(printf "%4s" ${SocTemp})°C"
;;
littlebig)
BigFreq=$(awk '{printf ("%0.0f",$1/1000); }' </sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_cur_freq) 2>/dev/null
LittleFreq=$(awk '{printf ("%0.0f",$1/1000); }' </sys/devices/system/cpu/cpu4/cpufreq/cpuinfo_cur_freq) 2>/dev/null
echo -e "$(date "+%H:%M:%S"): $(printf "%4s" ${BigFreq})/$(printf "%4s" ${LittleFreq})MHz $(printf "%5s" ${LoadAvg}) ${procStats} $(printf "%4s" ${SocTemp})°C"
;;
normal)
CpuFreq=$(awk '{printf ("%0.0f",$1/1000); }' </sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_cur_freq) 2>/dev/null
echo -e "$(date "+%H:%M:%S"): $(printf "%4s" ${CpuFreq})MHz $(printf "%5s" ${LoadAvg}) ${procStats} $(printf "%4s" ${SocTemp})°C"
;;
notavailable)
echo -e "$(date "+%H:%M:%S"): --- $(printf "%5s" ${LoadAvg}) ${procStats} $(printf "%4s" ${SocTemp})°C"
;;
esac
sleep ${SleepInterval}
done
} # MonitorBoard
ProcessStats() {
# Generate detailed usage statistics based on /proc/stat
procStatLine=(`sed -n 's/^cpu\s//p' /proc/stat`)
UserStat=${procStatLine[0]}
NiceStat=${procStatLine[1]}
SystemStat=${procStatLine[2]}
IdleStat=${procStatLine[3]}
IOWaitStat=${procStatLine[4]}
IrqStat=${procStatLine[5]}
SoftIrqStat=${procStatLine[6]}
Total=0
for eachstat in ${procStatLine[@]}; do
Total=$(( ${Total} + ${eachstat} ))
done
UserDiff=$(( ${UserStat} - ${LastUserStat} ))
NiceDiff=$(( ${NiceStat} - ${LastNiceStat} ))
SystemDiff=$(( ${SystemStat} - ${LastSystemStat} ))
IOWaitDiff=$(( ${IOWaitStat} - ${LastIOWaitStat} ))
IrqDiff=$(( ${IrqStat} - ${LastIrqStat} ))
SoftIrqDiff=$(( ${SoftIrqStat} - ${LastSoftIrqStat} ))
diffIdle=$(( ${IdleStat} - ${LastIdleStat} ))
diffTotal=$(( ${Total} - ${LastTotal} ))
diffX=$(( ${diffTotal} - ${diffIdle} ))
CPULoad=$(( ${diffX}* 100 / ${diffTotal} ))
UserLoad=$(( ${UserDiff}* 100 / ${diffTotal} ))
SystemLoad=$(( ${SystemDiff}* 100 / ${diffTotal} ))
NiceLoad=$(( ${NiceDiff}* 100 / ${diffTotal} ))
IOWaitLoad=$(( ${IOWaitDiff}* 100 / ${diffTotal} ))
IrqCombined=$(( ${IrqDiff} + ${SoftIrqDiff} ))
IrqCombinedLoad=$(( ${IrqCombined}* 100 / ${diffTotal} ))
LastUserStat=${UserStat}
LastNiceStat=${NiceStat}
LastSystemStat=${SystemStat}
LastIdleStat=${IdleStat}
LastIOWaitStat=${IOWaitStat}
LastIrqStat=${IrqStat}
LastSoftIrqStat=${SoftIrqStat}
LastTotal=${Total}
procStats=$(echo -e "$(printf "%3s" ${CPULoad})%$(printf "%4s" ${SystemLoad})%$(printf "%4s" ${UserLoad})%$(printf "%4s" ${NiceLoad})%$(printf "%4s" ${IOWaitLoad})%$(printf "%4s" ${IrqCombinedLoad})%")
} # ProcessStats
CheckRelease() {
# Display warning when not executing on Debian Stretch or Ubuntu Bionic
Distro=$(lsb_release -c | awk -F" " '{print $2}' | tr '[:upper:]' '[:lower:]')
case ${Distro} in
stretch|bionic)
:
;;
*)
echo -e "${LRED}${BOLD}WARNING: this tool is meant to run only on Debian Stretch or Ubuntu Bionic.${NC}"
echo -e "When running on other distros results are partially meaningless or can't be collected.\nPress [ctrl]-[c] to stop or [enter] to continue."
read
;;
esac
} # CheckRelease
CheckLoad() {
# Only continue if average load is less than 0.1
AvgLoad1Min=$(awk -F" " '{print $1*100}' < /proc/loadavg)
[ $AvgLoad1Min -ge 10 ] && echo -e "\nAverage load is 0.1 or higher (way too much background activity). Waiting...\n"
while [ $AvgLoad1Min -ge 10 ]; do
echo -e "System too busy for benchmarking:$(uptime)"
sleep 5
AvgLoad1Min=$(awk -F" " '{print $1*100}' < /proc/loadavg)
done
echo ""
} # CheckLoad
BasicSetup() {
CPUArchitecture="$(lscpu | awk -F" " '/^Architecture/ {print $2}')"
case ${CPUArchitecture} in
arm*|aarch*)
[ -f /proc/device-tree/model ] && read DeviceName </proc/device-tree/model
# Try to switch to performance cpufreq governor on ARM with all CPU cores
CPUCores=$(grep -c '^processor' /proc/cpuinfo)
for ((i=0;i<CPUCores;i++)); do
echo performance >/sys/devices/system/cpu/cpu${i}/cpufreq/scaling_governor
done
;;
x86*|i686)
# Define CPUCores as 3 to prevent big.LITTLE handling and throttling reporting
CPUCores=3
# Try to get device name from CPU entry
DeviceName="$(lscpu | sed 's/ \{1,\}/ /g' | awk -F": " '/^Model name/ {print $2}')"
;;
*)
echo "${CPUArchitecture} not supported. Aborting." >&2
exit 1
;;
esac
} # BasicSetup
InstallPrerequisits() {
echo -e "sbc-bench v${Version}\n\nInstalling needed tools. This may take some time...\c"
SevenZip=$(which 7za || which 7zr)
[ -z "${SevenZip}" ] && apt -f -qq -y install p7zip >/dev/null 2>&1 && SevenZip=/usr/bin/7zr
[ -z "${SevenZip}" ] && (echo "No 7-zip binary found and could not be installed. Aborting" >&2 ; exit 1)
which iostat >/dev/null 2>&1 || apt -f -qq -y install sysstat >/dev/null 2>&1
which git >/dev/null 2>&1 || apt -f -qq -y install git >/dev/null 2>&1
which openssl >/dev/null 2>&1 || apt -f -qq -y install openssl >/dev/null 2>&1
which curl >/dev/null 2>&1 || apt -f -qq -y install curl >/dev/null 2>&1
# get/build tinymembench if not already there
[ -d "${InstallLocation}" ] || mkdir -p "${InstallLocation}"
if [ ! -x "${InstallLocation}"/tinymembench/tinymembench ]; then
cd "${InstallLocation}"
git clone https://github.com/ssvb/tinymembench >/dev/null 2>&1
cd tinymembench
make >/dev/null 2>&1
fi
# get/build mhz if not already there
if [ ! -x "${InstallLocation}"/mhz/mhz ]; then
cd "${InstallLocation}"
git clone http://git.1wt.eu/git/mhz.git/ >/dev/null 2>&1
cd mhz
make >/dev/null 2>&1
fi
# if called as 'sbc-bench neon' we also use cpuminer
if [ "${TestNEON}" = "yes" ]; then
InstallCpuminer >/dev/null 2>&1
fi
} # InstallPrerequisits
InstallCpuminer() {
# get/build cpuminer if not already there
if [ ! -x "${InstallLocation}"/cpuminer-multi/cpuminer ]; then
cd "${InstallLocation}"
apt -f -qq -y install automake autoconf pkg-config libcurl4-openssl-dev libjansson-dev libssl-dev libgmp-dev make g++ zlib1g-dev >/dev/null 2>&1
git clone https://github.com/tkinjo1985/cpuminer-multi.git
cd cpuminer-multi/
./build.sh
fi
if [ ! -x "${InstallLocation}"/cpuminer-multi/cpuminer ]; then
echo -e " (can't build cpuminer)\c"
fi
} # InstallCpuminer
InitialMonitoring() {
# log benchmark start in dmesg output
echo "sbc-bench started" >/dev/kmsg
# Create temporary files
TempDir="$(mktemp -d /tmp/${0##*/}.XXXXXX)"
TempLog="${TempDir}/temp.log"
ResultLog="${TempDir}/results.log"
MonitorLog="${TempDir}/monitor.log"
trap "rm -rf \"${TempDir}\" ; exit 0" 0 1 2 3 15
# Log version and device info
echo -e "sbc-bench v${Version} ${DeviceName} ($(date -R))\n" >${ResultLog}
# Log distribution info
[ -f /etc/armbian-release ] && . /etc/armbian-release
which lsb_release >/dev/null 2>&1 && (lsb_release -a 2>/dev/null) >>${ResultLog}
if [ -n "${BOARDFAMILY}" ]; then
echo -e "\nArmbian release info:\n$(grep -v "#" /etc/armbian-release)" >>${ResultLog}
else
echo -e "Architecture:\t$(dpkg --print-architecture)" >>${ResultLog}
fi
# On Raspberries we also collect 'firmware' information:
if [ -f /usr/bin/vcgencmd ]; then
echo -e "\nRaspberry Pi ThreadX version:\n$(/usr/bin/vcgencmd version)" >>${ResultLog}
[ -f /boot/config.txt ] && echo -e "\nThreadX configuration (/boot/config.txt):\n$(grep -v '#' /boot/config.txt | sed '/^\s*$/d')" >>${ResultLog}
echo -e "\nActual ThreadX settings:\n$(vcgencmd get_config int)" >>${ResultLog}
fi
# Log gcc version
echo -e "\n$(which gcc) $(gcc --version | sed 's/gcc\ //' | head -n1)" >>${ResultLog}
# Some basic system info needed to interpret system health later
echo -e "\nUptime:$(uptime)\n\n$(iostat)\n\n$(free -h)\n\n$(swapon -s)" >>${ResultLog}
} # InitialMonitoring
CheckClockspeeds() {
echo -e " Done.\nChecking cpufreq OPP...\c"
echo -e "\n##########################################################################" >>${ResultLog}
if [ -f ${MonitorLog} ]; then
# 2nd check after most demanding benchmark has been run.
echo -e "\nTesting clockspeeds again. System health now:\n" >>${ResultLog}
grep 'Time' ${MonitorLog} | tail -n 1 >"${TempDir}/systemhealth.now" >>${ResultLog}
grep ':' ${MonitorLog} | tail -n 1 >>"${TempDir}/systemhealth.now" >>${ResultLog}
fi
if [ "X${BOARDFAMILY}" = "Xs5p6818" -o ${CPUCores} -le 4 ]; then
# no big.LITTLE, checking cluster 0 is sufficient
echo -e "\nChecking cpufreq OPP:\n" >>${ResultLog}
CheckCPUCluster 0 >>${ResultLog}
else
# big.LITTLE or something else (Amlogic S912)
echo -e "\nChecking cpufreq OPP for cpu0-cpu3:\n" >>${ResultLog}
CheckCPUCluster 0 >>${ResultLog}
echo -e "\nChecking cpufreq OPP for cpu4-cpu$(( ${CPUCores} - 1 )):\n" >>${ResultLog}
CheckCPUCluster 4 >>${ResultLog}
fi
} # CheckClockspeeds
CheckTimeInState() {
# Check cpufreq statistics prior and after benchmark to detect throttling (won't work
# with all kernels and especially not on the RPi since RPi Trading people are cheating.
# Cpufreq support via sysfs is bogus and with latest ThreadX (firmware) update they
# even cheat wrt querying the 'firmware' via 'vcgencmd get_throttled':
# https://www.raspberrypi.org/forums/viewtopic.php?f=63&t=217056#p1334921
find /sys -type f -name time_in_state | sort | grep 'cpufreq' | while read ; do
StatFile="${REPLY}"
Number=$(echo ${StatFile} | tr -c -d '[:digit:]')
read MaxSpeed </sys/devices/system/cpu/cpu${Number}/cpufreq/cpuinfo_max_freq
sort -n <${StatFile} | while read ; do
Cpufreq=$(awk '{print $1}' <<<${REPLY})
if [ ${Cpufreq} -lt ${MaxSpeed} ]; then
echo ${REPLY} >>${TempDir}/time_in_state_${1}_${Number}
echo ${REPLY} >>${TempDir}/full_time_in_state_${1}_${Number}
elif [ ${Cpufreq} -le ${MaxSpeed} ]; then
echo ${REPLY} >>${TempDir}/full_time_in_state_${1}_${Number}
fi
done
done
} # CheckTimeInState
CheckCPUCluster() {
# check whether there's cpufreq support or not
if [ -d /sys/devices/system/cpu/cpu${1}/cpufreq ]; then
# walk through all cpufreq OPP and report clockspeeds (kernel vs. measured)
read MinSpeed </sys/devices/system/cpu/cpu${1}/cpufreq/cpuinfo_min_freq
read MaxSpeed </sys/devices/system/cpu/cpu${1}/cpufreq/cpuinfo_max_freq
echo ${MinSpeed} >/sys/devices/system/cpu/cpu${1}/cpufreq/scaling_min_freq
for i in $(tr " " "\n" </sys/devices/system/cpu/cpu${1}/cpufreq/scaling_available_frequencies | sort -n -r) ; do
echo ${i} >/sys/devices/system/cpu/cpu${1}/cpufreq/scaling_max_freq
sleep 0.1
RealSpeed=$(taskset -c $1 "${InstallLocation}"/mhz/mhz 3 100000 | awk -F" cpu_MHz=" '{print $2}' | awk -F" " '{print $1}' | tr '\n' '/' | sed 's|/$||')
SysfsSpeed=$(( $i / 1000 ))
if [ -f /usr/bin/vcgencmd ]; then
# On RPi we query ThreadX about clockspeeds too
ThreadXFreq=$(/usr/bin/vcgencmd measure_clock arm | awk -F"=" '{printf ("%0.0f",$2/1000000); }' )
echo -e "Cpufreq OPP: $(printf "%4s" ${SysfsSpeed}) ThreadX: $(printf "%4s" ${ThreadXFreq}) Measured: ${RealSpeed}"
else
echo -e "Cpufreq OPP: $(printf "%4s" ${SysfsSpeed}) Measured: ${RealSpeed}"
fi
done
echo ${MaxSpeed} >/sys/devices/system/cpu/cpu${1}/cpufreq/scaling_max_freq
else
# no cpufreq support
RealSpeed=$(taskset -c $(( $1 + 1 )) "${InstallLocation}"/mhz/mhz 3 100000 | awk -F" cpu_MHz=" '{print $2}' | awk -F" " '{print $1}' | tr '\n' '/' | sed 's|/$||')
echo -e "No cpufreq support available. Measured on cpu$(( $1 + 1 )): ${RealSpeed}"
fi
} # CheckCPUCluster
RunTinyMemBench() {
echo -e " Done.\nExecuting tinymembench. This will take a long time...\c"
echo -e "System health while running tinymembench:\n" >${MonitorLog}
/bin/bash "${PathToMe}" m 60 >>${MonitorLog} &
MonitoringPID=$!
if [ ${CPUCores} -gt 4 ]; then
if [ "X${BOARDFAMILY}" = "Xs5p6818" ]; then
# S5P6816 octa-core SoC is not a big.LITTLE design
"${InstallLocation}"/tinymembench/tinymembench >${TempLog} 2>&1
else
# big.LITTLE SoC, we execute one time on a little and one time on a big core
echo -e "Executing tinymembench on a little core:\n" >${TempLog}
taskset -c 0 "${InstallLocation}"/tinymembench/tinymembench >>${TempLog} 2>&1
echo -e "\nExecuting tinymembench on a big core:\n" >>${TempLog}
taskset -c $(( ${CPUCores} -1 )) "${InstallLocation}"/tinymembench/tinymembench >>${TempLog} 2>&1
fi
else
"${InstallLocation}"/tinymembench/tinymembench >${TempLog} 2>&1
fi
kill ${MonitoringPID}
echo -e "\n##########################################################################\n" >>${ResultLog}
cat ${TempLog} >>${ResultLog}
} # RunTinyMemBench
Run7ZipBenchmark() {
echo -e " Done.\nExecuting 7-zip benchmark. This will take a long time...\c"
echo -e "\nSystem health while running 7-zip single core benchmark:\n" >>${MonitorLog}
echo -e "\c" >${TempLog}
/bin/bash "${PathToMe}" m 30 >>${MonitorLog} &
MonitoringPID=$!
if [ ${CPUCores} -eq 1 ]; then
# Do not measure single threaded result since useless
:
elif [ ${CPUCores} -gt 4 ]; then
if [ "X${BOARDFAMILY}" = "Xs5p6818" ]; then
# S5P6816 octa-core SoC is not a big.LITTLE design
taskset -c 0 "${SevenZip}" b -mmt 1 >>${TempLog}
else
# big.LITTLE SoC, we execute one time on a little and one time on a big core
taskset -c 0 "${SevenZip}" b -mmt 1 >>${TempLog}
taskset -c $(( ${CPUCores} - 1 )) "${SevenZip}" b -mmt 1 >>${TempLog}
fi
else
taskset -c 0 "${SevenZip}" b -mmt 1 >>${TempLog}
fi
kill ${MonitoringPID}
echo -e "\n##########################################################################" >>${ResultLog}
cat ${TempLog} >>${ResultLog}
echo -e "\nSystem health while running 7-zip multi core benchmark:\n" >>${MonitorLog}
echo -e "\c" >${TempLog}
/bin/bash "${PathToMe}" m 30 >>${MonitorLog} &
MonitoringPID=$!
RunHowManyTimes=3
for ((i=1;i<=RunHowManyTimes;i++)); do
"${SevenZip}" b >>${TempLog}
done
kill ${MonitoringPID}
echo -e "\n##########################################################################" >>${ResultLog}
cat ${TempLog} >>${ResultLog}
sed -i 's/|//' ${TempLog}
CompScore=$(awk -F" " '/^Avr:/ {print $4}' <${TempLog} | tr '\n' ', ' | sed 's/,$//')
DecompScore=$(awk -F" " '/^Avr:/ {print $7}' <${TempLog} | tr '\n' ', ' | sed 's/,$//')
TotScore=$(awk -F" " '/^Tot:/ {print $4}' <${TempLog} | tr '\n' ', ' | sed 's/,$//')
echo -e "\nCompression: ${CompScore}" >>${ResultLog}
echo -e "Decompression: ${DecompScore}" >>${ResultLog}
echo -e "Total: ${TotScore}" >>${ResultLog}
} # Run7ZipBenchmark
RunOpenSSLBenchmark() {
echo -e " Done.\nExecuting OpenSSL benchmark. This will take 3 minutes...\c"
echo -e "\nSystem health while running OpenSSL benchmark:\n" >>${MonitorLog}
/bin/bash "${PathToMe}" m 10 >>${MonitorLog} &
MonitoringPID=$!
OpenSSLLog="${TempDir}/openssl.log"
for i in 128 192 256 ; do
if [ ${CPUCores} -gt 4 ]; then
# big.LITTLE SoC, we execute one time on a little and one time on a big core
taskset -c 0 openssl speed -elapsed -evp aes-${i}-cbc 2>/dev/null
taskset -c $(( ${CPUCores} -1 )) openssl speed -elapsed -evp aes-${i}-cbc 2>/dev/null
else
openssl speed -elapsed -evp aes-${i}-cbc 2>/dev/null
openssl speed -elapsed -evp aes-${i}-cbc 2>/dev/null
fi
done >${OpenSSLLog}
kill ${MonitoringPID}
echo -e "\n##########################################################################\n" >>${ResultLog}
echo -e "$(openssl version | awk -F" " '{print $1" "$2", built on "$3" "$4" "$5" "$6" "$7" "$8" "$9" "$10" "$11" "$12" "$13" "$14" "$15}' | sed 's/ *$//')\n$(grep '^type' ${OpenSSLLog} | head -n1)" >>${ResultLog}
grep '^aes-' ${OpenSSLLog} >>${ResultLog}
} # RunOpenSSLBenchmark
RunCpuminerBenchmark() {
echo -e " Done.\nExecuting cpuminer. This will take 5 minutes...\c"
echo -e "\nSystem health while running cpuminer:\n" >>${MonitorLog}
/bin/bash "${PathToMe}" m 10 >>${MonitorLog} &
MonitoringPID=$!
"${InstallLocation}"/cpuminer-multi/cpuminer --benchmark --cpu-priority=2 >${TempLog} &
MinerPID=$!
sleep 300
kill ${MinerPID} ${MonitoringPID}
echo -e "\n##########################################################################\n" >>${ResultLog}
sed "s,\x1B\[[0-9;]*[a-zA-Z],,g" <${TempLog} >>${ResultLog}
# Summarized 'Total:' scores
TotalScores="$(awk -F" " '/Total:/ {print $4}' ${TempLog} | sort -r -n | uniq | tr '\n' ', ' | sed 's/,$//')"
echo -e "\nTotal Scores: ${TotalScores}" >>${ResultLog}
} # RunCpuminerBenchmark
DisplayResults() {
echo -e " Done.\n\007\007\007"
# Check for throttling on normal ARM SBC
find /sys -type f -name time_in_state | sort | grep 'cpufreq' | while read ; do
Number=$(echo ${REPLY} | tr -c -d '[:digit:]')
diff ${TempDir}/time_in_state_after_${Number} ${TempDir}/time_in_state_before_${Number} >/dev/null 2>&1
if [ $? -ne 0 ]; then
case ${CPUCores} in
1|2|4)
ReportCpufreqStatistics ${Number}
echo -e "${LRED}${BOLD}ATTENTION: Throttling might have occured. Check the uploaded log for details.${NC}\n"
;;
6)
if [ ${Number} -eq 0 ]; then
ReportCpufreqStatistics ${Number} " for CPUs 0-3"
echo -e "${LRED}${BOLD}ATTENTION: Throttling might have occured on CPUs 0-3. Check the uploaded log for details.${NC}\n"
else
ReportCpufreqStatistics ${Number} " for CPUs 4-5"
echo -e "${LRED}${BOLD}ATTENTION: Throttling might have occured on CPUs 4-5. Check the uploaded log for details.${NC}\n"
fi
;;
8)
if [ "X${BOARDFAMILY}" = "Xs5p6818" ]; then
ReportCpufreqStatistics ${Number}
echo -e "${LRED}${BOLD}ATTENTION: Throttling might have occured. Check the uploaded log for details.${NC}\n"
elif [ ${Number} -eq 0 ]; then
ReportCpufreqStatistics ${Number} " for CPUs 0-3"
echo -e "${LRED}${BOLD}ATTENTION: Throttling might have occured on CPUs 0-3. Check the uploaded log for details.${NC}\n"
else
ReportCpufreqStatistics ${Number} " for CPUs 4-7"
echo -e "${LRED}${BOLD}ATTENTION: Throttling might have occured on CPUs 4-7. Check the uploaded log for details.${NC}\n"
fi
;;
esac
fi
done
# Check for killed CPU cores. Some unfortunate users might still use Allwinner BSP kernels
CPUCoresNow=$(grep -c '^processor' /proc/cpuinfo)
if [ ${CPUCoresNow} -lt ${CPUCores} ]; then
echo -e "${LRED}${BOLD}ATTENTION: Due to overheating prevention $(( ${CPUCores} - ${CPUCoresNow} )) CPU cores have been killed. Check the uploaded log for details.${NC}\n"
fi
# Check for throttling/undervoltage on Raspberry Pi
grep -q '/1200MHz' ${MonitorLog} && Warning="ATTENTION: Silent throttling has occured. Check the uploaded log for details."
if [ -f /usr/bin/vcgencmd ]; then
Health="$(perl -e "printf \"%19b\n\", $(/usr/bin/vcgencmd get_throttled | cut -f2 -d=)" | tr -d '[:blank:]')"
# https://forum.armbian.com/topic/7763-benchmarking-cpus/?do=findComment&comment=59042
HealthLength=$(wc -c <<<"${Health}")
[ ${HealthLength} -eq 19 ] && Health="0${Health}"
case ${Health} in
10*)
Warning="ATTENTION: Frequency capping to 600 MHz has occured. Check the uploaded log for details."
ReportRPiHealth ${Health} >>${TempDir}/throttling_info.txt
;;
01*)
Warning="ATTENTION: Throttling has occured. Check the uploaded log for details."
ReportRPiHealth ${Health} >>${TempDir}/throttling_info.txt
;;
11*)
Warning="ATTENTION: Throttling and frequency capping has occured. Check the uploaded log for details."
ReportRPiHealth ${Health} >>${TempDir}/throttling_info.txt
;;
esac
if [ "X${Warning}" = "X" ]; then
echo -e "${LGREEN}It seems neither throttling nor frequency capping has occured.${NC}\n"
else
echo -e "${LRED}${BOLD}${Warning}${NC}\n"
fi
fi
# Prepare benchmark results
echo -e "\n##########################################################################\n" >>${ResultLog}
cat ${MonitorLog} >>${ResultLog}
[ -f ${TempDir}/throttling_info.txt ] && cat ${TempDir}/throttling_info.txt >>${ResultLog}
# add dmesg output since start of the benchmark if something relevant is there
TimeStamp="$(dmesg | tr -d '[' | tr -d ']' | awk -F" " '/sbc-bench started/ {print $1}' | tail -n1)"
dmesg | sed "/${TimeStamp}/,\$!d" | grep -v 'sbc-bench started' >"${TempDir}/dmesg"
if [ -s "${TempDir}/dmesg" ]; then
echo -e "\n##########################################################################\n\ndmesg output while running the benchmarks:\n" >>${ResultLog}
cat "${TempDir}/dmesg" >>${ResultLog}
fi
echo -e "\n##########################################################################\n" >>${ResultLog}
echo -e "$(iostat)\n\n$(free -h)\n\n$(swapon -s)\n\n$(lscpu)" >>${ResultLog}
ReportCacheSizes >>${ResultLog}
UploadURL=$(curl -s -F 'f:1=<-' ix.io <${ResultLog} 2>/dev/null || curl -s -F 'f:1=<-' ix.io <${ResultLog})
# Display benchmark results
[ ${CPUCores} -gt 4 ] && BigLittle=" (big.LITTLE cores measured individually)"
echo -e "${BOLD}Memory performance${NC}${BigLittle}:"
awk -F" " '/^ standard/ {print $2": "$4" "$5" "$6}' <${ResultLog}
if [ "${TestNEON}" = "yes" -a -x "${InstallLocation}"/cpuminer-multi/cpuminer ]; then
echo -e "\n${BOLD}Cpuminer total scores${NC} (5 minutes execution): $(awk -F"Total Scores: " '/^Total Scores: / {print $2}' ${ResultLog}) kH/s"
fi
echo -e "\n${BOLD}7-zip total scores${NC} (3 consecutive runs): $(awk -F" " '/^Total:/ {print $2}' ${ResultLog})"
echo -e "\n${BOLD}OpenSSL results${NC}${BigLittle}:\n$(grep '^type' ${OpenSSLLog} | head -n1)"
grep '^aes-' ${OpenSSLLog}
if [ "X${UploadURL}" = "X" ]; then
echo -e "\nUnable to upload full test results. Please copy&paste the below stuff to pastebin.com and\nprovide the URL. Check the output for throttling and swapping please.\n\n"
cat ${ResultLog}
echo -e "\n"
else
echo -e "\nFull results uploaded to ${UploadURL}. Please check the log for anomalies (e.g. swapping\nor throttling happenend) and otherwise share this URL.\n"
fi
} # DisplayResults
ReportCpufreqStatistics() {
# Displays cpufreq driver statistics from before and after the benchmark as comparison
if [ -f ${TempDir}/full_time_in_state_before_${1} ]; then
echo -e "\nThrottling statistics (time spent on each cpufreq OPP)${2}:\n" >>${TempDir}/throttling_info.txt
awk -F" " '{print $1}' <${TempDir}/full_time_in_state_before_${1} | sort -r -n | while read ; do
BeforeValue=$(awk -F" " "/^${REPLY}/ {print \$2}" <${TempDir}/full_time_in_state_before_${1})
AfterValue=$(awk -F" " "/^${REPLY}/ {print \$2}" <${TempDir}/full_time_in_state_after_${1})
if [ ${AfterValue} -eq ${BeforeValue} ]; then
Duration=0
else
Duration=$(awk '{printf ("%0.2f",$1/100); }' <<<$(( ${AfterValue} - ${BeforeValue} )) )
fi
echo -e "$(printf "%4s" $(( ${REPLY} / 1000 )) ) MHz: $(printf "%7s" ${Duration}) sec" >>${TempDir}/throttling_info.txt
done
fi
} # ReportCpufreqStatistics
ReportRPiHealth() {
# Displaying the 'vcgencmd get_throttled' output in an understandable way
echo -e "\nQuerying ThreadX on RPi for thermal or undervoltage issues:\n\n${1}"
cat <<-EOF
||| |||_ under-voltage
||| ||_ currently throttled
||| |_ arm frequency capped
|||_ under-voltage has occurred since last reboot
||_ throttling has occurred since last reboot
|_ arm frequency capped has occurred since last reboot
EOF
} # ReportRPiHealth
ReportCacheSizes() {
find /sys/devices/system/cpu -name "cache" -type d | sort -n | while read ; do
find "${REPLY}" -name size -type f | while read ; do
echo -e "\n${REPLY}: $(cat ${REPLY})\c"
[ -f ${REPLY%/*}/level ] && echo -e ", level: $(cat ${REPLY%/*}/level)\c"
[ -f ${REPLY%/*}/type ] && echo -e ", type: $(cat ${REPLY%/*}/type)\c"
done
done
} # ReportCacheSizes
Main "$@"