tc_tm_protocols.lua 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517
  1. -- Define the protocol
  2. TC_TM_PROTOCOL = Proto("tc_tm", "TC_TM")
  3. local f_sof1 = ProtoField.uint8("TC_TM_PROTOCOL.sof1", "SOF1", base.HEX)
  4. local f_sof2 = ProtoField.uint8("TC_TM_PROTOCOL.sof2", "SOF2", base.HEX)
  5. local f_tm_ctrl = ProtoField.uint8("TC_TM_PROTOCOL.tm_ctrl", "TM Ctrl", base.HEX)
  6. local f_GS_Id = ProtoField.uint8("TC_TM_PROTOCOL.GS_Id", "GS Id", base.HEX)
  7. local f_timestamp = ProtoField.uint32("TC_TM_PROTOCOL.timestamp", "Timestamp", base.DEC)
  8. local f_sequence_no = ProtoField.uint16("TC_TM_PROTOCOL.sequence_no", "Sequence No", base.DEC)
  9. local f_sat_id = ProtoField.uint16("TC_TM_PROTOCOL.sat_id", "SAT Id", base.DEC)
  10. local f_qos = ProtoField.uint8("TC_TM_PROTOCOL.qos", "QoS", base.DEC)
  11. local f_sa_id = ProtoField.uint16("TC_TM_PROTOCOL.sa_id", "SA Id", base.DEC)
  12. local f_da_id = ProtoField.uint16("TC_TM_PROTOCOL.da_id", "DA Id", base.DEC)
  13. local f_rm_id = ProtoField.uint8("TC_TM_PROTOCOL.rm_id", "RM Id", base.DEC)
  14. local f_tm_id = ProtoField.uint16("TC_TM_PROTOCOL.tm_id", "TM Id", base.DEC)
  15. local f_co_id = ProtoField.uint16("TC_TM_PROTOCOL.co_id", "Co Id", base.DEC)
  16. local f_tm_len = ProtoField.uint16("TC_TM_PROTOCOL.tm_len", "TM Len", base.DEC)
  17. local f_tm_data = ProtoField.uint8("TC_TM_PROTOCOL.tm_data", "TM Data", base.DEC)
  18. local f_ext_header_len = ProtoField.uint8("TC_TM_PROTOCOL.ext_header_len", "Ext Header Len", base.DEC)
  19. local f_ext_header_data = ProtoField.uint8("TC_TM_PROTOCOL.ext_header_data", "Ext Header Data", base.DEC)
  20. local f_crc = ProtoField.uint8("TC_TM_PROTOCOL.crc", "CRC", base.DEC)
  21. local f_c_mac = ProtoField.uint8("TC_TM_PROTOCOL.c_mac", "C MAC", base.DEC)
  22. local f_eof1 = ProtoField.uint8("TC_TM_PROTOCOL.eof1", "EOF1", base.HEX)
  23. local f_tc_id = ProtoField.uint16("TC_TM_PROTOCOL.tc_id", "TC Id", base.DEC)
  24. local f_tc_len = ProtoField.uint16("TC_TM_PROTOCOL.tc_len", "TC Len", base.DEC)
  25. local f_tc_data = ProtoField.uint32("TC_TM_PROTOCOL.tc_data", "TC Data", base.DEC)
  26. -- Add the fields to the protocol
  27. TC_TM_PROTOCOL.fields = {
  28. f_sof1, f_sof2, f_tm_ctrl,f_GS_Id, f_timestamp, f_sequence_no, f_sat_id,
  29. f_qos, f_sa_id, f_da_id, f_rm_id, f_tm_id, f_co_id, f_tm_len, f_tm_data,
  30. f_ext_header_len, f_ext_header_data, f_crc, f_c_mac, f_eof1,f_tc_id ,f_tc_len ,f_tc_data
  31. }
  32. ------------------------------------------------------------------------------------------------------------------------------------
  33. function TC_dissector(buffer, pinfo, tree)
  34. --Set the protocol coloumn name in wireshark
  35. local subtree = tree:add(TC_TM_PROTOCOL, buffer(), "TELECOMMAND")
  36. -- Dissect SOF1 and SOF2
  37. subtree:add(f_sof1 , buffer(4, 1))
  38. subtree:add(f_sof2, buffer(5, 1))
  39. -- Dissect TC Control byte (TC_Ctrl)
  40. local offset = 6
  41. local tc_ctrl = buffer(offset, 1):le_uint() -- Get the value of TC Ctrl byte
  42. subtree:add(f_tm_ctrl , tc_ctrl)
  43. offset = offset + 1
  44. -- Dissect Timestamp (4 bytes)
  45. if buffer:len() >= offset + 4 then
  46. local timestamp = buffer(offset, 4):le_uint()
  47. subtree:add(f_timestamp, timestamp)
  48. offset = offset + 4
  49. else
  50. subtree:add(f_timestamp, "Invalid data (timestamp)")
  51. end
  52. -- Dissect Sequence No (2 bytes)
  53. if buffer:len() >= offset + 2 then
  54. local sequence = buffer(offset, 2):le_uint()
  55. subtree:add(f_sequence_no, sequence)
  56. offset = offset + 2
  57. else
  58. subtree:add(f_sequence_no, "Invalid data (sequence number)")
  59. end
  60. -- Dissect SAT_Id (1 byte)
  61. if buffer:len() >= offset + 1 then
  62. local sat_id = buffer(offset, 1):uint() -- Get SAT Id (1st byte)
  63. --offset = offset + 1
  64. if bit.band(sat_id, 0x80) == 0x80 then --if 7th bit is set and
  65. if buffer:len() >= offset + 1 then
  66. local second = buffer(offset, 2):le_uint()
  67. if bit.band(second, 0x8000) == 0x0000 then --and 15th bit is set then unset 15th bit
  68. sat_id = buffer(offset, 2):le_uint()
  69. sat_id = bit.band(sat_id, 0xFF7F)
  70. offset = offset + 1
  71. else
  72. sat_id = buffer(offset, 2):le_uint()
  73. sat_id = bit.band(sat_id, 0x7FFF) -- and 15th bit is unset then unset the 7th bit
  74. offset = offset + 1
  75. end
  76. end
  77. end
  78. subtree:add(f_sat_id, sat_id)
  79. offset = offset + 1
  80. end
  81. -- Dissect GS_Id (1 byte)
  82. if buffer:len() >= offset + 1 then
  83. local gs_id = buffer(offset, 1):uint() -- Get SAT Id (1st byte)
  84. --offset = offset + 1
  85. if bit.band(gs_id, 0x80) == 0x80 then
  86. if buffer:len() >= offset + 1 then
  87. local second = buffer(offset, 2):le_uint()
  88. if bit.band(second, 0x8000) == 0x0000 then
  89. gs_id = buffer(offset, 2):le_uint()
  90. gs_id = bit.band(gs_id, 0xFF7F)
  91. offset = offset + 1
  92. else
  93. gs_id = buffer(offset, 2):le_uint()
  94. gs_id = bit.band(gs_id, 0x7FFF)
  95. offset = offset + 1
  96. end
  97. end
  98. end
  99. subtree:add(f_GS_Id, gs_id)
  100. offset = offset + 1
  101. end
  102. -- Dissect QoS (1 byte)
  103. if buffer:len() >= offset + 1 then
  104. subtree:add(f_qos, buffer(offset, 1))
  105. offset = offset + 1
  106. end
  107. -- Dissect SA_Id (Source Application ID) and DA_Id (Destination Application ID)
  108. if buffer:len() >= offset + 1 then
  109. local sa_id = buffer(offset, 1):uint() -- Get source Application Id (1st byte)
  110. --offset = offset + 1
  111. if bit.band(sa_id, 0x80) == 0x80 then
  112. if buffer:len() >= offset + 1 then
  113. local second = buffer(offset, 2):le_uint()
  114. if bit.band(second, 0x8000) == 0x0000 then
  115. sa_id = buffer(offset, 2):le_uint()
  116. sa_id = bit.band(sa_id, 0xFF7F)
  117. offset = offset + 1
  118. else
  119. sa_id = buffer(offset, 2):le_uint()
  120. sa_id = bit.band(sa_id, 0x7FFF)
  121. offset = offset + 1
  122. end
  123. end
  124. end
  125. subtree:add(f_sa_id, sa_id)
  126. offset = offset + 1
  127. end
  128. if buffer:len() >= offset + 1 then
  129. --subtree:add(f_DA_Id, buffer(offset, 1))
  130. local da_id = buffer(offset, 1):le_uint() -- Get Destination Application Id (1st byte)
  131. --offset = offset + 1
  132. if bit.band(da_id, 0x80) == 0x80 then
  133. if buffer:len() >= offset + 1 then
  134. local second = buffer(offset, 2):le_uint()
  135. if bit.band(second, 0x8000) == 0x0000 then
  136. da_id = buffer(offset, 2):le_uint()
  137. da_id = bit.band(da_id, 0xFF7F)
  138. offset = offset + 1
  139. else
  140. da_id = buffer(offset, 2):le_uint()
  141. da_id = bit.band(da_id, 0x7FFF)
  142. offset = offset + 1
  143. end
  144. end
  145. end
  146. subtree:add(f_da_id, da_id)
  147. offset = offset + 1
  148. end
  149. -- Dissect RM_Id (1 byte)
  150. if buffer:len() >= offset + 1 then
  151. subtree:add(f_rm_id, buffer(offset, 1))
  152. offset = offset + 1
  153. end
  154. -- Dissect TC_Id (2 bytes)
  155. -- Dissect TC_Id (2 bytes)
  156. local info_display
  157. local tc_id = buffer(offset, 1):le_uint() -- Read 1 byte for tc_id
  158. if buffer:len() >= offset + 2 then -- Ensure enough data is available to read 2 bytes
  159. -- Check if the 7th bit (0x80) is set in tc_id
  160. if bit.band(tc_id, 0x80) == 0x80 then
  161. if buffer:len() >= offset + 2 then -- Ensure enough data for the next 2 bytes
  162. local second = buffer(offset, 2):le_uint()
  163. if bit.band(second, 0x8000) == 0x0000 then
  164. tc_id = buffer(offset, 2):le_uint()
  165. tc_id = bit.band(tc_id, 0xFF7F)
  166. offset = offset + 1
  167. else
  168. tc_id = buffer(offset, 2):le_uint()
  169. tc_id = bit.band(tc_id, 0x7FFF)
  170. offset = offset + 1
  171. end
  172. end
  173. end
  174. -- Add tc_id to the subtree (display in dissection tree)
  175. subtree:add(f_tc_id, tc_id)
  176. -- Move the offset forward by 1 (since we initially read 1 byte)
  177. offset = offset + 1
  178. end
  179. -- Dissect TC_Len (2 bytes)
  180. local tc_len = 0 -- Declare tc_len as a local variable here
  181. if buffer:len() >= offset + 2 then
  182. tc_len = buffer(offset, 2):le_uint()
  183. subtree:add(f_tc_len, tc_len)
  184. offset = offset + 2
  185. end
  186. -- Dissect TC_Data (variable length)
  187. --local subtree2 = subtree:add(TC_TM_PROTOCOL, buffer(), TC_DATA())
  188. local data
  189. if buffer:len() >= offset + tc_len then
  190. if tc_len == 0 then
  191. subtree:add("TC DATA : NIL")
  192. else
  193. if tc_id == 621 then
  194. require("HM_dissector")
  195. data = HM_dissector(buffer, pinfo, tree, TC_TM_PROTOCOL, offset)
  196. offset = offset + tc_len
  197. else
  198. for i=0,tc_len - 1 do
  199. subtree:add(f_tc_data, buffer(offset, 1))
  200. offset = offset + 1
  201. end
  202. end
  203. end
  204. end
  205. -- Check if we can dissect Ext_Header_Len
  206. if bit.band(tc_ctrl, 0x80) == 0x80 and buffer:len() >= offset + 1 then
  207. subtree:add(f_ext_header_len, buffer(offset, 1))
  208. local ext_header_len = buffer(offset, 1):uint()
  209. offset = offset + 1
  210. if buffer:len() >= offset + ext_header_len then
  211. for i=1,ext_header_len do
  212. subtree:add(f_ext_header_data, buffer(offset, 1))
  213. offset = offset + 1
  214. end
  215. end
  216. end
  217. -- Dissect CRC (1 byte)
  218. if buffer:len() >= offset + 1 then
  219. subtree:add(f_crc, buffer(offset, 1))
  220. offset = offset + 1
  221. end
  222. -- Dissect C-MAC/H-MAC (if TC Ctrl bit 4 is set)
  223. if bit.band(tc_ctrl, 0x10) == 0x10 and buffer:len() >= offset + 32 then
  224. for i=1,32 do
  225. subtree:add(f_c_mac, buffer(offset, 1))
  226. offset = offset + 1
  227. end
  228. end
  229. -- Dissect EOF (1 byte)
  230. if buffer:len() >= offset + 1 then
  231. subtree:add(f_eof1, buffer(offset, 1))
  232. end
  233. local string
  234. if data == 9 then
  235. string = "EPS"
  236. elseif data == 22 then
  237. string = "ADCS"
  238. elseif data == 14 then
  239. string = "COMMS_SBAND_CNTRL"
  240. elseif data == 15 then
  241. string = "COMMS_UHF_CNTRL"
  242. elseif data == 42 then
  243. string = "SENSOR"
  244. elseif data == 27 then
  245. string = "OBC"
  246. else
  247. string = "All module"
  248. end
  249. info_display= {buffer(11, 2):le_uint(),buffer(7, 4):le_uint() ,tc_id, string} --- for info display in wireshark
  250. pinfo.cols.info ="seq_no = " ..info_display[1] .." TS = " ..info_display[2].." TC_ID = "..info_display[3].." Module = " ..info_display[4] -- for info display in wireshark
  251. end
  252. -----------------------------------------------------------------------------------------------------------------------------------
  253. function TM_dissector(buffer, pinfo, tree) -- OBC to GS
  254. local subtree = tree:add(TC_TM_PROTOCOL, buffer(), "TELEMETRY")
  255. -- Set the protocol column name in Wireshark
  256. -- Ensure the buffer is large enough to contain the entire structure (40 bytes)
  257. -- if buffer:len() < 40 then
  258. -- return -- Not enough data to dissect
  259. --end
  260. -- Extract the fields in order as defined in the structure
  261. subtree:add(f_sof1, buffer(4, 1):uint()) -- SOF1 in HEX
  262. subtree:add(f_sof2, buffer(5, 1):uint()) -- SOF2 in HEX
  263. subtree:add(f_tm_ctrl, buffer(6, 1):le_uint()) -- TM Ctrl in HEX
  264. subtree:add(f_timestamp, buffer(7, 4):le_uint())
  265. subtree:add(f_sequence_no, buffer(11, 2):le_uint()) -- Sequence No in HEX
  266. -- My logic for SAT Id
  267. local magic = buffer(13, 1):le_uint()
  268. local add=13
  269. if bit.band(magic, 0x80) == 0 then
  270. subtree:add(f_sat_id, buffer(add, 1):le_uint()) -- SAT Id in HEX
  271. add=add+1
  272. else
  273. magic=buffer(add, 2):le_uint()
  274. local dummy=magic & 0x8000 --to check 7th byte is 1
  275. if dummy == 0 then
  276. magic = magic & 0xff7f -- if 15th bit is 0 set 7th bit 0
  277. else
  278. magic = magic & 0x7fff -- if 15 bit is 1 set 15th bit to 0
  279. end
  280. subtree:add(f_sat_id, magic) -- SAT Id in HEX
  281. add=add+2
  282. end
  283. -- QoS in HEX
  284. subtree:add(f_qos, buffer(add, 1):le_uint())
  285. add=add+1
  286. -- My logic for SA Id
  287. magic = buffer(add, 1):le_uint()
  288. if bit.band(magic, 0x80) == 0 then
  289. subtree:add(f_sa_id, buffer(add, 1):le_uint()) -- SA Id in HEX
  290. add=add+1
  291. else
  292. magic=buffer(add, 2):le_uint()
  293. local dummy=magic & 0x8000
  294. if dummy == 0 then
  295. magic = magic & 0xff7f
  296. else
  297. magic = magic & 0x7fff
  298. end
  299. subtree:add(f_sa_id,magic) -- SA Id in HEX
  300. add=add+2
  301. end
  302. -- My logic for DA Id
  303. magic = buffer(add, 1):uint()
  304. if bit.band(magic, 0x80) == 0 then
  305. subtree:add(f_da_id, buffer(add, 1):le_uint()) -- DA Id in HEX
  306. add=add+1
  307. else
  308. magic=buffer(add, 2):le_uint()
  309. local dummy=magic & 0x8000
  310. if dummy == 0 then
  311. magic = magic & 0xff7f
  312. else
  313. magic = magic & 0x7fff
  314. end
  315. subtree:add(f_da_id, magic) -- DA Id in HEX
  316. add=add+2
  317. end
  318. -- RM Id in HEX
  319. subtree:add(f_rm_id, buffer(add, 1):le_uint())
  320. add=add+1
  321. -- My logic for TM Id
  322. local info_display -- for display in wireshark
  323. local tm_id
  324. magic = buffer(add, 1):uint()
  325. if bit.band(magic, 0x80) == 0 then
  326. subtree:add(f_tm_id, buffer(add, 1):le_uint()) -- TM Id in HEX
  327. info_display= {buffer(11, 2):le_uint(),buffer(7, 4):le_uint() ,buffer(add, 1):le_uint()} --- for info display in wireshark
  328. add=add+1
  329. else
  330. magic=buffer(add, 2):le_uint()
  331. local dummy=magic & 0x8000
  332. if dummy == 0 then
  333. magic = magic & 0xff7f
  334. else
  335. magic = magic & 0x7fff
  336. end
  337. subtree:add(f_tm_id, magic) -- TM Id in HEX
  338. info_display= {buffer(11, 2):le_uint(),buffer(7, 4):le_uint() ,magic} --- for info display in wireshark
  339. add=add+2
  340. end
  341. tm_id = magic
  342. -- Co Id in HEX
  343. subtree:add(f_co_id, buffer(add, 2):le_uint())
  344. add=add+2
  345. -- TM Len in HEX
  346. local tm_len
  347. magic = buffer(add, 2):le_uint()
  348. tm_len = magic
  349. subtree:add(f_tm_len, buffer(add, 2):le_uint())
  350. add=add+2
  351. -- logic for TM Data
  352. local subtree2 = subtree:add(TC_TM_PROTOCOL, buffer(), "TM_DATA")
  353. -- Dissect TM_Data (variable length)
  354. if tm_id == 621 then
  355. require("HM_dissector")
  356. HM_TM_dissector(buffer, pinfo, tree, TC_TM_PROTOCOL, add, tm_len)
  357. add = add + tm_len
  358. else
  359. for i=0,magic-1 do
  360. subtree2:add(f_tm_data, buffer(add, 1):le_uint())
  361. add=add+1
  362. end
  363. end
  364. -- Ext Header Len in HEX
  365. magic = buffer(6, 1):le_uint() -- to check tm control
  366. if bit.band(magic, 0x80) == 0x80 then
  367. magic= buffer(add, 1):uint()
  368. subtree:add(f_ext_header_len, buffer(add, 1):uint())
  369. add=add+1
  370. -- logic for Ext Header Data
  371. for i=0,magic-1 do
  372. subtree:add(f_ext_header_data, buffer(add, 1):uint()) -- Ext Header Data in HEX
  373. add=add+1
  374. end
  375. end
  376. -- CRC in HEX
  377. subtree:add(f_crc, buffer(add, 1):le_uint())
  378. add=add+1
  379. -- C MAC in HEX
  380. magic = buffer(6, 1):le_uint() -- tm control
  381. if bit.band(magic, 0x10) == 0x10 then
  382. for i=1,32 do
  383. subtree:add(f_c_mac, buffer(add, 1):uint())
  384. add=add+1
  385. end
  386. end
  387. -- EOF1 in HEX
  388. subtree:add(f_eof1, buffer(add, 1):uint())
  389. pinfo.cols.info ="seq_no=" ..info_display[1] .." TS=" ..info_display[2].." TM_ID="..info_display[3] -- for info display in wireshark
  390. -- Check if the buffer contains enough data for the expected total length
  391. --if buffer:len() < 38 + buffer(21, 2):uint() then
  392. -- subtree:add_expert_info(PI_MALFORMED, PI_WARN, "Incomplete data")
  393. --end
  394. end
  395. -- Register the dissector for udp port
  396. -------------------------------------------------------------------------------------------------------------------------------
  397. -- Dissector function
  398. function TC_TM_PROTOCOL.dissector(buffer, pinfo, tree)
  399. -- Register protocol in preferences (add this part)
  400. --my_udp_protocol:register_preference()
  401. pinfo.cols.protocol = "TC_TM"
  402. -- Create a subtree for this protocol
  403. --local subtree = tree:add(my_udp_protocol, buffer(), "TE")
  404. local direction = buffer(6, 1):uint() -- taking tM or tc control byte
  405. if direction & 0x40 == 0x40 then -- if 6th bit is set then it is TM
  406. pinfo.cols.protocol = "TELEMETRY"
  407. TM_dissector(buffer, pinfo, tree) -- OBC to GS
  408. else
  409. pinfo.cols.protocol = "TELECOMMAND"
  410. TC_dissector(buffer, pinfo, tree) -- GS to OBC
  411. end
  412. end
  413. ------------------------------------------------------------------------------------------------------------------------------------------------
  414. -- Add a preference for the port number
  415. TC_TM_PROTOCOL.prefs.port = Pref.uint("UDP Port", 6767, "UDP port for TC_TM protocol")
  416. -- Register the dissector based on the preference
  417. function TC_TM_PROTOCOL.init()
  418. local udp_table = DissectorTable.get("udp.port")
  419. udp_table:add(TC_TM_PROTOCOL.prefs.port, TC_TM_PROTOCOL)
  420. end