mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
cmd/internal/obj/arm64: make assembler almost concurrency-safe
CL 39922 made the arm assembler concurrency-safe. This CL does the same, but for arm64. The approach is similar: introduce ctxt7 to hold function-local state and thread it through the assembler as necessary. One race remains after this CL, deep in aclass, in the check that a Prog does not take the address of a TLS variable. That race is conceptually unrelated to this refactoring, and will be addressed in a separate CL. Passes toolstash-check -all. Updates #15756 Change-Id: Icab1ef70008468f9a5b8bf728a77c4520bbcb67d Reviewed-on: https://go-review.googlesource.com/40252 Run-TryBot: Josh Bleecher Snyder <josharian@gmail.com> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Cherry Zhang <cherryyz@google.com>
This commit is contained in:
parent
c4135d61bb
commit
b5931020b7
3 changed files with 457 additions and 444 deletions
File diff suppressed because it is too large
Load diff
|
|
@ -47,16 +47,16 @@ var complements = []obj.As{
|
||||||
ACMNW: ACMPW,
|
ACMNW: ACMPW,
|
||||||
}
|
}
|
||||||
|
|
||||||
func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize int32) *obj.Prog {
|
func (c *ctxt7) stacksplit(p *obj.Prog, framesize int32) *obj.Prog {
|
||||||
// MOV g_stackguard(g), R1
|
// MOV g_stackguard(g), R1
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
|
|
||||||
p.As = AMOVD
|
p.As = AMOVD
|
||||||
p.From.Type = obj.TYPE_MEM
|
p.From.Type = obj.TYPE_MEM
|
||||||
p.From.Reg = REGG
|
p.From.Reg = REGG
|
||||||
p.From.Offset = 2 * int64(ctxt.Arch.PtrSize) // G.stackguard0
|
p.From.Offset = 2 * int64(c.ctxt.Arch.PtrSize) // G.stackguard0
|
||||||
if ctxt.Cursym.CFunc() {
|
if c.cursym.CFunc() {
|
||||||
p.From.Offset = 3 * int64(ctxt.Arch.PtrSize) // G.stackguard1
|
p.From.Offset = 3 * int64(c.ctxt.Arch.PtrSize) // G.stackguard1
|
||||||
}
|
}
|
||||||
p.To.Type = obj.TYPE_REG
|
p.To.Type = obj.TYPE_REG
|
||||||
p.To.Reg = REG_R1
|
p.To.Reg = REG_R1
|
||||||
|
|
@ -66,7 +66,7 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
// small stack: SP < stackguard
|
// small stack: SP < stackguard
|
||||||
// MOV SP, R2
|
// MOV SP, R2
|
||||||
// CMP stackguard, R2
|
// CMP stackguard, R2
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
|
|
||||||
p.As = AMOVD
|
p.As = AMOVD
|
||||||
p.From.Type = obj.TYPE_REG
|
p.From.Type = obj.TYPE_REG
|
||||||
|
|
@ -74,7 +74,7 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
p.To.Type = obj.TYPE_REG
|
p.To.Type = obj.TYPE_REG
|
||||||
p.To.Reg = REG_R2
|
p.To.Reg = REG_R2
|
||||||
|
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
p.As = ACMP
|
p.As = ACMP
|
||||||
p.From.Type = obj.TYPE_REG
|
p.From.Type = obj.TYPE_REG
|
||||||
p.From.Reg = REG_R1
|
p.From.Reg = REG_R1
|
||||||
|
|
@ -83,7 +83,7 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
// large stack: SP-framesize < stackguard-StackSmall
|
// large stack: SP-framesize < stackguard-StackSmall
|
||||||
// SUB $(framesize-StackSmall), SP, R2
|
// SUB $(framesize-StackSmall), SP, R2
|
||||||
// CMP stackguard, R2
|
// CMP stackguard, R2
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
|
|
||||||
p.As = ASUB
|
p.As = ASUB
|
||||||
p.From.Type = obj.TYPE_CONST
|
p.From.Type = obj.TYPE_CONST
|
||||||
|
|
@ -92,7 +92,7 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
p.To.Type = obj.TYPE_REG
|
p.To.Type = obj.TYPE_REG
|
||||||
p.To.Reg = REG_R2
|
p.To.Reg = REG_R2
|
||||||
|
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
p.As = ACMP
|
p.As = ACMP
|
||||||
p.From.Type = obj.TYPE_REG
|
p.From.Type = obj.TYPE_REG
|
||||||
p.From.Reg = REG_R1
|
p.From.Reg = REG_R1
|
||||||
|
|
@ -109,19 +109,19 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
// SUB R1, R2
|
// SUB R1, R2
|
||||||
// MOV $(framesize+(StackGuard-StackSmall)), R3
|
// MOV $(framesize+(StackGuard-StackSmall)), R3
|
||||||
// CMP R3, R2
|
// CMP R3, R2
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
|
|
||||||
p.As = ACMP
|
p.As = ACMP
|
||||||
p.From.Type = obj.TYPE_CONST
|
p.From.Type = obj.TYPE_CONST
|
||||||
p.From.Offset = obj.StackPreempt
|
p.From.Offset = obj.StackPreempt
|
||||||
p.Reg = REG_R1
|
p.Reg = REG_R1
|
||||||
|
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
q = p
|
q = p
|
||||||
p.As = ABEQ
|
p.As = ABEQ
|
||||||
p.To.Type = obj.TYPE_BRANCH
|
p.To.Type = obj.TYPE_BRANCH
|
||||||
|
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
p.As = AADD
|
p.As = AADD
|
||||||
p.From.Type = obj.TYPE_CONST
|
p.From.Type = obj.TYPE_CONST
|
||||||
p.From.Offset = obj.StackGuard
|
p.From.Offset = obj.StackGuard
|
||||||
|
|
@ -129,21 +129,21 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
p.To.Type = obj.TYPE_REG
|
p.To.Type = obj.TYPE_REG
|
||||||
p.To.Reg = REG_R2
|
p.To.Reg = REG_R2
|
||||||
|
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
p.As = ASUB
|
p.As = ASUB
|
||||||
p.From.Type = obj.TYPE_REG
|
p.From.Type = obj.TYPE_REG
|
||||||
p.From.Reg = REG_R1
|
p.From.Reg = REG_R1
|
||||||
p.To.Type = obj.TYPE_REG
|
p.To.Type = obj.TYPE_REG
|
||||||
p.To.Reg = REG_R2
|
p.To.Reg = REG_R2
|
||||||
|
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
p.As = AMOVD
|
p.As = AMOVD
|
||||||
p.From.Type = obj.TYPE_CONST
|
p.From.Type = obj.TYPE_CONST
|
||||||
p.From.Offset = int64(framesize) + (obj.StackGuard - obj.StackSmall)
|
p.From.Offset = int64(framesize) + (obj.StackGuard - obj.StackSmall)
|
||||||
p.To.Type = obj.TYPE_REG
|
p.To.Type = obj.TYPE_REG
|
||||||
p.To.Reg = REG_R3
|
p.To.Reg = REG_R3
|
||||||
|
|
||||||
p = obj.Appendp(p, newprog)
|
p = obj.Appendp(p, c.newprog)
|
||||||
p.As = ACMP
|
p.As = ACMP
|
||||||
p.From.Type = obj.TYPE_REG
|
p.From.Type = obj.TYPE_REG
|
||||||
p.From.Reg = REG_R3
|
p.From.Reg = REG_R3
|
||||||
|
|
@ -151,23 +151,23 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
}
|
}
|
||||||
|
|
||||||
// BLS do-morestack
|
// BLS do-morestack
|
||||||
bls := obj.Appendp(p, newprog)
|
bls := obj.Appendp(p, c.newprog)
|
||||||
bls.As = ABLS
|
bls.As = ABLS
|
||||||
bls.To.Type = obj.TYPE_BRANCH
|
bls.To.Type = obj.TYPE_BRANCH
|
||||||
|
|
||||||
var last *obj.Prog
|
var last *obj.Prog
|
||||||
for last = ctxt.Cursym.Text; last.Link != nil; last = last.Link {
|
for last = c.cursym.Text; last.Link != nil; last = last.Link {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now we are at the end of the function, but logically
|
// Now we are at the end of the function, but logically
|
||||||
// we are still in function prologue. We need to fix the
|
// we are still in function prologue. We need to fix the
|
||||||
// SP data and PCDATA.
|
// SP data and PCDATA.
|
||||||
spfix := obj.Appendp(last, newprog)
|
spfix := obj.Appendp(last, c.newprog)
|
||||||
spfix.As = obj.ANOP
|
spfix.As = obj.ANOP
|
||||||
spfix.Spadj = -framesize
|
spfix.Spadj = -framesize
|
||||||
|
|
||||||
pcdata := obj.Appendp(spfix, newprog)
|
pcdata := obj.Appendp(spfix, c.newprog)
|
||||||
pcdata.Pos = ctxt.Cursym.Text.Pos
|
pcdata.Pos = c.cursym.Text.Pos
|
||||||
pcdata.As = obj.APCDATA
|
pcdata.As = obj.APCDATA
|
||||||
pcdata.From.Type = obj.TYPE_CONST
|
pcdata.From.Type = obj.TYPE_CONST
|
||||||
pcdata.From.Offset = obj.PCDATA_StackMapIndex
|
pcdata.From.Offset = obj.PCDATA_StackMapIndex
|
||||||
|
|
@ -175,7 +175,7 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
pcdata.To.Offset = -1 // pcdata starts at -1 at function entry
|
pcdata.To.Offset = -1 // pcdata starts at -1 at function entry
|
||||||
|
|
||||||
// MOV LR, R3
|
// MOV LR, R3
|
||||||
movlr := obj.Appendp(pcdata, newprog)
|
movlr := obj.Appendp(pcdata, c.newprog)
|
||||||
movlr.As = AMOVD
|
movlr.As = AMOVD
|
||||||
movlr.From.Type = obj.TYPE_REG
|
movlr.From.Type = obj.TYPE_REG
|
||||||
movlr.From.Reg = REGLINK
|
movlr.From.Reg = REGLINK
|
||||||
|
|
@ -188,7 +188,7 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
|
|
||||||
debug := movlr
|
debug := movlr
|
||||||
if false {
|
if false {
|
||||||
debug = obj.Appendp(debug, newprog)
|
debug = obj.Appendp(debug, c.newprog)
|
||||||
debug.As = AMOVD
|
debug.As = AMOVD
|
||||||
debug.From.Type = obj.TYPE_CONST
|
debug.From.Type = obj.TYPE_CONST
|
||||||
debug.From.Offset = int64(framesize)
|
debug.From.Offset = int64(framesize)
|
||||||
|
|
@ -197,23 +197,23 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
}
|
}
|
||||||
|
|
||||||
// BL runtime.morestack(SB)
|
// BL runtime.morestack(SB)
|
||||||
call := obj.Appendp(debug, newprog)
|
call := obj.Appendp(debug, c.newprog)
|
||||||
call.As = ABL
|
call.As = ABL
|
||||||
call.To.Type = obj.TYPE_BRANCH
|
call.To.Type = obj.TYPE_BRANCH
|
||||||
morestack := "runtime.morestack"
|
morestack := "runtime.morestack"
|
||||||
switch {
|
switch {
|
||||||
case ctxt.Cursym.CFunc():
|
case c.cursym.CFunc():
|
||||||
morestack = "runtime.morestackc"
|
morestack = "runtime.morestackc"
|
||||||
case ctxt.Cursym.Text.From3.Offset&obj.NEEDCTXT == 0:
|
case c.cursym.Text.From3.Offset&obj.NEEDCTXT == 0:
|
||||||
morestack = "runtime.morestack_noctxt"
|
morestack = "runtime.morestack_noctxt"
|
||||||
}
|
}
|
||||||
call.To.Sym = ctxt.Lookup(morestack, 0)
|
call.To.Sym = c.ctxt.Lookup(morestack, 0)
|
||||||
|
|
||||||
// B start
|
// B start
|
||||||
jmp := obj.Appendp(call, newprog)
|
jmp := obj.Appendp(call, c.newprog)
|
||||||
jmp.As = AB
|
jmp.As = AB
|
||||||
jmp.To.Type = obj.TYPE_BRANCH
|
jmp.To.Type = obj.TYPE_BRANCH
|
||||||
jmp.Pcond = ctxt.Cursym.Text.Link
|
jmp.Pcond = c.cursym.Text.Link
|
||||||
jmp.Spadj = +framesize
|
jmp.Spadj = +framesize
|
||||||
|
|
||||||
// placeholder for bls's jump target
|
// placeholder for bls's jump target
|
||||||
|
|
@ -224,6 +224,8 @@ func stacksplit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc, framesize in
|
||||||
}
|
}
|
||||||
|
|
||||||
func progedit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
func progedit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
|
c := ctxt7{ctxt: ctxt, newprog: newprog}
|
||||||
|
|
||||||
p.From.Class = 0
|
p.From.Class = 0
|
||||||
p.To.Class = 0
|
p.To.Class = 0
|
||||||
|
|
||||||
|
|
@ -264,7 +266,7 @@ func progedit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
p.From.Type = obj.TYPE_MEM
|
p.From.Type = obj.TYPE_MEM
|
||||||
p.From.Sym = ctxt.Float32Sym(f32)
|
p.From.Sym = c.ctxt.Float32Sym(f32)
|
||||||
p.From.Name = obj.NAME_EXTERN
|
p.From.Name = obj.NAME_EXTERN
|
||||||
p.From.Offset = 0
|
p.From.Offset = 0
|
||||||
}
|
}
|
||||||
|
|
@ -278,7 +280,7 @@ func progedit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
p.From.Type = obj.TYPE_MEM
|
p.From.Type = obj.TYPE_MEM
|
||||||
p.From.Sym = ctxt.Float64Sym(f64)
|
p.From.Sym = c.ctxt.Float64Sym(f64)
|
||||||
p.From.Name = obj.NAME_EXTERN
|
p.From.Name = obj.NAME_EXTERN
|
||||||
p.From.Offset = 0
|
p.From.Offset = 0
|
||||||
}
|
}
|
||||||
|
|
@ -315,13 +317,13 @@ func progedit(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ctxt.Flag_dynlink {
|
if c.ctxt.Flag_dynlink {
|
||||||
rewriteToUseGot(ctxt, p, newprog)
|
c.rewriteToUseGot(p)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Rewrite p, if necessary, to access global data via the global offset table.
|
// Rewrite p, if necessary, to access global data via the global offset table.
|
||||||
func rewriteToUseGot(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
func (c *ctxt7) rewriteToUseGot(p *obj.Prog) {
|
||||||
if p.As == obj.ADUFFCOPY || p.As == obj.ADUFFZERO {
|
if p.As == obj.ADUFFCOPY || p.As == obj.ADUFFZERO {
|
||||||
// ADUFFxxx $offset
|
// ADUFFxxx $offset
|
||||||
// becomes
|
// becomes
|
||||||
|
|
@ -330,9 +332,9 @@ func rewriteToUseGot(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
// CALL REGTMP
|
// CALL REGTMP
|
||||||
var sym *obj.LSym
|
var sym *obj.LSym
|
||||||
if p.As == obj.ADUFFZERO {
|
if p.As == obj.ADUFFZERO {
|
||||||
sym = ctxt.Lookup("runtime.duffzero", 0)
|
sym = c.ctxt.Lookup("runtime.duffzero", 0)
|
||||||
} else {
|
} else {
|
||||||
sym = ctxt.Lookup("runtime.duffcopy", 0)
|
sym = c.ctxt.Lookup("runtime.duffcopy", 0)
|
||||||
}
|
}
|
||||||
offset := p.To.Offset
|
offset := p.To.Offset
|
||||||
p.As = AMOVD
|
p.As = AMOVD
|
||||||
|
|
@ -344,13 +346,13 @@ func rewriteToUseGot(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
p.To.Name = obj.NAME_NONE
|
p.To.Name = obj.NAME_NONE
|
||||||
p.To.Offset = 0
|
p.To.Offset = 0
|
||||||
p.To.Sym = nil
|
p.To.Sym = nil
|
||||||
p1 := obj.Appendp(p, newprog)
|
p1 := obj.Appendp(p, c.newprog)
|
||||||
p1.As = AADD
|
p1.As = AADD
|
||||||
p1.From.Type = obj.TYPE_CONST
|
p1.From.Type = obj.TYPE_CONST
|
||||||
p1.From.Offset = offset
|
p1.From.Offset = offset
|
||||||
p1.To.Type = obj.TYPE_REG
|
p1.To.Type = obj.TYPE_REG
|
||||||
p1.To.Reg = REGTMP
|
p1.To.Reg = REGTMP
|
||||||
p2 := obj.Appendp(p1, newprog)
|
p2 := obj.Appendp(p1, c.newprog)
|
||||||
p2.As = obj.ACALL
|
p2.As = obj.ACALL
|
||||||
p2.To.Type = obj.TYPE_REG
|
p2.To.Type = obj.TYPE_REG
|
||||||
p2.To.Reg = REGTMP
|
p2.To.Reg = REGTMP
|
||||||
|
|
@ -363,15 +365,15 @@ func rewriteToUseGot(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
// MOVD $sym, Rx becomes MOVD sym@GOT, Rx
|
// MOVD $sym, Rx becomes MOVD sym@GOT, Rx
|
||||||
// MOVD $sym+<off>, Rx becomes MOVD sym@GOT, Rx; ADD <off>, Rx
|
// MOVD $sym+<off>, Rx becomes MOVD sym@GOT, Rx; ADD <off>, Rx
|
||||||
if p.As != AMOVD {
|
if p.As != AMOVD {
|
||||||
ctxt.Diag("do not know how to handle TYPE_ADDR in %v with -dynlink", p)
|
c.ctxt.Diag("do not know how to handle TYPE_ADDR in %v with -dynlink", p)
|
||||||
}
|
}
|
||||||
if p.To.Type != obj.TYPE_REG {
|
if p.To.Type != obj.TYPE_REG {
|
||||||
ctxt.Diag("do not know how to handle LEAQ-type insn to non-register in %v with -dynlink", p)
|
c.ctxt.Diag("do not know how to handle LEAQ-type insn to non-register in %v with -dynlink", p)
|
||||||
}
|
}
|
||||||
p.From.Type = obj.TYPE_MEM
|
p.From.Type = obj.TYPE_MEM
|
||||||
p.From.Name = obj.NAME_GOTREF
|
p.From.Name = obj.NAME_GOTREF
|
||||||
if p.From.Offset != 0 {
|
if p.From.Offset != 0 {
|
||||||
q := obj.Appendp(p, newprog)
|
q := obj.Appendp(p, c.newprog)
|
||||||
q.As = AADD
|
q.As = AADD
|
||||||
q.From.Type = obj.TYPE_CONST
|
q.From.Type = obj.TYPE_CONST
|
||||||
q.From.Offset = p.From.Offset
|
q.From.Offset = p.From.Offset
|
||||||
|
|
@ -380,7 +382,7 @@ func rewriteToUseGot(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if p.From3 != nil && p.From3.Name == obj.NAME_EXTERN {
|
if p.From3 != nil && p.From3.Name == obj.NAME_EXTERN {
|
||||||
ctxt.Diag("don't know how to handle %v with -dynlink", p)
|
c.ctxt.Diag("don't know how to handle %v with -dynlink", p)
|
||||||
}
|
}
|
||||||
var source *obj.Addr
|
var source *obj.Addr
|
||||||
// MOVx sym, Ry becomes MOVD sym@GOT, REGTMP; MOVx (REGTMP), Ry
|
// MOVx sym, Ry becomes MOVD sym@GOT, REGTMP; MOVx (REGTMP), Ry
|
||||||
|
|
@ -388,7 +390,7 @@ func rewriteToUseGot(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
// An addition may be inserted between the two MOVs if there is an offset.
|
// An addition may be inserted between the two MOVs if there is an offset.
|
||||||
if p.From.Name == obj.NAME_EXTERN && !p.From.Sym.Local() {
|
if p.From.Name == obj.NAME_EXTERN && !p.From.Sym.Local() {
|
||||||
if p.To.Name == obj.NAME_EXTERN && !p.To.Sym.Local() {
|
if p.To.Name == obj.NAME_EXTERN && !p.To.Sym.Local() {
|
||||||
ctxt.Diag("cannot handle NAME_EXTERN on both sides in %v with -dynlink", p)
|
c.ctxt.Diag("cannot handle NAME_EXTERN on both sides in %v with -dynlink", p)
|
||||||
}
|
}
|
||||||
source = &p.From
|
source = &p.From
|
||||||
} else if p.To.Name == obj.NAME_EXTERN && !p.To.Sym.Local() {
|
} else if p.To.Name == obj.NAME_EXTERN && !p.To.Sym.Local() {
|
||||||
|
|
@ -403,10 +405,10 @@ func rewriteToUseGot(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if source.Type != obj.TYPE_MEM {
|
if source.Type != obj.TYPE_MEM {
|
||||||
ctxt.Diag("don't know how to handle %v with -dynlink", p)
|
c.ctxt.Diag("don't know how to handle %v with -dynlink", p)
|
||||||
}
|
}
|
||||||
p1 := obj.Appendp(p, newprog)
|
p1 := obj.Appendp(p, c.newprog)
|
||||||
p2 := obj.Appendp(p1, newprog)
|
p2 := obj.Appendp(p1, c.newprog)
|
||||||
p1.As = AMOVD
|
p1.As = AMOVD
|
||||||
p1.From.Type = obj.TYPE_MEM
|
p1.From.Type = obj.TYPE_MEM
|
||||||
p1.From.Sym = source.Sym
|
p1.From.Sym = source.Sym
|
||||||
|
|
@ -432,18 +434,18 @@ func rewriteToUseGot(ctxt *obj.Link, p *obj.Prog, newprog obj.ProgAlloc) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
ctxt.Cursym = cursym
|
|
||||||
|
|
||||||
if cursym.Text == nil || cursym.Text.Link == nil {
|
if cursym.Text == nil || cursym.Text.Link == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
p := cursym.Text
|
c := ctxt7{ctxt: ctxt, newprog: newprog, cursym: cursym}
|
||||||
|
|
||||||
|
p := c.cursym.Text
|
||||||
textstksiz := p.To.Offset
|
textstksiz := p.To.Offset
|
||||||
aoffset := int32(textstksiz)
|
aoffset := int32(textstksiz)
|
||||||
|
|
||||||
cursym.Args = p.To.Val.(int32)
|
c.cursym.Args = p.To.Val.(int32)
|
||||||
cursym.Locals = int32(textstksiz)
|
c.cursym.Locals = int32(textstksiz)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* find leaf subroutines
|
* find leaf subroutines
|
||||||
|
|
@ -452,7 +454,7 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
*/
|
*/
|
||||||
q := (*obj.Prog)(nil)
|
q := (*obj.Prog)(nil)
|
||||||
var q1 *obj.Prog
|
var q1 *obj.Prog
|
||||||
for p := cursym.Text; p != nil; p = p.Link {
|
for p := c.cursym.Text; p != nil; p = p.Link {
|
||||||
switch p.As {
|
switch p.As {
|
||||||
case obj.ATEXT:
|
case obj.ATEXT:
|
||||||
p.Mark |= LEAF
|
p.Mark |= LEAF
|
||||||
|
|
@ -469,7 +471,7 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
case ABL,
|
case ABL,
|
||||||
obj.ADUFFZERO,
|
obj.ADUFFZERO,
|
||||||
obj.ADUFFCOPY:
|
obj.ADUFFCOPY:
|
||||||
cursym.Text.Mark &^= LEAF
|
c.cursym.Text.Mark &^= LEAF
|
||||||
fallthrough
|
fallthrough
|
||||||
|
|
||||||
case ACBNZ,
|
case ACBNZ,
|
||||||
|
|
@ -514,19 +516,19 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
|
|
||||||
var q2 *obj.Prog
|
var q2 *obj.Prog
|
||||||
var retjmp *obj.LSym
|
var retjmp *obj.LSym
|
||||||
for p := cursym.Text; p != nil; p = p.Link {
|
for p := c.cursym.Text; p != nil; p = p.Link {
|
||||||
o := p.As
|
o := p.As
|
||||||
switch o {
|
switch o {
|
||||||
case obj.ATEXT:
|
case obj.ATEXT:
|
||||||
cursym.Text = p
|
c.cursym.Text = p
|
||||||
if textstksiz < 0 {
|
if textstksiz < 0 {
|
||||||
ctxt.Autosize = 0
|
c.autosize = 0
|
||||||
} else {
|
} else {
|
||||||
ctxt.Autosize = int32(textstksiz + 8)
|
c.autosize = int32(textstksiz + 8)
|
||||||
}
|
}
|
||||||
if (cursym.Text.Mark&LEAF != 0) && ctxt.Autosize <= 8 {
|
if (c.cursym.Text.Mark&LEAF != 0) && c.autosize <= 8 {
|
||||||
ctxt.Autosize = 0
|
c.autosize = 0
|
||||||
} else if ctxt.Autosize&(16-1) != 0 {
|
} else if c.autosize&(16-1) != 0 {
|
||||||
// The frame includes an LR.
|
// The frame includes an LR.
|
||||||
// If the frame size is 8, it's only an LR,
|
// If the frame size is 8, it's only an LR,
|
||||||
// so there's no potential for breaking references to
|
// so there's no potential for breaking references to
|
||||||
|
|
@ -535,32 +537,32 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
// But otherwise, if there is a non-empty locals section,
|
// But otherwise, if there is a non-empty locals section,
|
||||||
// the author of the code is responsible for making sure
|
// the author of the code is responsible for making sure
|
||||||
// that the frame size is 8 mod 16.
|
// that the frame size is 8 mod 16.
|
||||||
if ctxt.Autosize == 8 {
|
if c.autosize == 8 {
|
||||||
ctxt.Autosize += 8
|
c.autosize += 8
|
||||||
cursym.Locals += 8
|
c.cursym.Locals += 8
|
||||||
} else {
|
} else {
|
||||||
ctxt.Diag("%v: unaligned frame size %d - must be 8 mod 16 (or 0)", p, ctxt.Autosize-8)
|
c.ctxt.Diag("%v: unaligned frame size %d - must be 8 mod 16 (or 0)", p, c.autosize-8)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
p.To.Offset = int64(ctxt.Autosize) - 8
|
p.To.Offset = int64(c.autosize) - 8
|
||||||
if ctxt.Autosize == 0 && !(cursym.Text.Mark&LEAF != 0) {
|
if c.autosize == 0 && !(c.cursym.Text.Mark&LEAF != 0) {
|
||||||
if ctxt.Debugvlog {
|
if c.ctxt.Debugvlog {
|
||||||
ctxt.Logf("save suppressed in: %s\n", cursym.Text.From.Sym.Name)
|
c.ctxt.Logf("save suppressed in: %s\n", c.cursym.Text.From.Sym.Name)
|
||||||
}
|
}
|
||||||
cursym.Text.Mark |= LEAF
|
c.cursym.Text.Mark |= LEAF
|
||||||
}
|
}
|
||||||
|
|
||||||
if !(p.From3.Offset&obj.NOSPLIT != 0) {
|
if !(p.From3.Offset&obj.NOSPLIT != 0) {
|
||||||
p = stacksplit(ctxt, p, newprog, ctxt.Autosize) // emit split check
|
p = c.stacksplit(p, c.autosize) // emit split check
|
||||||
}
|
}
|
||||||
|
|
||||||
aoffset = ctxt.Autosize
|
aoffset = c.autosize
|
||||||
if aoffset > 0xF0 {
|
if aoffset > 0xF0 {
|
||||||
aoffset = 0xF0
|
aoffset = 0xF0
|
||||||
}
|
}
|
||||||
if cursym.Text.Mark&LEAF != 0 {
|
if c.cursym.Text.Mark&LEAF != 0 {
|
||||||
cursym.Set(obj.AttrLeaf, true)
|
c.cursym.Set(obj.AttrLeaf, true)
|
||||||
if ctxt.Autosize == 0 {
|
if c.autosize == 0 {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -568,21 +570,21 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
// Frame is non-empty. Make sure to save link register, even if
|
// Frame is non-empty. Make sure to save link register, even if
|
||||||
// it is a leaf function, so that traceback works.
|
// it is a leaf function, so that traceback works.
|
||||||
q = p
|
q = p
|
||||||
if ctxt.Autosize > aoffset {
|
if c.autosize > aoffset {
|
||||||
// Frame size is too large for a MOVD.W instruction.
|
// Frame size is too large for a MOVD.W instruction.
|
||||||
// Store link register before decrementing SP, so if a signal comes
|
// Store link register before decrementing SP, so if a signal comes
|
||||||
// during the execution of the function prologue, the traceback
|
// during the execution of the function prologue, the traceback
|
||||||
// code will not see a half-updated stack frame.
|
// code will not see a half-updated stack frame.
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.Pos = p.Pos
|
q.Pos = p.Pos
|
||||||
q.As = ASUB
|
q.As = ASUB
|
||||||
q.From.Type = obj.TYPE_CONST
|
q.From.Type = obj.TYPE_CONST
|
||||||
q.From.Offset = int64(ctxt.Autosize)
|
q.From.Offset = int64(c.autosize)
|
||||||
q.Reg = REGSP
|
q.Reg = REGSP
|
||||||
q.To.Type = obj.TYPE_REG
|
q.To.Type = obj.TYPE_REG
|
||||||
q.To.Reg = REGTMP
|
q.To.Reg = REGTMP
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.Pos = p.Pos
|
q.Pos = p.Pos
|
||||||
q.As = AMOVD
|
q.As = AMOVD
|
||||||
q.From.Type = obj.TYPE_REG
|
q.From.Type = obj.TYPE_REG
|
||||||
|
|
@ -590,17 +592,17 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
q.To.Type = obj.TYPE_MEM
|
q.To.Type = obj.TYPE_MEM
|
||||||
q.To.Reg = REGTMP
|
q.To.Reg = REGTMP
|
||||||
|
|
||||||
q1 = obj.Appendp(q, newprog)
|
q1 = obj.Appendp(q, c.newprog)
|
||||||
q1.Pos = p.Pos
|
q1.Pos = p.Pos
|
||||||
q1.As = AMOVD
|
q1.As = AMOVD
|
||||||
q1.From.Type = obj.TYPE_REG
|
q1.From.Type = obj.TYPE_REG
|
||||||
q1.From.Reg = REGTMP
|
q1.From.Reg = REGTMP
|
||||||
q1.To.Type = obj.TYPE_REG
|
q1.To.Type = obj.TYPE_REG
|
||||||
q1.To.Reg = REGSP
|
q1.To.Reg = REGSP
|
||||||
q1.Spadj = ctxt.Autosize
|
q1.Spadj = c.autosize
|
||||||
} else {
|
} else {
|
||||||
// small frame, update SP and save LR in a single MOVD.W instruction
|
// small frame, update SP and save LR in a single MOVD.W instruction
|
||||||
q1 = obj.Appendp(q, newprog)
|
q1 = obj.Appendp(q, c.newprog)
|
||||||
q1.As = AMOVD
|
q1.As = AMOVD
|
||||||
q1.Pos = p.Pos
|
q1.Pos = p.Pos
|
||||||
q1.From.Type = obj.TYPE_REG
|
q1.From.Type = obj.TYPE_REG
|
||||||
|
|
@ -612,7 +614,7 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
q1.Spadj = aoffset
|
q1.Spadj = aoffset
|
||||||
}
|
}
|
||||||
|
|
||||||
if cursym.Text.From3.Offset&obj.WRAPPER != 0 {
|
if c.cursym.Text.From3.Offset&obj.WRAPPER != 0 {
|
||||||
// if(g->panic != nil && g->panic->argp == FP) g->panic->argp = bottom-of-frame
|
// if(g->panic != nil && g->panic->argp == FP) g->panic->argp = bottom-of-frame
|
||||||
//
|
//
|
||||||
// MOV g_panic(g), R1
|
// MOV g_panic(g), R1
|
||||||
|
|
@ -631,26 +633,26 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
// It is a liblink NOP, not a ARM64 NOP: it encodes to 0 instruction bytes.
|
// It is a liblink NOP, not a ARM64 NOP: it encodes to 0 instruction bytes.
|
||||||
q = q1
|
q = q1
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.As = AMOVD
|
q.As = AMOVD
|
||||||
q.From.Type = obj.TYPE_MEM
|
q.From.Type = obj.TYPE_MEM
|
||||||
q.From.Reg = REGG
|
q.From.Reg = REGG
|
||||||
q.From.Offset = 4 * int64(ctxt.Arch.PtrSize) // G.panic
|
q.From.Offset = 4 * int64(c.ctxt.Arch.PtrSize) // G.panic
|
||||||
q.To.Type = obj.TYPE_REG
|
q.To.Type = obj.TYPE_REG
|
||||||
q.To.Reg = REG_R1
|
q.To.Reg = REG_R1
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.As = ACMP
|
q.As = ACMP
|
||||||
q.From.Type = obj.TYPE_REG
|
q.From.Type = obj.TYPE_REG
|
||||||
q.From.Reg = REGZERO
|
q.From.Reg = REGZERO
|
||||||
q.Reg = REG_R1
|
q.Reg = REG_R1
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.As = ABEQ
|
q.As = ABEQ
|
||||||
q.To.Type = obj.TYPE_BRANCH
|
q.To.Type = obj.TYPE_BRANCH
|
||||||
q1 = q
|
q1 = q
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.As = AMOVD
|
q.As = AMOVD
|
||||||
q.From.Type = obj.TYPE_MEM
|
q.From.Type = obj.TYPE_MEM
|
||||||
q.From.Reg = REG_R1
|
q.From.Reg = REG_R1
|
||||||
|
|
@ -658,26 +660,26 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
q.To.Type = obj.TYPE_REG
|
q.To.Type = obj.TYPE_REG
|
||||||
q.To.Reg = REG_R2
|
q.To.Reg = REG_R2
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.As = AADD
|
q.As = AADD
|
||||||
q.From.Type = obj.TYPE_CONST
|
q.From.Type = obj.TYPE_CONST
|
||||||
q.From.Offset = int64(ctxt.Autosize) + 8
|
q.From.Offset = int64(c.autosize) + 8
|
||||||
q.Reg = REGSP
|
q.Reg = REGSP
|
||||||
q.To.Type = obj.TYPE_REG
|
q.To.Type = obj.TYPE_REG
|
||||||
q.To.Reg = REG_R3
|
q.To.Reg = REG_R3
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.As = ACMP
|
q.As = ACMP
|
||||||
q.From.Type = obj.TYPE_REG
|
q.From.Type = obj.TYPE_REG
|
||||||
q.From.Reg = REG_R2
|
q.From.Reg = REG_R2
|
||||||
q.Reg = REG_R3
|
q.Reg = REG_R3
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.As = ABNE
|
q.As = ABNE
|
||||||
q.To.Type = obj.TYPE_BRANCH
|
q.To.Type = obj.TYPE_BRANCH
|
||||||
q2 = q
|
q2 = q
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.As = AADD
|
q.As = AADD
|
||||||
q.From.Type = obj.TYPE_CONST
|
q.From.Type = obj.TYPE_CONST
|
||||||
q.From.Offset = 8
|
q.From.Offset = 8
|
||||||
|
|
@ -685,7 +687,7 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
q.To.Type = obj.TYPE_REG
|
q.To.Type = obj.TYPE_REG
|
||||||
q.To.Reg = REG_R4
|
q.To.Reg = REG_R4
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
q.As = AMOVD
|
q.As = AMOVD
|
||||||
q.From.Type = obj.TYPE_REG
|
q.From.Type = obj.TYPE_REG
|
||||||
q.From.Reg = REG_R4
|
q.From.Reg = REG_R4
|
||||||
|
|
@ -693,7 +695,7 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
q.To.Reg = REG_R1
|
q.To.Reg = REG_R1
|
||||||
q.To.Offset = 0 // Panic.argp
|
q.To.Offset = 0 // Panic.argp
|
||||||
|
|
||||||
q = obj.Appendp(q, newprog)
|
q = obj.Appendp(q, c.newprog)
|
||||||
|
|
||||||
q.As = obj.ANOP
|
q.As = obj.ANOP
|
||||||
q1.Pcond = q
|
q1.Pcond = q
|
||||||
|
|
@ -703,24 +705,24 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
case obj.ARET:
|
case obj.ARET:
|
||||||
nocache(p)
|
nocache(p)
|
||||||
if p.From.Type == obj.TYPE_CONST {
|
if p.From.Type == obj.TYPE_CONST {
|
||||||
ctxt.Diag("using BECOME (%v) is not supported!", p)
|
c.ctxt.Diag("using BECOME (%v) is not supported!", p)
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
retjmp = p.To.Sym
|
retjmp = p.To.Sym
|
||||||
p.To = obj.Addr{}
|
p.To = obj.Addr{}
|
||||||
if cursym.Text.Mark&LEAF != 0 {
|
if c.cursym.Text.Mark&LEAF != 0 {
|
||||||
if ctxt.Autosize != 0 {
|
if c.autosize != 0 {
|
||||||
p.As = AADD
|
p.As = AADD
|
||||||
p.From.Type = obj.TYPE_CONST
|
p.From.Type = obj.TYPE_CONST
|
||||||
p.From.Offset = int64(ctxt.Autosize)
|
p.From.Offset = int64(c.autosize)
|
||||||
p.To.Type = obj.TYPE_REG
|
p.To.Type = obj.TYPE_REG
|
||||||
p.To.Reg = REGSP
|
p.To.Reg = REGSP
|
||||||
p.Spadj = -ctxt.Autosize
|
p.Spadj = -c.autosize
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* want write-back pre-indexed SP+autosize -> SP, loading REGLINK*/
|
/* want write-back pre-indexed SP+autosize -> SP, loading REGLINK*/
|
||||||
aoffset = ctxt.Autosize
|
aoffset = c.autosize
|
||||||
|
|
||||||
if aoffset > 0xF0 {
|
if aoffset > 0xF0 {
|
||||||
aoffset = 0xF0
|
aoffset = 0xF0
|
||||||
|
|
@ -733,11 +735,11 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
p.To.Type = obj.TYPE_REG
|
p.To.Type = obj.TYPE_REG
|
||||||
p.To.Reg = REGLINK
|
p.To.Reg = REGLINK
|
||||||
p.Spadj = -aoffset
|
p.Spadj = -aoffset
|
||||||
if ctxt.Autosize > aoffset {
|
if c.autosize > aoffset {
|
||||||
q = newprog()
|
q = newprog()
|
||||||
q.As = AADD
|
q.As = AADD
|
||||||
q.From.Type = obj.TYPE_CONST
|
q.From.Type = obj.TYPE_CONST
|
||||||
q.From.Offset = int64(ctxt.Autosize) - int64(aoffset)
|
q.From.Offset = int64(c.autosize) - int64(aoffset)
|
||||||
q.To.Type = obj.TYPE_REG
|
q.To.Type = obj.TYPE_REG
|
||||||
q.To.Reg = REGSP
|
q.To.Reg = REGSP
|
||||||
q.Link = p.Link
|
q.Link = p.Link
|
||||||
|
|
@ -760,7 +762,7 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
p.As = AB
|
p.As = AB
|
||||||
p.To.Type = obj.TYPE_BRANCH
|
p.To.Type = obj.TYPE_BRANCH
|
||||||
p.To.Sym = retjmp
|
p.To.Sym = retjmp
|
||||||
p.Spadj = +ctxt.Autosize
|
p.Spadj = +c.autosize
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -768,7 +770,7 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym, newprog obj.ProgAlloc) {
|
||||||
p.To.Type = obj.TYPE_MEM
|
p.To.Type = obj.TYPE_MEM
|
||||||
p.To.Offset = 0
|
p.To.Offset = 0
|
||||||
p.To.Reg = REGLINK
|
p.To.Reg = REGLINK
|
||||||
p.Spadj = +ctxt.Autosize
|
p.Spadj = +c.autosize
|
||||||
|
|
||||||
case AADD, ASUB:
|
case AADD, ASUB:
|
||||||
if p.To.Type == obj.TYPE_REG && p.To.Reg == REGSP && p.From.Type == obj.TYPE_CONST {
|
if p.To.Type == obj.TYPE_REG && p.To.Reg == REGSP && p.From.Type == obj.TYPE_CONST {
|
||||||
|
|
|
||||||
|
|
@ -728,8 +728,6 @@ type Link struct {
|
||||||
Imports []string
|
Imports []string
|
||||||
Plan9privates *LSym
|
Plan9privates *LSym
|
||||||
Printp *Prog
|
Printp *Prog
|
||||||
Blitrl *Prog
|
|
||||||
Elitrl *Prog
|
|
||||||
Instoffset int64
|
Instoffset int64
|
||||||
Autosize int32
|
Autosize int32
|
||||||
Pc int64
|
Pc int64
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue