1 /*
2 * Copyright (c) 2018 Intel Corporation
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/kernel.h>
8 #include <SEGGER_SYSVIEW.h>
9 #include <ksched.h>
10
11 extern const SEGGER_SYSVIEW_OS_API SYSVIEW_X_OS_TraceAPI;
12
13 #if CONFIG_THREAD_MAX_NAME_LEN
14 #define THREAD_NAME_LEN CONFIG_THREAD_MAX_NAME_LEN
15 #else
16 #define THREAD_NAME_LEN 20
17 #endif
18
19
set_thread_name(char * name,struct k_thread * thread)20 static void set_thread_name(char *name, struct k_thread *thread)
21 {
22 const char *tname = k_thread_name_get(thread);
23
24 if (tname != NULL && tname[0] != '\0') {
25 memcpy(name, tname, THREAD_NAME_LEN);
26 name[THREAD_NAME_LEN - 1] = '\0';
27 } else {
28 snprintk(name, THREAD_NAME_LEN, "T%pE%p",
29 thread, &thread->entry);
30 }
31 }
32
sys_trace_thread_info(struct k_thread * thread)33 void sys_trace_thread_info(struct k_thread *thread)
34 {
35 char name[THREAD_NAME_LEN];
36
37 set_thread_name(name, thread);
38
39 SEGGER_SYSVIEW_TASKINFO Info;
40
41 Info.TaskID = (uint32_t)(uintptr_t)thread;
42 Info.sName = name;
43 Info.Prio = thread->base.prio;
44 Info.StackBase = thread->stack_info.size;
45 Info.StackSize = thread->stack_info.start;
46 SEGGER_SYSVIEW_SendTaskInfo(&Info);
47 }
48
49
cbSendSystemDesc(void)50 static void cbSendSystemDesc(void)
51 {
52 SEGGER_SYSVIEW_SendSysDesc("N=" CONFIG_SEGGER_SYSVIEW_APP_NAME);
53 SEGGER_SYSVIEW_SendSysDesc("D=" CONFIG_BOARD " "
54 CONFIG_SOC_FAMILY " " CONFIG_ARCH);
55 SEGGER_SYSVIEW_SendSysDesc("O=Zephyr");
56 }
57
send_task_list_cb(void)58 static void send_task_list_cb(void)
59 {
60 struct k_thread *thread;
61
62 for (thread = _kernel.threads; thread; thread = thread->next_thread) {
63 char name[THREAD_NAME_LEN];
64
65 if (z_is_idle_thread_object(thread)) {
66 continue;
67 }
68
69 set_thread_name(name, thread);
70
71 SEGGER_SYSVIEW_SendTaskInfo(&(SEGGER_SYSVIEW_TASKINFO) {
72 .TaskID = (uint32_t)(uintptr_t)thread,
73 .sName = name,
74 .StackSize = thread->stack_info.size,
75 .StackBase = thread->stack_info.start,
76 .Prio = thread->base.prio,
77 });
78 }
79 }
80
81
get_time_cb(void)82 static U64 get_time_cb(void)
83 {
84 return (U64)k_cycle_get_32();
85 }
86
87
88 const SEGGER_SYSVIEW_OS_API SYSVIEW_X_OS_TraceAPI = {
89 get_time_cb,
90 send_task_list_cb,
91 };
92
93
94
SEGGER_SYSVIEW_Conf(void)95 void SEGGER_SYSVIEW_Conf(void)
96 {
97 SEGGER_SYSVIEW_Init(sys_clock_hw_cycles_per_sec(),
98 sys_clock_hw_cycles_per_sec(),
99 &SYSVIEW_X_OS_TraceAPI, cbSendSystemDesc);
100 #if DT_NODE_HAS_STATUS_OKAY(DT_CHOSEN(zephyr_sram))
101 SEGGER_SYSVIEW_SetRAMBase(DT_REG_ADDR(DT_CHOSEN(zephyr_sram)));
102 #else
103 /* Setting RAMBase is just an optimization: this value is subtracted
104 * from all pointers in order to save bandwidth. It's not an error
105 * if a platform does not set this value.
106 */
107 #endif
108 }
109