1*3a37aee3SSimon Glass# Copyright (c) 2013, Google Inc. 2*3a37aee3SSimon Glass# 3*3a37aee3SSimon Glass# SPDX-License-Identifier: GPL-2.0+ 4*3a37aee3SSimon Glass# 5*3a37aee3SSimon Glass# Sanity check of the FIT handling in U-Boot 6*3a37aee3SSimon Glass 7*3a37aee3SSimon Glassimport os 8*3a37aee3SSimon Glassimport pytest 9*3a37aee3SSimon Glassimport struct 10*3a37aee3SSimon Glassimport u_boot_utils as util 11*3a37aee3SSimon Glass 12*3a37aee3SSimon Glass# Define a base ITS which we can adjust using % and a dictionary 13*3a37aee3SSimon Glassbase_its = ''' 14*3a37aee3SSimon Glass/dts-v1/; 15*3a37aee3SSimon Glass 16*3a37aee3SSimon Glass/ { 17*3a37aee3SSimon Glass description = "Chrome OS kernel image with one or more FDT blobs"; 18*3a37aee3SSimon Glass #address-cells = <1>; 19*3a37aee3SSimon Glass 20*3a37aee3SSimon Glass images { 21*3a37aee3SSimon Glass kernel@1 { 22*3a37aee3SSimon Glass data = /incbin/("%(kernel)s"); 23*3a37aee3SSimon Glass type = "kernel"; 24*3a37aee3SSimon Glass arch = "sandbox"; 25*3a37aee3SSimon Glass os = "linux"; 26*3a37aee3SSimon Glass compression = "none"; 27*3a37aee3SSimon Glass load = <0x40000>; 28*3a37aee3SSimon Glass entry = <0x8>; 29*3a37aee3SSimon Glass }; 30*3a37aee3SSimon Glass kernel@2 { 31*3a37aee3SSimon Glass data = /incbin/("%(loadables1)s"); 32*3a37aee3SSimon Glass type = "kernel"; 33*3a37aee3SSimon Glass arch = "sandbox"; 34*3a37aee3SSimon Glass os = "linux"; 35*3a37aee3SSimon Glass compression = "none"; 36*3a37aee3SSimon Glass %(loadables1_load)s 37*3a37aee3SSimon Glass entry = <0x0>; 38*3a37aee3SSimon Glass }; 39*3a37aee3SSimon Glass fdt@1 { 40*3a37aee3SSimon Glass description = "snow"; 41*3a37aee3SSimon Glass data = /incbin/("u-boot.dtb"); 42*3a37aee3SSimon Glass type = "flat_dt"; 43*3a37aee3SSimon Glass arch = "sandbox"; 44*3a37aee3SSimon Glass %(fdt_load)s 45*3a37aee3SSimon Glass compression = "none"; 46*3a37aee3SSimon Glass signature@1 { 47*3a37aee3SSimon Glass algo = "sha1,rsa2048"; 48*3a37aee3SSimon Glass key-name-hint = "dev"; 49*3a37aee3SSimon Glass }; 50*3a37aee3SSimon Glass }; 51*3a37aee3SSimon Glass ramdisk@1 { 52*3a37aee3SSimon Glass description = "snow"; 53*3a37aee3SSimon Glass data = /incbin/("%(ramdisk)s"); 54*3a37aee3SSimon Glass type = "ramdisk"; 55*3a37aee3SSimon Glass arch = "sandbox"; 56*3a37aee3SSimon Glass os = "linux"; 57*3a37aee3SSimon Glass %(ramdisk_load)s 58*3a37aee3SSimon Glass compression = "none"; 59*3a37aee3SSimon Glass }; 60*3a37aee3SSimon Glass ramdisk@2 { 61*3a37aee3SSimon Glass description = "snow"; 62*3a37aee3SSimon Glass data = /incbin/("%(loadables2)s"); 63*3a37aee3SSimon Glass type = "ramdisk"; 64*3a37aee3SSimon Glass arch = "sandbox"; 65*3a37aee3SSimon Glass os = "linux"; 66*3a37aee3SSimon Glass %(loadables2_load)s 67*3a37aee3SSimon Glass compression = "none"; 68*3a37aee3SSimon Glass }; 69*3a37aee3SSimon Glass }; 70*3a37aee3SSimon Glass configurations { 71*3a37aee3SSimon Glass default = "conf@1"; 72*3a37aee3SSimon Glass conf@1 { 73*3a37aee3SSimon Glass kernel = "kernel@1"; 74*3a37aee3SSimon Glass fdt = "fdt@1"; 75*3a37aee3SSimon Glass %(ramdisk_config)s 76*3a37aee3SSimon Glass %(loadables_config)s 77*3a37aee3SSimon Glass }; 78*3a37aee3SSimon Glass }; 79*3a37aee3SSimon Glass}; 80*3a37aee3SSimon Glass''' 81*3a37aee3SSimon Glass 82*3a37aee3SSimon Glass# Define a base FDT - currently we don't use anything in this 83*3a37aee3SSimon Glassbase_fdt = ''' 84*3a37aee3SSimon Glass/dts-v1/; 85*3a37aee3SSimon Glass 86*3a37aee3SSimon Glass/ { 87*3a37aee3SSimon Glass model = "Sandbox Verified Boot Test"; 88*3a37aee3SSimon Glass compatible = "sandbox"; 89*3a37aee3SSimon Glass 90*3a37aee3SSimon Glass reset@0 { 91*3a37aee3SSimon Glass compatible = "sandbox,reset"; 92*3a37aee3SSimon Glass }; 93*3a37aee3SSimon Glass 94*3a37aee3SSimon Glass}; 95*3a37aee3SSimon Glass''' 96*3a37aee3SSimon Glass 97*3a37aee3SSimon Glass# This is the U-Boot script that is run for each test. First load the FIT, 98*3a37aee3SSimon Glass# then run the 'bootm' command, then save out memory from the places where 99*3a37aee3SSimon Glass# we expect 'bootm' to write things. Then quit. 100*3a37aee3SSimon Glassbase_script = ''' 101*3a37aee3SSimon Glasssb load hostfs 0 %(fit_addr)x %(fit)s 102*3a37aee3SSimon Glassfdt addr %(fit_addr)x 103*3a37aee3SSimon Glassbootm start %(fit_addr)x 104*3a37aee3SSimon Glassbootm loados 105*3a37aee3SSimon Glasssb save hostfs 0 %(kernel_addr)x %(kernel_out)s %(kernel_size)x 106*3a37aee3SSimon Glasssb save hostfs 0 %(fdt_addr)x %(fdt_out)s %(fdt_size)x 107*3a37aee3SSimon Glasssb save hostfs 0 %(ramdisk_addr)x %(ramdisk_out)s %(ramdisk_size)x 108*3a37aee3SSimon Glasssb save hostfs 0 %(loadables1_addr)x %(loadables1_out)s %(loadables1_size)x 109*3a37aee3SSimon Glasssb save hostfs 0 %(loadables2_addr)x %(loadables2_out)s %(loadables2_size)x 110*3a37aee3SSimon Glass''' 111*3a37aee3SSimon Glass 112*3a37aee3SSimon Glass@pytest.mark.boardspec('sandbox') 113*3a37aee3SSimon Glass@pytest.mark.buildconfigspec('fit_signature') 114*3a37aee3SSimon Glassdef test_fit(u_boot_console): 115*3a37aee3SSimon Glass def make_fname(leaf): 116*3a37aee3SSimon Glass """Make a temporary filename 117*3a37aee3SSimon Glass 118*3a37aee3SSimon Glass Args: 119*3a37aee3SSimon Glass leaf: Leaf name of file to create (within temporary directory) 120*3a37aee3SSimon Glass Return: 121*3a37aee3SSimon Glass Temporary filename 122*3a37aee3SSimon Glass """ 123*3a37aee3SSimon Glass 124*3a37aee3SSimon Glass return os.path.join(cons.config.build_dir, leaf) 125*3a37aee3SSimon Glass 126*3a37aee3SSimon Glass def filesize(fname): 127*3a37aee3SSimon Glass """Get the size of a file 128*3a37aee3SSimon Glass 129*3a37aee3SSimon Glass Args: 130*3a37aee3SSimon Glass fname: Filename to check 131*3a37aee3SSimon Glass Return: 132*3a37aee3SSimon Glass Size of file in bytes 133*3a37aee3SSimon Glass """ 134*3a37aee3SSimon Glass return os.stat(fname).st_size 135*3a37aee3SSimon Glass 136*3a37aee3SSimon Glass def read_file(fname): 137*3a37aee3SSimon Glass """Read the contents of a file 138*3a37aee3SSimon Glass 139*3a37aee3SSimon Glass Args: 140*3a37aee3SSimon Glass fname: Filename to read 141*3a37aee3SSimon Glass Returns: 142*3a37aee3SSimon Glass Contents of file as a string 143*3a37aee3SSimon Glass """ 144*3a37aee3SSimon Glass with open(fname, 'r') as fd: 145*3a37aee3SSimon Glass return fd.read() 146*3a37aee3SSimon Glass 147*3a37aee3SSimon Glass def make_dtb(): 148*3a37aee3SSimon Glass """Make a sample .dts file and compile it to a .dtb 149*3a37aee3SSimon Glass 150*3a37aee3SSimon Glass Returns: 151*3a37aee3SSimon Glass Filename of .dtb file created 152*3a37aee3SSimon Glass """ 153*3a37aee3SSimon Glass src = make_fname('u-boot.dts') 154*3a37aee3SSimon Glass dtb = make_fname('u-boot.dtb') 155*3a37aee3SSimon Glass with open(src, 'w') as fd: 156*3a37aee3SSimon Glass print >> fd, base_fdt 157*3a37aee3SSimon Glass util.run_and_log(cons, ['dtc', src, '-O', 'dtb', '-o', dtb]) 158*3a37aee3SSimon Glass return dtb 159*3a37aee3SSimon Glass 160*3a37aee3SSimon Glass def make_its(params): 161*3a37aee3SSimon Glass """Make a sample .its file with parameters embedded 162*3a37aee3SSimon Glass 163*3a37aee3SSimon Glass Args: 164*3a37aee3SSimon Glass params: Dictionary containing parameters to embed in the %() strings 165*3a37aee3SSimon Glass Returns: 166*3a37aee3SSimon Glass Filename of .its file created 167*3a37aee3SSimon Glass """ 168*3a37aee3SSimon Glass its = make_fname('test.its') 169*3a37aee3SSimon Glass with open(its, 'w') as fd: 170*3a37aee3SSimon Glass print >> fd, base_its % params 171*3a37aee3SSimon Glass return its 172*3a37aee3SSimon Glass 173*3a37aee3SSimon Glass def make_fit(mkimage, params): 174*3a37aee3SSimon Glass """Make a sample .fit file ready for loading 175*3a37aee3SSimon Glass 176*3a37aee3SSimon Glass This creates a .its script with the selected parameters and uses mkimage to 177*3a37aee3SSimon Glass turn this into a .fit image. 178*3a37aee3SSimon Glass 179*3a37aee3SSimon Glass Args: 180*3a37aee3SSimon Glass mkimage: Filename of 'mkimage' utility 181*3a37aee3SSimon Glass params: Dictionary containing parameters to embed in the %() strings 182*3a37aee3SSimon Glass Return: 183*3a37aee3SSimon Glass Filename of .fit file created 184*3a37aee3SSimon Glass """ 185*3a37aee3SSimon Glass fit = make_fname('test.fit') 186*3a37aee3SSimon Glass its = make_its(params) 187*3a37aee3SSimon Glass util.run_and_log(cons, [mkimage, '-f', its, fit]) 188*3a37aee3SSimon Glass with open(make_fname('u-boot.dts'), 'w') as fd: 189*3a37aee3SSimon Glass print >> fd, base_fdt 190*3a37aee3SSimon Glass return fit 191*3a37aee3SSimon Glass 192*3a37aee3SSimon Glass def make_kernel(filename, text): 193*3a37aee3SSimon Glass """Make a sample kernel with test data 194*3a37aee3SSimon Glass 195*3a37aee3SSimon Glass Args: 196*3a37aee3SSimon Glass filename: the name of the file you want to create 197*3a37aee3SSimon Glass Returns: 198*3a37aee3SSimon Glass Full path and filename of the kernel it created 199*3a37aee3SSimon Glass """ 200*3a37aee3SSimon Glass fname = make_fname(filename) 201*3a37aee3SSimon Glass data = '' 202*3a37aee3SSimon Glass for i in range(100): 203*3a37aee3SSimon Glass data += 'this %s %d is unlikely to boot\n' % (text, i) 204*3a37aee3SSimon Glass with open(fname, 'w') as fd: 205*3a37aee3SSimon Glass print >> fd, data 206*3a37aee3SSimon Glass return fname 207*3a37aee3SSimon Glass 208*3a37aee3SSimon Glass def make_ramdisk(filename, text): 209*3a37aee3SSimon Glass """Make a sample ramdisk with test data 210*3a37aee3SSimon Glass 211*3a37aee3SSimon Glass Returns: 212*3a37aee3SSimon Glass Filename of ramdisk created 213*3a37aee3SSimon Glass """ 214*3a37aee3SSimon Glass fname = make_fname(filename) 215*3a37aee3SSimon Glass data = '' 216*3a37aee3SSimon Glass for i in range(100): 217*3a37aee3SSimon Glass data += '%s %d was seldom used in the middle ages\n' % (text, i) 218*3a37aee3SSimon Glass with open(fname, 'w') as fd: 219*3a37aee3SSimon Glass print >> fd, data 220*3a37aee3SSimon Glass return fname 221*3a37aee3SSimon Glass 222*3a37aee3SSimon Glass def find_matching(text, match): 223*3a37aee3SSimon Glass """Find a match in a line of text, and return the unmatched line portion 224*3a37aee3SSimon Glass 225*3a37aee3SSimon Glass This is used to extract a part of a line from some text. The match string 226*3a37aee3SSimon Glass is used to locate the line - we use the first line that contains that 227*3a37aee3SSimon Glass match text. 228*3a37aee3SSimon Glass 229*3a37aee3SSimon Glass Once we find a match, we discard the match string itself from the line, 230*3a37aee3SSimon Glass and return what remains. 231*3a37aee3SSimon Glass 232*3a37aee3SSimon Glass TODO: If this function becomes more generally useful, we could change it 233*3a37aee3SSimon Glass to use regex and return groups. 234*3a37aee3SSimon Glass 235*3a37aee3SSimon Glass Args: 236*3a37aee3SSimon Glass text: Text to check (list of strings, one for each command issued) 237*3a37aee3SSimon Glass match: String to search for 238*3a37aee3SSimon Glass Return: 239*3a37aee3SSimon Glass String containing unmatched portion of line 240*3a37aee3SSimon Glass Exceptions: 241*3a37aee3SSimon Glass ValueError: If match is not found 242*3a37aee3SSimon Glass 243*3a37aee3SSimon Glass >>> find_matching(['first line:10', 'second_line:20'], 'first line:') 244*3a37aee3SSimon Glass '10' 245*3a37aee3SSimon Glass >>> find_matching(['first line:10', 'second_line:20'], 'second line') 246*3a37aee3SSimon Glass Traceback (most recent call last): 247*3a37aee3SSimon Glass ... 248*3a37aee3SSimon Glass ValueError: Test aborted 249*3a37aee3SSimon Glass >>> find_matching('first line:10\', 'second_line:20'], 'second_line:') 250*3a37aee3SSimon Glass '20' 251*3a37aee3SSimon Glass >>> find_matching('first line:10\', 'second_line:20\nthird_line:30'], 252*3a37aee3SSimon Glass 'third_line:') 253*3a37aee3SSimon Glass '30' 254*3a37aee3SSimon Glass """ 255*3a37aee3SSimon Glass __tracebackhide__ = True 256*3a37aee3SSimon Glass for line in '\n'.join(text).splitlines(): 257*3a37aee3SSimon Glass pos = line.find(match) 258*3a37aee3SSimon Glass if pos != -1: 259*3a37aee3SSimon Glass return line[:pos] + line[pos + len(match):] 260*3a37aee3SSimon Glass 261*3a37aee3SSimon Glass pytest.fail("Expected '%s' but not found in output") 262*3a37aee3SSimon Glass 263*3a37aee3SSimon Glass def check_equal(expected_fname, actual_fname, failure_msg): 264*3a37aee3SSimon Glass """Check that a file matches its expected contents 265*3a37aee3SSimon Glass 266*3a37aee3SSimon Glass Args: 267*3a37aee3SSimon Glass expected_fname: Filename containing expected contents 268*3a37aee3SSimon Glass actual_fname: Filename containing actual contents 269*3a37aee3SSimon Glass failure_msg: Message to print on failure 270*3a37aee3SSimon Glass """ 271*3a37aee3SSimon Glass expected_data = read_file(expected_fname) 272*3a37aee3SSimon Glass actual_data = read_file(actual_fname) 273*3a37aee3SSimon Glass assert expected_data == actual_data, failure_msg 274*3a37aee3SSimon Glass 275*3a37aee3SSimon Glass def check_not_equal(expected_fname, actual_fname, failure_msg): 276*3a37aee3SSimon Glass """Check that a file does not match its expected contents 277*3a37aee3SSimon Glass 278*3a37aee3SSimon Glass Args: 279*3a37aee3SSimon Glass expected_fname: Filename containing expected contents 280*3a37aee3SSimon Glass actual_fname: Filename containing actual contents 281*3a37aee3SSimon Glass failure_msg: Message to print on failure 282*3a37aee3SSimon Glass """ 283*3a37aee3SSimon Glass expected_data = read_file(expected_fname) 284*3a37aee3SSimon Glass actual_data = read_file(actual_fname) 285*3a37aee3SSimon Glass assert expected_data != actual_data, failure_msg 286*3a37aee3SSimon Glass 287*3a37aee3SSimon Glass def run_fit_test(mkimage): 288*3a37aee3SSimon Glass """Basic sanity check of FIT loading in U-Boot 289*3a37aee3SSimon Glass 290*3a37aee3SSimon Glass TODO: Almost everything: 291*3a37aee3SSimon Glass - hash algorithms - invalid hash/contents should be detected 292*3a37aee3SSimon Glass - signature algorithms - invalid sig/contents should be detected 293*3a37aee3SSimon Glass - compression 294*3a37aee3SSimon Glass - checking that errors are detected like: 295*3a37aee3SSimon Glass - image overwriting 296*3a37aee3SSimon Glass - missing images 297*3a37aee3SSimon Glass - invalid configurations 298*3a37aee3SSimon Glass - incorrect os/arch/type fields 299*3a37aee3SSimon Glass - empty data 300*3a37aee3SSimon Glass - images too large/small 301*3a37aee3SSimon Glass - invalid FDT (e.g. putting a random binary in instead) 302*3a37aee3SSimon Glass - default configuration selection 303*3a37aee3SSimon Glass - bootm command line parameters should have desired effect 304*3a37aee3SSimon Glass - run code coverage to make sure we are testing all the code 305*3a37aee3SSimon Glass """ 306*3a37aee3SSimon Glass # Set up invariant files 307*3a37aee3SSimon Glass control_dtb = make_dtb() 308*3a37aee3SSimon Glass kernel = make_kernel('test-kernel.bin', 'kernel') 309*3a37aee3SSimon Glass ramdisk = make_ramdisk('test-ramdisk.bin', 'ramdisk') 310*3a37aee3SSimon Glass loadables1 = make_kernel('test-loadables1.bin', 'lenrek') 311*3a37aee3SSimon Glass loadables2 = make_ramdisk('test-loadables2.bin', 'ksidmar') 312*3a37aee3SSimon Glass kernel_out = make_fname('kernel-out.bin') 313*3a37aee3SSimon Glass fdt_out = make_fname('fdt-out.dtb') 314*3a37aee3SSimon Glass ramdisk_out = make_fname('ramdisk-out.bin') 315*3a37aee3SSimon Glass loadables1_out = make_fname('loadables1-out.bin') 316*3a37aee3SSimon Glass loadables2_out = make_fname('loadables2-out.bin') 317*3a37aee3SSimon Glass 318*3a37aee3SSimon Glass # Set up basic parameters with default values 319*3a37aee3SSimon Glass params = { 320*3a37aee3SSimon Glass 'fit_addr' : 0x1000, 321*3a37aee3SSimon Glass 322*3a37aee3SSimon Glass 'kernel' : kernel, 323*3a37aee3SSimon Glass 'kernel_out' : kernel_out, 324*3a37aee3SSimon Glass 'kernel_addr' : 0x40000, 325*3a37aee3SSimon Glass 'kernel_size' : filesize(kernel), 326*3a37aee3SSimon Glass 327*3a37aee3SSimon Glass 'fdt_out' : fdt_out, 328*3a37aee3SSimon Glass 'fdt_addr' : 0x80000, 329*3a37aee3SSimon Glass 'fdt_size' : filesize(control_dtb), 330*3a37aee3SSimon Glass 'fdt_load' : '', 331*3a37aee3SSimon Glass 332*3a37aee3SSimon Glass 'ramdisk' : ramdisk, 333*3a37aee3SSimon Glass 'ramdisk_out' : ramdisk_out, 334*3a37aee3SSimon Glass 'ramdisk_addr' : 0xc0000, 335*3a37aee3SSimon Glass 'ramdisk_size' : filesize(ramdisk), 336*3a37aee3SSimon Glass 'ramdisk_load' : '', 337*3a37aee3SSimon Glass 'ramdisk_config' : '', 338*3a37aee3SSimon Glass 339*3a37aee3SSimon Glass 'loadables1' : loadables1, 340*3a37aee3SSimon Glass 'loadables1_out' : loadables1_out, 341*3a37aee3SSimon Glass 'loadables1_addr' : 0x100000, 342*3a37aee3SSimon Glass 'loadables1_size' : filesize(loadables1), 343*3a37aee3SSimon Glass 'loadables1_load' : '', 344*3a37aee3SSimon Glass 345*3a37aee3SSimon Glass 'loadables2' : loadables2, 346*3a37aee3SSimon Glass 'loadables2_out' : loadables2_out, 347*3a37aee3SSimon Glass 'loadables2_addr' : 0x140000, 348*3a37aee3SSimon Glass 'loadables2_size' : filesize(loadables2), 349*3a37aee3SSimon Glass 'loadables2_load' : '', 350*3a37aee3SSimon Glass 351*3a37aee3SSimon Glass 'loadables_config' : '', 352*3a37aee3SSimon Glass } 353*3a37aee3SSimon Glass 354*3a37aee3SSimon Glass # Make a basic FIT and a script to load it 355*3a37aee3SSimon Glass fit = make_fit(mkimage, params) 356*3a37aee3SSimon Glass params['fit'] = fit 357*3a37aee3SSimon Glass cmd = base_script % params 358*3a37aee3SSimon Glass 359*3a37aee3SSimon Glass # First check that we can load a kernel 360*3a37aee3SSimon Glass # We could perhaps reduce duplication with some loss of readability 361*3a37aee3SSimon Glass cons.config.dtb = control_dtb 362*3a37aee3SSimon Glass cons.restart_uboot() 363*3a37aee3SSimon Glass with cons.log.section('Kernel load'): 364*3a37aee3SSimon Glass output = cons.run_command_list(cmd.splitlines()) 365*3a37aee3SSimon Glass check_equal(kernel, kernel_out, 'Kernel not loaded') 366*3a37aee3SSimon Glass check_not_equal(control_dtb, fdt_out, 367*3a37aee3SSimon Glass 'FDT loaded but should be ignored') 368*3a37aee3SSimon Glass check_not_equal(ramdisk, ramdisk_out, 369*3a37aee3SSimon Glass 'Ramdisk loaded but should not be') 370*3a37aee3SSimon Glass 371*3a37aee3SSimon Glass # Find out the offset in the FIT where U-Boot has found the FDT 372*3a37aee3SSimon Glass line = find_matching(output, 'Booting using the fdt blob at ') 373*3a37aee3SSimon Glass fit_offset = int(line, 16) - params['fit_addr'] 374*3a37aee3SSimon Glass fdt_magic = struct.pack('>L', 0xd00dfeed) 375*3a37aee3SSimon Glass data = read_file(fit) 376*3a37aee3SSimon Glass 377*3a37aee3SSimon Glass # Now find where it actually is in the FIT (skip the first word) 378*3a37aee3SSimon Glass real_fit_offset = data.find(fdt_magic, 4) 379*3a37aee3SSimon Glass assert fit_offset == real_fit_offset, ( 380*3a37aee3SSimon Glass 'U-Boot loaded FDT from offset %#x, FDT is actually at %#x' % 381*3a37aee3SSimon Glass (fit_offset, real_fit_offset)) 382*3a37aee3SSimon Glass 383*3a37aee3SSimon Glass # Now a kernel and an FDT 384*3a37aee3SSimon Glass with cons.log.section('Kernel + FDT load'): 385*3a37aee3SSimon Glass params['fdt_load'] = 'load = <%#x>;' % params['fdt_addr'] 386*3a37aee3SSimon Glass fit = make_fit(mkimage, params) 387*3a37aee3SSimon Glass cons.restart_uboot() 388*3a37aee3SSimon Glass output = cons.run_command_list(cmd.splitlines()) 389*3a37aee3SSimon Glass check_equal(kernel, kernel_out, 'Kernel not loaded') 390*3a37aee3SSimon Glass check_equal(control_dtb, fdt_out, 'FDT not loaded') 391*3a37aee3SSimon Glass check_not_equal(ramdisk, ramdisk_out, 392*3a37aee3SSimon Glass 'Ramdisk loaded but should not be') 393*3a37aee3SSimon Glass 394*3a37aee3SSimon Glass # Try a ramdisk 395*3a37aee3SSimon Glass with cons.log.section('Kernel + FDT + Ramdisk load'): 396*3a37aee3SSimon Glass params['ramdisk_config'] = 'ramdisk = "ramdisk@1";' 397*3a37aee3SSimon Glass params['ramdisk_load'] = 'load = <%#x>;' % params['ramdisk_addr'] 398*3a37aee3SSimon Glass fit = make_fit(mkimage, params) 399*3a37aee3SSimon Glass cons.restart_uboot() 400*3a37aee3SSimon Glass output = cons.run_command_list(cmd.splitlines()) 401*3a37aee3SSimon Glass check_equal(ramdisk, ramdisk_out, 'Ramdisk not loaded') 402*3a37aee3SSimon Glass 403*3a37aee3SSimon Glass # Configuration with some Loadables 404*3a37aee3SSimon Glass with cons.log.section('Kernel + FDT + Ramdisk load + Loadables'): 405*3a37aee3SSimon Glass params['loadables_config'] = 'loadables = "kernel@2", "ramdisk@2";' 406*3a37aee3SSimon Glass params['loadables1_load'] = ('load = <%#x>;' % 407*3a37aee3SSimon Glass params['loadables1_addr']) 408*3a37aee3SSimon Glass params['loadables2_load'] = ('load = <%#x>;' % 409*3a37aee3SSimon Glass params['loadables2_addr']) 410*3a37aee3SSimon Glass fit = make_fit(mkimage, params) 411*3a37aee3SSimon Glass cons.restart_uboot() 412*3a37aee3SSimon Glass output = cons.run_command_list(cmd.splitlines()) 413*3a37aee3SSimon Glass check_equal(loadables1, loadables1_out, 414*3a37aee3SSimon Glass 'Loadables1 (kernel) not loaded') 415*3a37aee3SSimon Glass check_equal(loadables2, loadables2_out, 416*3a37aee3SSimon Glass 'Loadables2 (ramdisk) not loaded') 417*3a37aee3SSimon Glass 418*3a37aee3SSimon Glass cons = u_boot_console 419*3a37aee3SSimon Glass try: 420*3a37aee3SSimon Glass # We need to use our own device tree file. Remember to restore it 421*3a37aee3SSimon Glass # afterwards. 422*3a37aee3SSimon Glass old_dtb = cons.config.dtb 423*3a37aee3SSimon Glass mkimage = cons.config.build_dir + '/tools/mkimage' 424*3a37aee3SSimon Glass run_fit_test(mkimage) 425*3a37aee3SSimon Glass finally: 426*3a37aee3SSimon Glass # Go back to the original U-Boot with the correct dtb. 427*3a37aee3SSimon Glass cons.config.dtb = old_dtb 428*3a37aee3SSimon Glass cons.restart_uboot() 429