# ***** BEGIN GPL LICENSE BLOCK ***** # # This program 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 program 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 program; if not, write to the Free Software Foundation, # Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. # # ***** END GPL LICENCE BLOCK ***** # # (c) 2009, At Mind B.V. - Jeroen Bakker # (c) 2014, Blender Foundation - Campbell Barton # (c) 2018, Blender Foundation - Sybren A. Stüvel """Read-write utility functions.""" import struct import typing class EndianIO: # TODO(Sybren): note as UCHAR: struct.Struct = None and move actual structs to LittleEndianTypes UCHAR = struct.Struct(b' int: """Write a (truncated) string as UTF-8. The string will always be written 0-terminated. :param fileobj: the file to write to. :param astring: the string to write. :param fieldlen: the field length in bytes. :returns: the number of bytes written. """ assert isinstance(astring, str) encoded = astring.encode('utf-8') # Take into account we also need space for a trailing 0-byte. maxlen = fieldlen - 1 if len(encoded) >= maxlen: encoded = encoded[:maxlen] # Keep stripping off the last byte until the string # is valid UTF-8 again. while True: try: encoded.decode('utf8') except UnicodeDecodeError: encoded = encoded[:-1] else: break return fileobj.write(encoded + b'\0') @classmethod def write_bytes(cls, fileobj: typing.IO[bytes], data: bytes, fieldlen: int) -> int: """Write (truncated) bytes. When len(data) < fieldlen, a terminating b'\0' will be appended. :returns: the number of bytes written. """ assert isinstance(data, (bytes, bytearray)) if len(data) >= fieldlen: to_write = data[0:fieldlen] else: to_write = data + b'\0' return fileobj.write(to_write) @classmethod def read_bytes0(cls, fileobj, length): data = fileobj.read(length) return cls.read_data0(data) @classmethod def read_data0_offset(cls, data, offset): add = data.find(b'\0', offset) - offset return data[offset:offset + add] @classmethod def read_data0(cls, data): add = data.find(b'\0') if add < 0: return data return data[:add] class LittleEndianTypes(EndianIO): pass class BigEndianTypes(LittleEndianTypes): UCHAR = struct.Struct(b'>B') USHORT = struct.Struct(b'>H') USHORT2 = struct.Struct(b'>HH') # two shorts in a row SSHORT = struct.Struct(b'>h') UINT = struct.Struct(b'>I') SINT = struct.Struct(b'>i') FLOAT = struct.Struct(b'>f') ULONG = struct.Struct(b'>Q')