1 /* 2 * Copyright (c) 2022 G-Technologies Sdn. Bhd. 3 * 4 * SPDX-License-Identifier: Apache-2.0 5 */ 6 7 #ifndef ZEPHYR_INCLUDE_SHELL_MQTT_H_ 8 #define ZEPHYR_INCLUDE_SHELL_MQTT_H_ 9 10 #include <zephyr/kernel.h> 11 #include <zephyr/shell/shell.h> 12 #include <zephyr/net/socket.h> 13 #include <zephyr/net/net_mgmt.h> 14 #include <zephyr/net/net_event.h> 15 #include <zephyr/net/conn_mgr_monitor.h> 16 #include <zephyr/net/mqtt.h> 17 #include <zephyr/sys/ring_buffer.h> 18 19 #ifdef __cplusplus 20 extern "C" { 21 #endif 22 23 #define RX_RB_SIZE CONFIG_SHELL_MQTT_RX_BUF_SIZE 24 #define TX_BUF_SIZE CONFIG_SHELL_MQTT_TX_BUF_SIZE 25 #define SH_MQTT_BUFFER_SIZE 64 26 #define DEVICE_ID_BIN_MAX_SIZE 3 27 #define DEVICE_ID_HEX_MAX_SIZE ((DEVICE_ID_BIN_MAX_SIZE * 2) + 1) 28 #define SH_MQTT_TOPIC_MAX_SIZE DEVICE_ID_HEX_MAX_SIZE + 3 29 30 extern const struct shell_transport_api shell_mqtt_transport_api; 31 32 struct shell_mqtt_tx_buf { 33 /** tx buffer. */ 34 char buf[TX_BUF_SIZE]; 35 36 /** Current tx buf length. */ 37 uint16_t len; 38 }; 39 40 /** MQTT-based shell transport. */ 41 struct shell_mqtt { 42 char device_id[DEVICE_ID_HEX_MAX_SIZE]; 43 char sub_topic[SH_MQTT_TOPIC_MAX_SIZE]; 44 char pub_topic[SH_MQTT_TOPIC_MAX_SIZE]; 45 46 /** Handler function registered by shell. */ 47 shell_transport_handler_t shell_handler; 48 49 struct ring_buf rx_rb; 50 uint8_t rx_rb_buf[RX_RB_SIZE]; 51 uint8_t *rx_rb_ptr; 52 53 struct shell_mqtt_tx_buf tx_buf; 54 55 /** Context registered by shell. */ 56 void *shell_context; 57 58 /** The mqtt client struct */ 59 struct mqtt_client mqtt_cli; 60 61 /* Buffers for MQTT client. */ 62 struct buffer { 63 uint8_t rx[SH_MQTT_BUFFER_SIZE]; 64 uint8_t tx[SH_MQTT_BUFFER_SIZE]; 65 } buf; 66 67 struct k_mutex lock; 68 69 /** MQTT Broker details. */ 70 struct sockaddr_storage broker; 71 72 struct zsock_addrinfo *haddr; 73 struct zsock_pollfd fds[1]; 74 int nfds; 75 76 struct mqtt_publish_param pub_data; 77 78 struct net_mgmt_event_callback mgmt_cb; 79 80 /** work */ 81 struct k_work_q workq; 82 struct k_work net_disconnected_work; 83 struct k_work_delayable connect_dwork; 84 struct k_work_delayable subscribe_dwork; 85 struct k_work_delayable process_dwork; 86 struct k_work_delayable publish_dwork; 87 88 /** MQTT connection states */ 89 enum sh_mqtt_transport_state { 90 SHELL_MQTT_TRANSPORT_DISCONNECTED, 91 SHELL_MQTT_TRANSPORT_CONNECTED, 92 } transport_state; 93 94 /** MQTT subscription states */ 95 enum sh_mqtt_subscribe_state { 96 SHELL_MQTT_NOT_SUBSCRIBED, 97 SHELL_MQTT_SUBSCRIBED, 98 } subscribe_state; 99 100 /** Network states */ 101 enum sh_mqtt_network_state { 102 SHELL_MQTT_NETWORK_DISCONNECTED, 103 SHELL_MQTT_NETWORK_CONNECTED, 104 } network_state; 105 }; 106 107 #define SHELL_MQTT_DEFINE(_name) \ 108 static struct shell_mqtt _name##_shell_mqtt; \ 109 struct shell_transport _name = { .api = &shell_mqtt_transport_api, \ 110 .ctx = (struct shell_mqtt *)&_name##_shell_mqtt } 111 112 /** 113 * @brief This function provides pointer to shell mqtt backend instance. 114 * 115 * Function returns pointer to the shell mqtt instance. This instance can be 116 * next used with shell_execute_cmd function in order to test commands behavior. 117 * 118 * @returns Pointer to the shell instance. 119 */ 120 const struct shell *shell_backend_mqtt_get_ptr(void); 121 122 /** 123 * @brief Function to define the device ID (devid) for which the shell mqtt backend uses as a 124 * client ID when it connects to the broker. It will publish its output to devid_tx and subscribe 125 * to devid_rx for input . 126 * 127 * @note This is a weak-linked function, and can be overridden if desired. 128 * 129 * @param id Pointer to the devid buffer 130 * @param id_max_len Maximum size of the devid buffer defined by DEVICE_ID_HEX_MAX_SIZE 131 * 132 * @return true if length of devid > 0 133 */ 134 bool shell_mqtt_get_devid(char *id, int id_max_len); 135 136 #ifdef __cplusplus 137 } 138 #endif 139 140 #endif /* ZEPHYR_INCLUDE_SHELL_MQTT_H_ */ 141