1 /*
2  * Copyright (c) 2017 Linaro Limited
3  * Copyright (c) 2018-2019 Foundries.io
4  * Copyright (c) 2022 Laird Connectivity
5  *
6  * SPDX-License-Identifier: Apache-2.0
7  */
8 
9 /*
10  * Source material for uCIFI battery object (3411):
11  * https://raw.githubusercontent.com/OpenMobileAlliance/lwm2m-registry/prod/3411.xml
12  */
13 
14 #define LOG_MODULE_NAME net_ucifi_battery
15 #define LOG_LEVEL CONFIG_LWM2M_LOG_LEVEL
16 
17 #include <zephyr/logging/log.h>
18 LOG_MODULE_REGISTER(LOG_MODULE_NAME);
19 
20 #include <stdint.h>
21 #include <zephyr/init.h>
22 
23 #include "lwm2m_object.h"
24 #include "lwm2m_engine.h"
25 #include "lwm2m_resource_ids.h"
26 #include "ucifi_battery.h"
27 
28 #define BATTERY_VERSION_MAJOR 1
29 #define BATTERY_VERSION_MINOR 0
30 
31 #define MAX_INSTANCE_COUNT CONFIG_LWM2M_UCIFI_BATTERY_INSTANCE_COUNT
32 
33 #define BATTERY_MAX_ID 12
34 #define RESOURCE_INSTANCE_COUNT (BATTERY_MAX_ID)
35 
36 /* resource state variables */
37 static uint8_t battery_level[MAX_INSTANCE_COUNT];
38 static uint32_t supply_loss_counter[MAX_INSTANCE_COUNT];
39 
40 static struct lwm2m_engine_obj battery;
41 static struct lwm2m_engine_obj_field fields[] = {
42 	OBJ_FIELD_DATA(UCIFI_BATTERY_LEVEL_RID, R, U8),
43 	OBJ_FIELD_DATA(UCIFI_BATTERY_CAPACITY_RID, R_OPT, FLOAT),
44 	OBJ_FIELD_DATA(UCIFI_BATTERY_VOLTAGE_RID, R_OPT, FLOAT),
45 	OBJ_FIELD_DATA(UCIFI_BATTERY_TYPE_RID, RW_OPT, STRING),
46 	OBJ_FIELD_DATA(UCIFI_BATTERY_LOW_THESHOLD_RID, RW_OPT, U8),
47 	OBJ_FIELD_DATA(UCIFI_BATTERY_LEVEL_TOO_LOW_RID, R_OPT, BOOL),
48 	OBJ_FIELD_DATA(UCIFI_BATTERY_SHUTDOWN_RID, RW_OPT, BOOL),
49 	OBJ_FIELD_DATA(UCIFI_BATTERY_NUM_CYCLES_RID, R_OPT, U32),
50 	OBJ_FIELD_DATA(UCIFI_BATTERY_SUPPLY_LOSS_RID, R_OPT, BOOL),
51 	OBJ_FIELD_DATA(UCIFI_BATTERY_SUPPLY_LOSS_COUNTER_RID, R_OPT, U32),
52 	OBJ_FIELD_EXECUTE_OPT(UCIFI_BATTERY_SUPPLY_LOSS_COUNTER_RESET_RID),
53 	OBJ_FIELD_DATA(UCIFI_BATTERY_SUPPLY_LOSS_REASON_RID, R_OPT, STRING),
54 };
55 
56 static struct lwm2m_engine_obj_inst inst[MAX_INSTANCE_COUNT];
57 static struct lwm2m_engine_res res[MAX_INSTANCE_COUNT][BATTERY_MAX_ID];
58 static struct lwm2m_engine_res_inst res_inst[MAX_INSTANCE_COUNT][RESOURCE_INSTANCE_COUNT];
59 
clear_supply_loss_counter(uint16_t obj_inst_id,int index)60 static void clear_supply_loss_counter(uint16_t obj_inst_id, int index)
61 {
62 	supply_loss_counter[index] = 0;
63 	lwm2m_notify_observer(UCIFI_OBJECT_BATTERY_ID, obj_inst_id,
64 			UCIFI_BATTERY_SUPPLY_LOSS_COUNTER_RID);
65 }
66 
supply_loss_counter_reset_cb(uint16_t obj_inst_id,uint8_t * args,uint16_t args_len)67 static int supply_loss_counter_reset_cb(uint16_t obj_inst_id, uint8_t *args, uint16_t args_len)
68 {
69 	int i;
70 
71 	LOG_DBG("RESET supply loss counter %d", obj_inst_id);
72 	for (i = 0; i < MAX_INSTANCE_COUNT; i++) {
73 		if (inst[i].obj && inst[i].obj_inst_id == obj_inst_id) {
74 			clear_supply_loss_counter(obj_inst_id, i);
75 			return 0;
76 		}
77 	}
78 
79 	return -ENOENT;
80 }
81 
battery_create(uint16_t obj_inst_id)82 static struct lwm2m_engine_obj_inst *battery_create(uint16_t obj_inst_id)
83 {
84 	int index, i = 0, j = 0;
85 
86 	/* Check that there is no other instance with this ID */
87 	for (index = 0; index < MAX_INSTANCE_COUNT; index++) {
88 		if (inst[index].obj && inst[index].obj_inst_id == obj_inst_id) {
89 			LOG_ERR("Can not create instance - "
90 				"already existing: %u",
91 				obj_inst_id);
92 			return NULL;
93 		}
94 	}
95 
96 	for (index = 0; index < MAX_INSTANCE_COUNT; index++) {
97 		if (!inst[index].obj) {
98 			break;
99 		}
100 	}
101 
102 	if (index >= MAX_INSTANCE_COUNT) {
103 		LOG_ERR("Can not create instance - no more room: %u", obj_inst_id);
104 		return NULL;
105 	}
106 
107 	/* Set default values */
108 	battery_level[index] = 0;
109 	supply_loss_counter[index] = 0;
110 
111 	(void)memset(res[index], 0, sizeof(res[index][0]) * ARRAY_SIZE(res[index]));
112 	init_res_instance(res_inst[index], ARRAY_SIZE(res_inst[index]));
113 
114 	/* initialize instance resource data */
115 	INIT_OBJ_RES_DATA(UCIFI_BATTERY_LEVEL_RID, res[index], i, res_inst[index], j,
116 			  &battery_level[index], sizeof(*battery_level));
117 	INIT_OBJ_RES_OPTDATA(UCIFI_BATTERY_CAPACITY_RID, res[index], i, res_inst[index], j);
118 	INIT_OBJ_RES_OPTDATA(UCIFI_BATTERY_VOLTAGE_RID, res[index], i, res_inst[index], j);
119 	INIT_OBJ_RES_OPTDATA(UCIFI_BATTERY_TYPE_RID, res[index], i, res_inst[index], j);
120 	INIT_OBJ_RES_OPTDATA(UCIFI_BATTERY_LOW_THESHOLD_RID, res[index], i, res_inst[index], j);
121 	INIT_OBJ_RES_OPTDATA(UCIFI_BATTERY_LEVEL_TOO_LOW_RID, res[index], i, res_inst[index], j);
122 	INIT_OBJ_RES_OPTDATA(UCIFI_BATTERY_SHUTDOWN_RID, res[index], i, res_inst[index], j);
123 	INIT_OBJ_RES_OPTDATA(UCIFI_BATTERY_NUM_CYCLES_RID, res[index], i, res_inst[index], j);
124 	INIT_OBJ_RES_OPTDATA(UCIFI_BATTERY_SUPPLY_LOSS_RID, res[index], i, res_inst[index], j);
125 	INIT_OBJ_RES_DATA(UCIFI_BATTERY_SUPPLY_LOSS_COUNTER_RID, res[index], i, res_inst[index], j,
126 			  &supply_loss_counter[index], sizeof(*supply_loss_counter));
127 	INIT_OBJ_RES_EXECUTE(UCIFI_BATTERY_SUPPLY_LOSS_COUNTER_RESET_RID, res[index], i,
128 			     supply_loss_counter_reset_cb);
129 	INIT_OBJ_RES_OPTDATA(UCIFI_BATTERY_SUPPLY_LOSS_REASON_RID, res[index], i, res_inst[index],
130 			     j);
131 
132 	inst[index].resources = res[index];
133 	inst[index].resource_count = i;
134 	LOG_DBG("Create uCIFI Battery instance: %d", obj_inst_id);
135 	return &inst[index];
136 }
137 
ucifi_battery_init(void)138 static int ucifi_battery_init(void)
139 {
140 	battery.obj_id = UCIFI_OBJECT_BATTERY_ID;
141 	battery.version_major = BATTERY_VERSION_MAJOR;
142 	battery.version_minor = BATTERY_VERSION_MINOR;
143 	battery.is_core = false;
144 	battery.fields = fields;
145 	battery.field_count = ARRAY_SIZE(fields);
146 	battery.max_instance_count = MAX_INSTANCE_COUNT;
147 	battery.create_cb = battery_create;
148 	lwm2m_register_obj(&battery);
149 
150 	return 0;
151 }
152 
153 SYS_INIT(ucifi_battery_init, APPLICATION, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);
154