tests: lwm2m_rd_client: Added fff tests
Added unittests for lwm2m_rd_client.c using FFF framework.
Signed-off-by: Andreas Chmielewski <andreas.chmielewski@grandcentrix.net>
diff --git a/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/CMakeLists.txt b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/CMakeLists.txt
new file mode 100644
index 0000000..5c7d106
--- /dev/null
+++ b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/CMakeLists.txt
@@ -0,0 +1,30 @@
+cmake_minimum_required(VERSION 3.20.0)
+
+find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
+project(lwm2m_rd_client)
+
+set(APP_SRC_DIR ${CMAKE_CURRENT_SOURCE_DIR}/src)
+
+# Add test sources
+target_sources(app PRIVATE ${APP_SRC_DIR}/main.c)
+target_sources(app PRIVATE ${APP_SRC_DIR}/stubs.c)
+target_sources(app PRIVATE ${ZEPHYR_BASE}/subsys/net/lib/lwm2m/lwm2m_rd_client.c)
+
+# Add includes directories
+target_include_directories(app PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/src)
+target_include_directories(app PRIVATE ${ZEPHYR_BASE}/include/)
+target_include_directories(app PRIVATE ${ZEPHYR_BASE}/subsys/net/lib/lwm2m/)
+
+add_compile_definitions(CONFIG_LWM2M_ENGINE_MAX_PENDING=2)
+add_compile_definitions(CONFIG_LWM2M_ENGINE_MAX_REPLIES=2)
+add_compile_definitions(CONFIG_LWM2M_ENGINE_VALIDATION_BUFFER_SIZE=512)
+add_compile_definitions(CONFIG_LWM2M_ENGINE_MESSAGE_HEADER_SIZE=512)
+add_compile_definitions(CONFIG_LWM2M_RD_CLIENT_ENDPOINT_NAME_MAX_LENGTH=32)
+add_compile_definitions(CONFIG_LWM2M_RD_CLIENT_MAX_RETRIES=2)
+add_compile_definitions(CONFIG_LWM2M_COAP_BLOCK_SIZE=256)
+add_compile_definitions(CONFIG_LWM2M_COAP_MAX_MSG_SIZE=512)
+add_compile_definitions(CONFIG_LWM2M_ENGINE_DEFAULT_LIFETIME=60)
+add_compile_definitions(CONFIG_LWM2M_SECURITY_INSTANCE_COUNT=1)
+add_compile_definitions(CONFIG_LWM2M_SECONDS_TO_UPDATE_EARLY=30)
+add_compile_definitions(CONFIG_LWM2M_QUEUE_MODE_UPTIME=30)
+add_compile_definitions(CONFIG_LWM2M_LOG_LEVEL=4)
diff --git a/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/prj.conf b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/prj.conf
new file mode 100644
index 0000000..fce1a579
--- /dev/null
+++ b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/prj.conf
@@ -0,0 +1,3 @@
+CONFIG_ZTEST=y
+CONFIG_ZTEST_NEW_API=y
+CONFIG_ZTEST_STACK_SIZE=4096
diff --git a/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/src/main.c b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/src/main.c
new file mode 100644
index 0000000..d5fe2ed
--- /dev/null
+++ b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/src/main.c
@@ -0,0 +1,361 @@
+/*
+ * Copyright (c) 2022 grandcentrix GmbH
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "stubs.h"
+
+#include <zephyr/fff.h>
+#include <zephyr/logging/log.h>
+#include <zephyr/ztest.h>
+
+#include <lwm2m_rd_client.h>
+
+LOG_MODULE_REGISTER(lwm2m_rd_client_test);
+
+DEFINE_FFF_GLOBALS;
+
+/* Maximum number of iterations within the state machine of RD Client
+ * service that is waited for until a possible event occurs
+ */
+static const uint8_t RD_CLIENT_MAX_LOOKUP_ITERATIONS = 10;
+
+FAKE_VOID_FUNC(show_lwm2m_event, enum lwm2m_rd_client_event);
+FAKE_VOID_FUNC(show_lwm2m_observe, enum lwm2m_observe_event);
+
+bool check_lwm2m_rd_client_event(uint8_t expected_val, uint8_t arg_index)
+{
+ int max_service_iterations = RD_CLIENT_MAX_LOOKUP_ITERATIONS;
+
+ while (max_service_iterations > 0) {
+ if (show_lwm2m_event_fake.call_count > arg_index) {
+ return show_lwm2m_event_fake.arg0_history[arg_index] == expected_val;
+ }
+
+ wait_for_service(1);
+ max_service_iterations--;
+ }
+
+ return false;
+}
+
+bool check_lwm2m_observe_event(uint8_t expected_val, uint8_t arg_index)
+{
+ int max_service_iterations = RD_CLIENT_MAX_LOOKUP_ITERATIONS;
+
+ while (max_service_iterations > 0) {
+ if (show_lwm2m_observe_fake.call_count > arg_index) {
+ return show_lwm2m_observe_fake.arg0_history[arg_index] == expected_val;
+ }
+
+ wait_for_service(1);
+ max_service_iterations--;
+ }
+
+ return false;
+}
+
+static void lwm2m_event_cb(struct lwm2m_ctx *client, enum lwm2m_rd_client_event client_event)
+{
+ ARG_UNUSED(client);
+
+ switch (client_event) {
+ case LWM2M_RD_CLIENT_EVENT_ENGINE_SUSPENDED:
+ LOG_INF("**** LWM2M_RD_CLIENT_EVENT_ENGINE_SUSPENDED");
+ break;
+ case LWM2M_RD_CLIENT_EVENT_REGISTRATION_FAILURE:
+ LOG_INF("**** LWM2M_RD_CLIENT_EVENT_REGISTRATION_FAILURE");
+ break;
+ case LWM2M_RD_CLIENT_EVENT_REG_TIMEOUT:
+ LOG_INF("**** LWM2M_RD_CLIENT_EVENT_REG_TIMEOUT");
+ break;
+ case LWM2M_RD_CLIENT_EVENT_DISCONNECT:
+ LOG_INF("**** LWM2M_RD_CLIENT_EVENT_DISCONNECT");
+ break;
+ case LWM2M_RD_CLIENT_EVENT_REGISTRATION_COMPLETE:
+ LOG_INF("**** LWM2M_RD_CLIENT_EVENT_REGISTRATION_COMPLETE");
+ break;
+ case LWM2M_RD_CLIENT_EVENT_REG_UPDATE_COMPLETE:
+ LOG_INF("**** LWM2M_RD_CLIENT_EVENT_REG_UPDATE_COMPLETE");
+ break;
+ case LWM2M_RD_CLIENT_EVENT_NETWORK_ERROR:
+ LOG_INF("**** LWM2M_RD_CLIENT_EVENT_NETWORK_ERROR");
+ break;
+ default:
+ break;
+ }
+
+ show_lwm2m_event(client_event);
+}
+
+static void lwm2m_observe_cb(enum lwm2m_observe_event event, struct lwm2m_obj_path *path,
+ void *user_data)
+{
+ switch (event) {
+ case LWM2M_OBSERVE_EVENT_OBSERVER_ADDED:
+ LOG_INF("**** LWM2M_OBSERVE_EVENT_OBSERVER_ADDED");
+ break;
+ case LWM2M_OBSERVE_EVENT_NOTIFY_TIMEOUT:
+ LOG_INF("**** LWM2M_OBSERVE_EVENT_NOTIFY_TIMEOUT");
+ break;
+ case LWM2M_OBSERVE_EVENT_OBSERVER_REMOVED:
+ LOG_INF("**** LWM2M_OBSERVE_EVENT_OBSERVER_REMOVED");
+ break;
+ case LWM2M_OBSERVE_EVENT_NOTIFY_ACK:
+ LOG_INF("**** LWM2M_OBSERVE_EVENT_NOTIFY_ACK");
+ break;
+ default:
+ break;
+ }
+
+ show_lwm2m_observe(event);
+}
+
+#define FFF_FAKES_LIST(FAKE)
+
+static void my_suite_before(void *data)
+{
+ /* Register resets */
+ DO_FOREACH_FAKE(RESET_FAKE);
+
+ /* reset common FFF internal structures */
+ FFF_RESET_HISTORY();
+
+ RESET_FAKE(show_lwm2m_event);
+ RESET_FAKE(show_lwm2m_observe);
+
+ test_lwm2m_engine_stop_service();
+}
+
+void message_reply_cb_default(struct lwm2m_message *msg)
+{
+ struct coap_packet response;
+ struct coap_reply reply;
+ struct sockaddr from;
+
+ memset(&response, 0, sizeof(struct coap_packet));
+ memset(&reply, 0, sizeof(struct coap_reply));
+ memset(&from, 0, sizeof(struct sockaddr));
+
+ msg->reply_cb(&response, &reply, &from);
+}
+
+void message_reply_timeout_cb_default(struct lwm2m_message *msg)
+{
+ msg->message_timeout_cb(msg);
+}
+
+ZTEST_SUITE(lwm2m_rd_client, NULL, NULL, my_suite_before, NULL, NULL);
+
+ZTEST(lwm2m_rd_client, test_start_registration_ok)
+{
+ struct lwm2m_ctx ctx;
+
+ (void)memset(&ctx, 0x0, sizeof(ctx));
+
+ test_prepare_pending_message_cb(&message_reply_cb_default);
+
+ lwm2m_engine_add_service_fake.custom_fake = lwm2m_engine_add_service_fake_default;
+ lwm2m_rd_client_init();
+ test_lwm2m_engine_start_service();
+ wait_for_service(1);
+
+ lwm2m_get_bool_fake.custom_fake = lwm2m_get_bool_fake_default;
+ lwm2m_sprint_ip_addr_fake.custom_fake = lwm2m_sprint_ip_addr_fake_default;
+ lwm2m_init_message_fake.custom_fake = lwm2m_init_message_fake_default;
+ coap_header_get_code_fake.custom_fake = coap_header_get_code_fake_created;
+ coap_find_options_fake.custom_fake = coap_find_options_do_registration_reply_cb_ok;
+ zassert_true(lwm2m_rd_client_start(&ctx, "Test", 0, lwm2m_event_cb, lwm2m_observe_cb) == 0,
+ NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REGISTRATION_COMPLETE, 0),
+ NULL);
+
+ coap_header_get_code_fake.custom_fake = coap_header_get_code_fake_deleted;
+ zassert_true(lwm2m_rd_client_stop(&ctx, lwm2m_event_cb, true) == 0, NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_DISCONNECT, 1), NULL);
+}
+
+ZTEST(lwm2m_rd_client, test_start_registration_timeout)
+{
+ struct lwm2m_ctx ctx;
+
+ (void)memset(&ctx, 0x0, sizeof(ctx));
+
+ test_prepare_pending_message_cb(&message_reply_timeout_cb_default);
+
+ lwm2m_engine_add_service_fake.custom_fake = lwm2m_engine_add_service_fake_default;
+ lwm2m_rd_client_init();
+ test_lwm2m_engine_start_service();
+ wait_for_service(1);
+
+ lwm2m_get_bool_fake.custom_fake = lwm2m_get_bool_fake_default;
+ lwm2m_sprint_ip_addr_fake.custom_fake = lwm2m_sprint_ip_addr_fake_default;
+ lwm2m_init_message_fake.custom_fake = lwm2m_init_message_fake_default;
+ zassert_true(lwm2m_rd_client_start(&ctx, "Test", 0, lwm2m_event_cb, lwm2m_observe_cb) == 0,
+ NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_DISCONNECT, 0), NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REG_TIMEOUT, 1), NULL);
+}
+
+ZTEST(lwm2m_rd_client, test_start_registration_fail)
+{
+ struct lwm2m_ctx ctx;
+
+ (void)memset(&ctx, 0x0, sizeof(ctx));
+
+ test_prepare_pending_message_cb(&message_reply_cb_default);
+
+ lwm2m_engine_add_service_fake.custom_fake = lwm2m_engine_add_service_fake_default;
+ lwm2m_rd_client_init();
+ test_lwm2m_engine_start_service();
+ wait_for_service(1);
+
+ lwm2m_get_bool_fake.custom_fake = lwm2m_get_bool_fake_default;
+ lwm2m_sprint_ip_addr_fake.custom_fake = lwm2m_sprint_ip_addr_fake_default;
+ lwm2m_init_message_fake.custom_fake = lwm2m_init_message_fake_default;
+ zassert_true(lwm2m_rd_client_start(&ctx, "Test", 0, lwm2m_event_cb, lwm2m_observe_cb) == 0,
+ NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REGISTRATION_FAILURE, 0),
+ NULL);
+}
+
+ZTEST(lwm2m_rd_client, test_start_registration_update)
+{
+ struct lwm2m_ctx ctx;
+
+ (void)memset(&ctx, 0x0, sizeof(ctx));
+
+ test_prepare_pending_message_cb(&message_reply_cb_default);
+
+ lwm2m_engine_add_service_fake.custom_fake = lwm2m_engine_add_service_fake_default;
+ lwm2m_rd_client_init();
+ test_lwm2m_engine_start_service();
+ wait_for_service(1);
+
+ lwm2m_get_bool_fake.custom_fake = lwm2m_get_bool_fake_default;
+ lwm2m_sprint_ip_addr_fake.custom_fake = lwm2m_sprint_ip_addr_fake_default;
+ lwm2m_init_message_fake.custom_fake = lwm2m_init_message_fake_default;
+ coap_header_get_code_fake.custom_fake = coap_header_get_code_fake_created;
+ coap_find_options_fake.custom_fake = coap_find_options_do_registration_reply_cb_ok;
+ zassert_true(lwm2m_rd_client_start(&ctx, "Test", 0, lwm2m_event_cb, lwm2m_observe_cb) == 0,
+ NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REGISTRATION_COMPLETE, 0),
+ NULL);
+
+ lwm2m_rd_client_update();
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REG_UPDATE_COMPLETE, 1),
+ NULL);
+}
+
+ZTEST(lwm2m_rd_client, test_start_registration_update_fail)
+{
+ struct lwm2m_ctx ctx;
+
+ (void)memset(&ctx, 0x0, sizeof(ctx));
+
+ test_prepare_pending_message_cb(&message_reply_cb_default);
+
+ lwm2m_engine_add_service_fake.custom_fake = lwm2m_engine_add_service_fake_default;
+ lwm2m_rd_client_init();
+ test_lwm2m_engine_start_service();
+ wait_for_service(1);
+
+ lwm2m_get_bool_fake.custom_fake = lwm2m_get_bool_fake_default;
+ lwm2m_sprint_ip_addr_fake.custom_fake = lwm2m_sprint_ip_addr_fake_default;
+ lwm2m_init_message_fake.custom_fake = lwm2m_init_message_fake_default;
+ coap_header_get_code_fake.custom_fake = coap_header_get_code_fake_created;
+ coap_find_options_fake.custom_fake = coap_find_options_do_registration_reply_cb_ok;
+ zassert_true(lwm2m_rd_client_start(&ctx, "Test", 0, lwm2m_event_cb, lwm2m_observe_cb) == 0,
+ NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REGISTRATION_COMPLETE, 0),
+ NULL);
+
+ RESET_FAKE(coap_header_get_code);
+
+ lwm2m_rd_client_update();
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REGISTRATION_FAILURE, 1),
+ NULL);
+}
+
+ZTEST(lwm2m_rd_client, test_error_on_registration_update)
+{
+ struct lwm2m_ctx ctx;
+
+ (void)memset(&ctx, 0x0, sizeof(ctx));
+
+ test_prepare_pending_message_cb(&message_reply_cb_default);
+
+ lwm2m_engine_add_service_fake.custom_fake = lwm2m_engine_add_service_fake_default;
+ lwm2m_rd_client_init();
+ test_lwm2m_engine_start_service();
+ wait_for_service(1);
+
+ lwm2m_get_bool_fake.custom_fake = lwm2m_get_bool_fake_default;
+ lwm2m_sprint_ip_addr_fake.custom_fake = lwm2m_sprint_ip_addr_fake_default;
+ lwm2m_init_message_fake.custom_fake = lwm2m_init_message_fake_default;
+ coap_header_get_code_fake.custom_fake = coap_header_get_code_fake_created;
+ coap_find_options_fake.custom_fake = coap_find_options_do_registration_reply_cb_ok;
+ zassert_true(lwm2m_rd_client_start(&ctx, "Test", 0, lwm2m_event_cb, lwm2m_observe_cb) == 0,
+ NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REGISTRATION_COMPLETE, 0),
+ NULL);
+
+ coap_packet_append_option_fake.custom_fake = coap_packet_append_option_fake_err;
+ lwm2m_rd_client_update();
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REGISTRATION_COMPLETE, 1),
+ NULL);
+}
+
+ZTEST(lwm2m_rd_client, test_network_error_on_registration)
+{
+ struct lwm2m_ctx ctx;
+
+ (void)memset(&ctx, 0x0, sizeof(ctx));
+
+ lwm2m_engine_add_service_fake.custom_fake = lwm2m_engine_add_service_fake_default;
+ lwm2m_rd_client_init();
+ test_lwm2m_engine_start_service();
+ wait_for_service(1);
+
+ lwm2m_get_bool_fake.custom_fake = lwm2m_get_bool_fake_default;
+ lwm2m_sprint_ip_addr_fake.custom_fake = lwm2m_sprint_ip_addr_fake_default;
+ lwm2m_init_message_fake.custom_fake = lwm2m_init_message_fake_default;
+ coap_header_get_code_fake.custom_fake = coap_header_get_code_fake_created;
+ coap_find_options_fake.custom_fake = coap_find_options_do_registration_reply_cb_ok;
+ coap_packet_append_option_fake.custom_fake = coap_packet_append_option_fake_err;
+ zassert_true(lwm2m_rd_client_start(&ctx, "Test", 0, lwm2m_event_cb, lwm2m_observe_cb) == 0,
+ NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_NETWORK_ERROR, 0), NULL);
+}
+
+ZTEST(lwm2m_rd_client, test_suspend_resume_registration)
+{
+ struct lwm2m_ctx ctx;
+
+ (void)memset(&ctx, 0x0, sizeof(ctx));
+
+ test_prepare_pending_message_cb(&message_reply_cb_default);
+
+ lwm2m_engine_add_service_fake.custom_fake = lwm2m_engine_add_service_fake_default;
+ lwm2m_rd_client_init();
+ test_lwm2m_engine_start_service();
+ wait_for_service(1);
+
+ lwm2m_get_bool_fake.custom_fake = lwm2m_get_bool_fake_default;
+ lwm2m_sprint_ip_addr_fake.custom_fake = lwm2m_sprint_ip_addr_fake_default;
+ lwm2m_init_message_fake.custom_fake = lwm2m_init_message_fake_default;
+ coap_header_get_code_fake.custom_fake = coap_header_get_code_fake_created;
+ coap_find_options_fake.custom_fake = coap_find_options_do_registration_reply_cb_ok;
+ zassert_true(lwm2m_rd_client_start(&ctx, "Test", 0, lwm2m_event_cb, lwm2m_observe_cb) == 0,
+ NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REGISTRATION_COMPLETE, 0),
+ NULL);
+
+ zassert_true(lwm2m_rd_client_pause() == 0, NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_ENGINE_SUSPENDED, 1), NULL);
+
+ zassert_true(lwm2m_rd_client_resume() == 0, NULL);
+ zassert_true(check_lwm2m_rd_client_event(LWM2M_RD_CLIENT_EVENT_REG_UPDATE_COMPLETE, 2),
+ NULL);
+}
diff --git a/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/src/stubs.c b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/src/stubs.c
new file mode 100644
index 0000000..fe21254
--- /dev/null
+++ b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/src/stubs.c
@@ -0,0 +1,204 @@
+/*
+ * Copyright (c) 2022 grandcentrix GmbH
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include <zephyr/logging/log.h>
+
+#include <stubs.h>
+
+LOG_MODULE_DECLARE(lwm2m_rd_client_test);
+
+/* zephyr/net/coap.h */
+DEFINE_FAKE_VALUE_FUNC(uint8_t, coap_header_get_code, const struct coap_packet *);
+uint8_t coap_header_get_code_fake_created(const struct coap_packet *cpkt)
+{
+ return COAP_RESPONSE_CODE_CREATED;
+}
+uint8_t coap_header_get_code_fake_deleted(const struct coap_packet *cpkt)
+{
+ return COAP_RESPONSE_CODE_DELETED;
+}
+
+DEFINE_FAKE_VALUE_FUNC(int, coap_append_option_int, struct coap_packet *, uint16_t, unsigned int);
+DEFINE_FAKE_VALUE_FUNC(int, coap_packet_append_option, struct coap_packet *, uint16_t,
+ const uint8_t *, uint16_t);
+int coap_packet_append_option_fake_err(struct coap_packet *cpkt, uint16_t code,
+ const uint8_t *value, uint16_t len)
+{
+ return -1;
+}
+
+DEFINE_FAKE_VALUE_FUNC(int, coap_packet_append_payload_marker, struct coap_packet *);
+DEFINE_FAKE_VALUE_FUNC(int, coap_find_options, const struct coap_packet *, uint16_t,
+ struct coap_option *, uint16_t);
+int coap_find_options_do_registration_reply_cb_ok(const struct coap_packet *cpkt, uint16_t code,
+ struct coap_option *options, uint16_t veclen)
+{
+ char options0[] = "rd";
+ char options1[] = "jATO2yn9u7";
+
+ options[1].len = sizeof(options0);
+ memcpy(&options[1].value, &options0, sizeof(options0));
+ options[1].len = sizeof(options1);
+ memcpy(&options[1].value, &options1, sizeof(options1));
+ return 3;
+}
+
+DEFINE_FAKE_VALUE_FUNC(uint16_t, coap_next_id);
+
+/* zephyr/net/lwm2m.h */
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_engine_start, struct lwm2m_ctx *);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_engine_stop, struct lwm2m_ctx *);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_open_socket, struct lwm2m_ctx *);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_get_u32, const struct lwm2m_obj_path *, uint32_t *);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_get_u16, const struct lwm2m_obj_path *, uint16_t *);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_get_bool, const struct lwm2m_obj_path *, bool *);
+int lwm2m_get_bool_fake_default(const struct lwm2m_obj_path *path, bool *value)
+{
+ *value = false;
+ return 0;
+}
+
+/* subsys/net/lib/lwm2m/lwm2m_engine.h */
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_socket_start, struct lwm2m_ctx *);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_socket_close, struct lwm2m_ctx *);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_close_socket, struct lwm2m_ctx *);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_security_inst_id_to_index, uint16_t);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_engine_connection_resume, struct lwm2m_ctx *);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_push_queued_buffers, struct lwm2m_ctx *);
+DEFINE_FAKE_VOID_FUNC(lwm2m_engine_context_init, struct lwm2m_ctx *);
+DEFINE_FAKE_VOID_FUNC(lwm2m_engine_context_close, struct lwm2m_ctx *);
+DEFINE_FAKE_VALUE_FUNC(char *, lwm2m_sprint_ip_addr, const struct sockaddr *);
+char *lwm2m_sprint_ip_addr_fake_default(const struct sockaddr *addr)
+{
+ return "192.168.1.1:4444";
+}
+
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_server_short_id_to_inst, uint16_t);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_security_index_to_inst_id, int);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_engine_add_service, k_work_handler_t, uint32_t);
+
+k_work_handler_t lwm2m_engine_add_service_service;
+uint32_t lwm2m_engine_add_service_period_ms = 20;
+int lwm2m_engine_add_service_fake_default(k_work_handler_t service, uint32_t period_ms)
+{
+ lwm2m_engine_add_service_service = service;
+ lwm2m_engine_add_service_period_ms = period_ms;
+ return 0;
+}
+
+uint16_t counter = RD_CLIENT_MAX_SERVICE_ITERATIONS;
+struct lwm2m_message *pending_message;
+void *(*pending_message_cb)();
+
+static void service_work_fn(struct k_work *work)
+{
+ while (lwm2m_engine_add_service_service != NULL) {
+ if (pending_message != NULL && pending_message_cb != NULL) {
+ pending_message_cb(pending_message);
+ pending_message = NULL;
+ }
+
+ lwm2m_engine_add_service_service(work);
+ k_sleep(K_MSEC(lwm2m_engine_add_service_period_ms));
+ counter--;
+
+ /* avoid endless loop if rd client is stuck somewhere */
+ if (counter == 0) {
+ break;
+ }
+ }
+}
+
+void wait_for_service(uint16_t cycles)
+{
+ uint16_t end = counter - cycles;
+
+ while (counter > end) {
+ k_sleep(K_MSEC(1));
+ }
+}
+
+K_WORK_DEFINE(service_work, service_work_fn);
+
+void test_lwm2m_engine_start_service(void)
+{
+ counter = RD_CLIENT_MAX_SERVICE_ITERATIONS;
+ k_work_submit(&service_work);
+}
+
+void test_lwm2m_engine_stop_service(void)
+{
+ pending_message_cb = NULL;
+ k_work_cancel(&service_work);
+}
+
+/* subsys/net/lib/lwm2m/lwm2m_message_handling.h */
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_init_message, struct lwm2m_message *);
+int lwm2m_init_message_fake_default(struct lwm2m_message *msg)
+{
+ pending_message = msg;
+ return 0;
+}
+
+void test_prepare_pending_message_cb(void *cb)
+{
+ pending_message_cb = cb;
+}
+
+DEFINE_FAKE_VOID_FUNC(lwm2m_reset_message, struct lwm2m_message *, bool);
+DEFINE_FAKE_VALUE_FUNC(int, lwm2m_send_message_async, struct lwm2m_message *);
+
+/* subsys/net/lib/lwm2m/lwm2m_registry.h */
+DEFINE_FAKE_VOID_FUNC(lwm2m_engine_get_binding, char *);
+DEFINE_FAKE_VOID_FUNC(lwm2m_engine_get_queue_mode, char *);
+
+/* subsys/net/lib/lwm2m/lwm2m_rw_link_format.h */
+FAKE_VALUE_FUNC(int, put_begin, struct lwm2m_output_context *, struct lwm2m_obj_path *);
+FAKE_VALUE_FUNC(int, put_end, struct lwm2m_output_context *, struct lwm2m_obj_path *);
+FAKE_VALUE_FUNC(int, put_begin_oi, struct lwm2m_output_context *, struct lwm2m_obj_path *);
+FAKE_VALUE_FUNC(int, put_end_oi, struct lwm2m_output_context *, struct lwm2m_obj_path *);
+FAKE_VALUE_FUNC(int, put_begin_r, struct lwm2m_output_context *, struct lwm2m_obj_path *);
+FAKE_VALUE_FUNC(int, put_end_r, struct lwm2m_output_context *, struct lwm2m_obj_path *);
+FAKE_VALUE_FUNC(int, put_begin_ri, struct lwm2m_output_context *, struct lwm2m_obj_path *);
+FAKE_VALUE_FUNC(int, put_end_ri, struct lwm2m_output_context *, struct lwm2m_obj_path *);
+FAKE_VALUE_FUNC(int, put_s8, struct lwm2m_output_context *, struct lwm2m_obj_path *, int8_t);
+FAKE_VALUE_FUNC(int, put_s16, struct lwm2m_output_context *, struct lwm2m_obj_path *, int16_t);
+FAKE_VALUE_FUNC(int, put_s32, struct lwm2m_output_context *, struct lwm2m_obj_path *, int32_t);
+FAKE_VALUE_FUNC(int, put_s64, struct lwm2m_output_context *, struct lwm2m_obj_path *, int64_t);
+FAKE_VALUE_FUNC(int, put_time, struct lwm2m_output_context *, struct lwm2m_obj_path *, time_t);
+FAKE_VALUE_FUNC(int, put_string, struct lwm2m_output_context *, struct lwm2m_obj_path *, char *,
+ size_t);
+FAKE_VALUE_FUNC(int, put_float, struct lwm2m_output_context *, struct lwm2m_obj_path *, double *);
+FAKE_VALUE_FUNC(int, put_bool, struct lwm2m_output_context *, struct lwm2m_obj_path *, bool);
+FAKE_VALUE_FUNC(int, put_opaque, struct lwm2m_output_context *, struct lwm2m_obj_path *, char *,
+ size_t);
+FAKE_VALUE_FUNC(int, put_objlnk, struct lwm2m_output_context *, struct lwm2m_obj_path *,
+ struct lwm2m_objlnk *);
+FAKE_VALUE_FUNC(int, put_corelink, struct lwm2m_output_context *, const struct lwm2m_obj_path *);
+
+const struct lwm2m_writer link_format_writer = {
+ .put_begin = put_begin,
+ .put_end = put_end,
+ .put_begin_oi = put_begin_oi,
+ .put_end_oi = put_end_oi,
+ .put_begin_r = put_begin_r,
+ .put_end_r = put_end_r,
+ .put_begin_ri = put_begin_oi,
+ .put_end_ri = put_end_oi,
+ .put_s8 = put_s8,
+ .put_s16 = put_s16,
+ .put_s32 = put_s32,
+ .put_s64 = put_s64,
+ .put_time = put_time,
+ .put_string = put_string,
+ .put_float = put_float,
+ .put_bool = put_bool,
+ .put_opaque = put_opaque,
+ .put_objlnk = put_objlnk,
+ .put_corelink = put_corelink,
+};
+
+DEFINE_FAKE_VALUE_FUNC(int, do_register_op_link_format, struct lwm2m_message *);
diff --git a/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/src/stubs.h b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/src/stubs.h
new file mode 100644
index 0000000..9a87ebf
--- /dev/null
+++ b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/src/stubs.h
@@ -0,0 +1,116 @@
+/*
+ * Copyright (c) 2022 grandcentrix GmbH
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef STUBS_H
+#define STUBS_H
+
+#include <stdbool.h>
+#include <stdint.h>
+
+#include <zephyr/fff.h>
+#include <zephyr/net/lwm2m.h>
+#include <zephyr/ztest.h>
+
+#include <lwm2m_engine.h>
+
+/* Number of iterations the state machine within the RDClient service
+ * is triggered
+ */
+static const uint8_t RD_CLIENT_MAX_SERVICE_ITERATIONS = 50;
+
+/* zephyr/net/coap.h */
+DECLARE_FAKE_VALUE_FUNC(uint8_t, coap_header_get_code, const struct coap_packet *);
+uint8_t coap_header_get_code_fake_created(const struct coap_packet *cpkt);
+uint8_t coap_header_get_code_fake_deleted(const struct coap_packet *cpkt);
+DECLARE_FAKE_VALUE_FUNC(int, coap_append_option_int, struct coap_packet *, uint16_t, unsigned int);
+DECLARE_FAKE_VALUE_FUNC(int, coap_packet_append_option, struct coap_packet *, uint16_t,
+ const uint8_t *, uint16_t);
+int coap_packet_append_option_fake_err(struct coap_packet *cpkt, uint16_t code,
+ const uint8_t *value, uint16_t len);
+DECLARE_FAKE_VALUE_FUNC(int, coap_packet_append_payload_marker, struct coap_packet *);
+DECLARE_FAKE_VALUE_FUNC(int, coap_find_options, const struct coap_packet *, uint16_t,
+ struct coap_option *, uint16_t);
+int coap_find_options_do_registration_reply_cb_ok(const struct coap_packet *cpkt, uint16_t code,
+ struct coap_option *options, uint16_t veclen);
+DECLARE_FAKE_VALUE_FUNC(uint16_t, coap_next_id);
+
+/* zephyr/net/lwm2m.h */
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_engine_start, struct lwm2m_ctx *);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_engine_stop, struct lwm2m_ctx *);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_open_socket, struct lwm2m_ctx *);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_get_u32, const struct lwm2m_obj_path *, uint32_t *);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_get_u16, const struct lwm2m_obj_path *, uint16_t *);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_get_bool, const struct lwm2m_obj_path *, bool *);
+int lwm2m_get_bool_fake_default(const struct lwm2m_obj_path *path, bool *value);
+
+/* subsys/net/lib/lwm2m/lwm2m_engine.h */
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_socket_start, struct lwm2m_ctx *);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_socket_close, struct lwm2m_ctx *);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_close_socket, struct lwm2m_ctx *);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_security_inst_id_to_index, uint16_t);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_engine_connection_resume, struct lwm2m_ctx *);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_push_queued_buffers, struct lwm2m_ctx *);
+DECLARE_FAKE_VOID_FUNC(lwm2m_engine_context_init, struct lwm2m_ctx *);
+DECLARE_FAKE_VOID_FUNC(lwm2m_engine_context_close, struct lwm2m_ctx *);
+DECLARE_FAKE_VALUE_FUNC(char *, lwm2m_sprint_ip_addr, const struct sockaddr *);
+char *lwm2m_sprint_ip_addr_fake_default(const struct sockaddr *addr);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_server_short_id_to_inst, uint16_t);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_security_index_to_inst_id, int);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_engine_add_service, k_work_handler_t, uint32_t);
+int lwm2m_engine_add_service_fake_default(k_work_handler_t service, uint32_t period_ms);
+void wait_for_service(uint16_t cycles);
+void test_lwm2m_engine_start_service(void);
+void test_lwm2m_engine_stop_service(void);
+
+/* subsys/net/lib/lwm2m/lwm2m_message_handling.h */
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_init_message, struct lwm2m_message *);
+int lwm2m_init_message_fake_default(struct lwm2m_message *msg);
+void test_prepare_pending_message_cb(void *cb);
+
+DECLARE_FAKE_VOID_FUNC(lwm2m_reset_message, struct lwm2m_message *, bool);
+DECLARE_FAKE_VALUE_FUNC(int, lwm2m_send_message_async, struct lwm2m_message *);
+
+/* subsys/net/lib/lwm2m/lwm2m_registry.h */
+DECLARE_FAKE_VOID_FUNC(lwm2m_engine_get_binding, char *);
+DECLARE_FAKE_VOID_FUNC(lwm2m_engine_get_queue_mode, char *);
+
+/* subsys/net/lib/lwm2m/lwm2m_rw_link_format.h */
+DECLARE_FAKE_VALUE_FUNC(int, do_register_op_link_format, struct lwm2m_message *);
+
+#define DO_FOREACH_FAKE(FUNC) \
+ do { \
+ FUNC(coap_header_get_code) \
+ FUNC(coap_append_option_int) \
+ FUNC(coap_packet_append_option) \
+ FUNC(coap_packet_append_payload_marker) \
+ FUNC(coap_find_options) \
+ FUNC(coap_next_id) \
+ FUNC(lwm2m_engine_start) \
+ FUNC(lwm2m_engine_stop) \
+ FUNC(lwm2m_get_u32) \
+ FUNC(lwm2m_get_u16) \
+ FUNC(lwm2m_get_bool) \
+ FUNC(lwm2m_socket_start) \
+ FUNC(lwm2m_socket_close) \
+ FUNC(lwm2m_close_socket) \
+ FUNC(lwm2m_security_inst_id_to_index) \
+ FUNC(lwm2m_engine_connection_resume) \
+ FUNC(lwm2m_push_queued_buffers) \
+ FUNC(lwm2m_engine_context_init) \
+ FUNC(lwm2m_engine_context_close) \
+ FUNC(lwm2m_sprint_ip_addr) \
+ FUNC(lwm2m_server_short_id_to_inst) \
+ FUNC(lwm2m_security_index_to_inst_id) \
+ FUNC(lwm2m_engine_add_service) \
+ FUNC(lwm2m_init_message) \
+ FUNC(lwm2m_reset_message) \
+ FUNC(lwm2m_send_message_async) \
+ FUNC(lwm2m_engine_get_binding) \
+ FUNC(lwm2m_engine_get_queue_mode) \
+ FUNC(do_register_op_link_format) \
+ } while (0)
+
+#endif /* STUBS_H */
diff --git a/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/testcase.yaml b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/testcase.yaml
new file mode 100644
index 0000000..a5af87c
--- /dev/null
+++ b/tests/subsys/net/lib/lwm2m/lwm2m_rd_client/testcase.yaml
@@ -0,0 +1,9 @@
+common:
+ depends_on: netif
+tests:
+ subsys.net.lib.lwm2m_rd_client:
+ tags: net lwm2m
+ platform_allow: native_posix qemu_x86 qemu_x86_64
+ integration_platforms:
+ - native_posix
+ - qemu_x86