1 /* Common target dependent code for GDB on ARM systems.
3 Copyright (C) 1988-2024 Free Software Foundation, Inc.
5 This file is part of GDB.
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with this program. If not, see <http://www.gnu.org/licenses/>. */
20 #include "gdbsupport/common-regcache.h"
23 #include "../features/arm/arm-core.c"
24 #include "../features/arm/arm-tls.c"
25 #include "../features/arm/arm-vfpv2.c"
26 #include "../features/arm/arm-vfpv3.c"
27 #include "../features/arm/xscale-iwmmxt.c"
28 #include "../features/arm/arm-m-profile.c"
29 #include "../features/arm/arm-m-profile-with-fpa.c"
30 #include "../features/arm/arm-m-profile-mve.c"
31 #include "../features/arm/arm-m-system.c"
36 thumb_insn_size (unsigned short inst1
)
38 if ((inst1
& 0xe000) == 0xe000 && (inst1
& 0x1800) != 0)
47 condition_true (unsigned long cond
, unsigned long status_reg
)
49 if (cond
== INST_AL
|| cond
== INST_NV
)
55 return ((status_reg
& FLAG_Z
) != 0);
57 return ((status_reg
& FLAG_Z
) == 0);
59 return ((status_reg
& FLAG_C
) != 0);
61 return ((status_reg
& FLAG_C
) == 0);
63 return ((status_reg
& FLAG_N
) != 0);
65 return ((status_reg
& FLAG_N
) == 0);
67 return ((status_reg
& FLAG_V
) != 0);
69 return ((status_reg
& FLAG_V
) == 0);
71 return ((status_reg
& (FLAG_C
| FLAG_Z
)) == FLAG_C
);
73 return ((status_reg
& (FLAG_C
| FLAG_Z
)) != FLAG_C
);
75 return (((status_reg
& FLAG_N
) == 0) == ((status_reg
& FLAG_V
) == 0));
77 return (((status_reg
& FLAG_N
) == 0) != ((status_reg
& FLAG_V
) == 0));
79 return (((status_reg
& FLAG_Z
) == 0)
80 && (((status_reg
& FLAG_N
) == 0)
81 == ((status_reg
& FLAG_V
) == 0)));
83 return (((status_reg
& FLAG_Z
) != 0)
84 || (((status_reg
& FLAG_N
) == 0)
85 != ((status_reg
& FLAG_V
) == 0)));
94 thumb_advance_itstate (unsigned int itstate
)
96 /* Preserve IT[7:5], the first three bits of the condition. Shift
97 the upcoming condition flags left by one bit. */
98 itstate
= (itstate
& 0xe0) | ((itstate
<< 1) & 0x1f);
100 /* If we have finished the IT block, clear the state. */
101 if ((itstate
& 0x0f) == 0)
110 arm_instruction_changes_pc (uint32_t this_instr
)
112 if (bits (this_instr
, 28, 31) == INST_NV
)
113 /* Unconditional instructions. */
114 switch (bits (this_instr
, 24, 27))
118 /* Branch with Link and change to Thumb. */
123 /* Coprocessor register transfer. */
124 if (bits (this_instr
, 12, 15) == 15)
125 error (_("Invalid update to pc in instruction"));
131 switch (bits (this_instr
, 25, 27))
134 if (bits (this_instr
, 23, 24) == 2 && bit (this_instr
, 20) == 0)
136 /* Multiplies and extra load/stores. */
137 if (bit (this_instr
, 4) == 1 && bit (this_instr
, 7) == 1)
138 /* Neither multiplies nor extension load/stores are allowed
142 /* Otherwise, miscellaneous instructions. */
144 /* BX <reg>, BXJ <reg>, BLX <reg> */
145 if (bits (this_instr
, 4, 27) == 0x12fff1
146 || bits (this_instr
, 4, 27) == 0x12fff2
147 || bits (this_instr
, 4, 27) == 0x12fff3)
150 /* Other miscellaneous instructions are unpredictable if they
154 /* Data processing instruction. */
158 if (bits (this_instr
, 12, 15) == 15)
165 /* Media instructions and architecturally undefined instructions. */
166 if (bits (this_instr
, 25, 27) == 3 && bit (this_instr
, 4) == 1)
170 if (bit (this_instr
, 20) == 0)
174 if (bits (this_instr
, 12, 15) == ARM_PC_REGNUM
)
180 /* Load/store multiple. */
181 if (bit (this_instr
, 20) == 1 && bit (this_instr
, 15) == 1)
187 /* Branch and branch with link. */
192 /* Coprocessor transfers or SWIs can not affect PC. */
196 internal_error (_("bad value in switch"));
203 thumb_instruction_changes_pc (unsigned short inst
)
205 if ((inst
& 0xff00) == 0xbd00) /* pop {rlist, pc} */
208 if ((inst
& 0xf000) == 0xd000) /* conditional branch */
211 if ((inst
& 0xf800) == 0xe000) /* unconditional branch */
214 if ((inst
& 0xff00) == 0x4700) /* bx REG, blx REG */
217 if ((inst
& 0xff87) == 0x4687) /* mov pc, REG */
220 if ((inst
& 0xf500) == 0xb100) /* CBNZ or CBZ. */
230 thumb2_instruction_changes_pc (unsigned short inst1
, unsigned short inst2
)
232 if ((inst1
& 0xf800) == 0xf000 && (inst2
& 0x8000) == 0x8000)
234 /* Branches and miscellaneous control instructions. */
236 if ((inst2
& 0x1000) != 0 || (inst2
& 0xd001) == 0xc000)
241 else if (inst1
== 0xf3de && (inst2
& 0xff00) == 0x3f00)
243 /* SUBS PC, LR, #imm8. */
246 else if ((inst2
& 0xd000) == 0x8000 && (inst1
& 0x0380) != 0x0380)
248 /* Conditional branch. */
255 if ((inst1
& 0xfe50) == 0xe810)
257 /* Load multiple or RFE. */
259 if (bit (inst1
, 7) && !bit (inst1
, 8))
265 else if (!bit (inst1
, 7) && bit (inst1
, 8))
271 else if (bit (inst1
, 7) && bit (inst1
, 8))
276 else if (!bit (inst1
, 7) && !bit (inst1
, 8))
285 if ((inst1
& 0xffef) == 0xea4f && (inst2
& 0xfff0) == 0x0f00)
287 /* MOV PC or MOVS PC. */
291 if ((inst1
& 0xff70) == 0xf850 && (inst2
& 0xf000) == 0xf000)
294 if (bits (inst1
, 0, 3) == 15)
300 if ((inst2
& 0x0fc0) == 0x0000)
306 if ((inst1
& 0xfff0) == 0xe8d0 && (inst2
& 0xfff0) == 0xf000)
312 if ((inst1
& 0xfff0) == 0xe8d0 && (inst2
& 0xfff0) == 0xf010)
324 shifted_reg_val (reg_buffer_common
*regcache
, unsigned long inst
,
325 int carry
, unsigned long pc_val
, unsigned long status_reg
)
327 unsigned long res
, shift
;
328 int rm
= bits (inst
, 0, 3);
329 unsigned long shifttype
= bits (inst
, 5, 6);
333 int rs
= bits (inst
, 8, 11);
336 : regcache_raw_get_unsigned (regcache
, rs
)) & 0xFF;
339 shift
= bits (inst
, 7, 11);
341 res
= (rm
== ARM_PC_REGNUM
342 ? (pc_val
+ (bit (inst
, 4) ? 12 : 8))
343 : regcache_raw_get_unsigned (regcache
, rm
));
348 res
= shift
>= 32 ? 0 : res
<< shift
;
352 res
= shift
>= 32 ? 0 : res
>> shift
;
358 res
= ((res
& 0x80000000L
)
359 ? ~((~res
) >> shift
) : res
>> shift
);
362 case 3: /* ROR/RRX */
365 res
= (res
>> 1) | (carry
? 0x80000000L
: 0);
367 res
= (res
>> shift
) | (res
<< (32 - shift
));
371 return res
& 0xffffffff;
374 /* See arch/arm.h. */
377 arm_create_target_description (arm_fp_type fp_type
, bool tls
)
379 target_desc_up tdesc
= allocate_target_description ();
381 #ifndef IN_PROCESS_AGENT
382 if (fp_type
== ARM_FP_TYPE_IWMMXT
)
383 set_tdesc_architecture (tdesc
.get (), "iwmmxt");
385 set_tdesc_architecture (tdesc
.get (), "arm");
390 regnum
= create_feature_arm_arm_core (tdesc
.get (), regnum
);
394 case ARM_FP_TYPE_NONE
:
397 case ARM_FP_TYPE_VFPV2
:
398 regnum
= create_feature_arm_arm_vfpv2 (tdesc
.get (), regnum
);
401 case ARM_FP_TYPE_VFPV3
:
402 regnum
= create_feature_arm_arm_vfpv3 (tdesc
.get (), regnum
);
405 case ARM_FP_TYPE_IWMMXT
:
406 regnum
= create_feature_arm_xscale_iwmmxt (tdesc
.get (), regnum
);
410 error (_("Invalid Arm FP type: %d"), fp_type
);
414 regnum
= create_feature_arm_arm_tls (tdesc
.get (), regnum
);
416 return tdesc
.release ();
419 /* See arch/arm.h. */
422 arm_create_mprofile_target_description (arm_m_profile_type m_type
)
424 target_desc
*tdesc
= allocate_target_description ().release ();
426 #ifndef IN_PROCESS_AGENT
427 set_tdesc_architecture (tdesc
, "arm");
434 case ARM_M_TYPE_M_PROFILE
:
435 regnum
= create_feature_arm_arm_m_profile (tdesc
, regnum
);
438 case ARM_M_TYPE_VFP_D16
:
439 regnum
= create_feature_arm_arm_m_profile (tdesc
, regnum
);
440 regnum
= create_feature_arm_arm_vfpv2 (tdesc
, regnum
);
443 case ARM_M_TYPE_WITH_FPA
:
444 regnum
= create_feature_arm_arm_m_profile_with_fpa (tdesc
, regnum
);
448 regnum
= create_feature_arm_arm_m_profile (tdesc
, regnum
);
449 regnum
= create_feature_arm_arm_vfpv2 (tdesc
, regnum
);
450 regnum
= create_feature_arm_arm_m_profile_mve (tdesc
, regnum
);
453 case ARM_M_TYPE_SYSTEM
:
454 regnum
= create_feature_arm_arm_m_profile (tdesc
, regnum
);
455 regnum
= create_feature_arm_arm_m_system (tdesc
, regnum
);
459 error (_("Invalid Arm M type: %d"), m_type
);