main.lua 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. --[[
  2. IPv6服务端演示, 仅EC618系列支持, 例如Air780E/Air600E/Air780UG/Air700E
  3. ]]
  4. -- LuaTools需要PROJECT和VERSION这两个信息
  5. PROJECT = "ipv6_server"
  6. VERSION = "1.0.0"
  7. log.info("main", PROJECT, VERSION)
  8. -- 一定要添加sys.lua !!!!
  9. sys = require("sys")
  10. sysplus = require("sysplus")
  11. libnet = require "libnet"
  12. -- Air780E的AT固件默认会为开机键防抖, 导致部分用户刷机很麻烦
  13. if rtos.bsp() == "EC618" and pm and pm.PWK_MODE then
  14. pm.power(pm.PWK_MODE, false)
  15. end
  16. -- 处理未识别的网络消息
  17. local function netCB(msg)
  18. log.info("未处理消息", msg[1], msg[2], msg[3], msg[4])
  19. end
  20. LED = gpio.setup(27, 0)
  21. GPIO1 = gpio.setup(1, 0)
  22. GPIO24 = gpio.setup(24, 0)
  23. HTTP_200_EMTRY = "HTTP/1.0 200 OK\r\nServer: LuatOS\r\nConnection: close\r\nContent-Length: 0\r\n\r\n"
  24. -- 演示task
  25. function ipv6test()
  26. -- 4G模组均支持, 简单判断一下完事
  27. if mobile == nil then
  28. while 1 do
  29. log.info("ipv6", "当前仅4G模组的4G网络支持ipv6功能")
  30. sys.wait(1000)
  31. end
  32. end
  33. -- 启用IPv6, 默认关闭状态,必须在驻网前开启
  34. -- 注意, 启用IPv6, 联网速度会慢2~3秒
  35. mobile.ipv6(true)
  36. log.info("ipv6", "等待联网")
  37. sys.waitUntil("IP_READY")
  38. log.info("ipv6", "联网完成")
  39. sys.wait(1000)
  40. -- 打印一下本地ip, 一般只有ipv6才可能是公网ip, ipv4基本不可能
  41. -- 而且ipv6不一样是外网ip, 这是运营商决定的, 模块无能为力
  42. ip, mask, gw, ipv6 = socket.localIP()
  43. log.info("本地IP地址", ip, ipv6)
  44. if not ipv6 then
  45. log.info("没有IPV6地址,无法演示")
  46. -- return
  47. end
  48. log.info("shell", "telnet -6 " .. ipv6 .. " 14000")
  49. -- 开始正在的逻辑, 发起socket链接,等待数据/上报心跳
  50. local taskName = "ipv6server"
  51. local topic = taskName .. "_txrx"
  52. local txqueue = {}
  53. sysplus.taskInitEx(ipv6task, taskName, netCB, taskName, txqueue, topic)
  54. while 1 do
  55. local result, tp, data = sys.waitUntil(topic, 60000)
  56. if not result then
  57. -- 等很久了,没数据上传/下发, 发个日期心跳包吧
  58. --table.insert(txqueue, string.char(0))
  59. --sys_send(taskName, socket.EVENT, 0)
  60. elseif tp == "uplink" then
  61. -- 上行数据, 主动上报的数据,那就发送呀
  62. table.insert(txqueue, data)
  63. sys_send(taskName, socket.EVENT, 0)
  64. elseif tp == "downlink" then
  65. -- 下行数据,接收的数据, 从ipv6task来的
  66. -- 其他代码可以通过 sys.publish()
  67. log.info("socket", "收到下发的数据了", #data, data)
  68. -- 下面是模拟一个http服务, 因为httpsrv库还没好,先用着吧
  69. if data:startsWith("GET / ") then
  70. local httpresp = "HTTP/1.0 200 OK\r\n"
  71. httpresp = httpresp .. "Server: LuatOS\r\nContent-Type: text/html\r\nConnection: close\r\n"
  72. local fdata = io.readFile("/luadb/index.html")
  73. httpresp = httpresp .. string.format("Content-Length: %d\r\n\r\n", #fdata)
  74. httpresp = httpresp .. fdata
  75. table.insert(txqueue, httpresp)
  76. table.insert(txqueue, "close")
  77. sys_send(taskName, socket.EVENT, 0)
  78. elseif data:startsWith("GET /led/") then
  79. if data:startsWith("GET /led/1") then
  80. log.info("led", "亮起")
  81. LED(1)
  82. else
  83. log.info("led", "熄灭")
  84. LED(0)
  85. end
  86. table.insert(txqueue, HTTP_200_EMTRY)
  87. table.insert(txqueue, "close")
  88. sys_send(taskName, socket.EVENT, 0)
  89. elseif data:startsWith("GET /gpio24/") then
  90. if data:startsWith("GET /gpio24/1") then
  91. log.info("gpio24", "亮起")
  92. GPIO24(1)
  93. else
  94. log.info("gpio24", "熄灭")
  95. GPIO24(0)
  96. end
  97. table.insert(txqueue, HTTP_200_EMTRY)
  98. table.insert(txqueue, "close")
  99. sys_send(taskName, socket.EVENT, 0)
  100. elseif data:startsWith("GET /gpio1/") then
  101. if data:startsWith("GET /gpio1/1") then
  102. log.info("gpio1", "亮起")
  103. GPIO1(1)
  104. else
  105. log.info("gpio1", "熄灭")
  106. GPIO1(0)
  107. end
  108. table.insert(txqueue, HTTP_200_EMTRY)
  109. table.insert(txqueue, "close")
  110. sys_send(taskName, socket.EVENT, 0)
  111. elseif data:startsWith("GET ") or data:startsWith("POST ") or data:startsWith("HEAD ") then
  112. table.insert(txqueue, HTTP_200_EMTRY)
  113. table.insert(txqueue, "close")
  114. sys_send(taskName, socket.EVENT, 0)
  115. end
  116. end
  117. end
  118. end
  119. function ipv6task(d1Name, txqueue, rxtopic)
  120. -- 本地监听的端口
  121. local port = 14000
  122. local rx_buff = zbuff.create(1024)
  123. local tx_buff = zbuff.create(4 * 1024)
  124. local netc = socket.create(nil, d1Name)
  125. socket.config(netc, 14000)
  126. log.info("任务id", d1Name)
  127. while true do
  128. log.info("socket", "开始监控")
  129. local result = libnet.listen(d1Name, 0, netc)
  130. if result then
  131. log.info("socket", "监听成功")
  132. result = socket.accept(netc, nil) --只支持1对1
  133. if result then
  134. log.info("客户端连上了")
  135. end
  136. else
  137. log.info("socket", "监听失败!!")
  138. end
  139. while result do
  140. -- log.info("socket", "调用rx接收数据")
  141. local succ, param = socket.rx(netc, rx_buff)
  142. if not succ then
  143. log.info("客户端断开了", succ, param, ip, port)
  144. break
  145. end
  146. if rx_buff:used() > 0 then
  147. log.info("socket", "收到客户端数据,长度", rx_buff:used())
  148. local data = rx_buff:query() -- 获取数据
  149. sys.publish(rxtopic, "downlink", data)
  150. rx_buff:del()
  151. end
  152. -- log.info("libnet", "调用wait开始等待消息")
  153. result, param, param2 = libnet.wait(d1Name, 15000, netc)
  154. log.info("libnet", "wait", result, param, param2)
  155. if not result then
  156. -- 网络异常了
  157. log.info("socket", "客户端断开了", result, param)
  158. break
  159. elseif #txqueue > 0 then
  160. local force_close = false
  161. while #txqueue > 0 do
  162. local data = table.remove(txqueue, 1)
  163. if not data then
  164. break
  165. end
  166. log.info("socket", "上行数据长度", #data)
  167. if data == "close" then
  168. --sys.wait(1000)
  169. force_close = true
  170. break
  171. end
  172. tx_buff:del()
  173. tx_buff:copy(nil, data)
  174. result,param = libnet.tx(d1Name, 15000, netc, tx_buff)
  175. log.info("libnet", "发送数据的结果", result, param)
  176. if not result then
  177. log.info("socket", "数据发送异常", result, param)
  178. break
  179. end
  180. end
  181. if force_close then
  182. break
  183. end
  184. end
  185. end
  186. log.info("socket", "连接已断开,继续下一个循环")
  187. libnet.close(d1Name, 5000, netc)
  188. -- log.info(rtos.meminfo("sys"))
  189. sys.wait(50)
  190. end
  191. end
  192. sys.taskInit(ipv6test)
  193. -- 用户代码已结束---------------------------------------------
  194. -- 结尾总是这一句
  195. sys.run()
  196. -- sys.run()之后后面不要加任何语句!!!!!