ZuluSCSI_platform_network.cpp 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. /*
  2. * Copyright (c) 2023 joshua stein <jcs@jcs.org>
  3. *
  4. * Permission to use, copy, modify, and distribute this software for any
  5. * purpose with or without fee is hereby granted, provided that the above
  6. * copyright notice and this permission notice appear in all copies.
  7. *
  8. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  9. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  10. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
  11. * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  12. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  13. * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  14. * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  15. */
  16. #ifdef ZULUSCSI_NETWORK
  17. #include <Arduino.h>
  18. #include "ZuluSCSI_platform_network.h"
  19. #include "ZuluSCSI_log.h"
  20. #include "ZuluSCSI_config.h"
  21. #include <scsi.h>
  22. #include <network.h>
  23. extern "C" {
  24. #include <cyw43.h>
  25. #include <pico/cyw43_arch.h>
  26. #include <pico/unique_id.h>
  27. #ifndef CYW43_IOCTL_GET_RSSI
  28. #define CYW43_IOCTL_GET_RSSI (0xfe)
  29. #endif
  30. #define PICO_W_GPIO_LED_PIN 0
  31. #define PICO_W_LED_ON() cyw43_arch_gpio_put(PICO_W_GPIO_LED_PIN, 1)
  32. #define PICO_W_LED_OFF() cyw43_arch_gpio_put(PICO_W_GPIO_LED_PIN, 0)
  33. #define PICO_W_LONG_BLINK_DELAY 200
  34. #define PICO_W_SHORT_BLINK_DELAY 75
  35. // A default DaynaPort-compatible MAC
  36. static const char defaultMAC[] = { 0x00, 0x80, 0x19, 0xc0, 0xff, 0xee };
  37. static bool network_in_use = false;
  38. bool platform_network_supported()
  39. {
  40. /* from cores/rp2040/RP2040Support.h */
  41. #if !defined(ARDUINO_RASPBERRY_PI_PICO_W)
  42. return false;
  43. #else
  44. extern bool __isPicoW;
  45. return __isPicoW;
  46. #endif
  47. }
  48. int platform_network_init(char *mac)
  49. {
  50. pico_unique_board_id_t board_id;
  51. uint8_t set_mac[6], read_mac[6];
  52. if (!platform_network_supported())
  53. return -1;
  54. // long signal blink at network initialization
  55. PICO_W_LED_OFF();
  56. PICO_W_LED_ON();
  57. delay(PICO_W_LONG_BLINK_DELAY);
  58. PICO_W_LED_OFF();
  59. logmsg(" ");
  60. logmsg("=== Network Initialization ===");
  61. memset(wifi_network_list, 0, sizeof(wifi_network_list));
  62. cyw43_deinit(&cyw43_state);
  63. cyw43_init(&cyw43_state);
  64. if (mac == NULL || (mac[0] == 0 && mac[1] == 0 && mac[2] == 0 && mac[3] == 0 && mac[4] == 0 && mac[5] == 0))
  65. {
  66. mac = (char *)&set_mac;
  67. memcpy(mac, defaultMAC, sizeof(set_mac));
  68. // retain Dayna vendor but use a device id specific to this board
  69. pico_get_unique_board_id(&board_id);
  70. if (g_log_debug)
  71. logmsg("Unique board id: ", board_id.id[0], " ", board_id.id[1], " ", board_id.id[2], " ", board_id.id[3], " ",
  72. board_id.id[4], " ", board_id.id[5], " ", board_id.id[6], " ", board_id.id[7]);
  73. if (board_id.id[3] != 0 && board_id.id[4] != 0 && board_id.id[5] != 0)
  74. {
  75. mac[3] = board_id.id[3];
  76. mac[4] = board_id.id[4];
  77. mac[5] = board_id.id[5];
  78. }
  79. memcpy(scsiDev.boardCfg.wifiMACAddress, mac, sizeof(scsiDev.boardCfg.wifiMACAddress));
  80. }
  81. // setting the MAC requires libpico to be compiled with CYW43_USE_OTP_MAC=0
  82. memcpy(cyw43_state.mac, mac, sizeof(cyw43_state.mac));
  83. cyw43_arch_enable_sta_mode();
  84. cyw43_wifi_get_mac(&cyw43_state, CYW43_ITF_STA, read_mac);
  85. logmsg("Wi-Fi MAC: ", read_mac[0],":",read_mac[1], ":", read_mac[2], ":", read_mac[3], ":", read_mac[4], ":", read_mac[5]);
  86. if (memcmp(mac, read_mac, sizeof(read_mac)) != 0)
  87. logmsg("WARNING: Wi-Fi MAC is not what was requested (",
  88. (uint8_t)mac[0], ":", (uint8_t)mac[1], ":", (uint8_t)mac[2], ":", (uint8_t)mac[3], ":", (uint8_t)mac[4], ":", (uint8_t)mac[5],
  89. "), is libpico not compiled with CYW43_USE_OTP_MAC=0?");
  90. network_in_use = true;
  91. return 0;
  92. }
  93. void platform_network_add_multicast_address(uint8_t *mac)
  94. {
  95. int ret;
  96. if ((ret = cyw43_wifi_update_multicast_filter(&cyw43_state, mac, true)) != 0)
  97. logmsg( __func__, ": cyw43_wifi_update_multicast_filter: ", ret);
  98. }
  99. bool platform_network_wifi_join(char *ssid, char *password)
  100. {
  101. int ret;
  102. if (!platform_network_supported())
  103. return false;
  104. if (password == NULL || password[0] == 0)
  105. {
  106. logmsg("Connecting to Wi-Fi SSID \"", ssid, "\" with no authentication");
  107. ret = cyw43_arch_wifi_connect_async(ssid, NULL, CYW43_AUTH_OPEN);
  108. }
  109. else
  110. {
  111. logmsg("Connecting to Wi-Fi SSID \"", ssid, "\" with WPA/WPA2 PSK");
  112. ret = cyw43_arch_wifi_connect_async(ssid, password, CYW43_AUTH_WPA2_MIXED_PSK);
  113. }
  114. if (ret != 0)
  115. {
  116. logmsg("Wi-Fi connection failed: ", ret);
  117. }
  118. else
  119. {
  120. // Short single blink at start of connection sequence
  121. PICO_W_LED_OFF();
  122. delay(PICO_W_SHORT_BLINK_DELAY);
  123. PICO_W_LED_ON();
  124. delay(PICO_W_SHORT_BLINK_DELAY);
  125. PICO_W_LED_OFF();
  126. }
  127. return (ret == 0);
  128. }
  129. void platform_network_poll()
  130. {
  131. if (!network_in_use)
  132. return;
  133. scsiNetworkPurge();
  134. cyw43_arch_poll();
  135. }
  136. int platform_network_send(uint8_t *buf, size_t len)
  137. {
  138. int ret = cyw43_send_ethernet(&cyw43_state, 0, len, buf, 0);
  139. if (ret != 0)
  140. logmsg("cyw43_send_ethernet failed: ", ret);
  141. return ret;
  142. }
  143. static int platform_network_wifi_scan_result(void *env, const cyw43_ev_scan_result_t *result)
  144. {
  145. struct wifi_network_entry *entry = NULL;
  146. if (!result || !result->ssid_len || !result->ssid[0])
  147. return 0;
  148. for (int i = 0; i < WIFI_NETWORK_LIST_ENTRY_COUNT; i++)
  149. {
  150. // take first available
  151. if (wifi_network_list[i].ssid[0] == '\0')
  152. {
  153. entry = &wifi_network_list[i];
  154. break;
  155. }
  156. // or if we've seen this network before, use this slot
  157. else if (strcmp((char *)result->ssid, wifi_network_list[i].ssid) == 0)
  158. {
  159. entry = &wifi_network_list[i];
  160. break;
  161. }
  162. }
  163. if (!entry)
  164. {
  165. // no available slots, insert according to our RSSI
  166. for (int i = 0; i < WIFI_NETWORK_LIST_ENTRY_COUNT; i++)
  167. {
  168. if (result->rssi > wifi_network_list[i].rssi)
  169. {
  170. // shift everything else down
  171. for (int j = WIFI_NETWORK_LIST_ENTRY_COUNT - 1; j > i; j--)
  172. wifi_network_list[j] = wifi_network_list[j - 1];
  173. entry = &wifi_network_list[i];
  174. memset(entry, 0, sizeof(struct wifi_network_entry));
  175. break;
  176. }
  177. }
  178. }
  179. if (entry == NULL)
  180. return 0;
  181. if (entry->rssi == 0 || result->rssi > entry->rssi)
  182. {
  183. entry->channel = result->channel;
  184. entry->rssi = result->rssi;
  185. }
  186. if (result->auth_mode & 7)
  187. entry->flags = WIFI_NETWORK_FLAG_AUTH;
  188. strncpy(entry->ssid, (const char *)result->ssid, sizeof(entry->ssid));
  189. entry->ssid[sizeof(entry->ssid) - 1] = '\0';
  190. memcpy(entry->bssid, result->bssid, sizeof(entry->bssid));
  191. return 0;
  192. }
  193. int platform_network_wifi_start_scan()
  194. {
  195. if (cyw43_wifi_scan_active(&cyw43_state))
  196. return -1;
  197. cyw43_wifi_scan_options_t scan_options = { 0 };
  198. memset(wifi_network_list, 0, sizeof(wifi_network_list));
  199. return cyw43_wifi_scan(&cyw43_state, &scan_options, NULL, platform_network_wifi_scan_result);
  200. }
  201. int platform_network_wifi_scan_finished()
  202. {
  203. return !cyw43_wifi_scan_active(&cyw43_state);
  204. }
  205. void platform_network_wifi_dump_scan_list()
  206. {
  207. struct wifi_network_entry *entry = NULL;
  208. for (int i = 0; i < WIFI_NETWORK_LIST_ENTRY_COUNT; i++)
  209. {
  210. entry = &wifi_network_list[i];
  211. if (entry->ssid[0] == '\0')
  212. break;
  213. logmsg("wifi[",i,"] = ",entry->ssid,", channel ",(int)entry->channel,", rssi ",(int)entry->rssi,
  214. ", bssid ",(uint8_t) entry->bssid[0],":",(uint8_t) entry->bssid[1],":",(uint8_t) entry->bssid[2],":",
  215. (uint8_t) entry->bssid[3],":",(uint8_t) entry->bssid[4],":",(uint8_t) entry->bssid[5],", flags ", entry->flags);
  216. }
  217. }
  218. int platform_network_wifi_rssi()
  219. {
  220. int32_t rssi = 0;
  221. cyw43_ioctl(&cyw43_state, CYW43_IOCTL_GET_RSSI, sizeof(rssi), (uint8_t *)&rssi, CYW43_ITF_STA);
  222. return rssi;
  223. }
  224. char * platform_network_wifi_ssid()
  225. {
  226. struct ssid_t {
  227. uint32_t ssid_len;
  228. uint8_t ssid[32 + 1];
  229. } ssid;
  230. static char cur_ssid[32 + 1];
  231. memset(cur_ssid, 0, sizeof(cur_ssid));
  232. int ret = cyw43_ioctl(&cyw43_state, CYW43_IOCTL_GET_SSID, sizeof(ssid), (uint8_t *)&ssid, CYW43_ITF_STA);
  233. if (ret)
  234. {
  235. logmsg("Failed getting Wi-Fi SSID: ", ret);
  236. return NULL;
  237. }
  238. ssid.ssid[sizeof(ssid.ssid) - 1] = '\0';
  239. if (ssid.ssid_len < sizeof(ssid.ssid))
  240. ssid.ssid[ssid.ssid_len] = '\0';
  241. strlcpy(cur_ssid, (char *)ssid.ssid, sizeof(cur_ssid));
  242. return cur_ssid;
  243. }
  244. char * platform_network_wifi_bssid()
  245. {
  246. static char bssid[6];
  247. memset(bssid, 0, sizeof(bssid));
  248. /* TODO */
  249. return bssid;
  250. }
  251. int platform_network_wifi_channel()
  252. {
  253. int32_t channel = 0;
  254. cyw43_ioctl(&cyw43_state, CYW43_IOCTL_GET_CHANNEL, sizeof(channel), (uint8_t *)&channel, CYW43_ITF_STA);
  255. return channel;
  256. }
  257. // these override weakly-defined functions in pico-sdk
  258. void cyw43_cb_process_ethernet(void *cb_data, int itf, size_t len, const uint8_t *buf)
  259. {
  260. scsiNetworkEnqueue(buf, len);
  261. }
  262. void cyw43_cb_tcpip_set_link_down(cyw43_t *self, int itf)
  263. {
  264. logmsg("Disassociated from Wi-Fi SSID \"", (char *)self->ap_ssid,"\"");
  265. }
  266. void cyw43_cb_tcpip_set_link_up(cyw43_t *self, int itf)
  267. {
  268. char *ssid = platform_network_wifi_ssid();
  269. if (ssid)
  270. {
  271. logmsg("Successfully connected to Wi-Fi SSID \"",ssid,"\"");
  272. // blink LED 3 times when connected
  273. PICO_W_LED_OFF();
  274. for (uint8_t i = 0; i < 3; i++)
  275. {
  276. delay(PICO_W_SHORT_BLINK_DELAY);
  277. PICO_W_LED_ON();
  278. delay(PICO_W_SHORT_BLINK_DELAY);
  279. PICO_W_LED_OFF();
  280. }
  281. }
  282. }
  283. }
  284. #endif // ZULUSCSI_NETWORK