stats.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589
  1. /*
  2. * Copyright (c) 2017 Red Hat, Inc.
  3. *
  4. * All rights reserved.
  5. *
  6. * Authors: Christine Caulfield (ccaulfie@redhat.com)
  7. *
  8. * This software licensed under BSD license, the text of which follows:
  9. *
  10. * Redistribution and use in source and binary forms, with or without
  11. * modification, are permitted provided that the following conditions are met:
  12. *
  13. * - Redistributions of source code must retain the above copyright notice,
  14. * this list of conditions and the following disclaimer.
  15. * - Redistributions in binary form must reproduce the above copyright notice,
  16. * this list of conditions and the following disclaimer in the documentation
  17. * and/or other materials provided with the distribution.
  18. * - Neither the name of the MontaVista Software, Inc. nor the names of its
  19. * contributors may be used to endorse or promote products derived from this
  20. * software without specific prior written permission.
  21. *
  22. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTIBUTORS "AS IS"
  23. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  24. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  25. * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
  26. * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
  27. * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
  28. * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
  29. * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
  30. * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  31. * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
  32. * THE POSSIBILITY OF SUCH DAMAGE.
  33. */
  34. #include <config.h>
  35. #include <sys/types.h>
  36. #include <sys/stat.h>
  37. #include <fcntl.h>
  38. #include <stdint.h>
  39. #include <stddef.h>
  40. #include <unistd.h>
  41. #include <libknet.h>
  42. #include <qb/qblist.h>
  43. #include <qb/qbipcs.h>
  44. #include <qb/qbipc_common.h>
  45. #include <corosync/corodefs.h>
  46. #include <corosync/coroapi.h>
  47. #include <corosync/logsys.h>
  48. #include <corosync/icmap.h>
  49. #include <corosync/totem/totemstats.h>
  50. #include "util.h"
  51. #include "ipcs_stats.h"
  52. #include "stats.h"
  53. LOGSYS_DECLARE_SUBSYS ("STATS");
  54. static qb_map_t *stats_map;
  55. /* Convert iterator number to text and a stats pointer */
  56. struct cs_stats_conv {
  57. enum {STAT_PG, STAT_SRP, STAT_KNET, STAT_IPCSC, STAT_IPCSG} type;
  58. const char *name;
  59. const size_t offset;
  60. const icmap_value_types_t value_type;
  61. };
  62. struct cs_stats_conv cs_pg_stats[] = {
  63. { STAT_PG, "msg_queue_avail", offsetof(totempg_stats_t, msg_queue_avail), ICMAP_VALUETYPE_UINT32},
  64. { STAT_PG, "msg_reserved", offsetof(totempg_stats_t, msg_reserved), ICMAP_VALUETYPE_UINT32},
  65. };
  66. struct cs_stats_conv cs_srp_stats[] = {
  67. { STAT_SRP, "orf_token_tx", offsetof(totemsrp_stats_t, orf_token_tx), ICMAP_VALUETYPE_UINT64},
  68. { STAT_SRP, "orf_token_rx", offsetof(totemsrp_stats_t, orf_token_rx), ICMAP_VALUETYPE_UINT64},
  69. { STAT_SRP, "memb_merge_detect_tx", offsetof(totemsrp_stats_t, memb_merge_detect_tx), ICMAP_VALUETYPE_UINT64},
  70. { STAT_SRP, "memb_merge_detect_rx", offsetof(totemsrp_stats_t, memb_merge_detect_rx), ICMAP_VALUETYPE_UINT64},
  71. { STAT_SRP, "memb_join_tx", offsetof(totemsrp_stats_t, memb_join_tx), ICMAP_VALUETYPE_UINT64},
  72. { STAT_SRP, "memb_join_rx", offsetof(totemsrp_stats_t, memb_join_rx), ICMAP_VALUETYPE_UINT64},
  73. { STAT_SRP, "mcast_tx", offsetof(totemsrp_stats_t, mcast_tx), ICMAP_VALUETYPE_UINT64},
  74. { STAT_SRP, "mcast_retx", offsetof(totemsrp_stats_t, mcast_retx), ICMAP_VALUETYPE_UINT64},
  75. { STAT_SRP, "mcast_rx", offsetof(totemsrp_stats_t, mcast_rx), ICMAP_VALUETYPE_UINT64},
  76. { STAT_SRP, "memb_commit_token_tx", offsetof(totemsrp_stats_t, memb_commit_token_tx), ICMAP_VALUETYPE_UINT64},
  77. { STAT_SRP, "memb_commit_token_rx", offsetof(totemsrp_stats_t, memb_commit_token_rx), ICMAP_VALUETYPE_UINT64},
  78. { STAT_SRP, "token_hold_cancel_tx", offsetof(totemsrp_stats_t, token_hold_cancel_tx), ICMAP_VALUETYPE_UINT64},
  79. { STAT_SRP, "token_hold_cancel_rx", offsetof(totemsrp_stats_t, token_hold_cancel_rx), ICMAP_VALUETYPE_UINT64},
  80. { STAT_SRP, "operational_entered", offsetof(totemsrp_stats_t, operational_entered), ICMAP_VALUETYPE_UINT64},
  81. { STAT_SRP, "operational_token_lost", offsetof(totemsrp_stats_t, operational_token_lost), ICMAP_VALUETYPE_UINT64},
  82. { STAT_SRP, "gather_entered", offsetof(totemsrp_stats_t, gather_entered), ICMAP_VALUETYPE_UINT64},
  83. { STAT_SRP, "gather_token_lost", offsetof(totemsrp_stats_t, gather_token_lost), ICMAP_VALUETYPE_UINT64},
  84. { STAT_SRP, "commit_entered", offsetof(totemsrp_stats_t, commit_entered), ICMAP_VALUETYPE_UINT64},
  85. { STAT_SRP, "commit_token_lost", offsetof(totemsrp_stats_t, commit_token_lost), ICMAP_VALUETYPE_UINT64},
  86. { STAT_SRP, "recovery_entered", offsetof(totemsrp_stats_t, recovery_entered), ICMAP_VALUETYPE_UINT64},
  87. { STAT_SRP, "recovery_token_lost", offsetof(totemsrp_stats_t, recovery_token_lost), ICMAP_VALUETYPE_UINT64},
  88. { STAT_SRP, "consensus_timeouts", offsetof(totemsrp_stats_t, consensus_timeouts), ICMAP_VALUETYPE_UINT64},
  89. { STAT_SRP, "rx_msg_dropped", offsetof(totemsrp_stats_t, rx_msg_dropped), ICMAP_VALUETYPE_UINT64},
  90. { STAT_SRP, "continuous_gather", offsetof(totemsrp_stats_t, continuous_gather), ICMAP_VALUETYPE_UINT32},
  91. { STAT_SRP, "continuous_sendmsg_failures", offsetof(totemsrp_stats_t, continuous_sendmsg_failures), ICMAP_VALUETYPE_UINT32},
  92. { STAT_SRP, "firewall_enabled_or_nic_failure", offsetof(totemsrp_stats_t, firewall_enabled_or_nic_failure), ICMAP_VALUETYPE_UINT8},
  93. { STAT_SRP, "mtt_rx_token", offsetof(totemsrp_stats_t, mtt_rx_token), ICMAP_VALUETYPE_UINT32},
  94. { STAT_SRP, "avg_token_workload", offsetof(totemsrp_stats_t, avg_token_workload), ICMAP_VALUETYPE_UINT32},
  95. { STAT_SRP, "avg_backlog_calc", offsetof(totemsrp_stats_t, avg_backlog_calc), ICMAP_VALUETYPE_UINT32},
  96. };
  97. struct cs_stats_conv cs_knet_stats[] = {
  98. { STAT_KNET, "enabled", offsetof(struct knet_link_status, enabled), ICMAP_VALUETYPE_UINT8},
  99. { STAT_KNET, "connected", offsetof(struct knet_link_status, connected), ICMAP_VALUETYPE_UINT8},
  100. { STAT_KNET, "mtu", offsetof(struct knet_link_status, mtu), ICMAP_VALUETYPE_UINT32},
  101. { STAT_KNET, "tx_data_packets", offsetof(struct knet_link_status, stats.tx_data_packets), ICMAP_VALUETYPE_UINT64},
  102. { STAT_KNET, "rx_data_packets", offsetof(struct knet_link_status, stats.rx_data_packets), ICMAP_VALUETYPE_UINT64},
  103. { STAT_KNET, "tx_data_bytes", offsetof(struct knet_link_status, stats.tx_data_bytes), ICMAP_VALUETYPE_UINT64},
  104. { STAT_KNET, "rx_data_bytes", offsetof(struct knet_link_status, stats.rx_data_bytes), ICMAP_VALUETYPE_UINT64},
  105. { STAT_KNET, "tx_ping_packets", offsetof(struct knet_link_status, stats.tx_ping_packets), ICMAP_VALUETYPE_UINT64},
  106. { STAT_KNET, "rx_ping_packets", offsetof(struct knet_link_status, stats.rx_ping_packets), ICMAP_VALUETYPE_UINT64},
  107. { STAT_KNET, "tx_ping_bytes", offsetof(struct knet_link_status, stats.tx_ping_bytes), ICMAP_VALUETYPE_UINT64},
  108. { STAT_KNET, "rx_ping_bytes", offsetof(struct knet_link_status, stats.rx_ping_bytes), ICMAP_VALUETYPE_UINT64},
  109. { STAT_KNET, "tx_pong_packets", offsetof(struct knet_link_status, stats.tx_pong_packets), ICMAP_VALUETYPE_UINT64},
  110. { STAT_KNET, "rx_pong_packets", offsetof(struct knet_link_status, stats.rx_pong_packets), ICMAP_VALUETYPE_UINT64},
  111. { STAT_KNET, "tx_pong_bytes", offsetof(struct knet_link_status, stats.tx_pong_bytes), ICMAP_VALUETYPE_UINT64},
  112. { STAT_KNET, "rx_pong_bytes", offsetof(struct knet_link_status, stats.rx_pong_bytes), ICMAP_VALUETYPE_UINT64},
  113. { STAT_KNET, "tx_pmtu_packets", offsetof(struct knet_link_status, stats.tx_pmtu_packets), ICMAP_VALUETYPE_UINT64},
  114. { STAT_KNET, "rx_pmtu_packets", offsetof(struct knet_link_status, stats.rx_pmtu_packets), ICMAP_VALUETYPE_UINT64},
  115. { STAT_KNET, "tx_pmtu_bytes", offsetof(struct knet_link_status, stats.tx_pmtu_bytes), ICMAP_VALUETYPE_UINT64},
  116. { STAT_KNET, "rx_pmtu_bytes", offsetof(struct knet_link_status, stats.rx_pmtu_bytes), ICMAP_VALUETYPE_UINT64},
  117. { STAT_KNET, "tx_total_packets", offsetof(struct knet_link_status, stats.tx_total_packets), ICMAP_VALUETYPE_UINT64},
  118. { STAT_KNET, "rx_total_packets", offsetof(struct knet_link_status, stats.rx_total_packets), ICMAP_VALUETYPE_UINT64},
  119. { STAT_KNET, "tx_total_bytes", offsetof(struct knet_link_status, stats.tx_total_bytes), ICMAP_VALUETYPE_UINT64},
  120. { STAT_KNET, "rx_total_bytes", offsetof(struct knet_link_status, stats.rx_total_bytes), ICMAP_VALUETYPE_UINT64},
  121. { STAT_KNET, "tx_total_errors", offsetof(struct knet_link_status, stats.tx_total_errors), ICMAP_VALUETYPE_UINT64},
  122. { STAT_KNET, "rx_total_retries", offsetof(struct knet_link_status, stats.tx_total_retries), ICMAP_VALUETYPE_UINT64},
  123. { STAT_KNET, "tx_pmtu_errors", offsetof(struct knet_link_status, stats.tx_pmtu_errors), ICMAP_VALUETYPE_UINT32},
  124. { STAT_KNET, "tx_pmtu_retries", offsetof(struct knet_link_status, stats.tx_pmtu_retries), ICMAP_VALUETYPE_UINT32},
  125. { STAT_KNET, "tx_ping_errors", offsetof(struct knet_link_status, stats.tx_ping_errors), ICMAP_VALUETYPE_UINT32},
  126. { STAT_KNET, "tx_ping_retries", offsetof(struct knet_link_status, stats.tx_ping_retries), ICMAP_VALUETYPE_UINT32},
  127. { STAT_KNET, "tx_pong_errors", offsetof(struct knet_link_status, stats.tx_pong_errors), ICMAP_VALUETYPE_UINT32},
  128. { STAT_KNET, "tx_pong_retries", offsetof(struct knet_link_status, stats.tx_pong_retries), ICMAP_VALUETYPE_UINT32},
  129. { STAT_KNET, "tx_data_errors", offsetof(struct knet_link_status, stats.tx_data_errors), ICMAP_VALUETYPE_UINT32},
  130. { STAT_KNET, "tx_data_retries", offsetof(struct knet_link_status, stats.tx_data_retries), ICMAP_VALUETYPE_UINT32},
  131. { STAT_KNET, "latency_min", offsetof(struct knet_link_status, stats.latency_min), ICMAP_VALUETYPE_UINT32},
  132. { STAT_KNET, "latency_max", offsetof(struct knet_link_status, stats.latency_max), ICMAP_VALUETYPE_UINT32},
  133. { STAT_KNET, "latency_ave", offsetof(struct knet_link_status, stats.latency_ave), ICMAP_VALUETYPE_UINT32},
  134. { STAT_KNET, "latency_samples", offsetof(struct knet_link_status, stats.latency_samples), ICMAP_VALUETYPE_UINT32},
  135. { STAT_KNET, "down_count", offsetof(struct knet_link_status, stats.down_count), ICMAP_VALUETYPE_UINT32},
  136. { STAT_KNET, "up_count", offsetof(struct knet_link_status, stats.up_count), ICMAP_VALUETYPE_UINT32},
  137. };
  138. struct cs_stats_conv cs_ipcs_conn_stats[] = {
  139. { STAT_IPCSC, "queueing", offsetof(struct ipcs_conn_stats, cnx.queuing), ICMAP_VALUETYPE_INT32},
  140. { STAT_IPCSC, "queued", offsetof(struct ipcs_conn_stats, cnx.queued), ICMAP_VALUETYPE_UINT32},
  141. { STAT_IPCSC, "invalid_request", offsetof(struct ipcs_conn_stats, cnx.invalid_request), ICMAP_VALUETYPE_UINT64},
  142. { STAT_IPCSC, "overload", offsetof(struct ipcs_conn_stats, cnx.overload), ICMAP_VALUETYPE_UINT64},
  143. { STAT_IPCSC, "sent", offsetof(struct ipcs_conn_stats, cnx.sent), ICMAP_VALUETYPE_UINT32},
  144. { STAT_IPCSC, "procname", offsetof(struct ipcs_conn_stats, cnx.proc_name), ICMAP_VALUETYPE_STRING},
  145. { STAT_IPCSC, "requests", offsetof(struct ipcs_conn_stats, conn.requests), ICMAP_VALUETYPE_UINT64},
  146. { STAT_IPCSC, "responses", offsetof(struct ipcs_conn_stats, conn.responses), ICMAP_VALUETYPE_UINT64},
  147. { STAT_IPCSC, "dispatched", offsetof(struct ipcs_conn_stats, conn.events), ICMAP_VALUETYPE_UINT64},
  148. { STAT_IPCSC, "send_retries", offsetof(struct ipcs_conn_stats, conn.send_retries), ICMAP_VALUETYPE_UINT64},
  149. { STAT_IPCSC, "recv_retries", offsetof(struct ipcs_conn_stats, conn.recv_retries), ICMAP_VALUETYPE_UINT64},
  150. { STAT_IPCSC, "flow_control", offsetof(struct ipcs_conn_stats, conn.flow_control_state), ICMAP_VALUETYPE_UINT32},
  151. { STAT_IPCSC, "flow_control_count", offsetof(struct ipcs_conn_stats, conn.flow_control_count), ICMAP_VALUETYPE_UINT64},
  152. };
  153. struct cs_stats_conv cs_ipcs_global_stats[] = {
  154. { STAT_IPCSG, "global.active", offsetof(struct ipcs_global_stats, active), ICMAP_VALUETYPE_UINT64},
  155. { STAT_IPCSG, "global.closed", offsetof(struct ipcs_global_stats, closed), ICMAP_VALUETYPE_UINT64},
  156. };
  157. #define NUM_PG_STATS (sizeof(cs_pg_stats) / sizeof(struct cs_stats_conv))
  158. #define NUM_SRP_STATS (sizeof(cs_srp_stats) / sizeof(struct cs_stats_conv))
  159. #define NUM_KNET_STATS (sizeof(cs_knet_stats) / sizeof(struct cs_stats_conv))
  160. #define NUM_IPCSC_STATS (sizeof(cs_ipcs_conn_stats) / sizeof(struct cs_stats_conv))
  161. #define NUM_IPCSG_STATS (sizeof(cs_ipcs_global_stats) / sizeof(struct cs_stats_conv))
  162. /* What goes in the trie */
  163. struct stats_item {
  164. char *key_name;
  165. struct cs_stats_conv * cs_conv;
  166. };
  167. /* One of these per tracker */
  168. struct cs_stats_tracker
  169. {
  170. char *key_name;
  171. void *user_data;
  172. int32_t events;
  173. icmap_notify_fn_t notify_fn;
  174. uint64_t old_value;
  175. struct qb_list_head list;
  176. };
  177. QB_LIST_DECLARE (stats_tracker_list_head);
  178. static const struct corosync_api_v1 *api;
  179. static void stats_map_set_value(struct cs_stats_conv *conv,
  180. void *stat_array,
  181. void *value,
  182. size_t *value_len,
  183. icmap_value_types_t *type)
  184. {
  185. if (value_len) {
  186. *value_len = icmap_get_valuetype_len(conv->value_type);
  187. }
  188. if (type) {
  189. *type = conv->value_type;
  190. if ((*type == ICMAP_VALUETYPE_STRING) && value_len && stat_array) {
  191. *value_len = strlen((char *)(stat_array) + conv->offset)+1;
  192. }
  193. }
  194. if (value) {
  195. memcpy(value, (char *)(stat_array) + conv->offset, *value_len);
  196. }
  197. }
  198. static void stats_add_entry(const char *key, struct cs_stats_conv *cs_conv)
  199. {
  200. struct stats_item *item = malloc(sizeof(struct stats_item));
  201. if (item) {
  202. item->cs_conv = cs_conv;
  203. item->key_name = strdup(key);
  204. qb_map_put(stats_map, item->key_name, item);
  205. }
  206. }
  207. static void stats_rm_entry(const char *key)
  208. {
  209. struct stats_item *item = qb_map_get(stats_map, key);
  210. if (item) {
  211. qb_map_rm(stats_map, item->key_name);
  212. free(item->key_name);
  213. free(item);
  214. }
  215. }
  216. cs_error_t stats_map_init(const struct corosync_api_v1 *corosync_api)
  217. {
  218. int i;
  219. char param[ICMAP_KEYNAME_MAXLEN];
  220. api = corosync_api;
  221. stats_map = qb_trie_create();
  222. if (!stats_map) {
  223. return CS_ERR_INIT;
  224. }
  225. /* Populate the static portions of the trie */
  226. for (i = 0; i<NUM_PG_STATS; i++) {
  227. sprintf(param, "stats.pg.%s", cs_pg_stats[i].name);
  228. stats_add_entry(param, &cs_pg_stats[i]);
  229. }
  230. for (i = 0; i<NUM_SRP_STATS; i++) {
  231. sprintf(param, "stats.srp.%s", cs_srp_stats[i].name);
  232. stats_add_entry(param, &cs_srp_stats[i]);
  233. }
  234. for (i = 0; i<NUM_IPCSG_STATS; i++) {
  235. sprintf(param, "stats.ipcs.%s", cs_ipcs_global_stats[i].name);
  236. stats_add_entry(param, &cs_ipcs_global_stats[i]);
  237. }
  238. /* KNET and IPCS stats are added when appropriate */
  239. return CS_OK;
  240. }
  241. cs_error_t stats_map_get(const char *key_name,
  242. void *value,
  243. size_t *value_len,
  244. icmap_value_types_t *type)
  245. {
  246. struct cs_stats_conv *statinfo;
  247. struct stats_item *item;
  248. totempg_stats_t *pg_stats;
  249. struct knet_link_status link_status;
  250. struct ipcs_conn_stats ipcs_conn_stats;
  251. struct ipcs_global_stats ipcs_global_stats;
  252. int res;
  253. int nodeid;
  254. int link_no;
  255. int service_id;
  256. uint32_t pid;
  257. void *conn_ptr;
  258. item = qb_map_get(stats_map, key_name);
  259. if (!item) {
  260. return CS_ERR_NOT_EXIST;
  261. }
  262. statinfo = item->cs_conv;
  263. switch (statinfo->type) {
  264. case STAT_PG:
  265. pg_stats = api->totem_get_stats();
  266. stats_map_set_value(statinfo, pg_stats, value, value_len, type);
  267. break;
  268. case STAT_SRP:
  269. pg_stats = api->totem_get_stats();
  270. stats_map_set_value(statinfo, pg_stats->srp, value, value_len, type);
  271. break;
  272. case STAT_KNET:
  273. if (sscanf(key_name, "stats.knet.node%d.link%d", &nodeid, &link_no) != 2) {
  274. return CS_ERR_NOT_EXIST;
  275. }
  276. /* Validate node & link IDs */
  277. if (nodeid <= 0 || nodeid > KNET_MAX_HOST ||
  278. link_no < 0 || link_no > KNET_MAX_LINK) {
  279. return CS_ERR_NOT_EXIST;
  280. }
  281. /* Always get the latest stats */
  282. res = totemknet_link_get_status((knet_node_id_t)nodeid, (uint8_t)link_no, &link_status);
  283. if (res != CS_OK) {
  284. return CS_ERR_LIBRARY;
  285. }
  286. stats_map_set_value(statinfo, &link_status, value, value_len, type);
  287. break;
  288. case STAT_IPCSC:
  289. if (sscanf(key_name, "stats.ipcs.service%d.%d.%p", &service_id, &pid, &conn_ptr) != 3) {
  290. return CS_ERR_NOT_EXIST;
  291. }
  292. res = cs_ipcs_get_conn_stats(service_id, pid, conn_ptr, &ipcs_conn_stats);
  293. if (res != CS_OK) {
  294. return res;
  295. }
  296. stats_map_set_value(statinfo, &ipcs_conn_stats, value, value_len, type);
  297. break;
  298. case STAT_IPCSG:
  299. cs_ipcs_get_global_stats(&ipcs_global_stats);
  300. stats_map_set_value(statinfo, &ipcs_global_stats, value, value_len, type);
  301. break;
  302. default:
  303. return CS_ERR_LIBRARY;
  304. }
  305. return CS_OK;
  306. }
  307. cs_error_t stats_map_set(const char *key_name,
  308. const void *value,
  309. size_t value_len,
  310. icmap_value_types_t type)
  311. {
  312. return CS_ERR_NOT_SUPPORTED;
  313. }
  314. cs_error_t stats_map_adjust_int(const char *key_name, int32_t step)
  315. {
  316. return CS_ERR_NOT_SUPPORTED;
  317. }
  318. cs_error_t stats_map_delete(const char *key_name)
  319. {
  320. return CS_ERR_NOT_SUPPORTED;
  321. }
  322. int stats_map_is_key_ro(const char *key_name)
  323. {
  324. /* It's all read-only */
  325. return 1;
  326. }
  327. icmap_iter_t stats_map_iter_init(const char *prefix)
  328. {
  329. return (qb_map_pref_iter_create(stats_map, prefix));
  330. }
  331. const char *stats_map_iter_next(icmap_iter_t iter, size_t *value_len, icmap_value_types_t *type)
  332. {
  333. const char *res;
  334. struct stats_item *item;
  335. res = qb_map_iter_next(iter, (void **)&item);
  336. if (res == NULL) {
  337. return (res);
  338. }
  339. stats_map_set_value(item->cs_conv, NULL, NULL, value_len, type);
  340. return res;
  341. }
  342. void stats_map_iter_finalize(icmap_iter_t iter)
  343. {
  344. qb_map_iter_free(iter);
  345. }
  346. void stats_trigger_trackers()
  347. {
  348. struct cs_stats_tracker *tracker;
  349. struct qb_list_head *iter;
  350. cs_error_t res;
  351. size_t value_len;
  352. icmap_value_types_t type;
  353. uint64_t value;
  354. struct icmap_notify_value new_val;
  355. struct icmap_notify_value old_val;
  356. qb_list_for_each(iter, &stats_tracker_list_head) {
  357. tracker = qb_list_entry(iter, struct cs_stats_tracker, list);
  358. if (tracker->events & ICMAP_TRACK_PREFIX || !tracker->key_name ) {
  359. continue;
  360. }
  361. res = stats_map_get(tracker->key_name,
  362. &value, &value_len, &type);
  363. /* Check if it has changed */
  364. if ((res == CS_OK) && (memcmp(&value, &tracker->old_value, value_len) != 0)) {
  365. old_val.type = new_val.type = type;
  366. old_val.len = new_val.len = value_len;
  367. old_val.data = new_val.data = &value;
  368. tracker->notify_fn(ICMAP_TRACK_MODIFY, tracker->key_name,
  369. old_val, new_val, tracker->user_data);
  370. memcpy(&tracker->old_value, &value, value_len);
  371. }
  372. }
  373. }
  374. /* Callback from libqb when a key is added/removed */
  375. static void stats_map_notify_fn(uint32_t event, char *key, void *old_value, void *value, void *user_data)
  376. {
  377. struct cs_stats_tracker *tracker = user_data;
  378. struct icmap_notify_value new_val;
  379. struct icmap_notify_value old_val;
  380. char new_value[64];
  381. if (value == NULL && old_value == NULL) {
  382. return ;
  383. }
  384. new_val.data = new_value;
  385. if (stats_map_get(key,
  386. &new_value,
  387. &new_val.len,
  388. &new_val.type) != CS_OK) {
  389. }
  390. /* We don't know what the old value was
  391. but as this only tracks ADD & DELETE I'm not worried
  392. about it */
  393. memcpy(&old_val, &new_val, sizeof(new_val));
  394. tracker->notify_fn(icmap_qbtt_to_tt(event),
  395. key,
  396. new_val,
  397. old_val,
  398. tracker->user_data);
  399. }
  400. cs_error_t stats_map_track_add(const char *key_name,
  401. int32_t track_type,
  402. icmap_notify_fn_t notify_fn,
  403. void *user_data,
  404. icmap_track_t *icmap_track)
  405. {
  406. struct cs_stats_tracker *tracker;
  407. size_t value_len;
  408. icmap_value_types_t type;
  409. cs_error_t err;
  410. /* We can track adding or deleting a key under a prefix */
  411. if ((track_type & ICMAP_TRACK_PREFIX) &&
  412. (!(track_type & ICMAP_TRACK_DELETE) ||
  413. !(track_type & ICMAP_TRACK_ADD))) {
  414. return CS_ERR_NOT_SUPPORTED;
  415. }
  416. tracker = malloc(sizeof(struct cs_stats_tracker));
  417. if (!tracker) {
  418. return CS_ERR_NO_MEMORY;
  419. }
  420. tracker->notify_fn = notify_fn;
  421. tracker->user_data = user_data;
  422. if (key_name) {
  423. tracker->key_name = strdup(key_name);
  424. if (!tracker->key_name) {
  425. free(tracker);
  426. return CS_ERR_NO_MEMORY;
  427. }
  428. /* Get initial value */
  429. if (stats_map_get(tracker->key_name,
  430. &tracker->old_value, &value_len, &type) == CS_OK) {
  431. tracker->old_value = 0ULL;
  432. }
  433. } else {
  434. tracker->key_name = NULL;
  435. tracker->old_value = 0ULL;
  436. }
  437. /* Add/delete trackers can use the qb_map tracking */
  438. if ((track_type & ICMAP_TRACK_ADD) ||
  439. (track_type & ICMAP_TRACK_DELETE)) {
  440. err = qb_map_notify_add(stats_map, tracker->key_name,
  441. stats_map_notify_fn,
  442. icmap_tt_to_qbtt(track_type),
  443. tracker);
  444. if (err != 0) {
  445. log_printf(LOGSYS_LEVEL_ERROR, "creating stats tracker %s failed. %d\n", tracker->key_name, err);
  446. free(tracker->key_name);
  447. free(tracker);
  448. return (qb_to_cs_error(err));
  449. }
  450. tracker->events = track_type;
  451. }
  452. qb_list_add (&tracker->list, &stats_tracker_list_head);
  453. *icmap_track = (icmap_track_t)tracker;
  454. return CS_OK;
  455. }
  456. cs_error_t stats_map_track_delete(icmap_track_t icmap_track)
  457. {
  458. struct cs_stats_tracker *tracker = (struct cs_stats_tracker *)icmap_track;
  459. int err;
  460. if ((tracker->events & ICMAP_TRACK_ADD) ||
  461. (tracker->events & ICMAP_TRACK_DELETE)) {
  462. err = qb_map_notify_del_2(stats_map,
  463. tracker->key_name, stats_map_notify_fn,
  464. icmap_tt_to_qbtt(tracker->events), tracker);
  465. if (err) {
  466. log_printf(LOGSYS_LEVEL_ERROR, "deleting tracker %s failed. %d\n", tracker->key_name, err);
  467. }
  468. }
  469. qb_list_del(&tracker->list);
  470. free(tracker->key_name);
  471. free(tracker);
  472. return CS_OK;
  473. }
  474. void *stats_map_track_get_user_data(icmap_track_t icmap_track)
  475. {
  476. struct cs_stats_tracker *tracker = (struct cs_stats_tracker *)icmap_track;
  477. return tracker->user_data;
  478. }
  479. /* Called from totemknet to add/remove keys from our map */
  480. void stats_knet_add_member(knet_node_id_t nodeid, uint8_t link_no)
  481. {
  482. int i;
  483. char param[ICMAP_KEYNAME_MAXLEN];
  484. for (i = 0; i<NUM_KNET_STATS; i++) {
  485. sprintf(param, "stats.knet.node%d.link%d.%s", nodeid, link_no, cs_knet_stats[i].name);
  486. stats_add_entry(param, &cs_knet_stats[i]);
  487. }
  488. }
  489. void stats_knet_del_member(knet_node_id_t nodeid, uint8_t link_no)
  490. {
  491. int i;
  492. char param[ICMAP_KEYNAME_MAXLEN];
  493. for (i = 0; i<NUM_KNET_STATS; i++) {
  494. sprintf(param, "stats.knet.node%d.link%d.%s", nodeid, link_no, cs_knet_stats[i].name);
  495. stats_rm_entry(param);
  496. }
  497. }
  498. /* Called from ipc_glue to add/remove keys from our map */
  499. void stats_ipcs_add_connection(int service_id, uint32_t pid, void *ptr)
  500. {
  501. int i;
  502. char param[ICMAP_KEYNAME_MAXLEN];
  503. for (i = 0; i<NUM_IPCSC_STATS; i++) {
  504. sprintf(param, "stats.ipcs.service%d.%d.%p.%s", service_id, pid, ptr, cs_ipcs_conn_stats[i].name);
  505. stats_add_entry(param, &cs_ipcs_conn_stats[i]);
  506. }
  507. }
  508. void stats_ipcs_del_connection(int service_id, uint32_t pid, void *ptr)
  509. {
  510. int i;
  511. char param[ICMAP_KEYNAME_MAXLEN];
  512. for (i = 0; i<NUM_IPCSC_STATS; i++) {
  513. sprintf(param, "stats.ipcs.service%d.%d.%p.%s", service_id, pid, ptr, cs_ipcs_conn_stats[i].name);
  514. stats_rm_entry(param);
  515. }
  516. }