sysvars.c 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. #include "sysvars.h"
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <stdio.h>
  5. #include <time.h>
  6. static const char *sysvar_bool_tostr(sysvar_t from, char *buf)
  7. {
  8. buf[0] = '0' + from.v_bool;
  9. buf[1] = '\0';
  10. return buf;
  11. }
  12. static bool sysvar_bool_fromstr(sysvar_t *to, const char *from)
  13. {
  14. char c = from[0] | 0x20;
  15. switch (c) {
  16. case '1':
  17. case 't':
  18. case 'y':
  19. case 'j':
  20. case 's':
  21. to->v_bool = true;
  22. return true;
  23. case '0':
  24. case 'f':
  25. case 'n':
  26. to->v_bool = false;
  27. return true;
  28. default:
  29. return false;
  30. }
  31. }
  32. const struct sysvar_ops sysvar_bool_ops = {
  33. .tostr = sysvar_bool_tostr,
  34. .fromstr = sysvar_bool_fromstr,
  35. .buflen = BOOL_BUFLEN
  36. };
  37. static const char *sysvar_int_tostr(sysvar_t from, char *buf)
  38. {
  39. snprintf(buf, INT_BUFLEN, "%ld", from.v_int);
  40. return buf;
  41. }
  42. static bool sysvar_int_fromstr(sysvar_t *to, const char *from)
  43. {
  44. char *ep;
  45. long v = strtol(from, &ep, 0);
  46. if (ep == from || *ep)
  47. return false;
  48. to->v_int = v;
  49. return true;
  50. }
  51. const struct sysvar_ops sysvar_int_ops = {
  52. .tostr = sysvar_int_tostr,
  53. .fromstr = sysvar_int_fromstr,
  54. .buflen = INT_BUFLEN
  55. };
  56. static const char *sysvar_uint_tostr(sysvar_t from, char *buf)
  57. {
  58. snprintf(buf, UINT_BUFLEN, "%lu", from.v_uint);
  59. return buf;
  60. }
  61. static bool sysvar_uint_fromstr(sysvar_t *to, const char *from)
  62. {
  63. char *ep;
  64. unsigned long v = strtoul(from, &ep, 0);
  65. if (ep == from || *ep)
  66. return false;
  67. to->v_uint = v;
  68. return true;
  69. }
  70. const struct sysvar_ops sysvar_uint_ops = {
  71. .tostr = sysvar_uint_tostr,
  72. .fromstr = sysvar_uint_fromstr,
  73. .buflen = UINT_BUFLEN
  74. };
  75. static const char *sysvar_str_tostr(sysvar_t from, char *buf)
  76. {
  77. (void)buf;
  78. return from.v_str;
  79. }
  80. static bool sysvar_str_fromstr(sysvar_t *to, const char *from)
  81. {
  82. char *ns;
  83. if (!from) {
  84. ns = NULL;
  85. } else {
  86. ns = strdup(from);
  87. if (!ns)
  88. return false;
  89. }
  90. to->v_str = ns;
  91. return true;
  92. }
  93. static bool sysvar_str_set(sysvar_t *to, sysvar_t from)
  94. {
  95. return sysvar_str_fromstr(to, from.v_str);
  96. }
  97. const struct sysvar_ops sysvar_str_ops = {
  98. .set = sysvar_str_set,
  99. .tostr = sysvar_str_tostr,
  100. .fromstr = sysvar_str_fromstr,
  101. .is_ptr = true
  102. };
  103. static void sysvar_tz_update(sysvar_t val, bool isset)
  104. {
  105. if (isset)
  106. setenv("TZ", val.v_tz, 1);
  107. else
  108. unsetenv("TZ");
  109. tzset();
  110. }
  111. const struct sysvar_ops sysvar_tz_ops = {
  112. .set = sysvar_str_set,
  113. .tostr = sysvar_str_tostr,
  114. .fromstr = sysvar_str_fromstr,
  115. .update = sysvar_tz_update,
  116. .is_ptr = true
  117. };
  118. static const char *sysvar_ip_tostr(sysvar_t from, char *buf)
  119. {
  120. union ip_bytes {
  121. uint8_t b[4];
  122. uint32_t l;
  123. } ip;
  124. ip.l = from.v_ip;
  125. snprintf(buf, IP_BUFLEN,
  126. "%u.%u.%u.%u", ip.b[0], ip.b[1], ip.b[2], ip.b[3]);
  127. return buf;
  128. }
  129. static bool sysvar_ip_fromstr(sysvar_t *to, const char *str)
  130. {
  131. union ip_bytes {
  132. uint8_t b[4];
  133. uint32_t l;
  134. } ip;
  135. ip.l = 0;
  136. for (int i = 0; i < 4; i++) {
  137. char *ep;
  138. unsigned long v = strtoul(str, &ep, 10);
  139. if (ep == str || *ep != (i == 3) ? '\0' : '.' || v > 255)
  140. return false;
  141. str = ep + 1;
  142. ip.b[i] = v;
  143. }
  144. to->v_ip = ip.l;
  145. return true;
  146. }
  147. const struct sysvar_ops sysvar_ip_ops = {
  148. .tostr = sysvar_ip_tostr,
  149. .fromstr = sysvar_ip_fromstr,
  150. .buflen = IP_BUFLEN
  151. };
  152. static bool sysvar_mac_set(sysvar_t *to, sysvar_t from)
  153. {
  154. uint8_t *buf;
  155. if (!from.v_mac) {
  156. buf = NULL;
  157. } else {
  158. buf = malloc(6);
  159. if (!buf)
  160. return false;
  161. memcpy(buf, from.v_mac, 6);
  162. }
  163. to->v_mac = buf;
  164. return true;
  165. }
  166. static const char *sysvar_mac_tostr(sysvar_t from, char *buf)
  167. {
  168. const uint8_t *m = from.v_mac;
  169. if (!m)
  170. return NULL;
  171. snprintf(buf, MAC_BUFLEN, "%x:%x:%x:%x:%x:%x",
  172. m[0], m[1], m[2], m[3], m[4], m[5]);
  173. return buf;
  174. }
  175. static bool sysvar_mac_fromstr(sysvar_t *to, const char *str)
  176. {
  177. sysvar_t from;
  178. uint8_t m[6];
  179. if (!str) {
  180. from.v_mac = NULL;
  181. } else {
  182. for (int i = 0; i < 6; i++) {
  183. char *ep;
  184. unsigned long v = strtoul(str, &ep, 16);
  185. if (ep == str || *ep != (i == 5) ? '\0' : ':' || v > 255)
  186. return false;
  187. str = ep + 1;
  188. m[i] = v;
  189. }
  190. from.v_mac = m;
  191. }
  192. return sysvar_mac_set(to, from);
  193. }
  194. const struct sysvar_ops sysvar_mac_ops = {
  195. .set = sysvar_mac_set,
  196. .tostr = sysvar_mac_tostr,
  197. .fromstr = sysvar_mac_fromstr,
  198. .buflen = MAC_BUFLEN,
  199. .is_ptr = true
  200. };
  201. /* --- Generic getters/setters --- */
  202. /* Contains the lowest numbered sysvar changed; or sysvar_count if nothing */
  203. enum sysvars_enum sysvar_changed;
  204. sysvar_t getvar(size_t var)
  205. {
  206. if (var >= (size_t)sysvar_count)
  207. return sysvar_defval[sysvar_null];
  208. return sysvar_val[var];
  209. }
  210. static bool do_setvar(size_t var, sysvar_t val, bool is_set)
  211. {
  212. const struct sysvar_ops *type = sysvar_types[var];
  213. sysvar_t *to = &sysvar_val[var];
  214. void *free_ptr = NULL;
  215. if (type->is_ptr)
  216. free_ptr = to->v_ptr;
  217. if (!type->set || (type->is_ptr && !val.v_ptr)) {
  218. sysvar_val[var] = val;
  219. } else {
  220. if (!type->set(to, val))
  221. return false;
  222. }
  223. if (free_ptr)
  224. free(free_ptr);
  225. if (var < (size_t)sysvar_changed)
  226. sysvar_changed = (enum sysvars_enum)var;
  227. sysvar_isset[var] = is_set;
  228. if (type->update)
  229. type->update(*to, is_set);
  230. return true;
  231. }
  232. bool setvar(size_t var, sysvar_t val)
  233. {
  234. if (var >= (size_t)sysvar_count)
  235. return false;
  236. return do_setvar(var, val, true);
  237. }
  238. bool unsetvar(size_t var)
  239. {
  240. if (var >= (size_t)sysvar_count)
  241. return false;
  242. return do_setvar(var, sysvar_defval[var], false);
  243. }
  244. /* --- Getters/setters converting to/from strings --- */
  245. const char *getvar_tostr(size_t var)
  246. {
  247. static char buf[SYSVAR_BUFLEN];
  248. return getvar_tostr_r(var, buf);
  249. }
  250. const char *getvar_tostr_r(size_t var, char *buf)
  251. {
  252. if (var >= (size_t)sysvar_count)
  253. return NULL;
  254. const struct sysvar_ops *type = sysvar_types[var];
  255. /* A tostr method is required */
  256. return type->tostr(sysvar_val[var], buf);
  257. }
  258. bool setvar_fromstr(size_t var, const char *str)
  259. {
  260. if (var >= (size_t)sysvar_count)
  261. return NULL;
  262. if (!str)
  263. return unsetvar(var);
  264. const struct sysvar_ops *type = sysvar_types[var];
  265. sysvar_t *to = &sysvar_val[var];
  266. void *free_ptr = NULL;
  267. if (type->is_ptr)
  268. free_ptr = to->v_ptr;
  269. /* A fromstr method is required */
  270. if (!type->fromstr(to, str))
  271. return false;
  272. if (var < (size_t)sysvar_changed)
  273. sysvar_changed = (enum sysvars_enum)var;
  274. if (free_ptr)
  275. free(free_ptr);
  276. sysvar_isset[var] = true;
  277. return true;
  278. }
  279. /* --- Find the index of a specific variable --- */
  280. static int string_ptr_compare(const void *a, const void *b)
  281. {
  282. const char * const *aa = a;
  283. const char * const *bb = b;
  284. return strcmp(*aa, *bb);
  285. }
  286. size_t sysvar_find(size_t ns, const char *name)
  287. {
  288. if (ns >= (size_t)sysvar_nscount)
  289. return 0;
  290. const sysvar_ns_t *nsi = &sysvar_ns[ns];
  291. const char * const *varname;
  292. varname = bsearch(name, sysvar_name + nsi->first,
  293. nsi->count, sizeof(const char *),
  294. string_ptr_compare);
  295. if (!varname)
  296. return 0;
  297. return varname - sysvar_name;
  298. }
  299. /* --- Initialization/reset to defaults --- */
  300. void sysvar_reset(void)
  301. {
  302. for (size_t i = sysvar_null; i < (size_t)sysvar_count; i++)
  303. unsetvar(i);
  304. sysvar_changed = sysvar_null;
  305. }