160 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			160 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Python
		
	
	
	
| # SPDX-License-Identifier:  GPL-2.0+
 | |
| # Copyright (c) 2020
 | |
| # Author: Sam Protsenko <joe.skb7@gmail.com>
 | |
| 
 | |
| # Test U-Boot's "abootimg" commands.
 | |
| 
 | |
| import os
 | |
| import pytest
 | |
| import u_boot_utils
 | |
| 
 | |
| """
 | |
| These tests rely on disk image (boot.img), which is automatically created by
 | |
| the test from the stored hex dump. This is done to avoid the dependency on the
 | |
| most recent mkbootimg tool from AOSP/master. Here is the list of commands which
 | |
| was used to generate the boot.img and obtain compressed hex dump from it:
 | |
| 
 | |
|     $ echo '/dts-v1/; / { model = "x1"; compatible = "y1,z1"; };' > test1.dts
 | |
|     $ echo '/dts-v1/; / { model = "x2"; compatible = "y2,z2"; };' > test2.dts
 | |
|     $ dtc test1.dts > dt1.dtb
 | |
|     $ dtc test2.dts > dt2.dtb
 | |
|     $ cat dt1.dtb dt2.dtb > dtb.img
 | |
|     $ echo 'kernel payload' > kernel
 | |
|     $ echo 'ramdisk payload' > ramdisk.img
 | |
|     $ mkbootimg --kernel ./kernel --ramdisk ./ramdisk.img  \
 | |
|                 --cmdline "cmdline test" --dtb ./dtb.img   \
 | |
|                 --os_version R --os_patch_level 2019-06-05 \
 | |
|                 --header_version 2 --output boot.img
 | |
|     $ gzip -9 boot.img
 | |
|     $ xxd -p boot.img.gz > boot.img.gz.hex
 | |
| 
 | |
| Now one can obtain original boot.img from this hex dump like this:
 | |
| 
 | |
|     $ xxd -r -p boot.img.gz.hex boot.img.gz
 | |
|     $ gunzip -9 boot.img.gz
 | |
| """
 | |
| 
 | |
| # boot.img.gz hex dump
 | |
| img_hex = """1f8b08084844af5d0203626f6f742e696d670073f47309f2f77451e46700
 | |
| 820606010106301084501f04181819041838181898803c3346060c909c9b
 | |
| 92939997aa50925a5cc2300a461c3078b2e1793c4b876fd92db97939fb6c
 | |
| b7762ffff07d345446c1281805e8a0868d81e117a45e111c0d8dc101b253
 | |
| 8bf25273140a122b73f21353b8460364148c8251300a46c1281801a02831
 | |
| 3725b3387bb401300a46c1281805a360148c207081f7df5b20550bc41640
 | |
| 9c03c41a0c90f17fe85400986d82452b6c3680198a192a0ce17c3610ae34
 | |
| d4a9820881a70f3873f35352731892f3730b124b32937252a96bb9119ae5
 | |
| 463a5546f82c1f05a360148c8251300a462e000085bf67f200200000"""
 | |
| # Expected response for "abootimg dtb_dump" command
 | |
| dtb_dump_resp="""## DTB area contents (concat format):
 | |
|  - DTB #0:
 | |
|            (DTB)size = 125
 | |
|           (DTB)model = x1
 | |
|      (DTB)compatible = y1,z1
 | |
|  - DTB #1:
 | |
|            (DTB)size = 125
 | |
|           (DTB)model = x2
 | |
|      (DTB)compatible = y2,z2"""
 | |
| # Address in RAM where to load the boot image ('abootimg' looks in $loadaddr)
 | |
| loadaddr = 0x1000
 | |
| # Expected DTB #1 offset from the boot image start address
 | |
| dtb1_offset = 0x187d
 | |
| # DTB #1 start address in RAM
 | |
| dtb1_addr = loadaddr + dtb1_offset
 | |
| 
 | |
| class AbootimgTestDiskImage(object):
 | |
|     """Disk image used by abootimg tests."""
 | |
| 
 | |
|     def __init__(self, u_boot_console):
 | |
|         """Initialize a new AbootimgDiskImage object.
 | |
| 
 | |
|         Args:
 | |
|             u_boot_console: A U-Boot console.
 | |
| 
 | |
|         Returns:
 | |
|             Nothing.
 | |
|         """
 | |
| 
 | |
|         gz_hex = u_boot_console.config.persistent_data_dir + '/boot.img.gz.hex'
 | |
|         gz = u_boot_console.config.persistent_data_dir + '/boot.img.gz'
 | |
| 
 | |
|         filename = 'boot.img'
 | |
|         persistent = u_boot_console.config.persistent_data_dir + '/' + filename
 | |
|         self.path = u_boot_console.config.result_dir  + '/' + filename
 | |
| 
 | |
|         with u_boot_utils.persistent_file_helper(u_boot_console.log, persistent):
 | |
|             if os.path.exists(persistent):
 | |
|                 u_boot_console.log.action('Disk image file ' + persistent +
 | |
|                     ' already exists')
 | |
|             else:
 | |
|                 u_boot_console.log.action('Generating ' + persistent)
 | |
| 
 | |
|                 f = open(gz_hex, "w")
 | |
|                 f.write(img_hex)
 | |
|                 f.close()
 | |
| 
 | |
|                 cmd = ('xxd', '-r', '-p', gz_hex, gz)
 | |
|                 u_boot_utils.run_and_log(u_boot_console, cmd)
 | |
| 
 | |
|                 cmd = ('gunzip', '-9', gz)
 | |
|                 u_boot_utils.run_and_log(u_boot_console, cmd)
 | |
| 
 | |
|         cmd = ('cp', persistent, self.path)
 | |
|         u_boot_utils.run_and_log(u_boot_console, cmd)
 | |
| 
 | |
| gtdi = None
 | |
| @pytest.fixture(scope='function')
 | |
| def abootimg_disk_image(u_boot_console):
 | |
|     """pytest fixture to provide a AbootimgTestDiskImage object to tests.
 | |
|     This is function-scoped because it uses u_boot_console, which is also
 | |
|     function-scoped. However, we don't need to actually do any function-scope
 | |
|     work, so this simply returns the same object over and over each time."""
 | |
| 
 | |
|     global gtdi
 | |
|     if not gtdi:
 | |
|         gtdi = AbootimgTestDiskImage(u_boot_console)
 | |
|     return gtdi
 | |
| 
 | |
| @pytest.mark.boardspec('sandbox')
 | |
| @pytest.mark.buildconfigspec('android_boot_image')
 | |
| @pytest.mark.buildconfigspec('cmd_abootimg')
 | |
| @pytest.mark.buildconfigspec('cmd_fdt')
 | |
| @pytest.mark.requiredtool('xxd')
 | |
| @pytest.mark.requiredtool('gunzip')
 | |
| def test_abootimg(abootimg_disk_image, u_boot_console):
 | |
|     """Test the 'abootimg' command."""
 | |
| 
 | |
|     u_boot_console.log.action('Loading disk image to RAM...')
 | |
|     u_boot_console.run_command('setenv loadaddr 0x%x' % (loadaddr))
 | |
|     u_boot_console.run_command('host load hostfs - 0x%x %s' % (loadaddr,
 | |
|         abootimg_disk_image.path))
 | |
| 
 | |
|     u_boot_console.log.action('Testing \'abootimg get ver\'...')
 | |
|     response = u_boot_console.run_command('abootimg get ver')
 | |
|     assert response == "2"
 | |
|     u_boot_console.run_command('abootimg get ver v')
 | |
|     response = u_boot_console.run_command('env print v')
 | |
|     assert response == 'v=2'
 | |
| 
 | |
|     u_boot_console.log.action('Testing \'abootimg get recovery_dtbo\'...')
 | |
|     response = u_boot_console.run_command('abootimg get recovery_dtbo a')
 | |
|     assert response == 'Error: recovery_dtbo_size is 0'
 | |
| 
 | |
|     u_boot_console.log.action('Testing \'abootimg dump dtb\'...')
 | |
|     response = u_boot_console.run_command('abootimg dump dtb').replace('\r', '')
 | |
|     assert response == dtb_dump_resp
 | |
| 
 | |
|     u_boot_console.log.action('Testing \'abootimg get dtb_load_addr\'...')
 | |
|     u_boot_console.run_command('abootimg get dtb_load_addr a')
 | |
|     response = u_boot_console.run_command('env print a')
 | |
|     assert response == 'a=11f00000'
 | |
| 
 | |
|     u_boot_console.log.action('Testing \'abootimg get dtb --index\'...')
 | |
|     u_boot_console.run_command('abootimg get dtb --index=1 dtb1_start')
 | |
|     response = u_boot_console.run_command('env print dtb1_start')
 | |
|     correct_str = "dtb1_start=%x" % (dtb1_addr)
 | |
|     assert response == correct_str
 | |
|     u_boot_console.run_command('fdt addr $dtb1_start')
 | |
|     u_boot_console.run_command('fdt get value v / model')
 | |
|     response = u_boot_console.run_command('env print v')
 | |
|     assert response == 'v=x2'
 |