1
2 /*---------------------------------------------------------------*/
3 /*--- begin guest_ppc_helpers.c ---*/
4 /*---------------------------------------------------------------*/
5
6 /*
7 This file is part of Valgrind, a dynamic binary instrumentation
8 framework.
9
10 Copyright (C) 2004-2010 OpenWorks LLP
11 info@open-works.net
12
13 This program is free software; you can redistribute it and/or
14 modify it under the terms of the GNU General Public License as
15 published by the Free Software Foundation; either version 2 of the
16 License, or (at your option) any later version.
17
18 This program is distributed in the hope that it will be useful, but
19 WITHOUT ANY WARRANTY; without even the implied warranty of
20 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
21 General Public License for more details.
22
23 You should have received a copy of the GNU General Public License
24 along with this program; if not, write to the Free Software
25 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
26 02110-1301, USA.
27
28 The GNU General Public License is contained in the file COPYING.
29
30 Neither the names of the U.S. Department of Energy nor the
31 University of California nor the names of its contributors may be
32 used to endorse or promote products derived from this software
33 without prior written permission.
34 */
35
36 #include "libvex_basictypes.h"
37 #include "libvex_emwarn.h"
38 #include "libvex_guest_ppc32.h"
39 #include "libvex_guest_ppc64.h"
40 #include "libvex_ir.h"
41 #include "libvex.h"
42
43 #include "main_util.h"
44 #include "guest_generic_bb_to_IR.h"
45 #include "guest_ppc_defs.h"
46
47
48 /* This file contains helper functions for ppc32 and ppc64 guest code.
49 Calls to these functions are generated by the back end. These
50 calls are of course in the host machine code and this file will be
51 compiled to host machine code, so that all makes sense.
52
53 Only change the signatures of these helper functions very
54 carefully. If you change the signature here, you'll have to change
55 the parameters passed to it in the IR calls constructed by
56 guest-ppc/toIR.c.
57 */
58
59
60 /*---------------------------------------------------------------*/
61 /*--- Misc integer helpers. ---*/
62 /*---------------------------------------------------------------*/
63
64 /* CALLED FROM GENERATED CODE */
65 /* DIRTY HELPER (non-referentially-transparent) */
66 /* Horrible hack. On non-ppc platforms, return 1. */
67 /* Reads a complete, consistent 64-bit TB value. */
ppcg_dirtyhelper_MFTB(void)68 ULong ppcg_dirtyhelper_MFTB ( void )
69 {
70 # if defined(__powerpc__) || defined(_AIX)
71 ULong res;
72 UInt lo, hi1, hi2;
73 while (1) {
74 __asm__ __volatile__ ("\n"
75 "\tmftbu %0\n"
76 "\tmftb %1\n"
77 "\tmftbu %2\n"
78 : "=r" (hi1), "=r" (lo), "=r" (hi2)
79 );
80 if (hi1 == hi2) break;
81 }
82 res = ((ULong)hi1) << 32;
83 res |= (ULong)lo;
84 return res;
85 # else
86 return 1ULL;
87 # endif
88 }
89
90
91 /* CALLED FROM GENERATED CODE */
92 /* DIRTY HELPER (non-referentially transparent) */
ppc32g_dirtyhelper_MFSPR_268_269(UInt r269)93 UInt ppc32g_dirtyhelper_MFSPR_268_269 ( UInt r269 )
94 {
95 # if defined(__powerpc__) || defined(_AIX)
96 UInt spr;
97 if (r269) {
98 __asm__ __volatile__("mfspr %0,269" : "=b"(spr));
99 } else {
100 __asm__ __volatile__("mfspr %0,268" : "=b"(spr));
101 }
102 return spr;
103 # else
104 return 0;
105 # endif
106 }
107
108
109 /* CALLED FROM GENERATED CODE */
110 /* DIRTY HELPER (I'm not really sure what the side effects are) */
ppc32g_dirtyhelper_MFSPR_287(void)111 UInt ppc32g_dirtyhelper_MFSPR_287 ( void )
112 {
113 # if defined(__powerpc__) || defined(_AIX)
114 UInt spr;
115 __asm__ __volatile__("mfspr %0,287" : "=b"(spr));
116 return spr;
117 # else
118 return 0;
119 # endif
120 }
121
122
123 /* CALLED FROM GENERATED CODE */
124 /* DIRTY HELPER (reads guest state, writes guest mem) */
ppc32g_dirtyhelper_LVS(VexGuestPPC32State * gst,UInt vD_off,UInt sh,UInt shift_right)125 void ppc32g_dirtyhelper_LVS ( VexGuestPPC32State* gst,
126 UInt vD_off, UInt sh, UInt shift_right )
127 {
128 static
129 UChar ref[32] = { 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
130 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F,
131 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17,
132 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F };
133 U128* pU128_src;
134 U128* pU128_dst;
135
136 vassert( vD_off <= sizeof(VexGuestPPC32State)-8 );
137 vassert( sh <= 15 );
138 vassert( shift_right <= 1 );
139 if (shift_right)
140 sh = 16-sh;
141 /* else shift left */
142
143 pU128_src = (U128*)&ref[sh];
144 pU128_dst = (U128*)( ((UChar*)gst) + vD_off );
145
146 (*pU128_dst)[0] = (*pU128_src)[0];
147 (*pU128_dst)[1] = (*pU128_src)[1];
148 (*pU128_dst)[2] = (*pU128_src)[2];
149 (*pU128_dst)[3] = (*pU128_src)[3];
150 }
151
152 /* CALLED FROM GENERATED CODE */
153 /* DIRTY HELPER (reads guest state, writes guest mem) */
ppc64g_dirtyhelper_LVS(VexGuestPPC64State * gst,UInt vD_off,UInt sh,UInt shift_right)154 void ppc64g_dirtyhelper_LVS ( VexGuestPPC64State* gst,
155 UInt vD_off, UInt sh, UInt shift_right )
156 {
157 UChar ref[32];
158 ULong i;
159 /* ref[] used to be a static const array, but this doesn't work on
160 ppc64 because VEX doesn't load the TOC pointer for the call here,
161 and so we wind up picking up some totally random other data.
162 (It's a wonder we don't segfault.) So, just to be clear, this
163 "fix" (vex r2073) is really a kludgearound for the fact that
164 VEX's 64-bit ppc code generation doesn't provide a valid TOC
165 pointer for helper function calls. Ick. (Bug 250038) */
166 for (i = 0; i < 32; i++) ref[i] = i;
167
168 U128* pU128_src;
169 U128* pU128_dst;
170
171 vassert( vD_off <= sizeof(VexGuestPPC64State)-8 );
172 vassert( sh <= 15 );
173 vassert( shift_right <= 1 );
174 if (shift_right)
175 sh = 16-sh;
176 /* else shift left */
177
178 pU128_src = (U128*)&ref[sh];
179 pU128_dst = (U128*)( ((UChar*)gst) + vD_off );
180
181 (*pU128_dst)[0] = (*pU128_src)[0];
182 (*pU128_dst)[1] = (*pU128_src)[1];
183 (*pU128_dst)[2] = (*pU128_src)[2];
184 (*pU128_dst)[3] = (*pU128_src)[3];
185 }
186
187
188 /* Helper-function specialiser. */
189
guest_ppc32_spechelper(HChar * function_name,IRExpr ** args,IRStmt ** precedingStmts,Int n_precedingStmts)190 IRExpr* guest_ppc32_spechelper ( HChar* function_name,
191 IRExpr** args,
192 IRStmt** precedingStmts,
193 Int n_precedingStmts )
194 {
195 return NULL;
196 }
197
guest_ppc64_spechelper(HChar * function_name,IRExpr ** args,IRStmt ** precedingStmts,Int n_precedingStmts)198 IRExpr* guest_ppc64_spechelper ( HChar* function_name,
199 IRExpr** args,
200 IRStmt** precedingStmts,
201 Int n_precedingStmts )
202 {
203 return NULL;
204 }
205
206
207 /*----------------------------------------------*/
208 /*--- The exported fns .. ---*/
209 /*----------------------------------------------*/
210
211 /* VISIBLE TO LIBVEX CLIENT */
LibVEX_GuestPPC32_get_CR(VexGuestPPC32State * vex_state)212 UInt LibVEX_GuestPPC32_get_CR ( /*IN*/VexGuestPPC32State* vex_state )
213 {
214 # define FIELD(_n) \
215 ( ( (UInt) \
216 ( (vex_state->guest_CR##_n##_321 & (7<<1)) \
217 | (vex_state->guest_CR##_n##_0 & 1) \
218 ) \
219 ) \
220 << (4 * (7-(_n))) \
221 )
222
223 return
224 FIELD(0) | FIELD(1) | FIELD(2) | FIELD(3)
225 | FIELD(4) | FIELD(5) | FIELD(6) | FIELD(7);
226
227 # undef FIELD
228 }
229
230
231 /* VISIBLE TO LIBVEX CLIENT */
232 /* Note: %CR is 32 bits even for ppc64 */
LibVEX_GuestPPC64_get_CR(VexGuestPPC64State * vex_state)233 UInt LibVEX_GuestPPC64_get_CR ( /*IN*/VexGuestPPC64State* vex_state )
234 {
235 # define FIELD(_n) \
236 ( ( (UInt) \
237 ( (vex_state->guest_CR##_n##_321 & (7<<1)) \
238 | (vex_state->guest_CR##_n##_0 & 1) \
239 ) \
240 ) \
241 << (4 * (7-(_n))) \
242 )
243
244 return
245 FIELD(0) | FIELD(1) | FIELD(2) | FIELD(3)
246 | FIELD(4) | FIELD(5) | FIELD(6) | FIELD(7);
247
248 # undef FIELD
249 }
250
251
252 /* VISIBLE TO LIBVEX CLIENT */
LibVEX_GuestPPC32_put_CR(UInt cr_native,VexGuestPPC32State * vex_state)253 void LibVEX_GuestPPC32_put_CR ( UInt cr_native,
254 /*OUT*/VexGuestPPC32State* vex_state )
255 {
256 UInt t;
257
258 # define FIELD(_n) \
259 do { \
260 t = cr_native >> (4*(7-(_n))); \
261 vex_state->guest_CR##_n##_0 = toUChar(t & 1); \
262 vex_state->guest_CR##_n##_321 = toUChar(t & (7<<1)); \
263 } while (0)
264
265 FIELD(0);
266 FIELD(1);
267 FIELD(2);
268 FIELD(3);
269 FIELD(4);
270 FIELD(5);
271 FIELD(6);
272 FIELD(7);
273
274 # undef FIELD
275 }
276
277
278 /* VISIBLE TO LIBVEX CLIENT */
279 /* Note: %CR is 32 bits even for ppc64 */
LibVEX_GuestPPC64_put_CR(UInt cr_native,VexGuestPPC64State * vex_state)280 void LibVEX_GuestPPC64_put_CR ( UInt cr_native,
281 /*OUT*/VexGuestPPC64State* vex_state )
282 {
283 UInt t;
284
285 # define FIELD(_n) \
286 do { \
287 t = cr_native >> (4*(7-(_n))); \
288 vex_state->guest_CR##_n##_0 = toUChar(t & 1); \
289 vex_state->guest_CR##_n##_321 = toUChar(t & (7<<1)); \
290 } while (0)
291
292 FIELD(0);
293 FIELD(1);
294 FIELD(2);
295 FIELD(3);
296 FIELD(4);
297 FIELD(5);
298 FIELD(6);
299 FIELD(7);
300
301 # undef FIELD
302 }
303
304
305 /* VISIBLE TO LIBVEX CLIENT */
LibVEX_GuestPPC32_get_XER(VexGuestPPC32State * vex_state)306 UInt LibVEX_GuestPPC32_get_XER ( /*IN*/VexGuestPPC32State* vex_state )
307 {
308 UInt w = 0;
309 w |= ( ((UInt)vex_state->guest_XER_BC) & 0xFF );
310 w |= ( (((UInt)vex_state->guest_XER_SO) & 0x1) << 31 );
311 w |= ( (((UInt)vex_state->guest_XER_OV) & 0x1) << 30 );
312 w |= ( (((UInt)vex_state->guest_XER_CA) & 0x1) << 29 );
313 return w;
314 }
315
316
317 /* VISIBLE TO LIBVEX CLIENT */
318 /* Note: %XER is 32 bits even for ppc64 */
LibVEX_GuestPPC64_get_XER(VexGuestPPC64State * vex_state)319 UInt LibVEX_GuestPPC64_get_XER ( /*IN*/VexGuestPPC64State* vex_state )
320 {
321 UInt w = 0;
322 w |= ( ((UInt)vex_state->guest_XER_BC) & 0xFF );
323 w |= ( (((UInt)vex_state->guest_XER_SO) & 0x1) << 31 );
324 w |= ( (((UInt)vex_state->guest_XER_OV) & 0x1) << 30 );
325 w |= ( (((UInt)vex_state->guest_XER_CA) & 0x1) << 29 );
326 return w;
327 }
328
329
330 /* VISIBLE TO LIBVEX CLIENT */
LibVEX_GuestPPC32_put_XER(UInt xer_native,VexGuestPPC32State * vex_state)331 void LibVEX_GuestPPC32_put_XER ( UInt xer_native,
332 /*OUT*/VexGuestPPC32State* vex_state )
333 {
334 vex_state->guest_XER_BC = toUChar(xer_native & 0xFF);
335 vex_state->guest_XER_SO = toUChar((xer_native >> 31) & 0x1);
336 vex_state->guest_XER_OV = toUChar((xer_native >> 30) & 0x1);
337 vex_state->guest_XER_CA = toUChar((xer_native >> 29) & 0x1);
338 }
339
340 /* VISIBLE TO LIBVEX CLIENT */
341 /* Note: %XER is 32 bits even for ppc64 */
LibVEX_GuestPPC64_put_XER(UInt xer_native,VexGuestPPC64State * vex_state)342 void LibVEX_GuestPPC64_put_XER ( UInt xer_native,
343 /*OUT*/VexGuestPPC64State* vex_state )
344 {
345 vex_state->guest_XER_BC = toUChar(xer_native & 0xFF);
346 vex_state->guest_XER_SO = toUChar((xer_native >> 31) & 0x1);
347 vex_state->guest_XER_OV = toUChar((xer_native >> 30) & 0x1);
348 vex_state->guest_XER_CA = toUChar((xer_native >> 29) & 0x1);
349 }
350
351 /* VISIBLE TO LIBVEX CLIENT */
LibVEX_GuestPPC32_initialise(VexGuestPPC32State * vex_state)352 void LibVEX_GuestPPC32_initialise ( /*OUT*/VexGuestPPC32State* vex_state )
353 {
354 Int i;
355 vex_state->guest_GPR0 = 0;
356 vex_state->guest_GPR1 = 0;
357 vex_state->guest_GPR2 = 0;
358 vex_state->guest_GPR3 = 0;
359 vex_state->guest_GPR4 = 0;
360 vex_state->guest_GPR5 = 0;
361 vex_state->guest_GPR6 = 0;
362 vex_state->guest_GPR7 = 0;
363 vex_state->guest_GPR8 = 0;
364 vex_state->guest_GPR9 = 0;
365 vex_state->guest_GPR10 = 0;
366 vex_state->guest_GPR11 = 0;
367 vex_state->guest_GPR12 = 0;
368 vex_state->guest_GPR13 = 0;
369 vex_state->guest_GPR14 = 0;
370 vex_state->guest_GPR15 = 0;
371 vex_state->guest_GPR16 = 0;
372 vex_state->guest_GPR17 = 0;
373 vex_state->guest_GPR18 = 0;
374 vex_state->guest_GPR19 = 0;
375 vex_state->guest_GPR20 = 0;
376 vex_state->guest_GPR21 = 0;
377 vex_state->guest_GPR22 = 0;
378 vex_state->guest_GPR23 = 0;
379 vex_state->guest_GPR24 = 0;
380 vex_state->guest_GPR25 = 0;
381 vex_state->guest_GPR26 = 0;
382 vex_state->guest_GPR27 = 0;
383 vex_state->guest_GPR28 = 0;
384 vex_state->guest_GPR29 = 0;
385 vex_state->guest_GPR30 = 0;
386 vex_state->guest_GPR31 = 0;
387
388 vex_state->guest_FPR0 = 0;
389 vex_state->guest_FPR1 = 0;
390 vex_state->guest_FPR2 = 0;
391 vex_state->guest_FPR3 = 0;
392 vex_state->guest_FPR4 = 0;
393 vex_state->guest_FPR5 = 0;
394 vex_state->guest_FPR6 = 0;
395 vex_state->guest_FPR7 = 0;
396 vex_state->guest_FPR8 = 0;
397 vex_state->guest_FPR9 = 0;
398 vex_state->guest_FPR10 = 0;
399 vex_state->guest_FPR11 = 0;
400 vex_state->guest_FPR12 = 0;
401 vex_state->guest_FPR13 = 0;
402 vex_state->guest_FPR14 = 0;
403 vex_state->guest_FPR15 = 0;
404 vex_state->guest_FPR16 = 0;
405 vex_state->guest_FPR17 = 0;
406 vex_state->guest_FPR18 = 0;
407 vex_state->guest_FPR19 = 0;
408 vex_state->guest_FPR20 = 0;
409 vex_state->guest_FPR21 = 0;
410 vex_state->guest_FPR22 = 0;
411 vex_state->guest_FPR23 = 0;
412 vex_state->guest_FPR24 = 0;
413 vex_state->guest_FPR25 = 0;
414 vex_state->guest_FPR26 = 0;
415 vex_state->guest_FPR27 = 0;
416 vex_state->guest_FPR28 = 0;
417 vex_state->guest_FPR29 = 0;
418 vex_state->guest_FPR30 = 0;
419 vex_state->guest_FPR31 = 0;
420
421 /* Initialise the vector state. */
422 # define VECZERO(_vr) _vr[0]=_vr[1]=_vr[2]=_vr[3] = 0;
423
424 VECZERO(vex_state->guest_VR0 );
425 VECZERO(vex_state->guest_VR1 );
426 VECZERO(vex_state->guest_VR2 );
427 VECZERO(vex_state->guest_VR3 );
428 VECZERO(vex_state->guest_VR4 );
429 VECZERO(vex_state->guest_VR5 );
430 VECZERO(vex_state->guest_VR6 );
431 VECZERO(vex_state->guest_VR7 );
432 VECZERO(vex_state->guest_VR8 );
433 VECZERO(vex_state->guest_VR9 );
434 VECZERO(vex_state->guest_VR10);
435 VECZERO(vex_state->guest_VR11);
436 VECZERO(vex_state->guest_VR12);
437 VECZERO(vex_state->guest_VR13);
438 VECZERO(vex_state->guest_VR14);
439 VECZERO(vex_state->guest_VR15);
440 VECZERO(vex_state->guest_VR16);
441 VECZERO(vex_state->guest_VR17);
442 VECZERO(vex_state->guest_VR18);
443 VECZERO(vex_state->guest_VR19);
444 VECZERO(vex_state->guest_VR20);
445 VECZERO(vex_state->guest_VR21);
446 VECZERO(vex_state->guest_VR22);
447 VECZERO(vex_state->guest_VR23);
448 VECZERO(vex_state->guest_VR24);
449 VECZERO(vex_state->guest_VR25);
450 VECZERO(vex_state->guest_VR26);
451 VECZERO(vex_state->guest_VR27);
452 VECZERO(vex_state->guest_VR28);
453 VECZERO(vex_state->guest_VR29);
454 VECZERO(vex_state->guest_VR30);
455 VECZERO(vex_state->guest_VR31);
456
457 # undef VECZERO
458
459 vex_state->guest_CIA = 0;
460 vex_state->guest_LR = 0;
461 vex_state->guest_CTR = 0;
462
463 vex_state->guest_XER_SO = 0;
464 vex_state->guest_XER_OV = 0;
465 vex_state->guest_XER_CA = 0;
466 vex_state->guest_XER_BC = 0;
467
468 vex_state->guest_CR0_321 = 0;
469 vex_state->guest_CR0_0 = 0;
470 vex_state->guest_CR1_321 = 0;
471 vex_state->guest_CR1_0 = 0;
472 vex_state->guest_CR2_321 = 0;
473 vex_state->guest_CR2_0 = 0;
474 vex_state->guest_CR3_321 = 0;
475 vex_state->guest_CR3_0 = 0;
476 vex_state->guest_CR4_321 = 0;
477 vex_state->guest_CR4_0 = 0;
478 vex_state->guest_CR5_321 = 0;
479 vex_state->guest_CR5_0 = 0;
480 vex_state->guest_CR6_321 = 0;
481 vex_state->guest_CR6_0 = 0;
482 vex_state->guest_CR7_321 = 0;
483 vex_state->guest_CR7_0 = 0;
484
485 vex_state->guest_FPROUND = (UInt)PPCrm_NEAREST;
486
487 vex_state->guest_VRSAVE = 0;
488
489 vex_state->guest_VSCR = 0x0; // Non-Java mode = 0
490
491 vex_state->guest_EMWARN = EmWarn_NONE;
492
493 vex_state->guest_TISTART = 0;
494 vex_state->guest_TILEN = 0;
495
496 vex_state->guest_NRADDR = 0;
497 vex_state->guest_NRADDR_GPR2 = 0;
498
499 vex_state->guest_REDIR_SP = -1;
500 for (i = 0; i < VEX_GUEST_PPC32_REDIR_STACK_SIZE; i++)
501 vex_state->guest_REDIR_STACK[i] = 0;
502
503 vex_state->guest_IP_AT_SYSCALL = 0;
504 vex_state->guest_SPRG3_RO = 0;
505 }
506
507
508 /* VISIBLE TO LIBVEX CLIENT */
LibVEX_GuestPPC64_initialise(VexGuestPPC64State * vex_state)509 void LibVEX_GuestPPC64_initialise ( /*OUT*/VexGuestPPC64State* vex_state )
510 {
511 Int i;
512 vex_state->guest_GPR0 = 0;
513 vex_state->guest_GPR1 = 0;
514 vex_state->guest_GPR2 = 0;
515 vex_state->guest_GPR3 = 0;
516 vex_state->guest_GPR4 = 0;
517 vex_state->guest_GPR5 = 0;
518 vex_state->guest_GPR6 = 0;
519 vex_state->guest_GPR7 = 0;
520 vex_state->guest_GPR8 = 0;
521 vex_state->guest_GPR9 = 0;
522 vex_state->guest_GPR10 = 0;
523 vex_state->guest_GPR11 = 0;
524 vex_state->guest_GPR12 = 0;
525 vex_state->guest_GPR13 = 0;
526 vex_state->guest_GPR14 = 0;
527 vex_state->guest_GPR15 = 0;
528 vex_state->guest_GPR16 = 0;
529 vex_state->guest_GPR17 = 0;
530 vex_state->guest_GPR18 = 0;
531 vex_state->guest_GPR19 = 0;
532 vex_state->guest_GPR20 = 0;
533 vex_state->guest_GPR21 = 0;
534 vex_state->guest_GPR22 = 0;
535 vex_state->guest_GPR23 = 0;
536 vex_state->guest_GPR24 = 0;
537 vex_state->guest_GPR25 = 0;
538 vex_state->guest_GPR26 = 0;
539 vex_state->guest_GPR27 = 0;
540 vex_state->guest_GPR28 = 0;
541 vex_state->guest_GPR29 = 0;
542 vex_state->guest_GPR30 = 0;
543 vex_state->guest_GPR31 = 0;
544
545 vex_state->guest_FPR0 = 0;
546 vex_state->guest_FPR1 = 0;
547 vex_state->guest_FPR2 = 0;
548 vex_state->guest_FPR3 = 0;
549 vex_state->guest_FPR4 = 0;
550 vex_state->guest_FPR5 = 0;
551 vex_state->guest_FPR6 = 0;
552 vex_state->guest_FPR7 = 0;
553 vex_state->guest_FPR8 = 0;
554 vex_state->guest_FPR9 = 0;
555 vex_state->guest_FPR10 = 0;
556 vex_state->guest_FPR11 = 0;
557 vex_state->guest_FPR12 = 0;
558 vex_state->guest_FPR13 = 0;
559 vex_state->guest_FPR14 = 0;
560 vex_state->guest_FPR15 = 0;
561 vex_state->guest_FPR16 = 0;
562 vex_state->guest_FPR17 = 0;
563 vex_state->guest_FPR18 = 0;
564 vex_state->guest_FPR19 = 0;
565 vex_state->guest_FPR20 = 0;
566 vex_state->guest_FPR21 = 0;
567 vex_state->guest_FPR22 = 0;
568 vex_state->guest_FPR23 = 0;
569 vex_state->guest_FPR24 = 0;
570 vex_state->guest_FPR25 = 0;
571 vex_state->guest_FPR26 = 0;
572 vex_state->guest_FPR27 = 0;
573 vex_state->guest_FPR28 = 0;
574 vex_state->guest_FPR29 = 0;
575 vex_state->guest_FPR30 = 0;
576 vex_state->guest_FPR31 = 0;
577
578 /* Initialise the vector state. */
579 # define VECZERO(_vr) _vr[0]=_vr[1]=_vr[2]=_vr[3] = 0;
580
581 VECZERO(vex_state->guest_VR0 );
582 VECZERO(vex_state->guest_VR1 );
583 VECZERO(vex_state->guest_VR2 );
584 VECZERO(vex_state->guest_VR3 );
585 VECZERO(vex_state->guest_VR4 );
586 VECZERO(vex_state->guest_VR5 );
587 VECZERO(vex_state->guest_VR6 );
588 VECZERO(vex_state->guest_VR7 );
589 VECZERO(vex_state->guest_VR8 );
590 VECZERO(vex_state->guest_VR9 );
591 VECZERO(vex_state->guest_VR10);
592 VECZERO(vex_state->guest_VR11);
593 VECZERO(vex_state->guest_VR12);
594 VECZERO(vex_state->guest_VR13);
595 VECZERO(vex_state->guest_VR14);
596 VECZERO(vex_state->guest_VR15);
597 VECZERO(vex_state->guest_VR16);
598 VECZERO(vex_state->guest_VR17);
599 VECZERO(vex_state->guest_VR18);
600 VECZERO(vex_state->guest_VR19);
601 VECZERO(vex_state->guest_VR20);
602 VECZERO(vex_state->guest_VR21);
603 VECZERO(vex_state->guest_VR22);
604 VECZERO(vex_state->guest_VR23);
605 VECZERO(vex_state->guest_VR24);
606 VECZERO(vex_state->guest_VR25);
607 VECZERO(vex_state->guest_VR26);
608 VECZERO(vex_state->guest_VR27);
609 VECZERO(vex_state->guest_VR28);
610 VECZERO(vex_state->guest_VR29);
611 VECZERO(vex_state->guest_VR30);
612 VECZERO(vex_state->guest_VR31);
613
614 # undef VECZERO
615
616 vex_state->guest_CIA = 0;
617 vex_state->guest_LR = 0;
618 vex_state->guest_CTR = 0;
619
620 vex_state->guest_XER_SO = 0;
621 vex_state->guest_XER_OV = 0;
622 vex_state->guest_XER_CA = 0;
623 vex_state->guest_XER_BC = 0;
624
625 vex_state->guest_CR0_321 = 0;
626 vex_state->guest_CR0_0 = 0;
627 vex_state->guest_CR1_321 = 0;
628 vex_state->guest_CR1_0 = 0;
629 vex_state->guest_CR2_321 = 0;
630 vex_state->guest_CR2_0 = 0;
631 vex_state->guest_CR3_321 = 0;
632 vex_state->guest_CR3_0 = 0;
633 vex_state->guest_CR4_321 = 0;
634 vex_state->guest_CR4_0 = 0;
635 vex_state->guest_CR5_321 = 0;
636 vex_state->guest_CR5_0 = 0;
637 vex_state->guest_CR6_321 = 0;
638 vex_state->guest_CR6_0 = 0;
639 vex_state->guest_CR7_321 = 0;
640 vex_state->guest_CR7_0 = 0;
641
642 vex_state->guest_FPROUND = (UInt)PPCrm_NEAREST;
643
644 vex_state->guest_VRSAVE = 0;
645
646 vex_state->guest_VSCR = 0x0; // Non-Java mode = 0
647
648 vex_state->guest_EMWARN = EmWarn_NONE;
649
650 vex_state->padding = 0;
651
652 vex_state->guest_TISTART = 0;
653 vex_state->guest_TILEN = 0;
654
655 vex_state->guest_NRADDR = 0;
656 vex_state->guest_NRADDR_GPR2 = 0;
657
658 vex_state->guest_REDIR_SP = -1;
659 for (i = 0; i < VEX_GUEST_PPC64_REDIR_STACK_SIZE; i++)
660 vex_state->guest_REDIR_STACK[i] = 0;
661
662 vex_state->guest_IP_AT_SYSCALL = 0;
663 vex_state->guest_SPRG3_RO = 0;
664
665 vex_state->padding2 = 0;
666 }
667
668
669 /*-----------------------------------------------------------*/
670 /*--- Describing the ppc guest state, for the benefit ---*/
671 /*--- of iropt and instrumenters. ---*/
672 /*-----------------------------------------------------------*/
673
674 /* Figure out if any part of the guest state contained in minoff
675 .. maxoff requires precise memory exceptions. If in doubt return
676 True (but this is generates significantly slower code).
677
678 By default we enforce precise exns for guest R1 (stack pointer),
679 CIA (current insn address) and LR (link register). These are the
680 minimum needed to extract correct stack backtraces from ppc
681 code. [[NB: not sure if keeping LR up to date is actually
682 necessary.]]
683 */
guest_ppc32_state_requires_precise_mem_exns(Int minoff,Int maxoff)684 Bool guest_ppc32_state_requires_precise_mem_exns ( Int minoff,
685 Int maxoff )
686 {
687 Int lr_min = offsetof(VexGuestPPC32State, guest_LR);
688 Int lr_max = lr_min + 4 - 1;
689 Int r1_min = offsetof(VexGuestPPC32State, guest_GPR1);
690 Int r1_max = r1_min + 4 - 1;
691 Int cia_min = offsetof(VexGuestPPC32State, guest_CIA);
692 Int cia_max = cia_min + 4 - 1;
693
694 if (maxoff < lr_min || minoff > lr_max) {
695 /* no overlap with LR */
696 } else {
697 return True;
698 }
699
700 if (maxoff < r1_min || minoff > r1_max) {
701 /* no overlap with R1 */
702 } else {
703 return True;
704 }
705
706 if (maxoff < cia_min || minoff > cia_max) {
707 /* no overlap with CIA */
708 } else {
709 return True;
710 }
711
712 return False;
713 }
714
guest_ppc64_state_requires_precise_mem_exns(Int minoff,Int maxoff)715 Bool guest_ppc64_state_requires_precise_mem_exns ( Int minoff,
716 Int maxoff )
717 {
718 /* Given that R2 is a Big Deal in the ELF ppc64 ABI, it seems
719 prudent to be conservative with it, even though thus far there
720 is no evidence to suggest that it actually needs to be kept up
721 to date wrt possible exceptions. */
722 Int lr_min = offsetof(VexGuestPPC64State, guest_LR);
723 Int lr_max = lr_min + 8 - 1;
724 Int r1_min = offsetof(VexGuestPPC64State, guest_GPR1);
725 Int r1_max = r1_min + 8 - 1;
726 Int r2_min = offsetof(VexGuestPPC64State, guest_GPR2);
727 Int r2_max = r2_min + 8 - 1;
728 Int cia_min = offsetof(VexGuestPPC64State, guest_CIA);
729 Int cia_max = cia_min + 8 - 1;
730
731 if (maxoff < lr_min || minoff > lr_max) {
732 /* no overlap with LR */
733 } else {
734 return True;
735 }
736
737 if (maxoff < r1_min || minoff > r1_max) {
738 /* no overlap with R1 */
739 } else {
740 return True;
741 }
742
743 if (maxoff < r2_min || minoff > r2_max) {
744 /* no overlap with R2 */
745 } else {
746 return True;
747 }
748
749 if (maxoff < cia_min || minoff > cia_max) {
750 /* no overlap with CIA */
751 } else {
752 return True;
753 }
754
755 return False;
756 }
757
758
759 #define ALWAYSDEFD32(field) \
760 { offsetof(VexGuestPPC32State, field), \
761 (sizeof ((VexGuestPPC32State*)0)->field) }
762
763 VexGuestLayout
764 ppc32Guest_layout
765 = {
766 /* Total size of the guest state, in bytes. */
767 .total_sizeB = sizeof(VexGuestPPC32State),
768
769 /* Describe the stack pointer. */
770 .offset_SP = offsetof(VexGuestPPC32State,guest_GPR1),
771 .sizeof_SP = 4,
772
773 /* Describe the frame pointer. */
774 .offset_FP = offsetof(VexGuestPPC32State,guest_GPR1),
775 .sizeof_FP = 4,
776
777 /* Describe the instruction pointer. */
778 .offset_IP = offsetof(VexGuestPPC32State,guest_CIA),
779 .sizeof_IP = 4,
780
781 /* Describe any sections to be regarded by Memcheck as
782 'always-defined'. */
783 .n_alwaysDefd = 11,
784
785 .alwaysDefd
786 = { /* 0 */ ALWAYSDEFD32(guest_CIA),
787 /* 1 */ ALWAYSDEFD32(guest_EMWARN),
788 /* 2 */ ALWAYSDEFD32(guest_TISTART),
789 /* 3 */ ALWAYSDEFD32(guest_TILEN),
790 /* 4 */ ALWAYSDEFD32(guest_VSCR),
791 /* 5 */ ALWAYSDEFD32(guest_FPROUND),
792 /* 6 */ ALWAYSDEFD32(guest_NRADDR),
793 /* 7 */ ALWAYSDEFD32(guest_NRADDR_GPR2),
794 /* 8 */ ALWAYSDEFD32(guest_REDIR_SP),
795 /* 9 */ ALWAYSDEFD32(guest_REDIR_STACK),
796 /* 10 */ ALWAYSDEFD32(guest_IP_AT_SYSCALL)
797 }
798 };
799
800 #define ALWAYSDEFD64(field) \
801 { offsetof(VexGuestPPC64State, field), \
802 (sizeof ((VexGuestPPC64State*)0)->field) }
803
804 VexGuestLayout
805 ppc64Guest_layout
806 = {
807 /* Total size of the guest state, in bytes. */
808 .total_sizeB = sizeof(VexGuestPPC64State),
809
810 /* Describe the stack pointer. */
811 .offset_SP = offsetof(VexGuestPPC64State,guest_GPR1),
812 .sizeof_SP = 8,
813
814 /* Describe the frame pointer. */
815 .offset_FP = offsetof(VexGuestPPC64State,guest_GPR1),
816 .sizeof_FP = 8,
817
818 /* Describe the instruction pointer. */
819 .offset_IP = offsetof(VexGuestPPC64State,guest_CIA),
820 .sizeof_IP = 8,
821
822 /* Describe any sections to be regarded by Memcheck as
823 'always-defined'. */
824 .n_alwaysDefd = 11,
825
826 .alwaysDefd
827 = { /* 0 */ ALWAYSDEFD64(guest_CIA),
828 /* 1 */ ALWAYSDEFD64(guest_EMWARN),
829 /* 2 */ ALWAYSDEFD64(guest_TISTART),
830 /* 3 */ ALWAYSDEFD64(guest_TILEN),
831 /* 4 */ ALWAYSDEFD64(guest_VSCR),
832 /* 5 */ ALWAYSDEFD64(guest_FPROUND),
833 /* 6 */ ALWAYSDEFD64(guest_NRADDR),
834 /* 7 */ ALWAYSDEFD64(guest_NRADDR_GPR2),
835 /* 8 */ ALWAYSDEFD64(guest_REDIR_SP),
836 /* 9 */ ALWAYSDEFD64(guest_REDIR_STACK),
837 /* 10 */ ALWAYSDEFD64(guest_IP_AT_SYSCALL)
838 }
839 };
840
841 /*---------------------------------------------------------------*/
842 /*--- end guest_ppc_helpers.c ---*/
843 /*---------------------------------------------------------------*/
844