blob: 1a85ebcf81ab3add113b54c1c433bba6a718e43c [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
Simon Glassbd310df2023-02-23 18:18:16 -070016import pathlib
Simon Glass9d2eb922017-06-18 22:09:06 -060017import struct
18import unittest
19
Simon Glassa60cabd2020-12-28 20:34:47 -070020from dtoc import dtb_platdata
Simon Glassa997ea52020-04-17 18:09:04 -060021from dtoc import fdt
22from dtoc import fdt_util
Simon Glass768ff0a2021-02-03 06:00:51 -070023from dtoc import src_scan
Simon Glass2b160072022-02-11 13:23:20 -070024from dtoc.dtb_platdata import Ftype
25from dtoc.dtb_platdata import get_value
26from dtoc.dtb_platdata import tab_to
Simon Glass9065bc92020-12-28 20:35:06 -070027from dtoc.src_scan import conv_name_to_c
28from dtoc.src_scan import get_compat_name
Simon Glass14d64e32025-04-29 07:21:59 -060029from u_boot_pylib import terminal
Simon Glass131444f2023-02-23 18:18:04 -070030from u_boot_pylib import test_util
31from u_boot_pylib import tools
Simon Glass9d2eb922017-06-18 22:09:06 -060032
Simon Glassbd310df2023-02-23 18:18:16 -070033DTOC_DIR = pathlib.Path(__file__).parent
34TEST_DATA_DIR = DTOC_DIR / 'test/'
Simon Glass9d2eb922017-06-18 22:09:06 -060035
36
Simon Glasseb37e2d2017-11-12 21:52:17 -070037HEADER = '''/*
38 * DO NOT MODIFY
39 *
Simon Glass6b208842020-12-28 20:35:00 -070040 * Defines the structs used to hold devicetree data.
41 * This was generated by dtoc from a .dtb (device tree binary) file.
Simon Glasseb37e2d2017-11-12 21:52:17 -070042 */
43
44#include <stdbool.h>
Masahiro Yamada75f82d02018-03-05 01:20:11 +090045#include <linux/libfdt.h>'''
Simon Glasseb37e2d2017-11-12 21:52:17 -070046
Simon Glass3fa3bbb2021-02-03 06:01:14 -070047DECL_HEADER = '''/*
48 * DO NOT MODIFY
49 *
50 * Declares externs for all device/uclass instances.
51 * This was generated by dtoc from a .dtb (device tree binary) file.
52 */
53'''
54
Simon Glassbe749002021-02-03 06:01:15 -070055C_HEADER_PRE = '''/*
Simon Glasseb37e2d2017-11-12 21:52:17 -070056 * DO NOT MODIFY
57 *
Simon Glass6b208842020-12-28 20:35:00 -070058 * Declares the U_BOOT_DRIVER() records and platform data.
59 * This was generated by dtoc from a .dtb (device tree binary) file.
Simon Glasseb37e2d2017-11-12 21:52:17 -070060 */
Simon Glassbe749002021-02-03 06:01:15 -070061'''
Simon Glasseb37e2d2017-11-12 21:52:17 -070062
Simon Glassbe749002021-02-03 06:01:15 -070063C_HEADER = C_HEADER_PRE + '''
Simon Glass1d8364a2020-12-28 20:34:54 -070064/* Allow use of U_BOOT_DRVINFO() in this file */
Simon Glassbeddd7a2020-12-28 20:35:01 -070065#define DT_PLAT_C
Simon Glass4c73d7b2020-10-03 11:31:41 -060066
Simon Glasseb37e2d2017-11-12 21:52:17 -070067#include <dm.h>
68#include <dt-structs.h>
69'''
70
Simon Glassc7b4b832021-02-03 06:01:20 -070071UCLASS_HEADER_COMMON = '''/*
72 * DO NOT MODIFY
73 *
74 * Declares the uclass instances (struct uclass).
75 * This was generated by dtoc from a .dtb (device tree binary) file.
76 */
77'''
78
Simon Glass768ff0a2021-02-03 06:00:51 -070079# Scanner saved from a previous run of the tests (to speed things up)
80saved_scan = None
81
Simon Glassa60cabd2020-12-28 20:34:47 -070082# This is a test so is allowed to access private things in the module it is
83# testing
84# pylint: disable=W0212
Simon Glass3bce93d2018-07-06 10:27:37 -060085
86def get_dtb_file(dts_fname, capture_stderr=False):
Simon Glass9d2eb922017-06-18 22:09:06 -060087 """Compile a .dts file to a .dtb
88
89 Args:
Simon Glassa60cabd2020-12-28 20:34:47 -070090 dts_fname (str): Filename of .dts file in the current directory
91 capture_stderr (bool): True to capture and discard stderr output
Simon Glass9d2eb922017-06-18 22:09:06 -060092
93 Returns:
Simon Glassa60cabd2020-12-28 20:34:47 -070094 str: Filename of compiled file in output directory
Simon Glass9d2eb922017-06-18 22:09:06 -060095 """
Simon Glassbd310df2023-02-23 18:18:16 -070096 return fdt_util.EnsureCompiled(str(TEST_DATA_DIR / dts_fname),
Simon Glass3bce93d2018-07-06 10:27:37 -060097 capture_stderr=capture_stderr)
Simon Glass9d2eb922017-06-18 22:09:06 -060098
Simon Glass768ff0a2021-02-03 06:00:51 -070099
100def setup():
101 global saved_scan
102
103 # Disable warnings so that calls to get_normalized_compat_name() will not
104 # output things.
Simon Glass695077d2021-03-26 16:17:25 +1300105 saved_scan = src_scan.Scanner(None, False)
Simon Glass768ff0a2021-02-03 06:00:51 -0700106 saved_scan.scan_drivers()
107
108def copy_scan():
109 """Get a copy of saved_scan so that each test can start clean"""
110 return copy.deepcopy(saved_scan)
111
Simon Glass9d2eb922017-06-18 22:09:06 -0600112
113class TestDtoc(unittest.TestCase):
114 """Tests for dtoc"""
115 @classmethod
116 def setUpClass(cls):
Simon Glass80025522022-01-29 14:14:04 -0700117 tools.prepare_output_dir(None)
Simon Glass7f5e2262020-07-07 21:32:06 -0600118 cls.maxDiff = None
Simon Glass9d2eb922017-06-18 22:09:06 -0600119
120 @classmethod
121 def tearDownClass(cls):
Simon Glass80025522022-01-29 14:14:04 -0700122 tools.finalise_output_dir()
Simon Glass9d2eb922017-06-18 22:09:06 -0600123
Simon Glassa60cabd2020-12-28 20:34:47 -0700124 @staticmethod
125 def _write_python_string(fname, data):
Simon Glassc47c2b32018-07-06 10:27:25 -0600126 """Write a string with tabs expanded as done in this Python file
127
128 Args:
Simon Glassa60cabd2020-12-28 20:34:47 -0700129 fname (str): Filename to write to
130 data (str): Raw string to convert
Simon Glassc47c2b32018-07-06 10:27:25 -0600131 """
132 data = data.replace('\t', '\\t')
Simon Glassa60cabd2020-12-28 20:34:47 -0700133 with open(fname, 'w') as fout:
134 fout.write(data)
Simon Glassc47c2b32018-07-06 10:27:25 -0600135
Simon Glassa60cabd2020-12-28 20:34:47 -0700136 def _check_strings(self, expected, actual):
Simon Glassc47c2b32018-07-06 10:27:25 -0600137 """Check that a string matches its expected value
138
139 If the strings do not match, they are written to the /tmp directory in
140 the same Python format as is used here in the test. This allows for
141 easy comparison and update of the tests.
142
143 Args:
Simon Glassa60cabd2020-12-28 20:34:47 -0700144 expected (str): Expected string
145 actual (str): Actual string
Simon Glassc47c2b32018-07-06 10:27:25 -0600146 """
147 if expected != actual:
Simon Glassa60cabd2020-12-28 20:34:47 -0700148 self._write_python_string('/tmp/binman.expected', expected)
149 self._write_python_string('/tmp/binman.actual', actual)
Simon Glass61b88e52019-05-17 22:00:31 -0600150 print('Failures written to /tmp/binman.{expected,actual}')
Simon Glassa60cabd2020-12-28 20:34:47 -0700151 self.assertEqual(expected, actual)
Simon Glassc47c2b32018-07-06 10:27:25 -0600152
Simon Glassa60cabd2020-12-28 20:34:47 -0700153 @staticmethod
Simon Glass3809ad92021-02-03 06:01:12 -0700154 def run_test(args, dtb_file, output, instantiate=False):
Simon Glassa60cabd2020-12-28 20:34:47 -0700155 """Run a test using dtoc
Walter Lozanoa324e412020-06-25 01:10:08 -0300156
Simon Glassa60cabd2020-12-28 20:34:47 -0700157 Args:
158 args (list of str): List of arguments for dtoc
159 dtb_file (str): Filename of .dtb file
160 output (str): Filename of output file
Simon Glassbe88d2f2021-02-03 06:01:07 -0700161
162 Returns:
163 DtbPlatdata object
Simon Glassa60cabd2020-12-28 20:34:47 -0700164 """
Simon Glass80d782c42021-02-03 06:01:10 -0700165 # Make a copy of the 'scan' object, since it includes uclasses and
166 # drivers, which get updated during execution.
Simon Glass3809ad92021-02-03 06:01:12 -0700167 return dtb_platdata.run_steps(
168 args, dtb_file, False, output, [], None, instantiate,
169 warning_disabled=True, scan=copy_scan())
Walter Lozanoa324e412020-06-25 01:10:08 -0300170
Simon Glass9d2eb922017-06-18 22:09:06 -0600171 def test_name(self):
172 """Test conversion of device tree names to C identifiers"""
173 self.assertEqual('serial_at_0x12', conv_name_to_c('serial@0x12'))
174 self.assertEqual('vendor_clock_frequency',
175 conv_name_to_c('vendor,clock-frequency'))
176 self.assertEqual('rockchip_rk3399_sdhci_5_1',
177 conv_name_to_c('rockchip,rk3399-sdhci-5.1'))
178
179 def test_tab_to(self):
180 """Test operation of tab_to() function"""
181 self.assertEqual('fred ', tab_to(0, 'fred'))
182 self.assertEqual('fred\t', tab_to(1, 'fred'))
183 self.assertEqual('fred was here ', tab_to(1, 'fred was here'))
184 self.assertEqual('fred was here\t\t', tab_to(3, 'fred was here'))
185 self.assertEqual('exactly8 ', tab_to(1, 'exactly8'))
186 self.assertEqual('exactly8\t', tab_to(2, 'exactly8'))
187
188 def test_get_value(self):
189 """Test operation of get_value() function"""
190 self.assertEqual('0x45',
Simon Glassc9a032c2020-11-08 20:36:17 -0700191 get_value(fdt.Type.INT, struct.pack('>I', 0x45)))
Simon Glass9d2eb922017-06-18 22:09:06 -0600192 self.assertEqual('0x45',
Simon Glassc9a032c2020-11-08 20:36:17 -0700193 get_value(fdt.Type.BYTE, struct.pack('<I', 0x45)))
Simon Glass9d2eb922017-06-18 22:09:06 -0600194 self.assertEqual('0x0',
Simon Glassc9a032c2020-11-08 20:36:17 -0700195 get_value(fdt.Type.BYTE, struct.pack('>I', 0x45)))
196 self.assertEqual('"test"', get_value(fdt.Type.STRING, 'test'))
197 self.assertEqual('true', get_value(fdt.Type.BOOL, None))
Simon Glass9d2eb922017-06-18 22:09:06 -0600198
199 def test_get_compat_name(self):
200 """Test operation of get_compat_name() function"""
201 Prop = collections.namedtuple('Prop', ['value'])
202 Node = collections.namedtuple('Node', ['props'])
203
204 prop = Prop(['rockchip,rk3399-sdhci-5.1', 'arasan,sdhci-5.1'])
205 node = Node({'compatible': prop})
Walter Lozano5fe734c2020-07-23 00:22:03 -0300206 self.assertEqual((['rockchip_rk3399_sdhci_5_1', 'arasan_sdhci_5_1']),
Simon Glass9d2eb922017-06-18 22:09:06 -0600207 get_compat_name(node))
208
209 prop = Prop(['rockchip,rk3399-sdhci-5.1'])
210 node = Node({'compatible': prop})
Walter Lozano5fe734c2020-07-23 00:22:03 -0300211 self.assertEqual((['rockchip_rk3399_sdhci_5_1']),
Simon Glass9d2eb922017-06-18 22:09:06 -0600212 get_compat_name(node))
213
214 prop = Prop(['rockchip,rk3399-sdhci-5.1', 'arasan,sdhci-5.1', 'third'])
215 node = Node({'compatible': prop})
Walter Lozano5fe734c2020-07-23 00:22:03 -0300216 self.assertEqual((['rockchip_rk3399_sdhci_5_1',
Simon Glassa60cabd2020-12-28 20:34:47 -0700217 'arasan_sdhci_5_1', 'third']),
Simon Glass9d2eb922017-06-18 22:09:06 -0600218 get_compat_name(node))
219
220 def test_empty_file(self):
221 """Test output from a device tree file with no nodes"""
222 dtb_file = get_dtb_file('dtoc_test_empty.dts')
Simon Glass80025522022-01-29 14:14:04 -0700223 output = tools.get_output_filename('output')
Simon Glass768ff0a2021-02-03 06:00:51 -0700224
225 # Run this one without saved_scan to complete test coverage
Simon Glass3809ad92021-02-03 06:01:12 -0700226 dtb_platdata.run_steps(['struct'], dtb_file, False, output, [], None,
227 False)
Simon Glass9d2eb922017-06-18 22:09:06 -0600228 with open(output) as infile:
229 lines = infile.read().splitlines()
Simon Glasseb37e2d2017-11-12 21:52:17 -0700230 self.assertEqual(HEADER.splitlines(), lines)
Simon Glass9d2eb922017-06-18 22:09:06 -0600231
Walter Lozanoa324e412020-06-25 01:10:08 -0300232 self.run_test(['platdata'], dtb_file, output)
Simon Glass9d2eb922017-06-18 22:09:06 -0600233 with open(output) as infile:
234 lines = infile.read().splitlines()
Simon Glass2500de22020-12-28 20:35:05 -0700235 self.assertEqual(C_HEADER.splitlines() + [''], lines)
Simon Glass9d2eb922017-06-18 22:09:06 -0600236
Simon Glass3fa3bbb2021-02-03 06:01:14 -0700237 decl_text = DECL_HEADER + '''
238#include <dm/device-internal.h>
239#include <dm/uclass-internal.h>
240
241/* driver declarations - these allow DM_DRIVER_GET() to be used */
Simon Glassed5e6ae2021-03-15 17:25:11 +1300242extern U_BOOT_DRIVER(sandbox_i2c);
243extern U_BOOT_DRIVER(sandbox_pmic);
244extern U_BOOT_DRIVER(sandbox_spl_test);
245extern U_BOOT_DRIVER(sandbox_spl_test);
246extern U_BOOT_DRIVER(sandbox_spl_test);
Simon Glass3fa3bbb2021-02-03 06:01:14 -0700247
248/* uclass driver declarations - needed for DM_UCLASS_DRIVER_REF() */
Simon Glassed5e6ae2021-03-15 17:25:11 +1300249extern UCLASS_DRIVER(i2c);
250extern UCLASS_DRIVER(misc);
251extern UCLASS_DRIVER(pmic);
Simon Glass3fa3bbb2021-02-03 06:01:14 -0700252'''
253 decl_text_inst = DECL_HEADER + '''
254#include <dm/device-internal.h>
255#include <dm/uclass-internal.h>
256
257/* driver declarations - these allow DM_DRIVER_GET() to be used */
Simon Glassed5e6ae2021-03-15 17:25:11 +1300258extern U_BOOT_DRIVER(sandbox_i2c);
259extern U_BOOT_DRIVER(root_driver);
260extern U_BOOT_DRIVER(denx_u_boot_test_bus);
261extern U_BOOT_DRIVER(sandbox_spl_test);
262extern U_BOOT_DRIVER(sandbox_spl_test);
263extern U_BOOT_DRIVER(denx_u_boot_fdt_test);
264extern U_BOOT_DRIVER(denx_u_boot_fdt_test);
Simon Glass3fa3bbb2021-02-03 06:01:14 -0700265
266/* device declarations - these allow DM_DEVICE_REF() to be used */
Simon Glassed5e6ae2021-03-15 17:25:11 +1300267extern DM_DEVICE_INST(i2c);
268extern DM_DEVICE_INST(root);
269extern DM_DEVICE_INST(some_bus);
270extern DM_DEVICE_INST(spl_test);
271extern DM_DEVICE_INST(spl_test3);
272extern DM_DEVICE_INST(test);
273extern DM_DEVICE_INST(test0);
Simon Glass3fa3bbb2021-02-03 06:01:14 -0700274
275/* uclass driver declarations - needed for DM_UCLASS_DRIVER_REF() */
Simon Glassed5e6ae2021-03-15 17:25:11 +1300276extern UCLASS_DRIVER(i2c);
277extern UCLASS_DRIVER(misc);
278extern UCLASS_DRIVER(root);
279extern UCLASS_DRIVER(testbus);
280extern UCLASS_DRIVER(testfdt);
Simon Glass3fa3bbb2021-02-03 06:01:14 -0700281
282/* uclass declarations - needed for DM_UCLASS_REF() */
Simon Glassed5e6ae2021-03-15 17:25:11 +1300283extern DM_UCLASS_INST(i2c);
284extern DM_UCLASS_INST(misc);
285extern DM_UCLASS_INST(root);
286extern DM_UCLASS_INST(testbus);
287extern DM_UCLASS_INST(testfdt);
Simon Glass3fa3bbb2021-02-03 06:01:14 -0700288'''
Simon Glass07ee48e2020-12-28 20:34:49 -0700289 struct_text = HEADER + '''
Simon Glass32a8d222021-02-03 06:00:57 -0700290struct dtd_sandbox_i2c {
Simon Glass90e5f0a2017-08-29 14:15:51 -0600291};
Simon Glass32a8d222021-02-03 06:00:57 -0700292struct dtd_sandbox_pmic {
Simon Glass90e5f0a2017-08-29 14:15:51 -0600293\tbool\t\tlow_power;
Simon Glass94ba59e2021-03-26 16:17:26 +1300294\tfdt32_t\t\treg[1];
Simon Glass90e5f0a2017-08-29 14:15:51 -0600295};
Simon Glass9d2eb922017-06-18 22:09:06 -0600296struct dtd_sandbox_spl_test {
Simon Glass7f5e2262020-07-07 21:32:06 -0600297\tconst char * acpi_name;
Simon Glass9d2eb922017-06-18 22:09:06 -0600298\tbool\t\tboolval;
299\tunsigned char\tbytearray[3];
300\tunsigned char\tbyteval;
Simon Glass3b55e3f2021-11-23 11:03:39 -0700301\tfdt32_t\t\tint64val[2];
Simon Glassa7d66982021-07-28 19:23:10 -0600302\tfdt32_t\t\tintarray[3];
Simon Glass9d2eb922017-06-18 22:09:06 -0600303\tfdt32_t\t\tintval;
304\tunsigned char\tlongbytearray[9];
Simon Glass43118322021-07-28 19:23:11 -0600305\tfdt32_t\t\tmaybe_empty_int[1];
Simon Glass9c526332018-07-06 10:27:28 -0600306\tunsigned char\tnotstring[5];
Simon Glass9d2eb922017-06-18 22:09:06 -0600307\tconst char *\tstringarray[3];
308\tconst char *\tstringval;
309};
Simon Glass07ee48e2020-12-28 20:34:49 -0700310'''
Simon Glass07ee48e2020-12-28 20:34:49 -0700311 platdata_text = C_HEADER + '''
Simon Glassbe749002021-02-03 06:01:15 -0700312/*
Simon Glasse2119082021-02-03 06:01:19 -0700313 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
314 *
315 * idx driver_info driver
316 * --- -------------------- --------------------
317 * 0: i2c_at_0 sandbox_i2c
318 * 1: pmic_at_9 sandbox_pmic
319 * 2: spl_test sandbox_spl_test
320 * 3: spl_test2 sandbox_spl_test
321 * 4: spl_test3 sandbox_spl_test
322 * --- -------------------- --------------------
323 */
324
325/*
Simon Glassbe749002021-02-03 06:01:15 -0700326 * Node /i2c@0 index 0
327 * driver sandbox_i2c parent None
328 */
Simon Glass32a8d222021-02-03 06:00:57 -0700329static struct dtd_sandbox_i2c dtv_i2c_at_0 = {
Simon Glass192f8132020-10-03 11:31:25 -0600330};
Simon Glass1d8364a2020-12-28 20:34:54 -0700331U_BOOT_DRVINFO(i2c_at_0) = {
Simon Glass32a8d222021-02-03 06:00:57 -0700332\t.name\t\t= "sandbox_i2c",
Simon Glasse2119082021-02-03 06:01:19 -0700333\t.plat\t\t= &dtv_i2c_at_0,
Simon Glass39edb952020-12-03 16:55:19 -0700334\t.plat_size\t= sizeof(dtv_i2c_at_0),
Simon Glass36b15e22020-10-03 11:31:35 -0600335\t.parent_idx\t= -1,
Simon Glass192f8132020-10-03 11:31:25 -0600336};
337
Simon Glassbe749002021-02-03 06:01:15 -0700338/*
339 * Node /i2c@0/pmic@9 index 1
340 * driver sandbox_pmic parent sandbox_i2c
341 */
Simon Glass32a8d222021-02-03 06:00:57 -0700342static struct dtd_sandbox_pmic dtv_pmic_at_9 = {
Simon Glass192f8132020-10-03 11:31:25 -0600343\t.low_power\t\t= true,
Simon Glass94ba59e2021-03-26 16:17:26 +1300344\t.reg\t\t\t= {0x9},
Simon Glass192f8132020-10-03 11:31:25 -0600345};
Simon Glass1d8364a2020-12-28 20:34:54 -0700346U_BOOT_DRVINFO(pmic_at_9) = {
Simon Glass32a8d222021-02-03 06:00:57 -0700347\t.name\t\t= "sandbox_pmic",
Simon Glasse2119082021-02-03 06:01:19 -0700348\t.plat\t\t= &dtv_pmic_at_9,
Simon Glass39edb952020-12-03 16:55:19 -0700349\t.plat_size\t= sizeof(dtv_pmic_at_9),
Simon Glass36b15e22020-10-03 11:31:35 -0600350\t.parent_idx\t= 0,
Simon Glass192f8132020-10-03 11:31:25 -0600351};
352
Simon Glassbe749002021-02-03 06:01:15 -0700353/*
354 * Node /spl-test index 2
355 * driver sandbox_spl_test parent None
356 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300357static struct dtd_sandbox_spl_test dtv_spl_test = {
Simon Glassc82de562019-05-17 22:00:32 -0600358\t.boolval\t\t= true,
Simon Glass9d2eb922017-06-18 22:09:06 -0600359\t.bytearray\t\t= {0x6, 0x0, 0x0},
360\t.byteval\t\t= 0x5,
Simon Glass3b55e3f2021-11-23 11:03:39 -0700361\t.int64val\t\t= {0x12345678, 0x9abcdef0},
Simon Glassa7d66982021-07-28 19:23:10 -0600362\t.intarray\t\t= {0x2, 0x3, 0x4},
Simon Glass9d2eb922017-06-18 22:09:06 -0600363\t.intval\t\t\t= 0x1,
Simon Glass131e0b02017-08-29 14:15:49 -0600364\t.longbytearray\t\t= {0x9, 0xa, 0xb, 0xc, 0xd, 0xe, 0xf, 0x10,
365\t\t0x11},
Simon Glass43118322021-07-28 19:23:11 -0600366\t.maybe_empty_int\t= {0x0},
Simon Glassc82de562019-05-17 22:00:32 -0600367\t.notstring\t\t= {0x20, 0x21, 0x22, 0x10, 0x0},
Simon Glass9d2eb922017-06-18 22:09:06 -0600368\t.stringarray\t\t= {"multi-word", "message", ""},
Simon Glassc82de562019-05-17 22:00:32 -0600369\t.stringval\t\t= "message",
Simon Glass9d2eb922017-06-18 22:09:06 -0600370};
Simon Glass1d8364a2020-12-28 20:34:54 -0700371U_BOOT_DRVINFO(spl_test) = {
Simon Glass9d2eb922017-06-18 22:09:06 -0600372\t.name\t\t= "sandbox_spl_test",
Simon Glasse2119082021-02-03 06:01:19 -0700373\t.plat\t\t= &dtv_spl_test,
Simon Glass39edb952020-12-03 16:55:19 -0700374\t.plat_size\t= sizeof(dtv_spl_test),
Simon Glass36b15e22020-10-03 11:31:35 -0600375\t.parent_idx\t= -1,
Simon Glass9d2eb922017-06-18 22:09:06 -0600376};
377
Simon Glassbe749002021-02-03 06:01:15 -0700378/*
379 * Node /spl-test2 index 3
380 * driver sandbox_spl_test parent None
381 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300382static struct dtd_sandbox_spl_test dtv_spl_test2 = {
Simon Glass7f5e2262020-07-07 21:32:06 -0600383\t.acpi_name\t\t= "\\\\_SB.GPO0",
Simon Glass9d2eb922017-06-18 22:09:06 -0600384\t.bytearray\t\t= {0x1, 0x23, 0x34},
385\t.byteval\t\t= 0x8,
Simon Glassa7d66982021-07-28 19:23:10 -0600386\t.intarray\t\t= {0x5, 0x0, 0x0},
Simon Glass9d2eb922017-06-18 22:09:06 -0600387\t.intval\t\t\t= 0x3,
Simon Glass8034e4d2020-10-03 11:31:27 -0600388\t.longbytearray\t\t= {0x9, 0xa, 0xb, 0xc, 0x0, 0x0, 0x0, 0x0,
Simon Glass131e0b02017-08-29 14:15:49 -0600389\t\t0x0},
Simon Glass9d2eb922017-06-18 22:09:06 -0600390\t.stringarray\t\t= {"another", "multi-word", "message"},
Simon Glassc82de562019-05-17 22:00:32 -0600391\t.stringval\t\t= "message2",
Simon Glass9d2eb922017-06-18 22:09:06 -0600392};
Simon Glass1d8364a2020-12-28 20:34:54 -0700393U_BOOT_DRVINFO(spl_test2) = {
Simon Glass9d2eb922017-06-18 22:09:06 -0600394\t.name\t\t= "sandbox_spl_test",
Simon Glasse2119082021-02-03 06:01:19 -0700395\t.plat\t\t= &dtv_spl_test2,
Simon Glass39edb952020-12-03 16:55:19 -0700396\t.plat_size\t= sizeof(dtv_spl_test2),
Simon Glass36b15e22020-10-03 11:31:35 -0600397\t.parent_idx\t= -1,
Simon Glass9d2eb922017-06-18 22:09:06 -0600398};
399
Simon Glassbe749002021-02-03 06:01:15 -0700400/*
401 * Node /spl-test3 index 4
402 * driver sandbox_spl_test parent None
403 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300404static struct dtd_sandbox_spl_test dtv_spl_test3 = {
Simon Glass8034e4d2020-10-03 11:31:27 -0600405\t.longbytearray\t\t= {0x9, 0xa, 0xb, 0xc, 0xd, 0xe, 0xf, 0x10,
406\t\t0x0},
Simon Glass43118322021-07-28 19:23:11 -0600407\t.maybe_empty_int\t= {0x1},
Simon Glass9d2eb922017-06-18 22:09:06 -0600408\t.stringarray\t\t= {"one", "", ""},
409};
Simon Glass1d8364a2020-12-28 20:34:54 -0700410U_BOOT_DRVINFO(spl_test3) = {
Simon Glass9d2eb922017-06-18 22:09:06 -0600411\t.name\t\t= "sandbox_spl_test",
Simon Glasse2119082021-02-03 06:01:19 -0700412\t.plat\t\t= &dtv_spl_test3,
Simon Glass39edb952020-12-03 16:55:19 -0700413\t.plat_size\t= sizeof(dtv_spl_test3),
Simon Glass36b15e22020-10-03 11:31:35 -0600414\t.parent_idx\t= -1,
Simon Glass9d2eb922017-06-18 22:09:06 -0600415};
416
Simon Glass2500de22020-12-28 20:35:05 -0700417'''
Simon Glassc7b4b832021-02-03 06:01:20 -0700418 uclass_text_inst = '''
419
Simon Glassc7b4b832021-02-03 06:01:20 -0700420#include <dm.h>
421#include <dt-structs.h>
422
423/*
Simon Glassfea2f252021-02-03 06:01:21 -0700424 * uclass declarations, ordered by 'struct uclass' linker_list idx:
425 * 0: i2c
426 * 1: misc
427 * 2: root
428 * 3: testbus
429 * 4: testfdt
Simon Glassc7b4b832021-02-03 06:01:20 -0700430 *
Simon Glassfea2f252021-02-03 06:01:21 -0700431 * Sequence numbers allocated in each uclass:
Simon Glassc7b4b832021-02-03 06:01:20 -0700432 * i2c: UCLASS_I2C
433 * 4: /i2c
434 * misc: UCLASS_MISC
435 * 0: /spl-test
436 * 1: /spl-test3
437 * root: UCLASS_ROOT
438 * 0: /
439 * testbus: UCLASS_TEST_BUS
440 * 2: /some-bus
441 * testfdt: UCLASS_TEST_FDT
442 * 1: /some-bus/test
443 * 2: /some-bus/test0
444 */
445
446struct list_head uclass_head = {
447 .prev = &DM_UCLASS_REF(testfdt)->sibling_node,
448 .next = &DM_UCLASS_REF(i2c)->sibling_node,
449};
450
451DM_UCLASS_INST(i2c) = {
452 .uc_drv = DM_UCLASS_DRIVER_REF(i2c),
453 .sibling_node = {
454 .prev = &uclass_head,
455 .next = &DM_UCLASS_REF(misc)->sibling_node,
456 },
457 .dev_head = {
458 .prev = &DM_DEVICE_REF(i2c)->uclass_node,
459 .next = &DM_DEVICE_REF(i2c)->uclass_node,
460 },
461};
462
463DM_UCLASS_INST(misc) = {
464 .uc_drv = DM_UCLASS_DRIVER_REF(misc),
465 .sibling_node = {
466 .prev = &DM_UCLASS_REF(i2c)->sibling_node,
467 .next = &DM_UCLASS_REF(root)->sibling_node,
468 },
469 .dev_head = {
470 .prev = &DM_DEVICE_REF(spl_test3)->uclass_node,
471 .next = &DM_DEVICE_REF(spl_test)->uclass_node,
472 },
473};
474
475DM_UCLASS_INST(root) = {
476 .uc_drv = DM_UCLASS_DRIVER_REF(root),
477 .sibling_node = {
478 .prev = &DM_UCLASS_REF(misc)->sibling_node,
479 .next = &DM_UCLASS_REF(testbus)->sibling_node,
480 },
481 .dev_head = {
482 .prev = &DM_DEVICE_REF(root)->uclass_node,
483 .next = &DM_DEVICE_REF(root)->uclass_node,
484 },
485};
486
487DM_UCLASS_INST(testbus) = {
488 .uc_drv = DM_UCLASS_DRIVER_REF(testbus),
489 .sibling_node = {
490 .prev = &DM_UCLASS_REF(root)->sibling_node,
491 .next = &DM_UCLASS_REF(testfdt)->sibling_node,
492 },
493 .dev_head = {
494 .prev = &DM_DEVICE_REF(some_bus)->uclass_node,
495 .next = &DM_DEVICE_REF(some_bus)->uclass_node,
496 },
497};
498
499#include <dm/test.h>
500u8 _testfdt_priv_[sizeof(struct dm_test_uc_priv)]
501 __attribute__ ((section (".priv_data")));
502DM_UCLASS_INST(testfdt) = {
503 .priv_ = _testfdt_priv_,
504 .uc_drv = DM_UCLASS_DRIVER_REF(testfdt),
505 .sibling_node = {
506 .prev = &DM_UCLASS_REF(testbus)->sibling_node,
507 .next = &uclass_head,
508 },
509 .dev_head = {
510 .prev = &DM_DEVICE_REF(test0)->uclass_node,
511 .next = &DM_DEVICE_REF(test)->uclass_node,
512 },
513};
514
Simon Glassfea2f252021-02-03 06:01:21 -0700515'''
516 device_text_inst = '''/*
517 * DO NOT MODIFY
518 *
519 * Declares the DM_DEVICE_INST() records.
520 * This was generated by dtoc from a .dtb (device tree binary) file.
521 */
522
Simon Glassfea2f252021-02-03 06:01:21 -0700523#include <dm.h>
524#include <dt-structs.h>
525
526/*
527 * udevice declarations, ordered by 'struct udevice' linker_list position:
528 *
529 * idx udevice driver
530 * --- -------------------- --------------------
531 * 0: i2c sandbox_i2c
532 * 1: root root_driver
533 * 2: some_bus denx_u_boot_test_bus
534 * 3: spl_test sandbox_spl_test
535 * 4: spl_test3 sandbox_spl_test
536 * 5: test denx_u_boot_fdt_test
537 * 6: test0 denx_u_boot_fdt_test
538 * --- -------------------- --------------------
539 */
540
541/*
542 * Node /i2c index 0
543 * driver sandbox_i2c parent root_driver
544*/
545static struct dtd_sandbox_i2c dtv_i2c = {
546\t.intval\t\t\t= 0x3,
547};
548
549#include <asm/i2c.h>
550u8 _sandbox_i2c_priv_i2c[sizeof(struct sandbox_i2c_priv)]
551\t__attribute__ ((section (".priv_data")));
552#include <i2c.h>
553u8 _sandbox_i2c_uc_priv_i2c[sizeof(struct dm_i2c_bus)]
554\t__attribute__ ((section (".priv_data")));
555
556DM_DEVICE_INST(i2c) = {
557\t.driver\t\t= DM_DRIVER_REF(sandbox_i2c),
558\t.name\t\t= "sandbox_i2c",
559\t.plat_\t\t= &dtv_i2c,
560\t.priv_\t\t= _sandbox_i2c_priv_i2c,
561\t.uclass\t\t= DM_UCLASS_REF(i2c),
562\t.uclass_priv_ = _sandbox_i2c_uc_priv_i2c,
563\t.uclass_node\t= {
564\t\t.prev = &DM_UCLASS_REF(i2c)->dev_head,
565\t\t.next = &DM_UCLASS_REF(i2c)->dev_head,
566\t},
567\t.child_head\t= {
568\t\t.prev = &DM_DEVICE_REF(i2c)->child_head,
569\t\t.next = &DM_DEVICE_REF(i2c)->child_head,
570\t},
571\t.sibling_node\t= {
572\t\t.prev = &DM_DEVICE_REF(root)->child_head,
573\t\t.next = &DM_DEVICE_REF(some_bus)->sibling_node,
574\t},
575\t.seq_ = 4,
576};
577
578/*
579 * Node / index 1
580 * driver root_driver parent None
581*/
582static struct dtd_root_driver dtv_root = {
583};
584
585DM_DEVICE_INST(root) = {
586\t.driver\t\t= DM_DRIVER_REF(root_driver),
587\t.name\t\t= "root_driver",
588\t.plat_\t\t= &dtv_root,
589\t.uclass\t\t= DM_UCLASS_REF(root),
590\t.uclass_node\t= {
591\t\t.prev = &DM_UCLASS_REF(root)->dev_head,
592\t\t.next = &DM_UCLASS_REF(root)->dev_head,
593\t},
594\t.child_head\t= {
595\t\t.prev = &DM_DEVICE_REF(spl_test3)->sibling_node,
596\t\t.next = &DM_DEVICE_REF(i2c)->sibling_node,
597\t},
598\t.seq_ = 0,
599};
600
601/*
602 * Node /some-bus index 2
603 * driver denx_u_boot_test_bus parent root_driver
604*/
605
606#include <dm/test.h>
607struct dm_test_pdata __attribute__ ((section (".priv_data")))
608\t_denx_u_boot_test_bus_plat_some_bus = {
609\t.dtplat = {
610\t\t.ping_add\t\t= 0x4,
611\t\t.ping_expect\t\t= 0x4,
612\t\t.reg\t\t\t= {0x3, 0x1},
613\t},
614};
615#include <dm/test.h>
616u8 _denx_u_boot_test_bus_priv_some_bus[sizeof(struct dm_test_priv)]
617\t__attribute__ ((section (".priv_data")));
618#include <dm/test.h>
Simon Glasse3304472022-05-08 04:39:23 -0600619u8 _denx_u_boot_test_bus_ucplat_some_bus[sizeof(struct dm_test_uclass_plat)]
Simon Glassfea2f252021-02-03 06:01:21 -0700620\t__attribute__ ((section (".priv_data")));
Simon Glassd1f12cf2022-05-08 04:39:24 -0600621#include <dm/test.h>
622u8 _denx_u_boot_test_bus_uc_priv_some_bus[sizeof(struct dm_test_uclass_priv)]
623 __attribute__ ((section (".priv_data")));
Simon Glassfea2f252021-02-03 06:01:21 -0700624#include <test.h>
625
626DM_DEVICE_INST(some_bus) = {
627\t.driver\t\t= DM_DRIVER_REF(denx_u_boot_test_bus),
628\t.name\t\t= "denx_u_boot_test_bus",
629\t.plat_\t\t= &_denx_u_boot_test_bus_plat_some_bus,
630\t.uclass_plat_\t= _denx_u_boot_test_bus_ucplat_some_bus,
631\t.driver_data\t= DM_TEST_TYPE_FIRST,
632\t.priv_\t\t= _denx_u_boot_test_bus_priv_some_bus,
633\t.uclass\t\t= DM_UCLASS_REF(testbus),
Simon Glassd1f12cf2022-05-08 04:39:24 -0600634\t.uclass_priv_ = _denx_u_boot_test_bus_uc_priv_some_bus,
Simon Glassfea2f252021-02-03 06:01:21 -0700635\t.uclass_node\t= {
636\t\t.prev = &DM_UCLASS_REF(testbus)->dev_head,
637\t\t.next = &DM_UCLASS_REF(testbus)->dev_head,
638\t},
639\t.child_head\t= {
640\t\t.prev = &DM_DEVICE_REF(test0)->sibling_node,
641\t\t.next = &DM_DEVICE_REF(test)->sibling_node,
642\t},
643\t.sibling_node\t= {
644\t\t.prev = &DM_DEVICE_REF(i2c)->sibling_node,
645\t\t.next = &DM_DEVICE_REF(spl_test)->sibling_node,
646\t},
647\t.seq_ = 2,
648};
649
650/*
651 * Node /spl-test index 3
652 * driver sandbox_spl_test parent root_driver
653*/
654static struct dtd_sandbox_spl_test dtv_spl_test = {
655\t.boolval\t\t= true,
656\t.intval\t\t\t= 0x1,
657};
658
659DM_DEVICE_INST(spl_test) = {
660\t.driver\t\t= DM_DRIVER_REF(sandbox_spl_test),
661\t.name\t\t= "sandbox_spl_test",
662\t.plat_\t\t= &dtv_spl_test,
663\t.uclass\t\t= DM_UCLASS_REF(misc),
664\t.uclass_node\t= {
665\t\t.prev = &DM_UCLASS_REF(misc)->dev_head,
666\t\t.next = &DM_DEVICE_REF(spl_test3)->uclass_node,
667\t},
668\t.child_head\t= {
669\t\t.prev = &DM_DEVICE_REF(spl_test)->child_head,
670\t\t.next = &DM_DEVICE_REF(spl_test)->child_head,
671\t},
672\t.sibling_node\t= {
673\t\t.prev = &DM_DEVICE_REF(some_bus)->sibling_node,
674\t\t.next = &DM_DEVICE_REF(spl_test3)->sibling_node,
675\t},
676\t.seq_ = 0,
677};
678
679/*
680 * Node /spl-test3 index 4
681 * driver sandbox_spl_test parent root_driver
682*/
683static struct dtd_sandbox_spl_test dtv_spl_test3 = {
684\t.longbytearray\t\t= {0x90a0b0c, 0xd0e0f10},
685\t.stringarray\t\t= "one",
686};
687
688DM_DEVICE_INST(spl_test3) = {
689\t.driver\t\t= DM_DRIVER_REF(sandbox_spl_test),
690\t.name\t\t= "sandbox_spl_test",
691\t.plat_\t\t= &dtv_spl_test3,
692\t.uclass\t\t= DM_UCLASS_REF(misc),
693\t.uclass_node\t= {
694\t\t.prev = &DM_DEVICE_REF(spl_test)->uclass_node,
695\t\t.next = &DM_UCLASS_REF(misc)->dev_head,
696\t},
697\t.child_head\t= {
698\t\t.prev = &DM_DEVICE_REF(spl_test3)->child_head,
699\t\t.next = &DM_DEVICE_REF(spl_test3)->child_head,
700\t},
701\t.sibling_node\t= {
702\t\t.prev = &DM_DEVICE_REF(spl_test)->sibling_node,
703\t\t.next = &DM_DEVICE_REF(root)->child_head,
704\t},
705\t.seq_ = 1,
706};
707
708/*
709 * Node /some-bus/test index 5
710 * driver denx_u_boot_fdt_test parent denx_u_boot_test_bus
711*/
712
713#include <dm/test.h>
714struct dm_test_pdata __attribute__ ((section (".priv_data")))
715\t_denx_u_boot_fdt_test_plat_test = {
716\t.dtplat = {
717\t\t.ping_add\t\t= 0x5,
718\t\t.ping_expect\t\t= 0x5,
Simon Glass94ba59e2021-03-26 16:17:26 +1300719\t\t.reg\t\t\t= {0x5},
Simon Glassfea2f252021-02-03 06:01:21 -0700720\t},
721};
722#include <dm/test.h>
723u8 _denx_u_boot_fdt_test_priv_test[sizeof(struct dm_test_priv)]
724\t__attribute__ ((section (".priv_data")));
725#include <dm/test.h>
726u8 _denx_u_boot_fdt_test_parent_plat_test[sizeof(struct dm_test_parent_plat)]
727\t__attribute__ ((section (".priv_data")));
728#include <dm/test.h>
729u8 _denx_u_boot_fdt_test_parent_priv_test[sizeof(struct dm_test_parent_data)]
730\t__attribute__ ((section (".priv_data")));
731
732DM_DEVICE_INST(test) = {
733\t.driver\t\t= DM_DRIVER_REF(denx_u_boot_fdt_test),
734\t.name\t\t= "denx_u_boot_fdt_test",
735\t.plat_\t\t= &_denx_u_boot_fdt_test_plat_test,
736\t.parent_plat_\t= _denx_u_boot_fdt_test_parent_plat_test,
737\t.driver_data\t= DM_TEST_TYPE_FIRST,
738\t.parent\t\t= DM_DEVICE_REF(some_bus),
739\t.priv_\t\t= _denx_u_boot_fdt_test_priv_test,
740\t.uclass\t\t= DM_UCLASS_REF(testfdt),
741\t.parent_priv_\t= _denx_u_boot_fdt_test_parent_priv_test,
742\t.uclass_node\t= {
743\t\t.prev = &DM_UCLASS_REF(testfdt)->dev_head,
744\t\t.next = &DM_DEVICE_REF(test0)->uclass_node,
745\t},
746\t.child_head\t= {
747\t\t.prev = &DM_DEVICE_REF(test)->child_head,
748\t\t.next = &DM_DEVICE_REF(test)->child_head,
749\t},
750\t.sibling_node\t= {
751\t\t.prev = &DM_DEVICE_REF(some_bus)->child_head,
752\t\t.next = &DM_DEVICE_REF(test0)->sibling_node,
753\t},
754\t.seq_ = 1,
755};
756
757/*
758 * Node /some-bus/test0 index 6
759 * driver denx_u_boot_fdt_test parent denx_u_boot_test_bus
760*/
761
762#include <dm/test.h>
763struct dm_test_pdata __attribute__ ((section (".priv_data")))
764\t_denx_u_boot_fdt_test_plat_test0 = {
765\t.dtplat = {
766\t},
767};
768#include <dm/test.h>
769u8 _denx_u_boot_fdt_test_priv_test0[sizeof(struct dm_test_priv)]
770\t__attribute__ ((section (".priv_data")));
771#include <dm/test.h>
772u8 _denx_u_boot_fdt_test_parent_plat_test0[sizeof(struct dm_test_parent_plat)]
773\t__attribute__ ((section (".priv_data")));
774#include <dm/test.h>
775u8 _denx_u_boot_fdt_test_parent_priv_test0[sizeof(struct dm_test_parent_data)]
776\t__attribute__ ((section (".priv_data")));
777
778DM_DEVICE_INST(test0) = {
779\t.driver\t\t= DM_DRIVER_REF(denx_u_boot_fdt_test),
780\t.name\t\t= "denx_u_boot_fdt_test",
781\t.plat_\t\t= &_denx_u_boot_fdt_test_plat_test0,
782\t.parent_plat_\t= _denx_u_boot_fdt_test_parent_plat_test0,
783\t.driver_data\t= DM_TEST_TYPE_SECOND,
784\t.parent\t\t= DM_DEVICE_REF(some_bus),
785\t.priv_\t\t= _denx_u_boot_fdt_test_priv_test0,
786\t.uclass\t\t= DM_UCLASS_REF(testfdt),
787\t.parent_priv_\t= _denx_u_boot_fdt_test_parent_priv_test0,
788\t.uclass_node\t= {
789\t\t.prev = &DM_DEVICE_REF(test)->uclass_node,
790\t\t.next = &DM_UCLASS_REF(testfdt)->dev_head,
791\t},
792\t.child_head\t= {
793\t\t.prev = &DM_DEVICE_REF(test0)->child_head,
794\t\t.next = &DM_DEVICE_REF(test0)->child_head,
795\t},
796\t.sibling_node\t= {
797\t\t.prev = &DM_DEVICE_REF(test)->sibling_node,
798\t\t.next = &DM_DEVICE_REF(some_bus)->child_head,
799\t},
800\t.seq_ = 2,
801};
802
803'''
Simon Glass07ee48e2020-12-28 20:34:49 -0700804
805 def test_simple(self):
806 """Test output from some simple nodes with various types of data"""
807 dtb_file = get_dtb_file('dtoc_test_simple.dts')
Simon Glass80025522022-01-29 14:14:04 -0700808 output = tools.get_output_filename('output')
Simon Glass07ee48e2020-12-28 20:34:49 -0700809 self.run_test(['struct'], dtb_file, output)
810 with open(output) as infile:
811 data = infile.read()
812
813 self._check_strings(self.struct_text, data)
814
815 self.run_test(['platdata'], dtb_file, output)
816 with open(output) as infile:
817 data = infile.read()
818
819 self._check_strings(self.platdata_text, data)
Simon Glass9d2eb922017-06-18 22:09:06 -0600820
Simon Glass3fa3bbb2021-02-03 06:01:14 -0700821 self.run_test(['decl'], dtb_file, output)
822 with open(output) as infile:
823 data = infile.read()
824
825 self._check_strings(self.decl_text, data)
826
Simon Glass4e8e8462020-12-28 20:34:52 -0700827 # Try the 'all' command
828 self.run_test(['all'], dtb_file, output)
Simon Glass80025522022-01-29 14:14:04 -0700829 data = tools.read_file(output, binary=False)
Simon Glassfea2f252021-02-03 06:01:21 -0700830 self._check_strings(
Simon Glass705b84b2021-04-27 08:19:48 +1200831 self.decl_text + self.platdata_text + self.struct_text, data)
Simon Glass4e8e8462020-12-28 20:34:52 -0700832
Walter Lozanoe675d962020-07-03 08:07:17 -0300833 def test_driver_alias(self):
834 """Test output from a device tree file with a driver alias"""
835 dtb_file = get_dtb_file('dtoc_test_driver_alias.dts')
Simon Glass80025522022-01-29 14:14:04 -0700836 output = tools.get_output_filename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300837 self.run_test(['struct'], dtb_file, output)
Walter Lozanoe675d962020-07-03 08:07:17 -0300838 with open(output) as infile:
839 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700840 self._check_strings(HEADER + '''
Walter Lozanoe675d962020-07-03 08:07:17 -0300841struct dtd_sandbox_gpio {
842\tconst char *\tgpio_bank_name;
843\tbool\t\tgpio_controller;
844\tfdt32_t\t\tsandbox_gpio_count;
845};
Walter Lozanoe675d962020-07-03 08:07:17 -0300846''', data)
847
Walter Lozanoa324e412020-06-25 01:10:08 -0300848 self.run_test(['platdata'], dtb_file, output)
Walter Lozanoe675d962020-07-03 08:07:17 -0300849 with open(output) as infile:
850 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700851 self._check_strings(C_HEADER + '''
Simon Glassbe749002021-02-03 06:01:15 -0700852/*
Simon Glasse2119082021-02-03 06:01:19 -0700853 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
854 *
855 * idx driver_info driver
856 * --- -------------------- --------------------
857 * 0: gpios_at_0 sandbox_gpio
858 * --- -------------------- --------------------
859 */
860
861/*
Simon Glassbe749002021-02-03 06:01:15 -0700862 * Node /gpios@0 index 0
863 * driver sandbox_gpio parent None
864 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300865static struct dtd_sandbox_gpio dtv_gpios_at_0 = {
Walter Lozanoe675d962020-07-03 08:07:17 -0300866\t.gpio_bank_name\t\t= "a",
867\t.gpio_controller\t= true,
868\t.sandbox_gpio_count\t= 0x14,
869};
Simon Glass1d8364a2020-12-28 20:34:54 -0700870U_BOOT_DRVINFO(gpios_at_0) = {
Walter Lozanoe675d962020-07-03 08:07:17 -0300871\t.name\t\t= "sandbox_gpio",
Simon Glasse2119082021-02-03 06:01:19 -0700872\t.plat\t\t= &dtv_gpios_at_0,
Simon Glass39edb952020-12-03 16:55:19 -0700873\t.plat_size\t= sizeof(dtv_gpios_at_0),
Simon Glass36b15e22020-10-03 11:31:35 -0600874\t.parent_idx\t= -1,
Walter Lozanoe675d962020-07-03 08:07:17 -0300875};
876
877''', data)
878
Walter Lozanoa324e412020-06-25 01:10:08 -0300879 def test_invalid_driver(self):
880 """Test output from a device tree file with an invalid driver"""
881 dtb_file = get_dtb_file('dtoc_test_invalid_driver.dts')
Simon Glass80025522022-01-29 14:14:04 -0700882 output = tools.get_output_filename('output')
Simon Glass14d64e32025-04-29 07:21:59 -0600883 with terminal.capture() as _:
Simon Glass3809ad92021-02-03 06:01:12 -0700884 dtb_platdata.run_steps(
885 ['struct'], dtb_file, False, output, [], None, False,
886 scan=copy_scan())
Walter Lozanoa324e412020-06-25 01:10:08 -0300887 with open(output) as infile:
888 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700889 self._check_strings(HEADER + '''
Walter Lozanoa324e412020-06-25 01:10:08 -0300890struct dtd_invalid {
891};
892''', data)
893
Simon Glass14d64e32025-04-29 07:21:59 -0600894 with terminal.capture() as _:
Simon Glass3809ad92021-02-03 06:01:12 -0700895 dtb_platdata.run_steps(
896 ['platdata'], dtb_file, False, output, [], None, False,
897 scan=copy_scan())
Walter Lozanoa324e412020-06-25 01:10:08 -0300898 with open(output) as infile:
899 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700900 self._check_strings(C_HEADER + '''
Simon Glasse2119082021-02-03 06:01:19 -0700901/*
902 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
903 *
904 * idx driver_info driver
905 * --- -------------------- --------------------
906 * 0: spl_test invalid
907 * --- -------------------- --------------------
908 */
909
Simon Glass192f8132020-10-03 11:31:25 -0600910/* Node /spl-test index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300911static struct dtd_invalid dtv_spl_test = {
Walter Lozanoa324e412020-06-25 01:10:08 -0300912};
Simon Glass1d8364a2020-12-28 20:34:54 -0700913U_BOOT_DRVINFO(spl_test) = {
Walter Lozanoa324e412020-06-25 01:10:08 -0300914\t.name\t\t= "invalid",
Simon Glasse2119082021-02-03 06:01:19 -0700915\t.plat\t\t= &dtv_spl_test,
Simon Glass39edb952020-12-03 16:55:19 -0700916\t.plat_size\t= sizeof(dtv_spl_test),
Simon Glass36b15e22020-10-03 11:31:35 -0600917\t.parent_idx\t= -1,
Walter Lozanoa324e412020-06-25 01:10:08 -0300918};
919
920''', data)
921
Simon Glass9d2eb922017-06-18 22:09:06 -0600922 def test_phandle(self):
923 """Test output from a node containing a phandle reference"""
924 dtb_file = get_dtb_file('dtoc_test_phandle.dts')
Simon Glass80025522022-01-29 14:14:04 -0700925 output = tools.get_output_filename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -0300926 self.run_test(['struct'], dtb_file, output)
Simon Glass9d2eb922017-06-18 22:09:06 -0600927 with open(output) as infile:
928 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700929 self._check_strings(HEADER + '''
Simon Glass9d2eb922017-06-18 22:09:06 -0600930struct dtd_source {
Simon Glass3deeb472017-08-29 14:15:59 -0600931\tstruct phandle_2_arg clocks[4];
Simon Glasse8cea0e2023-01-11 16:10:18 -0700932\tunsigned char phandle_name_offset[13];
Simon Glass9d2eb922017-06-18 22:09:06 -0600933};
934struct dtd_target {
935\tfdt32_t\t\tintval;
936};
937''', data)
938
Walter Lozanoa324e412020-06-25 01:10:08 -0300939 self.run_test(['platdata'], dtb_file, output)
Simon Glass9d2eb922017-06-18 22:09:06 -0600940 with open(output) as infile:
941 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -0700942 self._check_strings(C_HEADER + '''
Simon Glasse2119082021-02-03 06:01:19 -0700943/*
944 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
945 *
946 * idx driver_info driver
947 * --- -------------------- --------------------
948 * 0: phandle2_target target
949 * 1: phandle3_target target
950 * 2: phandle_source source
951 * 3: phandle_source2 source
952 * 4: phandle_target target
953 * --- -------------------- --------------------
954 */
955
Simon Glass192f8132020-10-03 11:31:25 -0600956/* Node /phandle2-target index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300957static struct dtd_target dtv_phandle2_target = {
Simon Glass3deeb472017-08-29 14:15:59 -0600958\t.intval\t\t\t= 0x1,
959};
Simon Glass1d8364a2020-12-28 20:34:54 -0700960U_BOOT_DRVINFO(phandle2_target) = {
Simon Glass3deeb472017-08-29 14:15:59 -0600961\t.name\t\t= "target",
Simon Glasse2119082021-02-03 06:01:19 -0700962\t.plat\t\t= &dtv_phandle2_target,
Simon Glass39edb952020-12-03 16:55:19 -0700963\t.plat_size\t= sizeof(dtv_phandle2_target),
Simon Glass36b15e22020-10-03 11:31:35 -0600964\t.parent_idx\t= -1,
Simon Glass3deeb472017-08-29 14:15:59 -0600965};
966
Simon Glass192f8132020-10-03 11:31:25 -0600967/* Node /phandle3-target index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300968static struct dtd_target dtv_phandle3_target = {
Simon Glass3deeb472017-08-29 14:15:59 -0600969\t.intval\t\t\t= 0x2,
970};
Simon Glass1d8364a2020-12-28 20:34:54 -0700971U_BOOT_DRVINFO(phandle3_target) = {
Simon Glass3deeb472017-08-29 14:15:59 -0600972\t.name\t\t= "target",
Simon Glasse2119082021-02-03 06:01:19 -0700973\t.plat\t\t= &dtv_phandle3_target,
Simon Glass39edb952020-12-03 16:55:19 -0700974\t.plat_size\t= sizeof(dtv_phandle3_target),
Simon Glass36b15e22020-10-03 11:31:35 -0600975\t.parent_idx\t= -1,
Simon Glass3deeb472017-08-29 14:15:59 -0600976};
977
Simon Glass192f8132020-10-03 11:31:25 -0600978/* Node /phandle-source index 2 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300979static struct dtd_source dtv_phandle_source = {
Simon Glassd0cd0752017-08-29 14:15:57 -0600980\t.clocks\t\t\t= {
Simon Glass5792f4b2020-10-03 11:31:40 -0600981\t\t\t{4, {}},
982\t\t\t{0, {11}},
983\t\t\t{1, {12, 13}},
984\t\t\t{4, {}},},
Simon Glasse8cea0e2023-01-11 16:10:18 -0700985\t.phandle_name_offset = {0x0, 0x0, 0x0, 0x3, 0x66, 0x72, 0x65, 0x64,
986\t\t0x0, 0x0, 0x0, 0x0, 0x7b},
Simon Glass9d2eb922017-06-18 22:09:06 -0600987};
Simon Glass1d8364a2020-12-28 20:34:54 -0700988U_BOOT_DRVINFO(phandle_source) = {
Simon Glass9d2eb922017-06-18 22:09:06 -0600989\t.name\t\t= "source",
Simon Glasse2119082021-02-03 06:01:19 -0700990\t.plat\t\t= &dtv_phandle_source,
Simon Glass39edb952020-12-03 16:55:19 -0700991\t.plat_size\t= sizeof(dtv_phandle_source),
Simon Glass36b15e22020-10-03 11:31:35 -0600992\t.parent_idx\t= -1,
Simon Glass9d2eb922017-06-18 22:09:06 -0600993};
994
Simon Glass192f8132020-10-03 11:31:25 -0600995/* Node /phandle-source2 index 3 */
Walter Lozanodc5b4372020-06-25 01:10:13 -0300996static struct dtd_source dtv_phandle_source2 = {
Simon Glass609e2b12018-07-06 10:27:31 -0600997\t.clocks\t\t\t= {
Simon Glass5792f4b2020-10-03 11:31:40 -0600998\t\t\t{4, {}},},
Simon Glass609e2b12018-07-06 10:27:31 -0600999};
Simon Glass1d8364a2020-12-28 20:34:54 -07001000U_BOOT_DRVINFO(phandle_source2) = {
Simon Glass609e2b12018-07-06 10:27:31 -06001001\t.name\t\t= "source",
Simon Glasse2119082021-02-03 06:01:19 -07001002\t.plat\t\t= &dtv_phandle_source2,
Simon Glass39edb952020-12-03 16:55:19 -07001003\t.plat_size\t= sizeof(dtv_phandle_source2),
Simon Glass36b15e22020-10-03 11:31:35 -06001004\t.parent_idx\t= -1,
Simon Glass609e2b12018-07-06 10:27:31 -06001005};
1006
Simon Glass16382ce2020-12-28 20:35:04 -07001007/* Node /phandle-target index 4 */
1008static struct dtd_target dtv_phandle_target = {
1009\t.intval\t\t\t= 0x0,
1010};
1011U_BOOT_DRVINFO(phandle_target) = {
1012\t.name\t\t= "target",
Simon Glasse2119082021-02-03 06:01:19 -07001013\t.plat\t\t= &dtv_phandle_target,
Simon Glass16382ce2020-12-28 20:35:04 -07001014\t.plat_size\t= sizeof(dtv_phandle_target),
1015\t.parent_idx\t= -1,
1016};
1017
Simon Glass9d2eb922017-06-18 22:09:06 -06001018''', data)
1019
Simon Glass961c1ce2018-07-06 10:27:35 -06001020 def test_phandle_single(self):
1021 """Test output from a node containing a phandle reference"""
1022 dtb_file = get_dtb_file('dtoc_test_phandle_single.dts')
Simon Glass80025522022-01-29 14:14:04 -07001023 output = tools.get_output_filename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -03001024 self.run_test(['struct'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -06001025 with open(output) as infile:
1026 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001027 self._check_strings(HEADER + '''
Simon Glass961c1ce2018-07-06 10:27:35 -06001028struct dtd_source {
1029\tstruct phandle_0_arg clocks[1];
1030};
1031struct dtd_target {
1032\tfdt32_t\t\tintval;
1033};
1034''', data)
1035
1036 def test_phandle_reorder(self):
1037 """Test that phandle targets are generated before their references"""
1038 dtb_file = get_dtb_file('dtoc_test_phandle_reorder.dts')
Simon Glass80025522022-01-29 14:14:04 -07001039 output = tools.get_output_filename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -03001040 self.run_test(['platdata'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -06001041 with open(output) as infile:
1042 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001043 self._check_strings(C_HEADER + '''
Simon Glasse2119082021-02-03 06:01:19 -07001044/*
1045 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
1046 *
1047 * idx driver_info driver
1048 * --- -------------------- --------------------
1049 * 0: phandle_source2 source
1050 * 1: phandle_target target
1051 * --- -------------------- --------------------
1052 */
1053
Simon Glass192f8132020-10-03 11:31:25 -06001054/* Node /phandle-source2 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001055static struct dtd_source dtv_phandle_source2 = {
Simon Glass961c1ce2018-07-06 10:27:35 -06001056\t.clocks\t\t\t= {
Simon Glass5792f4b2020-10-03 11:31:40 -06001057\t\t\t{1, {}},},
Simon Glass961c1ce2018-07-06 10:27:35 -06001058};
Simon Glass1d8364a2020-12-28 20:34:54 -07001059U_BOOT_DRVINFO(phandle_source2) = {
Simon Glass961c1ce2018-07-06 10:27:35 -06001060\t.name\t\t= "source",
Simon Glasse2119082021-02-03 06:01:19 -07001061\t.plat\t\t= &dtv_phandle_source2,
Simon Glass39edb952020-12-03 16:55:19 -07001062\t.plat_size\t= sizeof(dtv_phandle_source2),
Simon Glass36b15e22020-10-03 11:31:35 -06001063\t.parent_idx\t= -1,
Simon Glass961c1ce2018-07-06 10:27:35 -06001064};
1065
Simon Glass16382ce2020-12-28 20:35:04 -07001066/* Node /phandle-target index 1 */
1067static struct dtd_target dtv_phandle_target = {
1068};
1069U_BOOT_DRVINFO(phandle_target) = {
1070\t.name\t\t= "target",
Simon Glasse2119082021-02-03 06:01:19 -07001071\t.plat\t\t= &dtv_phandle_target,
Simon Glass16382ce2020-12-28 20:35:04 -07001072\t.plat_size\t= sizeof(dtv_phandle_target),
1073\t.parent_idx\t= -1,
1074};
1075
Simon Glass961c1ce2018-07-06 10:27:35 -06001076''', data)
1077
Walter Lozano5541fc02020-06-25 01:10:17 -03001078 def test_phandle_cd_gpio(self):
1079 """Test that phandle targets are generated when unsing cd-gpios"""
1080 dtb_file = get_dtb_file('dtoc_test_phandle_cd_gpios.dts')
Simon Glass80025522022-01-29 14:14:04 -07001081 output = tools.get_output_filename('output')
Simon Glass3809ad92021-02-03 06:01:12 -07001082 dtb_platdata.run_steps(
1083 ['platdata'], dtb_file, False, output, [], None, False,
1084 warning_disabled=True, scan=copy_scan())
Walter Lozano5541fc02020-06-25 01:10:17 -03001085 with open(output) as infile:
1086 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001087 self._check_strings(C_HEADER + '''
Simon Glasse2119082021-02-03 06:01:19 -07001088/*
1089 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
1090 *
1091 * idx driver_info driver
1092 * --- -------------------- --------------------
1093 * 0: phandle2_target target
1094 * 1: phandle3_target target
1095 * 2: phandle_source source
1096 * 3: phandle_source2 source
1097 * 4: phandle_target target
1098 * --- -------------------- --------------------
1099 */
1100
Simon Glass192f8132020-10-03 11:31:25 -06001101/* Node /phandle2-target index 0 */
Walter Lozano5541fc02020-06-25 01:10:17 -03001102static struct dtd_target dtv_phandle2_target = {
1103\t.intval\t\t\t= 0x1,
1104};
Simon Glass1d8364a2020-12-28 20:34:54 -07001105U_BOOT_DRVINFO(phandle2_target) = {
Walter Lozano5541fc02020-06-25 01:10:17 -03001106\t.name\t\t= "target",
Simon Glasse2119082021-02-03 06:01:19 -07001107\t.plat\t\t= &dtv_phandle2_target,
Simon Glass39edb952020-12-03 16:55:19 -07001108\t.plat_size\t= sizeof(dtv_phandle2_target),
Simon Glass36b15e22020-10-03 11:31:35 -06001109\t.parent_idx\t= -1,
Walter Lozano5541fc02020-06-25 01:10:17 -03001110};
1111
Simon Glass192f8132020-10-03 11:31:25 -06001112/* Node /phandle3-target index 1 */
Walter Lozano5541fc02020-06-25 01:10:17 -03001113static struct dtd_target dtv_phandle3_target = {
1114\t.intval\t\t\t= 0x2,
1115};
Simon Glass1d8364a2020-12-28 20:34:54 -07001116U_BOOT_DRVINFO(phandle3_target) = {
Walter Lozano5541fc02020-06-25 01:10:17 -03001117\t.name\t\t= "target",
Simon Glasse2119082021-02-03 06:01:19 -07001118\t.plat\t\t= &dtv_phandle3_target,
Simon Glass39edb952020-12-03 16:55:19 -07001119\t.plat_size\t= sizeof(dtv_phandle3_target),
Simon Glass36b15e22020-10-03 11:31:35 -06001120\t.parent_idx\t= -1,
Walter Lozano5541fc02020-06-25 01:10:17 -03001121};
1122
Simon Glass192f8132020-10-03 11:31:25 -06001123/* Node /phandle-source index 2 */
Walter Lozano5541fc02020-06-25 01:10:17 -03001124static struct dtd_source dtv_phandle_source = {
1125\t.cd_gpios\t\t= {
Simon Glass5792f4b2020-10-03 11:31:40 -06001126\t\t\t{4, {}},
1127\t\t\t{0, {11}},
1128\t\t\t{1, {12, 13}},
1129\t\t\t{4, {}},},
Walter Lozano5541fc02020-06-25 01:10:17 -03001130};
Simon Glass1d8364a2020-12-28 20:34:54 -07001131U_BOOT_DRVINFO(phandle_source) = {
Walter Lozano5541fc02020-06-25 01:10:17 -03001132\t.name\t\t= "source",
Simon Glasse2119082021-02-03 06:01:19 -07001133\t.plat\t\t= &dtv_phandle_source,
Simon Glass39edb952020-12-03 16:55:19 -07001134\t.plat_size\t= sizeof(dtv_phandle_source),
Simon Glass36b15e22020-10-03 11:31:35 -06001135\t.parent_idx\t= -1,
Walter Lozano5541fc02020-06-25 01:10:17 -03001136};
1137
Simon Glass192f8132020-10-03 11:31:25 -06001138/* Node /phandle-source2 index 3 */
Walter Lozano5541fc02020-06-25 01:10:17 -03001139static struct dtd_source dtv_phandle_source2 = {
1140\t.cd_gpios\t\t= {
Simon Glass5792f4b2020-10-03 11:31:40 -06001141\t\t\t{4, {}},},
Walter Lozano5541fc02020-06-25 01:10:17 -03001142};
Simon Glass1d8364a2020-12-28 20:34:54 -07001143U_BOOT_DRVINFO(phandle_source2) = {
Walter Lozano5541fc02020-06-25 01:10:17 -03001144\t.name\t\t= "source",
Simon Glasse2119082021-02-03 06:01:19 -07001145\t.plat\t\t= &dtv_phandle_source2,
Simon Glass39edb952020-12-03 16:55:19 -07001146\t.plat_size\t= sizeof(dtv_phandle_source2),
Simon Glass36b15e22020-10-03 11:31:35 -06001147\t.parent_idx\t= -1,
Walter Lozano5541fc02020-06-25 01:10:17 -03001148};
1149
Simon Glass16382ce2020-12-28 20:35:04 -07001150/* Node /phandle-target index 4 */
1151static struct dtd_target dtv_phandle_target = {
1152\t.intval\t\t\t= 0x0,
1153};
1154U_BOOT_DRVINFO(phandle_target) = {
1155\t.name\t\t= "target",
Simon Glasse2119082021-02-03 06:01:19 -07001156\t.plat\t\t= &dtv_phandle_target,
Simon Glass16382ce2020-12-28 20:35:04 -07001157\t.plat_size\t= sizeof(dtv_phandle_target),
1158\t.parent_idx\t= -1,
1159};
1160
Walter Lozano5541fc02020-06-25 01:10:17 -03001161''', data)
1162
Simon Glass961c1ce2018-07-06 10:27:35 -06001163 def test_phandle_bad(self):
1164 """Test a node containing an invalid phandle fails"""
Simon Glass04150022018-10-01 21:12:43 -06001165 dtb_file = get_dtb_file('dtoc_test_phandle_bad.dts',
1166 capture_stderr=True)
Simon Glass80025522022-01-29 14:14:04 -07001167 output = tools.get_output_filename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -07001168 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -03001169 self.run_test(['struct'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -06001170 self.assertIn("Cannot parse 'clocks' in node 'phandle-source'",
Simon Glassa60cabd2020-12-28 20:34:47 -07001171 str(exc.exception))
Simon Glass961c1ce2018-07-06 10:27:35 -06001172
1173 def test_phandle_bad2(self):
1174 """Test a phandle target missing its #*-cells property"""
Simon Glass04150022018-10-01 21:12:43 -06001175 dtb_file = get_dtb_file('dtoc_test_phandle_bad2.dts',
1176 capture_stderr=True)
Simon Glass80025522022-01-29 14:14:04 -07001177 output = tools.get_output_filename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -07001178 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -03001179 self.run_test(['struct'], dtb_file, output)
Walter Lozano179f0b62020-06-25 01:10:16 -03001180 self.assertIn("Node 'phandle-target' has no cells property",
Simon Glassa60cabd2020-12-28 20:34:47 -07001181 str(exc.exception))
Simon Glass961c1ce2018-07-06 10:27:35 -06001182
Simon Glass1b1fe412017-08-29 14:15:50 -06001183 def test_addresses64(self):
1184 """Test output from a node with a 'reg' property with na=2, ns=2"""
1185 dtb_file = get_dtb_file('dtoc_test_addr64.dts')
Simon Glass80025522022-01-29 14:14:04 -07001186 output = tools.get_output_filename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -03001187 self.run_test(['struct'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -06001188 with open(output) as infile:
1189 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001190 self._check_strings(HEADER + '''
Simon Glass1b1fe412017-08-29 14:15:50 -06001191struct dtd_test1 {
1192\tfdt64_t\t\treg[2];
1193};
1194struct dtd_test2 {
1195\tfdt64_t\t\treg[2];
1196};
1197struct dtd_test3 {
1198\tfdt64_t\t\treg[4];
1199};
1200''', data)
1201
Walter Lozanoa324e412020-06-25 01:10:08 -03001202 self.run_test(['platdata'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -06001203 with open(output) as infile:
1204 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001205 self._check_strings(C_HEADER + '''
Simon Glasse2119082021-02-03 06:01:19 -07001206/*
1207 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
1208 *
1209 * idx driver_info driver
1210 * --- -------------------- --------------------
1211 * 0: test1 test1
1212 * 1: test2 test2
1213 * 2: test3 test3
1214 * --- -------------------- --------------------
1215 */
1216
Simon Glass192f8132020-10-03 11:31:25 -06001217/* Node /test1 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001218static struct dtd_test1 dtv_test1 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001219\t.reg\t\t\t= {0x1234, 0x5678},
1220};
Simon Glass1d8364a2020-12-28 20:34:54 -07001221U_BOOT_DRVINFO(test1) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001222\t.name\t\t= "test1",
Simon Glasse2119082021-02-03 06:01:19 -07001223\t.plat\t\t= &dtv_test1,
Simon Glass39edb952020-12-03 16:55:19 -07001224\t.plat_size\t= sizeof(dtv_test1),
Simon Glass36b15e22020-10-03 11:31:35 -06001225\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001226};
1227
Simon Glass192f8132020-10-03 11:31:25 -06001228/* Node /test2 index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001229static struct dtd_test2 dtv_test2 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001230\t.reg\t\t\t= {0x1234567890123456, 0x9876543210987654},
1231};
Simon Glass1d8364a2020-12-28 20:34:54 -07001232U_BOOT_DRVINFO(test2) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001233\t.name\t\t= "test2",
Simon Glasse2119082021-02-03 06:01:19 -07001234\t.plat\t\t= &dtv_test2,
Simon Glass39edb952020-12-03 16:55:19 -07001235\t.plat_size\t= sizeof(dtv_test2),
Simon Glass36b15e22020-10-03 11:31:35 -06001236\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001237};
1238
Simon Glass192f8132020-10-03 11:31:25 -06001239/* Node /test3 index 2 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001240static struct dtd_test3 dtv_test3 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001241\t.reg\t\t\t= {0x1234567890123456, 0x9876543210987654, 0x2, 0x3},
1242};
Simon Glass1d8364a2020-12-28 20:34:54 -07001243U_BOOT_DRVINFO(test3) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001244\t.name\t\t= "test3",
Simon Glasse2119082021-02-03 06:01:19 -07001245\t.plat\t\t= &dtv_test3,
Simon Glass39edb952020-12-03 16:55:19 -07001246\t.plat_size\t= sizeof(dtv_test3),
Simon Glass36b15e22020-10-03 11:31:35 -06001247\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001248};
1249
Simon Glass2500de22020-12-28 20:35:05 -07001250''', data)
Simon Glass1b1fe412017-08-29 14:15:50 -06001251
1252 def test_addresses32(self):
1253 """Test output from a node with a 'reg' property with na=1, ns=1"""
1254 dtb_file = get_dtb_file('dtoc_test_addr32.dts')
Simon Glass80025522022-01-29 14:14:04 -07001255 output = tools.get_output_filename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -03001256 self.run_test(['struct'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -06001257 with open(output) as infile:
1258 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001259 self._check_strings(HEADER + '''
Simon Glass1b1fe412017-08-29 14:15:50 -06001260struct dtd_test1 {
1261\tfdt32_t\t\treg[2];
1262};
1263struct dtd_test2 {
1264\tfdt32_t\t\treg[4];
1265};
1266''', data)
1267
Walter Lozanoa324e412020-06-25 01:10:08 -03001268 self.run_test(['platdata'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -06001269 with open(output) as infile:
1270 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001271 self._check_strings(C_HEADER + '''
Simon Glasse2119082021-02-03 06:01:19 -07001272/*
1273 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
1274 *
1275 * idx driver_info driver
1276 * --- -------------------- --------------------
1277 * 0: test1 test1
1278 * 1: test2 test2
1279 * --- -------------------- --------------------
1280 */
1281
Simon Glass192f8132020-10-03 11:31:25 -06001282/* Node /test1 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001283static struct dtd_test1 dtv_test1 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001284\t.reg\t\t\t= {0x1234, 0x5678},
1285};
Simon Glass1d8364a2020-12-28 20:34:54 -07001286U_BOOT_DRVINFO(test1) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001287\t.name\t\t= "test1",
Simon Glasse2119082021-02-03 06:01:19 -07001288\t.plat\t\t= &dtv_test1,
Simon Glass39edb952020-12-03 16:55:19 -07001289\t.plat_size\t= sizeof(dtv_test1),
Simon Glass36b15e22020-10-03 11:31:35 -06001290\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001291};
1292
Simon Glass192f8132020-10-03 11:31:25 -06001293/* Node /test2 index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001294static struct dtd_test2 dtv_test2 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001295\t.reg\t\t\t= {0x12345678, 0x98765432, 0x2, 0x3},
1296};
Simon Glass1d8364a2020-12-28 20:34:54 -07001297U_BOOT_DRVINFO(test2) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001298\t.name\t\t= "test2",
Simon Glasse2119082021-02-03 06:01:19 -07001299\t.plat\t\t= &dtv_test2,
Simon Glass39edb952020-12-03 16:55:19 -07001300\t.plat_size\t= sizeof(dtv_test2),
Simon Glass36b15e22020-10-03 11:31:35 -06001301\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001302};
1303
Simon Glass2500de22020-12-28 20:35:05 -07001304''', data)
Simon Glass1b1fe412017-08-29 14:15:50 -06001305
1306 def test_addresses64_32(self):
1307 """Test output from a node with a 'reg' property with na=2, ns=1"""
1308 dtb_file = get_dtb_file('dtoc_test_addr64_32.dts')
Simon Glass80025522022-01-29 14:14:04 -07001309 output = tools.get_output_filename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -03001310 self.run_test(['struct'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -06001311 with open(output) as infile:
1312 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001313 self._check_strings(HEADER + '''
Simon Glass1b1fe412017-08-29 14:15:50 -06001314struct dtd_test1 {
1315\tfdt64_t\t\treg[2];
1316};
1317struct dtd_test2 {
1318\tfdt64_t\t\treg[2];
1319};
1320struct dtd_test3 {
1321\tfdt64_t\t\treg[4];
1322};
1323''', data)
1324
Walter Lozanoa324e412020-06-25 01:10:08 -03001325 self.run_test(['platdata'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -06001326 with open(output) as infile:
1327 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001328 self._check_strings(C_HEADER + '''
Simon Glasse2119082021-02-03 06:01:19 -07001329/*
1330 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
1331 *
1332 * idx driver_info driver
1333 * --- -------------------- --------------------
1334 * 0: test1 test1
1335 * 1: test2 test2
1336 * 2: test3 test3
1337 * --- -------------------- --------------------
1338 */
1339
Simon Glass192f8132020-10-03 11:31:25 -06001340/* Node /test1 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001341static struct dtd_test1 dtv_test1 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001342\t.reg\t\t\t= {0x123400000000, 0x5678},
1343};
Simon Glass1d8364a2020-12-28 20:34:54 -07001344U_BOOT_DRVINFO(test1) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001345\t.name\t\t= "test1",
Simon Glasse2119082021-02-03 06:01:19 -07001346\t.plat\t\t= &dtv_test1,
Simon Glass39edb952020-12-03 16:55:19 -07001347\t.plat_size\t= sizeof(dtv_test1),
Simon Glass36b15e22020-10-03 11:31:35 -06001348\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001349};
1350
Simon Glass192f8132020-10-03 11:31:25 -06001351/* Node /test2 index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001352static struct dtd_test2 dtv_test2 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001353\t.reg\t\t\t= {0x1234567890123456, 0x98765432},
1354};
Simon Glass1d8364a2020-12-28 20:34:54 -07001355U_BOOT_DRVINFO(test2) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001356\t.name\t\t= "test2",
Simon Glasse2119082021-02-03 06:01:19 -07001357\t.plat\t\t= &dtv_test2,
Simon Glass39edb952020-12-03 16:55:19 -07001358\t.plat_size\t= sizeof(dtv_test2),
Simon Glass36b15e22020-10-03 11:31:35 -06001359\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001360};
1361
Simon Glass192f8132020-10-03 11:31:25 -06001362/* Node /test3 index 2 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001363static struct dtd_test3 dtv_test3 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001364\t.reg\t\t\t= {0x1234567890123456, 0x98765432, 0x2, 0x3},
1365};
Simon Glass1d8364a2020-12-28 20:34:54 -07001366U_BOOT_DRVINFO(test3) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001367\t.name\t\t= "test3",
Simon Glasse2119082021-02-03 06:01:19 -07001368\t.plat\t\t= &dtv_test3,
Simon Glass39edb952020-12-03 16:55:19 -07001369\t.plat_size\t= sizeof(dtv_test3),
Simon Glass36b15e22020-10-03 11:31:35 -06001370\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001371};
1372
Simon Glass2500de22020-12-28 20:35:05 -07001373''', data)
Simon Glass1b1fe412017-08-29 14:15:50 -06001374
1375 def test_addresses32_64(self):
1376 """Test output from a node with a 'reg' property with na=1, ns=2"""
1377 dtb_file = get_dtb_file('dtoc_test_addr32_64.dts')
Simon Glass80025522022-01-29 14:14:04 -07001378 output = tools.get_output_filename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -03001379 self.run_test(['struct'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -06001380 with open(output) as infile:
1381 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001382 self._check_strings(HEADER + '''
Simon Glass1b1fe412017-08-29 14:15:50 -06001383struct dtd_test1 {
1384\tfdt64_t\t\treg[2];
1385};
1386struct dtd_test2 {
1387\tfdt64_t\t\treg[2];
1388};
1389struct dtd_test3 {
1390\tfdt64_t\t\treg[4];
1391};
1392''', data)
1393
Walter Lozanoa324e412020-06-25 01:10:08 -03001394 self.run_test(['platdata'], dtb_file, output)
Simon Glass1b1fe412017-08-29 14:15:50 -06001395 with open(output) as infile:
1396 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001397 self._check_strings(C_HEADER + '''
Simon Glasse2119082021-02-03 06:01:19 -07001398/*
1399 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
1400 *
1401 * idx driver_info driver
1402 * --- -------------------- --------------------
1403 * 0: test1 test1
1404 * 1: test2 test2
1405 * 2: test3 test3
1406 * --- -------------------- --------------------
1407 */
1408
Simon Glass192f8132020-10-03 11:31:25 -06001409/* Node /test1 index 0 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001410static struct dtd_test1 dtv_test1 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001411\t.reg\t\t\t= {0x1234, 0x567800000000},
1412};
Simon Glass1d8364a2020-12-28 20:34:54 -07001413U_BOOT_DRVINFO(test1) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001414\t.name\t\t= "test1",
Simon Glasse2119082021-02-03 06:01:19 -07001415\t.plat\t\t= &dtv_test1,
Simon Glass39edb952020-12-03 16:55:19 -07001416\t.plat_size\t= sizeof(dtv_test1),
Simon Glass36b15e22020-10-03 11:31:35 -06001417\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001418};
1419
Simon Glass192f8132020-10-03 11:31:25 -06001420/* Node /test2 index 1 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001421static struct dtd_test2 dtv_test2 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001422\t.reg\t\t\t= {0x12345678, 0x9876543210987654},
1423};
Simon Glass1d8364a2020-12-28 20:34:54 -07001424U_BOOT_DRVINFO(test2) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001425\t.name\t\t= "test2",
Simon Glasse2119082021-02-03 06:01:19 -07001426\t.plat\t\t= &dtv_test2,
Simon Glass39edb952020-12-03 16:55:19 -07001427\t.plat_size\t= sizeof(dtv_test2),
Simon Glass36b15e22020-10-03 11:31:35 -06001428\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001429};
1430
Simon Glass192f8132020-10-03 11:31:25 -06001431/* Node /test3 index 2 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001432static struct dtd_test3 dtv_test3 = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001433\t.reg\t\t\t= {0x12345678, 0x9876543210987654, 0x2, 0x3},
1434};
Simon Glass1d8364a2020-12-28 20:34:54 -07001435U_BOOT_DRVINFO(test3) = {
Simon Glass1b1fe412017-08-29 14:15:50 -06001436\t.name\t\t= "test3",
Simon Glasse2119082021-02-03 06:01:19 -07001437\t.plat\t\t= &dtv_test3,
Simon Glass39edb952020-12-03 16:55:19 -07001438\t.plat_size\t= sizeof(dtv_test3),
Simon Glass36b15e22020-10-03 11:31:35 -06001439\t.parent_idx\t= -1,
Simon Glass1b1fe412017-08-29 14:15:50 -06001440};
1441
Simon Glass2500de22020-12-28 20:35:05 -07001442''', data)
Simon Glass961c1ce2018-07-06 10:27:35 -06001443
1444 def test_bad_reg(self):
1445 """Test that a reg property with an invalid type generates an error"""
Simon Glass3bce93d2018-07-06 10:27:37 -06001446 # Capture stderr since dtc will emit warnings for this file
1447 dtb_file = get_dtb_file('dtoc_test_bad_reg.dts', capture_stderr=True)
Simon Glass80025522022-01-29 14:14:04 -07001448 output = tools.get_output_filename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -07001449 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -03001450 self.run_test(['struct'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -06001451 self.assertIn("Node 'spl-test' reg property is not an int",
Simon Glassa60cabd2020-12-28 20:34:47 -07001452 str(exc.exception))
Simon Glass961c1ce2018-07-06 10:27:35 -06001453
1454 def test_bad_reg2(self):
1455 """Test that a reg property with an invalid cell count is detected"""
Simon Glass3bce93d2018-07-06 10:27:37 -06001456 # Capture stderr since dtc will emit warnings for this file
1457 dtb_file = get_dtb_file('dtoc_test_bad_reg2.dts', capture_stderr=True)
Simon Glass80025522022-01-29 14:14:04 -07001458 output = tools.get_output_filename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -07001459 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -03001460 self.run_test(['struct'], dtb_file, output)
Simon Glassa60cabd2020-12-28 20:34:47 -07001461 self.assertIn(
Simon Glass4415dc12021-03-26 16:17:27 +13001462 "Node 'spl-test' (parent '/') reg property has 3 cells which is not a multiple of na + ns = 1 + 1)",
Simon Glassa60cabd2020-12-28 20:34:47 -07001463 str(exc.exception))
Simon Glass961c1ce2018-07-06 10:27:35 -06001464
1465 def test_add_prop(self):
1466 """Test that a subequent node can add a new property to a struct"""
1467 dtb_file = get_dtb_file('dtoc_test_add_prop.dts')
Simon Glass80025522022-01-29 14:14:04 -07001468 output = tools.get_output_filename('output')
Walter Lozanoa324e412020-06-25 01:10:08 -03001469 self.run_test(['struct'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -06001470 with open(output) as infile:
1471 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001472 self._check_strings(HEADER + '''
Simon Glass961c1ce2018-07-06 10:27:35 -06001473struct dtd_sandbox_spl_test {
1474\tfdt32_t\t\tintarray;
1475\tfdt32_t\t\tintval;
1476};
1477''', data)
1478
Walter Lozanoa324e412020-06-25 01:10:08 -03001479 self.run_test(['platdata'], dtb_file, output)
Simon Glass961c1ce2018-07-06 10:27:35 -06001480 with open(output) as infile:
1481 data = infile.read()
Simon Glassa60cabd2020-12-28 20:34:47 -07001482 self._check_strings(C_HEADER + '''
Simon Glassbe749002021-02-03 06:01:15 -07001483/*
Simon Glasse2119082021-02-03 06:01:19 -07001484 * driver_info declarations, ordered by 'struct driver_info' linker_list idx:
1485 *
1486 * idx driver_info driver
1487 * --- -------------------- --------------------
1488 * 0: spl_test sandbox_spl_test
1489 * 1: spl_test2 sandbox_spl_test
1490 * --- -------------------- --------------------
1491 */
1492
1493/*
Simon Glassbe749002021-02-03 06:01:15 -07001494 * Node /spl-test index 0
1495 * driver sandbox_spl_test parent None
1496 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001497static struct dtd_sandbox_spl_test dtv_spl_test = {
Simon Glass961c1ce2018-07-06 10:27:35 -06001498\t.intval\t\t\t= 0x1,
1499};
Simon Glass1d8364a2020-12-28 20:34:54 -07001500U_BOOT_DRVINFO(spl_test) = {
Simon Glass961c1ce2018-07-06 10:27:35 -06001501\t.name\t\t= "sandbox_spl_test",
Simon Glasse2119082021-02-03 06:01:19 -07001502\t.plat\t\t= &dtv_spl_test,
Simon Glass39edb952020-12-03 16:55:19 -07001503\t.plat_size\t= sizeof(dtv_spl_test),
Simon Glass36b15e22020-10-03 11:31:35 -06001504\t.parent_idx\t= -1,
Simon Glass961c1ce2018-07-06 10:27:35 -06001505};
1506
Simon Glassbe749002021-02-03 06:01:15 -07001507/*
1508 * Node /spl-test2 index 1
1509 * driver sandbox_spl_test parent None
1510 */
Walter Lozanodc5b4372020-06-25 01:10:13 -03001511static struct dtd_sandbox_spl_test dtv_spl_test2 = {
Simon Glass961c1ce2018-07-06 10:27:35 -06001512\t.intarray\t\t= 0x5,
1513};
Simon Glass1d8364a2020-12-28 20:34:54 -07001514U_BOOT_DRVINFO(spl_test2) = {
Simon Glass961c1ce2018-07-06 10:27:35 -06001515\t.name\t\t= "sandbox_spl_test",
Simon Glasse2119082021-02-03 06:01:19 -07001516\t.plat\t\t= &dtv_spl_test2,
Simon Glass39edb952020-12-03 16:55:19 -07001517\t.plat_size\t= sizeof(dtv_spl_test2),
Simon Glass36b15e22020-10-03 11:31:35 -06001518\t.parent_idx\t= -1,
Simon Glass961c1ce2018-07-06 10:27:35 -06001519};
1520
Simon Glass2500de22020-12-28 20:35:05 -07001521''', data)
Simon Glass961c1ce2018-07-06 10:27:35 -06001522
Simon Glassa60cabd2020-12-28 20:34:47 -07001523 def test_stdout(self):
Simon Glass961c1ce2018-07-06 10:27:35 -06001524 """Test output to stdout"""
1525 dtb_file = get_dtb_file('dtoc_test_simple.dts')
Simon Glass14d64e32025-04-29 07:21:59 -06001526 with terminal.capture() as (stdout, _):
Simon Glass6ca0c7a2020-12-28 20:34:48 -07001527 self.run_test(['struct'], dtb_file, None)
Simon Glass07ee48e2020-12-28 20:34:49 -07001528 self._check_strings(self.struct_text, stdout.getvalue())
Simon Glass961c1ce2018-07-06 10:27:35 -06001529
Simon Glassc3a310a82020-12-28 20:34:51 -07001530 def test_multi_to_file(self):
1531 """Test output of multiple pieces to a single file"""
1532 dtb_file = get_dtb_file('dtoc_test_simple.dts')
Simon Glass80025522022-01-29 14:14:04 -07001533 output = tools.get_output_filename('output')
Simon Glass4e8e8462020-12-28 20:34:52 -07001534 self.run_test(['all'], dtb_file, output)
Simon Glass80025522022-01-29 14:14:04 -07001535 data = tools.read_file(output, binary=False)
Simon Glassfea2f252021-02-03 06:01:21 -07001536 self._check_strings(
Simon Glass705b84b2021-04-27 08:19:48 +12001537 self.decl_text + self.platdata_text + self.struct_text, data)
Simon Glassc3a310a82020-12-28 20:34:51 -07001538
Simon Glassa60cabd2020-12-28 20:34:47 -07001539 def test_no_command(self):
Simon Glass961c1ce2018-07-06 10:27:35 -06001540 """Test running dtoc without a command"""
Simon Glassa60cabd2020-12-28 20:34:47 -07001541 with self.assertRaises(ValueError) as exc:
Walter Lozanoa324e412020-06-25 01:10:08 -03001542 self.run_test([], '', '')
Simon Glass961c1ce2018-07-06 10:27:35 -06001543 self.assertIn("Please specify a command: struct, platdata",
Simon Glassa60cabd2020-12-28 20:34:47 -07001544 str(exc.exception))
Simon Glass961c1ce2018-07-06 10:27:35 -06001545
Simon Glassa60cabd2020-12-28 20:34:47 -07001546 def test_bad_command(self):
Simon Glass961c1ce2018-07-06 10:27:35 -06001547 """Test running dtoc with an invalid command"""
1548 dtb_file = get_dtb_file('dtoc_test_simple.dts')
Simon Glass80025522022-01-29 14:14:04 -07001549 output = tools.get_output_filename('output')
Simon Glassa60cabd2020-12-28 20:34:47 -07001550 with self.assertRaises(ValueError) as exc:
Simon Glass3fa3bbb2021-02-03 06:01:14 -07001551 self.run_test(['invalid-cmd'], dtb_file, output)
1552 self.assertIn(
Simon Glassdb2b9ca2021-03-25 06:40:51 +13001553 "Unknown command 'invalid-cmd': (use: decl, platdata, struct)",
Simon Glass3fa3bbb2021-02-03 06:01:14 -07001554 str(exc.exception))
Walter Lozanod82062b2020-07-28 19:06:23 -03001555
Simon Glass4e8e8462020-12-28 20:34:52 -07001556 def test_output_conflict(self):
1557 """Test a conflict between and output dirs and output file"""
1558 with self.assertRaises(ValueError) as exc:
Simon Glass3809ad92021-02-03 06:01:12 -07001559 dtb_platdata.run_steps(
1560 ['all'], None, False, 'out', ['cdir'], None, False,
1561 warning_disabled=True, scan=copy_scan())
Simon Glass4e8e8462020-12-28 20:34:52 -07001562 self.assertIn("Must specify either output or output_dirs, not both",
1563 str(exc.exception))
1564
Simon Glass705b84b2021-04-27 08:19:48 +12001565 def check_output_dirs(self, instantiate):
Simon Glass4e8e8462020-12-28 20:34:52 -07001566 # Remove the directory so that files from other tests are not there
Simon Glass80025522022-01-29 14:14:04 -07001567 tools._remove_output_dir()
1568 tools.prepare_output_dir(None)
Simon Glass4e8e8462020-12-28 20:34:52 -07001569
1570 # This should create the .dts and .dtb in the output directory
1571 dtb_file = get_dtb_file('dtoc_test_simple.dts')
Simon Glass80025522022-01-29 14:14:04 -07001572 outdir = tools.get_output_dir()
Simon Glass4e8e8462020-12-28 20:34:52 -07001573 fnames = glob.glob(outdir + '/*')
1574 self.assertEqual(2, len(fnames))
1575
Simon Glass3809ad92021-02-03 06:01:12 -07001576 dtb_platdata.run_steps(
Simon Glass705b84b2021-04-27 08:19:48 +12001577 ['all'], dtb_file, False, None, [outdir], None, instantiate,
Simon Glass3809ad92021-02-03 06:01:12 -07001578 warning_disabled=True, scan=copy_scan())
Simon Glass4e8e8462020-12-28 20:34:52 -07001579 fnames = glob.glob(outdir + '/*')
Simon Glass705b84b2021-04-27 08:19:48 +12001580 return fnames
1581
1582 def test_output_dirs(self):
1583 """Test outputting files to a directory"""
1584 fnames = self.check_output_dirs(False)
1585 self.assertEqual(5, len(fnames))
Simon Glass4e8e8462020-12-28 20:34:52 -07001586
1587 leafs = set(os.path.basename(fname) for fname in fnames)
1588 self.assertEqual(
Simon Glass3fa3bbb2021-02-03 06:01:14 -07001589 {'dt-structs-gen.h', 'source.dts', 'dt-plat.c', 'source.dtb',
Simon Glass705b84b2021-04-27 08:19:48 +12001590 'dt-decl.h'},
1591 leafs)
1592
1593 def test_output_dirs_inst(self):
1594 """Test outputting files to a directory with instantiation"""
1595 fnames = self.check_output_dirs(True)
1596 self.assertEqual(6, len(fnames))
1597
1598 leafs = set(os.path.basename(fname) for fname in fnames)
1599 self.assertEqual(
1600 {'dt-structs-gen.h', 'source.dts', 'source.dtb',
Simon Glassfea2f252021-02-03 06:01:21 -07001601 'dt-uclass.c', 'dt-decl.h', 'dt-device.c'},
Simon Glass4e8e8462020-12-28 20:34:52 -07001602 leafs)
Simon Glass047a4802021-02-03 06:01:00 -07001603
1604 def setup_process_test(self):
1605 """Set up a test of process_nodes()
1606
1607 This uses saved_scan but returns a deep copy of it, so it is safe to
1608 modify it in these tests
1609
1610 Returns:
1611 tuple:
1612 DtbPlatdata: object to test
1613 Scanner: scanner to use
1614 """
1615 dtb_file = get_dtb_file('dtoc_test_simple.dts')
Simon Glass80025522022-01-29 14:14:04 -07001616 output = tools.get_output_filename('output')
Simon Glass047a4802021-02-03 06:01:00 -07001617
1618 # Take a copy before messing with it
Simon Glassc14fd0c2021-02-03 06:01:11 -07001619 scan = copy_scan()
Simon Glass047a4802021-02-03 06:01:00 -07001620 plat = dtb_platdata.DtbPlatdata(scan, dtb_file, False)
1621 plat.scan_dtb()
Simon Glassc14fd0c2021-02-03 06:01:11 -07001622 plat.scan_tree(False)
Simon Glass047a4802021-02-03 06:01:00 -07001623 plat.prepare_nodes()
1624 return plat, scan
1625
1626 def test_process_nodes(self):
1627 """Test processing nodes to add various info"""
1628 plat, scan = self.setup_process_test()
1629 plat.process_nodes(True)
1630
1631 i2c_node = plat._fdt.GetNode('/i2c@0')
1632 pmic_node = plat._fdt.GetNode('/i2c@0/pmic@9')
1633 pmic = scan._drivers['sandbox_pmic']
1634 i2c = scan._drivers['sandbox_i2c']
1635 self.assertEqual('DM_DEVICE_REF(pmic_at_9)', pmic_node.dev_ref)
1636 self.assertEqual(pmic, pmic_node.driver)
1637 self.assertEqual(i2c_node, pmic_node.parent)
1638 self.assertEqual(i2c, pmic_node.parent_driver)
1639
1640 # The pmic is the only child
1641 self.assertEqual(pmic_node.parent_seq, 0)
1642 self.assertEqual([pmic_node], i2c_node.child_devs)
1643
1644 # Start and end of the list should be the child_head
1645 ref = '&DM_DEVICE_REF(i2c_at_0)->child_head'
1646 self.assertEqual(
1647 {-1: ref, 0: '&DM_DEVICE_REF(pmic_at_9)->sibling_node', 1: ref},
1648 i2c_node.child_refs)
1649
1650 def test_process_nodes_bad_parent(self):
1651 # Pretend that i2c has a parent (the pmic) and delete that driver
1652 plat, scan = self.setup_process_test()
1653
1654 i2c_node = plat._fdt.GetNode('/i2c@0')
1655 pmic_node = plat._fdt.GetNode('/i2c@0/pmic@9')
1656 del scan._drivers['sandbox_pmic']
1657 i2c_node.parent = pmic_node
1658
1659 # Process twice, the second time to generate an exception
1660 plat.process_nodes(False)
1661 with self.assertRaises(ValueError) as exc:
1662 plat.process_nodes(True)
1663 self.assertIn(
1664 "Cannot parse/find parent driver 'sandbox_pmic' for 'sandbox_i2c",
1665 str(exc.exception))
1666
1667 def test_process_nodes_bad_node(self):
1668 plat, scan = self.setup_process_test()
1669
1670 # Now remove the pmic driver
1671 del scan._drivers['sandbox_pmic']
1672
1673 # Process twice, the second time to generate an exception
1674 plat.process_nodes(False)
1675 with self.assertRaises(ValueError) as exc:
1676 plat.process_nodes(True)
1677 self.assertIn("Cannot parse/find driver for 'sandbox_pmic",
1678 str(exc.exception))
Simon Glasseb3c2492021-02-03 06:01:01 -07001679
Simon Glass80d782c42021-02-03 06:01:10 -07001680 def test_process_nodes_bad_uclass(self):
1681 plat, scan = self.setup_process_test()
1682
1683 self.assertIn('UCLASS_I2C', scan._uclass)
1684 del scan._uclass['UCLASS_I2C']
1685 with self.assertRaises(ValueError) as exc:
1686 plat.process_nodes(True)
1687 self.assertIn("Cannot parse/find uclass 'UCLASS_I2C' for driver 'sandbox_i2c'",
1688 str(exc.exception))
1689
Simon Glasseb3c2492021-02-03 06:01:01 -07001690 def test_process_nodes_used(self):
1691 """Test processing nodes to add various info"""
1692 plat, scan = self.setup_process_test()
1693 plat.process_nodes(True)
1694
1695 pmic = scan._drivers['sandbox_pmic']
1696 self.assertTrue(pmic.used)
1697
1698 gpio = scan._drivers['sandbox_gpio']
1699 self.assertFalse(gpio.used)
Simon Glassbe88d2f2021-02-03 06:01:07 -07001700
1701 def test_alias_read(self):
1702 """Test obtaining aliases"""
1703 dtb_file = get_dtb_file('dtoc_test_inst.dts')
Simon Glass80025522022-01-29 14:14:04 -07001704 output = tools.get_output_filename('output')
Simon Glassbe88d2f2021-02-03 06:01:07 -07001705 plat = self.run_test(['struct'], dtb_file, output)
1706
1707 scan = plat._scan
1708 testfdt_node = plat._fdt.GetNode('/some-bus/test')
Simon Glass80d782c42021-02-03 06:01:10 -07001709 test0_node = plat._fdt.GetNode('/some-bus/test0')
Simon Glassbe88d2f2021-02-03 06:01:07 -07001710 self.assertIn('UCLASS_TEST_FDT', scan._uclass)
1711 uc = scan._uclass['UCLASS_TEST_FDT']
Simon Glass80d782c42021-02-03 06:01:10 -07001712 self.assertEqual({1: testfdt_node, 2: test0_node},
1713 uc.alias_num_to_node)
1714 self.assertEqual({'/some-bus/test': 1, '/some-bus/test0': 2},
1715 uc.alias_path_to_num)
Simon Glassbe88d2f2021-02-03 06:01:07 -07001716
1717 # Try adding an alias that doesn't exist
1718 self.assertFalse(scan.add_uclass_alias('fred', 3, None))
1719
1720 # Try adding an alias for a missing node
1721 self.assertIsNone(scan.add_uclass_alias('testfdt', 3, None))
1722
1723 def test_alias_read_bad(self):
1724 """Test invalid alias property name"""
1725 dtb_file = get_dtb_file('dtoc_test_alias_bad.dts')
Simon Glass80025522022-01-29 14:14:04 -07001726 output = tools.get_output_filename('output')
Simon Glassbe88d2f2021-02-03 06:01:07 -07001727 with self.assertRaises(ValueError) as exc:
1728 plat = self.run_test(['struct'], dtb_file, output)
1729 self.assertIn("Cannot decode alias 'i2c4-'", str(exc.exception))
1730
1731 def test_alias_read_bad_path(self):
1732 """Test alias pointing to a non-existent node"""
1733 # This line may produce a warning, so capture it:
1734 # Warning (alias_paths): /aliases:i2c4: aliases property is not a valid
1735 # node (/does/not/exist)
1736 dtb_file = get_dtb_file('dtoc_test_alias_bad_path.dts', True)
1737
Simon Glass80025522022-01-29 14:14:04 -07001738 output = tools.get_output_filename('output')
Simon Glassbe88d2f2021-02-03 06:01:07 -07001739 with self.assertRaises(ValueError) as exc:
1740 plat = self.run_test(['struct'], dtb_file, output)
1741 self.assertIn("Alias 'i2c4' path '/does/not/exist' not found",
1742 str(exc.exception))
1743
1744 def test_alias_read_bad_uclass(self):
1745 """Test alias for a uclass that doesn't exist"""
1746 dtb_file = get_dtb_file('dtoc_test_alias_bad_uc.dts')
Simon Glass80025522022-01-29 14:14:04 -07001747 output = tools.get_output_filename('output')
Simon Glass14d64e32025-04-29 07:21:59 -06001748 with terminal.capture() as (stdout, _):
Simon Glassbe88d2f2021-02-03 06:01:07 -07001749 plat = self.run_test(['struct'], dtb_file, output)
1750 self.assertEqual("Could not find uclass for alias 'other1'",
1751 stdout.getvalue().strip())
Simon Glassdf56e0b2021-02-03 06:01:09 -07001752
1753 def test_sequence(self):
1754 """Test assignment of sequence numnbers"""
1755 dtb_file = get_dtb_file('dtoc_test_inst.dts')
Simon Glass80025522022-01-29 14:14:04 -07001756 output = tools.get_output_filename('output')
Simon Glassdf56e0b2021-02-03 06:01:09 -07001757 plat = self.run_test(['struct'], dtb_file, output)
Simon Glass80d782c42021-02-03 06:01:10 -07001758
1759 scan = plat._scan
1760 testfdt = plat._fdt.GetNode('/some-bus/test')
1761 self.assertEqual(1, testfdt.seq)
1762 i2c = plat._fdt.GetNode('/i2c')
1763
1764 # For now this uclass is not compiled in, so no sequence is assigned
1765 self.assertEqual(4, i2c.seq)
1766 spl = plat._fdt.GetNode('/spl-test')
1767 self.assertEqual(0, spl.seq)
Simon Glassc14fd0c2021-02-03 06:01:11 -07001768
1769 def test_process_root(self):
1770 """Test assignment of sequence numnbers"""
1771 dtb_file = get_dtb_file('dtoc_test_simple.dts')
Simon Glass80025522022-01-29 14:14:04 -07001772 output = tools.get_output_filename('output')
Simon Glassc14fd0c2021-02-03 06:01:11 -07001773
1774 # Take a copy before messing with it
1775 scan = copy_scan()
1776 plat = dtb_platdata.DtbPlatdata(scan, dtb_file, False)
1777 plat.scan_dtb()
1778 root = plat._fdt.GetRoot()
1779
1780 plat.scan_tree(False)
1781 self.assertNotIn(root, plat._valid_nodes)
1782
1783 plat.scan_tree(True)
1784 self.assertIn(root, plat._valid_nodes)
1785 self.assertEqual('root_driver',
1786 scan.get_normalized_compat_name(root)[0])
Simon Glass3fa3bbb2021-02-03 06:01:14 -07001787
1788 def test_simple_inst(self):
1789 """Test output from some simple nodes with instantiate enabled"""
Simon Glassc7b4b832021-02-03 06:01:20 -07001790 dtb_file = get_dtb_file('dtoc_test_inst.dts')
Simon Glass80025522022-01-29 14:14:04 -07001791 output = tools.get_output_filename('output')
Simon Glass3fa3bbb2021-02-03 06:01:14 -07001792
1793 self.run_test(['decl'], dtb_file, output, True)
1794 with open(output) as infile:
1795 data = infile.read()
1796
1797 self._check_strings(self.decl_text_inst, data)
Simon Glassbe749002021-02-03 06:01:15 -07001798
Simon Glassc7b4b832021-02-03 06:01:20 -07001799 self.run_test(['uclass'], dtb_file, output, True)
1800 with open(output) as infile:
1801 data = infile.read()
1802
1803 self._check_strings(UCLASS_HEADER_COMMON + self.uclass_text_inst, data)
1804
Simon Glassfea2f252021-02-03 06:01:21 -07001805 self.run_test(['device'], dtb_file, output, True)
1806 with open(output) as infile:
1807 data = infile.read()
1808
1809 self._check_strings(self.device_text_inst, data)
1810
Simon Glassc7b4b832021-02-03 06:01:20 -07001811 def test_inst_no_hdr(self):
Simon Glassfea2f252021-02-03 06:01:21 -07001812 """Test dealing with a struct tsssshat has no header"""
Simon Glassc7b4b832021-02-03 06:01:20 -07001813 dtb_file = get_dtb_file('dtoc_test_inst.dts')
Simon Glass80025522022-01-29 14:14:04 -07001814 output = tools.get_output_filename('output')
Simon Glassc7b4b832021-02-03 06:01:20 -07001815
1816 # Run it once to set everything up
1817 plat = self.run_test(['decl'], dtb_file, output, True)
1818 scan = plat._scan
1819
1820 # Restart the output file and delete any record of the uclass' struct
1821 plat.setup_output(Ftype.SOURCE, output)
1822 del scan._structs['dm_test_uc_priv']
1823
1824 # Now generate the uclasses, which should provide a warning
Simon Glass14d64e32025-04-29 07:21:59 -06001825 with terminal.capture() as (stdout, _):
Simon Glassc7b4b832021-02-03 06:01:20 -07001826 plat.generate_uclasses()
1827 self.assertEqual(
1828 'Warning: Cannot find header file for struct dm_test_uc_priv',
1829 stdout.getvalue().strip())
Simon Glass4415dc12021-03-26 16:17:27 +13001830
1831 def test_missing_props(self):
1832 """Test detection of a parent node with no properties"""
1833 dtb_file = get_dtb_file('dtoc_test_noprops.dts', capture_stderr=True)
Simon Glass80025522022-01-29 14:14:04 -07001834 output = tools.get_output_filename('output')
Simon Glass4415dc12021-03-26 16:17:27 +13001835 with self.assertRaises(ValueError) as exc:
1836 self.run_test(['struct'], dtb_file, output)
1837 self.assertIn("Parent node '/i2c@0' has no properties - do you need",
1838 str(exc.exception))
1839
1840 def test_single_reg(self):
1841 """Test detection of a parent node with no properties"""
1842 dtb_file = get_dtb_file('dtoc_test_single_reg.dts')
Simon Glass80025522022-01-29 14:14:04 -07001843 output = tools.get_output_filename('output')
Simon Glass4415dc12021-03-26 16:17:27 +13001844 self.run_test(['struct'], dtb_file, output)
Simon Glass9ebd5512021-06-27 17:51:10 -06001845
1846 def test_missing_parent(self):
1847 """Test detection of a parent node with no properties"""
1848 dtb_file = get_dtb_file('dtoc_test_noparent.dts', capture_stderr=True)
Simon Glass80025522022-01-29 14:14:04 -07001849 output = tools.get_output_filename('output')
Simon Glass9ebd5512021-06-27 17:51:10 -06001850 with self.assertRaises(ValueError) as exc:
1851 self.run_test(['device'], dtb_file, output, instantiate=True)
1852 self.assertIn("Node '/i2c@0/spl-test/pmic@9' requires parent node "
1853 "'/i2c@0/spl-test' but it is not in the valid list",
1854 str(exc.exception))