Commit a2b1dc86 authored by Joel Sing's avatar Joel Sing

cmd/link,cmd/internal/objabi: factor out direct call identification

Factor out the direct CALL identification code from objabi.IsDirectJump and
use this in two places that have separately maintained lists of reloc types.
Provide an objabi.IsDirectCallOrJump function that implements the original
behaviour of objabi.IsDirectJump.

Change-Id: I48131bae92b2938fd7822110d53df0b4ffb35766
Reviewed-on: https://go-review.googlesource.com/c/go/+/196577
Run-TryBot: Joel Sing <joel@sing.id.au>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarCherry Zhang <cherryyz@google.com>
parent 4cb92600
...@@ -223,16 +223,34 @@ const ( ...@@ -223,16 +223,34 @@ const (
R_XCOFFREF R_XCOFFREF
) )
// IsDirectCall reports whether r is a relocation for a direct call.
// A direct call is a CALL instruction that takes the target address
// as an immediate. The address is embedded into the instruction, possibly
// with limited width. An indirect call is a CALL instruction that takes
// the target address in register or memory.
func (r RelocType) IsDirectCall() bool {
switch r {
case R_CALL, R_CALLARM, R_CALLARM64, R_CALLMIPS, R_CALLPOWER:
return true
}
return false
}
// IsDirectJump reports whether r is a relocation for a direct jump. // IsDirectJump reports whether r is a relocation for a direct jump.
// A direct jump is a CALL or JMP instruction that takes the target address // A direct jump is a JMP instruction that takes the target address
// as immediate. The address is embedded into the instruction, possibly // as an immediate. The address is embedded into the instruction, possibly
// with limited width. // with limited width. An indirect jump is a JMP instruction that takes
// An indirect jump is a CALL or JMP instruction that takes the target address // the target address in register or memory.
// in register or memory.
func (r RelocType) IsDirectJump() bool { func (r RelocType) IsDirectJump() bool {
switch r { switch r {
case R_CALL, R_CALLARM, R_CALLARM64, R_CALLPOWER, R_CALLMIPS, R_JMPMIPS: case R_JMPMIPS:
return true return true
} }
return false return false
} }
// IsDirectCallOrJump reports whether r is a relocation for a direct
// call or a direct jump.
func (r RelocType) IsDirectCallOrJump() bool {
return r.IsDirectCall() || r.IsDirectJump()
}
...@@ -74,7 +74,7 @@ func maxSizeTrampolinesPPC64(s *sym.Symbol, isTramp bool) uint64 { ...@@ -74,7 +74,7 @@ func maxSizeTrampolinesPPC64(s *sym.Symbol, isTramp bool) uint64 {
n := uint64(0) n := uint64(0)
for ri := range s.R { for ri := range s.R {
r := &s.R[ri] r := &s.R[ri]
if r.Type.IsDirectJump() { if r.Type.IsDirectCallOrJump() {
n++ n++
} }
} }
...@@ -93,7 +93,7 @@ func trampoline(ctxt *Link, s *sym.Symbol) { ...@@ -93,7 +93,7 @@ func trampoline(ctxt *Link, s *sym.Symbol) {
for ri := range s.R { for ri := range s.R {
r := &s.R[ri] r := &s.R[ri]
if !r.Type.IsDirectJump() { if !r.Type.IsDirectCallOrJump() {
continue continue
} }
if Symaddr(r.Sym) == 0 && (r.Sym.Type != sym.SDYNIMPORT && r.Sym.Type != sym.SUNDEFEXT) { if Symaddr(r.Sym) == 0 && (r.Sym.Type != sym.SDYNIMPORT && r.Sym.Type != sym.SUNDEFEXT) {
......
...@@ -2191,9 +2191,8 @@ func stkcheck(ctxt *Link, up *chain, depth int) int { ...@@ -2191,9 +2191,8 @@ func stkcheck(ctxt *Link, up *chain, depth int) int {
// Process calls in this span. // Process calls in this span.
for ; ri < endr && uint32(s.R[ri].Off) < pcsp.NextPC; ri++ { for ; ri < endr && uint32(s.R[ri].Off) < pcsp.NextPC; ri++ {
r = &s.R[ri] r = &s.R[ri]
switch r.Type { switch {
// Direct call. case r.Type.IsDirectCall():
case objabi.R_CALL, objabi.R_CALLARM, objabi.R_CALLARM64, objabi.R_CALLPOWER, objabi.R_CALLMIPS:
ch.limit = int(int32(limit) - pcsp.Value - int32(callsize(ctxt))) ch.limit = int(int32(limit) - pcsp.Value - int32(callsize(ctxt)))
ch.sym = r.Sym ch.sym = r.Sym
if stkcheck(ctxt, &ch, depth+1) < 0 { if stkcheck(ctxt, &ch, depth+1) < 0 {
...@@ -2204,9 +2203,8 @@ func stkcheck(ctxt *Link, up *chain, depth int) int { ...@@ -2204,9 +2203,8 @@ func stkcheck(ctxt *Link, up *chain, depth int) int {
// so we have to make sure it can call morestack. // so we have to make sure it can call morestack.
// Arrange the data structures to report both calls, so that // Arrange the data structures to report both calls, so that
// if there is an error, stkprint shows all the steps involved. // if there is an error, stkprint shows all the steps involved.
case objabi.R_CALLIND: case r.Type == objabi.R_CALLIND:
ch.limit = int(int32(limit) - pcsp.Value - int32(callsize(ctxt))) ch.limit = int(int32(limit) - pcsp.Value - int32(callsize(ctxt)))
ch.sym = nil ch.sym = nil
ch1.limit = ch.limit - callsize(ctxt) // for morestack in called prologue ch1.limit = ch.limit - callsize(ctxt) // for morestack in called prologue
ch1.up = &ch ch1.up = &ch
...@@ -2525,7 +2523,7 @@ func (ctxt *Link) callgraph() { ...@@ -2525,7 +2523,7 @@ func (ctxt *Link) callgraph() {
if r.Sym == nil { if r.Sym == nil {
continue continue
} }
if (r.Type == objabi.R_CALL || r.Type == objabi.R_CALLARM || r.Type == objabi.R_CALLARM64 || r.Type == objabi.R_CALLPOWER || r.Type == objabi.R_CALLMIPS) && r.Sym.Type == sym.STEXT { if r.Type.IsDirectCall() && r.Sym.Type == sym.STEXT {
ctxt.Logf("%s calls %s\n", s.Name, r.Sym.Name) ctxt.Logf("%s calls %s\n", s.Name, r.Sym.Name)
} }
} }
......
...@@ -254,7 +254,7 @@ func (ctxt *Link) pclntab() { ...@@ -254,7 +254,7 @@ func (ctxt *Link) pclntab() {
// set the resumption point to PC_B. // set the resumption point to PC_B.
lastWasmAddr = uint32(r.Add) lastWasmAddr = uint32(r.Add)
} }
if r.Type.IsDirectJump() && r.Sym != nil && r.Sym.Name == "runtime.deferreturn" { if r.Type.IsDirectCall() && r.Sym != nil && r.Sym.Name == "runtime.deferreturn" {
if ctxt.Arch.Family == sys.Wasm { if ctxt.Arch.Family == sys.Wasm {
deferreturn = lastWasmAddr - 1 deferreturn = lastWasmAddr - 1
} else { } else {
......
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