X-Git-Url: http://sigrok.org/gitweb/?a=blobdiff_plain;f=decoders%2Fcommon%2Fsrdhelper%2Fmod.py;h=b56cce6df3dbb2385136f462ac7af45fc11e706c;hb=3f77dc2aae971c4ba97ad81c851c63b681074410;hp=3c495b430458d0078c35662b2a4197ff62c3ec76;hpb=392a5d1ef4bb356be248589dad28d0bd83411c49;p=libsigrokdecode.git diff --git a/decoders/common/srdhelper/mod.py b/decoders/common/srdhelper/mod.py index 3c495b4..b56cce6 100644 --- a/decoders/common/srdhelper/mod.py +++ b/decoders/common/srdhelper/mod.py @@ -1,7 +1,7 @@ ## ## This file is part of the libsigrokdecode project. ## -## Copyright (C) 2012-2014 Uwe Hermann +## Copyright (C) 2012-2020 Uwe Hermann ## ## 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 @@ -17,6 +17,10 @@ ## along with this program; if not, see . ## +from enum import Enum, IntEnum, unique +from itertools import chain +import re + # Return the specified BCD number (max. 8 bits) as integer. def bcd2int(b): return (b & 0x0f) + ((b >> 4) * 10) @@ -25,10 +29,21 @@ def bin2int(s: str): return int('0b' + s, 2) def bitpack(bits): - res = 0 - for i, b in enumerate(bits): - res |= b << i - return res + return sum([b << i for i, b in enumerate(bits)]) + +def bitpack_lsb(bits, idx=None): + '''Conversion from LSB first bit sequence to integer.''' + if idx is not None: + bits = [b[idx] for b in bits] + return bitpack(bits) + +def bitpack_msb(bits, idx=None): + '''Conversion from MSB first bit sequence to integer.''' + bits = bits[:] + if idx is not None: + bits = [b[idx] for b in bits] + bits.reverse() + return bitpack(bits) def bitunpack(num, minbits=0): res = [] @@ -37,3 +52,47 @@ def bitunpack(num, minbits=0): num >>= 1 minbits -= 1 return tuple(res) + +@unique +class SrdStrEnum(Enum): + @classmethod + def from_list(cls, name, l): + # Keys are limited/converted to [A-Z0-9_], values can be any string. + items = [(re.sub('[^A-Z0-9_]', '_', l[i]), l[i]) for i in range(len(l))] + return cls(name, items) + + @classmethod + def from_str(cls, name, s): + return cls.from_list(name, s.split()) + +@unique +class SrdIntEnum(IntEnum): + @classmethod + def _prefix(cls, p): + return tuple([a.value for a in cls if a.name.startswith(p)]) + + @classmethod + def prefixes(cls, prefix_list): + if isinstance(prefix_list, str): + prefix_list = prefix_list.split() + return tuple(chain(*[cls._prefix(p) for p in prefix_list])) + + @classmethod + def _suffix(cls, s): + return tuple([a.value for a in cls if a.name.endswith(s)]) + + @classmethod + def suffixes(cls, suffix_list): + if isinstance(suffix_list, str): + suffix_list = suffix_list.split() + return tuple(chain(*[cls._suffix(s) for s in suffix_list])) + + @classmethod + def from_list(cls, name, l): + # Manually construct (Python 3.4 is missing the 'start' argument). + # Python defaults to start=1, but we want start=0. + return cls(name, [(l[i], i) for i in range(len(l))]) + + @classmethod + def from_str(cls, name, s): + return cls.from_list(name, s.split())