]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blob - gdb/gdbserver/regcache.c
Move savestring to common/common-utils.c, make gdbserver use it.
[thirdparty/binutils-gdb.git] / gdb / gdbserver / regcache.c
1 /* Register support routines for the remote server for GDB.
2 Copyright (C) 2001-2013 Free Software Foundation, Inc.
3
4 This file is part of GDB.
5
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 3 of the License, or
9 (at your option) any later version.
10
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.
15
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/>. */
18
19 #include "server.h"
20 #include "regdef.h"
21 #include "gdbthread.h"
22
23 #include <stdlib.h>
24 #include <string.h>
25
26 static int register_bytes;
27
28 static struct reg *reg_defs;
29 static int num_registers;
30
31 const char **gdbserver_expedite_regs;
32
33 #ifndef IN_PROCESS_AGENT
34
35 struct regcache *
36 get_thread_regcache (struct thread_info *thread, int fetch)
37 {
38 struct regcache *regcache;
39
40 regcache = (struct regcache *) inferior_regcache_data (thread);
41
42 if (regcache == NULL)
43 fatal ("no register cache");
44
45 if (fetch && regcache->registers_valid == 0)
46 {
47 struct thread_info *saved_inferior = current_inferior;
48
49 current_inferior = thread;
50 fetch_inferior_registers (regcache, -1);
51 current_inferior = saved_inferior;
52 regcache->registers_valid = 1;
53 }
54
55 return regcache;
56 }
57
58 void
59 regcache_invalidate_one (struct inferior_list_entry *entry)
60 {
61 struct thread_info *thread = (struct thread_info *) entry;
62 struct regcache *regcache;
63
64 regcache = (struct regcache *) inferior_regcache_data (thread);
65
66 if (regcache == NULL)
67 return;
68
69 if (regcache->registers_valid)
70 {
71 struct thread_info *saved_inferior = current_inferior;
72
73 current_inferior = thread;
74 store_inferior_registers (regcache, -1);
75 current_inferior = saved_inferior;
76 }
77
78 regcache->registers_valid = 0;
79 }
80
81 void
82 regcache_invalidate (void)
83 {
84 for_each_inferior (&all_threads, regcache_invalidate_one);
85 }
86
87 #endif
88
89 struct regcache *
90 init_register_cache (struct regcache *regcache, unsigned char *regbuf)
91 {
92 #ifndef IN_PROCESS_AGENT
93 if (regbuf == NULL)
94 {
95 /* Make sure to zero-initialize the register cache when it is
96 created, in case there are registers the target never
97 fetches. This way they'll read as zero instead of
98 garbage. */
99 regcache->registers = xcalloc (1, register_bytes);
100 regcache->registers_owned = 1;
101 regcache->register_status = xcalloc (1, num_registers);
102 gdb_assert (REG_UNAVAILABLE == 0);
103 }
104 else
105 #else
106 if (regbuf == NULL)
107 fatal ("init_register_cache: can't allocate memory from the heap");
108 else
109 #endif
110 {
111 regcache->registers = regbuf;
112 regcache->registers_owned = 0;
113 #ifndef IN_PROCESS_AGENT
114 regcache->register_status = NULL;
115 #endif
116 }
117
118 regcache->registers_valid = 0;
119
120 return regcache;
121 }
122
123 #ifndef IN_PROCESS_AGENT
124
125 struct regcache *
126 new_register_cache (void)
127 {
128 struct regcache *regcache;
129
130 if (register_bytes == 0)
131 return NULL; /* The architecture hasn't been initialized yet. */
132
133 regcache = xmalloc (sizeof (*regcache));
134 return init_register_cache (regcache, NULL);
135 }
136
137 void
138 free_register_cache (struct regcache *regcache)
139 {
140 if (regcache)
141 {
142 if (regcache->registers_owned)
143 free (regcache->registers);
144 free (regcache->register_status);
145 free (regcache);
146 }
147 }
148
149 #endif
150
151 void
152 regcache_cpy (struct regcache *dst, struct regcache *src)
153 {
154 memcpy (dst->registers, src->registers, register_bytes);
155 #ifndef IN_PROCESS_AGENT
156 if (dst->register_status != NULL && src->register_status != NULL)
157 memcpy (dst->register_status, src->register_status, num_registers);
158 #endif
159 dst->registers_valid = src->registers_valid;
160 }
161
162 #ifndef IN_PROCESS_AGENT
163 static void
164 realloc_register_cache (struct inferior_list_entry *thread_p)
165 {
166 struct thread_info *thread = (struct thread_info *) thread_p;
167 struct regcache *regcache
168 = (struct regcache *) inferior_regcache_data (thread);
169
170 if (regcache != NULL)
171 regcache_invalidate_one (thread_p);
172 free_register_cache (regcache);
173 set_inferior_regcache_data (thread, new_register_cache ());
174 }
175 #endif
176
177 void
178 set_register_cache (struct reg *regs, int n)
179 {
180 int offset, i;
181
182 #ifndef IN_PROCESS_AGENT
183 /* Before changing the register cache internal layout, flush the
184 contents of valid caches back to the threads. */
185 regcache_invalidate ();
186 #endif
187
188 reg_defs = regs;
189 num_registers = n;
190
191 offset = 0;
192 for (i = 0; i < n; i++)
193 {
194 regs[i].offset = offset;
195 offset += regs[i].size;
196 }
197
198 register_bytes = offset / 8;
199
200 /* Make sure PBUFSIZ is large enough to hold a full register packet. */
201 if (2 * register_bytes + 32 > PBUFSIZ)
202 fatal ("Register packet size exceeds PBUFSIZ.");
203
204 #ifndef IN_PROCESS_AGENT
205 /* Re-allocate all pre-existing register caches. */
206 for_each_inferior (&all_threads, realloc_register_cache);
207 #endif
208 }
209
210 int
211 register_cache_size (void)
212 {
213 return register_bytes;
214 }
215
216 #ifndef IN_PROCESS_AGENT
217
218 void
219 registers_to_string (struct regcache *regcache, char *buf)
220 {
221 unsigned char *registers = regcache->registers;
222 int i;
223
224 for (i = 0; i < num_registers; i++)
225 {
226 if (regcache->register_status[i] == REG_VALID)
227 {
228 convert_int_to_ascii (registers, buf, register_size (i));
229 buf += register_size (i) * 2;
230 }
231 else
232 {
233 memset (buf, 'x', register_size (i) * 2);
234 buf += register_size (i) * 2;
235 }
236 registers += register_size (i);
237 }
238 *buf = '\0';
239 }
240
241 void
242 registers_from_string (struct regcache *regcache, char *buf)
243 {
244 int len = strlen (buf);
245 unsigned char *registers = regcache->registers;
246
247 if (len != register_bytes * 2)
248 {
249 warning ("Wrong sized register packet (expected %d bytes, got %d)",
250 2*register_bytes, len);
251 if (len > register_bytes * 2)
252 len = register_bytes * 2;
253 }
254 convert_ascii_to_int (buf, registers, len / 2);
255 }
256
257 struct reg *
258 find_register_by_name (const char *name)
259 {
260 int i;
261
262 for (i = 0; i < num_registers; i++)
263 if (!strcmp (name, reg_defs[i].name))
264 return &reg_defs[i];
265 fatal ("Unknown register %s requested", name);
266 return 0;
267 }
268
269 int
270 find_regno (const char *name)
271 {
272 int i;
273
274 for (i = 0; i < num_registers; i++)
275 if (!strcmp (name, reg_defs[i].name))
276 return i;
277 fatal ("Unknown register %s requested", name);
278 return -1;
279 }
280
281 struct reg *
282 find_register_by_number (int n)
283 {
284 return &reg_defs[n];
285 }
286
287 #endif
288
289 int
290 register_size (int n)
291 {
292 return reg_defs[n].size / 8;
293 }
294
295 static unsigned char *
296 register_data (struct regcache *regcache, int n, int fetch)
297 {
298 return regcache->registers + (reg_defs[n].offset / 8);
299 }
300
301 /* Supply register N, whose contents are stored in BUF, to REGCACHE.
302 If BUF is NULL, the register's value is recorded as
303 unavailable. */
304
305 void
306 supply_register (struct regcache *regcache, int n, const void *buf)
307 {
308 if (buf)
309 {
310 memcpy (register_data (regcache, n, 0), buf, register_size (n));
311 #ifndef IN_PROCESS_AGENT
312 if (regcache->register_status != NULL)
313 regcache->register_status[n] = REG_VALID;
314 #endif
315 }
316 else
317 {
318 memset (register_data (regcache, n, 0), 0, register_size (n));
319 #ifndef IN_PROCESS_AGENT
320 if (regcache->register_status != NULL)
321 regcache->register_status[n] = REG_UNAVAILABLE;
322 #endif
323 }
324 }
325
326 /* Supply register N with value zero to REGCACHE. */
327
328 void
329 supply_register_zeroed (struct regcache *regcache, int n)
330 {
331 memset (register_data (regcache, n, 0), 0, register_size (n));
332 #ifndef IN_PROCESS_AGENT
333 if (regcache->register_status != NULL)
334 regcache->register_status[n] = REG_VALID;
335 #endif
336 }
337
338 /* Supply the whole register set whose contents are stored in BUF, to
339 REGCACHE. If BUF is NULL, all the registers' values are recorded
340 as unavailable. */
341
342 void
343 supply_regblock (struct regcache *regcache, const void *buf)
344 {
345 if (buf)
346 {
347 memcpy (regcache->registers, buf, register_bytes);
348 #ifndef IN_PROCESS_AGENT
349 {
350 int i;
351
352 for (i = 0; i < num_registers; i++)
353 regcache->register_status[i] = REG_VALID;
354 }
355 #endif
356 }
357 else
358 {
359 memset (regcache->registers, 0, register_bytes);
360 #ifndef IN_PROCESS_AGENT
361 {
362 int i;
363
364 for (i = 0; i < num_registers; i++)
365 regcache->register_status[i] = REG_UNAVAILABLE;
366 }
367 #endif
368 }
369 }
370
371 #ifndef IN_PROCESS_AGENT
372
373 void
374 supply_register_by_name (struct regcache *regcache,
375 const char *name, const void *buf)
376 {
377 supply_register (regcache, find_regno (name), buf);
378 }
379
380 #endif
381
382 void
383 collect_register (struct regcache *regcache, int n, void *buf)
384 {
385 memcpy (buf, register_data (regcache, n, 1), register_size (n));
386 }
387
388 #ifndef IN_PROCESS_AGENT
389
390 void
391 collect_register_as_string (struct regcache *regcache, int n, char *buf)
392 {
393 convert_int_to_ascii (register_data (regcache, n, 1),
394 buf, register_size (n));
395 }
396
397 void
398 collect_register_by_name (struct regcache *regcache,
399 const char *name, void *buf)
400 {
401 collect_register (regcache, find_regno (name), buf);
402 }
403
404 /* Special handling for register PC. */
405
406 CORE_ADDR
407 regcache_read_pc (struct regcache *regcache)
408 {
409 CORE_ADDR pc_val;
410
411 if (the_target->read_pc)
412 pc_val = the_target->read_pc (regcache);
413 else
414 internal_error (__FILE__, __LINE__,
415 "regcache_read_pc: Unable to find PC");
416
417 return pc_val;
418 }
419
420 void
421 regcache_write_pc (struct regcache *regcache, CORE_ADDR pc)
422 {
423 if (the_target->write_pc)
424 the_target->write_pc (regcache, pc);
425 else
426 internal_error (__FILE__, __LINE__,
427 "regcache_write_pc: Unable to update PC");
428 }
429
430 #endif