x86-64: improve handling of branches to absolute addresses
authorJan Beulich <jbeulich@suse.com>
Mon, 4 Jul 2022 06:31:21 +0000 (08:31 +0200)
committerJan Beulich <jbeulich@suse.com>
Mon, 4 Jul 2022 06:31:21 +0000 (08:31 +0200)
There are two related problems here: The use of "addr32" on a direct
branch would, besides causing a warning, result in operands to be
permitted which mistakenly are refused without "addr32". Plus at some
point not too long ago I'm afraid it may have been me who regressed the
relocation addends emitted for such branches. Correct both problems,
adding a testcase to guard against regressing this again.

gas/config/tc-i386.c
gas/testsuite/gas/i386/i386.exp
gas/testsuite/gas/i386/x86-64-branch-6.d [new file with mode: 0644]
gas/testsuite/gas/i386/x86-64-branch-6.e [new file with mode: 0644]
gas/testsuite/gas/i386/x86-64-branch-6.s [new file with mode: 0644]

index 5d486570bf8c998592a09a786e238fb70195a70b..8b245cc4d34edb0e205120a103611fc91506db75 100644 (file)
@@ -4975,7 +4975,9 @@ md_assemble (char *line)
   if (i.imm_operands)
     optimize_imm ();
 
-  if (i.disp_operands && !want_disp32 (current_templates->start))
+  if (i.disp_operands && !want_disp32 (current_templates->start)
+      && (!current_templates->start->opcode_modifier.jump
+         || i.jumpabsolute || i.types[0].bitfield.baseindex))
     {
       for (j = 0; j < i.operands; ++j)
        {
@@ -5985,7 +5987,9 @@ optimize_disp (void)
            /* Optimize 64-bit displacement to 32-bit for 64-bit BFD.  */
            if ((i.types[op].bitfield.disp32
                 || (flag_code == CODE_64BIT
-                    && want_disp32 (current_templates->start)))
+                    && want_disp32 (current_templates->start)
+                    && (!current_templates->start->opcode_modifier.jump
+                        || i.jumpabsolute || i.types[op].bitfield.baseindex)))
                && fits_in_unsigned_long (op_disp))
              {
                /* If this operand is at most 32 bits, convert
index a4fc5b4096bcafb2e95e368e17ccff253fe8da82..ebb1b49246239c6c9830bc7ca078afc3231a2d17 100644 (file)
@@ -1314,6 +1314,7 @@ if [gas_64_check] then {
        run_dump_test "x86-64-branch-3"
        run_list_test "x86-64-branch-4" "-al -mintel64"
        run_list_test "x86-64-branch-5" "-al"
+       run_dump_test "x86-64-branch-6"
 
        run_dump_test "x86-64-rip-2"
 
diff --git a/gas/testsuite/gas/i386/x86-64-branch-6.d b/gas/testsuite/gas/i386/x86-64-branch-6.d
new file mode 100644 (file)
index 0000000..a2d388f
--- /dev/null
@@ -0,0 +1,21 @@
+#objdump: -r
+#name: x86-64 branch 6
+#warning_output: x86-64-branch-6.e
+
+.*: +file format .*
+
+RELOCATION RECORDS FOR \[\.text\]:
+OFFSET +TYPE +VALUE *
+0+01 R_X86_64_PC32 +\*ABS\*\+0x000000008765431d
+0+11 R_X86_64_PC32 +\*ABS\*\+0x000000087654320c
+0+21 R_X86_64_PC32 +\*ABS\*\+0x000000008765431d
+0+31 R_X86_64_PC32 +\*ABS\*\+0x000000087654320c
+0+07 R_X86_64_PC32 +\*ABS\*\+0x000000008765431d
+0+0c R_X86_64_PC32 +\*ABS\*\+0x000000008765431d
+0+17 R_X86_64_PC32 +\*ABS\*\+0x000000087654320c
+0+1c R_X86_64_PC32 +\*ABS\*\+0x000000087654320c
+0+27 R_X86_64_PC32 +\*ABS\*\+0x000000008765431d
+0+2c R_X86_64_PC32 +\*ABS\*\+0x000000008765431d
+0+37 R_X86_64_PC32 +\*ABS\*\+0x000000087654320c
+0+3c R_X86_64_PC32 +\*ABS\*\+0x000000087654320c
+#pass
diff --git a/gas/testsuite/gas/i386/x86-64-branch-6.e b/gas/testsuite/gas/i386/x86-64-branch-6.e
new file mode 100644 (file)
index 0000000..1dc58ea
--- /dev/null
@@ -0,0 +1,7 @@
+.*: Assembler messages:
+.*:12: Warning: skipping prefixes on `call'
+.*:13: Warning: skipping prefixes on `je'
+.*:14: Warning: skipping prefixes on `jmp'
+.*:16: Warning: skipping prefixes on `call'
+.*:17: Warning: skipping prefixes on `je'
+.*:18: Warning: skipping prefixes on `jmp'
diff --git a/gas/testsuite/gas/i386/x86-64-branch-6.s b/gas/testsuite/gas/i386/x86-64-branch-6.s
new file mode 100644 (file)
index 0000000..a5b108e
--- /dev/null
@@ -0,0 +1,18 @@
+       .text
+
+branch_6:
+       call    0x87654321
+       je      0x87654321
+       jmp     0x87654321
+
+       call    0x876543210
+       je      0x876543210
+       jmp     0x876543210
+
+       addr32 call     0x87654321
+       addr32 je       0x87654321
+       addr32 jmp      0x87654321
+
+       addr32 call     0x876543210
+       addr32 je       0x876543210
+       addr32 jmp      0x876543210