Remove unnecessary casts
[openocd.git] / src / jtag / commands.c
1 /***************************************************************************
2 * Copyright (C) 2005 by Dominic Rath *
3 * Dominic.Rath@gmx.de *
4 * *
5 * Copyright (C) 2007,2008 Øyvind Harboe *
6 * oyvind.harboe@zylin.com *
7 * *
8 * Copyright (C) 2009 SoftPLC Corporation *
9 * http://softplc.com *
10 * dick@softplc.com *
11 * *
12 * Copyright (C) 2009 Zachary T Welch *
13 * zw@superlucidity.net *
14 * *
15 * This program is free software; you can redistribute it and/or modify *
16 * it under the terms of the GNU General Public License as published by *
17 * the Free Software Foundation; either version 2 of the License, or *
18 * (at your option) any later version. *
19 * *
20 * This program is distributed in the hope that it will be useful, *
21 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
22 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
23 * GNU General Public License for more details. *
24 * *
25 * You should have received a copy of the GNU General Public License *
26 * along with this program; if not, write to the *
27 * Free Software Foundation, Inc., *
28 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. *
29 ***************************************************************************/
30
31 #ifdef HAVE_CONFIG_H
32 #include "config.h"
33 #endif
34
35 #include <jtag/jtag.h>
36 #include "commands.h"
37
38 struct cmd_queue_page {
39 void *address;
40 size_t used;
41 struct cmd_queue_page *next;
42 };
43
44 #define CMD_QUEUE_PAGE_SIZE (1024 * 1024)
45 static struct cmd_queue_page *cmd_queue_pages;
46
47 struct jtag_command *jtag_command_queue;
48 static struct jtag_command **next_command_pointer = &jtag_command_queue;
49
50 void jtag_queue_command(struct jtag_command *cmd)
51 {
52 /* this command goes on the end, so ensure the queue terminates */
53 cmd->next = NULL;
54
55 struct jtag_command **last_cmd = next_command_pointer;
56 assert(NULL != last_cmd);
57 assert(NULL == *last_cmd);
58 *last_cmd = cmd;
59
60 /* store location where the next command pointer will be stored */
61 next_command_pointer = &cmd->next;
62 }
63
64 void *cmd_queue_alloc(size_t size)
65 {
66 struct cmd_queue_page **p_page = &cmd_queue_pages;
67 int offset;
68 uint8_t *t;
69
70 /*
71 * WARNING:
72 * We align/round the *SIZE* per below
73 * so that all pointers returned by
74 * this function are reasonably well
75 * aligned.
76 *
77 * If we did not, then an "odd-length" request would cause the
78 * *next* allocation to be at an *odd* address, and because
79 * this function has the same type of api as malloc() - we
80 * must also return pointers that have the same type of
81 * alignment.
82 *
83 * What I do not/have is a reasonable portable means
84 * to align by...
85 *
86 * The solution here, is based on these suggestions.
87 * http://gcc.gnu.org/ml/gcc-help/2008-12/msg00041.html
88 *
89 */
90 union worse_case_align {
91 int i;
92 long l;
93 float f;
94 void *v;
95 };
96 #define ALIGN_SIZE (sizeof(union worse_case_align))
97
98 /* The alignment process. */
99 size = (size + ALIGN_SIZE - 1) & (~(ALIGN_SIZE - 1));
100 /* Done... */
101
102 if (*p_page) {
103 while ((*p_page)->next)
104 p_page = &((*p_page)->next);
105 if (CMD_QUEUE_PAGE_SIZE - (*p_page)->used < size)
106 p_page = &((*p_page)->next);
107 }
108
109 if (!*p_page) {
110 *p_page = malloc(sizeof(struct cmd_queue_page));
111 (*p_page)->used = 0;
112 (*p_page)->address = malloc(CMD_QUEUE_PAGE_SIZE);
113 (*p_page)->next = NULL;
114 }
115
116 offset = (*p_page)->used;
117 (*p_page)->used += size;
118
119 t = (*p_page)->address;
120 return t + offset;
121 }
122
123 static void cmd_queue_free(void)
124 {
125 struct cmd_queue_page *page = cmd_queue_pages;
126
127 while (page) {
128 struct cmd_queue_page *last = page;
129 free(page->address);
130 page = page->next;
131 free(last);
132 }
133
134 cmd_queue_pages = NULL;
135 }
136
137 void jtag_command_queue_reset(void)
138 {
139 cmd_queue_free();
140
141 jtag_command_queue = NULL;
142 next_command_pointer = &jtag_command_queue;
143 }
144
145 enum scan_type jtag_scan_type(const struct scan_command *cmd)
146 {
147 int i;
148 int type = 0;
149
150 for (i = 0; i < cmd->num_fields; i++) {
151 if (cmd->fields[i].in_value)
152 type |= SCAN_IN;
153 if (cmd->fields[i].out_value)
154 type |= SCAN_OUT;
155 }
156
157 return type;
158 }
159
160 int jtag_scan_size(const struct scan_command *cmd)
161 {
162 int bit_count = 0;
163 int i;
164
165 /* count bits in scan command */
166 for (i = 0; i < cmd->num_fields; i++)
167 bit_count += cmd->fields[i].num_bits;
168
169 return bit_count;
170 }
171
172 int jtag_build_buffer(const struct scan_command *cmd, uint8_t **buffer)
173 {
174 int bit_count = 0;
175 int i;
176
177 bit_count = jtag_scan_size(cmd);
178 *buffer = calloc(1, DIV_ROUND_UP(bit_count, 8));
179
180 bit_count = 0;
181
182 DEBUG_JTAG_IO("%s num_fields: %i",
183 cmd->ir_scan ? "IRSCAN" : "DRSCAN",
184 cmd->num_fields);
185
186 for (i = 0; i < cmd->num_fields; i++) {
187 if (cmd->fields[i].out_value) {
188 #ifdef _DEBUG_JTAG_IO_
189 char *char_buf = buf_to_str(cmd->fields[i].out_value,
190 (cmd->fields[i].num_bits > DEBUG_JTAG_IOZ)
191 ? DEBUG_JTAG_IOZ
192 : cmd->fields[i].num_bits, 16);
193
194 LOG_DEBUG("fields[%i].out_value[%i]: 0x%s", i,
195 cmd->fields[i].num_bits, char_buf);
196 free(char_buf);
197 #endif
198 buf_set_buf(cmd->fields[i].out_value, 0, *buffer,
199 bit_count, cmd->fields[i].num_bits);
200 } else {
201 DEBUG_JTAG_IO("fields[%i].out_value[%i]: NULL",
202 i, cmd->fields[i].num_bits);
203 }
204
205 bit_count += cmd->fields[i].num_bits;
206 }
207
208 /*DEBUG_JTAG_IO("bit_count totalling: %i", bit_count); */
209
210 return bit_count;
211 }
212
213 int jtag_read_buffer(uint8_t *buffer, const struct scan_command *cmd)
214 {
215 int i;
216 int bit_count = 0;
217 int retval;
218
219 /* we return ERROR_OK, unless a check fails, or a handler reports a problem */
220 retval = ERROR_OK;
221
222 for (i = 0; i < cmd->num_fields; i++) {
223 /* if neither in_value nor in_handler
224 * are specified we don't have to examine this field
225 */
226 if (cmd->fields[i].in_value) {
227 int num_bits = cmd->fields[i].num_bits;
228 uint8_t *captured = buf_set_buf(buffer, bit_count,
229 malloc(DIV_ROUND_UP(num_bits, 8)), 0, num_bits);
230
231 #ifdef _DEBUG_JTAG_IO_
232 char *char_buf = buf_to_str(captured,
233 (num_bits > DEBUG_JTAG_IOZ)
234 ? DEBUG_JTAG_IOZ
235 : num_bits, 16);
236
237 LOG_DEBUG("fields[%i].in_value[%i]: 0x%s",
238 i, num_bits, char_buf);
239 free(char_buf);
240 #endif
241
242 if (cmd->fields[i].in_value)
243 buf_cpy(captured, cmd->fields[i].in_value, num_bits);
244
245 free(captured);
246 }
247 bit_count += cmd->fields[i].num_bits;
248 }
249
250 return retval;
251 }

Linking to existing account procedure

If you already have an account and want to add another login method you MUST first sign in with your existing account and then change URL to read https://review.openocd.org/login/?link to get to this page again but this time it'll work for linking. Thank you.

SSH host keys fingerprints

1024 SHA256:YKx8b7u5ZWdcbp7/4AeXNaqElP49m6QrwfXaqQGJAOk gerrit-code-review@openocd.zylin.com (DSA)
384 SHA256:jHIbSQa4REvwCFG4cq5LBlBLxmxSqelQPem/EXIrxjk gerrit-code-review@openocd.org (ECDSA)
521 SHA256:UAOPYkU9Fjtcao0Ul/Rrlnj/OsQvt+pgdYSZ4jOYdgs gerrit-code-review@openocd.org (ECDSA)
256 SHA256:A13M5QlnozFOvTllybRZH6vm7iSt0XLxbA48yfc2yfY gerrit-code-review@openocd.org (ECDSA)
256 SHA256:spYMBqEYoAOtK7yZBrcwE8ZpYt6b68Cfh9yEVetvbXg gerrit-code-review@openocd.org (ED25519)
+--[ED25519 256]--+
|=..              |
|+o..   .         |
|*.o   . .        |
|+B . . .         |
|Bo. = o S        |
|Oo.+ + =         |
|oB=.* = . o      |
| =+=.+   + E     |
|. .=o   . o      |
+----[SHA256]-----+
2048 SHA256:0Onrb7/PHjpo6iVZ7xQX2riKN83FJ3KGU0TvI0TaFG4 gerrit-code-review@openocd.zylin.com (RSA)