1
0
mirror of https://github.com/golang/go synced 2024-11-18 10:14:45 -07:00

cmd/asm: improve argument to obj.Prog assignment on PPC64

These can be simplified with the knowledge of how arguments are
assigned to obj.Prog objects on PPC64. If the argument is not
a register type, the Reg argument (a2 in optab) of obj.Prog is
not used, and those arguments are placed into RestArgs (a3, a4, a5
in optab).

This relaxes the special case handling enforced by IsPPC64RLD and
IsPPC64ISEL. Instead, arguments are assigned as noted above, and
incorrect usage of such opcodes is checked by optab rules, not by
the assembler front-end.

Likewise, add support for handling 6 argument opcodes, these do
not exist today, but will be added with ISA 3.1 (Power10).

Finally, to maintain backwards compatibility, some 4-arg opcodes
whose middle arguments are a register and immediate, could swap
these arguments and generate identical machine code. This likely
wasn't intended, but is possible. These are explicitly fixed up
in the backend, and the asm tests are extended to check these.

Change-Id: I5f8190212427dfe8e6f062185bfefb5fa4fd0e75
Reviewed-on: https://go-review.googlesource.com/c/go/+/427516
Reviewed-by: Cherry Mui <cherryyz@google.com>
Run-TryBot: Paul Murphy <murp@ibm.com>
Reviewed-by: Michael Knyszek <mknyszek@google.com>
TryBot-Result: Gopher Robot <gobot@golang.org>
Reviewed-by: Lynn Boger <laboger@linux.vnet.ibm.com>
This commit is contained in:
Paul E. Murphy 2022-09-01 09:23:49 -05:00 committed by Paul Murphy
parent 85c0d26c6a
commit a50c434308
4 changed files with 45 additions and 67 deletions

View File

@ -21,28 +21,6 @@ func jumpPPC64(word string) bool {
return false return false
} }
// IsPPC64RLD reports whether the op (as defined by an ppc64.A* constant) is
// one of the RLD-like instructions that require special handling.
// The FMADD-like instructions behave similarly.
func IsPPC64RLD(op obj.As) bool {
switch op {
case ppc64.ARLDC, ppc64.ARLDCCC, ppc64.ARLDCL, ppc64.ARLDCLCC,
ppc64.ARLDCR, ppc64.ARLDCRCC, ppc64.ARLDMI, ppc64.ARLDMICC,
ppc64.ARLWMI, ppc64.ARLWMICC, ppc64.ARLWNM, ppc64.ARLWNMCC:
return true
case ppc64.AFMADD, ppc64.AFMADDCC, ppc64.AFMADDS, ppc64.AFMADDSCC,
ppc64.AFMSUB, ppc64.AFMSUBCC, ppc64.AFMSUBS, ppc64.AFMSUBSCC,
ppc64.AFNMADD, ppc64.AFNMADDCC, ppc64.AFNMADDS, ppc64.AFNMADDSCC,
ppc64.AFNMSUB, ppc64.AFNMSUBCC, ppc64.AFNMSUBS, ppc64.AFNMSUBSCC:
return true
}
return false
}
func IsPPC64ISEL(op obj.As) bool {
return op == ppc64.AISEL
}
// IsPPC64CMP reports whether the op (as defined by an ppc64.A* constant) is // IsPPC64CMP reports whether the op (as defined by an ppc64.A* constant) is
// one of the CMP instructions that require special handling. // one of the CMP instructions that require special handling.
func IsPPC64CMP(op obj.As) bool { func IsPPC64CMP(op obj.As) bool {

View File

@ -727,23 +727,17 @@ func (p *Parser) asmInstruction(op obj.As, cond string, a []obj.Addr) {
prog.To = a[1] prog.To = a[1]
break break
} }
// Arithmetic. Choices are:
// reg reg reg prog.From = a[0]
// imm reg reg prog.To = a[2]
// reg imm reg
// If the immediate is the middle argument, use From3. // If the second argument is not a register argument, it must be
// passed RestArgs/SetFrom3
switch a[1].Type { switch a[1].Type {
case obj.TYPE_REG: case obj.TYPE_REG:
prog.From = a[0]
prog.Reg = p.getRegister(prog, op, &a[1]) prog.Reg = p.getRegister(prog, op, &a[1])
prog.To = a[2]
case obj.TYPE_CONST:
prog.From = a[0]
prog.SetFrom3(a[1])
prog.To = a[2]
default: default:
p.errorf("invalid addressing modes for %s instruction", op) prog.SetFrom3(a[1])
return
} }
case sys.RISCV64: case sys.RISCV64:
// RISCV64 instructions with one input and two outputs. // RISCV64 instructions with one input and two outputs.
@ -810,41 +804,18 @@ func (p *Parser) asmInstruction(op obj.As, cond string, a []obj.Addr) {
break break
} }
if p.arch.Family == sys.PPC64 { if p.arch.Family == sys.PPC64 {
if arch.IsPPC64RLD(op) { prog.From = a[0]
prog.From = a[0] prog.To = a[3]
prog.Reg = p.getRegister(prog, op, &a[1]) // If the second argument is not a register argument, it must be
prog.SetFrom3(a[2]) // passed RestArgs/SetFrom3
prog.To = a[3]
break
} else if arch.IsPPC64ISEL(op) {
// ISEL BC,RB,RA,RT becomes isel rt,ra,rb,bc
prog.SetFrom3(a[2]) // ra
prog.From = a[0] // bc
prog.Reg = p.getRegister(prog, op, &a[1]) // rb
prog.To = a[3] // rt
break
}
// Else, it is a VA-form instruction
// reg reg reg reg
// imm reg reg reg
// Or a VX-form instruction
// imm imm reg reg
if a[1].Type == obj.TYPE_REG { if a[1].Type == obj.TYPE_REG {
prog.From = a[0]
prog.Reg = p.getRegister(prog, op, &a[1]) prog.Reg = p.getRegister(prog, op, &a[1])
prog.SetFrom3(a[2]) prog.SetRestArgs([]obj.Addr{a[2]})
prog.To = a[3]
break
} else if a[1].Type == obj.TYPE_CONST {
prog.From = a[0]
prog.Reg = p.getRegister(prog, op, &a[2])
prog.SetFrom3(a[1])
prog.To = a[3]
break
} else { } else {
p.errorf("invalid addressing modes for %s instruction", op) // Don't set prog.Reg if a1 isn't a reg arg.
return prog.SetRestArgs([]obj.Addr{a[1], a[2]})
} }
break
} }
if p.arch.Family == sys.RISCV64 { if p.arch.Family == sys.RISCV64 {
prog.From = a[0] prog.From = a[0]
@ -909,6 +880,14 @@ func (p *Parser) asmInstruction(op obj.As, cond string, a []obj.Addr) {
prog.As = MRC // Both instructions are coded as MRC. prog.As = MRC // Both instructions are coded as MRC.
break break
} }
if p.arch.Family == sys.PPC64 {
prog.From = a[0]
// Second arg is always a register type on ppc64.
prog.Reg = p.getRegister(prog, op, &a[1])
prog.SetRestArgs([]obj.Addr{a[2], a[3], a[4]})
prog.To = a[5]
break
}
fallthrough fallthrough
default: default:
p.errorf("can't handle %s instruction with %d operands", op, len(a)) p.errorf("can't handle %s instruction with %d operands", op, len(a))

View File

@ -162,6 +162,7 @@ TEXT asmtest(SB),DUPOK|NOSPLIT,$0
ADD $1234567, R5 // 641f001263ffd6877cbf2a14 ADD $1234567, R5 // 641f001263ffd6877cbf2a14
ADD $1234567, R5, R6 // 641f001263ffd6877cdf2a14 ADD $1234567, R5, R6 // 641f001263ffd6877cdf2a14
ADDEX R3, R5, $3, R6 // 7cc32f54 ADDEX R3, R5, $3, R6 // 7cc32f54
ADDEX R3, $3, R5, R6 // 7cc32f54
ADDIS $8, R3 // 3c630008 ADDIS $8, R3 // 3c630008
ADDIS $1000, R3, R4 // 3c8303e8 ADDIS $1000, R3, R4 // 3c8303e8
@ -784,7 +785,9 @@ TEXT asmtest(SB),DUPOK|NOSPLIT,$0
VNCIPHERLAST V1, V2, V3 // 10611549 VNCIPHERLAST V1, V2, V3 // 10611549
VSBOX V1, V2 // 104105c8 VSBOX V1, V2 // 104105c8
VSHASIGMAW $1, V1, $15, V2 // 10418e82 VSHASIGMAW $1, V1, $15, V2 // 10418e82
VSHASIGMAW $1, $15, V1, V2 // 10418e82
VSHASIGMAD $2, V1, $15, V2 // 104196c2 VSHASIGMAD $2, V1, $15, V2 // 104196c2
VSHASIGMAD $2, $15, V1, V2 // 104196c2
LXVD2X (R3)(R4), VS1 // 7c241e98 LXVD2X (R3)(R4), VS1 // 7c241e98
LXVD2X (R3)(R0), VS1 // 7c201e98 LXVD2X (R3)(R0), VS1 // 7c201e98
@ -876,7 +879,11 @@ TEXT asmtest(SB),DUPOK|NOSPLIT,$0
XXPERM VS1, VS2, VS3 // f06110d0 XXPERM VS1, VS2, VS3 // f06110d0
XXSLDWI VS1, VS2, $1, VS3 // f0611110 XXSLDWI VS1, VS2, $1, VS3 // f0611110
XXSLDWI V1, V2, $1, V3 // f0611117 XXSLDWI V1, V2, $1, V3 // f0611117
XXSLDWI V1, $1, V2, V3 // f0611117
XXSLDWI VS33, VS34, $1, VS35 // f0611117 XXSLDWI VS33, VS34, $1, VS35 // f0611117
XXSLDWI VS33, $1, VS34, VS35 // f0611117
XXPERMDI VS33, VS34, $1, VS35 // f0611157
XXPERMDI VS33, $1, VS34, VS35 // f0611157
XSCVDPSP VS1, VS2 // f0400c24 XSCVDPSP VS1, VS2 // f0400c24
XVCVDPSP VS1, VS2 // f0400e24 XVCVDPSP VS1, VS2 // f0400e24
XSCVSXDDP VS1, VS2 // f0400de0 XSCVSXDDP VS1, VS2 // f0400de0

View File

@ -88,8 +88,8 @@ func progedit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
} }
} }
// Rewrite SUB constants into ADD.
switch p.As { switch p.As {
// Rewrite SUB constants into ADD.
case ASUBC: case ASUBC:
if p.From.Type == obj.TYPE_CONST { if p.From.Type == obj.TYPE_CONST {
p.From.Offset = -p.From.Offset p.From.Offset = -p.From.Offset
@ -107,7 +107,21 @@ func progedit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
p.From.Offset = -p.From.Offset p.From.Offset = -p.From.Offset
p.As = AADD p.As = AADD
} }
// To maintain backwards compatibility, we accept some 4 argument usage of
// several opcodes which was likely not intended, but did work. These are not
// added to optab to avoid the chance this behavior might be used with newer
// instructions.
//
// Rewrite argument ordering like "ADDEX R3, $3, R4, R5" into
// "ADDEX R3, R4, $3, R5"
case AVSHASIGMAW, AVSHASIGMAD, AADDEX, AXXSLDWI, AXXPERMDI:
if len(p.RestArgs) == 2 && p.Reg == 0 && p.RestArgs[0].Addr.Type == obj.TYPE_CONST && p.RestArgs[1].Addr.Type == obj.TYPE_REG {
p.Reg = p.RestArgs[1].Addr.Reg
p.RestArgs = p.RestArgs[:1]
}
} }
if c.ctxt.Headtype == objabi.Haix { if c.ctxt.Headtype == objabi.Haix {
c.rewriteToUseTOC(p) c.rewriteToUseTOC(p)
} else if c.ctxt.Flag_dynlink { } else if c.ctxt.Flag_dynlink {