jlink.c 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. // SPDX-License-Identifier: LGPL-2.1-or-later
  2. /*
  3. *
  4. * BlueZ - Bluetooth protocol stack for Linux
  5. *
  6. * Copyright (C) 2018 Codecoup
  7. *
  8. *
  9. */
  10. #ifdef HAVE_CONFIG_H
  11. #include <config.h>
  12. #endif
  13. #include <dlfcn.h>
  14. #include <errno.h>
  15. #include <stdio.h>
  16. #include <stdint.h>
  17. #include <stdlib.h>
  18. #include <string.h>
  19. #include <unistd.h>
  20. #include "jlink.h"
  21. #define RTT_CONTROL_START 0
  22. #define RTT_CONTROL_STOP 1
  23. #define RTT_CONTROL_GET_DESC 2
  24. #define RTT_CONTROL_GET_NUM_BUF 3
  25. #define RTT_CONTROL_GET_STAT 4
  26. #define RTT_DIRECTION_UP 0
  27. #define RTT_DIRECTION_DOWN 1
  28. static const char * const jlink_so_name[] = {
  29. "/usr/lib/libjlinkarm.so",
  30. "/usr/lib/libjlinkarm.so.6",
  31. "/opt/SEGGER/JLink/libjlinkarm.so",
  32. "/opt/SEGGER/JLink/libjlinkarm.so.6",
  33. };
  34. struct rtt_desc {
  35. uint32_t index;
  36. uint32_t direction;
  37. char name[32];
  38. uint32_t size;
  39. uint32_t flags;
  40. };
  41. static struct rtt_desc rtt_desc;
  42. typedef int (*jlink_emu_selectbyusbsn_func) (unsigned int sn);
  43. typedef int (*jlink_open_func) (void);
  44. typedef int (*jlink_execcommand_func) (char *in, char *out, int size);
  45. typedef int (*jlink_tif_select_func) (int);
  46. typedef void (*jlink_setspeed_func) (long int speed);
  47. typedef int (*jlink_connect_func) (void);
  48. typedef unsigned int (*jlink_getsn_func) (void);
  49. typedef void (*jlink_emu_getproductname_func) (char *out, int size);
  50. typedef int (*jlink_rtterminal_control_func) (int cmd, void *data);
  51. typedef int (*jlink_rtterminal_read_func) (int cmd, char *buf, int size);
  52. struct jlink {
  53. jlink_emu_selectbyusbsn_func emu_selectbyusbsn;
  54. jlink_open_func open;
  55. jlink_execcommand_func execcommand;
  56. jlink_tif_select_func tif_select;
  57. jlink_setspeed_func setspeed;
  58. jlink_connect_func connect;
  59. jlink_getsn_func getsn;
  60. jlink_emu_getproductname_func emu_getproductname;
  61. jlink_rtterminal_control_func rtterminal_control;
  62. jlink_rtterminal_read_func rtterminal_read;
  63. };
  64. static struct jlink jlink;
  65. #ifndef NELEM
  66. #define NELEM(x) (sizeof(x) / sizeof((x)[0]))
  67. #endif
  68. int jlink_init(void)
  69. {
  70. void *so;
  71. unsigned int i;
  72. for (i = 0; i < NELEM(jlink_so_name); i++) {
  73. so = dlopen(jlink_so_name[i], RTLD_LAZY);
  74. if (so)
  75. break;
  76. }
  77. if (!so)
  78. return -EIO;
  79. jlink.emu_selectbyusbsn = dlsym(so, "JLINK_EMU_SelectByUSBSN");
  80. jlink.open = dlsym(so, "JLINK_Open");
  81. jlink.execcommand = dlsym(so, "JLINK_ExecCommand");
  82. jlink.tif_select = dlsym(so, "JLINK_TIF_Select");
  83. jlink.setspeed = dlsym(so, "JLINK_SetSpeed");
  84. jlink.connect = dlsym(so, "JLINK_Connect");
  85. jlink.getsn = dlsym(so, "JLINK_GetSN");
  86. jlink.emu_getproductname = dlsym(so, "JLINK_EMU_GetProductName");
  87. jlink.rtterminal_control = dlsym(so, "JLINK_RTTERMINAL_Control");
  88. jlink.rtterminal_read = dlsym(so, "JLINK_RTTERMINAL_Read");
  89. if (!jlink.emu_selectbyusbsn || !jlink.open || !jlink.execcommand ||
  90. !jlink.tif_select || !jlink.setspeed ||
  91. !jlink.connect || !jlink.getsn ||
  92. !jlink.emu_getproductname ||
  93. !jlink.rtterminal_control || !jlink.rtterminal_read)
  94. return -EIO;
  95. return 0;
  96. }
  97. int jlink_connect(char *cfg)
  98. {
  99. const char *device = NULL;
  100. int tif = 1;
  101. unsigned int speed = 1000;
  102. unsigned int serial_no = 0;
  103. char *tok;
  104. char buf[64];
  105. tok = strtok(cfg, ",");
  106. device = tok;
  107. tok = strtok(NULL, ",");
  108. if (!tok)
  109. goto connect;
  110. if (strlen(tok))
  111. serial_no = atoi(tok);
  112. tok = strtok(NULL, ",");
  113. if (!tok)
  114. goto connect;
  115. if (strlen(tok)) {
  116. if (!strcasecmp("swd", tok))
  117. tif = 1;
  118. else
  119. return -EINVAL;
  120. }
  121. tok = strtok(NULL, ",");
  122. if (!tok)
  123. goto connect;
  124. if (strlen(tok))
  125. speed = atoi(tok);
  126. connect:
  127. if (serial_no)
  128. if (jlink.emu_selectbyusbsn(serial_no) < 0) {
  129. fprintf(stderr, "Failed to select emu by SN\n");
  130. return -ENODEV;
  131. }
  132. if (jlink.open() < 0) {
  133. fprintf(stderr, "Failed to open J-Link\n");
  134. return -ENODEV;
  135. }
  136. snprintf(buf, sizeof(buf), "device=%s", device);
  137. if (jlink.execcommand(buf, NULL, 0) < 0) {
  138. fprintf(stderr, "Failed to select target device\n");
  139. return -ENODEV;
  140. }
  141. if (jlink.tif_select(tif) < 0) {
  142. fprintf(stderr, "Failed to select target interface\n");
  143. return -ENODEV;
  144. }
  145. jlink.setspeed(speed);
  146. if (jlink.connect() < 0) {
  147. fprintf(stderr, "Failed to open target\n");
  148. return -EIO;
  149. }
  150. serial_no = jlink.getsn();
  151. jlink.emu_getproductname(buf, sizeof(buf));
  152. printf("Connected to %s (S/N: %u)\n", buf, serial_no);
  153. return 0;
  154. }
  155. int jlink_start_rtt(char *cfg)
  156. {
  157. unsigned int address = 0;
  158. unsigned int area_size = 0;
  159. const char *buffer = "btmonitor";
  160. char *tok;
  161. char cmd[64];
  162. int rtt_dir;
  163. int count;
  164. int i;
  165. if (!cfg)
  166. goto find_rttcb;
  167. tok = strtok(cfg, ",");
  168. if (strlen(tok)) {
  169. address = strtol(tok, NULL, 0);
  170. area_size = 0x1000;
  171. }
  172. tok = strtok(NULL, ",");
  173. if (!tok)
  174. goto find_rttcb;
  175. if (strlen(tok))
  176. area_size = strtol(tok, NULL, 0);
  177. tok = strtok(NULL, ",");
  178. if (!tok)
  179. goto find_rttcb;
  180. if (strlen(tok))
  181. buffer = tok;
  182. find_rttcb:
  183. if (address || area_size) {
  184. if (!area_size)
  185. snprintf(cmd, sizeof(cmd), "SetRTTAddr 0x%x", address);
  186. else
  187. snprintf(cmd, sizeof(cmd),
  188. "SetRTTSearchRanges 0x%x 0x%x",
  189. address, area_size);
  190. if (jlink.execcommand(cmd, NULL, 0) < 0)
  191. return -EIO;
  192. }
  193. if (jlink.rtterminal_control(RTT_CONTROL_START, NULL) < 0) {
  194. fprintf(stderr, "Failed to initialize RTT\n");
  195. return -1;
  196. }
  197. /* RTT may need some time to find control block so we need to wait */
  198. do {
  199. usleep(100);
  200. rtt_dir = RTT_DIRECTION_UP;
  201. count = jlink.rtterminal_control(RTT_CONTROL_GET_NUM_BUF,
  202. &rtt_dir);
  203. } while (count < 0);
  204. for (i = 0; i < count; i++) {
  205. memset(&rtt_desc, 0, sizeof(rtt_desc));
  206. rtt_desc.index = i;
  207. rtt_desc.direction = RTT_DIRECTION_UP;
  208. if (jlink.rtterminal_control(RTT_CONTROL_GET_DESC,
  209. &rtt_desc) < 0)
  210. continue;
  211. if (rtt_desc.size > 0 && !strcmp(buffer, rtt_desc.name))
  212. break;
  213. }
  214. if (i == count)
  215. return -ENODEV;
  216. printf("Using RTT up buffer #%d (size: %d)\n", i, rtt_desc.size);
  217. return 0;
  218. }
  219. int jlink_rtt_read(void *buf, size_t size)
  220. {
  221. return jlink.rtterminal_read(rtt_desc.index, buf, size);
  222. }