blob: 523f0a923ebc880de233c21bb6e665f06229210b [file] [log] [blame]
Simon Glassbfb0bb22019-10-31 07:42:55 -06001#!/usr/bin/env python3
Tom Rini10e47792018-05-06 17:58:06 -04002# SPDX-License-Identifier: GPL-2.0+
Simon Glass9d2eb922017-06-18 22:09:06 -06003# Copyright (c) 2012 The Chromium OS Authors.
4#
Simon Glass9d2eb922017-06-18 22:09:06 -06005
6"""Tests for the dtb_platdata module
7
Simon Glass70cd0d72018-07-06 10:27:20 -06008This includes unit tests for some functions and functional tests for the dtoc
9tool.
Simon Glass9d2eb922017-06-18 22:09:06 -060010"""
11
12import collections
Simon Glass768ff0a2021-02-03 06:00:51 -070013import copy
Simon Glass4e8e8462020-12-28 20:34:52 -070014import glob
Simon Glass9d2eb922017-06-18 22:09:06 -060015import os
16import struct
17import unittest
18
Simon Glass9d2eb922017-06-18 22:09:06 -060019from dtb_platdata import get_value
20from dtb_platdata import tab_to
Simon Glassa60cabd2020-12-28 20:34:47 -070021from dtoc import dtb_platdata
Simon Glassa997ea52020-04-17 18:09:04 -060022from dtoc import fdt
23from dtoc import fdt_util
Simon Glass768ff0a2021-02-03 06:00:51 -070024from dtoc import src_scan
Simon Glass9065bc92020-12-28 20:35:06 -070025from dtoc.src_scan import conv_name_to_c
26from dtoc.src_scan import get_compat_name
Simon Glassa997ea52020-04-17 18:09:04 -060027from patman import test_util
28from patman import tools
Simon Glass9d2eb922017-06-18 22:09:06 -060029
Simon Glassa60cabd2020-12-28 20:34:47 -070030OUR_PATH = os.path.dirname(os.path.realpath(__file__))
Simon Glass9d2eb922017-06-18 22:09:06 -060031
32
Simon Glasseb37e2d2017-11-12 21:52:17 -070033HEADER = '''/*
34 * DO NOT MODIFY
35 *
Simon Glass6b208842020-12-28 20:35:00 -070036 * Defines the structs used to hold devicetree data.
37 * This was generated by dtoc from a .dtb (device tree binary) file.
Simon Glasseb37e2d2017-11-12 21:52:17 -070038 */
39
40#include <stdbool.h>
Masahiro Yamada75f82d02018-03-05 01:20:11 +090041#include <linux/libfdt.h>'''
Simon Glasseb37e2d2017-11-12 21:52:17 -070042
43C_HEADER = '''/*
44 * DO NOT MODIFY
45 *
Simon Glass6b208842020-12-28 20:35:00 -070046 * Declares the U_BOOT_DRIVER() records and platform data.
47 * This was generated by dtoc from a .dtb (device tree binary) file.
Simon Glasseb37e2d2017-11-12 21:52:17 -070048 */
49
Simon Glass1d8364a2020-12-28 20:34:54 -070050/* Allow use of U_BOOT_DRVINFO() in this file */
Simon Glassbeddd7a2020-12-28 20:35:01 -070051#define DT_PLAT_C
Simon Glass4c73d7b2020-10-03 11:31:41 -060052
Simon Glasseb37e2d2017-11-12 21:52:17 -070053#include <common.h>
54#include <dm.h>
55#include <dt-structs.h>
56'''
57
Simon Glass768ff0a2021-02-03 06:00:51 -070058# Scanner saved from a previous run of the tests (to speed things up)
59saved_scan = None
60
Simon Glassa60cabd2020-12-28 20:34:47 -070061# This is a test so is allowed to access private things in the module it is
62# testing
63# pylint: disable=W0212
Simon Glass3bce93d2018-07-06 10:27:37 -060064
65def get_dtb_file(dts_fname, capture_stderr=False):
Simon Glass9d2eb922017-06-18 22:09:06 -060066 """Compile a .dts file to a .dtb
67
68 Args:
Simon Glassa60cabd2020-12-28 20:34:47 -070069 dts_fname (str): Filename of .dts file in the current directory
70 capture_stderr (bool): True to capture and discard stderr output
Simon Glass9d2eb922017-06-18 22:09:06 -060071
72 Returns:
Simon Glassa60cabd2020-12-28 20:34:47 -070073 str: Filename of compiled file in output directory
Simon Glass9d2eb922017-06-18 22:09:06 -060074 """
Simon Glass4f4b2402021-02-03 06:00:56 -070075 return fdt_util.EnsureCompiled(os.path.join(OUR_PATH, 'test', dts_fname),
Simon Glass3bce93d2018-07-06 10:27:37 -060076 capture_stderr=capture_stderr)
Simon Glass9d2eb922017-06-18 22:09:06 -060077
Simon Glass768ff0a2021-02-03 06:00:51 -070078
79def setup():
80 global saved_scan
81
82 # Disable warnings so that calls to get_normalized_compat_name() will not
83 # output things.
84 saved_scan = src_scan.Scanner(None, True, False)
85 saved_scan.scan_drivers()
86
87def copy_scan():
88 """Get a copy of saved_scan so that each test can start clean"""
89 return copy.deepcopy(saved_scan)
90
Simon Glass9d2eb922017-06-18 22:09:06 -060091
92class TestDtoc(unittest.TestCase):
93 """Tests for dtoc"""
94 @classmethod
95 def setUpClass(cls):
96 tools.PrepareOutputDir(None)
Simon Glass7f5e2262020-07-07 21:32:06 -060097 cls.maxDiff = None
Simon Glass9d2eb922017-06-18 22:09:06 -060098
99 @classmethod
100 def tearDownClass(cls):
Simon Glassa60cabd2020-12-28 20:34:47 -0700101 tools.FinaliseOutputDir()
Simon Glass9d2eb922017-06-18 22:09:06 -0600102
Simon Glassa60cabd2020-12-28 20:34:47 -0700103 @staticmethod
104 def _write_python_string(fname, data):
Simon Glassc47c2b32018-07-06 10:27:25 -0600105 """Write a string with tabs expanded as done in this Python file
106
107 Args:
Simon Glassa60cabd2020-12-28 20:34:47 -0700108 fname (str): Filename to write to
109 data (str): Raw string to convert
Simon Glassc47c2b32018-07-06 10:27:25 -0600110 """
111 data = data.replace('\t', '\\t')
Simon Glassa60cabd2020-12-28 20:34:47 -0700112 with open(fname, 'w') as fout:
113 fout.write(data)
Simon Glassc47c2b32018-07-06 10:27:25 -0600114
Simon Glassa60cabd2020-12-28 20:34:47 -0700115 def _check_strings(self, expected, actual):
Simon Glassc47c2b32018-07-06 10:27:25 -0600116 """Check that a string matches its expected value
117
118 If the strings do not match, they are written to the /tmp directory in
119 the same Python format as is used here in the test. This allows for
120 easy comparison and update of the tests.
121
122 Args:
Simon Glassa60cabd2020-12-28 20:34:47 -0700123 expected (str): Expected string
124 actual (str): Actual string
Simon Glassc47c2b32018-07-06 10:27:25 -0600125 """
126 if expected != actual:
Simon Glassa60cabd2020-12-28 20:34:47 -0700127 self._write_python_string('/tmp/binman.expected', expected)
128 self._write_python_string('/tmp/binman.actual', actual)
Simon Glass61b88e52019-05-17 22:00:31 -0600129 print('Failures written to /tmp/binman.{expected,actual}')
Simon Glassa60cabd2020-12-28 20:34:47 -0700130 self.assertEqual(expected, actual)
Simon Glassc47c2b32018-07-06 10:27:25 -0600131
Simon Glassa60cabd2020-12-28 20:34:47 -0700132 @staticmethod
133 def run_test(args, dtb_file, output):
134 """Run a test using dtoc
Walter Lozanoa324e412020-06-25 01:10:08 -0300135
Simon Glassa60cabd2020-12-28 20:34:47 -0700136 Args:
137 args (list of str): List of arguments for dtoc
138 dtb_file (str): Filename of .dtb file
139 output (str): Filename of output file
140 """
Simon Glass768ff0a2021-02-03 06:00:51 -0700141 dtb_platdata.run_steps(args, dtb_file, False, output, [], True,
142 None, None, scan=copy_scan())
Walter Lozanoa324e412020-06-25 01:10:08 -0300143
Simon Glass9d2eb922017-06-18 22:09:06 -0600144 def test_name(self):
145 """Test conversion of device tree names to C identifiers"""
146 self.assertEqual('serial_at_0x12', conv_name_to_c('serial@0x12'))
147 self.assertEqual('vendor_clock_frequency',
148 conv_name_to_c('vendor,clock-frequency'))
149 self.assertEqual('rockchip_rk3399_sdhci_5_1',
150 conv_name_to_c('rockchip,rk3399-sdhci-5.1'))
151
152 def test_tab_to(self):
153 """Test operation of tab_to() function"""
154 self.assertEqual('fred ', tab_to(0, 'fred'))
155 self.assertEqual('fred\t', tab_to(1, 'fred'))
156 self.assertEqual('fred was here ', tab_to(1, 'fred was here'))
157 self.assertEqual('fred was here\t\t', tab_to(3, 'fred was here'))
158 self.assertEqual('exactly8 ', tab_to(1, 'exactly8'))
159 self.assertEqual('exactly8\t', tab_to(2, 'exactly8'))
160
161 def test_get_value(self):
162 """Test operation of get_value() function"""
163 self.assertEqual('0x45',
Simon Glassc9a032c2020-11-08 20:36:17 -0700164 get_value(fdt.Type.INT, struct.pack('>I', 0x45)))
Simon Glass9d2eb922017-06-18 22:09:06 -0600165 self.assertEqual('0x45',
Simon Glassc9a032c2020-11-08 20:36:17 -0700166 get_value(fdt.Type.BYTE, struct.pack('<I', 0x45)))
Simon Glass9d2eb922017-06-18 22:09:06 -0600167 self.assertEqual('0x0',
Simon Glassc9a032c2020-11-08 20:36:17 -0700168 get_value(fdt.Type.BYTE, struct.pack('>I', 0x45)))
169 self.assertEqual('"test"', get_value(fdt.Type.STRING, 'test'))
170 self.assertEqual('true', get_value(fdt.Type.BOOL, None))
Simon Glass9d2eb922017-06-18 22:09:06 -0600171
172 def test_get_compat_name(self):
173 """Test operation of get_compat_name() function"""
174 Prop = collections.namedtuple('Prop', ['value'])
175 Node = collections.namedtuple('Node', ['props'])
176
177 prop = Prop(['rockchip,rk3399-sdhci-5.1', 'arasan,sdhci-5.1'])
178 node = Node({'compatible': prop})
Walter Lozano5fe734c2020-07-23 00:22:03 -0300179 self.assertEqual((['rockchip_rk3399_sdhci_5_1', 'arasan_sdhci_5_1']),
Simon Glass9d2eb922017-06-18 22:09:06 -0600180 get_compat_name(node))
181
182 prop = Prop(['rockchip,rk3399-sdhci-5.1'])
183 node = Node({'compatible': prop})
Walter Lozano5fe734c2020-07-23 00:22:03 -0300184 self.assertEqual((['rockchip_rk3399_sdhci_5_1']),
Simon Glass9d2eb922017-06-18 22:09:06 -0600185 get_compat_name(node))
186
187 prop = Prop(['rockchip,rk3399-sdhci-5.1', 'arasan,sdhci-5.1', 'third'])
188 node = Node({'compatible': prop})
Walter Lozano5fe734c2020-07-23 00:22:03 -0300189 self.assertEqual((['rockchip_rk3399_sdhci_5_1',
Simon Glassa60cabd2020-12-28 20:34:47 -0700190 'arasan_sdhci_5_1', 'third']),
Simon Glass9d2eb922017-06-18 22:09:06 -0600191 get_compat_name(node))
192
193 def test_empty_file(self):
194 """Test output from a device tree file with no nodes"""
195 dtb_file = get_dtb_file('dtoc_test_empty.dts')
196 output = tools.GetOutputFilename('output')
Simon Glass768ff0a2021-02-03 06:00:51 -0700197
198 # Run this one without saved_scan to complete test coverage
199 dtb_platdata.run_steps(['struct'], dtb_file, False, output, [], True)
Simon Glass9d2eb922017-06-18 22:09:06 -0600200 with open(output) as infile:
201 lines = infile.read().splitlines()
Simon Glasseb37e2d2017-11-12 21:52:17 -0700202 self.assertEqual(HEADER.splitlines(), lines)
Simon Glass9d2eb922017-06-18 22:09:06 -0600203
Walter Lozanoa324e412020-06-25 01:10:08 -0300204 self.run_test(['platdata'], dtb_file, output)
Simon Glass9d2eb922017-06-18 22:09:06 -0600205 with open(output) as infile:
206 lines = infile.read().splitlines()
Simon Glass2500de22020-12-28 20:35:05 -0700207 self.assertEqual(C_HEADER.splitlines() + [''], lines)
Simon Glass9d2eb922017-06-18 22:09:06 -0600208
Simon Glass07ee48e2020-12-28 20:34:49 -0700209 struct_text = HEADER + '''
Simon Glass90e5f0a2017-08-29 14:15:51 -0600210struct dtd_sandbox_i2c_test {
211};
212struct dtd_sandbox_pmic_test {
213\tbool\t\tlow_power;
214\tfdt64_t\t\treg[2];
215};
Simon Glass9d2eb922017-06-18 22:09:06 -0600216struct dtd_sandbox_spl_test {
Simon Glass7f5e2262020-07-07 21:32:06 -0600217\tconst char * acpi_name;
Simon Glass9d2eb922017-06-18 22:09:06 -0600218\tbool\t\tboolval;
219\tunsigned char\tbytearray[3];
220\tunsigned char\tbyteval;
221\tfdt32_t\t\tintarray[4];
222\tfdt32_t\t\tintval;
223\tunsigned char\tlongbytearray[9];
Simon Glass9c526332018-07-06 10:27:28 -0600224\tunsigned char\tnotstring[5];
Simon Glass9d2eb922017-06-18 22:09:06 -0600225\tconst char *\tstringarray[3];
226\tconst char *\tstringval;
227};
Simon Glass07ee48e2020-12-28 20:34:49 -0700228'''
Simon Glass9d2eb922017-06-18 22:09:06 -0600229
Simon Glass07ee48e2020-12-28 20:34:49 -0700230 platdata_text = C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600231/* Node /i2c@0 index 0 */
232static struct dtd_sandbox_i2c_test dtv_i2c_at_0 = {
233};
Simon Glass1d8364a2020-12-28 20:34:54 -0700234U_BOOT_DRVINFO(i2c_at_0) = {
Simon Glass192f8132020-10-03 11:31:25 -0600235\t.name\t\t= "sandbox_i2c_test",
Simon Glass71fa5b42020-12-03 16:55:18 -0700236\t.plat\t= &dtv_i2c_at_0,
Simon Glass39edb952020-12-03 16:55:19 -0700237\t.plat_size\t= sizeof(dtv_i2c_at_0),
Simon Glass36b15e22020-10-03 11:31:35 -0600238\t.parent_idx\t= -1,
Simon Glass192f8132020-10-03 11:31:25 -0600239};
240
241/* Node /i2c@0/pmic@9 index 1 */
242static struct dtd_sandbox_pmic_test dtv_pmic_at_9 = {
243\t.low_power\t\t= true,
244\t.reg\t\t\t= {0x9, 0x0},
245};
Simon Glass1d8364a2020-12-28 20:34:54 -0700246U_BOOT_DRVINFO(pmic_at_9) = {
Simon Glass192f8132020-10-03 11:31:25 -0600247\t.name\t\t= "sandbox_pmic_test",
Simon Glass71fa5b42020-12-03 16:55:18 -0700248\t.plat\t= &dtv_pmic_at_9,
Simon Glass39edb952020-12-03 16:55:19 -0700249\t.plat_size\t= sizeof(dtv_pmic_at_9),
Simon Glass36b15e22020-10-03 11:31:35 -0600250\t.parent_idx\t= 0,
Simon Glass192f8132020-10-03 11:31:25 -0600251};
252
253/* Node /spl-test index 2 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300254static struct dtd_sandbox_spl_test dtv_spl_test = {
Simon Glassc82de562019-05-17 22:00:32 -0600255\t.boolval\t\t= true,
Simon Glass9d2eb922017-06-18 22:09:06 -0600256\t.bytearray\t\t= {0x6, 0x0, 0x0},
257\t.byteval\t\t= 0x5,
Simon Glassc82de562019-05-17 22:00:32 -0600258\t.intarray\t\t= {0x2, 0x3, 0x4, 0x0},
Simon Glass9d2eb922017-06-18 22:09:06 -0600259\t.intval\t\t\t= 0x1,
Simon Glass131e0b02017-08-29 14:15:49 -0600260\t.longbytearray\t\t= {0x9, 0xa, 0xb, 0xc, 0xd, 0xe, 0xf, 0x10,
261\t\t0x11},
Simon Glassc82de562019-05-17 22:00:32 -0600262\t.notstring\t\t= {0x20, 0x21, 0x22, 0x10, 0x0},
Simon Glass9d2eb922017-06-18 22:09:06 -0600263\t.stringarray\t\t= {"multi-word", "message", ""},
Simon Glassc82de562019-05-17 22:00:32 -0600264\t.stringval\t\t= "message",
Simon Glass9d2eb922017-06-18 22:09:06 -0600265};
Simon Glass1d8364a2020-12-28 20:34:54 -0700266U_BOOT_DRVINFO(spl_test) = {
Simon Glass9d2eb922017-06-18 22:09:06 -0600267\t.name\t\t= "sandbox_spl_test",
Simon Glass71fa5b42020-12-03 16:55:18 -0700268\t.plat\t= &dtv_spl_test,
Simon Glass39edb952020-12-03 16:55:19 -0700269\t.plat_size\t= sizeof(dtv_spl_test),
Simon Glass36b15e22020-10-03 11:31:35 -0600270\t.parent_idx\t= -1,
Simon Glass9d2eb922017-06-18 22:09:06 -0600271};
272
Simon Glass192f8132020-10-03 11:31:25 -0600273/* Node /spl-test2 index 3 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300274static struct dtd_sandbox_spl_test dtv_spl_test2 = {
Simon Glass7f5e2262020-07-07 21:32:06 -0600275\t.acpi_name\t\t= "\\\\_SB.GPO0",
Simon Glass9d2eb922017-06-18 22:09:06 -0600276\t.bytearray\t\t= {0x1, 0x23, 0x34},
277\t.byteval\t\t= 0x8,
Simon Glassc82de562019-05-17 22:00:32 -0600278\t.intarray\t\t= {0x5, 0x0, 0x0, 0x0},
Simon Glass9d2eb922017-06-18 22:09:06 -0600279\t.intval\t\t\t= 0x3,
Simon Glass8034e4d2020-10-03 11:31:27 -0600280\t.longbytearray\t\t= {0x9, 0xa, 0xb, 0xc, 0x0, 0x0, 0x0, 0x0,
Simon Glass131e0b02017-08-29 14:15:49 -0600281\t\t0x0},
Simon Glass9d2eb922017-06-18 22:09:06 -0600282\t.stringarray\t\t= {"another", "multi-word", "message"},
Simon Glassc82de562019-05-17 22:00:32 -0600283\t.stringval\t\t= "message2",
Simon Glass9d2eb922017-06-18 22:09:06 -0600284};
Simon Glass1d8364a2020-12-28 20:34:54 -0700285U_BOOT_DRVINFO(spl_test2) = {
Simon Glass9d2eb922017-06-18 22:09:06 -0600286\t.name\t\t= "sandbox_spl_test",
Simon Glass71fa5b42020-12-03 16:55:18 -0700287\t.plat\t= &dtv_spl_test2,
Simon Glass39edb952020-12-03 16:55:19 -0700288\t.plat_size\t= sizeof(dtv_spl_test2),
Simon Glass36b15e22020-10-03 11:31:35 -0600289\t.parent_idx\t= -1,
Simon Glass9d2eb922017-06-18 22:09:06 -0600290};
291
Simon Glass192f8132020-10-03 11:31:25 -0600292/* Node /spl-test3 index 4 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300293static struct dtd_sandbox_spl_test dtv_spl_test3 = {
Simon Glass8034e4d2020-10-03 11:31:27 -0600294\t.longbytearray\t\t= {0x9, 0xa, 0xb, 0xc, 0xd, 0xe, 0xf, 0x10,
295\t\t0x0},
Simon Glass9d2eb922017-06-18 22:09:06 -0600296\t.stringarray\t\t= {"one", "", ""},
297};
Simon Glass1d8364a2020-12-28 20:34:54 -0700298U_BOOT_DRVINFO(spl_test3) = {
Simon Glass9d2eb922017-06-18 22:09:06 -0600299\t.name\t\t= "sandbox_spl_test",
Simon Glass71fa5b42020-12-03 16:55:18 -0700300\t.plat\t= &dtv_spl_test3,
Simon Glass39edb952020-12-03 16:55:19 -0700301\t.plat_size\t= sizeof(dtv_spl_test3),
Simon Glass36b15e22020-10-03 11:31:35 -0600302\t.parent_idx\t= -1,
Simon Glass9d2eb922017-06-18 22:09:06 -0600303};
304
Simon Glass2500de22020-12-28 20:35:05 -0700305'''
Simon Glass07ee48e2020-12-28 20:34:49 -0700306
307 def test_simple(self):
308 """Test output from some simple nodes with various types of data"""
309 dtb_file = get_dtb_file('dtoc_test_simple.dts')
310 output = tools.GetOutputFilename('output')
311 self.run_test(['struct'], dtb_file, output)
312 with open(output) as infile:
313 data = infile.read()
314
315 self._check_strings(self.struct_text, data)
316
317 self.run_test(['platdata'], dtb_file, output)
318 with open(output) as infile:
319 data = infile.read()
320
321 self._check_strings(self.platdata_text, data)
Simon Glass9d2eb922017-06-18 22:09:06 -0600322
Simon Glass4e8e8462020-12-28 20:34:52 -0700323 # Try the 'all' command
324 self.run_test(['all'], dtb_file, output)
325 data = tools.ReadFile(output, binary=False)
326 self._check_strings(self.platdata_text + self.struct_text, data)
327
Walter Lozanoe675d962020-07-03 08:07:17 -0300328 def test_driver_alias(self):
329 """Test output from a device tree file with a driver alias"""
330 dtb_file = get_dtb_file('dtoc_test_driver_alias.dts')
331 output = tools.GetOutputFilename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300332 self.run_test(['struct'], dtb_file, output)
Walter Lozanoe675d962020-07-03 08:07:17 -0300333 with open(output) as infile:
334 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700335 self._check_strings(HEADER + '''
Walter Lozanoe675d962020-07-03 08:07:17 -0300336struct dtd_sandbox_gpio {
337\tconst char *\tgpio_bank_name;
338\tbool\t\tgpio_controller;
339\tfdt32_t\t\tsandbox_gpio_count;
340};
Walter Lozanoe675d962020-07-03 08:07:17 -0300341''', data)
342
Walter Lozanoa324e412020-06-25 01:10:08 -0300343 self.run_test(['platdata'], dtb_file, output)
Walter Lozanoe675d962020-07-03 08:07:17 -0300344 with open(output) as infile:
345 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700346 self._check_strings(C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600347/* Node /gpios@0 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300348static struct dtd_sandbox_gpio dtv_gpios_at_0 = {
Walter Lozanoe675d962020-07-03 08:07:17 -0300349\t.gpio_bank_name\t\t= "a",
350\t.gpio_controller\t= true,
351\t.sandbox_gpio_count\t= 0x14,
352};
Simon Glass1d8364a2020-12-28 20:34:54 -0700353U_BOOT_DRVINFO(gpios_at_0) = {
Walter Lozanoe675d962020-07-03 08:07:17 -0300354\t.name\t\t= "sandbox_gpio",
Simon Glass71fa5b42020-12-03 16:55:18 -0700355\t.plat\t= &dtv_gpios_at_0,
Simon Glass39edb952020-12-03 16:55:19 -0700356\t.plat_size\t= sizeof(dtv_gpios_at_0),
Simon Glass36b15e22020-10-03 11:31:35 -0600357\t.parent_idx\t= -1,
Walter Lozanoe675d962020-07-03 08:07:17 -0300358};
359
360''', data)
361
Walter Lozanoa324e412020-06-25 01:10:08 -0300362 def test_invalid_driver(self):
363 """Test output from a device tree file with an invalid driver"""
364 dtb_file = get_dtb_file('dtoc_test_invalid_driver.dts')
365 output = tools.GetOutputFilename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -0700366 with test_util.capture_sys_output() as _:
Simon Glass768ff0a2021-02-03 06:00:51 -0700367 dtb_platdata.run_steps(['struct'], dtb_file, False, output, [],
368 scan=copy_scan())
Walter Lozanoa324e412020-06-25 01:10:08 -0300369 with open(output) as infile:
370 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700371 self._check_strings(HEADER + '''
Walter Lozanoa324e412020-06-25 01:10:08 -0300372struct dtd_invalid {
373};
374''', data)
375
Simon Glassa60cabd2020-12-28 20:34:47 -0700376 with test_util.capture_sys_output() as _:
Simon Glass768ff0a2021-02-03 06:00:51 -0700377 dtb_platdata.run_steps(['platdata'], dtb_file, False, output, [],
378 scan=copy_scan())
Walter Lozanoa324e412020-06-25 01:10:08 -0300379 with open(output) as infile:
380 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700381 self._check_strings(C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600382/* Node /spl-test index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300383static struct dtd_invalid dtv_spl_test = {
Walter Lozanoa324e412020-06-25 01:10:08 -0300384};
Simon Glass1d8364a2020-12-28 20:34:54 -0700385U_BOOT_DRVINFO(spl_test) = {
Walter Lozanoa324e412020-06-25 01:10:08 -0300386\t.name\t\t= "invalid",
Simon Glass71fa5b42020-12-03 16:55:18 -0700387\t.plat\t= &dtv_spl_test,
Simon Glass39edb952020-12-03 16:55:19 -0700388\t.plat_size\t= sizeof(dtv_spl_test),
Simon Glass36b15e22020-10-03 11:31:35 -0600389\t.parent_idx\t= -1,
Walter Lozanoa324e412020-06-25 01:10:08 -0300390};
391
392''', data)
393
Simon Glass9d2eb922017-06-18 22:09:06 -0600394 def test_phandle(self):
395 """Test output from a node containing a phandle reference"""
396 dtb_file = get_dtb_file('dtoc_test_phandle.dts')
397 output = tools.GetOutputFilename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300398 self.run_test(['struct'], dtb_file, output)
Simon Glass9d2eb922017-06-18 22:09:06 -0600399 with open(output) as infile:
400 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700401 self._check_strings(HEADER + '''
Simon Glass9d2eb922017-06-18 22:09:06 -0600402struct dtd_source {
Simon Glass3deeb472017-08-29 14:15:59 -0600403\tstruct phandle_2_arg clocks[4];
Simon Glass9d2eb922017-06-18 22:09:06 -0600404};
405struct dtd_target {
406\tfdt32_t\t\tintval;
407};
408''', data)
409
Walter Lozanoa324e412020-06-25 01:10:08 -0300410 self.run_test(['platdata'], dtb_file, output)
Simon Glass9d2eb922017-06-18 22:09:06 -0600411 with open(output) as infile:
412 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700413 self._check_strings(C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600414/* Node /phandle2-target index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300415static struct dtd_target dtv_phandle2_target = {
Simon Glass3deeb472017-08-29 14:15:59 -0600416\t.intval\t\t\t= 0x1,
417};
Simon Glass1d8364a2020-12-28 20:34:54 -0700418U_BOOT_DRVINFO(phandle2_target) = {
Simon Glass3deeb472017-08-29 14:15:59 -0600419\t.name\t\t= "target",
Simon Glass71fa5b42020-12-03 16:55:18 -0700420\t.plat\t= &dtv_phandle2_target,
Simon Glass39edb952020-12-03 16:55:19 -0700421\t.plat_size\t= sizeof(dtv_phandle2_target),
Simon Glass36b15e22020-10-03 11:31:35 -0600422\t.parent_idx\t= -1,
Simon Glass3deeb472017-08-29 14:15:59 -0600423};
424
Simon Glass192f8132020-10-03 11:31:25 -0600425/* Node /phandle3-target index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300426static struct dtd_target dtv_phandle3_target = {
Simon Glass3deeb472017-08-29 14:15:59 -0600427\t.intval\t\t\t= 0x2,
428};
Simon Glass1d8364a2020-12-28 20:34:54 -0700429U_BOOT_DRVINFO(phandle3_target) = {
Simon Glass3deeb472017-08-29 14:15:59 -0600430\t.name\t\t= "target",
Simon Glass71fa5b42020-12-03 16:55:18 -0700431\t.plat\t= &dtv_phandle3_target,
Simon Glass39edb952020-12-03 16:55:19 -0700432\t.plat_size\t= sizeof(dtv_phandle3_target),
Simon Glass36b15e22020-10-03 11:31:35 -0600433\t.parent_idx\t= -1,
Simon Glass3deeb472017-08-29 14:15:59 -0600434};
435
Simon Glass192f8132020-10-03 11:31:25 -0600436/* Node /phandle-source index 2 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300437static struct dtd_source dtv_phandle_source = {
Simon Glassd0cd0752017-08-29 14:15:57 -0600438\t.clocks\t\t\t= {
Simon Glass5792f4b2020-10-03 11:31:40 -0600439\t\t\t{4, {}},
440\t\t\t{0, {11}},
441\t\t\t{1, {12, 13}},
442\t\t\t{4, {}},},
Simon Glass9d2eb922017-06-18 22:09:06 -0600443};
Simon Glass1d8364a2020-12-28 20:34:54 -0700444U_BOOT_DRVINFO(phandle_source) = {
Simon Glass9d2eb922017-06-18 22:09:06 -0600445\t.name\t\t= "source",
Simon Glass71fa5b42020-12-03 16:55:18 -0700446\t.plat\t= &dtv_phandle_source,
Simon Glass39edb952020-12-03 16:55:19 -0700447\t.plat_size\t= sizeof(dtv_phandle_source),
Simon Glass36b15e22020-10-03 11:31:35 -0600448\t.parent_idx\t= -1,
Simon Glass9d2eb922017-06-18 22:09:06 -0600449};
450
Simon Glass192f8132020-10-03 11:31:25 -0600451/* Node /phandle-source2 index 3 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300452static struct dtd_source dtv_phandle_source2 = {
Simon Glass609e2b12018-07-06 10:27:31 -0600453\t.clocks\t\t\t= {
Simon Glass5792f4b2020-10-03 11:31:40 -0600454\t\t\t{4, {}},},
Simon Glass609e2b12018-07-06 10:27:31 -0600455};
Simon Glass1d8364a2020-12-28 20:34:54 -0700456U_BOOT_DRVINFO(phandle_source2) = {
Simon Glass609e2b12018-07-06 10:27:31 -0600457\t.name\t\t= "source",
Simon Glass71fa5b42020-12-03 16:55:18 -0700458\t.plat\t= &dtv_phandle_source2,
Simon Glass39edb952020-12-03 16:55:19 -0700459\t.plat_size\t= sizeof(dtv_phandle_source2),
Simon Glass36b15e22020-10-03 11:31:35 -0600460\t.parent_idx\t= -1,
Simon Glass609e2b12018-07-06 10:27:31 -0600461};
462
Simon Glass16382ce2020-12-28 20:35:04 -0700463/* Node /phandle-target index 4 */
464static struct dtd_target dtv_phandle_target = {
465\t.intval\t\t\t= 0x0,
466};
467U_BOOT_DRVINFO(phandle_target) = {
468\t.name\t\t= "target",
469\t.plat\t= &dtv_phandle_target,
470\t.plat_size\t= sizeof(dtv_phandle_target),
471\t.parent_idx\t= -1,
472};
473
Simon Glass9d2eb922017-06-18 22:09:06 -0600474''', data)
475
Simon Glass961c1ce2018-07-06 10:27:35 -0600476 def test_phandle_single(self):
477 """Test output from a node containing a phandle reference"""
478 dtb_file = get_dtb_file('dtoc_test_phandle_single.dts')
479 output = tools.GetOutputFilename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300480 self.run_test(['struct'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -0600481 with open(output) as infile:
482 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700483 self._check_strings(HEADER + '''
Simon Glass961c1ce2018-07-06 10:27:35 -0600484struct dtd_source {
485\tstruct phandle_0_arg clocks[1];
486};
487struct dtd_target {
488\tfdt32_t\t\tintval;
489};
490''', data)
491
492 def test_phandle_reorder(self):
493 """Test that phandle targets are generated before their references"""
494 dtb_file = get_dtb_file('dtoc_test_phandle_reorder.dts')
495 output = tools.GetOutputFilename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300496 self.run_test(['platdata'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -0600497 with open(output) as infile:
498 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700499 self._check_strings(C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600500/* Node /phandle-source2 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300501static struct dtd_source dtv_phandle_source2 = {
Simon Glass961c1ce2018-07-06 10:27:35 -0600502\t.clocks\t\t\t= {
Simon Glass5792f4b2020-10-03 11:31:40 -0600503\t\t\t{1, {}},},
Simon Glass961c1ce2018-07-06 10:27:35 -0600504};
Simon Glass1d8364a2020-12-28 20:34:54 -0700505U_BOOT_DRVINFO(phandle_source2) = {
Simon Glass961c1ce2018-07-06 10:27:35 -0600506\t.name\t\t= "source",
Simon Glass71fa5b42020-12-03 16:55:18 -0700507\t.plat\t= &dtv_phandle_source2,
Simon Glass39edb952020-12-03 16:55:19 -0700508\t.plat_size\t= sizeof(dtv_phandle_source2),
Simon Glass36b15e22020-10-03 11:31:35 -0600509\t.parent_idx\t= -1,
Simon Glass961c1ce2018-07-06 10:27:35 -0600510};
511
Simon Glass16382ce2020-12-28 20:35:04 -0700512/* Node /phandle-target index 1 */
513static struct dtd_target dtv_phandle_target = {
514};
515U_BOOT_DRVINFO(phandle_target) = {
516\t.name\t\t= "target",
517\t.plat\t= &dtv_phandle_target,
518\t.plat_size\t= sizeof(dtv_phandle_target),
519\t.parent_idx\t= -1,
520};
521
Simon Glass961c1ce2018-07-06 10:27:35 -0600522''', data)
523
Walter Lozano5541fc02020-06-25 01:10:17 -0300524 def test_phandle_cd_gpio(self):
525 """Test that phandle targets are generated when unsing cd-gpios"""
526 dtb_file = get_dtb_file('dtoc_test_phandle_cd_gpios.dts')
527 output = tools.GetOutputFilename('output')
Simon Glass768ff0a2021-02-03 06:00:51 -0700528 dtb_platdata.run_steps(['platdata'], dtb_file, False, output, [], True,
529 scan=copy_scan())
Walter Lozano5541fc02020-06-25 01:10:17 -0300530 with open(output) as infile:
531 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700532 self._check_strings(C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600533/* Node /phandle2-target index 0 */
Walter Lozano5541fc02020-06-25 01:10:17 -0300534static struct dtd_target dtv_phandle2_target = {
535\t.intval\t\t\t= 0x1,
536};
Simon Glass1d8364a2020-12-28 20:34:54 -0700537U_BOOT_DRVINFO(phandle2_target) = {
Walter Lozano5541fc02020-06-25 01:10:17 -0300538\t.name\t\t= "target",
Simon Glass71fa5b42020-12-03 16:55:18 -0700539\t.plat\t= &dtv_phandle2_target,
Simon Glass39edb952020-12-03 16:55:19 -0700540\t.plat_size\t= sizeof(dtv_phandle2_target),
Simon Glass36b15e22020-10-03 11:31:35 -0600541\t.parent_idx\t= -1,
Walter Lozano5541fc02020-06-25 01:10:17 -0300542};
543
Simon Glass192f8132020-10-03 11:31:25 -0600544/* Node /phandle3-target index 1 */
Walter Lozano5541fc02020-06-25 01:10:17 -0300545static struct dtd_target dtv_phandle3_target = {
546\t.intval\t\t\t= 0x2,
547};
Simon Glass1d8364a2020-12-28 20:34:54 -0700548U_BOOT_DRVINFO(phandle3_target) = {
Walter Lozano5541fc02020-06-25 01:10:17 -0300549\t.name\t\t= "target",
Simon Glass71fa5b42020-12-03 16:55:18 -0700550\t.plat\t= &dtv_phandle3_target,
Simon Glass39edb952020-12-03 16:55:19 -0700551\t.plat_size\t= sizeof(dtv_phandle3_target),
Simon Glass36b15e22020-10-03 11:31:35 -0600552\t.parent_idx\t= -1,
Walter Lozano5541fc02020-06-25 01:10:17 -0300553};
554
Simon Glass192f8132020-10-03 11:31:25 -0600555/* Node /phandle-source index 2 */
Walter Lozano5541fc02020-06-25 01:10:17 -0300556static struct dtd_source dtv_phandle_source = {
557\t.cd_gpios\t\t= {
Simon Glass5792f4b2020-10-03 11:31:40 -0600558\t\t\t{4, {}},
559\t\t\t{0, {11}},
560\t\t\t{1, {12, 13}},
561\t\t\t{4, {}},},
Walter Lozano5541fc02020-06-25 01:10:17 -0300562};
Simon Glass1d8364a2020-12-28 20:34:54 -0700563U_BOOT_DRVINFO(phandle_source) = {
Walter Lozano5541fc02020-06-25 01:10:17 -0300564\t.name\t\t= "source",
Simon Glass71fa5b42020-12-03 16:55:18 -0700565\t.plat\t= &dtv_phandle_source,
Simon Glass39edb952020-12-03 16:55:19 -0700566\t.plat_size\t= sizeof(dtv_phandle_source),
Simon Glass36b15e22020-10-03 11:31:35 -0600567\t.parent_idx\t= -1,
Walter Lozano5541fc02020-06-25 01:10:17 -0300568};
569
Simon Glass192f8132020-10-03 11:31:25 -0600570/* Node /phandle-source2 index 3 */
Walter Lozano5541fc02020-06-25 01:10:17 -0300571static struct dtd_source dtv_phandle_source2 = {
572\t.cd_gpios\t\t= {
Simon Glass5792f4b2020-10-03 11:31:40 -0600573\t\t\t{4, {}},},
Walter Lozano5541fc02020-06-25 01:10:17 -0300574};
Simon Glass1d8364a2020-12-28 20:34:54 -0700575U_BOOT_DRVINFO(phandle_source2) = {
Walter Lozano5541fc02020-06-25 01:10:17 -0300576\t.name\t\t= "source",
Simon Glass71fa5b42020-12-03 16:55:18 -0700577\t.plat\t= &dtv_phandle_source2,
Simon Glass39edb952020-12-03 16:55:19 -0700578\t.plat_size\t= sizeof(dtv_phandle_source2),
Simon Glass36b15e22020-10-03 11:31:35 -0600579\t.parent_idx\t= -1,
Walter Lozano5541fc02020-06-25 01:10:17 -0300580};
581
Simon Glass16382ce2020-12-28 20:35:04 -0700582/* Node /phandle-target index 4 */
583static struct dtd_target dtv_phandle_target = {
584\t.intval\t\t\t= 0x0,
585};
586U_BOOT_DRVINFO(phandle_target) = {
587\t.name\t\t= "target",
588\t.plat\t= &dtv_phandle_target,
589\t.plat_size\t= sizeof(dtv_phandle_target),
590\t.parent_idx\t= -1,
591};
592
Walter Lozano5541fc02020-06-25 01:10:17 -0300593''', data)
594
Simon Glass961c1ce2018-07-06 10:27:35 -0600595 def test_phandle_bad(self):
596 """Test a node containing an invalid phandle fails"""
Simon Glass04150022018-10-01 21:12:43 -0600597 dtb_file = get_dtb_file('dtoc_test_phandle_bad.dts',
598 capture_stderr=True)
Simon Glass961c1ce2018-07-06 10:27:35 -0600599 output = tools.GetOutputFilename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -0700600 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -0300601 self.run_test(['struct'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -0600602 self.assertIn("Cannot parse 'clocks' in node 'phandle-source'",
Simon Glassa60cabd2020-12-28 20:34:47 -0700603 str(exc.exception))
Simon Glass961c1ce2018-07-06 10:27:35 -0600604
605 def test_phandle_bad2(self):
606 """Test a phandle target missing its #*-cells property"""
Simon Glass04150022018-10-01 21:12:43 -0600607 dtb_file = get_dtb_file('dtoc_test_phandle_bad2.dts',
608 capture_stderr=True)
Simon Glass961c1ce2018-07-06 10:27:35 -0600609 output = tools.GetOutputFilename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -0700610 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -0300611 self.run_test(['struct'], dtb_file, output)
Walter Lozano179f0b62020-06-25 01:10:16 -0300612 self.assertIn("Node 'phandle-target' has no cells property",
Simon Glassa60cabd2020-12-28 20:34:47 -0700613 str(exc.exception))
Simon Glass961c1ce2018-07-06 10:27:35 -0600614
Simon Glass1b1fe412017-08-29 14:15:50 -0600615 def test_addresses64(self):
616 """Test output from a node with a 'reg' property with na=2, ns=2"""
617 dtb_file = get_dtb_file('dtoc_test_addr64.dts')
618 output = tools.GetOutputFilename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300619 self.run_test(['struct'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -0600620 with open(output) as infile:
621 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700622 self._check_strings(HEADER + '''
Simon Glass1b1fe412017-08-29 14:15:50 -0600623struct dtd_test1 {
624\tfdt64_t\t\treg[2];
625};
626struct dtd_test2 {
627\tfdt64_t\t\treg[2];
628};
629struct dtd_test3 {
630\tfdt64_t\t\treg[4];
631};
632''', data)
633
Walter Lozanoa324e412020-06-25 01:10:08 -0300634 self.run_test(['platdata'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -0600635 with open(output) as infile:
636 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700637 self._check_strings(C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600638/* Node /test1 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300639static struct dtd_test1 dtv_test1 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600640\t.reg\t\t\t= {0x1234, 0x5678},
641};
Simon Glass1d8364a2020-12-28 20:34:54 -0700642U_BOOT_DRVINFO(test1) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600643\t.name\t\t= "test1",
Simon Glass71fa5b42020-12-03 16:55:18 -0700644\t.plat\t= &dtv_test1,
Simon Glass39edb952020-12-03 16:55:19 -0700645\t.plat_size\t= sizeof(dtv_test1),
Simon Glass36b15e22020-10-03 11:31:35 -0600646\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600647};
648
Simon Glass192f8132020-10-03 11:31:25 -0600649/* Node /test2 index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300650static struct dtd_test2 dtv_test2 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600651\t.reg\t\t\t= {0x1234567890123456, 0x9876543210987654},
652};
Simon Glass1d8364a2020-12-28 20:34:54 -0700653U_BOOT_DRVINFO(test2) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600654\t.name\t\t= "test2",
Simon Glass71fa5b42020-12-03 16:55:18 -0700655\t.plat\t= &dtv_test2,
Simon Glass39edb952020-12-03 16:55:19 -0700656\t.plat_size\t= sizeof(dtv_test2),
Simon Glass36b15e22020-10-03 11:31:35 -0600657\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600658};
659
Simon Glass192f8132020-10-03 11:31:25 -0600660/* Node /test3 index 2 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300661static struct dtd_test3 dtv_test3 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600662\t.reg\t\t\t= {0x1234567890123456, 0x9876543210987654, 0x2, 0x3},
663};
Simon Glass1d8364a2020-12-28 20:34:54 -0700664U_BOOT_DRVINFO(test3) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600665\t.name\t\t= "test3",
Simon Glass71fa5b42020-12-03 16:55:18 -0700666\t.plat\t= &dtv_test3,
Simon Glass39edb952020-12-03 16:55:19 -0700667\t.plat_size\t= sizeof(dtv_test3),
Simon Glass36b15e22020-10-03 11:31:35 -0600668\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600669};
670
Simon Glass2500de22020-12-28 20:35:05 -0700671''', data)
Simon Glass1b1fe412017-08-29 14:15:50 -0600672
673 def test_addresses32(self):
674 """Test output from a node with a 'reg' property with na=1, ns=1"""
675 dtb_file = get_dtb_file('dtoc_test_addr32.dts')
676 output = tools.GetOutputFilename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300677 self.run_test(['struct'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -0600678 with open(output) as infile:
679 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700680 self._check_strings(HEADER + '''
Simon Glass1b1fe412017-08-29 14:15:50 -0600681struct dtd_test1 {
682\tfdt32_t\t\treg[2];
683};
684struct dtd_test2 {
685\tfdt32_t\t\treg[4];
686};
687''', data)
688
Walter Lozanoa324e412020-06-25 01:10:08 -0300689 self.run_test(['platdata'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -0600690 with open(output) as infile:
691 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700692 self._check_strings(C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600693/* Node /test1 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300694static struct dtd_test1 dtv_test1 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600695\t.reg\t\t\t= {0x1234, 0x5678},
696};
Simon Glass1d8364a2020-12-28 20:34:54 -0700697U_BOOT_DRVINFO(test1) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600698\t.name\t\t= "test1",
Simon Glass71fa5b42020-12-03 16:55:18 -0700699\t.plat\t= &dtv_test1,
Simon Glass39edb952020-12-03 16:55:19 -0700700\t.plat_size\t= sizeof(dtv_test1),
Simon Glass36b15e22020-10-03 11:31:35 -0600701\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600702};
703
Simon Glass192f8132020-10-03 11:31:25 -0600704/* Node /test2 index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300705static struct dtd_test2 dtv_test2 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600706\t.reg\t\t\t= {0x12345678, 0x98765432, 0x2, 0x3},
707};
Simon Glass1d8364a2020-12-28 20:34:54 -0700708U_BOOT_DRVINFO(test2) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600709\t.name\t\t= "test2",
Simon Glass71fa5b42020-12-03 16:55:18 -0700710\t.plat\t= &dtv_test2,
Simon Glass39edb952020-12-03 16:55:19 -0700711\t.plat_size\t= sizeof(dtv_test2),
Simon Glass36b15e22020-10-03 11:31:35 -0600712\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600713};
714
Simon Glass2500de22020-12-28 20:35:05 -0700715''', data)
Simon Glass1b1fe412017-08-29 14:15:50 -0600716
717 def test_addresses64_32(self):
718 """Test output from a node with a 'reg' property with na=2, ns=1"""
719 dtb_file = get_dtb_file('dtoc_test_addr64_32.dts')
720 output = tools.GetOutputFilename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300721 self.run_test(['struct'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -0600722 with open(output) as infile:
723 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700724 self._check_strings(HEADER + '''
Simon Glass1b1fe412017-08-29 14:15:50 -0600725struct dtd_test1 {
726\tfdt64_t\t\treg[2];
727};
728struct dtd_test2 {
729\tfdt64_t\t\treg[2];
730};
731struct dtd_test3 {
732\tfdt64_t\t\treg[4];
733};
734''', data)
735
Walter Lozanoa324e412020-06-25 01:10:08 -0300736 self.run_test(['platdata'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -0600737 with open(output) as infile:
738 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700739 self._check_strings(C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600740/* Node /test1 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300741static struct dtd_test1 dtv_test1 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600742\t.reg\t\t\t= {0x123400000000, 0x5678},
743};
Simon Glass1d8364a2020-12-28 20:34:54 -0700744U_BOOT_DRVINFO(test1) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600745\t.name\t\t= "test1",
Simon Glass71fa5b42020-12-03 16:55:18 -0700746\t.plat\t= &dtv_test1,
Simon Glass39edb952020-12-03 16:55:19 -0700747\t.plat_size\t= sizeof(dtv_test1),
Simon Glass36b15e22020-10-03 11:31:35 -0600748\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600749};
750
Simon Glass192f8132020-10-03 11:31:25 -0600751/* Node /test2 index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300752static struct dtd_test2 dtv_test2 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600753\t.reg\t\t\t= {0x1234567890123456, 0x98765432},
754};
Simon Glass1d8364a2020-12-28 20:34:54 -0700755U_BOOT_DRVINFO(test2) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600756\t.name\t\t= "test2",
Simon Glass71fa5b42020-12-03 16:55:18 -0700757\t.plat\t= &dtv_test2,
Simon Glass39edb952020-12-03 16:55:19 -0700758\t.plat_size\t= sizeof(dtv_test2),
Simon Glass36b15e22020-10-03 11:31:35 -0600759\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600760};
761
Simon Glass192f8132020-10-03 11:31:25 -0600762/* Node /test3 index 2 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300763static struct dtd_test3 dtv_test3 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600764\t.reg\t\t\t= {0x1234567890123456, 0x98765432, 0x2, 0x3},
765};
Simon Glass1d8364a2020-12-28 20:34:54 -0700766U_BOOT_DRVINFO(test3) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600767\t.name\t\t= "test3",
Simon Glass71fa5b42020-12-03 16:55:18 -0700768\t.plat\t= &dtv_test3,
Simon Glass39edb952020-12-03 16:55:19 -0700769\t.plat_size\t= sizeof(dtv_test3),
Simon Glass36b15e22020-10-03 11:31:35 -0600770\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600771};
772
Simon Glass2500de22020-12-28 20:35:05 -0700773''', data)
Simon Glass1b1fe412017-08-29 14:15:50 -0600774
775 def test_addresses32_64(self):
776 """Test output from a node with a 'reg' property with na=1, ns=2"""
777 dtb_file = get_dtb_file('dtoc_test_addr32_64.dts')
778 output = tools.GetOutputFilename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300779 self.run_test(['struct'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -0600780 with open(output) as infile:
781 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700782 self._check_strings(HEADER + '''
Simon Glass1b1fe412017-08-29 14:15:50 -0600783struct dtd_test1 {
784\tfdt64_t\t\treg[2];
785};
786struct dtd_test2 {
787\tfdt64_t\t\treg[2];
788};
789struct dtd_test3 {
790\tfdt64_t\t\treg[4];
791};
792''', data)
793
Walter Lozanoa324e412020-06-25 01:10:08 -0300794 self.run_test(['platdata'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -0600795 with open(output) as infile:
796 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700797 self._check_strings(C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600798/* Node /test1 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300799static struct dtd_test1 dtv_test1 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600800\t.reg\t\t\t= {0x1234, 0x567800000000},
801};
Simon Glass1d8364a2020-12-28 20:34:54 -0700802U_BOOT_DRVINFO(test1) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600803\t.name\t\t= "test1",
Simon Glass71fa5b42020-12-03 16:55:18 -0700804\t.plat\t= &dtv_test1,
Simon Glass39edb952020-12-03 16:55:19 -0700805\t.plat_size\t= sizeof(dtv_test1),
Simon Glass36b15e22020-10-03 11:31:35 -0600806\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600807};
808
Simon Glass192f8132020-10-03 11:31:25 -0600809/* Node /test2 index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300810static struct dtd_test2 dtv_test2 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600811\t.reg\t\t\t= {0x12345678, 0x9876543210987654},
812};
Simon Glass1d8364a2020-12-28 20:34:54 -0700813U_BOOT_DRVINFO(test2) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600814\t.name\t\t= "test2",
Simon Glass71fa5b42020-12-03 16:55:18 -0700815\t.plat\t= &dtv_test2,
Simon Glass39edb952020-12-03 16:55:19 -0700816\t.plat_size\t= sizeof(dtv_test2),
Simon Glass36b15e22020-10-03 11:31:35 -0600817\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600818};
819
Simon Glass192f8132020-10-03 11:31:25 -0600820/* Node /test3 index 2 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300821static struct dtd_test3 dtv_test3 = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600822\t.reg\t\t\t= {0x12345678, 0x9876543210987654, 0x2, 0x3},
823};
Simon Glass1d8364a2020-12-28 20:34:54 -0700824U_BOOT_DRVINFO(test3) = {
Simon Glass1b1fe412017-08-29 14:15:50 -0600825\t.name\t\t= "test3",
Simon Glass71fa5b42020-12-03 16:55:18 -0700826\t.plat\t= &dtv_test3,
Simon Glass39edb952020-12-03 16:55:19 -0700827\t.plat_size\t= sizeof(dtv_test3),
Simon Glass36b15e22020-10-03 11:31:35 -0600828\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -0600829};
830
Simon Glass2500de22020-12-28 20:35:05 -0700831''', data)
Simon Glass961c1ce2018-07-06 10:27:35 -0600832
833 def test_bad_reg(self):
834 """Test that a reg property with an invalid type generates an error"""
Simon Glass3bce93d2018-07-06 10:27:37 -0600835 # Capture stderr since dtc will emit warnings for this file
836 dtb_file = get_dtb_file('dtoc_test_bad_reg.dts', capture_stderr=True)
Simon Glass961c1ce2018-07-06 10:27:35 -0600837 output = tools.GetOutputFilename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -0700838 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -0300839 self.run_test(['struct'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -0600840 self.assertIn("Node 'spl-test' reg property is not an int",
Simon Glassa60cabd2020-12-28 20:34:47 -0700841 str(exc.exception))
Simon Glass961c1ce2018-07-06 10:27:35 -0600842
843 def test_bad_reg2(self):
844 """Test that a reg property with an invalid cell count is detected"""
Simon Glass3bce93d2018-07-06 10:27:37 -0600845 # Capture stderr since dtc will emit warnings for this file
846 dtb_file = get_dtb_file('dtoc_test_bad_reg2.dts', capture_stderr=True)
Simon Glass961c1ce2018-07-06 10:27:35 -0600847 output = tools.GetOutputFilename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -0700848 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -0300849 self.run_test(['struct'], dtb_file, output)
Simon Glassa60cabd2020-12-28 20:34:47 -0700850 self.assertIn(
851 "Node 'spl-test' reg property has 3 cells which is not a multiple of na + ns = 1 + 1)",
852 str(exc.exception))
Simon Glass961c1ce2018-07-06 10:27:35 -0600853
854 def test_add_prop(self):
855 """Test that a subequent node can add a new property to a struct"""
856 dtb_file = get_dtb_file('dtoc_test_add_prop.dts')
857 output = tools.GetOutputFilename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300858 self.run_test(['struct'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -0600859 with open(output) as infile:
860 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700861 self._check_strings(HEADER + '''
Simon Glass961c1ce2018-07-06 10:27:35 -0600862struct dtd_sandbox_spl_test {
863\tfdt32_t\t\tintarray;
864\tfdt32_t\t\tintval;
865};
866''', data)
867
Walter Lozanoa324e412020-06-25 01:10:08 -0300868 self.run_test(['platdata'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -0600869 with open(output) as infile:
870 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700871 self._check_strings(C_HEADER + '''
Simon Glass192f8132020-10-03 11:31:25 -0600872/* Node /spl-test index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300873static struct dtd_sandbox_spl_test dtv_spl_test = {
Simon Glass961c1ce2018-07-06 10:27:35 -0600874\t.intval\t\t\t= 0x1,
875};
Simon Glass1d8364a2020-12-28 20:34:54 -0700876U_BOOT_DRVINFO(spl_test) = {
Simon Glass961c1ce2018-07-06 10:27:35 -0600877\t.name\t\t= "sandbox_spl_test",
Simon Glass71fa5b42020-12-03 16:55:18 -0700878\t.plat\t= &dtv_spl_test,
Simon Glass39edb952020-12-03 16:55:19 -0700879\t.plat_size\t= sizeof(dtv_spl_test),
Simon Glass36b15e22020-10-03 11:31:35 -0600880\t.parent_idx\t= -1,
Simon Glass961c1ce2018-07-06 10:27:35 -0600881};
882
Simon Glass192f8132020-10-03 11:31:25 -0600883/* Node /spl-test2 index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300884static struct dtd_sandbox_spl_test dtv_spl_test2 = {
Simon Glass961c1ce2018-07-06 10:27:35 -0600885\t.intarray\t\t= 0x5,
886};
Simon Glass1d8364a2020-12-28 20:34:54 -0700887U_BOOT_DRVINFO(spl_test2) = {
Simon Glass961c1ce2018-07-06 10:27:35 -0600888\t.name\t\t= "sandbox_spl_test",
Simon Glass71fa5b42020-12-03 16:55:18 -0700889\t.plat\t= &dtv_spl_test2,
Simon Glass39edb952020-12-03 16:55:19 -0700890\t.plat_size\t= sizeof(dtv_spl_test2),
Simon Glass36b15e22020-10-03 11:31:35 -0600891\t.parent_idx\t= -1,
Simon Glass961c1ce2018-07-06 10:27:35 -0600892};
893
Simon Glass2500de22020-12-28 20:35:05 -0700894''', data)
Simon Glass961c1ce2018-07-06 10:27:35 -0600895
Simon Glassa60cabd2020-12-28 20:34:47 -0700896 def test_stdout(self):
Simon Glass961c1ce2018-07-06 10:27:35 -0600897 """Test output to stdout"""
898 dtb_file = get_dtb_file('dtoc_test_simple.dts')
Simon Glass07ee48e2020-12-28 20:34:49 -0700899 with test_util.capture_sys_output() as (stdout, _):
Simon Glass6ca0c7a2020-12-28 20:34:48 -0700900 self.run_test(['struct'], dtb_file, None)
Simon Glass07ee48e2020-12-28 20:34:49 -0700901 self._check_strings(self.struct_text, stdout.getvalue())
Simon Glass961c1ce2018-07-06 10:27:35 -0600902
Simon Glassc3a310a82020-12-28 20:34:51 -0700903 def test_multi_to_file(self):
904 """Test output of multiple pieces to a single file"""
905 dtb_file = get_dtb_file('dtoc_test_simple.dts')
906 output = tools.GetOutputFilename('output')
Simon Glass4e8e8462020-12-28 20:34:52 -0700907 self.run_test(['all'], dtb_file, output)
Simon Glassc3a310a82020-12-28 20:34:51 -0700908 data = tools.ReadFile(output, binary=False)
Simon Glass4e8e8462020-12-28 20:34:52 -0700909 self._check_strings(self.platdata_text + self.struct_text, data)
Simon Glassc3a310a82020-12-28 20:34:51 -0700910
Simon Glassa60cabd2020-12-28 20:34:47 -0700911 def test_no_command(self):
Simon Glass961c1ce2018-07-06 10:27:35 -0600912 """Test running dtoc without a command"""
Simon Glassa60cabd2020-12-28 20:34:47 -0700913 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -0300914 self.run_test([], '', '')
Simon Glass961c1ce2018-07-06 10:27:35 -0600915 self.assertIn("Please specify a command: struct, platdata",
Simon Glassa60cabd2020-12-28 20:34:47 -0700916 str(exc.exception))
Simon Glass961c1ce2018-07-06 10:27:35 -0600917
Simon Glassa60cabd2020-12-28 20:34:47 -0700918 def test_bad_command(self):
Simon Glass961c1ce2018-07-06 10:27:35 -0600919 """Test running dtoc with an invalid command"""
920 dtb_file = get_dtb_file('dtoc_test_simple.dts')
921 output = tools.GetOutputFilename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -0700922 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -0300923 self.run_test(['invalid-cmd'], dtb_file, output)
Simon Glass4e8e8462020-12-28 20:34:52 -0700924 self.assertIn("Unknown command 'invalid-cmd': (use: platdata, struct)",
Simon Glassa60cabd2020-12-28 20:34:47 -0700925 str(exc.exception))
Walter Lozanod82062b2020-07-28 19:06:23 -0300926
Simon Glass4e8e8462020-12-28 20:34:52 -0700927 def test_output_conflict(self):
928 """Test a conflict between and output dirs and output file"""
929 with self.assertRaises(ValueError) as exc:
Simon Glass768ff0a2021-02-03 06:00:51 -0700930 dtb_platdata.run_steps(['all'], None, False, 'out', ['cdir'], True,
931 scan=copy_scan())
Simon Glass4e8e8462020-12-28 20:34:52 -0700932 self.assertIn("Must specify either output or output_dirs, not both",
933 str(exc.exception))
934
935 def test_output_dirs(self):
936 """Test outputting files to a directory"""
937 # Remove the directory so that files from other tests are not there
938 tools._RemoveOutputDir()
939 tools.PrepareOutputDir(None)
940
941 # This should create the .dts and .dtb in the output directory
942 dtb_file = get_dtb_file('dtoc_test_simple.dts')
943 outdir = tools.GetOutputDir()
944 fnames = glob.glob(outdir + '/*')
945 self.assertEqual(2, len(fnames))
946
Simon Glass768ff0a2021-02-03 06:00:51 -0700947 dtb_platdata.run_steps(['all'], dtb_file, False, None, [outdir], True,
948 scan=copy_scan())
Simon Glass4e8e8462020-12-28 20:34:52 -0700949 fnames = glob.glob(outdir + '/*')
950 self.assertEqual(4, len(fnames))
951
952 leafs = set(os.path.basename(fname) for fname in fnames)
953 self.assertEqual(
Simon Glassbeddd7a2020-12-28 20:35:01 -0700954 {'dt-structs-gen.h', 'source.dts', 'dt-plat.c', 'source.dtb'},
Simon Glass4e8e8462020-12-28 20:34:52 -0700955 leafs)