config.c revision 1.12 1 1.12 joerg /* $NetBSD: config.c,v 1.12 2011/08/31 13:32:36 joerg Exp $ */
2 1.1 xtraeme
3 1.1 xtraeme /*-
4 1.1 xtraeme * Copyright (c) 2007 Juan Romero Pardines.
5 1.1 xtraeme * All rights reserved.
6 1.1 xtraeme *
7 1.1 xtraeme * Redistribution and use in source and binary forms, with or without
8 1.1 xtraeme * modification, are permitted provided that the following conditions
9 1.1 xtraeme * are met:
10 1.1 xtraeme * 1. Redistributions of source code must retain the above copyright
11 1.1 xtraeme * notice, this list of conditions and the following disclaimer.
12 1.1 xtraeme * 2. Redistributions in binary form must reproduce the above copyright
13 1.1 xtraeme * notice, this list of conditions and the following disclaimer in the
14 1.1 xtraeme * documentation and/or other materials provided with the distribution.
15 1.1 xtraeme *
16 1.1 xtraeme * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 1.1 xtraeme * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18 1.1 xtraeme * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19 1.1 xtraeme * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20 1.1 xtraeme * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21 1.1 xtraeme * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22 1.1 xtraeme * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23 1.1 xtraeme * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 1.1 xtraeme * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25 1.1 xtraeme * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 1.1 xtraeme */
27 1.1 xtraeme
28 1.1 xtraeme #include <sys/cdefs.h>
29 1.1 xtraeme #ifndef lint
30 1.12 joerg __RCSID("$NetBSD: config.c,v 1.12 2011/08/31 13:32:36 joerg Exp $");
31 1.1 xtraeme #endif /* not lint */
32 1.1 xtraeme
33 1.1 xtraeme #include <stdio.h>
34 1.1 xtraeme #include <string.h>
35 1.1 xtraeme #include <stdlib.h>
36 1.1 xtraeme #include <err.h>
37 1.1 xtraeme #include <errno.h>
38 1.1 xtraeme #include <sys/queue.h>
39 1.1 xtraeme #include <prop/proplib.h>
40 1.1 xtraeme
41 1.1 xtraeme #include "envstat.h"
42 1.1 xtraeme
43 1.1 xtraeme /*
44 1.1 xtraeme * Singly linked list for dictionaries that store properties
45 1.1 xtraeme * in a sensor.
46 1.1 xtraeme */
47 1.1 xtraeme static SLIST_HEAD(, sensor_block) sensor_block_list =
48 1.1 xtraeme SLIST_HEAD_INITIALIZER(&sensor_block_list);
49 1.1 xtraeme
50 1.1 xtraeme /*
51 1.1 xtraeme * Singly linked list for devices that store a proplib array
52 1.1 xtraeme * device with a device name.
53 1.1 xtraeme */
54 1.1 xtraeme static SLIST_HEAD(, device_block) device_block_list =
55 1.1 xtraeme SLIST_HEAD_INITIALIZER(&device_block_list);
56 1.1 xtraeme
57 1.1 xtraeme enum {
58 1.1 xtraeme VALUE_ERR,
59 1.1 xtraeme PROP_ERR,
60 1.1 xtraeme SENSOR_ERR,
61 1.1 xtraeme DEV_ERR
62 1.1 xtraeme };
63 1.1 xtraeme
64 1.6 xtraeme static prop_dictionary_t cfdict, sensordict, refreshdict;
65 1.12 joerg __dead static void config_errmsg(int, const char *, const char *);
66 1.1 xtraeme
67 1.1 xtraeme static void
68 1.1 xtraeme config_errmsg(int lvl, const char *key, const char *key2)
69 1.1 xtraeme {
70 1.6 xtraeme (void)printf("envstat: ");
71 1.1 xtraeme
72 1.1 xtraeme switch (lvl) {
73 1.1 xtraeme case VALUE_ERR:
74 1.1 xtraeme (void)printf("invalid value for '%s' in `%s'\n",
75 1.1 xtraeme key, key2);
76 1.1 xtraeme break;
77 1.1 xtraeme case PROP_ERR:
78 1.1 xtraeme (void)printf("the '%s' property is not allowed "
79 1.1 xtraeme "in `%s'\n", key, key2);
80 1.1 xtraeme break;
81 1.1 xtraeme case SENSOR_ERR:
82 1.1 xtraeme (void)printf("'%s' is not a valid sensor in the "
83 1.1 xtraeme "`%s' device\n", key, key2);
84 1.1 xtraeme break;
85 1.1 xtraeme case DEV_ERR:
86 1.1 xtraeme (void)printf("device `%s' doesn't exist\n", key);
87 1.1 xtraeme break;
88 1.1 xtraeme }
89 1.1 xtraeme
90 1.6 xtraeme (void)printf("envstat: please fix the configuration file!\n");
91 1.1 xtraeme exit(EXIT_FAILURE);
92 1.1 xtraeme }
93 1.1 xtraeme
94 1.1 xtraeme /*
95 1.6 xtraeme * Adds a property into a temporary dictionary.
96 1.1 xtraeme */
97 1.1 xtraeme void
98 1.1 xtraeme config_dict_add_prop(const char *key, char *value)
99 1.1 xtraeme {
100 1.10 pgoyette
101 1.1 xtraeme if (!key || !value)
102 1.1 xtraeme return;
103 1.1 xtraeme
104 1.2 xtraeme if (!sensordict) {
105 1.2 xtraeme sensordict = prop_dictionary_create();
106 1.2 xtraeme if (!sensordict)
107 1.7 xtraeme err(EXIT_FAILURE, "sensordict");
108 1.1 xtraeme }
109 1.1 xtraeme
110 1.2 xtraeme if (!prop_dictionary_set_cstring(sensordict, key, value))
111 1.1 xtraeme err(EXIT_FAILURE, "prop_dict_set_cstring");
112 1.1 xtraeme }
113 1.1 xtraeme
114 1.1 xtraeme /*
115 1.6 xtraeme * Marks sensor's dictionary to say that it's the last property
116 1.6 xtraeme * and the dictionary should be added into the singly linked list.
117 1.1 xtraeme */
118 1.1 xtraeme void
119 1.6 xtraeme config_dict_mark(void)
120 1.1 xtraeme {
121 1.1 xtraeme struct sensor_block *sb;
122 1.1 xtraeme
123 1.1 xtraeme sb = calloc(1, sizeof(*sb));
124 1.1 xtraeme if (!sb)
125 1.1 xtraeme err(EXIT_FAILURE, "!sb");
126 1.1 xtraeme
127 1.1 xtraeme sb->dict = prop_dictionary_create();
128 1.1 xtraeme if (!sb->dict)
129 1.1 xtraeme err(EXIT_FAILURE, "!sb->dict");
130 1.1 xtraeme
131 1.2 xtraeme sb->dict = prop_dictionary_copy(sensordict);
132 1.1 xtraeme SLIST_INSERT_HEAD(&sensor_block_list, sb, sb_head);
133 1.2 xtraeme config_dict_destroy(sensordict);
134 1.1 xtraeme }
135 1.1 xtraeme
136 1.1 xtraeme /*
137 1.1 xtraeme * Only used for debugging purposses.
138 1.1 xtraeme */
139 1.1 xtraeme void
140 1.1 xtraeme config_dict_dump(prop_dictionary_t d)
141 1.1 xtraeme {
142 1.1 xtraeme char *buf;
143 1.1 xtraeme
144 1.1 xtraeme buf = prop_dictionary_externalize(d);
145 1.1 xtraeme (void)printf("%s", buf);
146 1.1 xtraeme free(buf);
147 1.1 xtraeme }
148 1.1 xtraeme
149 1.1 xtraeme /*
150 1.1 xtraeme * Returns the global dictionary.
151 1.1 xtraeme */
152 1.1 xtraeme prop_dictionary_t
153 1.1 xtraeme config_dict_parsed(void)
154 1.1 xtraeme {
155 1.1 xtraeme return cfdict;
156 1.1 xtraeme }
157 1.1 xtraeme
158 1.1 xtraeme /*
159 1.6 xtraeme * To add device properties into the global array, for now only the
160 1.6 xtraeme * 'refresh-timeout' property is accepted.
161 1.6 xtraeme */
162 1.6 xtraeme void
163 1.6 xtraeme config_dict_adddev_prop(const char *key, const char *value, int line)
164 1.6 xtraeme {
165 1.6 xtraeme prop_dictionary_t d = NULL;
166 1.6 xtraeme uint64_t timo;
167 1.6 xtraeme size_t len;
168 1.6 xtraeme char *endptr, *tmp, *strval;
169 1.6 xtraeme bool minutes, hours;
170 1.6 xtraeme
171 1.6 xtraeme minutes = hours = false;
172 1.6 xtraeme
173 1.6 xtraeme /*
174 1.6 xtraeme * Check what was specified: seconds, minutes or hours.
175 1.6 xtraeme */
176 1.6 xtraeme if ((tmp = strchr(value, 's'))) {
177 1.6 xtraeme /*
178 1.6 xtraeme * do nothing, by default the value will be sent as seconds.
179 1.6 xtraeme */
180 1.6 xtraeme } else if ((tmp = strchr(value, 'm'))) {
181 1.6 xtraeme minutes = true;
182 1.6 xtraeme } else if ((tmp = strchr(value, 'h'))) {
183 1.6 xtraeme hours = true;
184 1.6 xtraeme } else
185 1.6 xtraeme goto bad;
186 1.6 xtraeme
187 1.6 xtraeme len = strlen(value);
188 1.6 xtraeme strval = calloc(len, sizeof(*value));
189 1.6 xtraeme if (!strval)
190 1.6 xtraeme err(EXIT_FAILURE, "calloc");
191 1.6 xtraeme
192 1.6 xtraeme (void)strlcpy(strval, value, len);
193 1.6 xtraeme
194 1.6 xtraeme timo = strtoul(strval, &endptr, 10);
195 1.6 xtraeme if (*endptr != '\0') {
196 1.6 xtraeme free(strval);
197 1.6 xtraeme goto bad;
198 1.6 xtraeme }
199 1.6 xtraeme
200 1.6 xtraeme free(strval);
201 1.6 xtraeme
202 1.7 xtraeme refreshdict = prop_dictionary_create();
203 1.7 xtraeme if (!refreshdict)
204 1.7 xtraeme err(EXIT_FAILURE, "prop_dict_create refresh");
205 1.6 xtraeme
206 1.6 xtraeme d = prop_dictionary_create();
207 1.6 xtraeme if (!d)
208 1.6 xtraeme err(EXIT_FAILURE, "prop_dict_create refresh 1");
209 1.6 xtraeme
210 1.6 xtraeme if (minutes)
211 1.6 xtraeme timo *= 60;
212 1.6 xtraeme else if (hours) {
213 1.6 xtraeme /*
214 1.6 xtraeme * Make sure the value is not too high...
215 1.6 xtraeme */
216 1.6 xtraeme if (timo > 999)
217 1.6 xtraeme goto bad;
218 1.6 xtraeme timo *= 60 * 60;
219 1.6 xtraeme } else {
220 1.6 xtraeme /*
221 1.6 xtraeme * 1 second is the lowest value allowed.
222 1.6 xtraeme */
223 1.6 xtraeme if (timo < 1)
224 1.6 xtraeme goto bad;
225 1.6 xtraeme }
226 1.6 xtraeme
227 1.6 xtraeme if (!prop_dictionary_set_uint64(d, key, timo))
228 1.6 xtraeme err(EXIT_FAILURE, "%s", key);
229 1.6 xtraeme
230 1.6 xtraeme if (!prop_dictionary_set(refreshdict, "device-properties", d))
231 1.6 xtraeme err(EXIT_FAILURE, "device-properties %s", key);
232 1.6 xtraeme
233 1.6 xtraeme prop_object_release(d);
234 1.6 xtraeme return;
235 1.6 xtraeme
236 1.6 xtraeme bad:
237 1.6 xtraeme (void)printf("envstat: invalid value for the '%s' "
238 1.6 xtraeme "property at line %d\n", key, line);
239 1.6 xtraeme (void)printf("envstat: please fix the configuration file!\n");
240 1.6 xtraeme if (d)
241 1.6 xtraeme prop_object_release(d);
242 1.6 xtraeme
243 1.6 xtraeme exit(EXIT_FAILURE);
244 1.6 xtraeme }
245 1.6 xtraeme
246 1.6 xtraeme /*
247 1.1 xtraeme * Destroys all objects from a dictionary.
248 1.1 xtraeme */
249 1.1 xtraeme void
250 1.1 xtraeme config_dict_destroy(prop_dictionary_t d)
251 1.1 xtraeme {
252 1.1 xtraeme prop_object_iterator_t iter;
253 1.1 xtraeme prop_object_t obj;
254 1.1 xtraeme
255 1.1 xtraeme iter = prop_dictionary_iterator(d);
256 1.1 xtraeme if (!iter)
257 1.1 xtraeme err(EXIT_FAILURE, "!iter");
258 1.1 xtraeme
259 1.1 xtraeme while ((obj = prop_object_iterator_next(iter)) != NULL) {
260 1.1 xtraeme prop_dictionary_remove(d,
261 1.1 xtraeme prop_dictionary_keysym_cstring_nocopy(obj));
262 1.1 xtraeme prop_object_iterator_reset(iter);
263 1.1 xtraeme }
264 1.1 xtraeme
265 1.1 xtraeme prop_object_iterator_release(iter);
266 1.1 xtraeme }
267 1.1 xtraeme
268 1.1 xtraeme /*
269 1.1 xtraeme * Parses all properties on the device and adds the device
270 1.1 xtraeme * into the singly linked list for devices and the global dictionary.
271 1.1 xtraeme */
272 1.1 xtraeme void
273 1.1 xtraeme config_devblock_add(const char *key, prop_dictionary_t kdict)
274 1.1 xtraeme {
275 1.1 xtraeme struct device_block *db;
276 1.1 xtraeme struct sensor_block *sb;
277 1.1 xtraeme prop_array_t array;
278 1.1 xtraeme prop_object_iterator_t iter;
279 1.1 xtraeme prop_dictionary_t sdict;
280 1.1 xtraeme prop_object_t obj;
281 1.1 xtraeme prop_string_t lindex;
282 1.1 xtraeme const char *sensor;
283 1.1 xtraeme bool sensor_found = false;
284 1.1 xtraeme
285 1.1 xtraeme if (!key)
286 1.1 xtraeme err(EXIT_FAILURE, "devblock !key");
287 1.1 xtraeme
288 1.1 xtraeme array = prop_dictionary_get(kdict, key);
289 1.1 xtraeme if (!array)
290 1.1 xtraeme config_errmsg(DEV_ERR, key, NULL);
291 1.1 xtraeme
292 1.1 xtraeme SLIST_FOREACH(sb, &sensor_block_list, sb_head) {
293 1.1 xtraeme /* get the index object value from configuration */
294 1.1 xtraeme lindex = prop_dictionary_get(sb->dict, "index");
295 1.1 xtraeme sensor = prop_string_cstring_nocopy(lindex);
296 1.1 xtraeme
297 1.1 xtraeme iter = prop_array_iterator(array);
298 1.1 xtraeme if (!iter)
299 1.1 xtraeme err(EXIT_FAILURE, "prop_array_iterator devblock");
300 1.1 xtraeme
301 1.1 xtraeme /*
302 1.1 xtraeme * Get the correct sensor's dictionary from kernel's
303 1.1 xtraeme * dictionary.
304 1.1 xtraeme */
305 1.1 xtraeme while ((sdict = prop_object_iterator_next(iter)) != NULL) {
306 1.1 xtraeme obj = prop_dictionary_get(sdict, "index");
307 1.1 xtraeme if (prop_string_equals(lindex, obj)) {
308 1.1 xtraeme sensor_found = true;
309 1.1 xtraeme break;
310 1.1 xtraeme }
311 1.1 xtraeme }
312 1.1 xtraeme
313 1.1 xtraeme if (!sensor_found) {
314 1.1 xtraeme prop_object_iterator_release(iter);
315 1.1 xtraeme config_errmsg(SENSOR_ERR, sensor, key);
316 1.1 xtraeme }
317 1.1 xtraeme
318 1.1 xtraeme config_devblock_check_sensorprops(sdict, sb->dict, sensor);
319 1.1 xtraeme prop_object_iterator_release(iter);
320 1.1 xtraeme }
321 1.1 xtraeme
322 1.1 xtraeme db = calloc(1, sizeof(*db));
323 1.1 xtraeme if (!db)
324 1.1 xtraeme err(EXIT_FAILURE, "calloc db");
325 1.1 xtraeme
326 1.1 xtraeme db->array = prop_array_create();
327 1.1 xtraeme if (!db->array)
328 1.1 xtraeme err(EXIT_FAILURE, "prop_array_create devblock");
329 1.1 xtraeme
330 1.1 xtraeme /*
331 1.1 xtraeme * Add all dictionaries into the array.
332 1.1 xtraeme */
333 1.1 xtraeme SLIST_FOREACH(sb, &sensor_block_list, sb_head)
334 1.1 xtraeme if (!prop_array_add(db->array, sb->dict))
335 1.1 xtraeme err(EXIT_FAILURE, "prop_array_add");
336 1.1 xtraeme
337 1.1 xtraeme /*
338 1.6 xtraeme * Add the device-properties dictionary into the array.
339 1.6 xtraeme */
340 1.6 xtraeme if (refreshdict) {
341 1.6 xtraeme if (!prop_array_add(db->array, refreshdict))
342 1.6 xtraeme err(EXIT_FAILURE, "prop_array_add refreshdict");
343 1.6 xtraeme prop_object_release(refreshdict);
344 1.6 xtraeme }
345 1.6 xtraeme
346 1.6 xtraeme /*
347 1.1 xtraeme * Add this device block into our list.
348 1.1 xtraeme */
349 1.1 xtraeme db->dev_key = strdup(key);
350 1.1 xtraeme SLIST_INSERT_HEAD(&device_block_list, db, db_head);
351 1.1 xtraeme
352 1.1 xtraeme /*
353 1.1 xtraeme * Remove all items in the list, but just decrement
354 1.1 xtraeme * the refcnt in the dictionaries... they are in use.
355 1.1 xtraeme */
356 1.1 xtraeme while (!SLIST_EMPTY(&sensor_block_list)) {
357 1.1 xtraeme sb = SLIST_FIRST(&sensor_block_list);
358 1.1 xtraeme SLIST_REMOVE_HEAD(&sensor_block_list, sb_head);
359 1.1 xtraeme prop_object_release(sb->dict);
360 1.1 xtraeme free(sb);
361 1.1 xtraeme }
362 1.1 xtraeme
363 1.1 xtraeme /*
364 1.1 xtraeme * Now the properties on the array has been parsed,
365 1.1 xtraeme * add it into the global dict.
366 1.1 xtraeme */
367 1.2 xtraeme if (!cfdict) {
368 1.2 xtraeme cfdict = prop_dictionary_create();
369 1.2 xtraeme if (!cfdict)
370 1.2 xtraeme err(EXIT_FAILURE, "prop_dictionary_create cfdict");
371 1.2 xtraeme }
372 1.2 xtraeme
373 1.1 xtraeme if (!prop_dictionary_set(cfdict, key, db->array))
374 1.1 xtraeme err(EXIT_FAILURE, "prop_dictionary_set db->array");
375 1.2 xtraeme
376 1.7 xtraeme /*
377 1.7 xtraeme * refreshdict must be NULLed to avoid false positives in
378 1.7 xtraeme * next matches.
379 1.7 xtraeme */
380 1.7 xtraeme refreshdict = NULL;
381 1.1 xtraeme }
382 1.1 xtraeme
383 1.1 xtraeme /*
384 1.1 xtraeme * Returns the dictionary that has 'sensor_key' in the 'dvname'
385 1.1 xtraeme * array.
386 1.1 xtraeme */
387 1.1 xtraeme prop_dictionary_t
388 1.1 xtraeme config_devblock_getdict(const char *dvname, const char *sensor_key)
389 1.1 xtraeme {
390 1.1 xtraeme struct device_block *db;
391 1.1 xtraeme prop_object_iterator_t iter;
392 1.1 xtraeme prop_object_t obj, obj2;
393 1.1 xtraeme
394 1.1 xtraeme if (!dvname || !sensor_key)
395 1.1 xtraeme return NULL;
396 1.1 xtraeme
397 1.1 xtraeme SLIST_FOREACH(db, &device_block_list, db_head)
398 1.1 xtraeme if (strcmp(db->dev_key, dvname) == 0)
399 1.1 xtraeme break;
400 1.1 xtraeme
401 1.1 xtraeme if (!db)
402 1.1 xtraeme return NULL;
403 1.1 xtraeme
404 1.1 xtraeme iter = prop_array_iterator(db->array);
405 1.1 xtraeme if (!iter)
406 1.1 xtraeme return NULL;
407 1.1 xtraeme
408 1.1 xtraeme while ((obj = prop_object_iterator_next(iter)) != NULL) {
409 1.1 xtraeme obj2 = prop_dictionary_get(obj, "index");
410 1.1 xtraeme if (prop_string_equals_cstring(obj2, sensor_key))
411 1.1 xtraeme break;
412 1.1 xtraeme }
413 1.1 xtraeme
414 1.1 xtraeme prop_object_iterator_release(iter);
415 1.1 xtraeme return obj;
416 1.1 xtraeme }
417 1.1 xtraeme
418 1.1 xtraeme /*
419 1.1 xtraeme * Checks that all properties specified in the configuration file
420 1.1 xtraeme * are valid and updates the objects with proper values.
421 1.1 xtraeme */
422 1.1 xtraeme void
423 1.1 xtraeme config_devblock_check_sensorprops(prop_dictionary_t ksdict,
424 1.1 xtraeme prop_dictionary_t csdict,
425 1.1 xtraeme const char *sensor)
426 1.1 xtraeme {
427 1.3 xtraeme prop_object_t obj, obj2, obj3;
428 1.1 xtraeme char *strval, *endptr;
429 1.1 xtraeme double val;
430 1.1 xtraeme
431 1.1 xtraeme /*
432 1.1 xtraeme * rfact property set?
433 1.1 xtraeme */
434 1.1 xtraeme obj = prop_dictionary_get(csdict, "rfact");
435 1.1 xtraeme if (obj) {
436 1.1 xtraeme obj2 = prop_dictionary_get(ksdict, "allow-rfact");
437 1.1 xtraeme if (prop_bool_true(obj2)) {
438 1.1 xtraeme strval = prop_string_cstring(obj);
439 1.1 xtraeme val = strtod(strval, &endptr);
440 1.1 xtraeme if (*endptr != '\0')
441 1.1 xtraeme config_errmsg(VALUE_ERR, "rfact", sensor);
442 1.1 xtraeme
443 1.1 xtraeme if (!prop_dictionary_set_uint32(csdict, "rfact", val))
444 1.1 xtraeme err(EXIT_FAILURE, "dict_set rfact");
445 1.1 xtraeme } else
446 1.1 xtraeme config_errmsg(PROP_ERR, "rfact", sensor);
447 1.1 xtraeme }
448 1.1 xtraeme
449 1.1 xtraeme /*
450 1.1 xtraeme * critical-capacity property set?
451 1.1 xtraeme */
452 1.1 xtraeme obj = prop_dictionary_get(csdict, "critical-capacity");
453 1.1 xtraeme if (obj) {
454 1.1 xtraeme obj2 = prop_dictionary_get(ksdict, "want-percentage");
455 1.3 xtraeme obj3 = prop_dictionary_get(ksdict, "monitoring-supported");
456 1.3 xtraeme if (prop_bool_true(obj2) && prop_bool_true(obj3)) {
457 1.1 xtraeme strval = prop_string_cstring(obj);
458 1.1 xtraeme val = strtod(strval, &endptr);
459 1.1 xtraeme if ((*endptr != '\0') || (val < 0 || val > 100))
460 1.1 xtraeme config_errmsg(VALUE_ERR,
461 1.1 xtraeme "critical-capacity",
462 1.1 xtraeme sensor);
463 1.1 xtraeme /*
464 1.1 xtraeme * Convert the value to a valid percentage.
465 1.1 xtraeme */
466 1.1 xtraeme obj = prop_dictionary_get(ksdict, "max-value");
467 1.1 xtraeme val = (val / 100) * prop_number_integer_value(obj);
468 1.1 xtraeme
469 1.4 xtraeme if (!prop_dictionary_set_uint32(csdict,
470 1.1 xtraeme "critical-capacity",
471 1.1 xtraeme val))
472 1.1 xtraeme err(EXIT_FAILURE, "dict_set critcap");
473 1.1 xtraeme } else
474 1.6 xtraeme config_errmsg(PROP_ERR, "critical-capacity", sensor);
475 1.1 xtraeme }
476 1.1 xtraeme
477 1.1 xtraeme /*
478 1.8 pgoyette * warning-capacity property set?
479 1.8 pgoyette */
480 1.8 pgoyette obj = prop_dictionary_get(csdict, "warning-capacity");
481 1.8 pgoyette if (obj) {
482 1.8 pgoyette obj2 = prop_dictionary_get(ksdict, "want-percentage");
483 1.8 pgoyette obj3 = prop_dictionary_get(ksdict, "monitoring-supported");
484 1.8 pgoyette if (prop_bool_true(obj2) && prop_bool_true(obj3)) {
485 1.8 pgoyette strval = prop_string_cstring(obj);
486 1.8 pgoyette val = strtod(strval, &endptr);
487 1.8 pgoyette if ((*endptr != '\0') || (val < 0 || val > 100))
488 1.8 pgoyette config_errmsg(VALUE_ERR,
489 1.8 pgoyette "warning-capacity",
490 1.8 pgoyette sensor);
491 1.8 pgoyette /*
492 1.8 pgoyette * Convert the value to a valid percentage.
493 1.8 pgoyette */
494 1.8 pgoyette obj = prop_dictionary_get(ksdict, "max-value");
495 1.8 pgoyette val = (val / 100) * prop_number_integer_value(obj);
496 1.8 pgoyette
497 1.8 pgoyette if (!prop_dictionary_set_uint32(csdict,
498 1.8 pgoyette "warning-capacity",
499 1.8 pgoyette val))
500 1.8 pgoyette err(EXIT_FAILURE, "dict_set warncap");
501 1.8 pgoyette } else
502 1.8 pgoyette config_errmsg(PROP_ERR, "warning-capacity", sensor);
503 1.8 pgoyette }
504 1.8 pgoyette
505 1.8 pgoyette /*
506 1.9 pgoyette * high-capacity property set?
507 1.9 pgoyette */
508 1.9 pgoyette obj = prop_dictionary_get(csdict, "high-capacity");
509 1.9 pgoyette if (obj) {
510 1.9 pgoyette obj2 = prop_dictionary_get(ksdict, "want-percentage");
511 1.9 pgoyette obj3 = prop_dictionary_get(ksdict, "monitoring-supported");
512 1.9 pgoyette if (prop_bool_true(obj2) && prop_bool_true(obj3)) {
513 1.9 pgoyette strval = prop_string_cstring(obj);
514 1.9 pgoyette val = strtod(strval, &endptr);
515 1.9 pgoyette if ((*endptr != '\0') || (val < 0 || val > 100))
516 1.9 pgoyette config_errmsg(VALUE_ERR,
517 1.9 pgoyette "high-capacity",
518 1.9 pgoyette sensor);
519 1.9 pgoyette /*
520 1.9 pgoyette * Convert the value to a valid percentage.
521 1.9 pgoyette */
522 1.9 pgoyette obj = prop_dictionary_get(ksdict, "max-value");
523 1.9 pgoyette val = (val / 100) * prop_number_integer_value(obj);
524 1.9 pgoyette
525 1.9 pgoyette if (!prop_dictionary_set_uint32(csdict,
526 1.9 pgoyette "high-capacity",
527 1.9 pgoyette val))
528 1.9 pgoyette err(EXIT_FAILURE, "dict_set highcap");
529 1.9 pgoyette } else
530 1.9 pgoyette config_errmsg(PROP_ERR, "high-capacity", sensor);
531 1.9 pgoyette }
532 1.9 pgoyette
533 1.9 pgoyette /*
534 1.9 pgoyette * maximum-capacity property set?
535 1.9 pgoyette */
536 1.9 pgoyette obj = prop_dictionary_get(csdict, "maximum-capacity");
537 1.9 pgoyette if (obj) {
538 1.9 pgoyette obj2 = prop_dictionary_get(ksdict, "want-percentage");
539 1.9 pgoyette obj3 = prop_dictionary_get(ksdict, "monitoring-supported");
540 1.9 pgoyette if (prop_bool_true(obj2) && prop_bool_true(obj3)) {
541 1.9 pgoyette strval = prop_string_cstring(obj);
542 1.9 pgoyette val = strtod(strval, &endptr);
543 1.9 pgoyette if ((*endptr != '\0') || (val < 0 || val > 100))
544 1.9 pgoyette config_errmsg(VALUE_ERR,
545 1.9 pgoyette "maximum-capacity",
546 1.9 pgoyette sensor);
547 1.9 pgoyette /*
548 1.9 pgoyette * Convert the value to a valid percentage.
549 1.9 pgoyette */
550 1.9 pgoyette obj = prop_dictionary_get(ksdict, "max-value");
551 1.9 pgoyette val = (val / 100) * prop_number_integer_value(obj);
552 1.9 pgoyette
553 1.9 pgoyette if (!prop_dictionary_set_uint32(csdict,
554 1.9 pgoyette "maximum-capacity",
555 1.9 pgoyette val))
556 1.9 pgoyette err(EXIT_FAILURE, "dict_set maxcap");
557 1.9 pgoyette } else
558 1.9 pgoyette config_errmsg(PROP_ERR, "maximum-capacity", sensor);
559 1.9 pgoyette }
560 1.9 pgoyette
561 1.9 pgoyette /*
562 1.1 xtraeme * critical-max property set?
563 1.1 xtraeme */
564 1.1 xtraeme obj = prop_dictionary_get(csdict, "critical-max");
565 1.1 xtraeme if (obj) {
566 1.1 xtraeme obj2 = prop_dictionary_get(ksdict, "monitoring-supported");
567 1.1 xtraeme if (!prop_bool_true(obj2))
568 1.1 xtraeme config_errmsg(PROP_ERR, "critical-max", sensor);
569 1.1 xtraeme
570 1.1 xtraeme strval = prop_string_cstring(obj);
571 1.1 xtraeme obj = convert_val_to_pnumber(ksdict, "critical-max",
572 1.1 xtraeme sensor, strval);
573 1.1 xtraeme if (!prop_dictionary_set(csdict, "critical-max", obj))
574 1.1 xtraeme err(EXIT_FAILURE, "prop_dict_set cmax");
575 1.1 xtraeme }
576 1.1 xtraeme
577 1.1 xtraeme /*
578 1.1 xtraeme * critical-min property set?
579 1.1 xtraeme */
580 1.1 xtraeme obj = prop_dictionary_get(csdict, "critical-min");
581 1.1 xtraeme if (obj) {
582 1.1 xtraeme obj2 = prop_dictionary_get(ksdict, "monitoring-supported");
583 1.1 xtraeme if (!prop_bool_true(obj2))
584 1.1 xtraeme config_errmsg(PROP_ERR, "critical-min", sensor);
585 1.1 xtraeme
586 1.1 xtraeme strval = prop_string_cstring(obj);
587 1.1 xtraeme obj = convert_val_to_pnumber(ksdict, "critical-min",
588 1.1 xtraeme sensor, strval);
589 1.1 xtraeme if (!prop_dictionary_set(csdict, "critical-min", obj))
590 1.1 xtraeme err(EXIT_FAILURE, "prop_dict_set cmin");
591 1.1 xtraeme }
592 1.8 pgoyette
593 1.8 pgoyette /*
594 1.8 pgoyette * warning-max property set?
595 1.8 pgoyette */
596 1.8 pgoyette obj = prop_dictionary_get(csdict, "warning-max");
597 1.8 pgoyette if (obj) {
598 1.8 pgoyette obj2 = prop_dictionary_get(ksdict, "monitoring-supported");
599 1.8 pgoyette if (!prop_bool_true(obj2))
600 1.8 pgoyette config_errmsg(PROP_ERR, "warning-max", sensor);
601 1.8 pgoyette
602 1.8 pgoyette strval = prop_string_cstring(obj);
603 1.8 pgoyette obj = convert_val_to_pnumber(ksdict, "warning-max",
604 1.8 pgoyette sensor, strval);
605 1.8 pgoyette if (!prop_dictionary_set(csdict, "warning-max", obj))
606 1.8 pgoyette err(EXIT_FAILURE, "prop_dict_set wmax");
607 1.8 pgoyette }
608 1.8 pgoyette /*
609 1.8 pgoyette * warning-min property set?
610 1.8 pgoyette */
611 1.8 pgoyette obj = prop_dictionary_get(csdict, "warning-min");
612 1.8 pgoyette if (obj) {
613 1.8 pgoyette obj2 = prop_dictionary_get(ksdict, "monitoring-supported");
614 1.8 pgoyette if (!prop_bool_true(obj2))
615 1.8 pgoyette config_errmsg(PROP_ERR, "warning-min", sensor);
616 1.8 pgoyette
617 1.8 pgoyette strval = prop_string_cstring(obj);
618 1.8 pgoyette obj = convert_val_to_pnumber(ksdict, "warning-min",
619 1.8 pgoyette sensor, strval);
620 1.8 pgoyette if (!prop_dictionary_set(csdict, "warning-min", obj))
621 1.8 pgoyette err(EXIT_FAILURE, "prop_dict_set wmin");
622 1.8 pgoyette }
623 1.1 xtraeme }
624 1.1 xtraeme
625 1.1 xtraeme /*
626 1.11 pgoyette * Conversions for {critical,warning}-{max,min} properties.
627 1.1 xtraeme */
628 1.1 xtraeme prop_number_t
629 1.1 xtraeme convert_val_to_pnumber(prop_dictionary_t kdict, const char *prop,
630 1.1 xtraeme const char *sensor, char *value)
631 1.1 xtraeme {
632 1.1 xtraeme prop_object_t obj;
633 1.1 xtraeme prop_number_t num;
634 1.5 xtraeme double val, max, min;
635 1.1 xtraeme char *strval, *tmp, *endptr;
636 1.1 xtraeme bool celsius;
637 1.1 xtraeme size_t len;
638 1.1 xtraeme
639 1.5 xtraeme val = max = min = 0;
640 1.5 xtraeme
641 1.5 xtraeme /*
642 1.11 pgoyette * Not allowed in battery sensors.
643 1.5 xtraeme */
644 1.11 pgoyette obj = prop_dictionary_get(kdict, "type");
645 1.11 pgoyette if (prop_string_equals_cstring(obj, "Battery capacity"))
646 1.5 xtraeme config_errmsg(PROP_ERR, prop, sensor);
647 1.5 xtraeme
648 1.1 xtraeme /*
649 1.1 xtraeme * Make the conversion for sensor's type.
650 1.1 xtraeme */
651 1.1 xtraeme if (prop_string_equals_cstring(obj, "Temperature")) {
652 1.1 xtraeme tmp = strchr(value, 'C');
653 1.1 xtraeme if (tmp)
654 1.1 xtraeme celsius = true;
655 1.1 xtraeme else {
656 1.1 xtraeme tmp = strchr(value, 'F');
657 1.1 xtraeme if (!tmp)
658 1.1 xtraeme config_errmsg(VALUE_ERR, prop, sensor);
659 1.1 xtraeme
660 1.1 xtraeme celsius = false;
661 1.1 xtraeme }
662 1.1 xtraeme
663 1.1 xtraeme len = strlen(value);
664 1.1 xtraeme strval = calloc(len, sizeof(*value));
665 1.1 xtraeme if (!strval)
666 1.1 xtraeme err(EXIT_FAILURE, "calloc");
667 1.1 xtraeme
668 1.1 xtraeme (void)strlcpy(strval, value, len);
669 1.1 xtraeme val = strtod(strval, &endptr);
670 1.6 xtraeme if (*endptr != '\0') {
671 1.6 xtraeme free(strval);
672 1.1 xtraeme config_errmsg(VALUE_ERR, prop, sensor);
673 1.6 xtraeme }
674 1.1 xtraeme
675 1.1 xtraeme /* convert to fahrenheit */
676 1.1 xtraeme if (!celsius)
677 1.1 xtraeme val = (val - 32.0) * (5.0 / 9.0);
678 1.1 xtraeme
679 1.1 xtraeme /* convert to microKelvin */
680 1.1 xtraeme val = val * 1000000 + 273150000;
681 1.1 xtraeme num = prop_number_create_unsigned_integer(val);
682 1.6 xtraeme free(strval);
683 1.1 xtraeme
684 1.10 pgoyette } else if (prop_string_equals_cstring(obj, "Fan") ||
685 1.10 pgoyette prop_string_equals_cstring(obj, "Integer")) {
686 1.1 xtraeme /* no conversion */
687 1.1 xtraeme val = strtod(value, &endptr);
688 1.1 xtraeme if (*endptr != '\0')
689 1.1 xtraeme config_errmsg(VALUE_ERR, prop, sensor);
690 1.1 xtraeme
691 1.1 xtraeme num = prop_number_create_unsigned_integer(val);
692 1.1 xtraeme
693 1.1 xtraeme } else {
694 1.5 xtraeme obj = prop_dictionary_get(kdict, "max-value");
695 1.5 xtraeme if (obj)
696 1.5 xtraeme max = prop_number_integer_value(obj);
697 1.5 xtraeme
698 1.5 xtraeme obj = prop_dictionary_get(kdict, "min-value");
699 1.5 xtraeme if (obj)
700 1.5 xtraeme min = prop_number_integer_value(obj);
701 1.5 xtraeme
702 1.1 xtraeme val = strtod(value, &endptr);
703 1.1 xtraeme if (*endptr != '\0')
704 1.1 xtraeme config_errmsg(VALUE_ERR, prop, sensor);
705 1.1 xtraeme
706 1.1 xtraeme /* convert to m[V,W,Ohms] again */
707 1.1 xtraeme val *= 1000000.0;
708 1.5 xtraeme
709 1.5 xtraeme /*
710 1.5 xtraeme * trying to set a value higher than the max
711 1.5 xtraeme * assigned?
712 1.5 xtraeme */
713 1.5 xtraeme if (max && val > max)
714 1.5 xtraeme config_errmsg(VALUE_ERR, prop, sensor);
715 1.5 xtraeme
716 1.5 xtraeme /*
717 1.5 xtraeme * trying to set a value lower than the min
718 1.5 xtraeme * assigned?
719 1.5 xtraeme */
720 1.5 xtraeme if (min && val < min)
721 1.5 xtraeme config_errmsg(VALUE_ERR, prop, sensor);
722 1.5 xtraeme
723 1.1 xtraeme num = prop_number_create_integer(val);
724 1.1 xtraeme }
725 1.1 xtraeme
726 1.1 xtraeme return num;
727 1.1 xtraeme }
728