262 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			262 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Python
		
	
	
	
| #
 | |
| # Some common functions and types.
 | |
| #
 | |
| 
 | |
| from __future__ import absolute_import, division, print_function, unicode_literals
 | |
| 
 | |
| from binascii import hexlify as _hexlify
 | |
| from struct import pack, unpack
 | |
| try:
 | |
| 	unicode
 | |
| 	# if Python2, unicode_literals will mess our first (un)pack() argument
 | |
| 	_pack_str = pack
 | |
| 	_unpack_str = unpack
 | |
| 	pack = lambda x, *args: _pack_str(str(x), *args)
 | |
| 	unpack = lambda x, *args: _unpack_str(str(x), *args)
 | |
| except:
 | |
| 	pass
 | |
| 
 | |
| try:
 | |
| 	unicode
 | |
| 	# this is certanly Python 2
 | |
| 	is_string = lambda d: isinstance(d, unicode) or isinstance(d, str)
 | |
| 	# no easy way to distinguish between b'' and '' :(
 | |
| 						# or (isinstance(d, str) \
 | |
| 						# 	and not any((chr(k) in d for k in range(0x00, 0x1F))) \
 | |
| 						# 	and not any((chr(k) in d for k in range(0x80, 0xFF))) \
 | |
| 						# 	)
 | |
| except:
 | |
| 	# this is certanly Python 3
 | |
| 	# In Py3, unicode and str are equal (the unicode object does not exist)
 | |
| 	is_string = lambda d: isinstance(d, str)
 | |
| 
 | |
| 
 | |
| class NamedInt(int):
 | |
| 	"""An reqular Python integer with an attached name.
 | |
| 
 | |
| 	Caution: comparison with strings will also match this NamedInt's name
 | |
| 	(case-insensitive)."""
 | |
| 
 | |
| 	def __new__(cls, value, name):
 | |
| 		assert is_string(name)
 | |
| 		obj = int.__new__(cls, value)
 | |
| 		obj.name = str(name)
 | |
| 		return obj
 | |
| 
 | |
| 	def bytes(self, count=2):
 | |
| 		return int2bytes(self, count)
 | |
| 
 | |
| 	def __eq__(self, other):
 | |
| 		if isinstance(other, NamedInt):
 | |
| 			return int(self) == int(other) and self.name == other.name
 | |
| 		if isinstance(other, int):
 | |
| 			return int(self) == int(other)
 | |
| 		if is_string(other):
 | |
| 			return self.name.lower() == other.lower()
 | |
| 		# this should catch comparisons with bytes in Py3
 | |
| 		if other is not None:
 | |
| 			raise TypeError("Unsupported type " + str(type(other)))
 | |
| 
 | |
| 	def __ne__(self, other):
 | |
| 		return not self.__eq__(other)
 | |
| 
 | |
| 	def __hash__(self):
 | |
| 		return int(self)
 | |
| 
 | |
| 	def __str__(self):
 | |
| 		return self.name
 | |
| 	__unicode__ = __str__
 | |
| 
 | |
| 	def __repr__(self):
 | |
| 		return 'NamedInt(%d, %r)' % (int(self), self.name)
 | |
| 
 | |
| 
 | |
| class NamedInts(object):
 | |
| 	"""An ordered set of NamedInt values.
 | |
| 
 | |
| 	Indexing can be made by int or string, and will return the corresponding
 | |
| 	NamedInt if it exists in this set, or `None`.
 | |
| 
 | |
| 	Extracting slices will return all present NamedInts in the given interval
 | |
| 	(extended slices are not supported).
 | |
| 
 | |
| 	Assigning a string to an indexed int will create a new NamedInt in this set;
 | |
| 	if the value already exists in the set (int or string), ValueError will be
 | |
| 	raised.
 | |
| 	"""
 | |
| 	__slots__ = ['__dict__', '_values', '_indexed', '_fallback']
 | |
| 
 | |
| 	def __init__(self, **kwargs):
 | |
| 		def _readable_name(n):
 | |
| 			if not is_string(n):
 | |
| 				raise TypeError("expected (unicode) string, got " + str(type(n)))
 | |
| 			return n.replace('__', '/').replace('_', ' ')
 | |
| 
 | |
| 		# print (repr(kwargs))
 | |
| 		values = {k: NamedInt(v, _readable_name(k)) for (k, v) in kwargs.items()}
 | |
| 		self.__dict__ = values
 | |
| 		self._values = sorted(list(values.values()))
 | |
| 		self._indexed = {int(v): v for v in self._values}
 | |
| 		# assert len(values) == len(self._indexed), "(%d) %r\n=> (%d) %r" % (len(values), values, len(self._indexed), self._indexed)
 | |
| 		self._fallback = None
 | |
| 
 | |
| 	@classmethod
 | |
| 	def range(cls, from_value, to_value, name_generator=lambda x: str(x), step=1):
 | |
| 		values = {name_generator(x): x for x in range(from_value, to_value + 1, step)}
 | |
| 		return NamedInts(**values)
 | |
| 
 | |
| 	def flag_names(self, value):
 | |
| 		unknown_bits = value
 | |
| 		for k in self._indexed:
 | |
| 			assert bin(k).count('1') == 1
 | |
| 			if k & value == k:
 | |
| 				unknown_bits &= ~k
 | |
| 				yield str(self._indexed[k])
 | |
| 
 | |
| 		if unknown_bits:
 | |
| 			yield 'unknown:%06X' % unknown_bits
 | |
| 
 | |
| 	def __getitem__(self, index):
 | |
| 		if isinstance(index, int):
 | |
| 			if index in self._indexed:
 | |
| 				return self._indexed[int(index)]
 | |
| 			if self._fallback and type(index) == int:
 | |
| 				value = NamedInt(index, self._fallback(index))
 | |
| 				self._indexed[index] = value
 | |
| 				self._values = sorted(self._values + [value])
 | |
| 				return value
 | |
| 
 | |
| 		elif is_string(index):
 | |
| 			if index in self.__dict__:
 | |
| 				return self.__dict__[index]
 | |
| 
 | |
| 		elif isinstance(index, slice):
 | |
| 			if index.start is None and index.stop is None:
 | |
| 				return self._values[:]
 | |
| 
 | |
| 			v_start = int(self._values[0]) if index.start is None else int(index.start)
 | |
| 			v_stop = (self._values[-1] + 1) if index.stop is None else int(index.stop)
 | |
| 
 | |
| 			if v_start > v_stop or v_start > self._values[-1] or v_stop <= self._values[0]:
 | |
| 				return []
 | |
| 
 | |
| 			if v_start <= self._values[0] and v_stop > self._values[-1]:
 | |
| 				return self._values[:]
 | |
| 
 | |
| 			start_index = 0
 | |
| 			stop_index = len(self._values)
 | |
| 			for i, value in enumerate(self._values):
 | |
| 				if value < v_start:
 | |
| 					start_index = i + 1
 | |
| 				elif index.stop is None:
 | |
| 					break
 | |
| 				if value >= v_stop:
 | |
| 					stop_index = i
 | |
| 					break
 | |
| 
 | |
| 			return self._values[start_index:stop_index]
 | |
| 
 | |
| 	def __setitem__(self, index, name):
 | |
| 		assert isinstance(index, int), type(index)
 | |
| 		if isinstance(name, NamedInt):
 | |
| 			assert int(index) == int(name), repr(index) + ' ' + repr(name)
 | |
| 			value = name
 | |
| 		elif is_string(name):
 | |
| 			value = NamedInt(index, name)
 | |
| 		else:
 | |
| 			raise TypeError('name must be a string')
 | |
| 
 | |
| 		if str(value) in self.__dict__:
 | |
| 			raise ValueError('%s (%d) already known' % (value, int(value)))
 | |
| 		if int(value) in self._indexed:
 | |
| 			raise ValueError('%d (%s) already known' % (int(value), value))
 | |
| 
 | |
| 		self._values = sorted(self._values + [value])
 | |
| 		self.__dict__[str(value)] = value
 | |
| 		self._indexed[int(value)] = value
 | |
| 
 | |
| 	def __contains__(self, value):
 | |
| 		if isinstance(value, int):
 | |
| 			return value in self._indexed
 | |
| 		elif is_string(value):
 | |
| 			return value in self.__dict__
 | |
| 
 | |
| 	def __iter__(self):
 | |
| 		for v in self._values:
 | |
| 			yield v
 | |
| 
 | |
| 	def __len__(self):
 | |
| 		return len(self._values)
 | |
| 
 | |
| 	def __repr__(self):
 | |
| 		return 'NamedInts(%s)' % ', '.join(repr(v) for v in self._values)
 | |
| 
 | |
| 
 | |
| def strhex(x):
 | |
| 	"""Produce a hex-string representation of a sequence of bytes."""
 | |
| 	return _hexlify(x).decode('ascii').upper()
 | |
| 
 | |
| 
 | |
| def bytes2int(x):
 | |
| 	"""Convert a bytes string to an int.
 | |
| 	The bytes are assumed to be in most-significant-first order.
 | |
| 	"""
 | |
| 	assert isinstance(x, bytes)
 | |
| 	assert len(x) < 9
 | |
| 	qx = (b'\x00' * 8) + x
 | |
| 	result, = unpack('!Q', qx[-8:])
 | |
| 	# assert x == int2bytes(result, len(x))
 | |
| 	return result
 | |
| 
 | |
| 
 | |
| def int2bytes(x, count=None):
 | |
| 	"""Convert an int to a bytes representation.
 | |
| 	The bytes are ordered in most-significant-first order.
 | |
| 	If 'count' is not given, the necessary number of bytes is computed.
 | |
| 	"""
 | |
| 	assert isinstance(x, int)
 | |
| 	result = pack('!Q', x)
 | |
| 	assert isinstance(result, bytes)
 | |
| 	# assert x == bytes2int(result)
 | |
| 
 | |
| 	if count is None:
 | |
| 		return result.lstrip(b'\x00')
 | |
| 
 | |
| 	assert isinstance(count, int)
 | |
| 	assert count > 0
 | |
| 	assert x.bit_length() <= count * 8
 | |
| 	return result[-count:]
 | |
| 
 | |
| 
 | |
| class KwException(Exception):
 | |
| 	"""An exception that remembers all arguments passed to the constructor.
 | |
| 	They can be later accessed by simple member access.
 | |
| 	"""
 | |
| 	def __init__(self, **kwargs):
 | |
| 		super(KwException, self).__init__(kwargs)
 | |
| 
 | |
| 	def __getattr__(self, k):
 | |
| 		try:
 | |
| 			return super(KwException, self).__getattr__(k)
 | |
| 		except AttributeError:
 | |
| 			return self.args[0][k]
 | |
| 
 | |
| 
 | |
| from collections import namedtuple
 | |
| 
 | |
| """Firmware information."""
 | |
| FirmwareInfo = namedtuple('FirmwareInfo', [
 | |
| 				'kind',
 | |
| 				'name',
 | |
| 				'version',
 | |
| 				'extras'])
 | |
| 
 | |
| """Reprogrammable keys informations."""
 | |
| ReprogrammableKeyInfo = namedtuple('ReprogrammableKeyInfo', [
 | |
| 				'index',
 | |
| 				'key',
 | |
| 				'task',
 | |
| 				'flags'])
 | |
| 
 | |
| del namedtuple
 |