1ec564b47SSimon Glass#!/usr/bin/python 2ec564b47SSimon Glass# 3ec564b47SSimon Glass# Copyright (C) 2016 Google, Inc 4ec564b47SSimon Glass# Written by Simon Glass <sjg@chromium.org> 5ec564b47SSimon Glass# 6ec564b47SSimon Glass# SPDX-License-Identifier: GPL-2.0+ 7ec564b47SSimon Glass# 8ec564b47SSimon Glass 9355c67c3SSimon Glassimport os 10ec564b47SSimon Glassimport struct 11c4c5f9eeSPaul Burtonimport sys 12355c67c3SSimon Glassimport tempfile 13355c67c3SSimon Glass 14355c67c3SSimon Glassimport command 15355c67c3SSimon Glassimport tools 16ec564b47SSimon Glass 17ec564b47SSimon Glassdef fdt32_to_cpu(val): 18ec564b47SSimon Glass """Convert a device tree cell to an integer 19ec564b47SSimon Glass 20ec564b47SSimon Glass Args: 21ec564b47SSimon Glass Value to convert (4-character string representing the cell value) 22ec564b47SSimon Glass 23ec564b47SSimon Glass Return: 24ec564b47SSimon Glass A native-endian integer value 25ec564b47SSimon Glass """ 26c4c5f9eeSPaul Burton if sys.version_info > (3, 0): 27f156b5b5SGeorge McCollister if isinstance(val, bytes): 28f156b5b5SGeorge McCollister val = val.decode('utf-8') 29c4c5f9eeSPaul Burton val = val.encode('raw_unicode_escape') 3020024daeSSimon Glass return struct.unpack('>I', val)[0] 31355c67c3SSimon Glass 32ef2715f4SSimon Glassdef fdt_cells_to_cpu(val, cells): 33ef2715f4SSimon Glass """Convert one or two cells to a long integer 34ef2715f4SSimon Glass 35ef2715f4SSimon Glass Args: 36ef2715f4SSimon Glass Value to convert (array of one or more 4-character strings) 37ef2715f4SSimon Glass 38ef2715f4SSimon Glass Return: 39ef2715f4SSimon Glass A native-endian long value 40ef2715f4SSimon Glass """ 41*a28bfcc3SSimon Glass if not cells: 42*a28bfcc3SSimon Glass return 0 43ef2715f4SSimon Glass out = long(fdt32_to_cpu(val[0])) 44ef2715f4SSimon Glass if cells == 2: 45ef2715f4SSimon Glass out = out << 32 | fdt32_to_cpu(val[1]) 46ef2715f4SSimon Glass return out 47ef2715f4SSimon Glass 48355c67c3SSimon Glassdef EnsureCompiled(fname): 49355c67c3SSimon Glass """Compile an fdt .dts source file into a .dtb binary blob if needed. 50355c67c3SSimon Glass 51355c67c3SSimon Glass Args: 52355c67c3SSimon Glass fname: Filename (if .dts it will be compiled). It not it will be 53355c67c3SSimon Glass left alone 54355c67c3SSimon Glass 55355c67c3SSimon Glass Returns: 56355c67c3SSimon Glass Filename of resulting .dtb file 57355c67c3SSimon Glass """ 58355c67c3SSimon Glass _, ext = os.path.splitext(fname) 59355c67c3SSimon Glass if ext != '.dts': 60355c67c3SSimon Glass return fname 61355c67c3SSimon Glass 62355c67c3SSimon Glass dts_input = tools.GetOutputFilename('source.dts') 63355c67c3SSimon Glass dtb_output = tools.GetOutputFilename('source.dtb') 64355c67c3SSimon Glass 65355c67c3SSimon Glass search_paths = [os.path.join(os.getcwd(), 'include')] 66355c67c3SSimon Glass root, _ = os.path.splitext(fname) 67355c67c3SSimon Glass args = ['-E', '-P', '-x', 'assembler-with-cpp', '-D__ASSEMBLY__'] 68355c67c3SSimon Glass args += ['-Ulinux'] 69355c67c3SSimon Glass for path in search_paths: 70355c67c3SSimon Glass args.extend(['-I', path]) 71355c67c3SSimon Glass args += ['-o', dts_input, fname] 72355c67c3SSimon Glass command.Run('cc', *args) 73355c67c3SSimon Glass 74355c67c3SSimon Glass # If we don't have a directory, put it in the tools tempdir 75355c67c3SSimon Glass search_list = [] 76355c67c3SSimon Glass for path in search_paths: 77355c67c3SSimon Glass search_list.extend(['-i', path]) 78355c67c3SSimon Glass args = ['-I', 'dts', '-o', dtb_output, '-O', 'dtb'] 79355c67c3SSimon Glass args.extend(search_list) 80355c67c3SSimon Glass args.append(dts_input) 81355c67c3SSimon Glass command.Run('dtc', *args) 82355c67c3SSimon Glass return dtb_output 838f224b37SSimon Glass 848f224b37SSimon Glassdef GetInt(node, propname, default=None): 858f224b37SSimon Glass prop = node.props.get(propname) 868f224b37SSimon Glass if not prop: 878f224b37SSimon Glass return default 888f224b37SSimon Glass value = fdt32_to_cpu(prop.value) 898f224b37SSimon Glass if type(value) == type(list): 908f224b37SSimon Glass raise ValueError("Node '%s' property '%' has list value: expecting" 918f224b37SSimon Glass "a single integer" % (node.name, propname)) 928f224b37SSimon Glass return value 938f224b37SSimon Glass 948f224b37SSimon Glassdef GetString(node, propname, default=None): 958f224b37SSimon Glass prop = node.props.get(propname) 968f224b37SSimon Glass if not prop: 978f224b37SSimon Glass return default 988f224b37SSimon Glass value = prop.value 998f224b37SSimon Glass if type(value) == type(list): 1008f224b37SSimon Glass raise ValueError("Node '%s' property '%' has list value: expecting" 1018f224b37SSimon Glass "a single string" % (node.name, propname)) 1028f224b37SSimon Glass return value 1038f224b37SSimon Glass 1048f224b37SSimon Glassdef GetBool(node, propname, default=False): 1058f224b37SSimon Glass if propname in node.props: 1068f224b37SSimon Glass return True 1078f224b37SSimon Glass return default 108