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
|
parameters:
- name: suiteName
type: string
- name: testOS
type: string
values:
- ubuntu1804
- ubuntu2004
- ubuntu2204
- name: toolset
type: string
values:
- make
- name: runPyTest
type: boolean
- name: runSystemImageTest
type: boolean
- name: runDevTest
type: boolean
- name: master_clock_rate
type: string
default: ''
- name: dutMatrix
type: object
- name: uhdArtifactSource
type: string
default: current
- name: xilinxLocation
type: string
- name: extra_rf_test_args
type: string
default: ''
- name: testLength
type: string
default: 'smoke'
- name: warmUpAfterBoot
type: number
default: 0
- name: timeoutInMinutes
type: number
default: 120
jobs:
- job: x4xx_hardware_test_${{ parameters.suiteName }}
displayName: x4xx hardware test ${{ parameters.suiteName }}
timeoutInMinutes: ${{ parameters.timeoutInMinutes }}
strategy:
matrix: ${{ parameters.dutMatrix }}
pool:
name: 'de-dre-lab'
demands: uhd_ats -equals $(pipelineAgent)
steps:
- checkout: self
- checkout: meta-ettus
- checkout: ettus-rts
- script: |
if [ "${{ parameters.uhdArtifactSource }}" = "current" ]; then
echo '##vso[task.setvariable variable=uhd_artifact_directory]'$(Agent.BuildDirectory)
else
echo '##vso[task.setvariable variable=uhd_artifact_directory]'$(Agent.BuildDirectory)/${{ parameters.uhdArtifactSource }}
fi
displayName: Set uhddev pipeline artifact location
- script: |
rm -rf $(uhd_artifact_directory)/$(dutEmbeddedImagesArtifact)
rm -rf $(uhd_artifact_directory)/uhddev-${{ parameters.testOS }}-${{ parameters.toolset }}
rm -rf $(uhd_artifact_directory)/gnuradio-${{ parameters.testOS }}-${{ parameters.toolset }}
displayName: Cleanup from prior run
- download: ${{ parameters.uhdArtifactSource }}
artifact: $(dutEmbeddedImagesArtifact)
displayName: Download $(dutEmbeddedImagesArtifact)
- task: ExtractFiles@1
inputs:
archiveFilePatterns: $(uhd_artifact_directory)/$(dutEmbeddedImagesArtifact)/u-boot-jtag-files.zip
destinationFolder: $(uhd_artifact_directory)/$(dutEmbeddedImagesArtifact)
cleanDestinationFolder: false
overwriteExistingFiles: true
displayName: Extract JTAG artifact
- download: ${{ parameters.uhdArtifactSource }}
artifact: uhddev-${{ parameters.testOS }}-${{ parameters.toolset }}
displayName: Download uhddev-${{ parameters.testOS }}-${{ parameters.toolset }} artifact
- task: ExtractFiles@1
inputs:
archiveFilePatterns: $(uhd_artifact_directory)/uhddev-${{ parameters.testOS }}-${{ parameters.toolset }}/uhddev-${{ parameters.testOS }}-${{ parameters.toolset }}.tar.gz
destinationFolder: $(uhd_artifact_directory)/uhddev-${{ parameters.testOS }}-${{ parameters.toolset }}
cleanDestinationFolder: false
overwriteExistingFiles: true
displayName: Extract uhddev-${{ parameters.testOS }}-${{ parameters.toolset }} artifact
- download: ${{ parameters.uhdArtifactSource }}
artifact: gnuradio-grettus-${{ parameters.testOS }}-${{ parameters.toolset }}
displayName: Download gnuradio-grettus-${{ parameters.testOS }}-${{ parameters.toolset }} artifact
- task: ExtractFiles@1
inputs:
archiveFilePatterns: $(uhd_artifact_directory)/gnuradio-grettus-${{ parameters.testOS }}-${{ parameters.toolset }}/gnuradio-grettus-${{ parameters.testOS }}-${{ parameters.toolset }}.tar.gz
destinationFolder: $(uhd_artifact_directory)/gnuradio-grettus-${{ parameters.testOS }}-${{ parameters.toolset }}
cleanDestinationFolder: false
overwriteExistingFiles: true
displayName: Extract gnuradio-grettus-${{ parameters.testOS }}-${{ parameters.toolset }} artifact
- script: |
cd $(Agent.TempDirectory)
python3 -m venv venv
source venv/bin/activate
echo '##vso[task.setvariable variable=USRP_EMB_VENV]'$(realpath venv)
echo '##vso[task.setvariable variable=USRP_EMB_IMAGE_PATH]'$(uhd_artifact_directory)/$(dutEmbeddedImagesArtifact)
echo '##vso[task.setvariable variable=USRP_EMB_FTDI_SERIAL]'$(uartSerial)
python3 -m pip install -U pip
pip3 install wheel
pip3 install --upgrade $(Build.SourcesDirectory)/meta-ettus-dev/contrib/test/usrp_emb
displayName: Create usrp_emb venv
- script: |
source ${{ parameters.xilinxLocation }}/settings64.sh
source $USRP_EMB_VENV/bin/activate
usrp_emb_x4xx_flash_emmc
displayName: Flash eMMC
- script: |
source $USRP_EMB_VENV/bin/activate
mkdir -p $(Common.TestResultsDirectory)/sysimg_tests/sysimg_boot_tests
cd $(Common.TestResultsDirectory)/sysimg_tests/sysimg_boot_tests
usrp_emb_test_x4xx
displayName: Run system image boot tests
condition: and(succeeded(), ${{ parameters.runSystemImageTest }})
- script: |
source $USRP_EMB_VENV/bin/activate
USRP_EMB_TARGET_IP=$(usrp_emb_x4xx_boot_linux)
echo Target IP is $USRP_EMB_TARGET_IP
echo '##vso[task.setvariable variable=USRP_EMB_TARGET_IP]'$USRP_EMB_TARGET_IP
displayName: Boot target, get mgmt IP
# Setting the timezone on the embedded device is
# required to make the test runtimes to look sane.
- script: |
export HOST_TIME_ZONE=$(cat /etc/timezone)
ssh-keygen -f ~/.ssh/known_hosts -R $USRP_EMB_TARGET_IP
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "timedatectl set-timezone $HOST_TIME_ZONE"
displayName: Set timezone to Host
- script: |
sleep 60
displayName: Wait for device to finish booting
- script: |
ssh-keygen -f ~/.ssh/known_hosts -R $USRP_EMB_TARGET_IP
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "uhd_image_loader --args addr=localhost,type=$(dutType),fpga=$(dutFPGA)"
EXITCODE=$?
sleep 60
exit $EXITCODE
displayName: Flash FPGA $(dutFPGA)
# When MPM reboots, the sfp0 address is reset to default
# Since rebooting MPM was added for clocking re-config we need
# to change sfp0 default on the device
# after the DUT responds to pings, give the device additional time for
# things like the RPC server to come up too
- script: |
source $USRP_EMB_VENV/bin/activate
ssh-keygen -f ~/.ssh/known_hosts -R $USRP_EMB_TARGET_IP
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "sed -i 's/192.168.10.2/$(sfp0ip)/g' /data/network/sfp0.network"
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "reboot"
timeout=240
echo "Waiting ${timeout}s for sfp0 at ip $(sfp0ip) to be reachable from host..."
wait_counter=0 ; until $(ping -c 1 -W 1 $(sfp0ip) &>/dev/null) ; do if [ $wait_counter -eq $timeout ]; then echo "Device with ip $(sfp0ip) did not start up within $timeout seconds!" ;break; fi; wait_counter=$((wait_counter+1)); done
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "ifconfig sfp0"
sleep 90
displayName: Set sfp0 IP
condition: and(succeeded(), ${{ parameters.runPyTest }})
- script: |
mkdir -p $(Common.TestResultsDirectory)/sysimg_tests/sysimg_subsystem_tests
cd $(Common.TestResultsDirectory)/sysimg_tests/sysimg_subsystem_tests
ssh-keygen -f ~/.ssh/known_hosts -R $USRP_EMB_TARGET_IP
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "rm -rf ~/usrp_sysimg_test && mkdir -p ~/usrp_sysimg_test"
scp -o StrictHostKeyChecking=no -r $(Build.SourcesDirectory)/meta-ettus-dev/contrib/test/tests root@$USRP_EMB_TARGET_IP:~/usrp_sysimg_test
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "pip3 install unittest-xml-reporting && cd ~/usrp_sysimg_test/tests && python3 -m xmlrunner discover . -v"
EXITCODE=$?
scp -o StrictHostKeyChecking=no -r root@$USRP_EMB_TARGET_IP:~/usrp_sysimg_test .
exit $EXITCODE
displayName: Run system image embedded subsystems tests
continueOnError: true
condition: and(succeeded(), ${{ parameters.runSystemImageTest }})
- script: |
mkdir -p $(Common.TestResultsDirectory)/mpm/mpm_unit_tests
cd $(Common.TestResultsDirectory)/mpm/mpm_unit_tests
ssh-keygen -f ~/.ssh/known_hosts -R $USRP_EMB_TARGET_IP
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "rm -rf ~/mpm_test_run && mkdir -p ~/mpm_test_run"
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "cd ~/mpm_test_run && python3 /usr/lib/usrp_mpm/tests/run_unit_tests.py -x x4xx"
EXITCODE=$?
scp -o StrictHostKeyChecking=no -r root@$USRP_EMB_TARGET_IP:~/mpm_test_run .
exit $EXITCODE
displayName: Run MPM unit tests
continueOnError: true
- script: |
mkdir -p $(Common.TestResultsDirectory)/devtest
cd $(Common.TestResultsDirectory)/devtest
if [ -z "$(master_clock_rate)" ]; then
args="addr=$USRP_EMB_TARGET_IP,type=$(dutType)"
else
args="addr=$USRP_EMB_TARGET_IP,type=$(dutType),master_clock_rate=$(master_clock_rate)"
fi
echo $args
python3 $(Build.SourcesDirectory)/uhddev/host/tests/devtest/run_testsuite.py \
--src-dir $(Build.SourcesDirectory)/uhddev/host/tests/devtest \
--devtest-pattern $(devtestPattern) --args $args \
--build-dir $(uhd_artifact_directory)/uhddev-${{ parameters.testOS }}-${{ parameters.toolset }}/uhddev/build \
--build-type Release --python-interp python3 --xml
displayName: Run devtest
continueOnError: true
condition: and(succeeded(), ${{ parameters.runDevTest }})
- script: |
cd $(Common.TestResultsDirectory)/devtest
python3 $(Build.SourcesDirectory)/uhddev/.ci/utils/format_devtest_junitxml.py \
$(Common.TestResultsDirectory)/devtest \
$(Common.TestResultsDirectory)/devtest/devtestresults.xml
displayName: Format devtest results
continueOnError: true
condition: and(succeeded(), ${{ parameters.runDevTest }})
- script: |
cat $(Common.TestResultsDirectory)/devtest/log*.log
echo ""
displayName: cat devtest log
condition: and(succeeded(), ${{ parameters.runDevTest }})
# warmUpAfterBoot optional parameter added to allow device to warm up before running tests
# For example, the phase test pass/fail requirements are based on declared specs measured after a warm up to 23 deg C
# https://www.ni.com/docs/en-US/bundle/ettus-usrp-x440-specs/page/specs.html#d668e1387
- script: |
export UHD_INSTALL_PATH=$(uhd_artifact_directory)/uhddev-${{ parameters.testOS }}-${{ parameters.toolset }}/uhddev/build-installed
export GNURADIO_GRETTUS_INSTALL_PATH=$(uhd_artifact_directory)/gnuradio-grettus-${{ parameters.testOS }}-${{ parameters.toolset }}/gnuradio/build-installed
export PATH=$UHD_INSTALL_PATH/bin:$PATH
export PATH=$GNURADIO_GRETTUS_INSTALL_PATH/bin:$PATH
export PYTHON_VERSION=$(python3 -c "import sys; print('python'+str(sys.version_info.major) + '.' + str(sys.version_info.minor))")
if [ -d "$UHD_INSTALL_PATH/lib64" ]; then
export PATH=$UHD_INSTALL_PATH/lib64/uhd/examples:$PATH
export LD_LIBRARY_PATH=$UHD_INSTALL_PATH/lib64:$LD_LIBRARY_PATH
export LD_LIBRARY_PATH=$GNURADIO_GRETTUS_INSTALL_PATH/lib64:$LD_LIBRARY_PATH
export PYTHONPATH=$UHD_INSTALL_PATH/lib64/$PYTHON_VERSION/site-packages:$PYTHONPATH
export PYTHONPATH=$GNURADIO_GRETTUS_INSTALL_PATH/lib64/python3/dist-packages:$PYTHONPATH
else
export PATH=$UHD_INSTALL_PATH/lib/uhd/examples:$PATH
export LD_LIBRARY_PATH=$UHD_INSTALL_PATH/lib:$LD_LIBRARY_PATH
export LD_LIBRARY_PATH=$GNURADIO_GRETTUS_INSTALL_PATH/lib:$LD_LIBRARY_PATH
export PYTHONPATH=$UHD_INSTALL_PATH/lib/$PYTHON_VERSION/site-packages:$PYTHONPATH
export PYTHONPATH=$GNURADIO_GRETTUS_INSTALL_PATH/lib/python3/dist-packages:$PYTHONPATH
fi
cd $(Build.SourcesDirectory)/ettus-rts/config/remote/python_tests
if [ ${{ parameters.warmUpAfterBoot }} > 0 ]; then
echo 'sleeping ${{ parameters.warmUpAfterBoot }} seconds before testing...'
sleep ${{ parameters.warmUpAfterBoot }}
fi
uhd_find_devices --args type=$(dutType)
python3 automated_main.py --ats_config $(pytestAtsConfig) --dut $(pytestDUT) --results_path '$(Common.TestResultsDirectory)/pytest/host' --test_length ${{ parameters.testLength }} ${{ parameters.extra_rf_test_args }}
displayName: Run pytest on host
continueOnError: true
condition: and(succeeded(), ${{ parameters.runPyTest }})
- script: |
mkdir -p $(Common.TestResultsDirectory)/pytest/embedded
cd $(Common.TestResultsDirectory)/pytest/embedded
ssh-keygen -f ~/.ssh/known_hosts -R $USRP_EMB_TARGET_IP
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "pip3 install --upgrade pytest pytest-helpers-namespace pytest-tempdir pytest-salt"
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "rm -rf ~/usrp_pytest && mkdir -p ~/usrp_pytest"
scp -o StrictHostKeyChecking=no -r $(Build.SourcesDirectory)/ettus-rts/config/remote/ root@$USRP_EMB_TARGET_IP:~/usrp_pytest
ssh -o StrictHostKeyChecking=no -tt root@$USRP_EMB_TARGET_IP "cd ~/usrp_pytest/remote/python_tests && python3 automated_main.py --ats_config $(pytestAtsConfig) --dut $(pytestDUT) --test_selector pytests/test_gnuradio_sanity.py --run_embedded"
EXITCODE=$?
scp -o StrictHostKeyChecking=no -r root@$USRP_EMB_TARGET_IP:~/usrp_pytest .
exit $EXITCODE
displayName: Run pytest on embedded
continueOnError: true
condition: and(succeeded(), ${{ parameters.runPyTest }})
# Run the mender tests last because they
# modify the flashed image on the device
- script: |
source $USRP_EMB_VENV/bin/activate
mkdir -p $(Common.TestResultsDirectory)/sysimg_tests/sysimg_mender_tests
cd $(Common.TestResultsDirectory)/sysimg_tests/sysimg_mender_tests
source $USRP_EMB_VENV/bin/activate
usrp_emb_test_x4xx_mender
displayName: Run system image mender tests
continueOnError: true
condition: and(succeeded(), ${{ parameters.runSystemImageTest }})
- task: PublishTestResults@2
inputs:
testResultsFormat: 'JUnit'
testResultsFiles: '$(Common.TestResultsDirectory)/sysimg_tests/**/*.xml'
testRunTitle: $(dutName) System Image Tests
mergeTestResults: true
failTaskOnFailedTests: true
displayName: Upload $(dutName) System Image Tests
condition: ${{ parameters.runSystemImageTest }}
- task: PublishTestResults@2
inputs:
testResultsFormat: 'JUnit'
testResultsFiles: '$(Common.TestResultsDirectory)/mpm/**/*.xml'
testRunTitle: $(dutName) mpm unit tests
mergeTestResults: true
failTaskOnFailedTests: true
displayName: Upload $(dutName) mpm unit tests
condition: always()
- task: PublishTestResults@2
inputs:
testResultsFormat: 'JUnit'
testResultsFiles: '$(Common.TestResultsDirectory)/devtest/devtestresults.xml'
testRunTitle: $(dutName) devtests
buildConfiguration: 'Release'
mergeTestResults: true
failTaskOnFailedTests: true
displayName: Upload $(dutName) devtests
condition: ${{ parameters.runDevTest }}
- task: PublishTestResults@2
inputs:
testResultsFormat: 'JUnit'
testResultsFiles: '$(Common.TestResultsDirectory)/pytest/host/**/*.xml'
testRunTitle: $(dutName) host pytests
mergeTestResults: true
failTaskOnFailedTests: true
displayName: Upload $(dutName) host pytests
condition: ${{ parameters.runPyTest }}
- task: PublishTestResults@2
inputs:
testResultsFormat: 'JUnit'
testResultsFiles: '$(Common.TestResultsDirectory)/pytest/embedded/**/*.xml'
testRunTitle: $(dutName) embedded pytests
mergeTestResults: true
failTaskOnFailedTests: true
displayName: Upload $(dutName) embedded pytests
condition: ${{ parameters.runPyTest }}
- publish: $(Common.TestResultsDirectory)
artifact: test-logs-x4xx-hardware-$(dutName)-run$(System.JobAttempt)
displayName: Upload Test Logs
condition: always()
- script: |
echo "A previous step failed. See steps with warnings."
echo "Marking the overall build status as a failure."
exit 1
condition: in(variables['Agent.JobStatus'], 'SucceededWithIssues')
displayName: Mark build as failure
|