Add Python bindings for a bare-bones set of libfdt functions. These allow navigating the tree and reading node names and properties. Signed-off-by: Simon Glass <sjg@xxxxxxxxxxxx> --- Changes in v4: - Make the library less pythonic to avoid a shaky illusion - Drop classes for Node and Prop, along with associated methods - Include libfdt.h instead of repeating it - Add support for fdt_getprop() - Bring in all libfdt functions (but Python support is missing for many) - Add full comments for Python methods Changes in v3: - Make the library more pythonic - Add classes for Node and Prop along with methods - Add an exception class - Use Python to generate exeptions instead of SWIG Changes in v2: - Add exceptions when functions return an error - Correct Python naming to following PEP8 - Use a class to encapsulate the various methods - Include fdt.h instead of redefining struct fdt_property - Use bytearray to avoid the SWIG warning 454 - Add comments pylibfdt/.gitignore | 3 + pylibfdt/Makefile.pylibfdt | 18 ++ pylibfdt/libfdt.swig | 565 +++++++++++++++++++++++++++++++++++++++++++++ pylibfdt/setup.py | 34 +++ 4 files changed, 620 insertions(+) create mode 100644 pylibfdt/.gitignore create mode 100644 pylibfdt/Makefile.pylibfdt create mode 100644 pylibfdt/libfdt.swig create mode 100644 pylibfdt/setup.py diff --git a/pylibfdt/.gitignore b/pylibfdt/.gitignore new file mode 100644 index 0000000..5e8c5e3 --- /dev/null +++ b/pylibfdt/.gitignore @@ -0,0 +1,3 @@ +libfdt.py +libfdt.pyc +libfdt_wrap.c diff --git a/pylibfdt/Makefile.pylibfdt b/pylibfdt/Makefile.pylibfdt new file mode 100644 index 0000000..fa74dd2 --- /dev/null +++ b/pylibfdt/Makefile.pylibfdt @@ -0,0 +1,18 @@ +# Makefile.pylibfdt +# + +PYLIBFDT_srcs = $(addprefix $(LIBFDT_srcdir)/,$(LIBFDT_SRCS)) +WRAP = $(PYLIBFDT_objdir)/libfdt_wrap.c +PYMODULE = $(PYLIBFDT_objdir)/_libfdt.so + +$(PYMODULE): $(PYLIBFDT_srcs) $(WRAP) + @$(VECHO) PYMOD $@ + python $(PYLIBFDT_objdir)/setup.py "$(CPPFLAGS)" $^ + mv _libfdt.so $(PYMODULE) + +$(WRAP): $(PYLIBFDT_srcdir)/libfdt.swig + @$(VECHO) SWIG $@ + swig -python -o $@ $< + +PYLIBFDT_cleanfiles = libfdt_wrap.c libfdt.py libfdt.pyc +PYLIBFDT_CLEANFILES = $(addprefix $(PYLIBFDT_objdir)/,$(PYLIBFDT_cleanfiles)) diff --git a/pylibfdt/libfdt.swig b/pylibfdt/libfdt.swig new file mode 100644 index 0000000..ac478d5 --- /dev/null +++ b/pylibfdt/libfdt.swig @@ -0,0 +1,565 @@ +/* + * pylibfdt - Flat Device Tree manipulation in Python + * Copyright (C) 2017 Google, Inc. + * Written by Simon Glass <sjg@xxxxxxxxxxxx> + * + * libfdt is dual licensed: you can use it either under the terms of + * the GPL, or the BSD license, at your option. + * + * a) This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or (at your option) any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this library; if not, write to the Free + * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, + * MA 02110-1301 USA + * + * Alternatively, + * + * b) Redistribution and use in source and binary forms, with or + * without modification, are permitted provided that the following + * conditions are met: + * + * 1. Redistributions of source code must retain the above + * copyright notice, this list of conditions and the following + * disclaimer. + * 2. Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials + * provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND + * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, + * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR + * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT + * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR + * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, + * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +%module libfdt + +%{ +#define SWIG_FILE_WITH_INIT +#include "libfdt.h" +%} + +%pythoncode %{ + +import struct + +# Error codes, corresponding to FDT_ERR_... in libfdt.h +(NOTFOUND, + EXISTS, + NOSPACE, + BADOFFSET, + BADPATH, + BADPHANDLE, + BADSTATE, + TRUNCATED, + BADMAGIC, + BADVERSION, + BADSTRUCTURE, + BADLAYOUT, + INTERNAL, + BADNCELLS, + BADVALUE, + BADOVERLAY) = range(1, 17) + +class FdtException(Exception): + """An exception caused by an error such as one of the codes above""" + def __init__(self, err): + self.err = err + + def __str__(self): + return 'pylibfdt error %d: %s' % (self.err, fdt_strerror(self.err)) + +def fdt32_to_cpu(val): + """Convert a device-tree cell value into a native integer""" + return struct.unpack("=I", struct.pack(">I", val))[0] + +def data(prop): + """Extract the data from a property + + This is an internal function only. + + Args: + prop: Property structure, as returned by get_property_by_offset() + + Returns: + The property data as a bytearray + """ + buf = bytearray(fdt32_to_cpu(prop.len)) + pylibfdt_copy_data(buf, prop) + return buf + +def strerror(fdt_err): + """Get the string for an error number + + Args: + fdt_err: Error number (-ve) + + Returns: + String containing the associated error + """ + return fdt_strerror(fdt_err) + +def check_err(val, quiet=False): + """Raise an error if the return value is -ve + + This is used to check for errors returned by libfdt C functions. + + Args: + val: Return value from a libfdt function + quiet: True to ignore the NOTFOUND error, False to raise on all errors + + Returns: + val if val >= 0 + + Raises + FdtException if val < 0 + """ + if val < 0: + if not quiet or val != -NOTFOUND: + raise FdtException(val) + return val + +def check_err_null(val, quiet=False): + """Raise an error if the return value is NULL + + This is used to check for a NULL return value from certain libfdt C + functions + + Args: + val: Return value from a libfdt function + quiet: True to ignore the NOTFOUND error, False to raise on all errors + + Returns: + val if val is a list, None if not + + Raises + FdtException if quiet is False and val indicates an error was + reported. If quiet if True then an FdtException is raised only if + the error is something other than -NOTFOUND. + """ + # Normally a tuple is returned which contains the data and its length. + # If we get just an integer error code, it means the function failed. + if not isinstance(val, list): + if not quiet or val != -NOTFOUND: + raise FdtException(val) + return None, + return val + +class Fdt: + """Device tree class, supporting all operations + + The Fdt object is created is created from a device tree binary file, + e.g. with something like: + + fdt = Fdt(open("filename.dtb").read()) + + Operations can then be performed using the methods in this class. Each + method xxx(args...) corresponds to a libfdt function fdt_xxx(fdt, args...). + + Almost all methods raise an FdtException if an error occurs. The + following does not: + + string() - since it has no error checking + + To avoid this behaviour a 'quiet' version is provided for some functions. + This behaves as for the normal version except that it will not raise + an exception in the case of an FDT_ERR_NOTFOUND error: it will simply + return the -NOTFOUND error code or None. + """ + def __init__(self, data): + self._fdt = bytearray(data) + + def string(self, offset): + """Get a string given its offset + + This is an internal function. + + Args: + offset: FDT offset in big-endian format + + Returns: + string value at that offset + """ + return fdt_string(self._fdt, fdt32_to_cpu(offset)) + + def path_offset(self, path): + """Get the offset for a given path + + Args: + path: Path to the required node, e.g. '/node@3/subnode@1' + + Returns: + Node offset + + Raises + FdtException if the path is not valid + """ + return check_err(fdt_path_offset(self._fdt, path)) + + def path_offset_quiet(self, path): + """Get the offset for a given path + + Args: + path: Path to the required node, e.g. '/node@3/subnode@1' + + Returns: + Node offset, or -NOTFOUND if the path is not value + + Raises + FdtException if any error occurs other than NOTFOUND + """ + return check_err(fdt_path_offset(self._fdt, path), True) + + def first_property_offset(self, nodeoffset): + """Get the offset of the first property in a node offset + + Args: + nodeoffset: Offset to the node to check + + Returns: + Offset of the first property + + Raises + FdtException if the associated node has no properties, or some + other error occurred + """ + return check_err(fdt_first_property_offset(self._fdt, nodeoffset)) + + def first_property_offset_quiet(self, nodeoffset): + """Get the offset of the first property in a node offset + + Args: + nodeoffset: Offset to the node to check + + Returns: + Offset of the first property, or -NOTFOUND if the node has no + properties + + Raises + FdtException if any other error occurs + """ + return check_err(fdt_first_property_offset(self._fdt, nodeoffset), True) + + def next_property_offset(self, prop_offset): + """Get the next property in a node + + Args: + prop_offset: Offset of the previous property + + Returns: + Offset of the next property + + Raises: + FdtException if the associated node has no more properties, or + some other error occurred + """ + return check_err(fdt_next_property_offset(self._fdt, prop_offset)) + + def next_property_offset_quiet(self, prop_offset): + """Get the next property in a node + + Args: + prop_offset: Offset of the previous property + + Returns: + Offset ot the next property, or -NOTFOUND if there are no more + properties + + Raises: + FdtException if any other error occurs + """ + return check_err(fdt_next_property_offset(self._fdt, prop_offset), True) + + def get_name(self, nodeoffset): + """Get the name of a node + + Args: + nodeoffset: Offset of node to check + + Returns: + Node name + + Raises: + FdtException on error (e.g. nodeoffset is invalid) + """ + return check_err_null(fdt_get_name(self._fdt, nodeoffset))[0] + + def get_property_by_offset(self, prop_offset): + """Obtains a property that can be examined + + TODO(sjg@xxxxxxxxxxxx): Consider returning a property object instead, + with the data extracted from the Fdt. + + Args: + prop_offset: Offset of property (e.g. from first_property_offset()) + + Returns: + Property object with members: + tag: Big-endian device tree tag value + len: Big-endian property length + nameoff: Big-endian string offset for use with string() + + Use data() on the return value to obtain the property value. + + Raises: + FdtException on error (e.g. invalid prop_offset or device + tree format) + """ + return check_err_null(fdt_get_property_by_offset(self._fdt, + prop_offset))[0] + + def first_subnode(self, nodeoffset): + """Find the first subnode of a parent node + + Args: + nodeoffset: Node offset of parent node + + Returns: + The offset of the first subnode, if any + + Raises: + FdtException if no subnode found or other error occurs + """ + return check_err(fdt_first_subnode(self._fdt, nodeoffset)) + + def first_subnode_quiet(self, nodeoffset): + """Find the first subnode of a node + + Args: + nodeoffset: Node offset of parent node + + Returns: + The offset of the first subnode, or -NOTFOUND if none + + Raises: + FdtException on error (e.g. invalid nodeoffset) + """ + return check_err(fdt_first_subnode(self._fdt, nodeoffset), True) + + def next_subnode(self, nodeoffset): + """Find the next subnode + + Args: + nodeoffset: Node offset of previous subnode + + Returns: + The offset of the next subnode, if any + + Raises: + FdtException if no more subnode found or other error occurs + """ + return check_err(fdt_next_subnode(self._fdt, nodeoffset)) + + def next_subnode_quiet(self, nodeoffset): + """Find the next subnode + + Args: + nodeoffset: Node offset of previous subnode + + Returns: + The offset of the next subnode, or -NOTFOUND if none + + Raises: + FdtException on error (e.g. invalid nodeoffset) + """ + return check_err(fdt_next_subnode(self._fdt, nodeoffset), True) + + def totalsize(self): + """Return the total size of the device tree + + Returns: + Total tree size in bytes + + Raises: + FdtException if any error occurs + """ + return check_err(fdt_totalsize(self._fdt)) + + def off_dt_struct(self): + """Return the start of the device tree struct area + + Returns: + Start offset of struct area + + Raises: + FdtException if any error occurs + """ + return check_err(fdt_off_dt_struct(self._fdt)) + + def pack(self): + """Pack the device tree to remove unused space + + This adjusts the tree in place. + + Raises: + FdtException if any error occurs + """ + return check_err(fdt_pack(self._fdt)) + + def delprop(self, nodeoffset, prop_name): + """Delete a property from a node + + Args: + nodeoffset: Node offset containing property to delete + prop_name: Name of property to delete + + Raises: + FdtError if the property does not exist, or another error occurs + """ + return check_err(fdt_delprop(self._fdt, nodeoffset, prop_name)) + + def getprop(self, nodeoffset, prop_name): + """Get a property from a node + + Args: + nodeoffset: Node offset containing property to get + prop_name: Name of property to get + + Returns: + Value of property as a string + + Raises: + FdtError if any error occurs (e.g. the property is not found) + """ + return check_err_null(fdt_getprop(self._fdt, nodeoffset, prop_name))[0] + + def getprop_quiet(self, nodeoffset, prop_name): + """Get a property from a node + + Args: + nodeoffset: Node offset containing property to get + prop_name: Name of property to get + + Returns: + Value of property as a string, or None if not found + + Raises: + FdtError if an error occurs (e.g. nodeoffset is invalid) + """ + return check_err_null(fdt_getprop(self._fdt, nodeoffset, prop_name), + True)[0] +%} + +%rename(fdt_property) fdt_property_func; + +typedef int fdt32_t; + +%include "libfdt/fdt.h" + +%include "typemaps.i" + +/* + * Unfortunately the defintiion of pybuffer_mutable_binary() in my Python + * version appears to be broken: + * pylibfdt/libfdt_wrap.c: In function ‘_wrap_pylibfdt_copy_data’: + * pylibfdt/libfdt_wrap.c:3603:22: error: ‘size’ undeclared (first use in this + * function) + * arg2 = (size_t) (size/sizeof(char)); + * + * This version works correctly. + */ +%define %mypybuffer_mutable_binary(TYPEMAP, SIZE) +%typemap(in) (TYPEMAP, SIZE)(int res, Py_ssize_t size = 0, void *buf = 0) +{ + res = PyObject_AsWriteBuffer($input, &buf, &size); + if (res < 0) { + PyErr_Clear(); + %argument_fail(res, "(TYPEMAP, SIZE)", $symname, $argnum); + } + $1 = ($1_ltype)buf; + $2 = ($2_ltype)(size1 / sizeof($*1_type)); +} +%enddef + +/* This is used to copy property data into a bytearray */ +%mypybuffer_mutable_binary(char *str, size_t size); +void pylibfdt_copy_data(char *str, size_t size, + const struct fdt_property *prop); + +/* Most functions don't change the device tree, so use a const void * */ +%typemap(in) (const void *) { + if (!PyByteArray_Check($input)) { + SWIG_exception_fail(SWIG_TypeError, "in method '" "$symname" + "', argument " "$argnum"" of type '" "$type""'"); + } + $1 = (void *)PyByteArray_AsString($input); +} + +/* Some functions do change the device tree, so use void * */ +%typemap(in) (void *) { + if (!PyByteArray_Check($input)) { + SWIG_exception_fail(SWIG_TypeError, "in method '" "$symname" + "', argument " "$argnum"" of type '" "$type""'"); + } + $1 = PyByteArray_AsString($input); +} + +%inline %{ + +/** + * pylibfdt_copy_data() - Copy data from a property to the given buffer + * + * This is used by the data() function to place the contents of a property + * into a bytearray. + * + * @buf: Destination pointer (typically the start of the bytearray) + * @size: Number of bytes to copy (size of bytearray) + * @prop: Property to copy + */ +void pylibfdt_copy_data(char *buf, size_t size, const struct fdt_property *prop) +{ + memcpy(buf, prop + 1, size); +} + +%} + +%apply int *OUTPUT { int *lenp }; + +/* Handle a few special cases which conflict with the typemap below */ +const void *fdt_offset_ptr(const void *fdt, int offset, unsigned int checklen); +const void *fdt_getprop_namelen(const void *fdt, int nodeoffset, + const char *name, int namelen, int *lenp); + +/* typemap used for fdt_getprop() */ +%typemap(out) (const void *) { + if (!$1) + $result = Py_None; + else + /* TODO(sjg@xxxxxxxxxxxx): Can we avoid the 'arg4'? */ + $result = Py_BuildValue("s#", $1, *arg4); +} + +/* We have both struct fdt_property and a function fdt_property() */ +%warnfilter(302) fdt_property; + +%include <../libfdt/libfdt.h> + +/* These are macros in the header so have to be redefined here */ +int fdt_magic(const void *fdt); +int fdt_totalsize(const void *fdt); +int fdt_off_dt_struct(const void *fdt); +int fdt_off_dt_strings(const void *fdt); +int fdt_off_mem_rsvmap(const void *fdt); +int fdt_version(const void *fdt); +int fdt_last_comp_version(const void *fdt); +int fdt_boot_cpuid_phys(const void *fdt); +int fdt_size_dt_strings(const void *fdt); +int fdt_size_dt_struct(const void *fdt); diff --git a/pylibfdt/setup.py b/pylibfdt/setup.py new file mode 100644 index 0000000..8f8618e --- /dev/null +++ b/pylibfdt/setup.py @@ -0,0 +1,34 @@ +#!/usr/bin/env python + +""" +setup.py file for SWIG libfdt +""" + +from distutils.core import setup, Extension +import os +import sys + +progname = sys.argv[0] +cflags = sys.argv[1] +files = sys.argv[2:] + +if cflags: + cflags = [flag for flag in cflags.split(' ') if flag] +else: + cflags = None + +libfdt_module = Extension( + '_libfdt', + sources = files, + extra_compile_args = cflags +) + +sys.argv = [progname, '--quiet', 'build_ext', '--inplace'] + +setup (name = 'libfdt', + version = '0.1', + author = "SWIG Docs", + description = """Simple swig libfdt from docs""", + ext_modules = [libfdt_module], + py_modules = ["libfdt"], + ) -- 2.11.0.483.g087da7b7c-goog -- To unsubscribe from this list: send the line "unsubscribe devicetree-compiler" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html