Implement a secure ICS protocol targeting LoRa Node151 microcontroller for controlling irrigation.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

164 lines
4.8 KiB

  1. # Copyright 2021 John-Mark Gurney.
  2. #
  3. # Redistribution and use in source and binary forms, with or without
  4. # modification, are permitted provided that the following conditions
  5. # are met:
  6. # 1. Redistributions of source code must retain the above copyright
  7. # notice, this list of conditions and the following disclaimer.
  8. # 2. Redistributions in binary form must reproduce the above copyright
  9. # notice, this list of conditions and the following disclaimer in the
  10. # documentation and/or other materials provided with the distribution.
  11. #
  12. # THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
  13. # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  14. # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  15. # ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
  16. # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  17. # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  18. # OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  19. # HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  20. # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  21. # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  22. # SUCH DAMAGE.
  23. #
  24. import os
  25. from ctypes import Structure, POINTER, CFUNCTYPE, pointer, sizeof
  26. from ctypes import c_uint8, c_uint16, c_ssize_t, c_size_t, c_uint64, c_int
  27. from ctypes import CDLL
  28. class StructureRepr(object):
  29. def __repr__(self): #pragma: no cover
  30. return '%s(%s)' % (self.__class__.__name__, ', '.join('%s=%s' %
  31. (k, getattr(self, k)) for k, v in self._fields_))
  32. class PktBuf(Structure):
  33. _fields_ = [
  34. ('pkt', POINTER(c_uint8)),
  35. ('pktlen', c_uint16),
  36. ]
  37. def _from(self):
  38. return bytes(self.pkt[:self.pktlen])
  39. def __repr__(self): #pragma: no cover
  40. return 'PktBuf(pkt=%s, pktlen=%s)' % (repr(self._from()),
  41. self.pktlen)
  42. def make_pktbuf(s):
  43. pb = PktBuf()
  44. if isinstance(s, bytearray):
  45. obj = s
  46. pb.pkt = pointer(c_uint8.from_buffer(s))
  47. else:
  48. obj = (c_uint8 * len(s))(*s)
  49. pb.pkt = obj
  50. pb.pktlen = len(s)
  51. pb._make_pktbuf_ref = (obj, s)
  52. return pb
  53. process_msgfunc_t = CFUNCTYPE(None, PktBuf, POINTER(PktBuf))
  54. try:
  55. _lib = CDLL('liblora_test.dylib')
  56. except OSError:
  57. _lib = None
  58. if _lib is not None:
  59. _lib._strobe_state_size.restype = c_size_t
  60. _lib._strobe_state_size.argtypes = ()
  61. _strobe_state_u64_cnt = (_lib._strobe_state_size() + 7) // 8
  62. else:
  63. _strobe_state_u64_cnt = 1
  64. class CommsSession(Structure,StructureRepr):
  65. _fields_ = [
  66. ('cs_crypto', c_uint64 * _strobe_state_u64_cnt),
  67. ('cs_state', c_int),
  68. ]
  69. class CommsState(Structure,StructureRepr):
  70. _fields_ = [
  71. # The alignment of these may be off
  72. ('cs_active', CommsSession),
  73. ('cs_pending', CommsSession),
  74. ('cs_start', c_uint64 * _strobe_state_u64_cnt),
  75. ('cs_procmsg', process_msgfunc_t),
  76. ('cs_prevmsg', PktBuf),
  77. ('cs_prevmsgresp', PktBuf),
  78. ('cs_prevmsgbuf', c_uint8 * 64),
  79. ('cs_prevmsgrespbuf', c_uint8 * 64),
  80. ]
  81. EC_PUBLIC_BYTES = 32
  82. EC_PRIVATE_BYTES = 32
  83. if _lib is not None:
  84. _lib._comms_state_size.restype = c_size_t
  85. _lib._comms_state_size.argtypes = ()
  86. if _lib._comms_state_size() != sizeof(CommsState): # pragma: no cover
  87. raise RuntimeError('CommsState structure size mismatch!')
  88. X25519_BASE_POINT = (c_uint8 * (256//8)).in_dll(_lib, 'X25519_BASE_POINT')
  89. for func, ret, args in [
  90. ('comms_init', None, (POINTER(CommsState), process_msgfunc_t,
  91. POINTER(PktBuf))),
  92. ('comms_process', None, (POINTER(CommsState), PktBuf, POINTER(PktBuf))),
  93. ('strobe_seed_prng', None, (POINTER(c_uint8), c_ssize_t)),
  94. ('x25519', c_int, (c_uint8 * EC_PUBLIC_BYTES, c_uint8 * EC_PRIVATE_BYTES, c_uint8 * EC_PUBLIC_BYTES, c_int)),
  95. ]:
  96. f = getattr(_lib, func)
  97. f.restype = ret
  98. f.argtypes = args
  99. locals()[func] = f
  100. def x25519_wrap(out, scalar, base, clamp):
  101. outptr = (c_uint8 * EC_PUBLIC_BYTES).from_buffer_copy(out)
  102. scalarptr = (c_uint8 * EC_PRIVATE_BYTES).from_buffer_copy(scalar)
  103. baseptr = (c_uint8 * EC_PRIVATE_BYTES).from_buffer_copy(base)
  104. r = x25519(outptr, scalarptr, baseptr, clamp)
  105. if r != 0:
  106. raise RuntimeError('x25519 failed')
  107. return bytes(outptr)
  108. def x25519_genkey():
  109. return os.urandom(EC_PRIVATE_BYTES)
  110. def x25519_base(scalar, clamp):
  111. out = bytearray(EC_PUBLIC_BYTES)
  112. outptr = (c_uint8 * EC_PUBLIC_BYTES).from_buffer(out)
  113. scalarptr = (c_uint8 * EC_PRIVATE_BYTES).from_buffer_copy(scalar)
  114. r = x25519(outptr, scalarptr, X25519_BASE_POINT, clamp)
  115. if r != 0:
  116. raise RuntimeError('x25519 failed')
  117. return bytes(out)
  118. def comms_process_wrap(state, input):
  119. '''A wrapper around comms_process that converts the argument
  120. into the buffer, and the returns the message as a bytes string.
  121. '''
  122. inpkt = make_pktbuf(input)
  123. outbytes = bytearray(64)
  124. outbuf = make_pktbuf(outbytes)
  125. comms_process(state, inpkt, outbuf)
  126. return outbuf._from()