Add project website URL
[libjaylink.git] / libjaylink / target.c
blob923e3ada40c97ba55e13330662166a81386409ea
1 /*
2 * This file is part of the libjaylink project.
4 * Copyright (C) 2014-2015 Marc Schink <jaylink-dev@marcschink.de>
6 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
20 #include <stdint.h>
21 #include <stdbool.h>
23 #include "libjaylink.h"
24 #include "libjaylink-internal.h"
26 /**
27 * @file
29 * Target related functions.
32 /** @cond PRIVATE */
33 #define CMD_SET_SPEED 0x05
34 #define CMD_SET_TARGET_POWER 0x08
35 #define CMD_GET_SPEEDS 0xc0
36 #define CMD_SELECT_TIF 0xc7
37 #define CMD_CLEAR_RESET 0xdc
38 #define CMD_SET_RESET 0xdd
40 #define TIF_GET_SELECTED 0xfe
41 #define TIF_GET_AVAILABLE 0xff
42 /** @endcond */
44 /**
45 * Set the target interface speed.
47 * @param[in,out] devh Device handle.
48 * @param[in] speed Speed in kHz or #JAYLINK_SPEED_ADAPTIVE_CLOCKING for
49 * adaptive clocking. Speed of 0 kHz is not allowed and
50 * adaptive clocking must only be used if the device has the
51 * #JAYLINK_DEV_CAP_ADAPTIVE_CLOCKING capability.
53 * @retval JAYLINK_OK Success.
54 * @retval JAYLINK_ERR_ARG Invalid arguments.
55 * @retval JAYLINK_ERR_TIMEOUT A timeout occurred.
56 * @retval JAYLINK_ERR_IO Input/output error.
57 * @retval JAYLINK_ERR Other error conditions.
59 * @see jaylink_get_speeds()
61 * @since 0.1.0
63 JAYLINK_API int jaylink_set_speed(struct jaylink_device_handle *devh,
64 uint16_t speed)
66 int ret;
67 struct jaylink_context *ctx;
68 uint8_t buf[3];
70 if (!devh || !speed)
71 return JAYLINK_ERR_ARG;
73 ctx = devh->dev->ctx;
74 ret = transport_start_write(devh, 3, true);
76 if (ret != JAYLINK_OK) {
77 log_err(ctx, "transport_start_write() failed: %s.",
78 jaylink_strerror(ret));
79 return ret;
82 buf[0] = CMD_SET_SPEED;
83 buffer_set_u16(buf, speed, 1);
85 ret = transport_write(devh, buf, 3);
87 if (ret != JAYLINK_OK) {
88 log_err(ctx, "transport_write() failed: %s.",
89 jaylink_strerror(ret));
90 return ret;
93 return JAYLINK_OK;
96 /**
97 * Retrieve target interface speeds.
99 * The speeds are applicable for the currently selected target interface only
100 * and calculated as follows:
102 * @par
103 * <tt>speeds = @a freq / n</tt> with <tt>n >= @a div</tt>, where @p n is an
104 * integer
106 * Assuming, for example, a base frequency @a freq of 4 MHz and a minimum
107 * divider @a div of 4 then the highest possible target interface speed is
108 * 4 MHz / 4 = 1 MHz. The next highest speed is 800 kHz for a divider of 5, and
109 * so on.
111 * @note This function must only be used if the device has the
112 * #JAYLINK_DEV_CAP_GET_SPEEDS capability.
114 * @param[in,out] devh Device handle.
115 * @param[out] speed Speed information on success, and undefined on failure.
117 * @retval JAYLINK_OK Success.
118 * @retval JAYLINK_ERR_ARG Invalid arguments.
119 * @retval JAYLINK_ERR_TIMEOUT A timeout occurred.
120 * @retval JAYLINK_ERR_PROTO Protocol violation.
121 * @retval JAYLINK_ERR_IO Input/output error.
122 * @retval JAYLINK_ERR Other error conditions.
124 * @see jaylink_select_interface()
126 * @since 0.1.0
128 JAYLINK_API int jaylink_get_speeds(struct jaylink_device_handle *devh,
129 struct jaylink_speed *speed)
131 int ret;
132 struct jaylink_context *ctx;
133 uint8_t buf[6];
134 uint16_t div;
136 if (!devh || !speed)
137 return JAYLINK_ERR_ARG;
139 ctx = devh->dev->ctx;
140 ret = transport_start_write_read(devh, 1, 6, true);
142 if (ret != JAYLINK_OK) {
143 log_err(ctx, "transport_start_write_read() failed: %s.",
144 jaylink_strerror(ret));
145 return ret;
148 buf[0] = CMD_GET_SPEEDS;
150 ret = transport_write(devh, buf, 1);
152 if (ret != JAYLINK_OK) {
153 log_err(ctx, "transport_write() failed: %s.",
154 jaylink_strerror(ret));
155 return ret;
158 ret = transport_read(devh, buf, 6);
160 if (ret != JAYLINK_OK) {
161 log_err(ctx, "transport_read() failed: %s.",
162 jaylink_strerror(ret));
163 return ret;
166 div = buffer_get_u16(buf, 4);
168 if (!div) {
169 log_err(ctx, "Minimum frequency divider is zero.");
170 return JAYLINK_ERR_PROTO;
173 speed->freq = buffer_get_u32(buf, 0);
174 speed->div = div;
176 return JAYLINK_OK;
180 * Select the target interface.
182 * @note This function must only be used if the device has the
183 * #JAYLINK_DEV_CAP_SELECT_TIF capability.
185 * @warning This function may return a value for @p prev_iface which is not
186 * covered by #jaylink_target_interface.
188 * @param[in,out] devh Device handle.
189 * @param[in] iface Target interface to select.
190 * @param[out] prev_iface Previously selected target interface on success, and
191 * undefined on failure. Can be NULL.
193 * @retval JAYLINK_OK Success.
194 * @retval JAYLINK_ERR_ARG Invalid arguments.
195 * @retval JAYLINK_ERR_TIMEOUT A timeout occurred.
196 * @retval JAYLINK_ERR_IO Input/output error.
197 * @retval JAYLINK_ERR Other error conditions.
199 * @see jaylink_get_available_interfaces()
201 * @since 0.1.0
203 JAYLINK_API int jaylink_select_interface(struct jaylink_device_handle *devh,
204 enum jaylink_target_interface iface,
205 enum jaylink_target_interface *prev_iface)
207 int ret;
208 struct jaylink_context *ctx;
209 uint8_t buf[4];
211 if (!devh)
212 return JAYLINK_ERR_ARG;
214 switch (iface) {
215 case JAYLINK_TIF_JTAG:
216 case JAYLINK_TIF_SWD:
217 case JAYLINK_TIF_BDM3:
218 case JAYLINK_TIF_FINE:
219 case JAYLINK_TIF_2W_JTAG_PIC32:
220 break;
221 default:
222 return JAYLINK_ERR_ARG;
225 ctx = devh->dev->ctx;
226 ret = transport_start_write_read(devh, 2, 4, true);
228 if (ret != JAYLINK_OK) {
229 log_err(ctx, "transport_start_write_read() failed: %s.",
230 jaylink_strerror(ret));
231 return ret;
234 buf[0] = CMD_SELECT_TIF;
235 buf[1] = iface;
237 ret = transport_write(devh, buf, 2);
239 if (ret != JAYLINK_OK) {
240 log_err(ctx, "transport_write() failed: %s.",
241 jaylink_strerror(ret));
242 return ret;
245 ret = transport_read(devh, buf, 4);
247 if (ret != JAYLINK_OK) {
248 log_err(ctx, "transport_read() failed: %s.",
249 jaylink_strerror(ret));
250 return ret;
253 if (prev_iface)
254 *prev_iface = buffer_get_u32(buf, 0);
256 return JAYLINK_OK;
260 * Retrieve the available target interfaces.
262 * The target interfaces are stored in a 32-bit bit field where each individual
263 * bit represents a target interface. A set bit indicates an available target
264 * interface. See #jaylink_target_interface for a description of the target
265 * interfaces and their bit positions.
267 * @note This function must only be used if the device has the
268 * #JAYLINK_DEV_CAP_SELECT_TIF capability.
270 * @param[in,out] devh Device handle.
271 * @param[out] interfaces Target interfaces on success, and undefined on
272 * failure.
274 * @retval JAYLINK_OK Success.
275 * @retval JAYLINK_ERR_ARG Invalid arguments.
276 * @retval JAYLINK_ERR_TIMEOUT A timeout occurred.
277 * @retval JAYLINK_ERR_IO Input/output error.
278 * @retval JAYLINK_ERR Other error conditions.
280 * @see jaylink_select_interface()
282 * @since 0.1.0
284 JAYLINK_API int jaylink_get_available_interfaces(
285 struct jaylink_device_handle *devh, uint32_t *interfaces)
287 int ret;
288 struct jaylink_context *ctx;
289 uint8_t buf[4];
291 if (!devh || !interfaces)
292 return JAYLINK_ERR_ARG;
294 ctx = devh->dev->ctx;
295 ret = transport_start_write_read(devh, 2, 4, true);
297 if (ret != JAYLINK_OK) {
298 log_err(ctx, "transport_start_write_read() failed: %s.",
299 jaylink_strerror(ret));
300 return ret;
303 buf[0] = CMD_SELECT_TIF;
304 buf[1] = TIF_GET_AVAILABLE;
306 ret = transport_write(devh, buf, 2);
308 if (ret != JAYLINK_OK) {
309 log_err(ctx, "transport_write() failed: %s.",
310 jaylink_strerror(ret));
311 return ret;
314 ret = transport_read(devh, buf, 4);
316 if (ret != JAYLINK_OK) {
317 log_err(ctx, "transport_read() failed: %s.",
318 jaylink_strerror(ret));
319 return ret;
322 *interfaces = buffer_get_u32(buf, 0);
324 return JAYLINK_OK;
328 * Retrieve the selected target interface.
330 * @note This function must only be used if the device has the
331 * #JAYLINK_DEV_CAP_SELECT_TIF capability.
333 * @warning This function may return a value for @p iface which is not covered
334 * by #jaylink_target_interface.
336 * @param[in,out] devh Device handle.
337 * @param[out] iface Selected target interface on success, and undefined on
338 * failure.
340 * @retval JAYLINK_OK Success.
341 * @retval JAYLINK_ERR_ARG Invalid arguments.
342 * @retval JAYLINK_ERR_TIMEOUT A timeout occurred.
343 * @retval JAYLINK_ERR_IO Input/output error.
344 * @retval JAYLINK_ERR Other error conditions.
346 * @see jaylink_select_interface()
348 * @since 0.1.0
350 JAYLINK_API int jaylink_get_selected_interface(
351 struct jaylink_device_handle *devh,
352 enum jaylink_target_interface *iface)
354 int ret;
355 struct jaylink_context *ctx;
356 uint8_t buf[4];
358 if (!devh || !iface)
359 return JAYLINK_ERR_ARG;
361 ctx = devh->dev->ctx;
362 ret = transport_start_write_read(devh, 2, 4, true);
364 if (ret != JAYLINK_OK) {
365 log_err(ctx, "transport_start_write_read() failed: %s.",
366 jaylink_strerror(ret));
367 return ret;
370 buf[0] = CMD_SELECT_TIF;
371 buf[1] = TIF_GET_SELECTED;
373 ret = transport_write(devh, buf, 2);
375 if (ret != JAYLINK_OK) {
376 log_err(ctx, "transport_write() failed: %s.",
377 jaylink_strerror(ret));
378 return ret;
381 ret = transport_read(devh, buf, 4);
383 if (ret != JAYLINK_OK) {
384 log_err(ctx, "transport_read() failed: %s.",
385 jaylink_strerror(ret));
386 return ret;
389 *iface = buffer_get_u32(buf, 0);
391 return JAYLINK_OK;
395 * Clear the target reset signal.
397 * @param[in,out] devh Device handle.
399 * @retval JAYLINK_OK Success.
400 * @retval JAYLINK_ERR_ARG Invalid arguments.
401 * @retval JAYLINK_ERR_TIMEOUT A timeout occurred.
402 * @retval JAYLINK_ERR_IO Input/output error.
403 * @retval JAYLINK_ERR Other error conditions.
405 * @since 0.1.0
407 JAYLINK_API int jaylink_clear_reset(struct jaylink_device_handle *devh)
409 int ret;
410 struct jaylink_context *ctx;
411 uint8_t buf[1];
413 if (!devh)
414 return JAYLINK_ERR_ARG;
416 ctx = devh->dev->ctx;
417 ret = transport_start_write(devh, 1, true);
419 if (ret != JAYLINK_OK) {
420 log_err(ctx, "transport_start_write() failed: %s.",
421 jaylink_strerror(ret));
422 return ret;
425 buf[0] = CMD_CLEAR_RESET;
427 ret = transport_write(devh, buf, 1);
429 if (ret != JAYLINK_OK) {
430 log_err(ctx, "transport_write() failed: %s.",
431 jaylink_strerror(ret));
432 return ret;
435 return JAYLINK_OK;
439 * Set the target reset signal.
441 * @param[in,out] devh Device handle.
443 * @retval JAYLINK_OK Success.
444 * @retval JAYLINK_ERR_ARG Invalid arguments.
445 * @retval JAYLINK_ERR_TIMEOUT A timeout occurred.
446 * @retval JAYLINK_ERR_IO Input/output error.
447 * @retval JAYLINK_ERR Other error conditions.
449 * @since 0.1.0
451 JAYLINK_API int jaylink_set_reset(struct jaylink_device_handle *devh)
453 int ret;
454 struct jaylink_context *ctx;
455 uint8_t buf[1];
457 if (!devh)
458 return JAYLINK_ERR_ARG;
460 ctx = devh->dev->ctx;
461 ret = transport_start_write(devh, 1, true);
463 if (ret != JAYLINK_OK) {
464 log_err(ctx, "transport_start_write() failed: %s.",
465 jaylink_strerror(ret));
466 return ret;
469 buf[0] = CMD_SET_RESET;
471 ret = transport_write(devh, buf, 1);
473 if (ret != JAYLINK_OK) {
474 log_err(ctx, "transport_write() failed: %s.",
475 jaylink_strerror(ret));
476 return ret;
479 return JAYLINK_OK;
483 * Set the target power supply.
485 * If enabled, the target is supplied with 5 V from pin 19 of the 20-pin
486 * JTAG / SWD connector.
488 * @note This function must only be used if the device has the
489 * #JAYLINK_DEV_CAP_SET_TARGET_POWER capability.
491 * @param[in,out] devh Device handle.
492 * @param[in] enable Determines whether to enable or disable the target power
493 * supply.
495 * @retval JAYLINK_OK Success.
496 * @retval JAYLINK_ERR_ARG Invalid arguments.
497 * @retval JAYLINK_ERR_TIMEOUT A timeout occurred.
498 * @retval JAYLINK_ERR_IO Input/output error.
499 * @retval JAYLINK_ERR Other error conditions.
501 * @since 0.1.0
503 JAYLINK_API int jaylink_set_target_power(struct jaylink_device_handle *devh,
504 bool enable)
506 int ret;
507 struct jaylink_context *ctx;
508 uint8_t buf[2];
510 if (!devh)
511 return JAYLINK_ERR_ARG;
513 ctx = devh->dev->ctx;
514 ret = transport_start_write(devh, 2, true);
516 if (ret != JAYLINK_OK) {
517 log_err(ctx, "transport_start_wrte() failed: %s.",
518 jaylink_strerror(ret));
519 return ret;
522 buf[0] = CMD_SET_TARGET_POWER;
523 buf[1] = enable;
525 ret = transport_write(devh, buf, 2);
527 if (ret != JAYLINK_OK) {
528 log_err(ctx, "transport_write() failed: %s.",
529 jaylink_strerror(ret));
530 return ret;
533 return JAYLINK_OK;