| 1 | /* | 
|---|
| 2 | * Copyright (c) 2010 Stanislav Kozina | 
|---|
| 3 | * Copyright (c) 2010 Martin Decky | 
|---|
| 4 | * All rights reserved. | 
|---|
| 5 | * | 
|---|
| 6 | * Redistribution and use in source and binary forms, with or without | 
|---|
| 7 | * modification, are permitted provided that the following conditions | 
|---|
| 8 | * are met: | 
|---|
| 9 | * | 
|---|
| 10 | * - Redistributions of source code must retain the above copyright | 
|---|
| 11 | *   notice, this list of conditions and the following disclaimer. | 
|---|
| 12 | * - Redistributions in binary form must reproduce the above copyright | 
|---|
| 13 | *   notice, this list of conditions and the following disclaimer in the | 
|---|
| 14 | *   documentation and/or other materials provided with the distribution. | 
|---|
| 15 | * - The name of the author may not be used to endorse or promote products | 
|---|
| 16 | *   derived from this software without specific prior written permission. | 
|---|
| 17 | * | 
|---|
| 18 | * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR | 
|---|
| 19 | * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | 
|---|
| 20 | * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. | 
|---|
| 21 | * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, | 
|---|
| 22 | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT | 
|---|
| 23 | * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
|---|
| 24 | * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
|---|
| 25 | * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
|---|
| 26 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF | 
|---|
| 27 | * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|---|
| 28 | */ | 
|---|
| 29 |  | 
|---|
| 30 | /** @addtogroup libc | 
|---|
| 31 | * @{ | 
|---|
| 32 | */ | 
|---|
| 33 | /** @file | 
|---|
| 34 | */ | 
|---|
| 35 |  | 
|---|
| 36 | #include <stats.h> | 
|---|
| 37 | #include <sysinfo.h> | 
|---|
| 38 | #include <errno.h> | 
|---|
| 39 | #include <stdio.h> | 
|---|
| 40 | #include <inttypes.h> | 
|---|
| 41 | #include <malloc.h> | 
|---|
| 42 |  | 
|---|
| 43 | #define SYSINFO_STATS_MAX_PATH  64 | 
|---|
| 44 |  | 
|---|
| 45 | /** Thread states | 
|---|
| 46 | * | 
|---|
| 47 | */ | 
|---|
| 48 | static const char *thread_states[] = { | 
|---|
| 49 | "Invalid", | 
|---|
| 50 | "Running", | 
|---|
| 51 | "Sleeping", | 
|---|
| 52 | "Ready", | 
|---|
| 53 | "Entering", | 
|---|
| 54 | "Exiting", | 
|---|
| 55 | "Lingering" | 
|---|
| 56 | }; | 
|---|
| 57 |  | 
|---|
| 58 | /** Get CPUs statistics | 
|---|
| 59 | * | 
|---|
| 60 | * @param count Number of records returned. | 
|---|
| 61 | * | 
|---|
| 62 | * @return Array of stats_cpu_t structures. | 
|---|
| 63 | *         If non-NULL then it should be eventually freed | 
|---|
| 64 | *         by free(). | 
|---|
| 65 | * | 
|---|
| 66 | */ | 
|---|
| 67 | stats_cpu_t *stats_get_cpus(size_t *count) | 
|---|
| 68 | { | 
|---|
| 69 | size_t size = 0; | 
|---|
| 70 | stats_cpu_t *stats_cpus = | 
|---|
| 71 | (stats_cpu_t *) sysinfo_get_data("system.cpus", &size); | 
|---|
| 72 |  | 
|---|
| 73 | if ((size % sizeof(stats_cpu_t)) != 0) { | 
|---|
| 74 | if (stats_cpus != NULL) | 
|---|
| 75 | free(stats_cpus); | 
|---|
| 76 | *count = 0; | 
|---|
| 77 | return NULL; | 
|---|
| 78 | } | 
|---|
| 79 |  | 
|---|
| 80 | *count = size / sizeof(stats_cpu_t); | 
|---|
| 81 | return stats_cpus; | 
|---|
| 82 | } | 
|---|
| 83 |  | 
|---|
| 84 | /** Get physical memory statistics | 
|---|
| 85 | * | 
|---|
| 86 | * | 
|---|
| 87 | * @return Pointer to the stats_physmem_t structure. | 
|---|
| 88 | *         If non-NULL then it should be eventually freed | 
|---|
| 89 | *         by free(). | 
|---|
| 90 | * | 
|---|
| 91 | */ | 
|---|
| 92 | stats_physmem_t *stats_get_physmem(void) | 
|---|
| 93 | { | 
|---|
| 94 | size_t size = 0; | 
|---|
| 95 | stats_physmem_t *stats_physmem = | 
|---|
| 96 | (stats_physmem_t *) sysinfo_get_data("system.physmem", &size); | 
|---|
| 97 |  | 
|---|
| 98 | if (size != sizeof(stats_physmem_t)) { | 
|---|
| 99 | if (stats_physmem != NULL) | 
|---|
| 100 | free(stats_physmem); | 
|---|
| 101 | return NULL; | 
|---|
| 102 | } | 
|---|
| 103 |  | 
|---|
| 104 | return stats_physmem; | 
|---|
| 105 | } | 
|---|
| 106 |  | 
|---|
| 107 | /** Get task statistics | 
|---|
| 108 | * | 
|---|
| 109 | * @param count Number of records returned. | 
|---|
| 110 | * | 
|---|
| 111 | * @return Array of stats_task_t structures. | 
|---|
| 112 | *         If non-NULL then it should be eventually freed | 
|---|
| 113 | *         by free(). | 
|---|
| 114 | * | 
|---|
| 115 | */ | 
|---|
| 116 | stats_task_t *stats_get_tasks(size_t *count) | 
|---|
| 117 | { | 
|---|
| 118 | size_t size = 0; | 
|---|
| 119 | stats_task_t *stats_tasks = | 
|---|
| 120 | (stats_task_t *) sysinfo_get_data("system.tasks", &size); | 
|---|
| 121 |  | 
|---|
| 122 | if ((size % sizeof(stats_task_t)) != 0) { | 
|---|
| 123 | if (stats_tasks != NULL) | 
|---|
| 124 | free(stats_tasks); | 
|---|
| 125 | *count = 0; | 
|---|
| 126 | return NULL; | 
|---|
| 127 | } | 
|---|
| 128 |  | 
|---|
| 129 | *count = size / sizeof(stats_task_t); | 
|---|
| 130 | return stats_tasks; | 
|---|
| 131 | } | 
|---|
| 132 |  | 
|---|
| 133 | /** Get single task statistics | 
|---|
| 134 | * | 
|---|
| 135 | * @param task_id Task ID we are interested in. | 
|---|
| 136 | * | 
|---|
| 137 | * @return Pointer to the stats_task_t structure. | 
|---|
| 138 | *         If non-NULL then it should be eventually freed | 
|---|
| 139 | *         by free(). | 
|---|
| 140 | * | 
|---|
| 141 | */ | 
|---|
| 142 | stats_task_t *stats_get_task(task_id_t task_id) | 
|---|
| 143 | { | 
|---|
| 144 | char name[SYSINFO_STATS_MAX_PATH]; | 
|---|
| 145 | snprintf(name, SYSINFO_STATS_MAX_PATH, "system.tasks.%" PRIu64, task_id); | 
|---|
| 146 |  | 
|---|
| 147 | size_t size = 0; | 
|---|
| 148 | stats_task_t *stats_task = | 
|---|
| 149 | (stats_task_t *) sysinfo_get_data(name, &size); | 
|---|
| 150 |  | 
|---|
| 151 | if (size != sizeof(stats_task_t)) { | 
|---|
| 152 | if (stats_task != NULL) | 
|---|
| 153 | free(stats_task); | 
|---|
| 154 | return NULL; | 
|---|
| 155 | } | 
|---|
| 156 |  | 
|---|
| 157 | return stats_task; | 
|---|
| 158 | } | 
|---|
| 159 |  | 
|---|
| 160 | /** Get thread statistics. | 
|---|
| 161 | * | 
|---|
| 162 | * @param count Number of records returned. | 
|---|
| 163 | * | 
|---|
| 164 | * @return Array of stats_thread_t structures. | 
|---|
| 165 | *         If non-NULL then it should be eventually freed | 
|---|
| 166 | *         by free(). | 
|---|
| 167 | * | 
|---|
| 168 | */ | 
|---|
| 169 | stats_thread_t *stats_get_threads(size_t *count) | 
|---|
| 170 | { | 
|---|
| 171 | size_t size = 0; | 
|---|
| 172 | stats_thread_t *stats_threads = | 
|---|
| 173 | (stats_thread_t *) sysinfo_get_data("system.threads", &size); | 
|---|
| 174 |  | 
|---|
| 175 | if ((size % sizeof(stats_thread_t)) != 0) { | 
|---|
| 176 | if (stats_threads != NULL) | 
|---|
| 177 | free(stats_threads); | 
|---|
| 178 | *count = 0; | 
|---|
| 179 | return NULL; | 
|---|
| 180 | } | 
|---|
| 181 |  | 
|---|
| 182 | *count = size / sizeof(stats_thread_t); | 
|---|
| 183 | return stats_threads; | 
|---|
| 184 | } | 
|---|
| 185 |  | 
|---|
| 186 | /** Get single thread statistics | 
|---|
| 187 | * | 
|---|
| 188 | * @param thread_id Thread ID we are interested in. | 
|---|
| 189 | * | 
|---|
| 190 | * @return Pointer to the stats_thread_t structure. | 
|---|
| 191 | *         If non-NULL then it should be eventually freed | 
|---|
| 192 | *         by free(). | 
|---|
| 193 | * | 
|---|
| 194 | */ | 
|---|
| 195 | stats_thread_t *stats_get_thread(thread_id_t thread_id) | 
|---|
| 196 | { | 
|---|
| 197 | char name[SYSINFO_STATS_MAX_PATH]; | 
|---|
| 198 | snprintf(name, SYSINFO_STATS_MAX_PATH, "system.threads.%" PRIu64, thread_id); | 
|---|
| 199 |  | 
|---|
| 200 | size_t size = 0; | 
|---|
| 201 | stats_thread_t *stats_thread = | 
|---|
| 202 | (stats_thread_t *) sysinfo_get_data(name, &size); | 
|---|
| 203 |  | 
|---|
| 204 | if (size != sizeof(stats_thread_t)) { | 
|---|
| 205 | if (stats_thread != NULL) | 
|---|
| 206 | free(stats_thread); | 
|---|
| 207 | return NULL; | 
|---|
| 208 | } | 
|---|
| 209 |  | 
|---|
| 210 | return stats_thread; | 
|---|
| 211 | } | 
|---|
| 212 |  | 
|---|
| 213 | /** Get exception statistics. | 
|---|
| 214 | * | 
|---|
| 215 | * @param count Number of records returned. | 
|---|
| 216 | * | 
|---|
| 217 | * @return Array of stats_exc_t structures. | 
|---|
| 218 | *         If non-NULL then it should be eventually freed | 
|---|
| 219 | *         by free(). | 
|---|
| 220 | * | 
|---|
| 221 | */ | 
|---|
| 222 | stats_exc_t *stats_get_exceptions(size_t *count) | 
|---|
| 223 | { | 
|---|
| 224 | size_t size = 0; | 
|---|
| 225 | stats_exc_t *stats_exceptions = | 
|---|
| 226 | (stats_exc_t *) sysinfo_get_data("system.exceptions", &size); | 
|---|
| 227 |  | 
|---|
| 228 | if ((size % sizeof(stats_exc_t)) != 0) { | 
|---|
| 229 | if (stats_exceptions != NULL) | 
|---|
| 230 | free(stats_exceptions); | 
|---|
| 231 | *count = 0; | 
|---|
| 232 | return NULL; | 
|---|
| 233 | } | 
|---|
| 234 |  | 
|---|
| 235 | *count = size / sizeof(stats_exc_t); | 
|---|
| 236 | return stats_exceptions; | 
|---|
| 237 | } | 
|---|
| 238 |  | 
|---|
| 239 | /** Get single exception statistics | 
|---|
| 240 | * | 
|---|
| 241 | * @param excn Exception number we are interested in. | 
|---|
| 242 | * | 
|---|
| 243 | * @return Pointer to the stats_exc_t structure. | 
|---|
| 244 | *         If non-NULL then it should be eventually freed | 
|---|
| 245 | *         by free(). | 
|---|
| 246 | * | 
|---|
| 247 | */ | 
|---|
| 248 | stats_exc_t *stats_get_exception(unsigned int excn) | 
|---|
| 249 | { | 
|---|
| 250 | char name[SYSINFO_STATS_MAX_PATH]; | 
|---|
| 251 | snprintf(name, SYSINFO_STATS_MAX_PATH, "system.exceptions.%u", excn); | 
|---|
| 252 |  | 
|---|
| 253 | size_t size = 0; | 
|---|
| 254 | stats_exc_t *stats_exception = | 
|---|
| 255 | (stats_exc_t *) sysinfo_get_data(name, &size); | 
|---|
| 256 |  | 
|---|
| 257 | if (size != sizeof(stats_exc_t)) { | 
|---|
| 258 | if (stats_exception != NULL) | 
|---|
| 259 | free(stats_exception); | 
|---|
| 260 | return NULL; | 
|---|
| 261 | } | 
|---|
| 262 |  | 
|---|
| 263 | return stats_exception; | 
|---|
| 264 | } | 
|---|
| 265 |  | 
|---|
| 266 | /** Get system load | 
|---|
| 267 | * | 
|---|
| 268 | * @param count Number of load records returned. | 
|---|
| 269 | * | 
|---|
| 270 | * @return Array of load records (load_t). | 
|---|
| 271 | *         If non-NULL then it should be eventually freed | 
|---|
| 272 | *         by free(). | 
|---|
| 273 | * | 
|---|
| 274 | */ | 
|---|
| 275 | load_t *stats_get_load(size_t *count) | 
|---|
| 276 | { | 
|---|
| 277 | size_t size = 0; | 
|---|
| 278 | load_t *load = | 
|---|
| 279 | (load_t *) sysinfo_get_data("system.load", &size); | 
|---|
| 280 |  | 
|---|
| 281 | if ((size % sizeof(load_t)) != 0) { | 
|---|
| 282 | if (load != NULL) | 
|---|
| 283 | free(load); | 
|---|
| 284 | *count = 0; | 
|---|
| 285 | return NULL; | 
|---|
| 286 | } | 
|---|
| 287 |  | 
|---|
| 288 | *count = size / sizeof(load_t); | 
|---|
| 289 | return load; | 
|---|
| 290 | } | 
|---|
| 291 |  | 
|---|
| 292 | /** Get system uptime | 
|---|
| 293 | * | 
|---|
| 294 | * @return System uptime (in seconds). | 
|---|
| 295 | * | 
|---|
| 296 | */ | 
|---|
| 297 | sysarg_t stats_get_uptime(void) | 
|---|
| 298 | { | 
|---|
| 299 | sysarg_t uptime; | 
|---|
| 300 | if (sysinfo_get_value("system.uptime", &uptime) != EOK) | 
|---|
| 301 | uptime = 0; | 
|---|
| 302 |  | 
|---|
| 303 | return uptime; | 
|---|
| 304 | } | 
|---|
| 305 |  | 
|---|
| 306 | /** Print load fixed-point value | 
|---|
| 307 | * | 
|---|
| 308 | * Print the load record fixed-point value in decimal | 
|---|
| 309 | * representation on stdout. | 
|---|
| 310 | * | 
|---|
| 311 | * @param upper      Load record. | 
|---|
| 312 | * @param dec_length Number of decimal digits to print. | 
|---|
| 313 | * | 
|---|
| 314 | */ | 
|---|
| 315 | void stats_print_load_fragment(load_t upper, unsigned int dec_length) | 
|---|
| 316 | { | 
|---|
| 317 | /* Magic value from BSD */ | 
|---|
| 318 | load_t lower = 65536; | 
|---|
| 319 |  | 
|---|
| 320 | /* Print the whole part */ | 
|---|
| 321 | printf("%u.", upper / lower); | 
|---|
| 322 |  | 
|---|
| 323 | load_t rest = (upper % lower) * 10; | 
|---|
| 324 |  | 
|---|
| 325 | unsigned int i; | 
|---|
| 326 | for (i = 0; i < dec_length; i++) { | 
|---|
| 327 | printf("%u", rest / lower); | 
|---|
| 328 | rest = (rest % lower) * 10; | 
|---|
| 329 | } | 
|---|
| 330 | } | 
|---|
| 331 |  | 
|---|
| 332 | const char *thread_get_state(state_t state) | 
|---|
| 333 | { | 
|---|
| 334 | if (state <= Lingering) | 
|---|
| 335 | return thread_states[state]; | 
|---|
| 336 |  | 
|---|
| 337 | return thread_states[Invalid]; | 
|---|
| 338 | } | 
|---|
| 339 |  | 
|---|
| 340 | /** @} | 
|---|
| 341 | */ | 
|---|