1 /* 2 * Copyright (c) 2017 Linaro Limited 3 * Copyright (c) 2018-2019 Foundries.io 4 * 5 * SPDX-License-Identifier: Apache-2.0 6 */ 7 8 #ifndef LWM2M_ENGINE_H 9 #define LWM2M_ENGINE_H 10 11 #include "lwm2m_message_handling.h" 12 #include "lwm2m_object.h" 13 #include "lwm2m_observation.h" 14 #include "lwm2m_registry.h" 15 #include <zephyr/kernel.h> 16 17 #define LWM2M_PROTOCOL_VERSION_MAJOR 1 18 #if defined(CONFIG_LWM2M_VERSION_1_1) 19 #define LWM2M_PROTOCOL_VERSION_MINOR 1 20 #else 21 #define LWM2M_PROTOCOL_VERSION_MINOR 0 22 #endif 23 24 #define LWM2M_PROTOCOL_VERSION_STRING \ 25 STRINGIFY(LWM2M_PROTOCOL_VERSION_MAJOR) \ 26 "." STRINGIFY(LWM2M_PROTOCOL_VERSION_MINOR) 27 28 /* Use this value to generate new token */ 29 #define LWM2M_MSG_TOKEN_GENERATE_NEW 0xFFU 30 31 /* length of time in milliseconds to wait for buffer allocations */ 32 #define BUF_ALLOC_TIMEOUT K_SECONDS(1) 33 34 /** initialization function */ 35 struct lwm2m_init_func { 36 int (*f)(void); 37 }; 38 /** 39 * @defgroup LWM2M_PRIO LwM2M initialization priorities 40 * @{ 41 */ 42 #define LWM2M_PRIO_ENGINE 0 /**< Engine initialization */ 43 #define LWM2M_PRIO_CORE 1 /**< Core object initialization */ 44 #define LWM2M_PRIO_OBJ 2 /**< Object initializations */ 45 #define LwM2M_PRIO_APP 3 /**< Application logic initialization */ 46 /** @} */ 47 48 /** 49 * @brief Declare an initialization function to be executed when LwM2M engine starts. 50 * 51 * When LwM2M engine starts up, it first executes all initialization functions in following 52 * priority order: 53 * 1. LWM2M_PRIO_ENGINE 54 * 2. LWM2M_PRIO_CORE, this is where all LwM2M core objects are initialized 55 * 3. LWM2M_PRIO_OBJ, this is where all other than core objects are initialized 56 * 4. LwM2M_PRIO_APP, application initialization. 57 * For example create sensor objects, and register object callbacks. 58 * 59 * @param[in] prio Priority, one of @ref LWM2M_PRIO macros. 60 * @param[in] init_function Initialization function 61 */ 62 #define LWM2M_ON_INIT(prio, init_function) \ 63 STRUCT_SECTION_ITERABLE(lwm2m_init_func, \ 64 CONCAT(LWM2M, prio, init_function)) = {.f = init_function} 65 66 /** 67 * @brief Declare engine initialization function. 68 * @sa LWM2M_ON_INIT 69 * @param[in] init_function Initialization function 70 */ 71 #define LWM2M_ENGINE_INIT(init_function) LWM2M_ON_INIT(LWM2M_PRIO_ENGINE, init_function) 72 73 /** 74 * @brief Declare core object initialization function. 75 * @sa LWM2M_ON_INIT 76 * @param[in] init_function Initialization function 77 */ 78 #define LWM2M_CORE_INIT(init_function) LWM2M_ON_INIT(LWM2M_PRIO_CORE, init_function) 79 80 /** 81 * @brief Declare object initialization function. 82 * @sa LWM2M_ON_INIT 83 * @param[in] init_function Initialization function 84 */ 85 #define LWM2M_OBJ_INIT(init_function) LWM2M_ON_INIT(LWM2M_PRIO_OBJ, init_function) 86 87 /** 88 * @brief Declare application specific initialization function. 89 * @sa LWM2M_ON_INIT 90 * @param[in] init_function Initialization function 91 */ 92 #define LWM2M_APP_INIT(init_function) LWM2M_ON_INIT(LWM2M_PRIO_APP, init_function) 93 94 /** 95 * @brief Validates that writing is a legal operation on the field given by the object in 96 * @p obj_inst and the resource id in @p msg. Returns the field to obj_field (if it exists). 97 * 98 * @param[in] msg lwm2m message to signal for which resource the write access should checked 99 * @param[in] obj_inst Engine object instance to signal which object the resource belongs to 100 * @param[out] obj_field Engine object field buffer pointer to store the field being checked 101 * @return 0 for successful validation and negative in all other cases 102 */ 103 int lwm2m_engine_validate_write_access(struct lwm2m_message *msg, 104 struct lwm2m_engine_obj_inst *obj_inst, 105 struct lwm2m_engine_obj_field **obj_field); 106 107 /* LwM2M context functions */ 108 /** 109 * @brief Removes all observes, clears all pending coap messages and resets all messages 110 * associated with this context. 111 * 112 * @param[in] client_ctx Context to close 113 */ 114 void lwm2m_engine_context_close(struct lwm2m_ctx *client_ctx); 115 116 /** 117 * @brief Initializes a lwm2m_ctx before a socket can be associated with it. 118 * 119 * @param[in] client_ctx Context to initialize 120 */ 121 void lwm2m_engine_context_init(struct lwm2m_ctx *client_ctx); 122 123 /* Message buffer functions */ 124 uint8_t *lwm2m_get_message_buf(void); 125 int lwm2m_put_message_buf(uint8_t *buf); 126 int lwm2m_perform_composite_observation_op(struct lwm2m_message *msg, uint8_t *token, 127 uint8_t token_length, sys_slist_t *lwm2m_path_list); 128 129 /** 130 * @brief Checks if context has access to the object specified by @p path 131 * 132 * @param[in] client_ctx lwm2m ctx of the connection 133 * @param[in] path Path of the object 134 * @return Returns false if the bootstrap flag of @p client_ctx is not set 135 */ 136 bool lwm2m_engine_bootstrap_override(struct lwm2m_ctx *client_ctx, struct lwm2m_obj_path *path); 137 138 /** 139 * @brief Deletes the object instance specified by msg->path. If the object instance id of the 140 * path is null (i.e path.level == 1), all object instances of that object will be deleted. 141 * Should only be called if delete request comes from a bootstrap server. 142 * 143 * @param[in] msg lwm2m message with operation delete 144 * @return 0 for success or negative in case of error 145 */ 146 int bootstrap_delete(struct lwm2m_message *msg); 147 148 /** 149 * @brief Adds the periodic @p service to the work queue. The period of the service becomes 150 * @p period_ms. 151 * 152 * @param[in] service Periodic service to be added 153 * @param[in] period_ms Period of the periodic service 154 * @return 0 for success or negative in case of error 155 */ 156 int lwm2m_engine_add_service(k_work_handler_t service, uint32_t period_ms); 157 158 /** 159 * @brief Update the period of a given service or remove it. 160 * 161 * Allow the period modification on an existing service created with 162 * lwm2m_engine_add_service(). When period is zero, service is removed. 163 * 164 * @param[in] service Handler of the periodic_service 165 * @param[in] period_ms New period for the periodic_service (in milliseconds) or zero. 166 * 167 * @return 0 for success, 1 when service was removed or negative in case of error. 168 */ 169 int lwm2m_engine_update_service_period(k_work_handler_t service, uint32_t period_ms); 170 171 /** 172 * @brief Call specific service handler only once at given timestamp. 173 * 174 * @param[in] service service to be called 175 * @param[in] timestamp Time when to call 176 * @return 0 for success or negative in case of error 177 */ 178 int lwm2m_engine_call_at(k_work_handler_t service, int64_t timestamp); 179 180 /** 181 * @brief Call given handler from engine context. 182 * 183 * @param[in] service Service callback to be called. 184 * @return 0 for success or negative in case of error 185 */ 186 int lwm2m_engine_call_now(k_work_handler_t service); 187 188 /** 189 * @brief Returns the index in the security objects list corresponding to the object instance 190 * id given by @p obj_inst_id 191 * 192 * @param[in] obj_inst_id object instance id of the security instance 193 * @return index in the list or negative in case of error 194 */ 195 int lwm2m_security_inst_id_to_index(uint16_t obj_inst_id); 196 197 /** 198 * @brief Returns the object instance id of the security having ssid given by @p short_id. 199 * 200 * @param[in] short_id ssid of the security object 201 * @return Object instance id or negative in case not found 202 */ 203 int lwm2m_security_short_id_to_inst(uint16_t short_id); 204 205 /** 206 * @brief Returns the object instance id of the security object instance at @p index 207 * in the security object list. 208 * 209 * @param[in] index Index in the security object list 210 * @return Object instance id of the security instance or negative in case of error 211 */ 212 int lwm2m_security_index_to_inst_id(int index); 213 214 #if defined(CONFIG_LWM2M_FIRMWARE_UPDATE_OBJ_SUPPORT) 215 /** 216 * @brief Sets the update state (as specified in LWM2M SPEC E.6 regarding the firmware update) 217 * of the Firmware object instance given by @p obj_inst_id to @p state. 218 * (i.e Sets the value of resource 5/ @p obj_inst_id /3) 219 * 220 * @param[in] obj_inst_id Object instance id of the firmware object instance 221 * @param[in] state (STATE_IDLE through STATE_UPDATING) 222 */ 223 void lwm2m_firmware_set_update_state_inst(uint16_t obj_inst_id, uint8_t state); 224 225 /** 226 * @brief Sets the result state (as specified in LWM2M SPEC E.6 regarding the firmware update) 227 * of the Firmware object instance given by @p obj_inst_id to @p result. 228 * (i.e Sets the value of resource 5/ @p obj_inst_id /5) 229 * 230 * @param[in] obj_inst_id Object instance id of the firmware object instance 231 * @param[in] result (RESULT_DEFAULT through RESULT_UNSUP_PROTO) 232 */ 233 void lwm2m_firmware_set_update_result_inst(uint16_t obj_inst_id, uint8_t result); 234 235 /** 236 * @brief Equivalent to lwm2m_firmware_set_update_state_inst(0, state). 237 * 238 * @param[in] state (STATE_IDLE through STATE_UPDATING) 239 */ 240 void lwm2m_firmware_set_update_state(uint8_t state); 241 242 /** 243 * @brief Equivalent to lwm2m_firmware_set_update_result_inst(0, result). 244 * 245 * @param[in] result (RESULT_DEFAULT through RESULT_UNSUP_PROTO) 246 */ 247 void lwm2m_firmware_set_update_result(uint8_t result); 248 249 /** 250 * @brief Returns the update state (as specified in LWM2M SPEC E.6 regarding the firmware update) 251 * of the Firmware object instance given by @p obj_inst_id. 252 * (i.e Gets the value of resource 5/ @p obj_inst_id /3) 253 * @param[in] obj_inst_id Object instance id of the firmware object 254 * @return (STATE_IDLE through STATE_UPDATING) 255 */ 256 uint8_t lwm2m_firmware_get_update_state_inst(uint16_t obj_inst_id); 257 258 /** 259 * @brief Equivalent to lwm2m_firmware_get_update_state_inst(0). 260 * 261 * @return (STATE_IDLE through STATE_UPDATING) 262 */ 263 uint8_t lwm2m_firmware_get_update_state(void); 264 265 /** 266 * @brief Returns the result state (as specified in LWM2M SPEC E.6 regarding the firmware update) 267 * of the Firmware object instance given by @p obj_inst_id. 268 * (i.e Gets the value of resource 5/ @p obj_inst_id /5) 269 * @param[in] obj_inst_id Object instance id of the firmware object 270 * @return (RESULT_DEFAULT through RESULT_UNSUP_PROTO) 271 */ 272 uint8_t lwm2m_firmware_get_update_result_inst(uint16_t obj_inst_id); 273 274 /** 275 * @brief Equivalent to lwm2m_firmware_get_update_result_inst(0). 276 * 277 * @return (RESULT_DEFAULT through RESULT_UNSUP_PROTO) 278 */ 279 uint8_t lwm2m_firmware_get_update_result(void); 280 #endif 281 282 /* Network Layer */ 283 /** 284 * @brief Opens a socket for the client_ctx if it does not exist. Saves the 285 * socket file descriptor to client_ctx->sock_fd. 286 * 287 * @param[in] client_ctx lwm2m context to open a socket for 288 * @return 0 for success or negative in case of error 289 */ 290 int lwm2m_open_socket(struct lwm2m_ctx *client_ctx); 291 292 /** 293 * @brief Closes the socket with the file descriptor given by client_ctx->sock_fd. 294 * Does nothing if the context's socket is already closed. 295 * 296 * @param[in, out] client_ctx lwm2m context whose socket is to be closed 297 * @return 0 for success or negative in case of error 298 */ 299 int lwm2m_close_socket(struct lwm2m_ctx *client_ctx); 300 301 /** 302 * @brief Removes the socket with the file descriptor given by client_ctx->sock_fd from the 303 * lwm2m work loop. Keeps the socket open. 304 * 305 * @param[in, out] client_ctx lwm2m context whose socket is being suspended 306 * @return 0 for success or negative in case of error 307 */ 308 int lwm2m_socket_suspend(struct lwm2m_ctx *client_ctx); 309 /** 310 * @brief Adds an existing socket to the lwm2m work loop. (The socket specified by the file 311 * descriptor in ctx->sock_fd) 312 * 313 * @param[in] ctx lwm2m context being added to the lwm2m work loop 314 * @return 0 for success or negative in case of error 315 */ 316 int lwm2m_socket_add(struct lwm2m_ctx *ctx); 317 318 /** 319 * @brief Removes a socket from the lwm2m work loop. 320 * 321 * @param[in] ctx lwm2m context to be removed from the lwm2m work loop 322 */ 323 void lwm2m_socket_del(struct lwm2m_ctx *ctx); 324 325 /** 326 * @brief Creates a socket for the @p client_ctx (if it does not exist) and adds it to 327 * the lwm2m work loop. 328 * 329 * @param client_ctx lwm2m context that the socket is being added for 330 * @return 0 for success or negative in case of error 331 */ 332 int lwm2m_socket_start(struct lwm2m_ctx *client_ctx); 333 334 /** 335 * @brief Closes the socket with file descriptor given by client_ctx->sock_fd. 336 * 337 * @param client_ctx Context whose socket is being closed 338 * @return 0 for success or negative in case of error 339 */ 340 int lwm2m_socket_close(struct lwm2m_ctx *client_ctx); 341 342 /** 343 * @brief Closes the socket connection when queue mode is enabled. 344 * 345 * @param[in, out] client_ctx lwm2m context to be closed 346 * @return 0 for success or negative in case of error 347 */ 348 int lwm2m_engine_close_socket_connection(struct lwm2m_ctx *client_ctx); 349 350 /** 351 * @brief Starts the socket of @p client_ctx. Requires CONFIG_LWM2M_DTLS_SUPPORT=y. 352 * 353 * @param[in] client_ctx lwm2m context to be re-added to the lwm2m work loop 354 * @return 0 for success or negative in case of error 355 */ 356 int lwm2m_engine_connection_resume(struct lwm2m_ctx *client_ctx); 357 358 /** 359 * @brief Moves all queued messages to pending. 360 * 361 * @param[in, out] client_ctx 362 * @return 0 for success or negative in case of error 363 */ 364 int lwm2m_push_queued_buffers(struct lwm2m_ctx *client_ctx); 365 366 /* Resources */ 367 struct lwm2m_ctx **lwm2m_sock_ctx(void); 368 int lwm2m_sock_nfds(void); 369 370 /** 371 * @brief Trigger the LwM2M engine to run. 372 * 373 * This function wakes up ongoing poll() from the socket-loop. 374 * It should be called when new transmissions are scheduled or service schedules are modified. 375 */ 376 void lwm2m_engine_wake_up(void); 377 378 /** 379 * @brief Locks the access to shared LwM2M engine variables. 380 */ 381 void lwm2m_engine_lock(void); 382 383 /** 384 * @brief Unlocks the access to shared LwM2M engine variables. 385 */ 386 void lwm2m_engine_unlock(void); 387 388 /** 389 * @brief Locks the client. 390 * 391 * @param[in] client_ctx LwM2M context 392 */ 393 void lwm2m_client_lock(struct lwm2m_ctx *ctx); 394 395 /** 396 * @brief Unlocks the client previously locked by lwm2m_client_lock(). 397 * 398 * @param[in] client_ctx LwM2M context 399 */ 400 void lwm2m_client_unlock(struct lwm2m_ctx *ctx); 401 402 #endif /* LWM2M_ENGINE_H */ 403