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.
 
 
 
 
 
 

253 lines
8.2 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. ARMOBJDUMP?= arm-none-eabi-objdump
  25. ARMCC?= arm-none-eabi-gcc
  26. ARMTARGET?= -mcpu=cortex-m3 -mthumb -DSTROBE_SINGLE_THREAD=1
  27. # Clang doesn't work due to no-nano libc
  28. #ARMCC?=clang-mp-9.0
  29. #ARMTARGET?= -nostdlib -ffreestanding -target arm-none-eabi -mcpu=cortex-m3 -mfloat-abi=soft -mthumb
  30. PLATFORM != uname -s
  31. .if $(PLATFORM) == "Darwin"
  32. SOEXT=dylib
  33. .else
  34. .error Unsupported platform: $(PLATFORM)
  35. .endif
  36. PROGEXT = .elf
  37. PROGS = lora.gw lora.irr
  38. SRCS.lora.gw = main.c
  39. SRCS.lora.irr = irr_main.c
  40. SRCS.lora.irr+= comms.c
  41. SRCS.lora.irr+= strobe_rng_init.c
  42. SRCS.lora.irr+= $(STROBE_SRCS)
  43. SRCS+= board.c
  44. SRCS+= misc.c
  45. .if 0
  46. SRCS+= rng_save.c
  47. .endif
  48. CFLAGS+= -I$(.CURDIR)
  49. CFLAGS+= -g
  50. #CFLAGS+= -DNDEBUG
  51. CFLAGS+= -I$(.OBJDIR) # for shared_key.h
  52. # Strobe
  53. .PATH: $(.CURDIR)/strobe
  54. CFLAGS+= -I$(.CURDIR)/strobe
  55. STROBE_SRCS+= strobe.c \
  56. x25519.c
  57. # LoRamac (SX1276) radio code
  58. LORAMAC_SRC = $(.CURDIR)/loramac/src
  59. .PATH: $(LORAMAC_SRC)/radio/sx1276 $(LORAMAC_SRC)/system $(LORAMAC_SRC)/boards/mcu $(LORAMAC_SRC)/boards/NucleoL152
  60. CFLAGS+= -I$(LORAMAC_SRC)/boards
  61. CFLAGS+= -I$(LORAMAC_SRC)/system
  62. CFLAGS+= -I$(LORAMAC_SRC)/radio
  63. CFLAGS+= -DUSE_HAL_DRIVER -DSX1276MB1LAS
  64. SRCS+= sx1276.c
  65. SRCS+= utilities.c
  66. SRCS+= adc.c timer.c delay.c gpio.c uart.c fifo.c
  67. SRCS+= adc-board.c delay-board.c gpio-board.c rtc-board.c lpm-board.c sx1276mb1las-board.c spi-board.c uart-board.c
  68. # Microcontroller
  69. STM32=$(.CURDIR)/stm32
  70. .PATH: $(STM32)/l151ccux
  71. LINKER_SCRIPT=$(STM32)/l151ccux/STM32L151CCUX_FLASH.ld
  72. SRCS+= \
  73. startup_stm32l151ccux.s \
  74. stm32l1xx_hal.c \
  75. stm32l1xx_hal_adc.c \
  76. stm32l1xx_hal_adc_ex.c \
  77. stm32l1xx_hal_cortex.c \
  78. stm32l1xx_hal_dma.c \
  79. stm32l1xx_hal_flash.c \
  80. stm32l1xx_hal_flash_ex.c \
  81. stm32l1xx_hal_gpio.c \
  82. stm32l1xx_hal_pcd.c \
  83. stm32l1xx_hal_pcd_ex.c \
  84. stm32l1xx_hal_pwr.c \
  85. stm32l1xx_hal_pwr_ex.c \
  86. stm32l1xx_hal_rcc.c \
  87. stm32l1xx_hal_rcc_ex.c \
  88. stm32l1xx_hal_rtc.c \
  89. stm32l1xx_hal_rtc_ex.c \
  90. stm32l1xx_hal_spi.c \
  91. stm32l1xx_hal_uart.c \
  92. system_stm32l1xx.c
  93. SRCS+= \
  94. stm32l1xx_it.c \
  95. stm32l1xx_hal_msp.c
  96. CFLAGS+= -I$(STM32)
  97. CFLAGS+= -I$(STM32)/l151ccux
  98. CFLAGS+= -DSTM32L151xC
  99. # USB
  100. .PATH: $(STM32)/usb
  101. SRCS+= \
  102. stm32l1xx_ll_usb.c \
  103. usb_device.c \
  104. usbd_cdc.c \
  105. usbd_cdc_if.c \
  106. usbd_conf.c \
  107. usbd_core.c \
  108. usbd_ctlreq.c \
  109. usbd_desc.c \
  110. usbd_ioreq.c
  111. CFLAGS+= -I$(STM32)/usb
  112. CFLAGS+= -Werror -Wall
  113. LIBLORA_TEST_SRCS= comms.c strobe.c x25519.c
  114. LIBLORA_TEST_OBJS= $(LIBLORA_TEST_SRCS:C/.c$/.no/)
  115. LIBLORA_TEST = liblora_test.$(SOEXT)
  116. $(LIBLORA_TEST): $(LIBLORA_TEST_OBJS)
  117. $(CC) -shared -o $@ $(.ALLSRC)
  118. .MAIN: all
  119. .PHONY: all
  120. DEPENDS = .arm_deps .test_deps
  121. .PHONY: depend
  122. depend: $(DEPENDS)
  123. .for i in $(DEPENDS)
  124. .sinclude "$i"
  125. .endfor
  126. # Temporarily disabled as OpenOCD can't program EEPROM on the STM32L1.
  127. # This could be used to add the data to flash.
  128. rng_save.c: Makefile
  129. ( echo '#include <strobe_rng_init.h>'; \
  130. echo 'const rng_word_t rng_save[roundup(32, sizeof(rng_word_t)) / sizeof(rng_word_t)] __attribute__ ((section (".eeprom"))) = {'; \
  131. dd if=/dev/random bs=32 count=1 | hexdump -e '4/4 " %3u," "\n"'; \
  132. echo '};' \
  133. ) > $@ || (rm $@ && false)
  134. .arm_deps:
  135. $(ARMCC) $(ARMTARGET) $(CFLAGS) $(.ALLSRC) -MM > $@ || (rm -f $@ && false)
  136. .test_deps: $(LIBLORA_TEST_SRCS)
  137. $(CC) $(CFLAGS) $(.ALLSRC) -MM | sed -e 's/\.o:/\.no:/' > $@ || (rm -f $@ && false)
  138. .for i in $(PROGS)
  139. ALLTGTS+= $(i)$(PROGEXT) $(i).list
  140. ASRCS.$(i) = $(SRCS) $(SRCS.$(i))
  141. OBJS.$(i) = $(ASRCS.$(i):C/.c$/.o/)
  142. .arm_deps: $(ASRCS.$(i))
  143. $(i)$(PROGEXT) $(i).map: $(OBJS.$(i))
  144. $(ARMCC) $(ARMTARGET) -o $(i)$(PROGEXT) $(.ALLSRC) -T$(LINKER_SCRIPT) --specs=nosys.specs -Wl,-Map="$(i).map" -Wl,--gc-sections -static --specs=nano.specs -Wl,--start-group -lc -lm -Wl,--end-group
  145. $(i).list: $(i)$(PROGEXT)
  146. $(ARMOBJDUMP) -h -S $(.ALLSRC) > $@ || (rm -f $@ && false)
  147. .endfor
  148. all: $(ALLTGTS)
  149. .PHONY: runbuild
  150. runbuild: $(SRCS)
  151. for i in $(.MAKEFILE_LIST) $(.ALLSRC) $$(cat $(DEPENDS) | gsed ':x; /\\$$/ { N; s/\\\n//; tx }' | sed -e 's/^[^:]*://'); do if [ "$$i" != ".." ]; then echo $$i; fi; done | sort -u | entr -d sh -c 'echo starting...; cd $(.CURDIR) && $(MAKE) $(.MAKEFLAGS) depend && $(MAKE) $(.MAKEFLAGS) all'
  152. .PHONY: runtests
  153. runtests: Makefile lora_comms.py lora.py loraserv.py multicast.py $(LIBLORA_TEST) $(LIBLORA_TEST_SRCS)
  154. ls $(.ALLSRC) | entr sh -c '(cd $(.CURDIR) && $(MAKE) $(.MAKEFLAGS) $(LIBLORA_TEST)) && ((PYTHONPATH="$(.CURDIR)" python -m coverage run -m unittest lora multicast loraserv && coverage report --omit=$(.CURDIR)/p/\* -m -i) 2>&1 | head -n 30)'
  155. .if empty(IRR_KEY) && exists($(.CURDIR)/.irr_key)
  156. IRR_KEY!= cat $(.CURDIR)/.irr_key
  157. .endif
  158. .PHONY: irrigation_key
  159. irrigation_key:
  160. @LANG=C tr -c -d a-zA-Z0-9 < /dev/urandom | dd bs=1 of=$(.CURDIR)/.irr_key count=32 2>/dev/null
  161. @echo 'Irrgation key created and put into .irr_key.'
  162. # make this a phony target so it's always run
  163. # dependancies will only be made when it's updated
  164. .PHONY: $(.OBJDIR)/shared_key.h
  165. $(.OBJDIR)/shared_key.h:
  166. @if [ "$(IRR_KEY)" = "" ]; then echo 'Must provide IRR_KEY make variable or have a non-empty file ".irr_key".'; false; fi
  167. @echo 'static uint8_t shared_key[] = {' $$(python -c 'import sys; print(", ".join(str(x) for x in sys.argv[1].encode("utf-8")))' $(IRR_KEY) ) "};" > shared_key.h.tmp
  168. @echo 'static struct pktbuf shared_key_buf = (struct pktbuf){ .pkt = shared_key, .pktlen = sizeof shared_key, };' >> shared_key.h.tmp
  169. (cmp shared_key.h.tmp shared_key.h >/dev/null 2>&1 && rm shared_key.h.tmp) || mv shared_key.h.tmp shared_key.h
  170. # native objects
  171. .SUFFIXES: .no
  172. .c.no:
  173. $(CC) $(CFLAGS) -c $< -o $@
  174. .c.o:
  175. $(ARMCC) $(ARMTARGET) $(CFLAGS) -c $< -o $@
  176. STROBE_NAME = strobe
  177. STROBE_REPO = https://git.code.sf.net/p/strobe/code
  178. STROBE_BRANCH = master
  179. LORAMAC_NAME = loramac
  180. LORAMAC_REPO = https://github.com/Lora-net/LoRaMac-node.git
  181. LORAMAC_BRANCH = master
  182. .for module in STROBE LORAMAC
  183. .PHONY: init-$($(module)_NAME)
  184. init-$($(module)_NAME):
  185. git subtree add -P $($(module)_NAME) --squash $($(module)_REPO) $($(module)_BRANCH)
  186. .PHONY: update-$($(module)_NAME)
  187. update-$($(module)_NAME):
  188. git subtree pull -P $($(module)_NAME) --squash $($(module)_REPO) $($(module)_BRANCH)
  189. .endfor
  190. # Making diagrams, imported from fbsdembdev
  191. # =========================================
  192. # https://github.com/ironcamel/Graph-Easy
  193. .SUFFIXES: .getxt .diag .html
  194. .getxt.diag: box.sh
  195. graph-easy < $< | sh $(.CURDIR)/box.sh > $@
  196. .diag.html:
  197. (cat $<; echo; echo '<!-- Markdeep: --><style class="fallback">body{visibility:hidden;white-space:pre;font-family:monospace}</style><script src="markdeep.min.js" charset="utf-8"></script><script src="https://casual-effects.com/markdeep/latest/markdeep.min.js" charset="utf-8"></script><script>window.alreadyProcessedMarkdeep||(document.body.style.visibility="visible")</script>') > $@
  198. DIAG?=diag
  199. test-diag:
  200. ls $(.CURDIR)/box.sh $(.CURDIR)/$(DIAG).getxt $(.CURDIR)/Makefile | entr sh -c 'cd $(.CURDIR) && $(MAKE) $(.MAKEFLAGS) $(DIAG).diag && cat $(.OBJDIR)/$(DIAG).diag'
  201. # hard coded dependancy for when "make depend" has not been run.
  202. irr_main.o: $(.OBJDIR)/shared_key.h