1 /*
2 * Copyright (c) 2017 Linaro Limited
3 * Copyright (c) 2017-2019 Foundries.io
4 *
5 * SPDX-License-Identifier: Apache-2.0
6 */
7
8 #define LOG_MODULE_NAME net_lwm2m_client_app
9 #define LOG_LEVEL LOG_LEVEL_DBG
10
11 #include <zephyr/logging/log.h>
12 LOG_MODULE_REGISTER(LOG_MODULE_NAME);
13
14 #include <zephyr/drivers/hwinfo.h>
15 #include <zephyr/kernel.h>
16 #include <zephyr/drivers/gpio.h>
17 #include <zephyr/drivers/sensor.h>
18 #include <zephyr/net/lwm2m.h>
19 #include <zephyr/net/conn_mgr_monitor.h>
20 #include <zephyr/net/conn_mgr_connectivity.h>
21 #include "modules.h"
22 #include "lwm2m_resource_ids.h"
23
24 #define APP_BANNER "Run LWM2M client"
25
26 #define WAIT_TIME K_SECONDS(10)
27 #define CONNECT_TIME K_SECONDS(10)
28
29 #define CLIENT_MANUFACTURER "Zephyr"
30 #define CLIENT_MODEL_NUMBER "OMA-LWM2M Sample Client"
31 #define CLIENT_SERIAL_NUMBER "345000123"
32 #define CLIENT_FIRMWARE_VER "1.0"
33 #define CLIENT_HW_VER "1.0.1"
34 #define TEMP_SENSOR_UNITS "Celcius"
35
36 /* Macros used to subscribe to specific Zephyr NET management events. */
37 #define L4_EVENT_MASK (NET_EVENT_L4_CONNECTED | NET_EVENT_L4_DISCONNECTED)
38 #define CONN_LAYER_EVENT_MASK (NET_EVENT_CONN_IF_FATAL_ERROR)
39
40 static uint8_t bat_idx = LWM2M_DEVICE_PWR_SRC_TYPE_BAT_INT;
41 static int bat_mv = 3800;
42 static int bat_ma = 125;
43 static uint8_t usb_idx = LWM2M_DEVICE_PWR_SRC_TYPE_USB;
44 static int usb_mv = 5000;
45 static int usb_ma = 900;
46 static uint8_t bat_level = 95;
47 static uint8_t bat_status = LWM2M_DEVICE_BATTERY_STATUS_CHARGING;
48 static int mem_free = 15;
49 static int mem_total = 25;
50 static double min_range = 0.0;
51 static double max_range = 100;
52
53 static struct lwm2m_ctx client_ctx;
54
55 static const char *endpoint =
56 (sizeof(CONFIG_LWM2M_APP_ID) > 1 ? CONFIG_LWM2M_APP_ID : CONFIG_BOARD);
57
58 #if defined(CONFIG_LWM2M_DTLS_SUPPORT)
59 BUILD_ASSERT(sizeof(endpoint) <= CONFIG_LWM2M_SECURITY_KEY_SIZE,
60 "Client ID length is too long");
61 #endif /* CONFIG_LWM2M_DTLS_SUPPORT */
62
63 static struct k_sem quit_lock;
64
65 /* Zephyr NET management event callback structures. */
66 static struct net_mgmt_event_callback l4_cb;
67 static struct net_mgmt_event_callback conn_cb;
68
69 static K_SEM_DEFINE(network_connected_sem, 0, 1);
70
device_reboot_cb(uint16_t obj_inst_id,uint8_t * args,uint16_t args_len)71 static int device_reboot_cb(uint16_t obj_inst_id,
72 uint8_t *args, uint16_t args_len)
73 {
74 LOG_INF("DEVICE: REBOOT");
75 /* Add an error for testing */
76 lwm2m_device_add_err(LWM2M_DEVICE_ERROR_LOW_POWER);
77 /* Change the battery voltage for testing */
78 lwm2m_set_s32(&LWM2M_OBJ(3, 0, 7, 0), (bat_mv - 1));
79
80 return 0;
81 }
82
device_factory_default_cb(uint16_t obj_inst_id,uint8_t * args,uint16_t args_len)83 static int device_factory_default_cb(uint16_t obj_inst_id,
84 uint8_t *args, uint16_t args_len)
85 {
86 LOG_INF("DEVICE: FACTORY DEFAULT");
87 /* Add an error for testing */
88 lwm2m_device_add_err(LWM2M_DEVICE_ERROR_GPS_FAILURE);
89 /* Change the USB current for testing */
90 lwm2m_set_s32(&LWM2M_OBJ(3, 0, 8, 1), (usb_ma - 1));
91
92 return 0;
93 }
94
lwm2m_setup(void)95 static int lwm2m_setup(void)
96 {
97 struct lwm2m_res_item temp_sensor_items[] = {
98 {&LWM2M_OBJ(IPSO_OBJECT_TEMP_SENSOR_ID, 0, MIN_RANGE_VALUE_RID), &min_range,
99 sizeof(min_range)},
100 {&LWM2M_OBJ(IPSO_OBJECT_TEMP_SENSOR_ID, 0, MAX_RANGE_VALUE_RID), &max_range,
101 sizeof(max_range)},
102 {&LWM2M_OBJ(IPSO_OBJECT_TEMP_SENSOR_ID, 0, SENSOR_UNITS_RID), TEMP_SENSOR_UNITS,
103 sizeof(TEMP_SENSOR_UNITS)}
104 };
105
106 /* setup SECURITY object */
107
108 /* Server URL */
109 lwm2m_set_string(&LWM2M_OBJ(0, 0, 0), CONFIG_LWM2M_APP_SERVER);
110
111 /* Security Mode */
112 lwm2m_set_u8(&LWM2M_OBJ(0, 0, 2), IS_ENABLED(CONFIG_LWM2M_DTLS_SUPPORT) ? 0 : 3);
113 #if defined(CONFIG_LWM2M_DTLS_SUPPORT)
114 lwm2m_set_string(&LWM2M_OBJ(0, 0, 3), endpoint);
115 if (sizeof(CONFIG_LWM2M_APP_PSK) > 1) {
116 char psk[1 + sizeof(CONFIG_LWM2M_APP_PSK) / 2];
117 /* Need to skip the nul terminator from string */
118 size_t len = hex2bin(CONFIG_LWM2M_APP_PSK, sizeof(CONFIG_LWM2M_APP_PSK) - 1, psk,
119 sizeof(psk));
120 if (len <= 0) {
121 return -EINVAL;
122 }
123 lwm2m_set_opaque(&LWM2M_OBJ(0, 0, 5), (void *)psk, len);
124 }
125 #endif /* CONFIG_LWM2M_DTLS_SUPPORT */
126
127 #if defined(CONFIG_LWM2M_RD_CLIENT_SUPPORT_BOOTSTRAP)
128 /* Mark 1st instance of security object as a bootstrap server */
129 lwm2m_set_u8(&LWM2M_OBJ(0, 0, 1), 1);
130
131 /* Create 2nd instance of security object needed for bootstrap */
132 lwm2m_create_object_inst(&LWM2M_OBJ(0, 1));
133 #else
134 /* Match Security object instance with a Server object instance with
135 * Short Server ID.
136 */
137 lwm2m_set_u16(&LWM2M_OBJ(0, 0, 10), CONFIG_LWM2M_SERVER_DEFAULT_SSID);
138 lwm2m_set_u16(&LWM2M_OBJ(1, 0, 0), CONFIG_LWM2M_SERVER_DEFAULT_SSID);
139 #endif
140
141 /* setup SERVER object */
142
143 /* setup DEVICE object */
144
145 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 0), CLIENT_MANUFACTURER, sizeof(CLIENT_MANUFACTURER),
146 sizeof(CLIENT_MANUFACTURER), LWM2M_RES_DATA_FLAG_RO);
147 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 1), CLIENT_MODEL_NUMBER, sizeof(CLIENT_MODEL_NUMBER),
148 sizeof(CLIENT_MODEL_NUMBER), LWM2M_RES_DATA_FLAG_RO);
149 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 2), CLIENT_SERIAL_NUMBER, sizeof(CLIENT_SERIAL_NUMBER),
150 sizeof(CLIENT_SERIAL_NUMBER), LWM2M_RES_DATA_FLAG_RO);
151 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 3), CLIENT_FIRMWARE_VER, sizeof(CLIENT_FIRMWARE_VER),
152 sizeof(CLIENT_FIRMWARE_VER), LWM2M_RES_DATA_FLAG_RO);
153 lwm2m_register_exec_callback(&LWM2M_OBJ(3, 0, 4), device_reboot_cb);
154 lwm2m_register_exec_callback(&LWM2M_OBJ(3, 0, 5), device_factory_default_cb);
155 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 9), &bat_level, sizeof(bat_level), sizeof(bat_level), 0);
156 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 10), &mem_free, sizeof(mem_free), sizeof(mem_free), 0);
157 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 17), CONFIG_BOARD, sizeof(CONFIG_BOARD),
158 sizeof(CONFIG_BOARD), LWM2M_RES_DATA_FLAG_RO);
159 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 18), CLIENT_HW_VER, sizeof(CLIENT_HW_VER),
160 sizeof(CLIENT_HW_VER), LWM2M_RES_DATA_FLAG_RO);
161 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 20), &bat_status, sizeof(bat_status),
162 sizeof(bat_status), 0);
163 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 21), &mem_total, sizeof(mem_total),
164 sizeof(mem_total), 0);
165
166 /* add power source resource instances */
167 lwm2m_create_res_inst(&LWM2M_OBJ(3, 0, 6, 0));
168 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 6, 0), &bat_idx, sizeof(bat_idx), sizeof(bat_idx), 0);
169 lwm2m_create_res_inst(&LWM2M_OBJ(3, 0, 7, 0));
170 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 7, 0), &bat_mv, sizeof(bat_mv), sizeof(bat_mv), 0);
171 lwm2m_create_res_inst(&LWM2M_OBJ(3, 0, 8, 0));
172 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 8, 0), &bat_ma, sizeof(bat_ma), sizeof(bat_ma), 0);
173 lwm2m_create_res_inst(&LWM2M_OBJ(3, 0, 6, 1));
174 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 6, 1), &usb_idx, sizeof(usb_idx), sizeof(usb_idx), 0);
175 lwm2m_create_res_inst(&LWM2M_OBJ(3, 0, 7, 1));
176 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 7, 1), &usb_mv, sizeof(usb_mv), sizeof(usb_mv), 0);
177 lwm2m_create_res_inst(&LWM2M_OBJ(3, 0, 8, 1));
178 lwm2m_set_res_buf(&LWM2M_OBJ(3, 0, 8, 1), &usb_ma, sizeof(usb_ma), sizeof(usb_ma), 0);
179
180 /* setup FIRMWARE object */
181 if (IS_ENABLED(CONFIG_LWM2M_FIRMWARE_UPDATE_OBJ_SUPPORT)) {
182 init_firmware_update();
183 }
184
185 /* setup TEMP SENSOR object */
186 init_temp_sensor();
187
188 /* Set multiple TEMP SENSOR resource values in one function call. */
189 int err = lwm2m_set_bulk(temp_sensor_items, ARRAY_SIZE(temp_sensor_items));
190
191 if (err) {
192 LOG_ERR("Failed to set TEMP SENSOR resources");
193 return err;
194 }
195
196 /* IPSO: Light Control object */
197 init_led_device();
198
199 /* IPSO: Timer object */
200 init_timer_object();
201
202 return 0;
203 }
204
rd_client_event(struct lwm2m_ctx * client,enum lwm2m_rd_client_event client_event)205 static void rd_client_event(struct lwm2m_ctx *client,
206 enum lwm2m_rd_client_event client_event)
207 {
208 switch (client_event) {
209
210 case LWM2M_RD_CLIENT_EVENT_NONE:
211 /* do nothing */
212 break;
213
214 case LWM2M_RD_CLIENT_EVENT_SERVER_DISABLED:
215 LOG_DBG("LwM2M server disabled");
216 break;
217
218 case LWM2M_RD_CLIENT_EVENT_BOOTSTRAP_REG_FAILURE:
219 LOG_DBG("Bootstrap registration failure!");
220 break;
221
222 case LWM2M_RD_CLIENT_EVENT_BOOTSTRAP_REG_COMPLETE:
223 LOG_DBG("Bootstrap registration complete");
224 break;
225
226 case LWM2M_RD_CLIENT_EVENT_BOOTSTRAP_TRANSFER_COMPLETE:
227 LOG_DBG("Bootstrap transfer complete");
228 break;
229
230 case LWM2M_RD_CLIENT_EVENT_REGISTRATION_FAILURE:
231 LOG_DBG("Registration failure!");
232 break;
233
234 case LWM2M_RD_CLIENT_EVENT_REGISTRATION_COMPLETE:
235 LOG_DBG("Registration complete");
236 break;
237
238 case LWM2M_RD_CLIENT_EVENT_REG_TIMEOUT:
239 LOG_DBG("Registration timeout!");
240 break;
241
242 case LWM2M_RD_CLIENT_EVENT_REG_UPDATE_COMPLETE:
243 LOG_DBG("Registration update complete");
244 break;
245
246 case LWM2M_RD_CLIENT_EVENT_DEREGISTER_FAILURE:
247 LOG_DBG("Deregister failure!");
248 break;
249
250 case LWM2M_RD_CLIENT_EVENT_DISCONNECT:
251 LOG_DBG("Disconnected");
252 break;
253
254 case LWM2M_RD_CLIENT_EVENT_QUEUE_MODE_RX_OFF:
255 LOG_DBG("Queue mode RX window closed");
256 break;
257
258 case LWM2M_RD_CLIENT_EVENT_ENGINE_SUSPENDED:
259 LOG_DBG("LwM2M engine suspended");
260 break;
261
262 case LWM2M_RD_CLIENT_EVENT_NETWORK_ERROR:
263 LOG_ERR("LwM2M engine reported a network error.");
264 lwm2m_rd_client_stop(client, rd_client_event, true);
265 break;
266
267 case LWM2M_RD_CLIENT_EVENT_REG_UPDATE:
268 LOG_DBG("Registration update");
269 break;
270 case LWM2M_RD_CLIENT_EVENT_DEREGISTER:
271 LOG_DBG("Client De-register");
272 break;
273 }
274 }
275
socket_state(int fd,enum lwm2m_socket_states state)276 static void socket_state(int fd, enum lwm2m_socket_states state)
277 {
278 (void) fd;
279 switch (state) {
280 case LWM2M_SOCKET_STATE_ONGOING:
281 LOG_DBG("LWM2M_SOCKET_STATE_ONGOING");
282 break;
283 case LWM2M_SOCKET_STATE_ONE_RESPONSE:
284 LOG_DBG("LWM2M_SOCKET_STATE_ONE_RESPONSE");
285 break;
286 case LWM2M_SOCKET_STATE_LAST:
287 LOG_DBG("LWM2M_SOCKET_STATE_LAST");
288 break;
289 case LWM2M_SOCKET_STATE_NO_DATA:
290 LOG_DBG("LWM2M_SOCKET_STATE_NO_DATA");
291 break;
292 }
293 }
294
observe_cb(enum lwm2m_observe_event event,struct lwm2m_obj_path * path,void * user_data)295 static void observe_cb(enum lwm2m_observe_event event,
296 struct lwm2m_obj_path *path, void *user_data)
297 {
298 char buf[LWM2M_MAX_PATH_STR_SIZE];
299
300 switch (event) {
301
302 case LWM2M_OBSERVE_EVENT_OBSERVER_ADDED:
303 LOG_INF("Observer added for %s", lwm2m_path_log_buf(buf, path));
304 break;
305
306 case LWM2M_OBSERVE_EVENT_OBSERVER_REMOVED:
307 LOG_INF("Observer removed for %s", lwm2m_path_log_buf(buf, path));
308 break;
309
310 case LWM2M_OBSERVE_EVENT_NOTIFY_ACK:
311 LOG_INF("Notify acknowledged for %s", lwm2m_path_log_buf(buf, path));
312 break;
313
314 case LWM2M_OBSERVE_EVENT_NOTIFY_TIMEOUT:
315 LOG_INF("Notify timeout for %s, trying registration update",
316 lwm2m_path_log_buf(buf, path));
317
318 lwm2m_rd_client_update();
319 break;
320 }
321 }
322
on_net_event_l4_disconnected(void)323 static void on_net_event_l4_disconnected(void)
324 {
325 LOG_INF("Disconnected from network");
326 lwm2m_engine_pause();
327 }
328
on_net_event_l4_connected(void)329 static void on_net_event_l4_connected(void)
330 {
331 LOG_INF("Connected to network");
332 k_sem_give(&network_connected_sem);
333 lwm2m_engine_resume();
334 }
335
l4_event_handler(struct net_mgmt_event_callback * cb,uint32_t event,struct net_if * iface)336 static void l4_event_handler(struct net_mgmt_event_callback *cb,
337 uint32_t event,
338 struct net_if *iface)
339 {
340 switch (event) {
341 case NET_EVENT_L4_CONNECTED:
342 LOG_INF("IP Up");
343 on_net_event_l4_connected();
344 break;
345 case NET_EVENT_L4_DISCONNECTED:
346 LOG_INF("IP down");
347 on_net_event_l4_disconnected();
348 break;
349 default:
350 break;
351 }
352 }
353
connectivity_event_handler(struct net_mgmt_event_callback * cb,uint32_t event,struct net_if * iface)354 static void connectivity_event_handler(struct net_mgmt_event_callback *cb,
355 uint32_t event,
356 struct net_if *iface)
357 {
358 if (event == NET_EVENT_CONN_IF_FATAL_ERROR) {
359 LOG_ERR("Fatal error received from the connectivity layer");
360 return;
361 }
362 }
363
main(void)364 int main(void)
365 {
366 uint32_t flags = IS_ENABLED(CONFIG_LWM2M_RD_CLIENT_SUPPORT_BOOTSTRAP) ?
367 LWM2M_RD_CLIENT_FLAG_BOOTSTRAP : 0;
368 int ret;
369
370 LOG_INF(APP_BANNER);
371
372 k_sem_init(&quit_lock, 0, K_SEM_MAX_LIMIT);
373
374 if (IS_ENABLED(CONFIG_NET_CONNECTION_MANAGER)) {
375 /* Setup handler for Zephyr NET Connection Manager events. */
376 net_mgmt_init_event_callback(&l4_cb, l4_event_handler, L4_EVENT_MASK);
377 net_mgmt_add_event_callback(&l4_cb);
378
379 /* Setup handler for Zephyr NET Connection Manager Connectivity layer. */
380 net_mgmt_init_event_callback(&conn_cb, connectivity_event_handler,
381 CONN_LAYER_EVENT_MASK);
382 net_mgmt_add_event_callback(&conn_cb);
383
384 ret = net_if_up(net_if_get_default());
385
386 if (ret < 0 && ret != -EALREADY) {
387 LOG_ERR("net_if_up, error: %d", ret);
388 return ret;
389 }
390
391 ret = conn_mgr_if_connect(net_if_get_default());
392 /* Ignore errors from interfaces not requiring connectivity */
393 if (ret == 0) {
394 LOG_INF("Connecting to network");
395 k_sem_take(&network_connected_sem, K_FOREVER);
396 }
397 }
398
399 ret = lwm2m_setup();
400 if (ret < 0) {
401 LOG_ERR("Cannot setup LWM2M fields (%d)", ret);
402 return 0;
403 }
404
405 (void)memset(&client_ctx, 0x0, sizeof(client_ctx));
406 #if defined(CONFIG_LWM2M_DTLS_SUPPORT)
407 client_ctx.tls_tag = CONFIG_LWM2M_APP_TLS_TAG;
408 #endif
409 client_ctx.set_socket_state = socket_state;
410
411 /* client_ctx.sec_obj_inst is 0 as a starting point */
412 lwm2m_rd_client_start(&client_ctx, endpoint, flags, rd_client_event, observe_cb);
413
414 k_sem_take(&quit_lock, K_FOREVER);
415 return 0;
416 }
417