cpuload_stat.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. /*
  2. * Copyright (c) 2017 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief cpu load statistic
  9. */
  10. #include <kernel.h>
  11. #include <kernel_structs.h>
  12. #include <toolchain.h>
  13. #include <linker/sections.h>
  14. #include <string.h>
  15. #include <ksched.h>
  16. #include <wait_q.h>
  17. #include <sys/dlist.h>
  18. #include <init.h>
  19. /* start flag */
  20. static int cpuload_started;
  21. /* cpu load poll interval, unit: ms */
  22. static int cpuload_interval;
  23. struct k_delayed_work cpuload_stat_work;
  24. void k_thread_runtime_clear(void);
  25. static void cpuload_stat_clear(void)
  26. {
  27. k_thread_runtime_clear();
  28. }
  29. void cpuload_stat(uint32_t interval)
  30. {
  31. k_thread_runtime_stats_t rt_stats_thread;
  32. k_thread_runtime_stats_t rt_stats_all;
  33. struct k_thread *thread_list = NULL;
  34. unsigned int key, ratio, running_cycles, time_us;
  35. uint32_t ms_all;
  36. int is_curr_thread, curr_prio;
  37. k_thread_runtime_stats_all_get(&rt_stats_all);
  38. ms_all = SYS_CLOCK_HW_CYCLES_TO_NS_AVG(rt_stats_all.execution_cycles, 1000000);
  39. printk("stat interval %d, %d\n", interval, ms_all);
  40. interval = ms_all;
  41. printk("\t\t\t\t prio \t run(us)\t %%cpu\n");
  42. key = irq_lock();
  43. thread_list = (struct k_thread *)(_kernel.threads);
  44. while (thread_list != NULL) {
  45. k_thread_runtime_stats_get(thread_list, &rt_stats_thread);
  46. running_cycles = rt_stats_thread.execution_cycles;
  47. is_curr_thread = thread_list == k_current_get();
  48. curr_prio = k_thread_priority_get(thread_list);
  49. irq_unlock(key);
  50. time_us = SYS_CLOCK_HW_CYCLES_TO_NS_AVG(running_cycles, 1000);
  51. ratio = time_us / 10 / interval;
  52. printk("%s%s[%p]:\n\t\t\t\t %d \t %d \t\t %d\n",
  53. is_curr_thread ? "*" : " ",
  54. k_thread_name_get(thread_list)? k_thread_name_get(thread_list):"NULL",
  55. thread_list,
  56. curr_prio,
  57. time_us,
  58. ratio);
  59. key = irq_lock();
  60. thread_list = (struct k_thread *)thread_list->next_thread;
  61. }
  62. irq_unlock(key);
  63. cpuload_stat_clear();
  64. }
  65. static void cpuload_stat_callback(struct k_work *work)
  66. {
  67. cpuload_stat(cpuload_interval);
  68. k_delayed_work_submit(&cpuload_stat_work, Z_TIMEOUT_MS(cpuload_interval));
  69. }
  70. void cpuload_stat_start(int interval_ms)
  71. {
  72. if (!interval_ms)
  73. return;
  74. if (cpuload_started)
  75. k_delayed_work_cancel(&cpuload_stat_work);
  76. cpuload_stat_clear();
  77. cpuload_interval = interval_ms;
  78. cpuload_started = 1;
  79. k_delayed_work_init(&cpuload_stat_work, cpuload_stat_callback);
  80. k_delayed_work_submit(&cpuload_stat_work, Z_TIMEOUT_MS(interval_ms));
  81. }
  82. void cpuload_stat_stop(void)
  83. {
  84. k_delayed_work_cancel(&cpuload_stat_work);
  85. cpuload_started = 0;
  86. }