Skip to content

Commit

Permalink
cmd/internal/obj/mips: replace MOVD with MOVF on 32-bit to avoid unal…
Browse files Browse the repository at this point in the history
…igned memory access

This is the simplest CL that I can make for Go 1.8. For Go 1.9, we can revisit it
and optimize the redundant address generation instructions or just fix golang#599 instead.

Fixes golang#18140.

Change-Id: Ie4804ab0e00dc6bb318da2bece8035c7c71caac3
Reviewed-on: https://go-review.googlesource.com/34193
Run-TryBot: Minux Ma <[email protected]>
TryBot-Result: Gobot Gobot <[email protected]>
Reviewed-by: Cherry Zhang <[email protected]>
  • Loading branch information
minux committed Dec 12, 2016
1 parent bc61026 commit 9fe2291
Showing 1 changed file with 38 additions and 0 deletions.
38 changes: 38 additions & 0 deletions src/cmd/internal/obj/mips/obj0.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ package mips
import (
"cmd/internal/obj"
"cmd/internal/sys"
"encoding/binary"
"fmt"
"math"
)
Expand Down Expand Up @@ -533,6 +534,43 @@ func preprocess(ctxt *obj.Link, cursym *obj.LSym) {
}
}

if ctxt.Mode&Mips32 != 0 {
// rewrite MOVD into two MOVF in 32-bit mode to avoid unaligned memory access
for p = cursym.Text; p != nil; p = p1 {
p1 = p.Link

if p.As != AMOVD {
continue
}
if p.From.Type != obj.TYPE_MEM && p.To.Type != obj.TYPE_MEM {
continue
}

p.As = AMOVF
q = ctxt.NewProg()
*q = *p
q.Link = p.Link
p.Link = q
p1 = q.Link

var regOff int16
if ctxt.Arch.ByteOrder == binary.BigEndian {
regOff = 1 // load odd register first
}
if p.From.Type == obj.TYPE_MEM {
reg := REG_F0 + (p.To.Reg-REG_F0)&^1
p.To.Reg = reg + regOff
q.To.Reg = reg + 1 - regOff
q.From.Offset += 4
} else if p.To.Type == obj.TYPE_MEM {
reg := REG_F0 + (p.From.Reg-REG_F0)&^1
p.From.Reg = reg + regOff
q.From.Reg = reg + 1 - regOff
q.To.Offset += 4
}
}
}

if nosched {
// if we don't do instruction scheduling, simply add
// NOP after each branch instruction.
Expand Down

0 comments on commit 9fe2291

Please sign in to comment.