1 /* 2 * Copyright (c) 2020 Nordic Semiconductor ASA 3 * 4 * SPDX-License-Identifier: Apache-2.0 5 */ 6 7 #ifndef ZEPHYR_INCLUDE_BLUETOOTH_MESH_BLOB_CLI_H_ 8 #define ZEPHYR_INCLUDE_BLUETOOTH_MESH_BLOB_CLI_H_ 9 10 #include <sys/types.h> 11 12 #include <zephyr/bluetooth/mesh/access.h> 13 #include <zephyr/bluetooth/mesh/blob.h> 14 15 #ifdef __cplusplus 16 extern "C" { 17 #endif 18 19 /** 20 * @defgroup bt_mesh_blob_cli Bluetooth Mesh BLOB Transfer Client model API 21 * @ingroup bt_mesh 22 * @{ 23 */ 24 25 struct bt_mesh_blob_cli; 26 27 /** 28 * 29 * @brief BLOB Transfer Client model Composition Data entry. 30 * 31 * @param _cli Pointer to a @ref bt_mesh_blob_cli instance. 32 */ 33 #define BT_MESH_MODEL_BLOB_CLI(_cli) \ 34 BT_MESH_MODEL_CB(BT_MESH_MODEL_ID_BLOB_CLI, _bt_mesh_blob_cli_op, \ 35 NULL, _cli, &_bt_mesh_blob_cli_cb) 36 37 /** Target node's Pull mode (Pull BLOB Transfer Mode) context used 38 * while sending chunks to the Target node. 39 */ 40 struct bt_mesh_blob_target_pull { 41 /** Timestamp when the Block Report Timeout Timer expires for this Target node. */ 42 int64_t block_report_timestamp; 43 44 /** Missing chunks reported by this Target node. */ 45 uint8_t missing[DIV_ROUND_UP(CONFIG_BT_MESH_BLOB_CHUNK_COUNT_MAX, 8)]; 46 }; 47 48 /** BLOB Transfer Client Target node. */ 49 struct bt_mesh_blob_target { 50 /** Linked list node */ 51 sys_snode_t n; 52 53 /** Target node address. */ 54 uint16_t addr; 55 56 /** Target node's Pull mode context. 57 * Needs to be initialized when sending a BLOB in Pull mode. 58 */ 59 struct bt_mesh_blob_target_pull *pull; 60 61 /** BLOB transfer status, see @ref bt_mesh_blob_status. */ 62 uint8_t status; 63 64 uint8_t procedure_complete:1, /* Procedure has been completed. */ 65 acked:1, /* Message has been acknowledged. Not used when sending. */ 66 timedout:1, /* Target node didn't respond after specified timeout. */ 67 skip:1; /* Skip Target node from broadcast. */ 68 }; 69 70 /** BLOB transfer information. 71 * 72 * If @c phase is @ref BT_MESH_BLOB_XFER_PHASE_INACTIVE, 73 * the fields below @c phase are not initialized. 74 * If @c phase is @ref BT_MESH_BLOB_XFER_PHASE_WAITING_FOR_START, 75 * the fields below @c id are not initialized. 76 */ 77 struct bt_mesh_blob_xfer_info { 78 /** BLOB transfer status. */ 79 enum bt_mesh_blob_status status; 80 81 /** BLOB transfer mode. */ 82 enum bt_mesh_blob_xfer_mode mode; 83 84 /** BLOB transfer phase. */ 85 enum bt_mesh_blob_xfer_phase phase; 86 87 /** BLOB ID. */ 88 uint64_t id; 89 90 /** BLOB size in octets. */ 91 uint32_t size; 92 93 /** Logarithmic representation of the block size. */ 94 uint8_t block_size_log; 95 96 /** MTU size in octets. */ 97 uint16_t mtu_size; 98 99 /** Bit field indicating blocks that were not received. */ 100 const uint8_t *missing_blocks; 101 }; 102 103 /** BLOB Transfer Client transfer inputs. */ 104 struct bt_mesh_blob_cli_inputs { 105 /** Linked list of Target nodes. Each node should point to @ref 106 * bt_mesh_blob_target::n. 107 */ 108 sys_slist_t targets; 109 110 /** AppKey index to send with. */ 111 uint16_t app_idx; 112 113 /** Group address destination for the BLOB transfer, or @ref 114 * BT_MESH_ADDR_UNASSIGNED to send every message to each Target 115 * node individually. 116 */ 117 uint16_t group; 118 119 /** Time to live value of BLOB transfer messages. */ 120 uint8_t ttl; 121 122 /** Additional response time for the Target nodes, in 10-second increments. 123 * 124 * The extra time can be used to give the Target nodes more time to respond 125 * to messages from the Client. The actual timeout will be calculated 126 * according to the following formula: 127 * 128 * @verbatim 129 * timeout = 20 seconds + (10 seconds * timeout_base) + (100 ms * TTL) 130 * @endverbatim 131 * 132 * If a Target node fails to respond to a message from the Client within the 133 * configured transfer timeout, the Target node is dropped. 134 */ 135 uint16_t timeout_base; 136 }; 137 138 /** Transfer capabilities of a Target node. */ 139 struct bt_mesh_blob_cli_caps { 140 /** Max BLOB size. */ 141 size_t max_size; 142 143 /** Logarithmic representation of the minimum block size. */ 144 uint8_t min_block_size_log; 145 146 /** Logarithmic representation of the maximum block size. */ 147 uint8_t max_block_size_log; 148 149 /** Max number of chunks per block. */ 150 uint16_t max_chunks; 151 152 /** Max chunk size. */ 153 uint16_t max_chunk_size; 154 155 /** Max MTU size. */ 156 uint16_t mtu_size; 157 158 /** Supported transfer modes. */ 159 enum bt_mesh_blob_xfer_mode modes; 160 }; 161 162 /** BLOB Transfer Client state. */ 163 enum bt_mesh_blob_cli_state { 164 /** No transfer is active. */ 165 BT_MESH_BLOB_CLI_STATE_NONE, 166 /** Retrieving transfer capabilities. */ 167 BT_MESH_BLOB_CLI_STATE_CAPS_GET, 168 /** Sending transfer start. */ 169 BT_MESH_BLOB_CLI_STATE_START, 170 /** Sending block start. */ 171 BT_MESH_BLOB_CLI_STATE_BLOCK_START, 172 /** Sending block chunks. */ 173 BT_MESH_BLOB_CLI_STATE_BLOCK_SEND, 174 /** Checking block status. */ 175 BT_MESH_BLOB_CLI_STATE_BLOCK_CHECK, 176 /** Checking transfer status. */ 177 BT_MESH_BLOB_CLI_STATE_XFER_CHECK, 178 /** Cancelling transfer. */ 179 BT_MESH_BLOB_CLI_STATE_CANCEL, 180 /** Transfer is suspended. */ 181 BT_MESH_BLOB_CLI_STATE_SUSPENDED, 182 /** Checking transfer progress. */ 183 BT_MESH_BLOB_CLI_STATE_XFER_PROGRESS_GET, 184 }; 185 186 /** Event handler callbacks for the BLOB Transfer Client model. 187 * 188 * All handlers are optional. 189 */ 190 struct bt_mesh_blob_cli_cb { 191 /** @brief Capabilities retrieval completion callback. 192 * 193 * Called when the capabilities retrieval procedure completes, indicating that 194 * a common set of acceptable transfer parameters have been established 195 * for the given list of Target nodes. All compatible Target nodes have 196 * status code @ref BT_MESH_BLOB_SUCCESS. 197 * 198 * @param cli BLOB Transfer Client instance. 199 * @param caps Safe transfer capabilities if the transfer capabilities 200 * of at least one Target node has satisfied the Client, or NULL otherwise. 201 */ 202 void (*caps)(struct bt_mesh_blob_cli *cli, 203 const struct bt_mesh_blob_cli_caps *caps); 204 205 /** @brief Target node loss callback. 206 * 207 * Called whenever a Target node has been lost due to some error in the 208 * transfer. Losing a Target node is not considered a fatal error for 209 * the Client until all Target nodes have been lost. 210 * 211 * @param cli BLOB Transfer Client instance. 212 * @param target Target node that was lost. 213 * @param reason Reason for the Target node loss. 214 */ 215 void (*lost_target)(struct bt_mesh_blob_cli *cli, 216 struct bt_mesh_blob_target *target, 217 enum bt_mesh_blob_status reason); 218 219 /** @brief Transfer is suspended. 220 * 221 * Called when the transfer is suspended due to response timeout from all Target nodes. 222 * 223 * @param cli BLOB Transfer Client instance. 224 */ 225 void (*suspended)(struct bt_mesh_blob_cli *cli); 226 227 /** @brief Transfer end callback. 228 * 229 * Called when the transfer ends. 230 * 231 * @param cli BLOB Transfer Client instance. 232 * @param xfer Completed transfer. 233 * @param success Status of the transfer. 234 * Is @c true if at least one Target 235 * node received the whole transfer. 236 */ 237 void (*end)(struct bt_mesh_blob_cli *cli, 238 const struct bt_mesh_blob_xfer *xfer, bool success); 239 240 /** @brief Transfer progress callback 241 * 242 * The content of @c info is invalidated upon exit from the callback. 243 * Therefore it needs to be copied if it is planned to be used later. 244 * 245 * @param cli BLOB Transfer Client instance. 246 * @param target Target node that responded to the request. 247 * @param info BLOB transfer information. 248 */ 249 void (*xfer_progress)(struct bt_mesh_blob_cli *cli, 250 struct bt_mesh_blob_target *target, 251 const struct bt_mesh_blob_xfer_info *info); 252 253 /** @brief End of Get Transfer Progress procedure. 254 * 255 * Called when all Target nodes have responded or the procedure timed-out. 256 * 257 * @param cli BLOB Transfer Client instance. 258 */ 259 void (*xfer_progress_complete)(struct bt_mesh_blob_cli *cli); 260 }; 261 262 /** @cond INTERNAL_HIDDEN */ 263 struct blob_cli_broadcast_ctx { 264 /** Called for every Target node in unicast mode, or once in case of multicast mode. */ 265 void (*send)(struct bt_mesh_blob_cli *cli, uint16_t dst); 266 /** Called after every @ref blob_cli_broadcast_ctx::send callback. */ 267 void (*send_complete)(struct bt_mesh_blob_cli *cli, uint16_t dst); 268 /** If @ref blob_cli_broadcast_ctx::acked is true, called after all Target nodes 269 * have confirmed reception by @ref blob_cli_broadcast_rsp. Otherwise, called 270 * after transmission has been completed. 271 */ 272 void (*next)(struct bt_mesh_blob_cli *cli); 273 /** If true, every transmission needs to be confirmed by @ref blob_cli_broadcast_rsp before 274 * @ref blob_cli_broadcast_ctx::next is called. 275 */ 276 bool acked; 277 /** If true, the message is always sent in a unicast way. */ 278 bool force_unicast; 279 /** If true, non-responsive Target nodes won't be dropped after transfer has timed out. */ 280 bool optional; 281 /** Set to true by the BLOB Transfer Client between blob_cli_broadcast 282 * and broadcast_complete calls. 283 */ 284 bool is_inited; 285 /* Defines a time in ms by which the broadcast API postpones sending the message to a next 286 * target or completing the broadcast. 287 */ 288 uint32_t post_send_delay_ms; 289 }; 290 /** INTERNAL_HIDDEN @endcond */ 291 292 /** BLOB Transfer Client model instance. */ 293 struct bt_mesh_blob_cli { 294 /** Event handler callbacks */ 295 const struct bt_mesh_blob_cli_cb *cb; 296 297 /* Runtime state */ 298 const struct bt_mesh_model *mod; 299 300 struct { 301 struct bt_mesh_blob_target *target; 302 struct blob_cli_broadcast_ctx ctx; 303 struct k_work_delayable retry; 304 /* Represents Client Timeout timer in a timestamp. Used in Pull mode only. */ 305 int64_t cli_timestamp; 306 struct k_work_delayable complete; 307 uint16_t pending; 308 uint8_t retries; 309 uint8_t sending : 1, 310 cancelled : 1; 311 } tx; 312 313 const struct bt_mesh_blob_io *io; 314 const struct bt_mesh_blob_cli_inputs *inputs; 315 const struct bt_mesh_blob_xfer *xfer; 316 uint32_t chunk_interval_ms; 317 uint16_t block_count; 318 uint16_t chunk_idx; 319 uint16_t mtu_size; 320 enum bt_mesh_blob_cli_state state; 321 struct bt_mesh_blob_block block; 322 struct bt_mesh_blob_cli_caps caps; 323 }; 324 325 /** @brief Retrieve transfer capabilities for a list of Target nodes. 326 * 327 * Queries the availability and capabilities of all Target nodes, producing a 328 * cumulative set of transfer capabilities for the Target nodes, and returning 329 * it through the @ref bt_mesh_blob_cli_cb::caps callback. 330 * 331 * Retrieving the capabilities may take several seconds, depending on the 332 * number of Target nodes and mesh network performance. The end of the procedure 333 * is indicated through the @ref bt_mesh_blob_cli_cb::caps callback. 334 * 335 * This procedure is not required, but strongly recommended as a 336 * preparation for a transfer to maximize performance and the chances of 337 * success. 338 * 339 * @param cli BLOB Transfer Client instance. 340 * @param inputs Statically allocated BLOB Transfer Client transfer inputs. 341 * 342 * @return 0 on success, or (negative) error code otherwise. 343 */ 344 int bt_mesh_blob_cli_caps_get(struct bt_mesh_blob_cli *cli, 345 const struct bt_mesh_blob_cli_inputs *inputs); 346 347 /** @brief Perform a BLOB transfer. 348 * 349 * Starts sending the transfer to the Target nodes. Only Target nodes with a 350 * @c status of @ref BT_MESH_BLOB_SUCCESS will be considered. 351 * 352 * The transfer will keep going either until all Target nodes have been dropped, or 353 * the full BLOB has been sent. 354 * 355 * The BLOB transfer may take several minutes, depending on the number of 356 * Target nodes, size of the BLOB and mesh network performance. The end of the 357 * transfer is indicated through the @ref bt_mesh_blob_cli_cb::end callback. 358 * 359 * A Client only supports one transfer at the time. 360 * 361 * @param cli BLOB Transfer Client instance. 362 * @param inputs Statically allocated BLOB Transfer Client transfer inputs. 363 * @param xfer Statically allocated transfer parameters. 364 * @param io BLOB stream to read the transfer from. 365 * 366 * @return 0 on success, or (negative) error code otherwise. 367 */ 368 int bt_mesh_blob_cli_send(struct bt_mesh_blob_cli *cli, 369 const struct bt_mesh_blob_cli_inputs *inputs, 370 const struct bt_mesh_blob_xfer *xfer, 371 const struct bt_mesh_blob_io *io); 372 373 /** @brief Suspend the active transfer. 374 * 375 * @param cli BLOB Transfer Client instance. 376 * 377 * @return 0 on success, or (negative) error code otherwise. 378 */ 379 int bt_mesh_blob_cli_suspend(struct bt_mesh_blob_cli *cli); 380 381 /** @brief Resume the suspended transfer. 382 * 383 * @param cli BLOB Transfer Client instance. 384 * 385 * @return 0 on success, or (negative) error code otherwise. 386 */ 387 int bt_mesh_blob_cli_resume(struct bt_mesh_blob_cli *cli); 388 389 /** @brief Cancel an ongoing transfer. 390 * 391 * @param cli BLOB Transfer Client instance. 392 */ 393 void bt_mesh_blob_cli_cancel(struct bt_mesh_blob_cli *cli); 394 395 /** @brief Get the progress of BLOB transfer. 396 * 397 * This function can only be used if the BLOB Transfer Client is currently 398 * not performing a BLOB transfer. 399 * To get progress of the active BLOB transfer, use the 400 * @ref bt_mesh_blob_cli_xfer_progress_active_get function. 401 * 402 * @param cli BLOB Transfer Client instance. 403 * @param inputs Statically allocated BLOB Transfer Client transfer inputs. 404 * 405 * @return 0 on success, or (negative) error code otherwise. 406 */ 407 int bt_mesh_blob_cli_xfer_progress_get(struct bt_mesh_blob_cli *cli, 408 const struct bt_mesh_blob_cli_inputs *inputs); 409 410 /** @brief Get the current progress of the active transfer in percent. 411 * 412 * @param cli BLOB Transfer Client instance. 413 * 414 * @return The current transfer progress, or 0 if no transfer is active. 415 */ 416 uint8_t bt_mesh_blob_cli_xfer_progress_active_get(struct bt_mesh_blob_cli *cli); 417 418 /** @brief Get the current state of the BLOB Transfer Client. 419 * 420 * @param cli BLOB Transfer Client instance. 421 * 422 * @return true if the BLOB Transfer Client is currently participating in a transfer or 423 * retrieving the capabilities and false otherwise. 424 */ 425 bool bt_mesh_blob_cli_is_busy(struct bt_mesh_blob_cli *cli); 426 427 /** @brief Set chunk sending interval in ms 428 * 429 * This function is optional, and can be used to define how fast chunks are sent in the BLOB Client 430 * Model. 431 * Without an added delay, for example a Bluetooth Mesh DFU can cause network blockage by 432 * constantly sending the next chunks, especially if the chunks are sent to group addresses or 433 * multiple unicast addresses. 434 * 435 * @note: Big intervals may cause timeouts. Increasing the @c timeout_base accordingly can 436 * circumvent this. 437 * 438 * @param cli BLOB Transfer Client instance. 439 * @param interval_ms the delay before each chunk is sent out in ms. 440 */ 441 void bt_mesh_blob_cli_set_chunk_interval_ms(struct bt_mesh_blob_cli *cli, uint32_t interval_ms); 442 443 /** @cond INTERNAL_HIDDEN */ 444 extern const struct bt_mesh_model_op _bt_mesh_blob_cli_op[]; 445 extern const struct bt_mesh_model_cb _bt_mesh_blob_cli_cb; 446 /** @endcond */ 447 448 /** @} */ 449 450 #ifdef __cplusplus 451 } 452 #endif 453 454 #endif /* ZEPHYR_INCLUDE_BLUETOOTH_MESH_BLOB_CLI_H_ */ 455