Commit c5ddc558 authored by Ben Shi's avatar Ben Shi Committed by Cherry Zhang

cmd/internal/obj/arm: support more ARMv5/ARMv6/ARMv7 instructions

REV/REV16/REVSH were introduced in ARMv6, they offered more efficient
byte reverse operatons.

MMUL/MMULA/MMULS were introduced in ARMv6, they simplified
a serial of mul->shift->add/sub operations into a single instruction.

RBIT was introduced in ARMv7, it inversed a 32-bit word's bit order.

MULS was introduced in ARMv7, it corresponded to MULA.

MULBB/MULABB were introduced in ARMv5TE, they performed 16-bit
multiplication (and accumulation).

Change-Id: I6365b17b3c4eaf382a657c210bb0094b423b11b8
Reviewed-on: https://go-review.googlesource.com/35565
Run-TryBot: Cherry Zhang <cherryyz@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarCherry Zhang <cherryyz@google.com>
parent 944d56d7
...@@ -158,10 +158,10 @@ func ARMMRCOffset(op obj.As, cond string, x0, x1, x2, x3, x4, x5 int64) (offset ...@@ -158,10 +158,10 @@ func ARMMRCOffset(op obj.As, cond string, x0, x1, x2, x3, x4, x5 int64) (offset
} }
// IsARMMULA reports whether the op (as defined by an arm.A* constant) is // IsARMMULA reports whether the op (as defined by an arm.A* constant) is
// MULA, MULAWT or MULAWB, the 4-operand instructions. // MULA, MULS, MMULA, MMULS, MULABB, MULAWB or MULAWT, the 4-operand instructions.
func IsARMMULA(op obj.As) bool { func IsARMMULA(op obj.As) bool {
switch op { switch op {
case arm.AMULA, arm.AMULAWB, arm.AMULAWT: case arm.AMULA, arm.AMULS, arm.AMMULA, arm.AMMULS, arm.AMULABB, arm.AMULAWB, arm.AMULAWT:
return true return true
} }
return false return false
......
...@@ -945,6 +945,26 @@ jmp_label_3: ...@@ -945,6 +945,26 @@ jmp_label_3:
SLL R5, R7 // 1775a0e1 SLL R5, R7 // 1775a0e1
SLL.S R5, R7 // 1775b0e1 SLL.S R5, R7 // 1775b0e1
// MULA / MULS
MULAWT R1, R2, R3, R4 // c23124e1
MULAWB R1, R2, R3, R4 // 823124e1
MULS R1, R2, R3, R4 // 923164e0
MMULA R1, R2, R3, R4 // 123154e7
MMULS R1, R2, R3, R4 // d23154e7
MULABB R1, R2, R3, R4 // 823104e1
// MUL
MMUL R1, R2, R3 // 12f153e7
MULBB R1, R2, R3 // 82f163e1
MULWB R1, R2, R3 // a20123e1
MULWT R1, R2, R3 // e20123e1
// REV
REV R1, R2 // 312fbfe6
REV16 R1, R2 // b12fbfe6
REVSH R1, R2 // b12fffe6
RBIT R1, R2 // 312fffe6
// //
// END // END
// //
......
...@@ -243,6 +243,7 @@ const ( ...@@ -243,6 +243,7 @@ const (
AMULU AMULU
ADIVU ADIVU
AMUL AMUL
AMMUL
ADIV ADIV
AMOD AMOD
AMODU AMODU
...@@ -261,6 +262,9 @@ const ( ...@@ -261,6 +262,9 @@ const (
ARFE ARFE
ASWI ASWI
AMULA AMULA
AMULS
AMMULA
AMMULS
AWORD AWORD
...@@ -281,11 +285,17 @@ const ( ...@@ -281,11 +285,17 @@ const (
APLD APLD
ACLZ ACLZ
AREV
AREV16
AREVSH
ARBIT
AMULWT AMULWT
AMULWB AMULWB
AMULBB
AMULAWT AMULAWT
AMULAWB AMULAWB
AMULABB
ADATABUNDLE ADATABUNDLE
ADATABUNDLEEND ADATABUNDLEEND
......
...@@ -67,6 +67,7 @@ var Anames = []string{ ...@@ -67,6 +67,7 @@ var Anames = []string{
"MULU", "MULU",
"DIVU", "DIVU",
"MUL", "MUL",
"MMUL",
"DIV", "DIV",
"MOD", "MOD",
"MODU", "MODU",
...@@ -83,6 +84,9 @@ var Anames = []string{ ...@@ -83,6 +84,9 @@ var Anames = []string{
"RFE", "RFE",
"SWI", "SWI",
"MULA", "MULA",
"MULS",
"MMULA",
"MMULS",
"WORD", "WORD",
"MULL", "MULL",
"MULAL", "MULAL",
...@@ -97,10 +101,16 @@ var Anames = []string{ ...@@ -97,10 +101,16 @@ var Anames = []string{
"STREXD", "STREXD",
"PLD", "PLD",
"CLZ", "CLZ",
"REV",
"REV16",
"REVSH",
"RBIT",
"MULWT", "MULWT",
"MULWB", "MULWB",
"MULBB",
"MULAWT", "MULAWT",
"MULAWB", "MULAWB",
"MULABB",
"DATABUNDLE", "DATABUNDLE",
"DATABUNDLEEND", "DATABUNDLEEND",
"MRC", "MRC",
......
...@@ -1440,9 +1440,21 @@ func buildop(ctxt *obj.Link) { ...@@ -1440,9 +1440,21 @@ func buildop(ctxt *obj.Link) {
case AMULWT: case AMULWT:
opset(AMULWB, r0) opset(AMULWB, r0)
opset(AMULBB, r0)
opset(AMMUL, r0)
case AMULAWT: case AMULAWT:
opset(AMULAWB, r0) opset(AMULAWB, r0)
opset(AMULABB, r0)
opset(AMULS, r0)
opset(AMMULA, r0)
opset(AMMULS, r0)
case ACLZ:
opset(AREV, r0)
opset(AREV16, r0)
opset(AREVSH, r0)
opset(ARBIT, r0)
case AMULA, case AMULA,
ALDREX, ALDREX,
...@@ -1452,7 +1464,6 @@ func buildop(ctxt *obj.Link) { ...@@ -1452,7 +1464,6 @@ func buildop(ctxt *obj.Link) {
ATST, ATST,
APLD, APLD,
obj.AUNDEF, obj.AUNDEF,
ACLZ,
obj.AFUNCDATA, obj.AFUNCDATA,
obj.APCDATA, obj.APCDATA,
obj.ANOP, obj.ANOP,
...@@ -2413,6 +2424,14 @@ func oprrr(ctxt *obj.Link, p *obj.Prog, a obj.As, sc int) uint32 { ...@@ -2413,6 +2424,14 @@ func oprrr(ctxt *obj.Link, p *obj.Prog, a obj.As, sc int) uint32 {
ctxt.Diag(".nil/.W on dp instruction") ctxt.Diag(".nil/.W on dp instruction")
} }
switch a { switch a {
case AMMUL:
return o | 0x75<<20 | 0xf<<12 | 0x1<<4
case AMULS:
return o | 0x6<<20 | 0x9<<4
case AMMULA:
return o | 0x75<<20 | 0x1<<4
case AMMULS:
return o | 0x75<<20 | 0xd<<4
case AMULU, AMUL: case AMULU, AMUL:
return o | 0x0<<21 | 0x9<<4 return o | 0x0<<21 | 0x9<<4
case AMULA: case AMULA:
...@@ -2547,18 +2566,36 @@ func oprrr(ctxt *obj.Link, p *obj.Prog, a obj.As, sc int) uint32 { ...@@ -2547,18 +2566,36 @@ func oprrr(ctxt *obj.Link, p *obj.Prog, a obj.As, sc int) uint32 {
case ACLZ: case ACLZ:
return o&(0xf<<28) | 0x16f<<16 | 0xf1<<4 return o&(0xf<<28) | 0x16f<<16 | 0xf1<<4
case AREV:
return o&(0xf<<28) | 0x6bf<<16 | 0xf3<<4
case AREV16:
return o&(0xf<<28) | 0x6bf<<16 | 0xfb<<4
case AREVSH:
return o&(0xf<<28) | 0x6ff<<16 | 0xfb<<4
case ARBIT:
return o&(0xf<<28) | 0x6ff<<16 | 0xf3<<4
case AMULWT: case AMULWT:
return o&(0xf<<28) | 0x12<<20 | 0xe<<4 return o&(0xf<<28) | 0x12<<20 | 0xe<<4
case AMULWB: case AMULWB:
return o&(0xf<<28) | 0x12<<20 | 0xa<<4 return o&(0xf<<28) | 0x12<<20 | 0xa<<4
case AMULBB:
return o&(0xf<<28) | 0x16<<20 | 0xf<<12 | 0x8<<4
case AMULAWT: case AMULAWT:
return o&(0xf<<28) | 0x12<<20 | 0xc<<4 return o&(0xf<<28) | 0x12<<20 | 0xc<<4
case AMULAWB: case AMULAWB:
return o&(0xf<<28) | 0x12<<20 | 0x8<<4 return o&(0xf<<28) | 0x12<<20 | 0x8<<4
case AMULABB:
return o&(0xf<<28) | 0x10<<20 | 0x8<<4
case ABL: // BLX REG case ABL: // BLX REG
return o&(0xf<<28) | 0x12fff3<<4 return o&(0xf<<28) | 0x12fff3<<4
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment