blob: 8d06f1e3ed50caf4134fc1a76f87fa2a8b7c80b0 [file] [log] [blame]
Keith Randalld2fd43a2015-04-15 15:51:25 -07001// Copyright 2015 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
5package gc
6
7import (
Josh Bleecher Snyder35fb5142015-08-10 12:15:52 -07008 "bytes"
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07009 "fmt"
Josh Bleecher Snyder35fb5142015-08-10 12:15:52 -070010 "html"
Josh Bleecher Snyderd2982092015-07-22 13:13:53 -070011 "os"
Brad Fitzpatrick7af53d92015-07-10 10:47:28 -060012 "strings"
Keith Randalld2fd43a2015-04-15 15:51:25 -070013
Keith Randall067e8df2015-05-28 13:49:20 -070014 "cmd/compile/internal/ssa"
Keith Randall083a6462015-05-12 11:06:44 -070015 "cmd/internal/obj"
Matthew Dempskyc6e11fe2016-04-06 12:01:40 -070016 "cmd/internal/sys"
Keith Randalld2fd43a2015-04-15 15:51:25 -070017)
18
Keith Randallc0740fe2016-03-03 22:06:57 -080019var ssaEnabled = true
20
Keith Randall2f57d0f2016-01-28 13:46:30 -080021var ssaConfig *ssa.Config
22var ssaExp ssaExport
23
David Chase378a8632016-02-25 13:10:51 -050024func initssa() *ssa.Config {
25 ssaExp.unimplemented = false
26 ssaExp.mustImplement = true
27 if ssaConfig == nil {
Matthew Dempskyc6e11fe2016-04-06 12:01:40 -070028 ssaConfig = ssa.NewConfig(Thearch.LinkArch.Name, &ssaExp, Ctxt, Debug['N'] == 0)
David Chase378a8632016-02-25 13:10:51 -050029 }
30 return ssaConfig
31}
32
Keith Randall5b355a72015-12-11 20:41:52 -080033func shouldssa(fn *Node) bool {
Matthew Dempskyc6e11fe2016-04-06 12:01:40 -070034 switch Thearch.LinkArch.Name {
Keith Randall4c9a4702016-03-21 22:57:26 -070035 default:
36 // Only available for testing.
37 if os.Getenv("SSATEST") == "" {
38 return false
39 }
40 // Generally available.
41 case "amd64":
Keith Randall5b355a72015-12-11 20:41:52 -080042 }
Keith Randallc0740fe2016-03-03 22:06:57 -080043 if !ssaEnabled {
44 return false
45 }
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -070046
David Chasee99dd522015-10-19 11:36:07 -040047 // Environment variable control of SSA CG
48 // 1. IF GOSSAFUNC == current function name THEN
49 // compile this function with SSA and log output to ssa.html
50
David Chase729abfa2015-10-26 17:34:06 -040051 // 2. IF GOSSAHASH == "" THEN
David Chasee99dd522015-10-19 11:36:07 -040052 // compile this function (and everything else) with SSA
53
David Chase729abfa2015-10-26 17:34:06 -040054 // 3. IF GOSSAHASH == "n" or "N"
David Chasee99dd522015-10-19 11:36:07 -040055 // IF GOSSAPKG == current package name THEN
56 // compile this function (and everything in this package) with SSA
57 // ELSE
58 // use the old back end for this function.
59 // This is for compatibility with existing test harness and should go away.
60
61 // 4. IF GOSSAHASH is a suffix of the binary-rendered SHA1 hash of the function name THEN
62 // compile this function with SSA
63 // ELSE
64 // compile this function with the old back end.
65
David Chase729abfa2015-10-26 17:34:06 -040066 // Plan is for 3 to be removed when the tests are revised.
67 // SSA is now default, and is disabled by setting
68 // GOSSAHASH to n or N, or selectively with strings of
69 // 0 and 1.
David Chasee99dd522015-10-19 11:36:07 -040070
Keith Randall5b355a72015-12-11 20:41:52 -080071 name := fn.Func.Nname.Sym.Name
72
73 funcname := os.Getenv("GOSSAFUNC")
74 if funcname != "" {
75 // If GOSSAFUNC is set, compile only that function.
76 return name == funcname
77 }
78
79 pkg := os.Getenv("GOSSAPKG")
80 if pkg != "" {
81 // If GOSSAPKG is set, compile only that package.
82 return localpkg.Name == pkg
83 }
84
David Chase378a8632016-02-25 13:10:51 -050085 return initssa().DebugHashMatch("GOSSAHASH", name)
Keith Randall5b355a72015-12-11 20:41:52 -080086}
87
88// buildssa builds an SSA function.
89func buildssa(fn *Node) *ssa.Func {
90 name := fn.Func.Nname.Sym.Name
Keith Randall59681802016-03-01 13:47:48 -080091 printssa := name == os.Getenv("GOSSAFUNC")
Keith Randall5b355a72015-12-11 20:41:52 -080092 if printssa {
Josh Bleecher Snydere0ac5c52015-07-20 18:42:45 -070093 fmt.Println("generating SSA for", name)
Ian Lance Taylor55c65d42016-03-04 13:16:48 -080094 dumplist("buildssa-enter", fn.Func.Enter)
95 dumplist("buildssa-body", fn.Nbody)
96 dumplist("buildssa-exit", fn.Func.Exit)
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -070097 }
Keith Randalld2fd43a2015-04-15 15:51:25 -070098
Keith Randallcfc2aa52015-05-18 16:44:20 -070099 var s state
Michael Matloob81ccf502015-05-30 01:03:06 -0400100 s.pushLine(fn.Lineno)
101 defer s.popLine()
102
Keith Randall6a8a9da2016-02-27 17:49:31 -0800103 if fn.Func.Pragma&CgoUnsafeArgs != 0 {
104 s.cgoUnsafeArgs = true
105 }
Keith Randall15ed37d2016-03-16 21:51:17 -0700106 if fn.Func.Pragma&Nowritebarrier != 0 {
107 s.noWB = true
108 }
109 defer func() {
110 if s.WBLineno != 0 {
111 fn.Func.WBLineno = s.WBLineno
112 }
113 }()
Keith Randalld2fd43a2015-04-15 15:51:25 -0700114 // TODO(khr): build config just once at the start of the compiler binary
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -0700115
Keith Randall2f57d0f2016-01-28 13:46:30 -0800116 ssaExp.log = printssa
David Chase378a8632016-02-25 13:10:51 -0500117
118 s.config = initssa()
Keith Randalld2fd43a2015-04-15 15:51:25 -0700119 s.f = s.config.NewFunc()
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -0700120 s.f.Name = name
David Chase8824dcc2015-10-08 12:39:56 -0400121 s.exitCode = fn.Func.Exit
Keith Randall74e568f2015-11-09 21:35:40 -0800122 s.panics = map[funcLine]*ssa.Block{}
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -0700123
Josh Bleecher Snyder35fb5142015-08-10 12:15:52 -0700124 if name == os.Getenv("GOSSAFUNC") {
125 // TODO: tempfile? it is handy to have the location
126 // of this file be stable, so you can just reload in the browser.
Keith Randallda8af472016-01-13 11:14:57 -0800127 s.config.HTML = ssa.NewHTMLWriter("ssa.html", s.config, name)
Josh Bleecher Snyder35fb5142015-08-10 12:15:52 -0700128 // TODO: generate and print a mapping from nodes to values and blocks
129 }
130 defer func() {
Keith Randall5b355a72015-12-11 20:41:52 -0800131 if !printssa {
Josh Bleecher Snyder35fb5142015-08-10 12:15:52 -0700132 s.config.HTML.Close()
133 }
134 }()
135
Keith Randalld2fd43a2015-04-15 15:51:25 -0700136 // Allocate starting block
137 s.f.Entry = s.f.NewBlock(ssa.BlockPlain)
138
Keith Randallcfc2aa52015-05-18 16:44:20 -0700139 // Allocate starting values
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700140 s.labels = map[string]*ssaLabel{}
141 s.labeledNodes = map[*Node]*ssaLabel{}
Keith Randall02f4d0a2015-11-02 08:10:26 -0800142 s.startmem = s.entryNewValue0(ssa.OpInitMem, ssa.TypeMem)
Josh Bleecher Snyder85e03292015-07-30 11:03:05 -0700143 s.sp = s.entryNewValue0(ssa.OpSP, Types[TUINTPTR]) // TODO: use generic pointer type (unsafe.Pointer?) instead
144 s.sb = s.entryNewValue0(ssa.OpSB, Types[TUINTPTR])
Keith Randall8c46aa52015-06-19 21:02:28 -0700145
David Chase956f3192015-09-11 16:40:05 -0400146 s.startBlock(s.f.Entry)
147 s.vars[&memVar] = s.startmem
148
Todd Neald076ef72015-10-15 20:25:32 -0500149 s.varsyms = map[*Node]interface{}{}
150
Keith Randall8c46aa52015-06-19 21:02:28 -0700151 // Generate addresses of local declarations
152 s.decladdrs = map[*Node]*ssa.Value{}
Keith Randall4fffd4562016-02-29 13:31:48 -0800153 for _, n := range fn.Func.Dcl {
Keith Randall8c46aa52015-06-19 21:02:28 -0700154 switch n.Class {
Keith Randall6a8a9da2016-02-27 17:49:31 -0800155 case PPARAM, PPARAMOUT:
Todd Neald076ef72015-10-15 20:25:32 -0500156 aux := s.lookupSymbol(n, &ssa.ArgSymbol{Typ: n.Type, Node: n})
Keith Randall8c46aa52015-06-19 21:02:28 -0700157 s.decladdrs[n] = s.entryNewValue1A(ssa.OpAddr, Ptrto(n.Type), aux, s.sp)
Keith Randall6a8a9da2016-02-27 17:49:31 -0800158 if n.Class == PPARAMOUT && s.canSSA(n) {
159 // Save ssa-able PPARAMOUT variables so we can
160 // store them back to the stack at the end of
161 // the function.
162 s.returns = append(s.returns, n)
163 }
Keith Randall3572c642016-04-21 19:28:28 -0700164 if n.Class == PPARAM && s.canSSA(n) && n.Type.IsPtrShaped() {
165 s.ptrargs = append(s.ptrargs, n)
166 n.SetNotLiveAtEnd(true) // SSA takes care of this explicitly
167 }
Keith Randall6a8a9da2016-02-27 17:49:31 -0800168 case PAUTO:
Keith Randalld2107fc2015-08-24 02:16:19 -0700169 // processed at each use, to prevent Addr coming
170 // before the decl.
Russ Coxb6dc3e62016-05-25 01:33:24 -0400171 case PAUTOHEAP:
172 // moved to heap - already handled by frontend
Keith Randallc3eb1a72015-09-06 13:42:26 -0700173 case PFUNC:
174 // local function - already handled by frontend
Daniel Morsingbe2a3e22015-07-01 20:37:25 +0100175 default:
Russ Coxb6dc3e62016-05-25 01:33:24 -0400176 s.Unimplementedf("local variable with class %s unimplemented", classnames[n.Class])
Keith Randall8c46aa52015-06-19 21:02:28 -0700177 }
178 }
Keith Randalld2fd43a2015-04-15 15:51:25 -0700179
180 // Convert the AST-based IR to the SSA-based IR
Keith Randall4fffd4562016-02-29 13:31:48 -0800181 s.stmts(fn.Func.Enter)
Keith Randall9d854fd2016-03-01 12:50:17 -0800182 s.stmts(fn.Nbody)
Keith Randalld2fd43a2015-04-15 15:51:25 -0700183
Keith Randallcfc2aa52015-05-18 16:44:20 -0700184 // fallthrough to exit
Keith Randalla7cfc7592015-09-08 16:04:37 -0700185 if s.curBlock != nil {
Keith Randallddc6b642016-03-09 19:27:57 -0800186 s.pushLine(fn.Func.Endlineno)
187 s.exit()
188 s.popLine()
Keith Randallcfc2aa52015-05-18 16:44:20 -0700189 }
190
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700191 // Check that we used all labels
192 for name, lab := range s.labels {
193 if !lab.used() && !lab.reported {
Robert Griesemerb83f3972016-03-02 11:01:25 -0800194 yyerrorl(lab.defNode.Lineno, "label %v defined and not used", name)
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700195 lab.reported = true
196 }
197 if lab.used() && !lab.defined() && !lab.reported {
Robert Griesemerb83f3972016-03-02 11:01:25 -0800198 yyerrorl(lab.useNode.Lineno, "label %v not defined", name)
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700199 lab.reported = true
200 }
201 }
202
203 // Check any forward gotos. Non-forward gotos have already been checked.
204 for _, n := range s.fwdGotos {
205 lab := s.labels[n.Left.Sym.Name]
206 // If the label is undefined, we have already have printed an error.
207 if lab.defined() {
208 s.checkgoto(n, lab.defNode)
209 }
210 }
211
212 if nerrors > 0 {
Keith Randall4c5459d2016-01-28 16:11:56 -0800213 s.f.Free()
Keith Randall5b355a72015-12-11 20:41:52 -0800214 return nil
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700215 }
216
David Chase6b99fb52016-04-21 13:24:58 -0400217 prelinkNumvars := s.f.NumValues()
218 sparseDefState := s.locatePotentialPhiFunctions(fn)
219
Keith Randalld2fd43a2015-04-15 15:51:25 -0700220 // Link up variable uses to variable definitions
David Chase6b99fb52016-04-21 13:24:58 -0400221 s.linkForwardReferences(sparseDefState)
222
223 if ssa.BuildStats > 0 {
224 s.f.LogStat("build", s.f.NumBlocks(), "blocks", prelinkNumvars, "vars_before",
225 s.f.NumValues(), "vars_after", prelinkNumvars*s.f.NumBlocks(), "ssa_phi_loc_cutoff_score")
226 }
Keith Randalld2fd43a2015-04-15 15:51:25 -0700227
David Chase8824dcc2015-10-08 12:39:56 -0400228 // Don't carry reference this around longer than necessary
Keith Randall4fffd4562016-02-29 13:31:48 -0800229 s.exitCode = Nodes{}
David Chase8824dcc2015-10-08 12:39:56 -0400230
Josh Bleecher Snyder983bc8d2015-07-17 16:47:43 +0000231 // Main call to ssa package to compile function
232 ssa.Compile(s.f)
233
Keith Randall5b355a72015-12-11 20:41:52 -0800234 return s.f
Keith Randalld2fd43a2015-04-15 15:51:25 -0700235}
236
Keith Randallcfc2aa52015-05-18 16:44:20 -0700237type state struct {
Keith Randalld2fd43a2015-04-15 15:51:25 -0700238 // configuration (arch) information
239 config *ssa.Config
240
241 // function we're building
242 f *ssa.Func
243
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700244 // labels and labeled control flow nodes (OFOR, OSWITCH, OSELECT) in f
245 labels map[string]*ssaLabel
246 labeledNodes map[*Node]*ssaLabel
247
248 // gotos that jump forward; required for deferred checkgoto calls
249 fwdGotos []*Node
David Chase8824dcc2015-10-08 12:39:56 -0400250 // Code that must precede any return
251 // (e.g., copying value of heap-escaped paramout back to true paramout)
Keith Randall4fffd4562016-02-29 13:31:48 -0800252 exitCode Nodes
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700253
254 // unlabeled break and continue statement tracking
255 breakTo *ssa.Block // current target for plain break statement
256 continueTo *ssa.Block // current target for plain continue statement
Keith Randalld2fd43a2015-04-15 15:51:25 -0700257
258 // current location where we're interpreting the AST
259 curBlock *ssa.Block
260
Keith Randall8c46aa52015-06-19 21:02:28 -0700261 // variable assignments in the current block (map from variable symbol to ssa value)
262 // *Node is the unique identifier (an ONAME Node) for the variable.
263 vars map[*Node]*ssa.Value
Keith Randalld2fd43a2015-04-15 15:51:25 -0700264
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +0000265 // all defined variables at the end of each block. Indexed by block ID.
Keith Randall8c46aa52015-06-19 21:02:28 -0700266 defvars []map[*Node]*ssa.Value
Keith Randalld2fd43a2015-04-15 15:51:25 -0700267
Keith Randalld2107fc2015-08-24 02:16:19 -0700268 // addresses of PPARAM and PPARAMOUT variables.
Keith Randall8c46aa52015-06-19 21:02:28 -0700269 decladdrs map[*Node]*ssa.Value
Keith Randallcfc2aa52015-05-18 16:44:20 -0700270
Todd Neald076ef72015-10-15 20:25:32 -0500271 // symbols for PEXTERN, PAUTO and PPARAMOUT variables so they can be reused.
272 varsyms map[*Node]interface{}
273
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +0000274 // starting values. Memory, stack pointer, and globals pointer
Keith Randallcfc2aa52015-05-18 16:44:20 -0700275 startmem *ssa.Value
Keith Randallcfc2aa52015-05-18 16:44:20 -0700276 sp *ssa.Value
Keith Randall8c46aa52015-06-19 21:02:28 -0700277 sb *ssa.Value
Michael Matloob81ccf502015-05-30 01:03:06 -0400278
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +0000279 // line number stack. The current line number is top of stack
Michael Matloob81ccf502015-05-30 01:03:06 -0400280 line []int32
Keith Randall74e568f2015-11-09 21:35:40 -0800281
282 // list of panic calls by function name and line number.
283 // Used to deduplicate panic calls.
284 panics map[funcLine]*ssa.Block
Keith Randallb5c5efd2016-01-14 16:02:23 -0800285
286 // list of FwdRef values.
287 fwdRefs []*ssa.Value
Keith Randall6a8a9da2016-02-27 17:49:31 -0800288
Russ Coxb6dc3e62016-05-25 01:33:24 -0400289 // list of PPARAMOUT (return) variables.
Keith Randall6a8a9da2016-02-27 17:49:31 -0800290 returns []*Node
291
Keith Randall3572c642016-04-21 19:28:28 -0700292 // list of PPARAM SSA-able pointer-shaped args. We ensure these are live
293 // throughout the function to help users avoid premature finalizers.
294 ptrargs []*Node
295
Keith Randall6a8a9da2016-02-27 17:49:31 -0800296 cgoUnsafeArgs bool
Keith Randall15ed37d2016-03-16 21:51:17 -0700297 noWB bool
298 WBLineno int32 // line number of first write barrier. 0=no write barriers
Keith Randall74e568f2015-11-09 21:35:40 -0800299}
300
301type funcLine struct {
302 f *Node
303 line int32
Keith Randalld2fd43a2015-04-15 15:51:25 -0700304}
305
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700306type ssaLabel struct {
307 target *ssa.Block // block identified by this label
308 breakTarget *ssa.Block // block to break to in control flow node identified by this label
309 continueTarget *ssa.Block // block to continue to in control flow node identified by this label
310 defNode *Node // label definition Node (OLABEL)
311 // Label use Node (OGOTO, OBREAK, OCONTINUE).
312 // Used only for error detection and reporting.
313 // There might be multiple uses, but we only need to track one.
314 useNode *Node
315 reported bool // reported indicates whether an error has already been reported for this label
316}
317
318// defined reports whether the label has a definition (OLABEL node).
319func (l *ssaLabel) defined() bool { return l.defNode != nil }
320
321// used reports whether the label has a use (OGOTO, OBREAK, or OCONTINUE node).
322func (l *ssaLabel) used() bool { return l.useNode != nil }
323
324// label returns the label associated with sym, creating it if necessary.
325func (s *state) label(sym *Sym) *ssaLabel {
326 lab := s.labels[sym.Name]
327 if lab == nil {
328 lab = new(ssaLabel)
329 s.labels[sym.Name] = lab
330 }
331 return lab
332}
333
Keith Randallda8af472016-01-13 11:14:57 -0800334func (s *state) Logf(msg string, args ...interface{}) { s.config.Logf(msg, args...) }
David Chase88b230e2016-01-29 14:44:15 -0500335func (s *state) Log() bool { return s.config.Log() }
Keith Randallda8af472016-01-13 11:14:57 -0800336func (s *state) Fatalf(msg string, args ...interface{}) { s.config.Fatalf(s.peekLine(), msg, args...) }
337func (s *state) Unimplementedf(msg string, args ...interface{}) {
338 s.config.Unimplementedf(s.peekLine(), msg, args...)
339}
Todd Neal98b88de2016-03-13 23:04:31 -0500340func (s *state) Warnl(line int32, msg string, args ...interface{}) { s.config.Warnl(line, msg, args...) }
341func (s *state) Debug_checknil() bool { return s.config.Debug_checknil() }
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -0700342
Keith Randall269baa92015-09-17 10:31:16 -0700343var (
344 // dummy node for the memory variable
Keith Randallc24681a2015-10-22 14:22:38 -0700345 memVar = Node{Op: ONAME, Class: Pxxx, Sym: &Sym{Name: "mem"}}
Keith Randall8c46aa52015-06-19 21:02:28 -0700346
Keith Randall269baa92015-09-17 10:31:16 -0700347 // dummy nodes for temporary variables
Josh Bleecher Snyder6b33b0e2016-04-10 09:08:00 -0700348 ptrVar = Node{Op: ONAME, Class: Pxxx, Sym: &Sym{Name: "ptr"}}
Josh Bleecher Snyder03e216f2016-04-18 09:40:30 -0700349 lenVar = Node{Op: ONAME, Class: Pxxx, Sym: &Sym{Name: "len"}}
Josh Bleecher Snyder6b33b0e2016-04-10 09:08:00 -0700350 newlenVar = Node{Op: ONAME, Class: Pxxx, Sym: &Sym{Name: "newlen"}}
351 capVar = Node{Op: ONAME, Class: Pxxx, Sym: &Sym{Name: "cap"}}
352 typVar = Node{Op: ONAME, Class: Pxxx, Sym: &Sym{Name: "typ"}}
353 idataVar = Node{Op: ONAME, Class: Pxxx, Sym: &Sym{Name: "idata"}}
354 okVar = Node{Op: ONAME, Class: Pxxx, Sym: &Sym{Name: "ok"}}
355 deltaVar = Node{Op: ONAME, Class: Pxxx, Sym: &Sym{Name: "delta"}}
Keith Randall269baa92015-09-17 10:31:16 -0700356)
Keith Randall5505e8c2015-09-12 23:27:26 -0700357
Keith Randalld2fd43a2015-04-15 15:51:25 -0700358// startBlock sets the current block we're generating code in to b.
Keith Randallcfc2aa52015-05-18 16:44:20 -0700359func (s *state) startBlock(b *ssa.Block) {
360 if s.curBlock != nil {
Josh Bleecher Snyder37ddc272015-06-24 14:03:39 -0700361 s.Fatalf("starting block %v when block %v has not ended", b, s.curBlock)
Keith Randallcfc2aa52015-05-18 16:44:20 -0700362 }
Keith Randalld2fd43a2015-04-15 15:51:25 -0700363 s.curBlock = b
Keith Randall8c46aa52015-06-19 21:02:28 -0700364 s.vars = map[*Node]*ssa.Value{}
Keith Randalld2fd43a2015-04-15 15:51:25 -0700365}
366
367// endBlock marks the end of generating code for the current block.
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +0000368// Returns the (former) current block. Returns nil if there is no current
Keith Randalld2fd43a2015-04-15 15:51:25 -0700369// block, i.e. if no code flows to the current execution point.
Keith Randallcfc2aa52015-05-18 16:44:20 -0700370func (s *state) endBlock() *ssa.Block {
Keith Randalld2fd43a2015-04-15 15:51:25 -0700371 b := s.curBlock
372 if b == nil {
373 return nil
374 }
375 for len(s.defvars) <= int(b.ID) {
376 s.defvars = append(s.defvars, nil)
377 }
378 s.defvars[b.ID] = s.vars
379 s.curBlock = nil
380 s.vars = nil
Michael Matloob81ccf502015-05-30 01:03:06 -0400381 b.Line = s.peekLine()
Keith Randalld2fd43a2015-04-15 15:51:25 -0700382 return b
383}
384
Michael Matloob81ccf502015-05-30 01:03:06 -0400385// pushLine pushes a line number on the line number stack.
386func (s *state) pushLine(line int32) {
387 s.line = append(s.line, line)
388}
389
390// popLine pops the top of the line number stack.
391func (s *state) popLine() {
392 s.line = s.line[:len(s.line)-1]
393}
394
395// peekLine peek the top of the line number stack.
396func (s *state) peekLine() int32 {
397 return s.line[len(s.line)-1]
398}
399
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700400func (s *state) Error(msg string, args ...interface{}) {
Robert Griesemerb83f3972016-03-02 11:01:25 -0800401 yyerrorl(s.peekLine(), msg, args...)
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700402}
403
Keith Randall8f22b522015-06-11 21:29:25 -0700404// newValue0 adds a new value with no arguments to the current block.
405func (s *state) newValue0(op ssa.Op, t ssa.Type) *ssa.Value {
406 return s.curBlock.NewValue0(s.peekLine(), op, t)
407}
408
409// newValue0A adds a new value with no arguments and an aux value to the current block.
410func (s *state) newValue0A(op ssa.Op, t ssa.Type, aux interface{}) *ssa.Value {
411 return s.curBlock.NewValue0A(s.peekLine(), op, t, aux)
Michael Matloob81ccf502015-05-30 01:03:06 -0400412}
413
Todd Neal991036a2015-09-03 18:24:22 -0500414// newValue0I adds a new value with no arguments and an auxint value to the current block.
415func (s *state) newValue0I(op ssa.Op, t ssa.Type, auxint int64) *ssa.Value {
416 return s.curBlock.NewValue0I(s.peekLine(), op, t, auxint)
417}
418
Michael Matloob81ccf502015-05-30 01:03:06 -0400419// newValue1 adds a new value with one argument to the current block.
Keith Randall8f22b522015-06-11 21:29:25 -0700420func (s *state) newValue1(op ssa.Op, t ssa.Type, arg *ssa.Value) *ssa.Value {
421 return s.curBlock.NewValue1(s.peekLine(), op, t, arg)
422}
423
424// newValue1A adds a new value with one argument and an aux value to the current block.
425func (s *state) newValue1A(op ssa.Op, t ssa.Type, aux interface{}, arg *ssa.Value) *ssa.Value {
426 return s.curBlock.NewValue1A(s.peekLine(), op, t, aux, arg)
Michael Matloob81ccf502015-05-30 01:03:06 -0400427}
428
Keith Randallcd7e0592015-07-15 21:33:49 -0700429// newValue1I adds a new value with one argument and an auxint value to the current block.
430func (s *state) newValue1I(op ssa.Op, t ssa.Type, aux int64, arg *ssa.Value) *ssa.Value {
431 return s.curBlock.NewValue1I(s.peekLine(), op, t, aux, arg)
432}
433
Michael Matloob81ccf502015-05-30 01:03:06 -0400434// newValue2 adds a new value with two arguments to the current block.
Keith Randall8f22b522015-06-11 21:29:25 -0700435func (s *state) newValue2(op ssa.Op, t ssa.Type, arg0, arg1 *ssa.Value) *ssa.Value {
436 return s.curBlock.NewValue2(s.peekLine(), op, t, arg0, arg1)
Michael Matloob81ccf502015-05-30 01:03:06 -0400437}
438
Daniel Morsing66b47812015-06-27 15:45:20 +0100439// newValue2I adds a new value with two arguments and an auxint value to the current block.
440func (s *state) newValue2I(op ssa.Op, t ssa.Type, aux int64, arg0, arg1 *ssa.Value) *ssa.Value {
441 return s.curBlock.NewValue2I(s.peekLine(), op, t, aux, arg0, arg1)
442}
443
Michael Matloob81ccf502015-05-30 01:03:06 -0400444// newValue3 adds a new value with three arguments to the current block.
Keith Randall8f22b522015-06-11 21:29:25 -0700445func (s *state) newValue3(op ssa.Op, t ssa.Type, arg0, arg1, arg2 *ssa.Value) *ssa.Value {
446 return s.curBlock.NewValue3(s.peekLine(), op, t, arg0, arg1, arg2)
Michael Matloob81ccf502015-05-30 01:03:06 -0400447}
448
Keith Randalld4cc51d2015-08-14 21:47:20 -0700449// newValue3I adds a new value with three arguments and an auxint value to the current block.
450func (s *state) newValue3I(op ssa.Op, t ssa.Type, aux int64, arg0, arg1, arg2 *ssa.Value) *ssa.Value {
451 return s.curBlock.NewValue3I(s.peekLine(), op, t, aux, arg0, arg1, arg2)
452}
453
Todd Neal991036a2015-09-03 18:24:22 -0500454// entryNewValue0 adds a new value with no arguments to the entry block.
Keith Randall8f22b522015-06-11 21:29:25 -0700455func (s *state) entryNewValue0(op ssa.Op, t ssa.Type) *ssa.Value {
456 return s.f.Entry.NewValue0(s.peekLine(), op, t)
457}
458
Todd Neal991036a2015-09-03 18:24:22 -0500459// entryNewValue0A adds a new value with no arguments and an aux value to the entry block.
Keith Randall8f22b522015-06-11 21:29:25 -0700460func (s *state) entryNewValue0A(op ssa.Op, t ssa.Type, aux interface{}) *ssa.Value {
461 return s.f.Entry.NewValue0A(s.peekLine(), op, t, aux)
Michael Matloob81ccf502015-05-30 01:03:06 -0400462}
463
Todd Neal991036a2015-09-03 18:24:22 -0500464// entryNewValue0I adds a new value with no arguments and an auxint value to the entry block.
465func (s *state) entryNewValue0I(op ssa.Op, t ssa.Type, auxint int64) *ssa.Value {
466 return s.f.Entry.NewValue0I(s.peekLine(), op, t, auxint)
467}
468
Michael Matloob81ccf502015-05-30 01:03:06 -0400469// entryNewValue1 adds a new value with one argument to the entry block.
Keith Randall8f22b522015-06-11 21:29:25 -0700470func (s *state) entryNewValue1(op ssa.Op, t ssa.Type, arg *ssa.Value) *ssa.Value {
471 return s.f.Entry.NewValue1(s.peekLine(), op, t, arg)
472}
473
474// entryNewValue1 adds a new value with one argument and an auxint value to the entry block.
475func (s *state) entryNewValue1I(op ssa.Op, t ssa.Type, auxint int64, arg *ssa.Value) *ssa.Value {
476 return s.f.Entry.NewValue1I(s.peekLine(), op, t, auxint, arg)
Michael Matloob81ccf502015-05-30 01:03:06 -0400477}
478
Keith Randall8c46aa52015-06-19 21:02:28 -0700479// entryNewValue1A adds a new value with one argument and an aux value to the entry block.
480func (s *state) entryNewValue1A(op ssa.Op, t ssa.Type, aux interface{}, arg *ssa.Value) *ssa.Value {
481 return s.f.Entry.NewValue1A(s.peekLine(), op, t, aux, arg)
482}
483
Michael Matloob81ccf502015-05-30 01:03:06 -0400484// entryNewValue2 adds a new value with two arguments to the entry block.
Keith Randall8f22b522015-06-11 21:29:25 -0700485func (s *state) entryNewValue2(op ssa.Op, t ssa.Type, arg0, arg1 *ssa.Value) *ssa.Value {
486 return s.f.Entry.NewValue2(s.peekLine(), op, t, arg0, arg1)
Michael Matloob81ccf502015-05-30 01:03:06 -0400487}
488
Josh Bleecher Snydercea44142015-09-08 16:52:25 -0700489// const* routines add a new const value to the entry block.
Josh Bleecher Snyder39214272016-03-06 18:06:09 -0800490func (s *state) constSlice(t ssa.Type) *ssa.Value { return s.f.ConstSlice(s.peekLine(), t) }
491func (s *state) constInterface(t ssa.Type) *ssa.Value { return s.f.ConstInterface(s.peekLine(), t) }
492func (s *state) constNil(t ssa.Type) *ssa.Value { return s.f.ConstNil(s.peekLine(), t) }
493func (s *state) constEmptyString(t ssa.Type) *ssa.Value { return s.f.ConstEmptyString(s.peekLine(), t) }
Josh Bleecher Snydercea44142015-09-08 16:52:25 -0700494func (s *state) constBool(c bool) *ssa.Value {
495 return s.f.ConstBool(s.peekLine(), Types[TBOOL], c)
496}
Keith Randall9cb332e2015-07-28 14:19:20 -0700497func (s *state) constInt8(t ssa.Type, c int8) *ssa.Value {
498 return s.f.ConstInt8(s.peekLine(), t, c)
499}
500func (s *state) constInt16(t ssa.Type, c int16) *ssa.Value {
501 return s.f.ConstInt16(s.peekLine(), t, c)
502}
503func (s *state) constInt32(t ssa.Type, c int32) *ssa.Value {
504 return s.f.ConstInt32(s.peekLine(), t, c)
505}
506func (s *state) constInt64(t ssa.Type, c int64) *ssa.Value {
507 return s.f.ConstInt64(s.peekLine(), t, c)
508}
David Chase997a9f32015-08-12 16:38:11 -0400509func (s *state) constFloat32(t ssa.Type, c float64) *ssa.Value {
510 return s.f.ConstFloat32(s.peekLine(), t, c)
511}
512func (s *state) constFloat64(t ssa.Type, c float64) *ssa.Value {
513 return s.f.ConstFloat64(s.peekLine(), t, c)
514}
Michael Matloob81ccf502015-05-30 01:03:06 -0400515func (s *state) constInt(t ssa.Type, c int64) *ssa.Value {
Keith Randall9cb332e2015-07-28 14:19:20 -0700516 if s.config.IntSize == 8 {
517 return s.constInt64(t, c)
518 }
519 if int64(int32(c)) != c {
520 s.Fatalf("integer constant too big %d", c)
521 }
522 return s.constInt32(t, int32(c))
Michael Matloob81ccf502015-05-30 01:03:06 -0400523}
524
Keith Randall4fffd4562016-02-29 13:31:48 -0800525func (s *state) stmts(a Nodes) {
526 for _, x := range a.Slice() {
527 s.stmt(x)
528 }
529}
530
Keith Randalld2fd43a2015-04-15 15:51:25 -0700531// ssaStmtList converts the statement n to SSA and adds it to s.
Ian Lance Taylorc4012b62016-03-08 10:26:20 -0800532func (s *state) stmtList(l Nodes) {
Ian Lance Taylor38921b32016-03-08 15:10:26 -0800533 for _, n := range l.Slice() {
534 s.stmt(n)
Keith Randalld2fd43a2015-04-15 15:51:25 -0700535 }
536}
537
538// ssaStmt converts the statement n to SSA and adds it to s.
Keith Randallcfc2aa52015-05-18 16:44:20 -0700539func (s *state) stmt(n *Node) {
Michael Matloob81ccf502015-05-30 01:03:06 -0400540 s.pushLine(n.Lineno)
541 defer s.popLine()
542
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700543 // If s.curBlock is nil, then we're about to generate dead code.
544 // We can't just short-circuit here, though,
545 // because we check labels and gotos as part of SSA generation.
546 // Provide a block for the dead code so that we don't have
547 // to add special cases everywhere else.
548 if s.curBlock == nil {
549 dead := s.f.NewBlock(ssa.BlockPlain)
550 s.startBlock(dead)
551 }
552
Keith Randalld2fd43a2015-04-15 15:51:25 -0700553 s.stmtList(n.Ninit)
554 switch n.Op {
555
556 case OBLOCK:
557 s.stmtList(n.List)
558
Josh Bleecher Snyder2574e4a2015-07-16 13:25:36 -0600559 // No-ops
Todd Neal67e43c12015-08-28 21:19:40 -0500560 case OEMPTY, ODCLCONST, ODCLTYPE, OFALL:
Brad Fitzpatrick7af53d92015-07-10 10:47:28 -0600561
Josh Bleecher Snyder2574e4a2015-07-16 13:25:36 -0600562 // Expression statements
563 case OCALLFUNC, OCALLMETH, OCALLINTER:
Keith Randalld24768e2015-09-09 23:56:59 -0700564 s.call(n, callNormal)
Keith Randallfb54e032016-02-24 16:19:20 -0800565 if n.Op == OCALLFUNC && n.Left.Op == ONAME && n.Left.Class == PFUNC &&
Matthew Dempsky980ab122016-04-13 18:37:18 -0700566 (compiling_runtime && n.Left.Sym.Name == "throw" ||
Keith Randall6a8a9da2016-02-27 17:49:31 -0800567 n.Left.Sym.Pkg == Runtimepkg && (n.Left.Sym.Name == "gopanic" || n.Left.Sym.Name == "selectgo" || n.Left.Sym.Name == "block")) {
Keith Randallfaf1bdb2016-02-06 22:35:34 -0800568 m := s.mem()
569 b := s.endBlock()
570 b.Kind = ssa.BlockExit
Keith Randall56e0ecc2016-03-15 20:45:50 -0700571 b.SetControl(m)
Keith Randallfaf1bdb2016-02-06 22:35:34 -0800572 // TODO: never rewrite OPANIC to OCALLFUNC in the
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +0000573 // first place. Need to wait until all backends
Keith Randallfaf1bdb2016-02-06 22:35:34 -0800574 // go through SSA.
575 }
Keith Randalld24768e2015-09-09 23:56:59 -0700576 case ODEFER:
577 s.call(n.Left, callDefer)
578 case OPROC:
579 s.call(n.Left, callGo)
Josh Bleecher Snyder2574e4a2015-07-16 13:25:36 -0600580
Keith Randall269baa92015-09-17 10:31:16 -0700581 case OAS2DOTTYPE:
Ian Lance Taylor38921b32016-03-08 15:10:26 -0800582 res, resok := s.dottype(n.Rlist.First(), true)
Keith Randalld4663e12016-03-21 10:22:03 -0700583 s.assign(n.List.First(), res, needwritebarrier(n.List.First(), n.Rlist.First()), false, n.Lineno, 0)
584 s.assign(n.List.Second(), resok, false, false, n.Lineno, 0)
Keith Randall269baa92015-09-17 10:31:16 -0700585 return
586
Keith Randalld2fd43a2015-04-15 15:51:25 -0700587 case ODCL:
Russ Coxb6dc3e62016-05-25 01:33:24 -0400588 if n.Left.Class == PAUTOHEAP {
589 Fatalf("DCL %v", n)
Daniel Morsingc31b6dd2015-06-12 14:23:29 +0100590 }
Keith Randalld2fd43a2015-04-15 15:51:25 -0700591
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700592 case OLABEL:
593 sym := n.Left.Sym
594
595 if isblanksym(sym) {
Keith Randall7e4c06d2015-07-12 11:52:09 -0700596 // Empty identifier is valid but useless.
597 // See issues 11589, 11593.
598 return
599 }
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700600
601 lab := s.label(sym)
602
603 // Associate label with its control flow node, if any
604 if ctl := n.Name.Defn; ctl != nil {
605 switch ctl.Op {
606 case OFOR, OSWITCH, OSELECT:
607 s.labeledNodes[ctl] = lab
608 }
Keith Randall0ad9c8c2015-06-12 16:24:33 -0700609 }
Keith Randalld2fd43a2015-04-15 15:51:25 -0700610
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700611 if !lab.defined() {
612 lab.defNode = n
613 } else {
Robert Griesemer2faf5bc2016-03-02 11:30:29 -0800614 s.Error("label %v already defined at %v", sym, linestr(lab.defNode.Lineno))
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700615 lab.reported = true
Keith Randalld2fd43a2015-04-15 15:51:25 -0700616 }
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700617 // The label might already have a target block via a goto.
618 if lab.target == nil {
619 lab.target = s.f.NewBlock(ssa.BlockPlain)
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -0700620 }
Keith Randalld2fd43a2015-04-15 15:51:25 -0700621
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700622 // go to that label (we pretend "label:" is preceded by "goto label")
623 b := s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -0500624 b.AddEdgeTo(lab.target)
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700625 s.startBlock(lab.target)
626
627 case OGOTO:
628 sym := n.Left.Sym
629
630 lab := s.label(sym)
631 if lab.target == nil {
632 lab.target = s.f.NewBlock(ssa.BlockPlain)
633 }
634 if !lab.used() {
635 lab.useNode = n
636 }
637
638 if lab.defined() {
639 s.checkgoto(n, lab.defNode)
640 } else {
641 s.fwdGotos = append(s.fwdGotos, n)
642 }
643
644 b := s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -0500645 b.AddEdgeTo(lab.target)
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700646
Keith Randall290d8fc2015-06-10 15:03:06 -0700647 case OAS, OASWB:
Josh Bleecher Snyder6b416652015-07-28 10:56:39 -0700648 // Check whether we can generate static data rather than code.
649 // If so, ignore n and defer data generation until codegen.
650 // Failure to do this causes writes to readonly symbols.
651 if gen_as_init(n, true) {
652 var data []*Node
653 if s.f.StaticData != nil {
654 data = s.f.StaticData.([]*Node)
655 }
656 s.f.StaticData = append(data, n)
657 return
658 }
Keith Randall5ba31942016-01-25 17:06:54 -0800659
Keith Randall309144b2016-04-01 11:05:30 -0700660 if n.Left == n.Right && n.Left.Op == ONAME {
661 // An x=x assignment. No point in doing anything
662 // here. In addition, skipping this assignment
663 // prevents generating:
664 // VARDEF x
665 // COPY x -> x
666 // which is bad because x is incorrectly considered
667 // dead before the vardef. See issue #14904.
668 return
669 }
670
Keith Randall5ba31942016-01-25 17:06:54 -0800671 var t *Type
Josh Bleecher Snyder07269312015-08-29 14:54:45 -0700672 if n.Right != nil {
Keith Randall5ba31942016-01-25 17:06:54 -0800673 t = n.Right.Type
674 } else {
675 t = n.Left.Type
676 }
677
678 // Evaluate RHS.
679 rhs := n.Right
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -0700680 if rhs != nil {
681 switch rhs.Op {
682 case OSTRUCTLIT, OARRAYLIT:
683 // All literals with nonzero fields have already been
684 // rewritten during walk. Any that remain are just T{}
685 // or equivalents. Use the zero value.
686 if !iszero(rhs) {
687 Fatalf("literal with nonzero value in SSA: %v", rhs)
688 }
689 rhs = nil
690 case OAPPEND:
691 // If we're writing the result of an append back to the same slice,
692 // handle it specially to avoid write barriers on the fast (non-growth) path.
693 // If the slice can be SSA'd, it'll be on the stack,
694 // so there will be no write barriers,
695 // so there's no need to attempt to prevent them.
Josh Bleecher Snyder03e216f2016-04-18 09:40:30 -0700696 if samesafeexpr(n.Left, rhs.List.First()) && !s.canSSA(n.Left) {
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -0700697 s.append(rhs, true)
698 return
699 }
Keith Randall5ba31942016-01-25 17:06:54 -0800700 }
Keith Randall5ba31942016-01-25 17:06:54 -0800701 }
702 var r *ssa.Value
703 needwb := n.Op == OASWB && rhs != nil
704 deref := !canSSAType(t)
705 if deref {
706 if rhs == nil {
707 r = nil // Signal assign to use OpZero.
Keith Randalld3886902015-09-18 22:12:38 -0700708 } else {
Keith Randall5ba31942016-01-25 17:06:54 -0800709 r = s.addr(rhs, false)
710 }
711 } else {
712 if rhs == nil {
713 r = s.zeroVal(t)
714 } else {
715 r = s.expr(rhs)
Keith Randalld3886902015-09-18 22:12:38 -0700716 }
Josh Bleecher Snyder07269312015-08-29 14:54:45 -0700717 }
Keith Randall5ba31942016-01-25 17:06:54 -0800718 if rhs != nil && rhs.Op == OAPPEND {
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -0700719 // The frontend gets rid of the write barrier to enable the special OAPPEND
720 // handling above, but since this is not a special case, we need it.
Keith Randall9d22c102015-09-11 11:02:57 -0700721 // TODO: just add a ptr graying to the end of growslice?
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -0700722 // TODO: check whether we need to provide special handling and a write barrier
723 // for ODOTTYPE and ORECV also.
Keith Randall9d22c102015-09-11 11:02:57 -0700724 // They get similar wb-removal treatment in walk.go:OAS.
Keith Randall5ba31942016-01-25 17:06:54 -0800725 needwb = true
Keith Randall9d22c102015-09-11 11:02:57 -0700726 }
Keith Randall5ba31942016-01-25 17:06:54 -0800727
Keith Randalld4663e12016-03-21 10:22:03 -0700728 var skip skipMask
729 if rhs != nil && (rhs.Op == OSLICE || rhs.Op == OSLICE3 || rhs.Op == OSLICESTR) && samesafeexpr(rhs.Left, n.Left) {
730 // We're assigning a slicing operation back to its source.
731 // Don't write back fields we aren't changing. See issue #14855.
Josh Bleecher Snyderf12bd8a2016-04-21 11:55:33 -0700732 i, j, k := rhs.SliceBounds()
Josh Bleecher Snyder5cab0162016-04-01 14:51:02 -0700733 if i != nil && (i.Op == OLITERAL && i.Val().Ctype() == CTINT && i.Int64() == 0) {
Keith Randalld4663e12016-03-21 10:22:03 -0700734 // [0:...] is the same as [:...]
735 i = nil
736 }
737 // TODO: detect defaults for len/cap also.
738 // Currently doesn't really work because (*p)[:len(*p)] appears here as:
739 // tmp = len(*p)
740 // (*p)[:tmp]
741 //if j != nil && (j.Op == OLEN && samesafeexpr(j.Left, n.Left)) {
742 // j = nil
743 //}
744 //if k != nil && (k.Op == OCAP && samesafeexpr(k.Left, n.Left)) {
745 // k = nil
746 //}
747 if i == nil {
748 skip |= skipPtr
749 if j == nil {
750 skip |= skipLen
751 }
752 if k == nil {
753 skip |= skipCap
754 }
755 }
756 }
757
758 s.assign(n.Left, r, needwb, deref, n.Lineno, skip)
Daniel Morsingc31b6dd2015-06-12 14:23:29 +0100759
Keith Randalld2fd43a2015-04-15 15:51:25 -0700760 case OIF:
Keith Randalld2fd43a2015-04-15 15:51:25 -0700761 bThen := s.f.NewBlock(ssa.BlockPlain)
762 bEnd := s.f.NewBlock(ssa.BlockPlain)
763 var bElse *ssa.Block
Ian Lance Taylor38921b32016-03-08 15:10:26 -0800764 if n.Rlist.Len() != 0 {
Keith Randalld2fd43a2015-04-15 15:51:25 -0700765 bElse = s.f.NewBlock(ssa.BlockPlain)
Keith Randall99187312015-11-02 16:56:53 -0800766 s.condBranch(n.Left, bThen, bElse, n.Likely)
767 } else {
768 s.condBranch(n.Left, bThen, bEnd, n.Likely)
Keith Randalld2fd43a2015-04-15 15:51:25 -0700769 }
770
771 s.startBlock(bThen)
Keith Randall9d854fd2016-03-01 12:50:17 -0800772 s.stmts(n.Nbody)
Josh Bleecher Snydere0ac5c52015-07-20 18:42:45 -0700773 if b := s.endBlock(); b != nil {
Todd Neal47d67992015-08-28 21:36:29 -0500774 b.AddEdgeTo(bEnd)
Keith Randalld2fd43a2015-04-15 15:51:25 -0700775 }
776
Ian Lance Taylor38921b32016-03-08 15:10:26 -0800777 if n.Rlist.Len() != 0 {
Keith Randalld2fd43a2015-04-15 15:51:25 -0700778 s.startBlock(bElse)
Keith Randalle707fbe2015-06-11 10:20:39 -0700779 s.stmtList(n.Rlist)
Josh Bleecher Snydere0ac5c52015-07-20 18:42:45 -0700780 if b := s.endBlock(); b != nil {
Todd Neal47d67992015-08-28 21:36:29 -0500781 b.AddEdgeTo(bEnd)
Keith Randalld2fd43a2015-04-15 15:51:25 -0700782 }
783 }
784 s.startBlock(bEnd)
785
786 case ORETURN:
787 s.stmtList(n.List)
Keith Randall6a8a9da2016-02-27 17:49:31 -0800788 s.exit()
Keith Randall8a1f6212015-09-08 21:28:44 -0700789 case ORETJMP:
790 s.stmtList(n.List)
Keith Randall6a8a9da2016-02-27 17:49:31 -0800791 b := s.exit()
792 b.Kind = ssa.BlockRetJmp // override BlockRet
Keith Randall8a1f6212015-09-08 21:28:44 -0700793 b.Aux = n.Left.Sym
Keith Randalld2fd43a2015-04-15 15:51:25 -0700794
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700795 case OCONTINUE, OBREAK:
796 var op string
797 var to *ssa.Block
798 switch n.Op {
799 case OCONTINUE:
800 op = "continue"
801 to = s.continueTo
802 case OBREAK:
803 op = "break"
804 to = s.breakTo
805 }
806 if n.Left == nil {
807 // plain break/continue
808 if to == nil {
809 s.Error("%s is not in a loop", op)
810 return
811 }
812 // nothing to do; "to" is already the correct target
813 } else {
814 // labeled break/continue; look up the target
815 sym := n.Left.Sym
816 lab := s.label(sym)
817 if !lab.used() {
818 lab.useNode = n.Left
819 }
820 if !lab.defined() {
821 s.Error("%s label not defined: %v", op, sym)
822 lab.reported = true
823 return
824 }
825 switch n.Op {
826 case OCONTINUE:
827 to = lab.continueTarget
828 case OBREAK:
829 to = lab.breakTarget
830 }
831 if to == nil {
832 // Valid label but not usable with a break/continue here, e.g.:
833 // for {
834 // continue abc
835 // }
836 // abc:
837 // for {}
838 s.Error("invalid %s label %v", op, sym)
839 lab.reported = true
840 return
841 }
842 }
843
844 b := s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -0500845 b.AddEdgeTo(to)
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700846
Keith Randalld2fd43a2015-04-15 15:51:25 -0700847 case OFOR:
Josh Bleecher Snyder51738682015-07-06 15:29:39 -0700848 // OFOR: for Ninit; Left; Right { Nbody }
Keith Randalld2fd43a2015-04-15 15:51:25 -0700849 bCond := s.f.NewBlock(ssa.BlockPlain)
850 bBody := s.f.NewBlock(ssa.BlockPlain)
Josh Bleecher Snyder51738682015-07-06 15:29:39 -0700851 bIncr := s.f.NewBlock(ssa.BlockPlain)
Keith Randalld2fd43a2015-04-15 15:51:25 -0700852 bEnd := s.f.NewBlock(ssa.BlockPlain)
853
854 // first, jump to condition test
855 b := s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -0500856 b.AddEdgeTo(bCond)
Keith Randalld2fd43a2015-04-15 15:51:25 -0700857
858 // generate code to test condition
Keith Randalld2fd43a2015-04-15 15:51:25 -0700859 s.startBlock(bCond)
Josh Bleecher Snyder51738682015-07-06 15:29:39 -0700860 if n.Left != nil {
Keith Randall99187312015-11-02 16:56:53 -0800861 s.condBranch(n.Left, bBody, bEnd, 1)
Josh Bleecher Snyder51738682015-07-06 15:29:39 -0700862 } else {
Keith Randall99187312015-11-02 16:56:53 -0800863 b := s.endBlock()
864 b.Kind = ssa.BlockPlain
865 b.AddEdgeTo(bBody)
Josh Bleecher Snyder51738682015-07-06 15:29:39 -0700866 }
Keith Randalld2fd43a2015-04-15 15:51:25 -0700867
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700868 // set up for continue/break in body
869 prevContinue := s.continueTo
870 prevBreak := s.breakTo
871 s.continueTo = bIncr
872 s.breakTo = bEnd
873 lab := s.labeledNodes[n]
874 if lab != nil {
875 // labeled for loop
876 lab.continueTarget = bIncr
877 lab.breakTarget = bEnd
878 }
879
Keith Randalld2fd43a2015-04-15 15:51:25 -0700880 // generate body
881 s.startBlock(bBody)
Keith Randall9d854fd2016-03-01 12:50:17 -0800882 s.stmts(n.Nbody)
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700883
884 // tear down continue/break
885 s.continueTo = prevContinue
886 s.breakTo = prevBreak
887 if lab != nil {
888 lab.continueTarget = nil
889 lab.breakTarget = nil
890 }
891
892 // done with body, goto incr
Josh Bleecher Snyder51738682015-07-06 15:29:39 -0700893 if b := s.endBlock(); b != nil {
Todd Neal47d67992015-08-28 21:36:29 -0500894 b.AddEdgeTo(bIncr)
Josh Bleecher Snyder51738682015-07-06 15:29:39 -0700895 }
896
897 // generate incr
898 s.startBlock(bIncr)
Josh Bleecher Snyder46815b92015-06-24 17:48:22 -0700899 if n.Right != nil {
900 s.stmt(n.Right)
901 }
Josh Bleecher Snyder51738682015-07-06 15:29:39 -0700902 if b := s.endBlock(); b != nil {
Todd Neal47d67992015-08-28 21:36:29 -0500903 b.AddEdgeTo(bCond)
Josh Bleecher Snyder6c140592015-07-04 09:07:54 -0700904 }
Keith Randalld2fd43a2015-04-15 15:51:25 -0700905 s.startBlock(bEnd)
906
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700907 case OSWITCH, OSELECT:
908 // These have been mostly rewritten by the front end into their Nbody fields.
909 // Our main task is to correctly hook up any break statements.
910 bEnd := s.f.NewBlock(ssa.BlockPlain)
911
912 prevBreak := s.breakTo
913 s.breakTo = bEnd
914 lab := s.labeledNodes[n]
915 if lab != nil {
916 // labeled
917 lab.breakTarget = bEnd
918 }
919
920 // generate body code
Keith Randall9d854fd2016-03-01 12:50:17 -0800921 s.stmts(n.Nbody)
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700922
923 s.breakTo = prevBreak
924 if lab != nil {
925 lab.breakTarget = nil
926 }
927
Keith Randalleb0cff92016-02-09 12:28:02 -0800928 // OSWITCH never falls through (s.curBlock == nil here).
929 // OSELECT does not fall through if we're calling selectgo.
930 // OSELECT does fall through if we're calling selectnb{send,recv}[2].
931 // In those latter cases, go to the code after the select.
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700932 if b := s.endBlock(); b != nil {
Todd Neal47d67992015-08-28 21:36:29 -0500933 b.AddEdgeTo(bEnd)
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -0700934 }
935 s.startBlock(bEnd)
936
Keith Randalld2fd43a2015-04-15 15:51:25 -0700937 case OVARKILL:
Keith Randalld2107fc2015-08-24 02:16:19 -0700938 // Insert a varkill op to record that a variable is no longer live.
939 // We only care about liveness info at call sites, so putting the
940 // varkill in the store chain is enough to keep it correctly ordered
941 // with respect to call ops.
Keith Randall6a8a9da2016-02-27 17:49:31 -0800942 if !s.canSSA(n.Left) {
Keith Randalld29e92b2015-09-19 12:01:39 -0700943 s.vars[&memVar] = s.newValue1A(ssa.OpVarKill, ssa.TypeMem, n.Left, s.mem())
944 }
Keith Randall9569b952015-08-28 22:51:01 -0700945
Keith Randall23d58102016-01-19 09:59:21 -0800946 case OVARLIVE:
947 // Insert a varlive op to record that a variable is still live.
948 if !n.Left.Addrtaken {
949 s.Fatalf("VARLIVE variable %s must have Addrtaken set", n.Left)
950 }
951 s.vars[&memVar] = s.newValue1A(ssa.OpVarLive, ssa.TypeMem, n.Left, s.mem())
952
Keith Randall46ffb022015-09-12 14:06:44 -0700953 case OCHECKNIL:
954 p := s.expr(n.Left)
955 s.nilCheck(p)
956
Keith Randalld2fd43a2015-04-15 15:51:25 -0700957 default:
Josh Bleecher Snyderf0272412016-04-22 07:14:10 -0700958 s.Unimplementedf("unhandled stmt %s", n.Op)
Keith Randalld2fd43a2015-04-15 15:51:25 -0700959 }
960}
961
Keith Randall6a8a9da2016-02-27 17:49:31 -0800962// exit processes any code that needs to be generated just before returning.
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +0000963// It returns a BlockRet block that ends the control flow. Its control value
Keith Randall6a8a9da2016-02-27 17:49:31 -0800964// will be set to the final memory state.
965func (s *state) exit() *ssa.Block {
Keith Randallddc6b642016-03-09 19:27:57 -0800966 if hasdefer {
967 s.rtcall(Deferreturn, true, nil)
968 }
969
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +0000970 // Run exit code. Typically, this code copies heap-allocated PPARAMOUT
Keith Randall6a8a9da2016-02-27 17:49:31 -0800971 // variables back to the stack.
972 s.stmts(s.exitCode)
973
974 // Store SSAable PPARAMOUT variables back to stack locations.
975 for _, n := range s.returns {
Keith Randall3572c642016-04-21 19:28:28 -0700976 addr := s.decladdrs[n]
Keith Randall6a8a9da2016-02-27 17:49:31 -0800977 val := s.variable(n, n.Type)
978 s.vars[&memVar] = s.newValue1A(ssa.OpVarDef, ssa.TypeMem, n, s.mem())
979 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, n.Type.Size(), addr, val, s.mem())
980 // TODO: if val is ever spilled, we'd like to use the
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +0000981 // PPARAMOUT slot for spilling it. That won't happen
Keith Randall6a8a9da2016-02-27 17:49:31 -0800982 // currently.
983 }
984
Keith Randall3572c642016-04-21 19:28:28 -0700985 // Keep input pointer args live until the return. This is a bandaid
986 // fix for 1.7 for what will become in 1.8 explicit runtime.KeepAlive calls.
987 // For <= 1.7 we guarantee that pointer input arguments live to the end of
988 // the function to prevent premature (from the user's point of view)
989 // execution of finalizers. See issue 15277.
990 // TODO: remove for 1.8?
991 for _, n := range s.ptrargs {
992 s.vars[&memVar] = s.newValue2(ssa.OpKeepAlive, ssa.TypeMem, s.variable(n, n.Type), s.mem())
993 }
994
Keith Randall6a8a9da2016-02-27 17:49:31 -0800995 // Do actual return.
996 m := s.mem()
997 b := s.endBlock()
998 b.Kind = ssa.BlockRet
Keith Randall56e0ecc2016-03-15 20:45:50 -0700999 b.SetControl(m)
Keith Randall6a8a9da2016-02-27 17:49:31 -08001000 return b
1001}
1002
Keith Randall67fdb0d2015-07-19 15:48:20 -07001003type opAndType struct {
Keith Randall4304fbc2015-11-16 13:20:16 -08001004 op Op
1005 etype EType
Keith Randall67fdb0d2015-07-19 15:48:20 -07001006}
1007
1008var opToSSA = map[opAndType]ssa.Op{
David Chase997a9f32015-08-12 16:38:11 -04001009 opAndType{OADD, TINT8}: ssa.OpAdd8,
1010 opAndType{OADD, TUINT8}: ssa.OpAdd8,
1011 opAndType{OADD, TINT16}: ssa.OpAdd16,
1012 opAndType{OADD, TUINT16}: ssa.OpAdd16,
1013 opAndType{OADD, TINT32}: ssa.OpAdd32,
1014 opAndType{OADD, TUINT32}: ssa.OpAdd32,
1015 opAndType{OADD, TPTR32}: ssa.OpAdd32,
1016 opAndType{OADD, TINT64}: ssa.OpAdd64,
1017 opAndType{OADD, TUINT64}: ssa.OpAdd64,
1018 opAndType{OADD, TPTR64}: ssa.OpAdd64,
1019 opAndType{OADD, TFLOAT32}: ssa.OpAdd32F,
1020 opAndType{OADD, TFLOAT64}: ssa.OpAdd64F,
Keith Randall67fdb0d2015-07-19 15:48:20 -07001021
David Chase997a9f32015-08-12 16:38:11 -04001022 opAndType{OSUB, TINT8}: ssa.OpSub8,
1023 opAndType{OSUB, TUINT8}: ssa.OpSub8,
1024 opAndType{OSUB, TINT16}: ssa.OpSub16,
1025 opAndType{OSUB, TUINT16}: ssa.OpSub16,
1026 opAndType{OSUB, TINT32}: ssa.OpSub32,
1027 opAndType{OSUB, TUINT32}: ssa.OpSub32,
1028 opAndType{OSUB, TINT64}: ssa.OpSub64,
1029 opAndType{OSUB, TUINT64}: ssa.OpSub64,
1030 opAndType{OSUB, TFLOAT32}: ssa.OpSub32F,
1031 opAndType{OSUB, TFLOAT64}: ssa.OpSub64F,
Keith Randall67fdb0d2015-07-19 15:48:20 -07001032
Josh Bleecher Snydere61e7c92015-07-22 19:19:40 -07001033 opAndType{ONOT, TBOOL}: ssa.OpNot,
1034
David Chase3a9d0ac2015-08-28 14:24:10 -04001035 opAndType{OMINUS, TINT8}: ssa.OpNeg8,
1036 opAndType{OMINUS, TUINT8}: ssa.OpNeg8,
1037 opAndType{OMINUS, TINT16}: ssa.OpNeg16,
1038 opAndType{OMINUS, TUINT16}: ssa.OpNeg16,
1039 opAndType{OMINUS, TINT32}: ssa.OpNeg32,
1040 opAndType{OMINUS, TUINT32}: ssa.OpNeg32,
1041 opAndType{OMINUS, TINT64}: ssa.OpNeg64,
1042 opAndType{OMINUS, TUINT64}: ssa.OpNeg64,
1043 opAndType{OMINUS, TFLOAT32}: ssa.OpNeg32F,
1044 opAndType{OMINUS, TFLOAT64}: ssa.OpNeg64F,
Alexandru Moșoi954d5ad2015-07-21 16:58:18 +02001045
Keith Randall4b803152015-07-29 17:07:09 -07001046 opAndType{OCOM, TINT8}: ssa.OpCom8,
1047 opAndType{OCOM, TUINT8}: ssa.OpCom8,
1048 opAndType{OCOM, TINT16}: ssa.OpCom16,
1049 opAndType{OCOM, TUINT16}: ssa.OpCom16,
1050 opAndType{OCOM, TINT32}: ssa.OpCom32,
1051 opAndType{OCOM, TUINT32}: ssa.OpCom32,
1052 opAndType{OCOM, TINT64}: ssa.OpCom64,
1053 opAndType{OCOM, TUINT64}: ssa.OpCom64,
1054
Josh Bleecher Snyderfa5fe192015-09-06 19:24:59 -07001055 opAndType{OIMAG, TCOMPLEX64}: ssa.OpComplexImag,
1056 opAndType{OIMAG, TCOMPLEX128}: ssa.OpComplexImag,
1057 opAndType{OREAL, TCOMPLEX64}: ssa.OpComplexReal,
1058 opAndType{OREAL, TCOMPLEX128}: ssa.OpComplexReal,
1059
David Chase997a9f32015-08-12 16:38:11 -04001060 opAndType{OMUL, TINT8}: ssa.OpMul8,
1061 opAndType{OMUL, TUINT8}: ssa.OpMul8,
1062 opAndType{OMUL, TINT16}: ssa.OpMul16,
1063 opAndType{OMUL, TUINT16}: ssa.OpMul16,
1064 opAndType{OMUL, TINT32}: ssa.OpMul32,
1065 opAndType{OMUL, TUINT32}: ssa.OpMul32,
1066 opAndType{OMUL, TINT64}: ssa.OpMul64,
1067 opAndType{OMUL, TUINT64}: ssa.OpMul64,
1068 opAndType{OMUL, TFLOAT32}: ssa.OpMul32F,
1069 opAndType{OMUL, TFLOAT64}: ssa.OpMul64F,
1070
1071 opAndType{ODIV, TFLOAT32}: ssa.OpDiv32F,
1072 opAndType{ODIV, TFLOAT64}: ssa.OpDiv64F,
Keith Randallbe1eb572015-07-22 13:46:15 -07001073
Todd Neal67cbd5b2015-08-18 19:14:47 -05001074 opAndType{OHMUL, TINT8}: ssa.OpHmul8,
1075 opAndType{OHMUL, TUINT8}: ssa.OpHmul8u,
1076 opAndType{OHMUL, TINT16}: ssa.OpHmul16,
1077 opAndType{OHMUL, TUINT16}: ssa.OpHmul16u,
1078 opAndType{OHMUL, TINT32}: ssa.OpHmul32,
1079 opAndType{OHMUL, TUINT32}: ssa.OpHmul32u,
1080
Todd Neala45f2d82015-08-17 17:46:06 -05001081 opAndType{ODIV, TINT8}: ssa.OpDiv8,
1082 opAndType{ODIV, TUINT8}: ssa.OpDiv8u,
1083 opAndType{ODIV, TINT16}: ssa.OpDiv16,
1084 opAndType{ODIV, TUINT16}: ssa.OpDiv16u,
1085 opAndType{ODIV, TINT32}: ssa.OpDiv32,
1086 opAndType{ODIV, TUINT32}: ssa.OpDiv32u,
1087 opAndType{ODIV, TINT64}: ssa.OpDiv64,
1088 opAndType{ODIV, TUINT64}: ssa.OpDiv64u,
1089
Todd Neal57d9e7e2015-08-18 19:51:44 -05001090 opAndType{OMOD, TINT8}: ssa.OpMod8,
1091 opAndType{OMOD, TUINT8}: ssa.OpMod8u,
1092 opAndType{OMOD, TINT16}: ssa.OpMod16,
1093 opAndType{OMOD, TUINT16}: ssa.OpMod16u,
1094 opAndType{OMOD, TINT32}: ssa.OpMod32,
1095 opAndType{OMOD, TUINT32}: ssa.OpMod32u,
1096 opAndType{OMOD, TINT64}: ssa.OpMod64,
1097 opAndType{OMOD, TUINT64}: ssa.OpMod64u,
1098
Alexandru Moșoiedff8812015-07-28 14:58:49 +02001099 opAndType{OAND, TINT8}: ssa.OpAnd8,
Keith Randall2a5e6c42015-07-23 14:35:02 -07001100 opAndType{OAND, TUINT8}: ssa.OpAnd8,
Alexandru Moșoiedff8812015-07-28 14:58:49 +02001101 opAndType{OAND, TINT16}: ssa.OpAnd16,
Keith Randall2a5e6c42015-07-23 14:35:02 -07001102 opAndType{OAND, TUINT16}: ssa.OpAnd16,
Alexandru Moșoiedff8812015-07-28 14:58:49 +02001103 opAndType{OAND, TINT32}: ssa.OpAnd32,
Keith Randall2a5e6c42015-07-23 14:35:02 -07001104 opAndType{OAND, TUINT32}: ssa.OpAnd32,
Alexandru Moșoiedff8812015-07-28 14:58:49 +02001105 opAndType{OAND, TINT64}: ssa.OpAnd64,
Keith Randall2a5e6c42015-07-23 14:35:02 -07001106 opAndType{OAND, TUINT64}: ssa.OpAnd64,
Alexandru Moșoiedff8812015-07-28 14:58:49 +02001107
Alexandru Moșoi74024162015-07-29 17:52:25 +02001108 opAndType{OOR, TINT8}: ssa.OpOr8,
1109 opAndType{OOR, TUINT8}: ssa.OpOr8,
1110 opAndType{OOR, TINT16}: ssa.OpOr16,
1111 opAndType{OOR, TUINT16}: ssa.OpOr16,
1112 opAndType{OOR, TINT32}: ssa.OpOr32,
1113 opAndType{OOR, TUINT32}: ssa.OpOr32,
1114 opAndType{OOR, TINT64}: ssa.OpOr64,
1115 opAndType{OOR, TUINT64}: ssa.OpOr64,
1116
Alexandru Moșoi6d9362a12015-07-30 12:33:36 +02001117 opAndType{OXOR, TINT8}: ssa.OpXor8,
1118 opAndType{OXOR, TUINT8}: ssa.OpXor8,
1119 opAndType{OXOR, TINT16}: ssa.OpXor16,
1120 opAndType{OXOR, TUINT16}: ssa.OpXor16,
1121 opAndType{OXOR, TINT32}: ssa.OpXor32,
1122 opAndType{OXOR, TUINT32}: ssa.OpXor32,
1123 opAndType{OXOR, TINT64}: ssa.OpXor64,
1124 opAndType{OXOR, TUINT64}: ssa.OpXor64,
1125
Alexandru Moșoi8b923972016-04-24 21:21:07 +02001126 opAndType{OEQ, TBOOL}: ssa.OpEqB,
Josh Bleecher Snyder1bab5b92015-07-28 14:14:25 -07001127 opAndType{OEQ, TINT8}: ssa.OpEq8,
1128 opAndType{OEQ, TUINT8}: ssa.OpEq8,
1129 opAndType{OEQ, TINT16}: ssa.OpEq16,
1130 opAndType{OEQ, TUINT16}: ssa.OpEq16,
1131 opAndType{OEQ, TINT32}: ssa.OpEq32,
1132 opAndType{OEQ, TUINT32}: ssa.OpEq32,
1133 opAndType{OEQ, TINT64}: ssa.OpEq64,
1134 opAndType{OEQ, TUINT64}: ssa.OpEq64,
Keith Randall1e4ebfd2015-09-10 13:53:27 -07001135 opAndType{OEQ, TINTER}: ssa.OpEqInter,
Matthew Dempsky40f1d0c2016-04-18 14:02:08 -07001136 opAndType{OEQ, TSLICE}: ssa.OpEqSlice,
Josh Bleecher Snyder1bab5b92015-07-28 14:14:25 -07001137 opAndType{OEQ, TFUNC}: ssa.OpEqPtr,
1138 opAndType{OEQ, TMAP}: ssa.OpEqPtr,
1139 opAndType{OEQ, TCHAN}: ssa.OpEqPtr,
Todd Neal5fdd4fe2015-08-30 20:47:26 -05001140 opAndType{OEQ, TPTR64}: ssa.OpEqPtr,
Josh Bleecher Snyder1bab5b92015-07-28 14:14:25 -07001141 opAndType{OEQ, TUINTPTR}: ssa.OpEqPtr,
1142 opAndType{OEQ, TUNSAFEPTR}: ssa.OpEqPtr,
David Chase8e601b22015-08-18 14:39:26 -04001143 opAndType{OEQ, TFLOAT64}: ssa.OpEq64F,
1144 opAndType{OEQ, TFLOAT32}: ssa.OpEq32F,
Keith Randall67fdb0d2015-07-19 15:48:20 -07001145
Alexandru Moșoi8b923972016-04-24 21:21:07 +02001146 opAndType{ONE, TBOOL}: ssa.OpNeqB,
Josh Bleecher Snyder1bab5b92015-07-28 14:14:25 -07001147 opAndType{ONE, TINT8}: ssa.OpNeq8,
1148 opAndType{ONE, TUINT8}: ssa.OpNeq8,
1149 opAndType{ONE, TINT16}: ssa.OpNeq16,
1150 opAndType{ONE, TUINT16}: ssa.OpNeq16,
1151 opAndType{ONE, TINT32}: ssa.OpNeq32,
1152 opAndType{ONE, TUINT32}: ssa.OpNeq32,
1153 opAndType{ONE, TINT64}: ssa.OpNeq64,
1154 opAndType{ONE, TUINT64}: ssa.OpNeq64,
Keith Randall1e4ebfd2015-09-10 13:53:27 -07001155 opAndType{ONE, TINTER}: ssa.OpNeqInter,
Matthew Dempsky40f1d0c2016-04-18 14:02:08 -07001156 opAndType{ONE, TSLICE}: ssa.OpNeqSlice,
Josh Bleecher Snyder1bab5b92015-07-28 14:14:25 -07001157 opAndType{ONE, TFUNC}: ssa.OpNeqPtr,
1158 opAndType{ONE, TMAP}: ssa.OpNeqPtr,
1159 opAndType{ONE, TCHAN}: ssa.OpNeqPtr,
Todd Neal5fdd4fe2015-08-30 20:47:26 -05001160 opAndType{ONE, TPTR64}: ssa.OpNeqPtr,
Josh Bleecher Snyder1bab5b92015-07-28 14:14:25 -07001161 opAndType{ONE, TUINTPTR}: ssa.OpNeqPtr,
1162 opAndType{ONE, TUNSAFEPTR}: ssa.OpNeqPtr,
David Chase8e601b22015-08-18 14:39:26 -04001163 opAndType{ONE, TFLOAT64}: ssa.OpNeq64F,
1164 opAndType{ONE, TFLOAT32}: ssa.OpNeq32F,
Keith Randall67fdb0d2015-07-19 15:48:20 -07001165
David Chase8e601b22015-08-18 14:39:26 -04001166 opAndType{OLT, TINT8}: ssa.OpLess8,
1167 opAndType{OLT, TUINT8}: ssa.OpLess8U,
1168 opAndType{OLT, TINT16}: ssa.OpLess16,
1169 opAndType{OLT, TUINT16}: ssa.OpLess16U,
1170 opAndType{OLT, TINT32}: ssa.OpLess32,
1171 opAndType{OLT, TUINT32}: ssa.OpLess32U,
1172 opAndType{OLT, TINT64}: ssa.OpLess64,
1173 opAndType{OLT, TUINT64}: ssa.OpLess64U,
1174 opAndType{OLT, TFLOAT64}: ssa.OpLess64F,
1175 opAndType{OLT, TFLOAT32}: ssa.OpLess32F,
Keith Randall67fdb0d2015-07-19 15:48:20 -07001176
David Chase8e601b22015-08-18 14:39:26 -04001177 opAndType{OGT, TINT8}: ssa.OpGreater8,
1178 opAndType{OGT, TUINT8}: ssa.OpGreater8U,
1179 opAndType{OGT, TINT16}: ssa.OpGreater16,
1180 opAndType{OGT, TUINT16}: ssa.OpGreater16U,
1181 opAndType{OGT, TINT32}: ssa.OpGreater32,
1182 opAndType{OGT, TUINT32}: ssa.OpGreater32U,
1183 opAndType{OGT, TINT64}: ssa.OpGreater64,
1184 opAndType{OGT, TUINT64}: ssa.OpGreater64U,
1185 opAndType{OGT, TFLOAT64}: ssa.OpGreater64F,
1186 opAndType{OGT, TFLOAT32}: ssa.OpGreater32F,
Keith Randall67fdb0d2015-07-19 15:48:20 -07001187
David Chase8e601b22015-08-18 14:39:26 -04001188 opAndType{OLE, TINT8}: ssa.OpLeq8,
1189 opAndType{OLE, TUINT8}: ssa.OpLeq8U,
1190 opAndType{OLE, TINT16}: ssa.OpLeq16,
1191 opAndType{OLE, TUINT16}: ssa.OpLeq16U,
1192 opAndType{OLE, TINT32}: ssa.OpLeq32,
1193 opAndType{OLE, TUINT32}: ssa.OpLeq32U,
1194 opAndType{OLE, TINT64}: ssa.OpLeq64,
1195 opAndType{OLE, TUINT64}: ssa.OpLeq64U,
1196 opAndType{OLE, TFLOAT64}: ssa.OpLeq64F,
1197 opAndType{OLE, TFLOAT32}: ssa.OpLeq32F,
Keith Randall67fdb0d2015-07-19 15:48:20 -07001198
David Chase8e601b22015-08-18 14:39:26 -04001199 opAndType{OGE, TINT8}: ssa.OpGeq8,
1200 opAndType{OGE, TUINT8}: ssa.OpGeq8U,
1201 opAndType{OGE, TINT16}: ssa.OpGeq16,
1202 opAndType{OGE, TUINT16}: ssa.OpGeq16U,
1203 opAndType{OGE, TINT32}: ssa.OpGeq32,
1204 opAndType{OGE, TUINT32}: ssa.OpGeq32U,
1205 opAndType{OGE, TINT64}: ssa.OpGeq64,
1206 opAndType{OGE, TUINT64}: ssa.OpGeq64U,
1207 opAndType{OGE, TFLOAT64}: ssa.OpGeq64F,
1208 opAndType{OGE, TFLOAT32}: ssa.OpGeq32F,
David Chase40aba8c2015-08-05 22:11:14 -04001209
1210 opAndType{OLROT, TUINT8}: ssa.OpLrot8,
1211 opAndType{OLROT, TUINT16}: ssa.OpLrot16,
1212 opAndType{OLROT, TUINT32}: ssa.OpLrot32,
1213 opAndType{OLROT, TUINT64}: ssa.OpLrot64,
Keith Randalla329e212015-09-12 13:26:57 -07001214
1215 opAndType{OSQRT, TFLOAT64}: ssa.OpSqrt,
Keith Randall67fdb0d2015-07-19 15:48:20 -07001216}
1217
Keith Randall4304fbc2015-11-16 13:20:16 -08001218func (s *state) concreteEtype(t *Type) EType {
Keith Randall2a5e6c42015-07-23 14:35:02 -07001219 e := t.Etype
1220 switch e {
1221 default:
1222 return e
Keith Randall67fdb0d2015-07-19 15:48:20 -07001223 case TINT:
Keith Randall2a5e6c42015-07-23 14:35:02 -07001224 if s.config.IntSize == 8 {
1225 return TINT64
Keith Randall67fdb0d2015-07-19 15:48:20 -07001226 }
Keith Randall2a5e6c42015-07-23 14:35:02 -07001227 return TINT32
Keith Randall67fdb0d2015-07-19 15:48:20 -07001228 case TUINT:
Keith Randall2a5e6c42015-07-23 14:35:02 -07001229 if s.config.IntSize == 8 {
1230 return TUINT64
Keith Randall67fdb0d2015-07-19 15:48:20 -07001231 }
Keith Randall2a5e6c42015-07-23 14:35:02 -07001232 return TUINT32
1233 case TUINTPTR:
1234 if s.config.PtrSize == 8 {
1235 return TUINT64
1236 }
1237 return TUINT32
Keith Randall67fdb0d2015-07-19 15:48:20 -07001238 }
Keith Randall2a5e6c42015-07-23 14:35:02 -07001239}
1240
Keith Randall4304fbc2015-11-16 13:20:16 -08001241func (s *state) ssaOp(op Op, t *Type) ssa.Op {
Keith Randall2a5e6c42015-07-23 14:35:02 -07001242 etype := s.concreteEtype(t)
Keith Randall67fdb0d2015-07-19 15:48:20 -07001243 x, ok := opToSSA[opAndType{op, etype}]
1244 if !ok {
Josh Bleecher Snyderfca0f332016-04-22 08:39:56 -07001245 s.Unimplementedf("unhandled binary op %s %s", op, etype)
Keith Randall67fdb0d2015-07-19 15:48:20 -07001246 }
1247 return x
Josh Bleecher Snyder46815b92015-06-24 17:48:22 -07001248}
1249
David Chase3a9d0ac2015-08-28 14:24:10 -04001250func floatForComplex(t *Type) *Type {
1251 if t.Size() == 8 {
1252 return Types[TFLOAT32]
1253 } else {
1254 return Types[TFLOAT64]
1255 }
1256}
1257
Keith Randall4b803152015-07-29 17:07:09 -07001258type opAndTwoTypes struct {
Keith Randall4304fbc2015-11-16 13:20:16 -08001259 op Op
1260 etype1 EType
1261 etype2 EType
Keith Randall4b803152015-07-29 17:07:09 -07001262}
1263
David Chased052bbd2015-09-01 17:09:00 -04001264type twoTypes struct {
Keith Randall4304fbc2015-11-16 13:20:16 -08001265 etype1 EType
1266 etype2 EType
David Chased052bbd2015-09-01 17:09:00 -04001267}
1268
1269type twoOpsAndType struct {
1270 op1 ssa.Op
1271 op2 ssa.Op
Keith Randall4304fbc2015-11-16 13:20:16 -08001272 intermediateType EType
David Chased052bbd2015-09-01 17:09:00 -04001273}
1274
1275var fpConvOpToSSA = map[twoTypes]twoOpsAndType{
1276
1277 twoTypes{TINT8, TFLOAT32}: twoOpsAndType{ssa.OpSignExt8to32, ssa.OpCvt32to32F, TINT32},
1278 twoTypes{TINT16, TFLOAT32}: twoOpsAndType{ssa.OpSignExt16to32, ssa.OpCvt32to32F, TINT32},
1279 twoTypes{TINT32, TFLOAT32}: twoOpsAndType{ssa.OpCopy, ssa.OpCvt32to32F, TINT32},
1280 twoTypes{TINT64, TFLOAT32}: twoOpsAndType{ssa.OpCopy, ssa.OpCvt64to32F, TINT64},
1281
1282 twoTypes{TINT8, TFLOAT64}: twoOpsAndType{ssa.OpSignExt8to32, ssa.OpCvt32to64F, TINT32},
1283 twoTypes{TINT16, TFLOAT64}: twoOpsAndType{ssa.OpSignExt16to32, ssa.OpCvt32to64F, TINT32},
1284 twoTypes{TINT32, TFLOAT64}: twoOpsAndType{ssa.OpCopy, ssa.OpCvt32to64F, TINT32},
1285 twoTypes{TINT64, TFLOAT64}: twoOpsAndType{ssa.OpCopy, ssa.OpCvt64to64F, TINT64},
1286
1287 twoTypes{TFLOAT32, TINT8}: twoOpsAndType{ssa.OpCvt32Fto32, ssa.OpTrunc32to8, TINT32},
1288 twoTypes{TFLOAT32, TINT16}: twoOpsAndType{ssa.OpCvt32Fto32, ssa.OpTrunc32to16, TINT32},
1289 twoTypes{TFLOAT32, TINT32}: twoOpsAndType{ssa.OpCvt32Fto32, ssa.OpCopy, TINT32},
1290 twoTypes{TFLOAT32, TINT64}: twoOpsAndType{ssa.OpCvt32Fto64, ssa.OpCopy, TINT64},
1291
1292 twoTypes{TFLOAT64, TINT8}: twoOpsAndType{ssa.OpCvt64Fto32, ssa.OpTrunc32to8, TINT32},
1293 twoTypes{TFLOAT64, TINT16}: twoOpsAndType{ssa.OpCvt64Fto32, ssa.OpTrunc32to16, TINT32},
1294 twoTypes{TFLOAT64, TINT32}: twoOpsAndType{ssa.OpCvt64Fto32, ssa.OpCopy, TINT32},
1295 twoTypes{TFLOAT64, TINT64}: twoOpsAndType{ssa.OpCvt64Fto64, ssa.OpCopy, TINT64},
1296 // unsigned
1297 twoTypes{TUINT8, TFLOAT32}: twoOpsAndType{ssa.OpZeroExt8to32, ssa.OpCvt32to32F, TINT32},
1298 twoTypes{TUINT16, TFLOAT32}: twoOpsAndType{ssa.OpZeroExt16to32, ssa.OpCvt32to32F, TINT32},
1299 twoTypes{TUINT32, TFLOAT32}: twoOpsAndType{ssa.OpZeroExt32to64, ssa.OpCvt64to32F, TINT64}, // go wide to dodge unsigned
1300 twoTypes{TUINT64, TFLOAT32}: twoOpsAndType{ssa.OpCopy, ssa.OpInvalid, TUINT64}, // Cvt64Uto32F, branchy code expansion instead
1301
1302 twoTypes{TUINT8, TFLOAT64}: twoOpsAndType{ssa.OpZeroExt8to32, ssa.OpCvt32to64F, TINT32},
1303 twoTypes{TUINT16, TFLOAT64}: twoOpsAndType{ssa.OpZeroExt16to32, ssa.OpCvt32to64F, TINT32},
1304 twoTypes{TUINT32, TFLOAT64}: twoOpsAndType{ssa.OpZeroExt32to64, ssa.OpCvt64to64F, TINT64}, // go wide to dodge unsigned
1305 twoTypes{TUINT64, TFLOAT64}: twoOpsAndType{ssa.OpCopy, ssa.OpInvalid, TUINT64}, // Cvt64Uto64F, branchy code expansion instead
1306
1307 twoTypes{TFLOAT32, TUINT8}: twoOpsAndType{ssa.OpCvt32Fto32, ssa.OpTrunc32to8, TINT32},
1308 twoTypes{TFLOAT32, TUINT16}: twoOpsAndType{ssa.OpCvt32Fto32, ssa.OpTrunc32to16, TINT32},
1309 twoTypes{TFLOAT32, TUINT32}: twoOpsAndType{ssa.OpCvt32Fto64, ssa.OpTrunc64to32, TINT64}, // go wide to dodge unsigned
1310 twoTypes{TFLOAT32, TUINT64}: twoOpsAndType{ssa.OpInvalid, ssa.OpCopy, TUINT64}, // Cvt32Fto64U, branchy code expansion instead
1311
1312 twoTypes{TFLOAT64, TUINT8}: twoOpsAndType{ssa.OpCvt64Fto32, ssa.OpTrunc32to8, TINT32},
1313 twoTypes{TFLOAT64, TUINT16}: twoOpsAndType{ssa.OpCvt64Fto32, ssa.OpTrunc32to16, TINT32},
1314 twoTypes{TFLOAT64, TUINT32}: twoOpsAndType{ssa.OpCvt64Fto64, ssa.OpTrunc64to32, TINT64}, // go wide to dodge unsigned
1315 twoTypes{TFLOAT64, TUINT64}: twoOpsAndType{ssa.OpInvalid, ssa.OpCopy, TUINT64}, // Cvt64Fto64U, branchy code expansion instead
1316
1317 // float
1318 twoTypes{TFLOAT64, TFLOAT32}: twoOpsAndType{ssa.OpCvt64Fto32F, ssa.OpCopy, TFLOAT32},
1319 twoTypes{TFLOAT64, TFLOAT64}: twoOpsAndType{ssa.OpCopy, ssa.OpCopy, TFLOAT64},
1320 twoTypes{TFLOAT32, TFLOAT32}: twoOpsAndType{ssa.OpCopy, ssa.OpCopy, TFLOAT32},
1321 twoTypes{TFLOAT32, TFLOAT64}: twoOpsAndType{ssa.OpCvt32Fto64F, ssa.OpCopy, TFLOAT64},
1322}
1323
Keith Randall4b803152015-07-29 17:07:09 -07001324var shiftOpToSSA = map[opAndTwoTypes]ssa.Op{
1325 opAndTwoTypes{OLSH, TINT8, TUINT8}: ssa.OpLsh8x8,
1326 opAndTwoTypes{OLSH, TUINT8, TUINT8}: ssa.OpLsh8x8,
1327 opAndTwoTypes{OLSH, TINT8, TUINT16}: ssa.OpLsh8x16,
1328 opAndTwoTypes{OLSH, TUINT8, TUINT16}: ssa.OpLsh8x16,
1329 opAndTwoTypes{OLSH, TINT8, TUINT32}: ssa.OpLsh8x32,
1330 opAndTwoTypes{OLSH, TUINT8, TUINT32}: ssa.OpLsh8x32,
1331 opAndTwoTypes{OLSH, TINT8, TUINT64}: ssa.OpLsh8x64,
1332 opAndTwoTypes{OLSH, TUINT8, TUINT64}: ssa.OpLsh8x64,
1333
1334 opAndTwoTypes{OLSH, TINT16, TUINT8}: ssa.OpLsh16x8,
1335 opAndTwoTypes{OLSH, TUINT16, TUINT8}: ssa.OpLsh16x8,
1336 opAndTwoTypes{OLSH, TINT16, TUINT16}: ssa.OpLsh16x16,
1337 opAndTwoTypes{OLSH, TUINT16, TUINT16}: ssa.OpLsh16x16,
1338 opAndTwoTypes{OLSH, TINT16, TUINT32}: ssa.OpLsh16x32,
1339 opAndTwoTypes{OLSH, TUINT16, TUINT32}: ssa.OpLsh16x32,
1340 opAndTwoTypes{OLSH, TINT16, TUINT64}: ssa.OpLsh16x64,
1341 opAndTwoTypes{OLSH, TUINT16, TUINT64}: ssa.OpLsh16x64,
1342
1343 opAndTwoTypes{OLSH, TINT32, TUINT8}: ssa.OpLsh32x8,
1344 opAndTwoTypes{OLSH, TUINT32, TUINT8}: ssa.OpLsh32x8,
1345 opAndTwoTypes{OLSH, TINT32, TUINT16}: ssa.OpLsh32x16,
1346 opAndTwoTypes{OLSH, TUINT32, TUINT16}: ssa.OpLsh32x16,
1347 opAndTwoTypes{OLSH, TINT32, TUINT32}: ssa.OpLsh32x32,
1348 opAndTwoTypes{OLSH, TUINT32, TUINT32}: ssa.OpLsh32x32,
1349 opAndTwoTypes{OLSH, TINT32, TUINT64}: ssa.OpLsh32x64,
1350 opAndTwoTypes{OLSH, TUINT32, TUINT64}: ssa.OpLsh32x64,
1351
1352 opAndTwoTypes{OLSH, TINT64, TUINT8}: ssa.OpLsh64x8,
1353 opAndTwoTypes{OLSH, TUINT64, TUINT8}: ssa.OpLsh64x8,
1354 opAndTwoTypes{OLSH, TINT64, TUINT16}: ssa.OpLsh64x16,
1355 opAndTwoTypes{OLSH, TUINT64, TUINT16}: ssa.OpLsh64x16,
1356 opAndTwoTypes{OLSH, TINT64, TUINT32}: ssa.OpLsh64x32,
1357 opAndTwoTypes{OLSH, TUINT64, TUINT32}: ssa.OpLsh64x32,
1358 opAndTwoTypes{OLSH, TINT64, TUINT64}: ssa.OpLsh64x64,
1359 opAndTwoTypes{OLSH, TUINT64, TUINT64}: ssa.OpLsh64x64,
1360
1361 opAndTwoTypes{ORSH, TINT8, TUINT8}: ssa.OpRsh8x8,
1362 opAndTwoTypes{ORSH, TUINT8, TUINT8}: ssa.OpRsh8Ux8,
1363 opAndTwoTypes{ORSH, TINT8, TUINT16}: ssa.OpRsh8x16,
1364 opAndTwoTypes{ORSH, TUINT8, TUINT16}: ssa.OpRsh8Ux16,
1365 opAndTwoTypes{ORSH, TINT8, TUINT32}: ssa.OpRsh8x32,
1366 opAndTwoTypes{ORSH, TUINT8, TUINT32}: ssa.OpRsh8Ux32,
1367 opAndTwoTypes{ORSH, TINT8, TUINT64}: ssa.OpRsh8x64,
1368 opAndTwoTypes{ORSH, TUINT8, TUINT64}: ssa.OpRsh8Ux64,
1369
1370 opAndTwoTypes{ORSH, TINT16, TUINT8}: ssa.OpRsh16x8,
1371 opAndTwoTypes{ORSH, TUINT16, TUINT8}: ssa.OpRsh16Ux8,
1372 opAndTwoTypes{ORSH, TINT16, TUINT16}: ssa.OpRsh16x16,
1373 opAndTwoTypes{ORSH, TUINT16, TUINT16}: ssa.OpRsh16Ux16,
1374 opAndTwoTypes{ORSH, TINT16, TUINT32}: ssa.OpRsh16x32,
1375 opAndTwoTypes{ORSH, TUINT16, TUINT32}: ssa.OpRsh16Ux32,
1376 opAndTwoTypes{ORSH, TINT16, TUINT64}: ssa.OpRsh16x64,
1377 opAndTwoTypes{ORSH, TUINT16, TUINT64}: ssa.OpRsh16Ux64,
1378
1379 opAndTwoTypes{ORSH, TINT32, TUINT8}: ssa.OpRsh32x8,
1380 opAndTwoTypes{ORSH, TUINT32, TUINT8}: ssa.OpRsh32Ux8,
1381 opAndTwoTypes{ORSH, TINT32, TUINT16}: ssa.OpRsh32x16,
1382 opAndTwoTypes{ORSH, TUINT32, TUINT16}: ssa.OpRsh32Ux16,
1383 opAndTwoTypes{ORSH, TINT32, TUINT32}: ssa.OpRsh32x32,
1384 opAndTwoTypes{ORSH, TUINT32, TUINT32}: ssa.OpRsh32Ux32,
1385 opAndTwoTypes{ORSH, TINT32, TUINT64}: ssa.OpRsh32x64,
1386 opAndTwoTypes{ORSH, TUINT32, TUINT64}: ssa.OpRsh32Ux64,
1387
1388 opAndTwoTypes{ORSH, TINT64, TUINT8}: ssa.OpRsh64x8,
1389 opAndTwoTypes{ORSH, TUINT64, TUINT8}: ssa.OpRsh64Ux8,
1390 opAndTwoTypes{ORSH, TINT64, TUINT16}: ssa.OpRsh64x16,
1391 opAndTwoTypes{ORSH, TUINT64, TUINT16}: ssa.OpRsh64Ux16,
1392 opAndTwoTypes{ORSH, TINT64, TUINT32}: ssa.OpRsh64x32,
1393 opAndTwoTypes{ORSH, TUINT64, TUINT32}: ssa.OpRsh64Ux32,
1394 opAndTwoTypes{ORSH, TINT64, TUINT64}: ssa.OpRsh64x64,
1395 opAndTwoTypes{ORSH, TUINT64, TUINT64}: ssa.OpRsh64Ux64,
1396}
1397
Keith Randall4304fbc2015-11-16 13:20:16 -08001398func (s *state) ssaShiftOp(op Op, t *Type, u *Type) ssa.Op {
Keith Randall4b803152015-07-29 17:07:09 -07001399 etype1 := s.concreteEtype(t)
1400 etype2 := s.concreteEtype(u)
1401 x, ok := shiftOpToSSA[opAndTwoTypes{op, etype1, etype2}]
1402 if !ok {
Josh Bleecher Snyderfca0f332016-04-22 08:39:56 -07001403 s.Unimplementedf("unhandled shift op %s etype=%s/%s", op, etype1, etype2)
Keith Randall4b803152015-07-29 17:07:09 -07001404 }
1405 return x
1406}
1407
Keith Randall4304fbc2015-11-16 13:20:16 -08001408func (s *state) ssaRotateOp(op Op, t *Type) ssa.Op {
David Chase40aba8c2015-08-05 22:11:14 -04001409 etype1 := s.concreteEtype(t)
1410 x, ok := opToSSA[opAndType{op, etype1}]
1411 if !ok {
Josh Bleecher Snyderfca0f332016-04-22 08:39:56 -07001412 s.Unimplementedf("unhandled rotate op %s etype=%s", op, etype1)
David Chase40aba8c2015-08-05 22:11:14 -04001413 }
1414 return x
1415}
1416
Keith Randalld2fd43a2015-04-15 15:51:25 -07001417// expr converts the expression n to ssa, adds it to s and returns the ssa result.
Keith Randallcfc2aa52015-05-18 16:44:20 -07001418func (s *state) expr(n *Node) *ssa.Value {
Keith Randalla19e60b2016-04-26 15:22:33 -07001419 if !(n.Op == ONAME || n.Op == OLITERAL && n.Sym != nil) {
1420 // ONAMEs and named OLITERALs have the line number
1421 // of the decl, not the use. See issue 14742.
1422 s.pushLine(n.Lineno)
1423 defer s.popLine()
1424 }
Michael Matloob81ccf502015-05-30 01:03:06 -04001425
Keith Randall06f32922015-07-11 11:39:12 -07001426 s.stmtList(n.Ninit)
Keith Randalld2fd43a2015-04-15 15:51:25 -07001427 switch n.Op {
Todd Nealdef7c652015-09-07 19:07:02 -05001428 case OCFUNC:
Keith Randallcd956572016-04-29 09:02:27 -07001429 aux := s.lookupSymbol(n, &ssa.ExternSymbol{Typ: n.Type, Sym: n.Left.Sym})
Todd Nealdef7c652015-09-07 19:07:02 -05001430 return s.entryNewValue1A(ssa.OpAddr, n.Type, aux, s.sb)
Keith Randalld2fd43a2015-04-15 15:51:25 -07001431 case ONAME:
Keith Randall290d8fc2015-06-10 15:03:06 -07001432 if n.Class == PFUNC {
1433 // "value" of a function is the address of the function's closure
Keith Randall8c46aa52015-06-19 21:02:28 -07001434 sym := funcsym(n.Sym)
Keith Randallcd956572016-04-29 09:02:27 -07001435 aux := &ssa.ExternSymbol{Typ: n.Type, Sym: sym}
Keith Randall8c46aa52015-06-19 21:02:28 -07001436 return s.entryNewValue1A(ssa.OpAddr, Ptrto(n.Type), aux, s.sb)
Keith Randall23df95b2015-05-12 15:16:52 -07001437 }
Keith Randall6a8a9da2016-02-27 17:49:31 -08001438 if s.canSSA(n) {
Keith Randall8c46aa52015-06-19 21:02:28 -07001439 return s.variable(n, n.Type)
Keith Randall290d8fc2015-06-10 15:03:06 -07001440 }
David Chase57670ad2015-10-09 16:48:30 -04001441 addr := s.addr(n, false)
Keith Randall8f22b522015-06-11 21:29:25 -07001442 return s.newValue2(ssa.OpLoad, n.Type, addr, s.mem())
David Chase956f3192015-09-11 16:40:05 -04001443 case OCLOSUREVAR:
David Chase57670ad2015-10-09 16:48:30 -04001444 addr := s.addr(n, false)
David Chase956f3192015-09-11 16:40:05 -04001445 return s.newValue2(ssa.OpLoad, n.Type, addr, s.mem())
Keith Randalld2fd43a2015-04-15 15:51:25 -07001446 case OLITERAL:
Matthew Dempsky97360092016-04-22 12:27:29 -07001447 switch u := n.Val().U.(type) {
1448 case *Mpint:
1449 i := u.Int64()
Keith Randall9cb332e2015-07-28 14:19:20 -07001450 switch n.Type.Size() {
1451 case 1:
1452 return s.constInt8(n.Type, int8(i))
1453 case 2:
1454 return s.constInt16(n.Type, int16(i))
1455 case 4:
1456 return s.constInt32(n.Type, int32(i))
1457 case 8:
1458 return s.constInt64(n.Type, i)
1459 default:
1460 s.Fatalf("bad integer size %d", n.Type.Size())
1461 return nil
1462 }
Matthew Dempsky97360092016-04-22 12:27:29 -07001463 case string:
1464 if u == "" {
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08001465 return s.constEmptyString(n.Type)
1466 }
Matthew Dempsky97360092016-04-22 12:27:29 -07001467 return s.entryNewValue0A(ssa.OpConstString, n.Type, u)
1468 case bool:
Keith Randalla19e60b2016-04-26 15:22:33 -07001469 return s.constBool(u)
Matthew Dempsky97360092016-04-22 12:27:29 -07001470 case *NilVal:
Keith Randall9f954db2015-08-18 10:26:28 -07001471 t := n.Type
1472 switch {
1473 case t.IsSlice():
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08001474 return s.constSlice(t)
Keith Randall9f954db2015-08-18 10:26:28 -07001475 case t.IsInterface():
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08001476 return s.constInterface(t)
Keith Randall9f954db2015-08-18 10:26:28 -07001477 default:
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08001478 return s.constNil(t)
Keith Randall9f954db2015-08-18 10:26:28 -07001479 }
Matthew Dempsky97360092016-04-22 12:27:29 -07001480 case *Mpflt:
David Chase997a9f32015-08-12 16:38:11 -04001481 switch n.Type.Size() {
1482 case 4:
Matthew Dempsky97360092016-04-22 12:27:29 -07001483 return s.constFloat32(n.Type, u.Float32())
David Chase997a9f32015-08-12 16:38:11 -04001484 case 8:
Matthew Dempsky97360092016-04-22 12:27:29 -07001485 return s.constFloat64(n.Type, u.Float64())
David Chase997a9f32015-08-12 16:38:11 -04001486 default:
1487 s.Fatalf("bad float size %d", n.Type.Size())
1488 return nil
1489 }
Matthew Dempsky97360092016-04-22 12:27:29 -07001490 case *Mpcplx:
1491 r := &u.Real
1492 i := &u.Imag
David Chase52578582015-08-28 14:24:10 -04001493 switch n.Type.Size() {
1494 case 8:
Matthew Dempsky97360092016-04-22 12:27:29 -07001495 pt := Types[TFLOAT32]
1496 return s.newValue2(ssa.OpComplexMake, n.Type,
1497 s.constFloat32(pt, r.Float32()),
1498 s.constFloat32(pt, i.Float32()))
David Chase52578582015-08-28 14:24:10 -04001499 case 16:
Matthew Dempsky97360092016-04-22 12:27:29 -07001500 pt := Types[TFLOAT64]
1501 return s.newValue2(ssa.OpComplexMake, n.Type,
1502 s.constFloat64(pt, r.Float64()),
1503 s.constFloat64(pt, i.Float64()))
David Chase52578582015-08-28 14:24:10 -04001504 default:
1505 s.Fatalf("bad float size %d", n.Type.Size())
1506 return nil
1507 }
David Chase997a9f32015-08-12 16:38:11 -04001508
Keith Randalld2fd43a2015-04-15 15:51:25 -07001509 default:
Josh Bleecher Snyder37ddc272015-06-24 14:03:39 -07001510 s.Unimplementedf("unhandled OLITERAL %v", n.Val().Ctype())
Keith Randalld2fd43a2015-04-15 15:51:25 -07001511 return nil
1512 }
Keith Randall0ad9c8c2015-06-12 16:24:33 -07001513 case OCONVNOP:
Josh Bleecher Snyder95aff4d2015-07-28 14:31:25 -07001514 to := n.Type
1515 from := n.Left.Type
Josh Bleecher Snyder95aff4d2015-07-28 14:31:25 -07001516
1517 // Assume everything will work out, so set up our return value.
1518 // Anything interesting that happens from here is a fatal.
Keith Randall0ad9c8c2015-06-12 16:24:33 -07001519 x := s.expr(n.Left)
David Chasee99dd522015-10-19 11:36:07 -04001520
1521 // Special case for not confusing GC and liveness.
1522 // We don't want pointers accidentally classified
1523 // as not-pointers or vice-versa because of copy
1524 // elision.
Matthew Dempsky788f1122016-03-28 10:55:44 -07001525 if to.IsPtrShaped() != from.IsPtrShaped() {
Keith Randall7807bda2015-11-10 15:35:36 -08001526 return s.newValue2(ssa.OpConvert, to, x, s.mem())
David Chasee99dd522015-10-19 11:36:07 -04001527 }
1528
Josh Bleecher Snyder95aff4d2015-07-28 14:31:25 -07001529 v := s.newValue1(ssa.OpCopy, to, x) // ensure that v has the right type
1530
Todd Nealdef7c652015-09-07 19:07:02 -05001531 // CONVNOP closure
Matthew Dempsky788f1122016-03-28 10:55:44 -07001532 if to.Etype == TFUNC && from.IsPtrShaped() {
Todd Nealdef7c652015-09-07 19:07:02 -05001533 return v
1534 }
1535
Josh Bleecher Snyder95aff4d2015-07-28 14:31:25 -07001536 // named <--> unnamed type or typed <--> untyped const
1537 if from.Etype == to.Etype {
1538 return v
1539 }
David Chasee99dd522015-10-19 11:36:07 -04001540
Josh Bleecher Snyder95aff4d2015-07-28 14:31:25 -07001541 // unsafe.Pointer <--> *T
1542 if to.Etype == TUNSAFEPTR && from.IsPtr() || from.Etype == TUNSAFEPTR && to.IsPtr() {
1543 return v
1544 }
1545
1546 dowidth(from)
1547 dowidth(to)
1548 if from.Width != to.Width {
1549 s.Fatalf("CONVNOP width mismatch %v (%d) -> %v (%d)\n", from, from.Width, to, to.Width)
1550 return nil
1551 }
1552 if etypesign(from.Etype) != etypesign(to.Etype) {
Josh Bleecher Snyderfca0f332016-04-22 08:39:56 -07001553 s.Fatalf("CONVNOP sign mismatch %v (%s) -> %v (%s)\n", from, from.Etype, to, to.Etype)
Josh Bleecher Snyder95aff4d2015-07-28 14:31:25 -07001554 return nil
1555 }
1556
Ian Lance Taylor88e18032016-03-01 15:17:34 -08001557 if instrumenting {
David Chase57670ad2015-10-09 16:48:30 -04001558 // These appear to be fine, but they fail the
1559 // integer constraint below, so okay them here.
1560 // Sample non-integer conversion: map[string]string -> *uint8
1561 return v
Josh Bleecher Snyder95aff4d2015-07-28 14:31:25 -07001562 }
1563
1564 if etypesign(from.Etype) == 0 {
1565 s.Fatalf("CONVNOP unrecognized non-integer %v -> %v\n", from, to)
1566 return nil
1567 }
1568
1569 // integer, same width, same sign
1570 return v
1571
Michael Matloob73054f52015-06-14 11:38:46 -07001572 case OCONV:
1573 x := s.expr(n.Left)
Keith Randall2a5e6c42015-07-23 14:35:02 -07001574 ft := n.Left.Type // from type
1575 tt := n.Type // to type
1576 if ft.IsInteger() && tt.IsInteger() {
1577 var op ssa.Op
1578 if tt.Size() == ft.Size() {
Josh Bleecher Snyder95aff4d2015-07-28 14:31:25 -07001579 op = ssa.OpCopy
Keith Randall2a5e6c42015-07-23 14:35:02 -07001580 } else if tt.Size() < ft.Size() {
1581 // truncation
1582 switch 10*ft.Size() + tt.Size() {
1583 case 21:
1584 op = ssa.OpTrunc16to8
1585 case 41:
1586 op = ssa.OpTrunc32to8
1587 case 42:
1588 op = ssa.OpTrunc32to16
1589 case 81:
1590 op = ssa.OpTrunc64to8
1591 case 82:
1592 op = ssa.OpTrunc64to16
1593 case 84:
1594 op = ssa.OpTrunc64to32
1595 default:
1596 s.Fatalf("weird integer truncation %s -> %s", ft, tt)
1597 }
1598 } else if ft.IsSigned() {
1599 // sign extension
1600 switch 10*ft.Size() + tt.Size() {
1601 case 12:
1602 op = ssa.OpSignExt8to16
1603 case 14:
1604 op = ssa.OpSignExt8to32
1605 case 18:
1606 op = ssa.OpSignExt8to64
1607 case 24:
1608 op = ssa.OpSignExt16to32
1609 case 28:
1610 op = ssa.OpSignExt16to64
1611 case 48:
1612 op = ssa.OpSignExt32to64
1613 default:
1614 s.Fatalf("bad integer sign extension %s -> %s", ft, tt)
1615 }
1616 } else {
1617 // zero extension
1618 switch 10*ft.Size() + tt.Size() {
1619 case 12:
1620 op = ssa.OpZeroExt8to16
1621 case 14:
1622 op = ssa.OpZeroExt8to32
1623 case 18:
1624 op = ssa.OpZeroExt8to64
1625 case 24:
1626 op = ssa.OpZeroExt16to32
1627 case 28:
1628 op = ssa.OpZeroExt16to64
1629 case 48:
1630 op = ssa.OpZeroExt32to64
1631 default:
1632 s.Fatalf("weird integer sign extension %s -> %s", ft, tt)
1633 }
1634 }
1635 return s.newValue1(op, n.Type, x)
1636 }
David Chase42825882015-08-20 15:14:20 -04001637
David Chased052bbd2015-09-01 17:09:00 -04001638 if ft.IsFloat() || tt.IsFloat() {
1639 conv, ok := fpConvOpToSSA[twoTypes{s.concreteEtype(ft), s.concreteEtype(tt)}]
1640 if !ok {
1641 s.Fatalf("weird float conversion %s -> %s", ft, tt)
David Chase42825882015-08-20 15:14:20 -04001642 }
David Chased052bbd2015-09-01 17:09:00 -04001643 op1, op2, it := conv.op1, conv.op2, conv.intermediateType
1644
1645 if op1 != ssa.OpInvalid && op2 != ssa.OpInvalid {
1646 // normal case, not tripping over unsigned 64
1647 if op1 == ssa.OpCopy {
1648 if op2 == ssa.OpCopy {
1649 return x
1650 }
1651 return s.newValue1(op2, n.Type, x)
1652 }
1653 if op2 == ssa.OpCopy {
1654 return s.newValue1(op1, n.Type, x)
1655 }
1656 return s.newValue1(op2, n.Type, s.newValue1(op1, Types[it], x))
1657 }
1658 // Tricky 64-bit unsigned cases.
1659 if ft.IsInteger() {
1660 // therefore tt is float32 or float64, and ft is also unsigned
David Chase42825882015-08-20 15:14:20 -04001661 if tt.Size() == 4 {
1662 return s.uint64Tofloat32(n, x, ft, tt)
1663 }
1664 if tt.Size() == 8 {
1665 return s.uint64Tofloat64(n, x, ft, tt)
1666 }
David Chased052bbd2015-09-01 17:09:00 -04001667 s.Fatalf("weird unsigned integer to float conversion %s -> %s", ft, tt)
David Chase42825882015-08-20 15:14:20 -04001668 }
David Chased052bbd2015-09-01 17:09:00 -04001669 // therefore ft is float32 or float64, and tt is unsigned integer
David Chase73151062015-08-26 14:25:40 -04001670 if ft.Size() == 4 {
David Chased052bbd2015-09-01 17:09:00 -04001671 return s.float32ToUint64(n, x, ft, tt)
David Chase73151062015-08-26 14:25:40 -04001672 }
David Chased052bbd2015-09-01 17:09:00 -04001673 if ft.Size() == 8 {
1674 return s.float64ToUint64(n, x, ft, tt)
David Chase73151062015-08-26 14:25:40 -04001675 }
David Chased052bbd2015-09-01 17:09:00 -04001676 s.Fatalf("weird float to unsigned integer conversion %s -> %s", ft, tt)
1677 return nil
David Chase42825882015-08-20 15:14:20 -04001678 }
David Chase3a9d0ac2015-08-28 14:24:10 -04001679
1680 if ft.IsComplex() && tt.IsComplex() {
1681 var op ssa.Op
1682 if ft.Size() == tt.Size() {
1683 op = ssa.OpCopy
1684 } else if ft.Size() == 8 && tt.Size() == 16 {
1685 op = ssa.OpCvt32Fto64F
1686 } else if ft.Size() == 16 && tt.Size() == 8 {
1687 op = ssa.OpCvt64Fto32F
1688 } else {
1689 s.Fatalf("weird complex conversion %s -> %s", ft, tt)
1690 }
1691 ftp := floatForComplex(ft)
1692 ttp := floatForComplex(tt)
1693 return s.newValue2(ssa.OpComplexMake, tt,
1694 s.newValue1(op, ttp, s.newValue1(ssa.OpComplexReal, ftp, x)),
1695 s.newValue1(op, ttp, s.newValue1(ssa.OpComplexImag, ftp, x)))
1696 }
David Chase42825882015-08-20 15:14:20 -04001697
Josh Bleecher Snyderfca0f332016-04-22 08:39:56 -07001698 s.Unimplementedf("unhandled OCONV %s -> %s", n.Left.Type.Etype, n.Type.Etype)
Keith Randall2a5e6c42015-07-23 14:35:02 -07001699 return nil
Keith Randallcfc2aa52015-05-18 16:44:20 -07001700
Keith Randall269baa92015-09-17 10:31:16 -07001701 case ODOTTYPE:
1702 res, _ := s.dottype(n, false)
1703 return res
1704
Josh Bleecher Snyder46815b92015-06-24 17:48:22 -07001705 // binary ops
1706 case OLT, OEQ, ONE, OLE, OGE, OGT:
Keith Randalld2fd43a2015-04-15 15:51:25 -07001707 a := s.expr(n.Left)
1708 b := s.expr(n.Right)
Keith Randalldb380bf2015-09-10 11:05:42 -07001709 if n.Left.Type.IsComplex() {
Keith Randallc244ce02015-09-10 14:59:00 -07001710 pt := floatForComplex(n.Left.Type)
Keith Randalldb380bf2015-09-10 11:05:42 -07001711 op := s.ssaOp(OEQ, pt)
1712 r := s.newValue2(op, Types[TBOOL], s.newValue1(ssa.OpComplexReal, pt, a), s.newValue1(ssa.OpComplexReal, pt, b))
1713 i := s.newValue2(op, Types[TBOOL], s.newValue1(ssa.OpComplexImag, pt, a), s.newValue1(ssa.OpComplexImag, pt, b))
1714 c := s.newValue2(ssa.OpAnd8, Types[TBOOL], r, i)
1715 switch n.Op {
1716 case OEQ:
1717 return c
1718 case ONE:
1719 return s.newValue1(ssa.OpNot, Types[TBOOL], c)
1720 default:
Josh Bleecher Snyderf0272412016-04-22 07:14:10 -07001721 s.Fatalf("ordered complex compare %s", n.Op)
Keith Randalldb380bf2015-09-10 11:05:42 -07001722 }
Keith Randalldb380bf2015-09-10 11:05:42 -07001723 }
Josh Bleecher Snyder85e03292015-07-30 11:03:05 -07001724 return s.newValue2(s.ssaOp(n.Op, n.Left.Type), Types[TBOOL], a, b)
David Chase3a9d0ac2015-08-28 14:24:10 -04001725 case OMUL:
1726 a := s.expr(n.Left)
1727 b := s.expr(n.Right)
1728 if n.Type.IsComplex() {
1729 mulop := ssa.OpMul64F
1730 addop := ssa.OpAdd64F
1731 subop := ssa.OpSub64F
1732 pt := floatForComplex(n.Type) // Could be Float32 or Float64
1733 wt := Types[TFLOAT64] // Compute in Float64 to minimize cancellation error
1734
1735 areal := s.newValue1(ssa.OpComplexReal, pt, a)
1736 breal := s.newValue1(ssa.OpComplexReal, pt, b)
1737 aimag := s.newValue1(ssa.OpComplexImag, pt, a)
1738 bimag := s.newValue1(ssa.OpComplexImag, pt, b)
1739
1740 if pt != wt { // Widen for calculation
1741 areal = s.newValue1(ssa.OpCvt32Fto64F, wt, areal)
1742 breal = s.newValue1(ssa.OpCvt32Fto64F, wt, breal)
1743 aimag = s.newValue1(ssa.OpCvt32Fto64F, wt, aimag)
1744 bimag = s.newValue1(ssa.OpCvt32Fto64F, wt, bimag)
1745 }
1746
1747 xreal := s.newValue2(subop, wt, s.newValue2(mulop, wt, areal, breal), s.newValue2(mulop, wt, aimag, bimag))
1748 ximag := s.newValue2(addop, wt, s.newValue2(mulop, wt, areal, bimag), s.newValue2(mulop, wt, aimag, breal))
1749
1750 if pt != wt { // Narrow to store back
1751 xreal = s.newValue1(ssa.OpCvt64Fto32F, pt, xreal)
1752 ximag = s.newValue1(ssa.OpCvt64Fto32F, pt, ximag)
1753 }
1754
1755 return s.newValue2(ssa.OpComplexMake, n.Type, xreal, ximag)
1756 }
1757 return s.newValue2(s.ssaOp(n.Op, n.Type), a.Type, a, b)
1758
1759 case ODIV:
1760 a := s.expr(n.Left)
1761 b := s.expr(n.Right)
1762 if n.Type.IsComplex() {
1763 // TODO this is not executed because the front-end substitutes a runtime call.
1764 // That probably ought to change; with modest optimization the widen/narrow
1765 // conversions could all be elided in larger expression trees.
1766 mulop := ssa.OpMul64F
1767 addop := ssa.OpAdd64F
1768 subop := ssa.OpSub64F
1769 divop := ssa.OpDiv64F
1770 pt := floatForComplex(n.Type) // Could be Float32 or Float64
1771 wt := Types[TFLOAT64] // Compute in Float64 to minimize cancellation error
1772
1773 areal := s.newValue1(ssa.OpComplexReal, pt, a)
1774 breal := s.newValue1(ssa.OpComplexReal, pt, b)
1775 aimag := s.newValue1(ssa.OpComplexImag, pt, a)
1776 bimag := s.newValue1(ssa.OpComplexImag, pt, b)
1777
1778 if pt != wt { // Widen for calculation
1779 areal = s.newValue1(ssa.OpCvt32Fto64F, wt, areal)
1780 breal = s.newValue1(ssa.OpCvt32Fto64F, wt, breal)
1781 aimag = s.newValue1(ssa.OpCvt32Fto64F, wt, aimag)
1782 bimag = s.newValue1(ssa.OpCvt32Fto64F, wt, bimag)
1783 }
1784
1785 denom := s.newValue2(addop, wt, s.newValue2(mulop, wt, breal, breal), s.newValue2(mulop, wt, bimag, bimag))
1786 xreal := s.newValue2(addop, wt, s.newValue2(mulop, wt, areal, breal), s.newValue2(mulop, wt, aimag, bimag))
1787 ximag := s.newValue2(subop, wt, s.newValue2(mulop, wt, aimag, breal), s.newValue2(mulop, wt, areal, bimag))
1788
1789 // TODO not sure if this is best done in wide precision or narrow
1790 // Double-rounding might be an issue.
1791 // Note that the pre-SSA implementation does the entire calculation
1792 // in wide format, so wide is compatible.
1793 xreal = s.newValue2(divop, wt, xreal, denom)
1794 ximag = s.newValue2(divop, wt, ximag, denom)
1795
1796 if pt != wt { // Narrow to store back
1797 xreal = s.newValue1(ssa.OpCvt64Fto32F, pt, xreal)
1798 ximag = s.newValue1(ssa.OpCvt64Fto32F, pt, ximag)
1799 }
David Chase3a9d0ac2015-08-28 14:24:10 -04001800 return s.newValue2(ssa.OpComplexMake, n.Type, xreal, ximag)
1801 }
David Chase18559e22015-10-28 13:55:46 -04001802 if n.Type.IsFloat() {
1803 return s.newValue2(s.ssaOp(n.Op, n.Type), a.Type, a, b)
1804 } else {
1805 // do a size-appropriate check for zero
1806 cmp := s.newValue2(s.ssaOp(ONE, n.Type), Types[TBOOL], b, s.zeroVal(n.Type))
1807 s.check(cmp, panicdivide)
1808 return s.newValue2(s.ssaOp(n.Op, n.Type), a.Type, a, b)
1809 }
1810 case OMOD:
1811 a := s.expr(n.Left)
1812 b := s.expr(n.Right)
1813 // do a size-appropriate check for zero
1814 cmp := s.newValue2(s.ssaOp(ONE, n.Type), Types[TBOOL], b, s.zeroVal(n.Type))
1815 s.check(cmp, panicdivide)
David Chase3a9d0ac2015-08-28 14:24:10 -04001816 return s.newValue2(s.ssaOp(n.Op, n.Type), a.Type, a, b)
1817 case OADD, OSUB:
1818 a := s.expr(n.Left)
1819 b := s.expr(n.Right)
1820 if n.Type.IsComplex() {
1821 pt := floatForComplex(n.Type)
1822 op := s.ssaOp(n.Op, pt)
1823 return s.newValue2(ssa.OpComplexMake, n.Type,
1824 s.newValue2(op, pt, s.newValue1(ssa.OpComplexReal, pt, a), s.newValue1(ssa.OpComplexReal, pt, b)),
1825 s.newValue2(op, pt, s.newValue1(ssa.OpComplexImag, pt, a), s.newValue1(ssa.OpComplexImag, pt, b)))
1826 }
1827 return s.newValue2(s.ssaOp(n.Op, n.Type), a.Type, a, b)
David Chase18559e22015-10-28 13:55:46 -04001828 case OAND, OOR, OHMUL, OXOR:
Keith Randalld2fd43a2015-04-15 15:51:25 -07001829 a := s.expr(n.Left)
1830 b := s.expr(n.Right)
Keith Randall67fdb0d2015-07-19 15:48:20 -07001831 return s.newValue2(s.ssaOp(n.Op, n.Type), a.Type, a, b)
Keith Randall4b803152015-07-29 17:07:09 -07001832 case OLSH, ORSH:
1833 a := s.expr(n.Left)
1834 b := s.expr(n.Right)
1835 return s.newValue2(s.ssaShiftOp(n.Op, n.Type, n.Right.Type), a.Type, a, b)
David Chase40aba8c2015-08-05 22:11:14 -04001836 case OLROT:
1837 a := s.expr(n.Left)
Josh Bleecher Snyder5cab0162016-04-01 14:51:02 -07001838 i := n.Right.Int64()
David Chase40aba8c2015-08-05 22:11:14 -04001839 if i <= 0 || i >= n.Type.Size()*8 {
1840 s.Fatalf("Wrong rotate distance for LROT, expected 1 through %d, saw %d", n.Type.Size()*8-1, i)
1841 }
1842 return s.newValue1I(s.ssaRotateOp(n.Op, n.Type), a.Type, i, a)
Brad Fitzpatricke8167112015-07-10 12:58:53 -06001843 case OANDAND, OOROR:
1844 // To implement OANDAND (and OOROR), we introduce a
1845 // new temporary variable to hold the result. The
1846 // variable is associated with the OANDAND node in the
1847 // s.vars table (normally variables are only
1848 // associated with ONAME nodes). We convert
1849 // A && B
1850 // to
1851 // var = A
1852 // if var {
1853 // var = B
1854 // }
1855 // Using var in the subsequent block introduces the
1856 // necessary phi variable.
1857 el := s.expr(n.Left)
1858 s.vars[n] = el
1859
1860 b := s.endBlock()
1861 b.Kind = ssa.BlockIf
Keith Randall56e0ecc2016-03-15 20:45:50 -07001862 b.SetControl(el)
Josh Bleecher Snyderbbf8c5c2015-08-11 17:28:56 -07001863 // In theory, we should set b.Likely here based on context.
1864 // However, gc only gives us likeliness hints
1865 // in a single place, for plain OIF statements,
1866 // and passing around context is finnicky, so don't bother for now.
Brad Fitzpatricke8167112015-07-10 12:58:53 -06001867
1868 bRight := s.f.NewBlock(ssa.BlockPlain)
1869 bResult := s.f.NewBlock(ssa.BlockPlain)
1870 if n.Op == OANDAND {
Todd Neal47d67992015-08-28 21:36:29 -05001871 b.AddEdgeTo(bRight)
1872 b.AddEdgeTo(bResult)
Brad Fitzpatricke8167112015-07-10 12:58:53 -06001873 } else if n.Op == OOROR {
Todd Neal47d67992015-08-28 21:36:29 -05001874 b.AddEdgeTo(bResult)
1875 b.AddEdgeTo(bRight)
Brad Fitzpatricke8167112015-07-10 12:58:53 -06001876 }
1877
1878 s.startBlock(bRight)
1879 er := s.expr(n.Right)
1880 s.vars[n] = er
1881
1882 b = s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -05001883 b.AddEdgeTo(bResult)
Brad Fitzpatricke8167112015-07-10 12:58:53 -06001884
1885 s.startBlock(bResult)
Josh Bleecher Snyder35ad1fc2015-08-27 10:11:08 -07001886 return s.variable(n, Types[TBOOL])
Keith Randall7e390722015-09-12 14:14:02 -07001887 case OCOMPLEX:
1888 r := s.expr(n.Left)
1889 i := s.expr(n.Right)
1890 return s.newValue2(ssa.OpComplexMake, n.Type, r, i)
Keith Randalld2fd43a2015-04-15 15:51:25 -07001891
Josh Bleecher Snyder4178f202015-09-05 19:28:00 -07001892 // unary ops
David Chase3a9d0ac2015-08-28 14:24:10 -04001893 case OMINUS:
1894 a := s.expr(n.Left)
1895 if n.Type.IsComplex() {
1896 tp := floatForComplex(n.Type)
1897 negop := s.ssaOp(n.Op, tp)
1898 return s.newValue2(ssa.OpComplexMake, n.Type,
1899 s.newValue1(negop, tp, s.newValue1(ssa.OpComplexReal, tp, a)),
1900 s.newValue1(negop, tp, s.newValue1(ssa.OpComplexImag, tp, a)))
1901 }
1902 return s.newValue1(s.ssaOp(n.Op, n.Type), a.Type, a)
Keith Randalla329e212015-09-12 13:26:57 -07001903 case ONOT, OCOM, OSQRT:
Brad Fitzpatrickd9c72d72015-07-10 11:25:48 -06001904 a := s.expr(n.Left)
Alexandru Moșoi954d5ad2015-07-21 16:58:18 +02001905 return s.newValue1(s.ssaOp(n.Op, n.Type), a.Type, a)
Keith Randall2f518072015-09-10 11:37:09 -07001906 case OIMAG, OREAL:
1907 a := s.expr(n.Left)
1908 return s.newValue1(s.ssaOp(n.Op, n.Left.Type), n.Type, a)
Josh Bleecher Snyder4178f202015-09-05 19:28:00 -07001909 case OPLUS:
1910 return s.expr(n.Left)
Brad Fitzpatrickd9c72d72015-07-10 11:25:48 -06001911
Keith Randallcfc2aa52015-05-18 16:44:20 -07001912 case OADDR:
David Chase57670ad2015-10-09 16:48:30 -04001913 return s.addr(n.Left, n.Bounded)
Keith Randallcfc2aa52015-05-18 16:44:20 -07001914
Josh Bleecher Snyder25d19162015-07-28 12:37:46 -07001915 case OINDREG:
1916 if int(n.Reg) != Thearch.REGSP {
1917 s.Unimplementedf("OINDREG of non-SP register %s in expr: %v", obj.Rconv(int(n.Reg)), n)
1918 return nil
1919 }
1920 addr := s.entryNewValue1I(ssa.OpOffPtr, Ptrto(n.Type), n.Xoffset, s.sp)
1921 return s.newValue2(ssa.OpLoad, n.Type, addr, s.mem())
1922
Keith Randalld2fd43a2015-04-15 15:51:25 -07001923 case OIND:
Keith Randall3c1a4c12016-04-19 21:06:53 -07001924 p := s.exprPtr(n.Left, false, n.Lineno)
Keith Randall8f22b522015-06-11 21:29:25 -07001925 return s.newValue2(ssa.OpLoad, n.Type, p, s.mem())
Keith Randallcfc2aa52015-05-18 16:44:20 -07001926
Keith Randallcd7e0592015-07-15 21:33:49 -07001927 case ODOT:
Keith Randalla734bbc2016-01-11 21:05:33 -08001928 t := n.Left.Type
1929 if canSSAType(t) {
1930 v := s.expr(n.Left)
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07001931 return s.newValue1I(ssa.OpStructSelect, n.Type, int64(fieldIdx(n)), v)
Keith Randalla734bbc2016-01-11 21:05:33 -08001932 }
David Chase57670ad2015-10-09 16:48:30 -04001933 p := s.addr(n, false)
Keith Randall97035642015-10-09 09:33:29 -07001934 return s.newValue2(ssa.OpLoad, n.Type, p, s.mem())
Keith Randallcd7e0592015-07-15 21:33:49 -07001935
Keith Randalld2fd43a2015-04-15 15:51:25 -07001936 case ODOTPTR:
Keith Randall3c1a4c12016-04-19 21:06:53 -07001937 p := s.exprPtr(n.Left, false, n.Lineno)
Josh Bleecher Snyderda1802f2016-03-04 12:34:43 -08001938 p = s.newValue1I(ssa.OpOffPtr, p.Type, n.Xoffset, p)
Keith Randall8f22b522015-06-11 21:29:25 -07001939 return s.newValue2(ssa.OpLoad, n.Type, p, s.mem())
Keith Randalld2fd43a2015-04-15 15:51:25 -07001940
1941 case OINDEX:
Keith Randall97035642015-10-09 09:33:29 -07001942 switch {
1943 case n.Left.Type.IsString():
Keith Randallcfc2aa52015-05-18 16:44:20 -07001944 a := s.expr(n.Left)
1945 i := s.expr(n.Right)
Keith Randall2a5e6c42015-07-23 14:35:02 -07001946 i = s.extendIndex(i)
Keith Randall97035642015-10-09 09:33:29 -07001947 if !n.Bounded {
1948 len := s.newValue1(ssa.OpStringLen, Types[TINT], a)
1949 s.boundsCheck(i, len)
Josh Bleecher Snydere00d6092015-06-02 09:16:22 -07001950 }
Keith Randall97035642015-10-09 09:33:29 -07001951 ptrtyp := Ptrto(Types[TUINT8])
1952 ptr := s.newValue1(ssa.OpStringPtr, ptrtyp, a)
Josh Bleecher Snyderda1802f2016-03-04 12:34:43 -08001953 if Isconst(n.Right, CTINT) {
Josh Bleecher Snyder5cab0162016-04-01 14:51:02 -07001954 ptr = s.newValue1I(ssa.OpOffPtr, ptrtyp, n.Right.Int64(), ptr)
Josh Bleecher Snyderda1802f2016-03-04 12:34:43 -08001955 } else {
1956 ptr = s.newValue2(ssa.OpAddPtr, ptrtyp, ptr, i)
1957 }
Keith Randall97035642015-10-09 09:33:29 -07001958 return s.newValue2(ssa.OpLoad, Types[TUINT8], ptr, s.mem())
1959 case n.Left.Type.IsSlice():
David Chase57670ad2015-10-09 16:48:30 -04001960 p := s.addr(n, false)
Josh Bleecher Snyder8640b512016-03-30 10:57:47 -07001961 return s.newValue2(ssa.OpLoad, n.Left.Type.Elem(), p, s.mem())
Keith Randall97035642015-10-09 09:33:29 -07001962 case n.Left.Type.IsArray():
1963 // TODO: fix when we can SSA arrays of length 1.
David Chase57670ad2015-10-09 16:48:30 -04001964 p := s.addr(n, false)
Josh Bleecher Snyder8640b512016-03-30 10:57:47 -07001965 return s.newValue2(ssa.OpLoad, n.Left.Type.Elem(), p, s.mem())
Keith Randall97035642015-10-09 09:33:29 -07001966 default:
1967 s.Fatalf("bad type for index %v", n.Left.Type)
1968 return nil
Keith Randallcfc2aa52015-05-18 16:44:20 -07001969 }
Keith Randalld2fd43a2015-04-15 15:51:25 -07001970
Brad Fitzpatrick7af53d92015-07-10 10:47:28 -06001971 case OLEN, OCAP:
Josh Bleecher Snydercc3f0312015-07-03 18:41:28 -07001972 switch {
Brad Fitzpatrick7af53d92015-07-10 10:47:28 -06001973 case n.Left.Type.IsSlice():
1974 op := ssa.OpSliceLen
1975 if n.Op == OCAP {
1976 op = ssa.OpSliceCap
1977 }
Josh Bleecher Snyder85e03292015-07-30 11:03:05 -07001978 return s.newValue1(op, Types[TINT], s.expr(n.Left))
Brad Fitzpatrick7af53d92015-07-10 10:47:28 -06001979 case n.Left.Type.IsString(): // string; not reachable for OCAP
Josh Bleecher Snyder85e03292015-07-30 11:03:05 -07001980 return s.newValue1(ssa.OpStringLen, Types[TINT], s.expr(n.Left))
Todd Neal707af252015-08-28 15:56:43 -05001981 case n.Left.Type.IsMap(), n.Left.Type.IsChan():
1982 return s.referenceTypeBuiltin(n, s.expr(n.Left))
Josh Bleecher Snydercc3f0312015-07-03 18:41:28 -07001983 default: // array
Josh Bleecher Snyder3a0783c2016-03-31 14:46:04 -07001984 return s.constInt(Types[TINT], n.Left.Type.NumElem())
Josh Bleecher Snydercc3f0312015-07-03 18:41:28 -07001985 }
1986
Josh Bleecher Snydera2d15802015-08-12 10:12:14 -07001987 case OSPTR:
1988 a := s.expr(n.Left)
1989 if n.Left.Type.IsSlice() {
1990 return s.newValue1(ssa.OpSlicePtr, n.Type, a)
1991 } else {
1992 return s.newValue1(ssa.OpStringPtr, n.Type, a)
1993 }
1994
Keith Randalld1c15a02015-08-04 15:47:22 -07001995 case OITAB:
1996 a := s.expr(n.Left)
1997 return s.newValue1(ssa.OpITab, n.Type, a)
1998
Josh Bleecher Snyder1792b362015-09-05 19:28:27 -07001999 case OEFACE:
2000 tab := s.expr(n.Left)
2001 data := s.expr(n.Right)
Keith Randall808d7c72015-10-07 14:35:25 -07002002 // The frontend allows putting things like struct{*byte} in
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00002003 // the data portion of an eface. But we don't want struct{*byte}
Keith Randall808d7c72015-10-07 14:35:25 -07002004 // as a register type because (among other reasons) the liveness
2005 // analysis is confused by the "fat" variables that result from
2006 // such types being spilled.
2007 // So here we ensure that we are selecting the underlying pointer
2008 // when we build an eface.
Keith Randalla734bbc2016-01-11 21:05:33 -08002009 // TODO: get rid of this now that structs can be SSA'd?
Matthew Dempsky788f1122016-03-28 10:55:44 -07002010 for !data.Type.IsPtrShaped() {
Keith Randall808d7c72015-10-07 14:35:25 -07002011 switch {
2012 case data.Type.IsArray():
Matthew Dempsky0b281872016-03-10 14:35:39 -08002013 data = s.newValue1I(ssa.OpArrayIndex, data.Type.ElemType(), 0, data)
Keith Randall808d7c72015-10-07 14:35:25 -07002014 case data.Type.IsStruct():
2015 for i := data.Type.NumFields() - 1; i >= 0; i-- {
2016 f := data.Type.FieldType(i)
2017 if f.Size() == 0 {
2018 // eface type could also be struct{p *byte; q [0]int}
2019 continue
2020 }
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07002021 data = s.newValue1I(ssa.OpStructSelect, f, int64(i), data)
Keith Randall808d7c72015-10-07 14:35:25 -07002022 break
2023 }
2024 default:
2025 s.Fatalf("type being put into an eface isn't a pointer")
2026 }
2027 }
Josh Bleecher Snyder1792b362015-09-05 19:28:27 -07002028 return s.newValue2(ssa.OpIMake, n.Type, tab, data)
2029
Josh Bleecher Snyderf12bd8a2016-04-21 11:55:33 -07002030 case OSLICE, OSLICEARR, OSLICE3, OSLICE3ARR:
Keith Randall5505e8c2015-09-12 23:27:26 -07002031 v := s.expr(n.Left)
Josh Bleecher Snyderf12bd8a2016-04-21 11:55:33 -07002032 var i, j, k *ssa.Value
2033 low, high, max := n.SliceBounds()
2034 if low != nil {
2035 i = s.extendIndex(s.expr(low))
Keith Randall5505e8c2015-09-12 23:27:26 -07002036 }
Josh Bleecher Snyderf12bd8a2016-04-21 11:55:33 -07002037 if high != nil {
2038 j = s.extendIndex(s.expr(high))
Keith Randall5505e8c2015-09-12 23:27:26 -07002039 }
Josh Bleecher Snyderf12bd8a2016-04-21 11:55:33 -07002040 if max != nil {
2041 k = s.extendIndex(s.expr(max))
2042 }
2043 p, l, c := s.slice(n.Left.Type, v, i, j, k)
Keith Randall5505e8c2015-09-12 23:27:26 -07002044 return s.newValue3(ssa.OpSliceMake, n.Type, p, l, c)
Josh Bleecher Snyderf12bd8a2016-04-21 11:55:33 -07002045
Keith Randall3526cf52015-08-24 23:52:03 -07002046 case OSLICESTR:
Keith Randall5505e8c2015-09-12 23:27:26 -07002047 v := s.expr(n.Left)
2048 var i, j *ssa.Value
Josh Bleecher Snyderf12bd8a2016-04-21 11:55:33 -07002049 low, high, _ := n.SliceBounds()
2050 if low != nil {
2051 i = s.extendIndex(s.expr(low))
Keith Randall3526cf52015-08-24 23:52:03 -07002052 }
Josh Bleecher Snyderf12bd8a2016-04-21 11:55:33 -07002053 if high != nil {
2054 j = s.extendIndex(s.expr(high))
Keith Randall3526cf52015-08-24 23:52:03 -07002055 }
Keith Randall5505e8c2015-09-12 23:27:26 -07002056 p, l, _ := s.slice(n.Left.Type, v, i, j, nil)
2057 return s.newValue2(ssa.OpStringMake, n.Type, p, l)
Keith Randall3526cf52015-08-24 23:52:03 -07002058
David Chase8eec2bb2016-03-11 00:10:52 -05002059 case OCALLFUNC:
2060 if isIntrinsicCall1(n) {
2061 return s.intrinsicCall1(n)
2062 }
2063 fallthrough
2064
2065 case OCALLINTER, OCALLMETH:
Keith Randall5ba31942016-01-25 17:06:54 -08002066 a := s.call(n, callNormal)
2067 return s.newValue2(ssa.OpLoad, n.Type, a, s.mem())
Josh Bleecher Snyder3d23afb2015-08-12 11:22:16 -07002068
2069 case OGETG:
Keith Randalld694f832015-10-19 18:54:40 -07002070 return s.newValue1(ssa.OpGetG, n.Type, s.mem())
Josh Bleecher Snyder3d23afb2015-08-12 11:22:16 -07002071
Keith Randall9d22c102015-09-11 11:02:57 -07002072 case OAPPEND:
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002073 return s.append(n, false)
Keith Randall9d22c102015-09-11 11:02:57 -07002074
Keith Randalld2fd43a2015-04-15 15:51:25 -07002075 default:
Josh Bleecher Snyderf0272412016-04-22 07:14:10 -07002076 s.Unimplementedf("unhandled expr %s", n.Op)
Keith Randalld2fd43a2015-04-15 15:51:25 -07002077 return nil
2078 }
2079}
2080
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002081// append converts an OAPPEND node to SSA.
2082// If inplace is false, it converts the OAPPEND expression n to an ssa.Value,
2083// adds it to s, and returns the Value.
2084// If inplace is true, it writes the result of the OAPPEND expression n
2085// back to the slice being appended to, and returns nil.
2086// inplace MUST be set to false if the slice can be SSA'd.
2087func (s *state) append(n *Node, inplace bool) *ssa.Value {
2088 // If inplace is false, process as expression "append(s, e1, e2, e3)":
2089 //
Josh Bleecher Snyder6b33b0e2016-04-10 09:08:00 -07002090 // ptr, len, cap := s
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002091 // newlen := len + 3
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002092 // if newlen > cap {
Josh Bleecher Snyder6b33b0e2016-04-10 09:08:00 -07002093 // ptr, len, cap = growslice(s, newlen)
2094 // newlen = len + 3 // recalculate to avoid a spill
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002095 // }
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002096 // // with write barriers, if needed:
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002097 // *(ptr+len) = e1
2098 // *(ptr+len+1) = e2
2099 // *(ptr+len+2) = e3
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002100 // return makeslice(ptr, newlen, cap)
2101 //
2102 //
2103 // If inplace is true, process as statement "s = append(s, e1, e2, e3)":
2104 //
2105 // a := &s
2106 // ptr, len, cap := s
2107 // newlen := len + 3
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002108 // if newlen > cap {
Josh Bleecher Snyder03e216f2016-04-18 09:40:30 -07002109 // newptr, len, newcap = growslice(ptr, len, cap, newlen)
2110 // vardef(a) // if necessary, advise liveness we are writing a new a
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002111 // *a.cap = newcap // write before ptr to avoid a spill
2112 // *a.ptr = newptr // with write barrier
2113 // }
Josh Bleecher Snyder03e216f2016-04-18 09:40:30 -07002114 // newlen = len + 3 // recalculate to avoid a spill
2115 // *a.len = newlen
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002116 // // with write barriers, if needed:
2117 // *(ptr+len) = e1
2118 // *(ptr+len+1) = e2
2119 // *(ptr+len+2) = e3
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002120
2121 et := n.Type.Elem()
2122 pt := Ptrto(et)
2123
2124 // Evaluate slice
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002125 sn := n.List.First() // the slice node is the first in the list
2126
2127 var slice, addr *ssa.Value
2128 if inplace {
2129 addr = s.addr(sn, false)
2130 slice = s.newValue2(ssa.OpLoad, n.Type, addr, s.mem())
2131 } else {
2132 slice = s.expr(sn)
2133 }
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002134
2135 // Allocate new blocks
2136 grow := s.f.NewBlock(ssa.BlockPlain)
2137 assign := s.f.NewBlock(ssa.BlockPlain)
2138
2139 // Decide if we need to grow
2140 nargs := int64(n.List.Len() - 1)
2141 p := s.newValue1(ssa.OpSlicePtr, pt, slice)
2142 l := s.newValue1(ssa.OpSliceLen, Types[TINT], slice)
2143 c := s.newValue1(ssa.OpSliceCap, Types[TINT], slice)
2144 nl := s.newValue2(s.ssaOp(OADD, Types[TINT]), Types[TINT], l, s.constInt(Types[TINT], nargs))
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002145
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002146 cmp := s.newValue2(s.ssaOp(OGT, Types[TINT]), Types[TBOOL], nl, c)
2147 s.vars[&ptrVar] = p
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002148
2149 if !inplace {
2150 s.vars[&newlenVar] = nl
2151 s.vars[&capVar] = c
Josh Bleecher Snyder03e216f2016-04-18 09:40:30 -07002152 } else {
2153 s.vars[&lenVar] = l
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002154 }
2155
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002156 b := s.endBlock()
2157 b.Kind = ssa.BlockIf
2158 b.Likely = ssa.BranchUnlikely
2159 b.SetControl(cmp)
2160 b.AddEdgeTo(grow)
2161 b.AddEdgeTo(assign)
2162
2163 // Call growslice
2164 s.startBlock(grow)
Keith Randallcd956572016-04-29 09:02:27 -07002165 taddr := s.newValue1A(ssa.OpAddr, Types[TUINTPTR], &ssa.ExternSymbol{Typ: Types[TUINTPTR], Sym: typenamesym(n.Type.Elem())}, s.sb)
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002166
2167 r := s.rtcall(growslice, true, []*Type{pt, Types[TINT], Types[TINT]}, taddr, p, l, c, nl)
2168
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002169 if inplace {
Josh Bleecher Snyder03e216f2016-04-18 09:40:30 -07002170 if sn.Op == ONAME {
2171 // Tell liveness we're about to build a new slice
2172 s.vars[&memVar] = s.newValue1A(ssa.OpVarDef, ssa.TypeMem, sn, s.mem())
2173 }
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002174 capaddr := s.newValue1I(ssa.OpOffPtr, pt, int64(Array_cap), addr)
2175 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, s.config.IntSize, capaddr, r[2], s.mem())
2176 s.insertWBstore(pt, addr, r[0], n.Lineno, 0)
2177 // load the value we just stored to avoid having to spill it
2178 s.vars[&ptrVar] = s.newValue2(ssa.OpLoad, pt, addr, s.mem())
Josh Bleecher Snyder03e216f2016-04-18 09:40:30 -07002179 s.vars[&lenVar] = r[1] // avoid a spill in the fast path
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002180 } else {
2181 s.vars[&ptrVar] = r[0]
2182 s.vars[&newlenVar] = s.newValue2(s.ssaOp(OADD, Types[TINT]), Types[TINT], r[1], s.constInt(Types[TINT], nargs))
2183 s.vars[&capVar] = r[2]
2184 }
2185
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002186 b = s.endBlock()
2187 b.AddEdgeTo(assign)
2188
2189 // assign new elements to slots
2190 s.startBlock(assign)
2191
Josh Bleecher Snyder03e216f2016-04-18 09:40:30 -07002192 if inplace {
2193 l = s.variable(&lenVar, Types[TINT]) // generates phi for len
2194 nl = s.newValue2(s.ssaOp(OADD, Types[TINT]), Types[TINT], l, s.constInt(Types[TINT], nargs))
2195 lenaddr := s.newValue1I(ssa.OpOffPtr, pt, int64(Array_nel), addr)
2196 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, s.config.IntSize, lenaddr, nl, s.mem())
2197 }
2198
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002199 // Evaluate args
2200 args := make([]*ssa.Value, 0, nargs)
2201 store := make([]bool, 0, nargs)
2202 for _, n := range n.List.Slice()[1:] {
2203 if canSSAType(n.Type) {
2204 args = append(args, s.expr(n))
2205 store = append(store, true)
2206 } else {
2207 args = append(args, s.addr(n, false))
2208 store = append(store, false)
2209 }
2210 }
2211
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002212 p = s.variable(&ptrVar, pt) // generates phi for ptr
2213 if !inplace {
2214 nl = s.variable(&newlenVar, Types[TINT]) // generates phi for nl
2215 c = s.variable(&capVar, Types[TINT]) // generates phi for cap
2216 }
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002217 p2 := s.newValue2(ssa.OpPtrIndex, pt, p, l)
2218 // TODO: just one write barrier call for all of these writes?
2219 // TODO: maybe just one writeBarrier.enabled check?
2220 for i, arg := range args {
2221 addr := s.newValue2(ssa.OpPtrIndex, pt, p2, s.constInt(Types[TINT], int64(i)))
2222 if store[i] {
2223 if haspointers(et) {
2224 s.insertWBstore(et, addr, arg, n.Lineno, 0)
2225 } else {
2226 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, et.Size(), addr, arg, s.mem())
2227 }
2228 } else {
2229 if haspointers(et) {
2230 s.insertWBmove(et, addr, arg, n.Lineno)
2231 } else {
2232 s.vars[&memVar] = s.newValue3I(ssa.OpMove, ssa.TypeMem, et.Size(), addr, arg, s.mem())
2233 }
2234 }
2235 }
2236
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002237 delete(s.vars, &ptrVar)
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002238 if inplace {
Josh Bleecher Snyder03e216f2016-04-18 09:40:30 -07002239 delete(s.vars, &lenVar)
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002240 return nil
2241 }
Josh Bleecher Snyder6b33b0e2016-04-10 09:08:00 -07002242 delete(s.vars, &newlenVar)
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002243 delete(s.vars, &capVar)
Josh Bleecher Snydera4650a22016-04-10 09:44:17 -07002244 // make result
Josh Bleecher Snyder5e1b7bd2016-04-04 10:58:21 -07002245 return s.newValue3(ssa.OpSliceMake, n.Type, p, nl, c)
2246}
2247
Keith Randall99187312015-11-02 16:56:53 -08002248// condBranch evaluates the boolean expression cond and branches to yes
2249// if cond is true and no if cond is false.
2250// This function is intended to handle && and || better than just calling
2251// s.expr(cond) and branching on the result.
2252func (s *state) condBranch(cond *Node, yes, no *ssa.Block, likely int8) {
2253 if cond.Op == OANDAND {
2254 mid := s.f.NewBlock(ssa.BlockPlain)
2255 s.stmtList(cond.Ninit)
2256 s.condBranch(cond.Left, mid, no, max8(likely, 0))
2257 s.startBlock(mid)
2258 s.condBranch(cond.Right, yes, no, likely)
2259 return
2260 // Note: if likely==1, then both recursive calls pass 1.
2261 // If likely==-1, then we don't have enough information to decide
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00002262 // whether the first branch is likely or not. So we pass 0 for
Keith Randall99187312015-11-02 16:56:53 -08002263 // the likeliness of the first branch.
2264 // TODO: have the frontend give us branch prediction hints for
2265 // OANDAND and OOROR nodes (if it ever has such info).
2266 }
2267 if cond.Op == OOROR {
2268 mid := s.f.NewBlock(ssa.BlockPlain)
2269 s.stmtList(cond.Ninit)
2270 s.condBranch(cond.Left, yes, mid, min8(likely, 0))
2271 s.startBlock(mid)
2272 s.condBranch(cond.Right, yes, no, likely)
2273 return
2274 // Note: if likely==-1, then both recursive calls pass -1.
2275 // If likely==1, then we don't have enough info to decide
2276 // the likelihood of the first branch.
2277 }
Keith Randalld19bfc32015-11-03 09:30:17 -08002278 if cond.Op == ONOT {
2279 s.stmtList(cond.Ninit)
2280 s.condBranch(cond.Left, no, yes, -likely)
2281 return
2282 }
Keith Randall99187312015-11-02 16:56:53 -08002283 c := s.expr(cond)
2284 b := s.endBlock()
2285 b.Kind = ssa.BlockIf
Keith Randall56e0ecc2016-03-15 20:45:50 -07002286 b.SetControl(c)
Keith Randall99187312015-11-02 16:56:53 -08002287 b.Likely = ssa.BranchPrediction(likely) // gc and ssa both use -1/0/+1 for likeliness
2288 b.AddEdgeTo(yes)
2289 b.AddEdgeTo(no)
2290}
2291
Keith Randalld4663e12016-03-21 10:22:03 -07002292type skipMask uint8
2293
2294const (
2295 skipPtr skipMask = 1 << iota
2296 skipLen
2297 skipCap
2298)
2299
Keith Randall5ba31942016-01-25 17:06:54 -08002300// assign does left = right.
2301// Right has already been evaluated to ssa, left has not.
2302// If deref is true, then we do left = *right instead (and right has already been nil-checked).
2303// If deref is true and right == nil, just do left = 0.
2304// Include a write barrier if wb is true.
Keith Randalld4663e12016-03-21 10:22:03 -07002305// skip indicates assignments (at the top level) that can be avoided.
2306func (s *state) assign(left *Node, right *ssa.Value, wb, deref bool, line int32, skip skipMask) {
Keith Randalld4cc51d2015-08-14 21:47:20 -07002307 if left.Op == ONAME && isblank(left) {
Keith Randalld4cc51d2015-08-14 21:47:20 -07002308 return
2309 }
Keith Randalld4cc51d2015-08-14 21:47:20 -07002310 t := left.Type
2311 dowidth(t)
Keith Randall6a8a9da2016-02-27 17:49:31 -08002312 if s.canSSA(left) {
Keith Randall5ba31942016-01-25 17:06:54 -08002313 if deref {
2314 s.Fatalf("can SSA LHS %s but not RHS %s", left, right)
2315 }
Keith Randalla734bbc2016-01-11 21:05:33 -08002316 if left.Op == ODOT {
2317 // We're assigning to a field of an ssa-able value.
2318 // We need to build a new structure with the new value for the
2319 // field we're assigning and the old values for the other fields.
2320 // For instance:
2321 // type T struct {a, b, c int}
2322 // var T x
2323 // x.b = 5
2324 // For the x.b = 5 assignment we want to generate x = T{x.a, 5, x.c}
2325
2326 // Grab information about the structure type.
2327 t := left.Left.Type
2328 nf := t.NumFields()
2329 idx := fieldIdx(left)
2330
2331 // Grab old value of structure.
2332 old := s.expr(left.Left)
2333
2334 // Make new structure.
2335 new := s.newValue0(ssa.StructMakeOp(t.NumFields()), t)
2336
2337 // Add fields as args.
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07002338 for i := 0; i < nf; i++ {
Keith Randalla734bbc2016-01-11 21:05:33 -08002339 if i == idx {
2340 new.AddArg(right)
2341 } else {
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07002342 new.AddArg(s.newValue1I(ssa.OpStructSelect, t.FieldType(i), int64(i), old))
Keith Randalla734bbc2016-01-11 21:05:33 -08002343 }
2344 }
2345
2346 // Recursively assign the new value we've made to the base of the dot op.
Keith Randalld4663e12016-03-21 10:22:03 -07002347 s.assign(left.Left, new, false, false, line, 0)
Keith Randalla734bbc2016-01-11 21:05:33 -08002348 // TODO: do we need to update named values here?
2349 return
2350 }
Daniel Morsingc31b6dd2015-06-12 14:23:29 +01002351 // Update variable assignment.
Josh Bleecher Snyder07269312015-08-29 14:54:45 -07002352 s.vars[left] = right
Keith Randallc24681a2015-10-22 14:22:38 -07002353 s.addNamedValue(left, right)
Daniel Morsingc31b6dd2015-06-12 14:23:29 +01002354 return
2355 }
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00002356 // Left is not ssa-able. Compute its address.
David Chase57670ad2015-10-09 16:48:30 -04002357 addr := s.addr(left, false)
Keith Randall5325fbc2016-04-29 12:09:32 -07002358 if left.Op == ONAME && skip == 0 {
Keith Randallb32217a2015-09-17 16:45:10 -07002359 s.vars[&memVar] = s.newValue1A(ssa.OpVarDef, ssa.TypeMem, left, s.mem())
Keith Randalld2107fc2015-08-24 02:16:19 -07002360 }
Keith Randall5ba31942016-01-25 17:06:54 -08002361 if deref {
2362 // Treat as a mem->mem move.
2363 if right == nil {
2364 s.vars[&memVar] = s.newValue2I(ssa.OpZero, ssa.TypeMem, t.Size(), addr, s.mem())
2365 return
2366 }
2367 if wb {
2368 s.insertWBmove(t, addr, right, line)
2369 return
2370 }
2371 s.vars[&memVar] = s.newValue3I(ssa.OpMove, ssa.TypeMem, t.Size(), addr, right, s.mem())
2372 return
Keith Randalle3869a62015-09-07 23:18:02 -07002373 }
Keith Randall5ba31942016-01-25 17:06:54 -08002374 // Treat as a store.
2375 if wb {
Keith Randalld4663e12016-03-21 10:22:03 -07002376 if skip&skipPtr != 0 {
2377 // Special case: if we don't write back the pointers, don't bother
2378 // doing the write barrier check.
2379 s.storeTypeScalars(t, addr, right, skip)
2380 return
2381 }
2382 s.insertWBstore(t, addr, right, line, skip)
2383 return
2384 }
2385 if skip != 0 {
2386 if skip&skipPtr == 0 {
2387 s.storeTypePtrs(t, addr, right)
2388 }
2389 s.storeTypeScalars(t, addr, right, skip)
Keith Randall5ba31942016-01-25 17:06:54 -08002390 return
2391 }
2392 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, t.Size(), addr, right, s.mem())
Daniel Morsingc31b6dd2015-06-12 14:23:29 +01002393}
2394
Josh Bleecher Snyder21bd4832015-07-20 15:30:52 -07002395// zeroVal returns the zero value for type t.
2396func (s *state) zeroVal(t *Type) *ssa.Value {
2397 switch {
Keith Randall9cb332e2015-07-28 14:19:20 -07002398 case t.IsInteger():
2399 switch t.Size() {
2400 case 1:
2401 return s.constInt8(t, 0)
2402 case 2:
2403 return s.constInt16(t, 0)
2404 case 4:
2405 return s.constInt32(t, 0)
2406 case 8:
2407 return s.constInt64(t, 0)
2408 default:
2409 s.Fatalf("bad sized integer type %s", t)
2410 }
Todd Neal752fe4d2015-08-25 19:21:45 -05002411 case t.IsFloat():
2412 switch t.Size() {
2413 case 4:
2414 return s.constFloat32(t, 0)
2415 case 8:
2416 return s.constFloat64(t, 0)
2417 default:
2418 s.Fatalf("bad sized float type %s", t)
2419 }
David Chase52578582015-08-28 14:24:10 -04002420 case t.IsComplex():
2421 switch t.Size() {
2422 case 8:
2423 z := s.constFloat32(Types[TFLOAT32], 0)
Keith Randalla5cffb62015-08-28 13:52:26 -07002424 return s.entryNewValue2(ssa.OpComplexMake, t, z, z)
David Chase52578582015-08-28 14:24:10 -04002425 case 16:
2426 z := s.constFloat64(Types[TFLOAT64], 0)
Keith Randalla5cffb62015-08-28 13:52:26 -07002427 return s.entryNewValue2(ssa.OpComplexMake, t, z, z)
David Chase52578582015-08-28 14:24:10 -04002428 default:
2429 s.Fatalf("bad sized complex type %s", t)
2430 }
2431
Josh Bleecher Snyder21bd4832015-07-20 15:30:52 -07002432 case t.IsString():
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08002433 return s.constEmptyString(t)
Matthew Dempsky788f1122016-03-28 10:55:44 -07002434 case t.IsPtrShaped():
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08002435 return s.constNil(t)
Josh Bleecher Snyder21bd4832015-07-20 15:30:52 -07002436 case t.IsBoolean():
Josh Bleecher Snydercea44142015-09-08 16:52:25 -07002437 return s.constBool(false)
Keith Randall9f954db2015-08-18 10:26:28 -07002438 case t.IsInterface():
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08002439 return s.constInterface(t)
Keith Randall9f954db2015-08-18 10:26:28 -07002440 case t.IsSlice():
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08002441 return s.constSlice(t)
Keith Randalla734bbc2016-01-11 21:05:33 -08002442 case t.IsStruct():
2443 n := t.NumFields()
2444 v := s.entryNewValue0(ssa.StructMakeOp(t.NumFields()), t)
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07002445 for i := 0; i < n; i++ {
Keith Randalla734bbc2016-01-11 21:05:33 -08002446 v.AddArg(s.zeroVal(t.FieldType(i).(*Type)))
2447 }
2448 return v
Josh Bleecher Snyder21bd4832015-07-20 15:30:52 -07002449 }
2450 s.Unimplementedf("zero for type %v not implemented", t)
2451 return nil
2452}
2453
Keith Randalld24768e2015-09-09 23:56:59 -07002454type callKind int8
2455
2456const (
2457 callNormal callKind = iota
2458 callDefer
2459 callGo
2460)
2461
David Chase8eec2bb2016-03-11 00:10:52 -05002462// isSSAIntrinsic1 returns true if n is a call to a recognized 1-arg intrinsic
2463// that can be handled by the SSA backend.
2464// SSA uses this, but so does the front end to see if should not
2465// inline a function because it is a candidate for intrinsic
2466// substitution.
2467func isSSAIntrinsic1(s *Sym) bool {
2468 // The test below is not quite accurate -- in the event that
2469 // a function is disabled on a per-function basis, for example
2470 // because of hash-keyed binary failure search, SSA might be
2471 // disabled for that function but it would not be noted here,
2472 // and thus an inlining would not occur (in practice, inlining
2473 // so far has only been noticed for Bswap32 and the 16-bit count
2474 // leading/trailing instructions, but heuristics might change
2475 // in the future or on different architectures).
Matthew Dempskyc6e11fe2016-04-06 12:01:40 -07002476 if !ssaEnabled || ssa.IntrinsicsDisable || Thearch.LinkArch.Family != sys.AMD64 {
David Chase8eec2bb2016-03-11 00:10:52 -05002477 return false
2478 }
2479 if s != nil && s.Pkg != nil && s.Pkg.Path == "runtime/internal/sys" {
2480 switch s.Name {
2481 case
2482 "Ctz64", "Ctz32", "Ctz16",
2483 "Bswap64", "Bswap32":
2484 return true
2485 }
2486 }
2487 return false
2488}
2489
2490func isIntrinsicCall1(n *Node) bool {
2491 if n == nil || n.Left == nil {
2492 return false
2493 }
2494 return isSSAIntrinsic1(n.Left.Sym)
2495}
2496
2497// intrinsicFirstArg extracts arg from n.List and eval
2498func (s *state) intrinsicFirstArg(n *Node) *ssa.Value {
2499 x := n.List.First()
2500 if x.Op == OAS {
2501 x = x.Right
2502 }
2503 return s.expr(x)
2504}
2505
2506// intrinsicCall1 converts a call to a recognized 1-arg intrinsic
2507// into the intrinsic
2508func (s *state) intrinsicCall1(n *Node) *ssa.Value {
2509 var result *ssa.Value
2510 switch n.Left.Sym.Name {
2511 case "Ctz64":
2512 result = s.newValue1(ssa.OpCtz64, Types[TUINT64], s.intrinsicFirstArg(n))
2513 case "Ctz32":
2514 result = s.newValue1(ssa.OpCtz32, Types[TUINT32], s.intrinsicFirstArg(n))
2515 case "Ctz16":
2516 result = s.newValue1(ssa.OpCtz16, Types[TUINT16], s.intrinsicFirstArg(n))
2517 case "Bswap64":
2518 result = s.newValue1(ssa.OpBswap64, Types[TUINT64], s.intrinsicFirstArg(n))
2519 case "Bswap32":
2520 result = s.newValue1(ssa.OpBswap32, Types[TUINT32], s.intrinsicFirstArg(n))
2521 }
2522 if result == nil {
2523 Fatalf("Unknown special call: %v", n.Left.Sym)
2524 }
2525 if ssa.IntrinsicsDebug > 0 {
2526 Warnl(n.Lineno, "intrinsic substitution for %v with %s", n.Left.Sym.Name, result.LongString())
2527 }
2528 return result
2529}
2530
Keith Randall5ba31942016-01-25 17:06:54 -08002531// Calls the function n using the specified call type.
2532// Returns the address of the return value (or nil if none).
Keith Randalld24768e2015-09-09 23:56:59 -07002533func (s *state) call(n *Node, k callKind) *ssa.Value {
2534 var sym *Sym // target symbol (if static)
2535 var closure *ssa.Value // ptr to closure to run (if dynamic)
2536 var codeptr *ssa.Value // ptr to target code (if dynamic)
2537 var rcvr *ssa.Value // receiver to set
2538 fn := n.Left
2539 switch n.Op {
2540 case OCALLFUNC:
2541 if k == callNormal && fn.Op == ONAME && fn.Class == PFUNC {
2542 sym = fn.Sym
2543 break
2544 }
2545 closure = s.expr(fn)
Keith Randalld24768e2015-09-09 23:56:59 -07002546 case OCALLMETH:
2547 if fn.Op != ODOTMETH {
2548 Fatalf("OCALLMETH: n.Left not an ODOTMETH: %v", fn)
2549 }
Keith Randalld24768e2015-09-09 23:56:59 -07002550 if k == callNormal {
Ian Lance Taylor5f525ca2016-03-18 16:52:30 -07002551 sym = fn.Sym
Keith Randalld24768e2015-09-09 23:56:59 -07002552 break
2553 }
Ian Lance Taylor5f525ca2016-03-18 16:52:30 -07002554 n2 := newname(fn.Sym)
Keith Randalld24768e2015-09-09 23:56:59 -07002555 n2.Class = PFUNC
Ian Lance Taylor5f525ca2016-03-18 16:52:30 -07002556 n2.Lineno = fn.Lineno
2557 closure = s.expr(n2)
Keith Randalld24768e2015-09-09 23:56:59 -07002558 // Note: receiver is already assigned in n.List, so we don't
2559 // want to set it here.
2560 case OCALLINTER:
2561 if fn.Op != ODOTINTER {
Dave Cheneyd3c79d32016-04-27 15:10:10 +10002562 Fatalf("OCALLINTER: n.Left not an ODOTINTER: %v", fn.Op)
Keith Randalld24768e2015-09-09 23:56:59 -07002563 }
2564 i := s.expr(fn.Left)
2565 itab := s.newValue1(ssa.OpITab, Types[TUINTPTR], i)
2566 itabidx := fn.Xoffset + 3*int64(Widthptr) + 8 // offset of fun field in runtime.itab
2567 itab = s.newValue1I(ssa.OpOffPtr, Types[TUINTPTR], itabidx, itab)
2568 if k == callNormal {
2569 codeptr = s.newValue2(ssa.OpLoad, Types[TUINTPTR], itab, s.mem())
2570 } else {
2571 closure = itab
2572 }
2573 rcvr = s.newValue1(ssa.OpIData, Types[TUINTPTR], i)
2574 }
2575 dowidth(fn.Type)
Josh Bleecher Snyder361b3342016-03-28 14:31:57 -07002576 stksize := fn.Type.ArgWidth() // includes receiver
Keith Randalld24768e2015-09-09 23:56:59 -07002577
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00002578 // Run all argument assignments. The arg slots have already
Keith Randalld24768e2015-09-09 23:56:59 -07002579 // been offset by the appropriate amount (+2*widthptr for go/defer,
2580 // +widthptr for interface calls).
2581 // For OCALLMETH, the receiver is set in these statements.
2582 s.stmtList(n.List)
2583
2584 // Set receiver (for interface calls)
2585 if rcvr != nil {
Keith Randall7c4fbb62015-10-19 13:56:55 -07002586 argStart := Ctxt.FixedFrameSize()
Keith Randalld24768e2015-09-09 23:56:59 -07002587 if k != callNormal {
2588 argStart += int64(2 * Widthptr)
2589 }
2590 addr := s.entryNewValue1I(ssa.OpOffPtr, Types[TUINTPTR], argStart, s.sp)
Keith Randallb32217a2015-09-17 16:45:10 -07002591 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, int64(Widthptr), addr, rcvr, s.mem())
Keith Randalld24768e2015-09-09 23:56:59 -07002592 }
2593
2594 // Defer/go args
2595 if k != callNormal {
2596 // Write argsize and closure (args to Newproc/Deferproc).
2597 argsize := s.constInt32(Types[TUINT32], int32(stksize))
Keith Randallb32217a2015-09-17 16:45:10 -07002598 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, 4, s.sp, argsize, s.mem())
Keith Randalld24768e2015-09-09 23:56:59 -07002599 addr := s.entryNewValue1I(ssa.OpOffPtr, Ptrto(Types[TUINTPTR]), int64(Widthptr), s.sp)
Keith Randallb32217a2015-09-17 16:45:10 -07002600 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, int64(Widthptr), addr, closure, s.mem())
Keith Randalld24768e2015-09-09 23:56:59 -07002601 stksize += 2 * int64(Widthptr)
2602 }
2603
2604 // call target
2605 bNext := s.f.NewBlock(ssa.BlockPlain)
2606 var call *ssa.Value
2607 switch {
2608 case k == callDefer:
2609 call = s.newValue1(ssa.OpDeferCall, ssa.TypeMem, s.mem())
2610 case k == callGo:
2611 call = s.newValue1(ssa.OpGoCall, ssa.TypeMem, s.mem())
2612 case closure != nil:
2613 codeptr = s.newValue2(ssa.OpLoad, Types[TUINTPTR], closure, s.mem())
2614 call = s.newValue3(ssa.OpClosureCall, ssa.TypeMem, codeptr, closure, s.mem())
2615 case codeptr != nil:
2616 call = s.newValue2(ssa.OpInterCall, ssa.TypeMem, codeptr, s.mem())
2617 case sym != nil:
2618 call = s.newValue1A(ssa.OpStaticCall, ssa.TypeMem, sym, s.mem())
2619 default:
Josh Bleecher Snyderf0272412016-04-22 07:14:10 -07002620 Fatalf("bad call type %s %v", n.Op, n)
Keith Randalld24768e2015-09-09 23:56:59 -07002621 }
2622 call.AuxInt = stksize // Call operations carry the argsize of the callee along with them
2623
2624 // Finish call block
Keith Randallb32217a2015-09-17 16:45:10 -07002625 s.vars[&memVar] = call
Keith Randalld24768e2015-09-09 23:56:59 -07002626 b := s.endBlock()
2627 b.Kind = ssa.BlockCall
Keith Randall56e0ecc2016-03-15 20:45:50 -07002628 b.SetControl(call)
Keith Randalld24768e2015-09-09 23:56:59 -07002629 b.AddEdgeTo(bNext)
Keith Randallddc6b642016-03-09 19:27:57 -08002630 if k == callDefer {
2631 // Add recover edge to exit code.
2632 b.Kind = ssa.BlockDefer
2633 r := s.f.NewBlock(ssa.BlockPlain)
2634 s.startBlock(r)
2635 s.exit()
2636 b.AddEdgeTo(r)
2637 b.Likely = ssa.BranchLikely
2638 }
Keith Randalld24768e2015-09-09 23:56:59 -07002639
Keith Randall5ba31942016-01-25 17:06:54 -08002640 // Start exit block, find address of result.
Keith Randalld24768e2015-09-09 23:56:59 -07002641 s.startBlock(bNext)
Keith Randall3572c642016-04-21 19:28:28 -07002642 // Keep input pointer args live across calls. This is a bandaid until 1.8.
2643 for _, n := range s.ptrargs {
2644 s.vars[&memVar] = s.newValue2(ssa.OpKeepAlive, ssa.TypeMem, s.variable(n, n.Type), s.mem())
2645 }
Matthew Dempskyf6fab932016-03-15 11:06:03 -07002646 res := n.Left.Type.Results()
2647 if res.NumFields() == 0 || k != callNormal {
Keith Randalld24768e2015-09-09 23:56:59 -07002648 // call has no return value. Continue with the next statement.
2649 return nil
2650 }
Matthew Dempskyf6fab932016-03-15 11:06:03 -07002651 fp := res.Field(0)
Cherry Zhang508a4242016-04-18 10:30:20 -04002652 return s.entryNewValue1I(ssa.OpOffPtr, Ptrto(fp.Type), fp.Offset+Ctxt.FixedFrameSize(), s.sp)
Keith Randalld24768e2015-09-09 23:56:59 -07002653}
2654
Josh Bleecher Snyder95aff4d2015-07-28 14:31:25 -07002655// etypesign returns the signed-ness of e, for integer/pointer etypes.
2656// -1 means signed, +1 means unsigned, 0 means non-integer/non-pointer.
Keith Randall4304fbc2015-11-16 13:20:16 -08002657func etypesign(e EType) int8 {
Josh Bleecher Snyder95aff4d2015-07-28 14:31:25 -07002658 switch e {
2659 case TINT8, TINT16, TINT32, TINT64, TINT:
2660 return -1
2661 case TUINT8, TUINT16, TUINT32, TUINT64, TUINT, TUINTPTR, TUNSAFEPTR:
2662 return +1
2663 }
2664 return 0
2665}
2666
Todd Neald076ef72015-10-15 20:25:32 -05002667// lookupSymbol is used to retrieve the symbol (Extern, Arg or Auto) used for a particular node.
2668// This improves the effectiveness of cse by using the same Aux values for the
2669// same symbols.
2670func (s *state) lookupSymbol(n *Node, sym interface{}) interface{} {
2671 switch sym.(type) {
2672 default:
2673 s.Fatalf("sym %v is of uknown type %T", sym, sym)
2674 case *ssa.ExternSymbol, *ssa.ArgSymbol, *ssa.AutoSymbol:
2675 // these are the only valid types
2676 }
2677
2678 if lsym, ok := s.varsyms[n]; ok {
2679 return lsym
2680 } else {
2681 s.varsyms[n] = sym
2682 return sym
2683 }
2684}
2685
Josh Bleecher Snydere00d6092015-06-02 09:16:22 -07002686// addr converts the address of the expression n to SSA, adds it to s and returns the SSA result.
Keith Randallc3c84a22015-07-13 15:55:37 -07002687// The value that the returned Value represents is guaranteed to be non-nil.
David Chase57670ad2015-10-09 16:48:30 -04002688// If bounded is true then this address does not require a nil check for its operand
2689// even if that would otherwise be implied.
2690func (s *state) addr(n *Node, bounded bool) *ssa.Value {
Keith Randallc24681a2015-10-22 14:22:38 -07002691 t := Ptrto(n.Type)
Keith Randallcfc2aa52015-05-18 16:44:20 -07002692 switch n.Op {
2693 case ONAME:
Keith Randall290d8fc2015-06-10 15:03:06 -07002694 switch n.Class {
2695 case PEXTERN:
Keith Randallcfc2aa52015-05-18 16:44:20 -07002696 // global variable
Keith Randallcd956572016-04-29 09:02:27 -07002697 aux := s.lookupSymbol(n, &ssa.ExternSymbol{Typ: n.Type, Sym: n.Sym})
Keith Randallc24681a2015-10-22 14:22:38 -07002698 v := s.entryNewValue1A(ssa.OpAddr, t, aux, s.sb)
Josh Bleecher Snyder67df7932015-07-28 11:08:44 -07002699 // TODO: Make OpAddr use AuxInt as well as Aux.
2700 if n.Xoffset != 0 {
2701 v = s.entryNewValue1I(ssa.OpOffPtr, v.Type, n.Xoffset, v)
2702 }
2703 return v
David Chase956f3192015-09-11 16:40:05 -04002704 case PPARAM:
2705 // parameter slot
Josh Bleecher Snyder596ddf42015-06-29 11:56:28 -07002706 v := s.decladdrs[n]
Keith Randall4304fbc2015-11-16 13:20:16 -08002707 if v != nil {
2708 return v
Josh Bleecher Snyder596ddf42015-06-29 11:56:28 -07002709 }
Keith Randall4304fbc2015-11-16 13:20:16 -08002710 if n.String() == ".fp" {
2711 // Special arg that points to the frame pointer.
2712 // (Used by the race detector, others?)
2713 aux := s.lookupSymbol(n, &ssa.ArgSymbol{Typ: n.Type, Node: n})
2714 return s.entryNewValue1A(ssa.OpAddr, t, aux, s.sp)
2715 }
2716 s.Fatalf("addr of undeclared ONAME %v. declared: %v", n, s.decladdrs)
2717 return nil
Keith Randalld2107fc2015-08-24 02:16:19 -07002718 case PAUTO:
Todd Neal40bfec02016-03-11 20:03:17 -06002719 aux := s.lookupSymbol(n, &ssa.AutoSymbol{Typ: n.Type, Node: n})
Keith Randallc24681a2015-10-22 14:22:38 -07002720 return s.newValue1A(ssa.OpAddr, t, aux, s.sp)
David Chase956f3192015-09-11 16:40:05 -04002721 case PPARAMOUT: // Same as PAUTO -- cannot generate LEA early.
Todd Neald076ef72015-10-15 20:25:32 -05002722 // ensure that we reuse symbols for out parameters so
2723 // that cse works on their addresses
2724 aux := s.lookupSymbol(n, &ssa.ArgSymbol{Typ: n.Type, Node: n})
Keith Randallc24681a2015-10-22 14:22:38 -07002725 return s.newValue1A(ssa.OpAddr, t, aux, s.sp)
Russ Coxb6dc3e62016-05-25 01:33:24 -04002726 case PPARAMREF:
Daniel Morsingc31b6dd2015-06-12 14:23:29 +01002727 return s.expr(n.Name.Heapaddr)
Keith Randall290d8fc2015-06-10 15:03:06 -07002728 default:
Russ Coxb6dc3e62016-05-25 01:33:24 -04002729 s.Unimplementedf("variable address class %v not implemented", classnames[n.Class])
Keith Randall290d8fc2015-06-10 15:03:06 -07002730 return nil
Keith Randallcfc2aa52015-05-18 16:44:20 -07002731 }
Keith Randallcfc2aa52015-05-18 16:44:20 -07002732 case OINDREG:
Josh Bleecher Snyder25d19162015-07-28 12:37:46 -07002733 // indirect off a register
Keith Randallcfc2aa52015-05-18 16:44:20 -07002734 // used for storing/loading arguments/returns to/from callees
Josh Bleecher Snyder25d19162015-07-28 12:37:46 -07002735 if int(n.Reg) != Thearch.REGSP {
2736 s.Unimplementedf("OINDREG of non-SP register %s in addr: %v", obj.Rconv(int(n.Reg)), n)
2737 return nil
2738 }
Keith Randallc24681a2015-10-22 14:22:38 -07002739 return s.entryNewValue1I(ssa.OpOffPtr, t, n.Xoffset, s.sp)
Keith Randallcfc2aa52015-05-18 16:44:20 -07002740 case OINDEX:
Brad Fitzpatrick7af53d92015-07-10 10:47:28 -06002741 if n.Left.Type.IsSlice() {
Keith Randallcfc2aa52015-05-18 16:44:20 -07002742 a := s.expr(n.Left)
2743 i := s.expr(n.Right)
Keith Randall2a5e6c42015-07-23 14:35:02 -07002744 i = s.extendIndex(i)
Keith Randallc24681a2015-10-22 14:22:38 -07002745 len := s.newValue1(ssa.OpSliceLen, Types[TINT], a)
Keith Randall46e62f82015-08-18 14:17:30 -07002746 if !n.Bounded {
2747 s.boundsCheck(i, len)
2748 }
Keith Randallc24681a2015-10-22 14:22:38 -07002749 p := s.newValue1(ssa.OpSlicePtr, t, a)
2750 return s.newValue2(ssa.OpPtrIndex, t, p, i)
Brad Fitzpatrick7af53d92015-07-10 10:47:28 -06002751 } else { // array
David Chase57670ad2015-10-09 16:48:30 -04002752 a := s.addr(n.Left, bounded)
Brad Fitzpatrick7af53d92015-07-10 10:47:28 -06002753 i := s.expr(n.Right)
Keith Randall2a5e6c42015-07-23 14:35:02 -07002754 i = s.extendIndex(i)
Josh Bleecher Snyder3a0783c2016-03-31 14:46:04 -07002755 len := s.constInt(Types[TINT], n.Left.Type.NumElem())
Keith Randall46e62f82015-08-18 14:17:30 -07002756 if !n.Bounded {
2757 s.boundsCheck(i, len)
2758 }
Josh Bleecher Snyder8640b512016-03-30 10:57:47 -07002759 return s.newValue2(ssa.OpPtrIndex, Ptrto(n.Left.Type.Elem()), a, i)
Keith Randallcfc2aa52015-05-18 16:44:20 -07002760 }
Todd Nealb383de22015-07-13 21:22:16 -05002761 case OIND:
Keith Randall3c1a4c12016-04-19 21:06:53 -07002762 return s.exprPtr(n.Left, bounded, n.Lineno)
Keith Randallc3c84a22015-07-13 15:55:37 -07002763 case ODOT:
David Chase57670ad2015-10-09 16:48:30 -04002764 p := s.addr(n.Left, bounded)
Josh Bleecher Snyderda1802f2016-03-04 12:34:43 -08002765 return s.newValue1I(ssa.OpOffPtr, t, n.Xoffset, p)
Keith Randallc3c84a22015-07-13 15:55:37 -07002766 case ODOTPTR:
Keith Randall3c1a4c12016-04-19 21:06:53 -07002767 p := s.exprPtr(n.Left, bounded, n.Lineno)
Josh Bleecher Snyderda1802f2016-03-04 12:34:43 -08002768 return s.newValue1I(ssa.OpOffPtr, t, n.Xoffset, p)
David Chase956f3192015-09-11 16:40:05 -04002769 case OCLOSUREVAR:
Josh Bleecher Snyderda1802f2016-03-04 12:34:43 -08002770 return s.newValue1I(ssa.OpOffPtr, t, n.Xoffset,
2771 s.entryNewValue0(ssa.OpGetClosurePtr, Ptrto(Types[TUINT8])))
David Chase57670ad2015-10-09 16:48:30 -04002772 case OCONVNOP:
2773 addr := s.addr(n.Left, bounded)
Keith Randallc24681a2015-10-22 14:22:38 -07002774 return s.newValue1(ssa.OpCopy, t, addr) // ensure that addr has the right type
Keith Randall5ba31942016-01-25 17:06:54 -08002775 case OCALLFUNC, OCALLINTER, OCALLMETH:
2776 return s.call(n, callNormal)
David Chase57670ad2015-10-09 16:48:30 -04002777
Keith Randallcfc2aa52015-05-18 16:44:20 -07002778 default:
Dave Cheneyd3c79d32016-04-27 15:10:10 +10002779 s.Unimplementedf("unhandled addr %v", n.Op)
Keith Randallcfc2aa52015-05-18 16:44:20 -07002780 return nil
2781 }
2782}
2783
Keith Randall290d8fc2015-06-10 15:03:06 -07002784// canSSA reports whether n is SSA-able.
Keith Randalla734bbc2016-01-11 21:05:33 -08002785// n must be an ONAME (or an ODOT sequence with an ONAME base).
Keith Randall6a8a9da2016-02-27 17:49:31 -08002786func (s *state) canSSA(n *Node) bool {
Keith Randall5325fbc2016-04-29 12:09:32 -07002787 if Debug['N'] != 0 {
2788 return false
2789 }
Keith Randalla734bbc2016-01-11 21:05:33 -08002790 for n.Op == ODOT {
2791 n = n.Left
2792 }
Keith Randall290d8fc2015-06-10 15:03:06 -07002793 if n.Op != ONAME {
Daniel Morsing66b47812015-06-27 15:45:20 +01002794 return false
Keith Randall290d8fc2015-06-10 15:03:06 -07002795 }
2796 if n.Addrtaken {
2797 return false
2798 }
Russ Coxb6dc3e62016-05-25 01:33:24 -04002799 if n.isParamHeapCopy() {
Keith Randall290d8fc2015-06-10 15:03:06 -07002800 return false
2801 }
Russ Coxb6dc3e62016-05-25 01:33:24 -04002802 if n.Class == PAUTOHEAP {
2803 Fatalf("canSSA of PAUTOHEAP %v", n)
2804 }
Josh Bleecher Snyder96548732015-08-28 13:35:32 -07002805 switch n.Class {
Keith Randall6a8a9da2016-02-27 17:49:31 -08002806 case PEXTERN, PPARAMREF:
2807 // TODO: maybe treat PPARAMREF with an Arg-like op to read from closure?
Keith Randall290d8fc2015-06-10 15:03:06 -07002808 return false
Keith Randall6a8a9da2016-02-27 17:49:31 -08002809 case PPARAMOUT:
2810 if hasdefer {
2811 // TODO: handle this case? Named return values must be
2812 // in memory so that the deferred function can see them.
2813 // Maybe do: if !strings.HasPrefix(n.String(), "~") { return false }
2814 return false
2815 }
2816 if s.cgoUnsafeArgs {
2817 // Cgo effectively takes the address of all result args,
2818 // but the compiler can't see that.
2819 return false
2820 }
Keith Randall290d8fc2015-06-10 15:03:06 -07002821 }
Keith Randall8a1f6212015-09-08 21:28:44 -07002822 if n.Class == PPARAM && n.String() == ".this" {
2823 // wrappers generated by genwrapper need to update
2824 // the .this pointer in place.
Keith Randall6a8a9da2016-02-27 17:49:31 -08002825 // TODO: treat as a PPARMOUT?
Keith Randall8a1f6212015-09-08 21:28:44 -07002826 return false
2827 }
Keith Randall9f954db2015-08-18 10:26:28 -07002828 return canSSAType(n.Type)
2829 // TODO: try to make more variables SSAable?
2830}
2831
2832// canSSA reports whether variables of type t are SSA-able.
2833func canSSAType(t *Type) bool {
2834 dowidth(t)
2835 if t.Width > int64(4*Widthptr) {
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00002836 // 4*Widthptr is an arbitrary constant. We want it
Keith Randall9f954db2015-08-18 10:26:28 -07002837 // to be at least 3*Widthptr so slices can be registerized.
2838 // Too big and we'll introduce too much register pressure.
Daniel Morsing66b47812015-06-27 15:45:20 +01002839 return false
2840 }
Keith Randall9f954db2015-08-18 10:26:28 -07002841 switch t.Etype {
2842 case TARRAY:
Keith Randall9f954db2015-08-18 10:26:28 -07002843 // We can't do arrays because dynamic indexing is
2844 // not supported on SSA variables.
2845 // TODO: maybe allow if length is <=1? All indexes
2846 // are constant? Might be good for the arrays
2847 // introduced by the compiler for variadic functions.
2848 return false
2849 case TSTRUCT:
Matthew Dempskydbed1c62016-03-17 13:26:08 -07002850 if t.NumFields() > ssa.MaxStruct {
Keith Randall9f954db2015-08-18 10:26:28 -07002851 return false
2852 }
Matthew Dempskyf6bca3f2016-03-17 01:32:18 -07002853 for _, t1 := range t.Fields().Slice() {
Keith Randall9f954db2015-08-18 10:26:28 -07002854 if !canSSAType(t1.Type) {
2855 return false
2856 }
2857 }
Keith Randalla734bbc2016-01-11 21:05:33 -08002858 return true
Keith Randall9f954db2015-08-18 10:26:28 -07002859 default:
2860 return true
2861 }
Keith Randall290d8fc2015-06-10 15:03:06 -07002862}
2863
Keith Randall3c1a4c12016-04-19 21:06:53 -07002864// exprPtr evaluates n to a pointer and nil-checks it.
2865func (s *state) exprPtr(n *Node, bounded bool, lineno int32) *ssa.Value {
2866 p := s.expr(n)
2867 if bounded || n.NonNil {
2868 if s.f.Config.Debug_checknil() && lineno > 1 {
2869 s.f.Config.Warnl(lineno, "removed nil check")
2870 }
2871 return p
2872 }
2873 s.nilCheck(p)
2874 return p
2875}
2876
Keith Randallcfc2aa52015-05-18 16:44:20 -07002877// nilCheck generates nil pointer checking code.
Josh Bleecher Snyder463858e2015-08-11 09:47:45 -07002878// Starts a new block on return, unless nil checks are disabled.
Josh Bleecher Snyder7e74e432015-07-24 11:55:52 -07002879// Used only for automatically inserted nil checks,
2880// not for user code like 'x != nil'.
Keith Randallcfc2aa52015-05-18 16:44:20 -07002881func (s *state) nilCheck(ptr *ssa.Value) {
Josh Bleecher Snyder463858e2015-08-11 09:47:45 -07002882 if Disable_checknil != 0 {
2883 return
2884 }
Keith Randall31115a52015-10-23 19:12:49 -07002885 chk := s.newValue2(ssa.OpNilCheck, ssa.TypeVoid, ptr, s.mem())
Keith Randallcfc2aa52015-05-18 16:44:20 -07002886 b := s.endBlock()
Keith Randall31115a52015-10-23 19:12:49 -07002887 b.Kind = ssa.BlockCheck
Keith Randall56e0ecc2016-03-15 20:45:50 -07002888 b.SetControl(chk)
Keith Randallcfc2aa52015-05-18 16:44:20 -07002889 bNext := s.f.NewBlock(ssa.BlockPlain)
Todd Neal47d67992015-08-28 21:36:29 -05002890 b.AddEdgeTo(bNext)
Josh Bleecher Snyder463858e2015-08-11 09:47:45 -07002891 s.startBlock(bNext)
Keith Randallcfc2aa52015-05-18 16:44:20 -07002892}
2893
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00002894// boundsCheck generates bounds checking code. Checks if 0 <= idx < len, branches to exit if not.
Keith Randallcfc2aa52015-05-18 16:44:20 -07002895// Starts a new block on return.
2896func (s *state) boundsCheck(idx, len *ssa.Value) {
Keith Randall8d236812015-08-18 15:25:40 -07002897 if Debug['B'] != 0 {
2898 return
2899 }
Keith Randallcfc2aa52015-05-18 16:44:20 -07002900 // TODO: convert index to full width?
2901 // TODO: if index is 64-bit and we're compiling to 32-bit, check that high 32 bits are zero.
2902
2903 // bounds check
Josh Bleecher Snyder85e03292015-07-30 11:03:05 -07002904 cmp := s.newValue2(ssa.OpIsInBounds, Types[TBOOL], idx, len)
Keith Randall3a70bf92015-09-17 16:54:15 -07002905 s.check(cmp, Panicindex)
Keith Randall3526cf52015-08-24 23:52:03 -07002906}
2907
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00002908// sliceBoundsCheck generates slice bounds checking code. Checks if 0 <= idx <= len, branches to exit if not.
Keith Randall3526cf52015-08-24 23:52:03 -07002909// Starts a new block on return.
2910func (s *state) sliceBoundsCheck(idx, len *ssa.Value) {
2911 if Debug['B'] != 0 {
2912 return
2913 }
2914 // TODO: convert index to full width?
2915 // TODO: if index is 64-bit and we're compiling to 32-bit, check that high 32 bits are zero.
2916
2917 // bounds check
2918 cmp := s.newValue2(ssa.OpIsSliceInBounds, Types[TBOOL], idx, len)
Keith Randall3a70bf92015-09-17 16:54:15 -07002919 s.check(cmp, panicslice)
Keith Randall3526cf52015-08-24 23:52:03 -07002920}
2921
Keith Randall3a70bf92015-09-17 16:54:15 -07002922// If cmp (a bool) is true, panic using the given function.
2923func (s *state) check(cmp *ssa.Value, fn *Node) {
Keith Randallcfc2aa52015-05-18 16:44:20 -07002924 b := s.endBlock()
2925 b.Kind = ssa.BlockIf
Keith Randall56e0ecc2016-03-15 20:45:50 -07002926 b.SetControl(cmp)
Josh Bleecher Snyderbbf8c5c2015-08-11 17:28:56 -07002927 b.Likely = ssa.BranchLikely
Keith Randallcfc2aa52015-05-18 16:44:20 -07002928 bNext := s.f.NewBlock(ssa.BlockPlain)
Keith Randall74e568f2015-11-09 21:35:40 -08002929 line := s.peekLine()
2930 bPanic := s.panics[funcLine{fn, line}]
2931 if bPanic == nil {
2932 bPanic = s.f.NewBlock(ssa.BlockPlain)
2933 s.panics[funcLine{fn, line}] = bPanic
2934 s.startBlock(bPanic)
2935 // The panic call takes/returns memory to ensure that the right
2936 // memory state is observed if the panic happens.
2937 s.rtcall(fn, false, nil)
2938 }
Todd Neal47d67992015-08-28 21:36:29 -05002939 b.AddEdgeTo(bNext)
2940 b.AddEdgeTo(bPanic)
Keith Randallcfc2aa52015-05-18 16:44:20 -07002941 s.startBlock(bNext)
2942}
2943
Keith Randall8c5bfcc2015-09-18 15:11:30 -07002944// rtcall issues a call to the given runtime function fn with the listed args.
2945// Returns a slice of results of the given result types.
2946// The call is added to the end of the current block.
2947// If returns is false, the block is marked as an exit block.
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00002948// If returns is true, the block is marked as a call block. A new block
Keith Randall8c5bfcc2015-09-18 15:11:30 -07002949// is started to load the return values.
2950func (s *state) rtcall(fn *Node, returns bool, results []*Type, args ...*ssa.Value) []*ssa.Value {
2951 // Write args to the stack
2952 var off int64 // TODO: arch-dependent starting offset?
2953 for _, arg := range args {
2954 t := arg.Type
2955 off = Rnd(off, t.Alignment())
2956 ptr := s.sp
2957 if off != 0 {
2958 ptr = s.newValue1I(ssa.OpOffPtr, Types[TUINTPTR], off, s.sp)
2959 }
2960 size := t.Size()
2961 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, size, ptr, arg, s.mem())
2962 off += size
2963 }
2964 off = Rnd(off, int64(Widthptr))
2965
2966 // Issue call
2967 call := s.newValue1A(ssa.OpStaticCall, ssa.TypeMem, fn.Sym, s.mem())
2968 s.vars[&memVar] = call
2969
2970 // Finish block
2971 b := s.endBlock()
2972 if !returns {
2973 b.Kind = ssa.BlockExit
Keith Randall56e0ecc2016-03-15 20:45:50 -07002974 b.SetControl(call)
Keith Randall8c5bfcc2015-09-18 15:11:30 -07002975 call.AuxInt = off
2976 if len(results) > 0 {
2977 Fatalf("panic call can't have results")
2978 }
2979 return nil
2980 }
2981 b.Kind = ssa.BlockCall
Keith Randall56e0ecc2016-03-15 20:45:50 -07002982 b.SetControl(call)
Keith Randall8c5bfcc2015-09-18 15:11:30 -07002983 bNext := s.f.NewBlock(ssa.BlockPlain)
2984 b.AddEdgeTo(bNext)
2985 s.startBlock(bNext)
2986
Keith Randall3572c642016-04-21 19:28:28 -07002987 // Keep input pointer args live across calls. This is a bandaid until 1.8.
2988 for _, n := range s.ptrargs {
2989 s.vars[&memVar] = s.newValue2(ssa.OpKeepAlive, ssa.TypeMem, s.variable(n, n.Type), s.mem())
2990 }
2991
Keith Randall8c5bfcc2015-09-18 15:11:30 -07002992 // Load results
2993 res := make([]*ssa.Value, len(results))
2994 for i, t := range results {
2995 off = Rnd(off, t.Alignment())
2996 ptr := s.sp
2997 if off != 0 {
2998 ptr = s.newValue1I(ssa.OpOffPtr, Types[TUINTPTR], off, s.sp)
2999 }
3000 res[i] = s.newValue2(ssa.OpLoad, t, ptr, s.mem())
3001 off += t.Size()
3002 }
3003 off = Rnd(off, int64(Widthptr))
3004
3005 // Remember how much callee stack space we needed.
3006 call.AuxInt = off
3007
3008 return res
3009}
3010
Keith Randall5ba31942016-01-25 17:06:54 -08003011// insertWBmove inserts the assignment *left = *right including a write barrier.
3012// t is the type being assigned.
3013func (s *state) insertWBmove(t *Type, left, right *ssa.Value, line int32) {
Keith Randall4304fbc2015-11-16 13:20:16 -08003014 // if writeBarrier.enabled {
Keith Randall5ba31942016-01-25 17:06:54 -08003015 // typedmemmove(&t, left, right)
3016 // } else {
3017 // *left = *right
Keith Randall9d22c102015-09-11 11:02:57 -07003018 // }
Keith Randall15ed37d2016-03-16 21:51:17 -07003019
3020 if s.noWB {
3021 s.Fatalf("write barrier prohibited")
3022 }
3023 if s.WBLineno == 0 {
3024 s.WBLineno = left.Line
3025 }
Keith Randall9d22c102015-09-11 11:02:57 -07003026 bThen := s.f.NewBlock(ssa.BlockPlain)
Keith Randall5ba31942016-01-25 17:06:54 -08003027 bElse := s.f.NewBlock(ssa.BlockPlain)
3028 bEnd := s.f.NewBlock(ssa.BlockPlain)
Keith Randall9d22c102015-09-11 11:02:57 -07003029
Keith Randallcd956572016-04-29 09:02:27 -07003030 aux := &ssa.ExternSymbol{Typ: Types[TBOOL], Sym: syslook("writeBarrier").Sym}
David Chase8107b002016-02-28 11:15:22 -05003031 flagaddr := s.newValue1A(ssa.OpAddr, Ptrto(Types[TUINT32]), aux, s.sb)
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003032 // TODO: select the .enabled field. It is currently first, so not needed for now.
David Chase8107b002016-02-28 11:15:22 -05003033 // Load word, test byte, avoiding partial register write from load byte.
3034 flag := s.newValue2(ssa.OpLoad, Types[TUINT32], flagaddr, s.mem())
3035 flag = s.newValue1(ssa.OpTrunc64to8, Types[TBOOL], flag)
Keith Randall9d22c102015-09-11 11:02:57 -07003036 b := s.endBlock()
3037 b.Kind = ssa.BlockIf
3038 b.Likely = ssa.BranchUnlikely
Keith Randall56e0ecc2016-03-15 20:45:50 -07003039 b.SetControl(flag)
Keith Randall9d22c102015-09-11 11:02:57 -07003040 b.AddEdgeTo(bThen)
Keith Randall5ba31942016-01-25 17:06:54 -08003041 b.AddEdgeTo(bElse)
Keith Randall9d22c102015-09-11 11:02:57 -07003042
3043 s.startBlock(bThen)
Keith Randallcd956572016-04-29 09:02:27 -07003044 taddr := s.newValue1A(ssa.OpAddr, Types[TUINTPTR], &ssa.ExternSymbol{Typ: Types[TUINTPTR], Sym: typenamesym(t)}, s.sb)
Keith Randall5ba31942016-01-25 17:06:54 -08003045 s.rtcall(typedmemmove, true, nil, taddr, left, right)
3046 s.endBlock().AddEdgeTo(bEnd)
3047
3048 s.startBlock(bElse)
3049 s.vars[&memVar] = s.newValue3I(ssa.OpMove, ssa.TypeMem, t.Size(), left, right, s.mem())
3050 s.endBlock().AddEdgeTo(bEnd)
3051
3052 s.startBlock(bEnd)
Keith Randall9d22c102015-09-11 11:02:57 -07003053
David Chase729abfa2015-10-26 17:34:06 -04003054 if Debug_wb > 0 {
Robert Griesemerb83f3972016-03-02 11:01:25 -08003055 Warnl(line, "write barrier")
David Chase729abfa2015-10-26 17:34:06 -04003056 }
Keith Randall5ba31942016-01-25 17:06:54 -08003057}
David Chase729abfa2015-10-26 17:34:06 -04003058
Keith Randall5ba31942016-01-25 17:06:54 -08003059// insertWBstore inserts the assignment *left = right including a write barrier.
3060// t is the type being assigned.
Keith Randalld4663e12016-03-21 10:22:03 -07003061func (s *state) insertWBstore(t *Type, left, right *ssa.Value, line int32, skip skipMask) {
Keith Randall5ba31942016-01-25 17:06:54 -08003062 // store scalar fields
3063 // if writeBarrier.enabled {
3064 // writebarrierptr for pointer fields
3065 // } else {
3066 // store pointer fields
3067 // }
3068
Keith Randall15ed37d2016-03-16 21:51:17 -07003069 if s.noWB {
3070 s.Fatalf("write barrier prohibited")
3071 }
3072 if s.WBLineno == 0 {
3073 s.WBLineno = left.Line
3074 }
Keith Randalld4663e12016-03-21 10:22:03 -07003075 s.storeTypeScalars(t, left, right, skip)
Keith Randall5ba31942016-01-25 17:06:54 -08003076
3077 bThen := s.f.NewBlock(ssa.BlockPlain)
3078 bElse := s.f.NewBlock(ssa.BlockPlain)
3079 bEnd := s.f.NewBlock(ssa.BlockPlain)
3080
Keith Randallcd956572016-04-29 09:02:27 -07003081 aux := &ssa.ExternSymbol{Typ: Types[TBOOL], Sym: syslook("writeBarrier").Sym}
David Chase8107b002016-02-28 11:15:22 -05003082 flagaddr := s.newValue1A(ssa.OpAddr, Ptrto(Types[TUINT32]), aux, s.sb)
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003083 // TODO: select the .enabled field. It is currently first, so not needed for now.
David Chase8107b002016-02-28 11:15:22 -05003084 // Load word, test byte, avoiding partial register write from load byte.
3085 flag := s.newValue2(ssa.OpLoad, Types[TUINT32], flagaddr, s.mem())
3086 flag = s.newValue1(ssa.OpTrunc64to8, Types[TBOOL], flag)
Keith Randall5ba31942016-01-25 17:06:54 -08003087 b := s.endBlock()
3088 b.Kind = ssa.BlockIf
3089 b.Likely = ssa.BranchUnlikely
Keith Randall56e0ecc2016-03-15 20:45:50 -07003090 b.SetControl(flag)
Keith Randall5ba31942016-01-25 17:06:54 -08003091 b.AddEdgeTo(bThen)
3092 b.AddEdgeTo(bElse)
3093
3094 // Issue write barriers for pointer writes.
3095 s.startBlock(bThen)
Keith Randallaebf6612016-01-29 21:57:57 -08003096 s.storeTypePtrsWB(t, left, right)
3097 s.endBlock().AddEdgeTo(bEnd)
3098
3099 // Issue regular stores for pointer writes.
3100 s.startBlock(bElse)
3101 s.storeTypePtrs(t, left, right)
3102 s.endBlock().AddEdgeTo(bEnd)
3103
3104 s.startBlock(bEnd)
3105
3106 if Debug_wb > 0 {
Robert Griesemerb83f3972016-03-02 11:01:25 -08003107 Warnl(line, "write barrier")
Keith Randallaebf6612016-01-29 21:57:57 -08003108 }
3109}
3110
3111// do *left = right for all scalar (non-pointer) parts of t.
Keith Randalld4663e12016-03-21 10:22:03 -07003112func (s *state) storeTypeScalars(t *Type, left, right *ssa.Value, skip skipMask) {
Keith Randallaebf6612016-01-29 21:57:57 -08003113 switch {
3114 case t.IsBoolean() || t.IsInteger() || t.IsFloat() || t.IsComplex():
3115 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, t.Size(), left, right, s.mem())
Matthew Dempsky788f1122016-03-28 10:55:44 -07003116 case t.IsPtrShaped():
Keith Randallaebf6612016-01-29 21:57:57 -08003117 // no scalar fields.
3118 case t.IsString():
Keith Randalld4663e12016-03-21 10:22:03 -07003119 if skip&skipLen != 0 {
3120 return
3121 }
Keith Randallaebf6612016-01-29 21:57:57 -08003122 len := s.newValue1(ssa.OpStringLen, Types[TINT], right)
3123 lenAddr := s.newValue1I(ssa.OpOffPtr, Ptrto(Types[TINT]), s.config.IntSize, left)
3124 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, s.config.IntSize, lenAddr, len, s.mem())
3125 case t.IsSlice():
Keith Randalld4663e12016-03-21 10:22:03 -07003126 if skip&skipLen == 0 {
3127 len := s.newValue1(ssa.OpSliceLen, Types[TINT], right)
3128 lenAddr := s.newValue1I(ssa.OpOffPtr, Ptrto(Types[TINT]), s.config.IntSize, left)
3129 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, s.config.IntSize, lenAddr, len, s.mem())
3130 }
3131 if skip&skipCap == 0 {
3132 cap := s.newValue1(ssa.OpSliceCap, Types[TINT], right)
3133 capAddr := s.newValue1I(ssa.OpOffPtr, Ptrto(Types[TINT]), 2*s.config.IntSize, left)
3134 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, s.config.IntSize, capAddr, cap, s.mem())
3135 }
Keith Randallaebf6612016-01-29 21:57:57 -08003136 case t.IsInterface():
3137 // itab field doesn't need a write barrier (even though it is a pointer).
3138 itab := s.newValue1(ssa.OpITab, Ptrto(Types[TUINT8]), right)
3139 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, s.config.IntSize, left, itab, s.mem())
3140 case t.IsStruct():
3141 n := t.NumFields()
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07003142 for i := 0; i < n; i++ {
Keith Randallaebf6612016-01-29 21:57:57 -08003143 ft := t.FieldType(i)
3144 addr := s.newValue1I(ssa.OpOffPtr, ft.PtrTo(), t.FieldOff(i), left)
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07003145 val := s.newValue1I(ssa.OpStructSelect, ft, int64(i), right)
Keith Randalld4663e12016-03-21 10:22:03 -07003146 s.storeTypeScalars(ft.(*Type), addr, val, 0)
Keith Randallaebf6612016-01-29 21:57:57 -08003147 }
3148 default:
3149 s.Fatalf("bad write barrier type %s", t)
3150 }
3151}
3152
3153// do *left = right for all pointer parts of t.
3154func (s *state) storeTypePtrs(t *Type, left, right *ssa.Value) {
3155 switch {
Matthew Dempsky788f1122016-03-28 10:55:44 -07003156 case t.IsPtrShaped():
Keith Randallaebf6612016-01-29 21:57:57 -08003157 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, s.config.PtrSize, left, right, s.mem())
3158 case t.IsString():
3159 ptr := s.newValue1(ssa.OpStringPtr, Ptrto(Types[TUINT8]), right)
3160 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, s.config.PtrSize, left, ptr, s.mem())
3161 case t.IsSlice():
3162 ptr := s.newValue1(ssa.OpSlicePtr, Ptrto(Types[TUINT8]), right)
3163 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, s.config.PtrSize, left, ptr, s.mem())
3164 case t.IsInterface():
3165 // itab field is treated as a scalar.
3166 idata := s.newValue1(ssa.OpIData, Ptrto(Types[TUINT8]), right)
3167 idataAddr := s.newValue1I(ssa.OpOffPtr, Ptrto(Types[TUINT8]), s.config.PtrSize, left)
3168 s.vars[&memVar] = s.newValue3I(ssa.OpStore, ssa.TypeMem, s.config.PtrSize, idataAddr, idata, s.mem())
3169 case t.IsStruct():
3170 n := t.NumFields()
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07003171 for i := 0; i < n; i++ {
Keith Randallaebf6612016-01-29 21:57:57 -08003172 ft := t.FieldType(i)
3173 if !haspointers(ft.(*Type)) {
3174 continue
3175 }
3176 addr := s.newValue1I(ssa.OpOffPtr, ft.PtrTo(), t.FieldOff(i), left)
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07003177 val := s.newValue1I(ssa.OpStructSelect, ft, int64(i), right)
Keith Randallaebf6612016-01-29 21:57:57 -08003178 s.storeTypePtrs(ft.(*Type), addr, val)
3179 }
3180 default:
3181 s.Fatalf("bad write barrier type %s", t)
3182 }
3183}
3184
3185// do *left = right with a write barrier for all pointer parts of t.
3186func (s *state) storeTypePtrsWB(t *Type, left, right *ssa.Value) {
Keith Randall5ba31942016-01-25 17:06:54 -08003187 switch {
Matthew Dempsky788f1122016-03-28 10:55:44 -07003188 case t.IsPtrShaped():
Keith Randall5ba31942016-01-25 17:06:54 -08003189 s.rtcall(writebarrierptr, true, nil, left, right)
3190 case t.IsString():
3191 ptr := s.newValue1(ssa.OpStringPtr, Ptrto(Types[TUINT8]), right)
3192 s.rtcall(writebarrierptr, true, nil, left, ptr)
3193 case t.IsSlice():
3194 ptr := s.newValue1(ssa.OpSlicePtr, Ptrto(Types[TUINT8]), right)
3195 s.rtcall(writebarrierptr, true, nil, left, ptr)
3196 case t.IsInterface():
3197 idata := s.newValue1(ssa.OpIData, Ptrto(Types[TUINT8]), right)
3198 idataAddr := s.newValue1I(ssa.OpOffPtr, Ptrto(Types[TUINT8]), s.config.PtrSize, left)
3199 s.rtcall(writebarrierptr, true, nil, idataAddr, idata)
Keith Randallaebf6612016-01-29 21:57:57 -08003200 case t.IsStruct():
3201 n := t.NumFields()
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07003202 for i := 0; i < n; i++ {
Keith Randallaebf6612016-01-29 21:57:57 -08003203 ft := t.FieldType(i)
3204 if !haspointers(ft.(*Type)) {
3205 continue
3206 }
3207 addr := s.newValue1I(ssa.OpOffPtr, ft.PtrTo(), t.FieldOff(i), left)
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07003208 val := s.newValue1I(ssa.OpStructSelect, ft, int64(i), right)
Keith Randallaebf6612016-01-29 21:57:57 -08003209 s.storeTypePtrsWB(ft.(*Type), addr, val)
3210 }
Keith Randall5ba31942016-01-25 17:06:54 -08003211 default:
3212 s.Fatalf("bad write barrier type %s", t)
3213 }
Keith Randall9d22c102015-09-11 11:02:57 -07003214}
3215
Keith Randall5505e8c2015-09-12 23:27:26 -07003216// slice computes the slice v[i:j:k] and returns ptr, len, and cap of result.
3217// i,j,k may be nil, in which case they are set to their default value.
3218// t is a slice, ptr to array, or string type.
3219func (s *state) slice(t *Type, v, i, j, k *ssa.Value) (p, l, c *ssa.Value) {
3220 var elemtype *Type
3221 var ptrtype *Type
3222 var ptr *ssa.Value
3223 var len *ssa.Value
3224 var cap *ssa.Value
3225 zero := s.constInt(Types[TINT], 0)
3226 switch {
3227 case t.IsSlice():
Josh Bleecher Snyder8640b512016-03-30 10:57:47 -07003228 elemtype = t.Elem()
Keith Randall5505e8c2015-09-12 23:27:26 -07003229 ptrtype = Ptrto(elemtype)
3230 ptr = s.newValue1(ssa.OpSlicePtr, ptrtype, v)
3231 len = s.newValue1(ssa.OpSliceLen, Types[TINT], v)
3232 cap = s.newValue1(ssa.OpSliceCap, Types[TINT], v)
3233 case t.IsString():
3234 elemtype = Types[TUINT8]
3235 ptrtype = Ptrto(elemtype)
3236 ptr = s.newValue1(ssa.OpStringPtr, ptrtype, v)
3237 len = s.newValue1(ssa.OpStringLen, Types[TINT], v)
3238 cap = len
3239 case t.IsPtr():
Josh Bleecher Snyder8640b512016-03-30 10:57:47 -07003240 if !t.Elem().IsArray() {
Keith Randall5505e8c2015-09-12 23:27:26 -07003241 s.Fatalf("bad ptr to array in slice %v\n", t)
3242 }
Josh Bleecher Snyder8640b512016-03-30 10:57:47 -07003243 elemtype = t.Elem().Elem()
Keith Randall5505e8c2015-09-12 23:27:26 -07003244 ptrtype = Ptrto(elemtype)
3245 s.nilCheck(v)
3246 ptr = v
Josh Bleecher Snyder3a0783c2016-03-31 14:46:04 -07003247 len = s.constInt(Types[TINT], t.Elem().NumElem())
Keith Randall5505e8c2015-09-12 23:27:26 -07003248 cap = len
3249 default:
3250 s.Fatalf("bad type in slice %v\n", t)
3251 }
3252
3253 // Set default values
3254 if i == nil {
3255 i = zero
3256 }
3257 if j == nil {
3258 j = len
3259 }
3260 if k == nil {
3261 k = cap
3262 }
3263
3264 // Panic if slice indices are not in bounds.
3265 s.sliceBoundsCheck(i, j)
3266 if j != k {
3267 s.sliceBoundsCheck(j, k)
3268 }
3269 if k != cap {
3270 s.sliceBoundsCheck(k, cap)
3271 }
3272
3273 // Generate the following code assuming that indexes are in bounds.
3274 // The conditional is to make sure that we don't generate a slice
3275 // that points to the next object in memory.
Keith Randall69a7c1522016-03-21 15:24:08 -07003276 // rlen = j-i
3277 // rcap = k-i
3278 // delta = i*elemsize
3279 // if rcap == 0 {
3280 // delta = 0
Keith Randall5505e8c2015-09-12 23:27:26 -07003281 // }
Keith Randall69a7c1522016-03-21 15:24:08 -07003282 // rptr = p+delta
3283 // result = (SliceMake rptr rlen rcap)
Keith Randall582baae2015-11-02 21:28:13 -08003284 subOp := s.ssaOp(OSUB, Types[TINT])
Keith Randall69a7c1522016-03-21 15:24:08 -07003285 eqOp := s.ssaOp(OEQ, Types[TINT])
Keith Randall582baae2015-11-02 21:28:13 -08003286 mulOp := s.ssaOp(OMUL, Types[TINT])
3287 rlen := s.newValue2(subOp, Types[TINT], j, i)
Keith Randall5505e8c2015-09-12 23:27:26 -07003288 var rcap *ssa.Value
3289 switch {
3290 case t.IsString():
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003291 // Capacity of the result is unimportant. However, we use
Keith Randall5505e8c2015-09-12 23:27:26 -07003292 // rcap to test if we've generated a zero-length slice.
3293 // Use length of strings for that.
3294 rcap = rlen
3295 case j == k:
3296 rcap = rlen
3297 default:
Keith Randall582baae2015-11-02 21:28:13 -08003298 rcap = s.newValue2(subOp, Types[TINT], k, i)
Keith Randall5505e8c2015-09-12 23:27:26 -07003299 }
3300
Keith Randall69a7c1522016-03-21 15:24:08 -07003301 // delta = # of elements to offset pointer by.
3302 s.vars[&deltaVar] = i
Keith Randall5505e8c2015-09-12 23:27:26 -07003303
Keith Randall69a7c1522016-03-21 15:24:08 -07003304 // Generate code to set delta=0 if the resulting capacity is zero.
3305 if !((i.Op == ssa.OpConst64 && i.AuxInt == 0) ||
3306 (i.Op == ssa.OpConst32 && int32(i.AuxInt) == 0)) {
3307 cmp := s.newValue2(eqOp, Types[TBOOL], rcap, zero)
Keith Randall5505e8c2015-09-12 23:27:26 -07003308
Keith Randall69a7c1522016-03-21 15:24:08 -07003309 b := s.endBlock()
3310 b.Kind = ssa.BlockIf
3311 b.Likely = ssa.BranchUnlikely
3312 b.SetControl(cmp)
Keith Randall5505e8c2015-09-12 23:27:26 -07003313
Keith Randall69a7c1522016-03-21 15:24:08 -07003314 // Generate block which zeros the delta variable.
3315 nz := s.f.NewBlock(ssa.BlockPlain)
3316 b.AddEdgeTo(nz)
3317 s.startBlock(nz)
3318 s.vars[&deltaVar] = zero
3319 s.endBlock()
3320
3321 // All done.
3322 merge := s.f.NewBlock(ssa.BlockPlain)
3323 b.AddEdgeTo(merge)
3324 nz.AddEdgeTo(merge)
3325 s.startBlock(merge)
3326
3327 // TODO: use conditional moves somehow?
Keith Randall5505e8c2015-09-12 23:27:26 -07003328 }
Keith Randall5505e8c2015-09-12 23:27:26 -07003329
Keith Randall69a7c1522016-03-21 15:24:08 -07003330 // Compute rptr = ptr + delta * elemsize
3331 rptr := s.newValue2(ssa.OpAddPtr, ptrtype, ptr, s.newValue2(mulOp, Types[TINT], s.variable(&deltaVar, Types[TINT]), s.constInt(Types[TINT], elemtype.Width)))
3332 delete(s.vars, &deltaVar)
Keith Randall5505e8c2015-09-12 23:27:26 -07003333 return rptr, rlen, rcap
3334}
3335
David Chase42825882015-08-20 15:14:20 -04003336type u2fcvtTab struct {
3337 geq, cvt2F, and, rsh, or, add ssa.Op
3338 one func(*state, ssa.Type, int64) *ssa.Value
3339}
3340
3341var u64_f64 u2fcvtTab = u2fcvtTab{
3342 geq: ssa.OpGeq64,
3343 cvt2F: ssa.OpCvt64to64F,
3344 and: ssa.OpAnd64,
3345 rsh: ssa.OpRsh64Ux64,
3346 or: ssa.OpOr64,
3347 add: ssa.OpAdd64F,
3348 one: (*state).constInt64,
3349}
3350
3351var u64_f32 u2fcvtTab = u2fcvtTab{
3352 geq: ssa.OpGeq64,
3353 cvt2F: ssa.OpCvt64to32F,
3354 and: ssa.OpAnd64,
3355 rsh: ssa.OpRsh64Ux64,
3356 or: ssa.OpOr64,
3357 add: ssa.OpAdd32F,
3358 one: (*state).constInt64,
3359}
3360
3361// Excess generality on a machine with 64-bit integer registers.
3362// Not used on AMD64.
3363var u32_f32 u2fcvtTab = u2fcvtTab{
3364 geq: ssa.OpGeq32,
3365 cvt2F: ssa.OpCvt32to32F,
3366 and: ssa.OpAnd32,
3367 rsh: ssa.OpRsh32Ux32,
3368 or: ssa.OpOr32,
3369 add: ssa.OpAdd32F,
3370 one: func(s *state, t ssa.Type, x int64) *ssa.Value {
3371 return s.constInt32(t, int32(x))
3372 },
3373}
3374
3375func (s *state) uint64Tofloat64(n *Node, x *ssa.Value, ft, tt *Type) *ssa.Value {
3376 return s.uintTofloat(&u64_f64, n, x, ft, tt)
3377}
3378
3379func (s *state) uint64Tofloat32(n *Node, x *ssa.Value, ft, tt *Type) *ssa.Value {
3380 return s.uintTofloat(&u64_f32, n, x, ft, tt)
3381}
3382
3383func (s *state) uintTofloat(cvttab *u2fcvtTab, n *Node, x *ssa.Value, ft, tt *Type) *ssa.Value {
3384 // if x >= 0 {
3385 // result = (floatY) x
3386 // } else {
3387 // y = uintX(x) ; y = x & 1
3388 // z = uintX(x) ; z = z >> 1
3389 // z = z >> 1
3390 // z = z | y
David Chase73151062015-08-26 14:25:40 -04003391 // result = floatY(z)
3392 // result = result + result
David Chase42825882015-08-20 15:14:20 -04003393 // }
3394 //
3395 // Code borrowed from old code generator.
3396 // What's going on: large 64-bit "unsigned" looks like
3397 // negative number to hardware's integer-to-float
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003398 // conversion. However, because the mantissa is only
David Chase42825882015-08-20 15:14:20 -04003399 // 63 bits, we don't need the LSB, so instead we do an
3400 // unsigned right shift (divide by two), convert, and
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003401 // double. However, before we do that, we need to be
David Chase42825882015-08-20 15:14:20 -04003402 // sure that we do not lose a "1" if that made the
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003403 // difference in the resulting rounding. Therefore, we
3404 // preserve it, and OR (not ADD) it back in. The case
David Chase42825882015-08-20 15:14:20 -04003405 // that matters is when the eleven discarded bits are
3406 // equal to 10000000001; that rounds up, and the 1 cannot
3407 // be lost else it would round down if the LSB of the
3408 // candidate mantissa is 0.
3409 cmp := s.newValue2(cvttab.geq, Types[TBOOL], x, s.zeroVal(ft))
3410 b := s.endBlock()
3411 b.Kind = ssa.BlockIf
Keith Randall56e0ecc2016-03-15 20:45:50 -07003412 b.SetControl(cmp)
David Chase42825882015-08-20 15:14:20 -04003413 b.Likely = ssa.BranchLikely
3414
3415 bThen := s.f.NewBlock(ssa.BlockPlain)
3416 bElse := s.f.NewBlock(ssa.BlockPlain)
3417 bAfter := s.f.NewBlock(ssa.BlockPlain)
3418
Todd Neal47d67992015-08-28 21:36:29 -05003419 b.AddEdgeTo(bThen)
David Chase42825882015-08-20 15:14:20 -04003420 s.startBlock(bThen)
3421 a0 := s.newValue1(cvttab.cvt2F, tt, x)
3422 s.vars[n] = a0
3423 s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -05003424 bThen.AddEdgeTo(bAfter)
David Chase42825882015-08-20 15:14:20 -04003425
Todd Neal47d67992015-08-28 21:36:29 -05003426 b.AddEdgeTo(bElse)
David Chase42825882015-08-20 15:14:20 -04003427 s.startBlock(bElse)
3428 one := cvttab.one(s, ft, 1)
3429 y := s.newValue2(cvttab.and, ft, x, one)
3430 z := s.newValue2(cvttab.rsh, ft, x, one)
3431 z = s.newValue2(cvttab.or, ft, z, y)
3432 a := s.newValue1(cvttab.cvt2F, tt, z)
3433 a1 := s.newValue2(cvttab.add, tt, a, a)
3434 s.vars[n] = a1
3435 s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -05003436 bElse.AddEdgeTo(bAfter)
David Chase42825882015-08-20 15:14:20 -04003437
3438 s.startBlock(bAfter)
3439 return s.variable(n, n.Type)
3440}
3441
Todd Neal707af252015-08-28 15:56:43 -05003442// referenceTypeBuiltin generates code for the len/cap builtins for maps and channels.
3443func (s *state) referenceTypeBuiltin(n *Node, x *ssa.Value) *ssa.Value {
3444 if !n.Left.Type.IsMap() && !n.Left.Type.IsChan() {
3445 s.Fatalf("node must be a map or a channel")
3446 }
Todd Neale0e40682015-08-26 18:40:52 -05003447 // if n == nil {
3448 // return 0
3449 // } else {
Todd Neal707af252015-08-28 15:56:43 -05003450 // // len
Todd Neale0e40682015-08-26 18:40:52 -05003451 // return *((*int)n)
Todd Neal707af252015-08-28 15:56:43 -05003452 // // cap
3453 // return *(((*int)n)+1)
Todd Neale0e40682015-08-26 18:40:52 -05003454 // }
3455 lenType := n.Type
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08003456 nilValue := s.constNil(Types[TUINTPTR])
Todd Neal67ac8a32015-08-28 15:20:54 -05003457 cmp := s.newValue2(ssa.OpEqPtr, Types[TBOOL], x, nilValue)
Todd Neale0e40682015-08-26 18:40:52 -05003458 b := s.endBlock()
3459 b.Kind = ssa.BlockIf
Keith Randall56e0ecc2016-03-15 20:45:50 -07003460 b.SetControl(cmp)
Todd Neale0e40682015-08-26 18:40:52 -05003461 b.Likely = ssa.BranchUnlikely
3462
3463 bThen := s.f.NewBlock(ssa.BlockPlain)
3464 bElse := s.f.NewBlock(ssa.BlockPlain)
3465 bAfter := s.f.NewBlock(ssa.BlockPlain)
3466
Todd Neal707af252015-08-28 15:56:43 -05003467 // length/capacity of a nil map/chan is zero
Todd Neal47d67992015-08-28 21:36:29 -05003468 b.AddEdgeTo(bThen)
Todd Neale0e40682015-08-26 18:40:52 -05003469 s.startBlock(bThen)
3470 s.vars[n] = s.zeroVal(lenType)
3471 s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -05003472 bThen.AddEdgeTo(bAfter)
Todd Neale0e40682015-08-26 18:40:52 -05003473
Todd Neal47d67992015-08-28 21:36:29 -05003474 b.AddEdgeTo(bElse)
Todd Neale0e40682015-08-26 18:40:52 -05003475 s.startBlock(bElse)
Todd Neal707af252015-08-28 15:56:43 -05003476 if n.Op == OLEN {
3477 // length is stored in the first word for map/chan
3478 s.vars[n] = s.newValue2(ssa.OpLoad, lenType, x, s.mem())
3479 } else if n.Op == OCAP {
3480 // capacity is stored in the second word for chan
3481 sw := s.newValue1I(ssa.OpOffPtr, lenType.PtrTo(), lenType.Width, x)
3482 s.vars[n] = s.newValue2(ssa.OpLoad, lenType, sw, s.mem())
3483 } else {
3484 s.Fatalf("op must be OLEN or OCAP")
3485 }
Todd Neale0e40682015-08-26 18:40:52 -05003486 s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -05003487 bElse.AddEdgeTo(bAfter)
Todd Neale0e40682015-08-26 18:40:52 -05003488
3489 s.startBlock(bAfter)
3490 return s.variable(n, lenType)
3491}
3492
David Chase73151062015-08-26 14:25:40 -04003493type f2uCvtTab struct {
3494 ltf, cvt2U, subf ssa.Op
3495 value func(*state, ssa.Type, float64) *ssa.Value
3496}
3497
3498var f32_u64 f2uCvtTab = f2uCvtTab{
3499 ltf: ssa.OpLess32F,
3500 cvt2U: ssa.OpCvt32Fto64,
3501 subf: ssa.OpSub32F,
3502 value: (*state).constFloat32,
3503}
3504
3505var f64_u64 f2uCvtTab = f2uCvtTab{
3506 ltf: ssa.OpLess64F,
3507 cvt2U: ssa.OpCvt64Fto64,
3508 subf: ssa.OpSub64F,
3509 value: (*state).constFloat64,
3510}
3511
3512func (s *state) float32ToUint64(n *Node, x *ssa.Value, ft, tt *Type) *ssa.Value {
3513 return s.floatToUint(&f32_u64, n, x, ft, tt)
3514}
3515func (s *state) float64ToUint64(n *Node, x *ssa.Value, ft, tt *Type) *ssa.Value {
3516 return s.floatToUint(&f64_u64, n, x, ft, tt)
3517}
3518
3519func (s *state) floatToUint(cvttab *f2uCvtTab, n *Node, x *ssa.Value, ft, tt *Type) *ssa.Value {
3520 // if x < 9223372036854775808.0 {
3521 // result = uintY(x)
3522 // } else {
3523 // y = x - 9223372036854775808.0
3524 // z = uintY(y)
3525 // result = z | -9223372036854775808
3526 // }
3527 twoToThe63 := cvttab.value(s, ft, 9223372036854775808.0)
3528 cmp := s.newValue2(cvttab.ltf, Types[TBOOL], x, twoToThe63)
3529 b := s.endBlock()
3530 b.Kind = ssa.BlockIf
Keith Randall56e0ecc2016-03-15 20:45:50 -07003531 b.SetControl(cmp)
David Chase73151062015-08-26 14:25:40 -04003532 b.Likely = ssa.BranchLikely
3533
3534 bThen := s.f.NewBlock(ssa.BlockPlain)
3535 bElse := s.f.NewBlock(ssa.BlockPlain)
3536 bAfter := s.f.NewBlock(ssa.BlockPlain)
3537
Todd Neal47d67992015-08-28 21:36:29 -05003538 b.AddEdgeTo(bThen)
David Chase73151062015-08-26 14:25:40 -04003539 s.startBlock(bThen)
3540 a0 := s.newValue1(cvttab.cvt2U, tt, x)
3541 s.vars[n] = a0
3542 s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -05003543 bThen.AddEdgeTo(bAfter)
David Chase73151062015-08-26 14:25:40 -04003544
Todd Neal47d67992015-08-28 21:36:29 -05003545 b.AddEdgeTo(bElse)
David Chase73151062015-08-26 14:25:40 -04003546 s.startBlock(bElse)
3547 y := s.newValue2(cvttab.subf, ft, x, twoToThe63)
3548 y = s.newValue1(cvttab.cvt2U, tt, y)
3549 z := s.constInt64(tt, -9223372036854775808)
3550 a1 := s.newValue2(ssa.OpOr64, tt, y, z)
3551 s.vars[n] = a1
3552 s.endBlock()
Todd Neal47d67992015-08-28 21:36:29 -05003553 bElse.AddEdgeTo(bAfter)
David Chase73151062015-08-26 14:25:40 -04003554
3555 s.startBlock(bAfter)
3556 return s.variable(n, n.Type)
3557}
3558
Keith Randall269baa92015-09-17 10:31:16 -07003559// ifaceType returns the value for the word containing the type.
3560// n is the node for the interface expression.
3561// v is the corresponding value.
3562func (s *state) ifaceType(n *Node, v *ssa.Value) *ssa.Value {
3563 byteptr := Ptrto(Types[TUINT8]) // type used in runtime prototypes for runtime type (*byte)
3564
Matthew Dempsky00e5a682016-04-01 13:36:24 -07003565 if n.Type.IsEmptyInterface() {
Keith Randall269baa92015-09-17 10:31:16 -07003566 // Have *eface. The type is the first word in the struct.
3567 return s.newValue1(ssa.OpITab, byteptr, v)
3568 }
3569
3570 // Have *iface.
3571 // The first word in the struct is the *itab.
3572 // If the *itab is nil, return 0.
3573 // Otherwise, the second word in the *itab is the type.
3574
3575 tab := s.newValue1(ssa.OpITab, byteptr, v)
3576 s.vars[&typVar] = tab
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08003577 isnonnil := s.newValue2(ssa.OpNeqPtr, Types[TBOOL], tab, s.constNil(byteptr))
Keith Randall269baa92015-09-17 10:31:16 -07003578 b := s.endBlock()
3579 b.Kind = ssa.BlockIf
Keith Randall56e0ecc2016-03-15 20:45:50 -07003580 b.SetControl(isnonnil)
Keith Randall269baa92015-09-17 10:31:16 -07003581 b.Likely = ssa.BranchLikely
3582
3583 bLoad := s.f.NewBlock(ssa.BlockPlain)
3584 bEnd := s.f.NewBlock(ssa.BlockPlain)
3585
3586 b.AddEdgeTo(bLoad)
3587 b.AddEdgeTo(bEnd)
3588 bLoad.AddEdgeTo(bEnd)
3589
3590 s.startBlock(bLoad)
3591 off := s.newValue1I(ssa.OpOffPtr, byteptr, int64(Widthptr), tab)
3592 s.vars[&typVar] = s.newValue2(ssa.OpLoad, byteptr, off, s.mem())
3593 s.endBlock()
3594
3595 s.startBlock(bEnd)
3596 typ := s.variable(&typVar, byteptr)
3597 delete(s.vars, &typVar)
3598 return typ
3599}
3600
3601// dottype generates SSA for a type assertion node.
3602// commaok indicates whether to panic or return a bool.
3603// If commaok is false, resok will be nil.
3604func (s *state) dottype(n *Node, commaok bool) (res, resok *ssa.Value) {
3605 iface := s.expr(n.Left)
3606 typ := s.ifaceType(n.Left, iface) // actual concrete type
3607 target := s.expr(typename(n.Type)) // target type
3608 if !isdirectiface(n.Type) {
3609 // walk rewrites ODOTTYPE/OAS2DOTTYPE into runtime calls except for this case.
3610 Fatalf("dottype needs a direct iface type %s", n.Type)
3611 }
3612
David Chase729abfa2015-10-26 17:34:06 -04003613 if Debug_typeassert > 0 {
Robert Griesemerb83f3972016-03-02 11:01:25 -08003614 Warnl(n.Lineno, "type assertion inlined")
David Chase729abfa2015-10-26 17:34:06 -04003615 }
3616
Keith Randall269baa92015-09-17 10:31:16 -07003617 // TODO: If we have a nonempty interface and its itab field is nil,
3618 // then this test is redundant and ifaceType should just branch directly to bFail.
3619 cond := s.newValue2(ssa.OpEqPtr, Types[TBOOL], typ, target)
3620 b := s.endBlock()
3621 b.Kind = ssa.BlockIf
Keith Randall56e0ecc2016-03-15 20:45:50 -07003622 b.SetControl(cond)
Keith Randall269baa92015-09-17 10:31:16 -07003623 b.Likely = ssa.BranchLikely
3624
3625 byteptr := Ptrto(Types[TUINT8])
3626
3627 bOk := s.f.NewBlock(ssa.BlockPlain)
3628 bFail := s.f.NewBlock(ssa.BlockPlain)
3629 b.AddEdgeTo(bOk)
3630 b.AddEdgeTo(bFail)
3631
3632 if !commaok {
3633 // on failure, panic by calling panicdottype
3634 s.startBlock(bFail)
Keith Randallcd956572016-04-29 09:02:27 -07003635 taddr := s.newValue1A(ssa.OpAddr, byteptr, &ssa.ExternSymbol{Typ: byteptr, Sym: typenamesym(n.Left.Type)}, s.sb)
Keith Randall8c5bfcc2015-09-18 15:11:30 -07003636 s.rtcall(panicdottype, false, nil, typ, target, taddr)
Keith Randall269baa92015-09-17 10:31:16 -07003637
3638 // on success, return idata field
3639 s.startBlock(bOk)
3640 return s.newValue1(ssa.OpIData, n.Type, iface), nil
3641 }
3642
3643 // commaok is the more complicated case because we have
3644 // a control flow merge point.
3645 bEnd := s.f.NewBlock(ssa.BlockPlain)
3646
3647 // type assertion succeeded
3648 s.startBlock(bOk)
3649 s.vars[&idataVar] = s.newValue1(ssa.OpIData, n.Type, iface)
3650 s.vars[&okVar] = s.constBool(true)
3651 s.endBlock()
3652 bOk.AddEdgeTo(bEnd)
3653
3654 // type assertion failed
3655 s.startBlock(bFail)
Josh Bleecher Snyder39214272016-03-06 18:06:09 -08003656 s.vars[&idataVar] = s.constNil(byteptr)
Keith Randall269baa92015-09-17 10:31:16 -07003657 s.vars[&okVar] = s.constBool(false)
3658 s.endBlock()
3659 bFail.AddEdgeTo(bEnd)
3660
3661 // merge point
3662 s.startBlock(bEnd)
3663 res = s.variable(&idataVar, byteptr)
3664 resok = s.variable(&okVar, Types[TBOOL])
3665 delete(s.vars, &idataVar)
3666 delete(s.vars, &okVar)
3667 return res, resok
3668}
3669
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -07003670// checkgoto checks that a goto from from to to does not
3671// jump into a block or jump over variable declarations.
3672// It is a copy of checkgoto in the pre-SSA backend,
3673// modified only for line number handling.
3674// TODO: document how this works and why it is designed the way it is.
3675func (s *state) checkgoto(from *Node, to *Node) {
3676 if from.Sym == to.Sym {
3677 return
3678 }
3679
3680 nf := 0
3681 for fs := from.Sym; fs != nil; fs = fs.Link {
3682 nf++
3683 }
3684 nt := 0
3685 for fs := to.Sym; fs != nil; fs = fs.Link {
3686 nt++
3687 }
3688 fs := from.Sym
3689 for ; nf > nt; nf-- {
3690 fs = fs.Link
3691 }
3692 if fs != to.Sym {
3693 // decide what to complain about.
3694 // prefer to complain about 'into block' over declarations,
3695 // so scan backward to find most recent block or else dcl.
3696 var block *Sym
3697
3698 var dcl *Sym
3699 ts := to.Sym
3700 for ; nt > nf; nt-- {
3701 if ts.Pkg == nil {
3702 block = ts
3703 } else {
3704 dcl = ts
3705 }
3706 ts = ts.Link
3707 }
3708
3709 for ts != fs {
3710 if ts.Pkg == nil {
3711 block = ts
3712 } else {
3713 dcl = ts
3714 }
3715 ts = ts.Link
3716 fs = fs.Link
3717 }
3718
Robert Griesemerb83f3972016-03-02 11:01:25 -08003719 lno := from.Left.Lineno
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -07003720 if block != nil {
Robert Griesemer2faf5bc2016-03-02 11:30:29 -08003721 yyerrorl(lno, "goto %v jumps into block starting at %v", from.Left.Sym, linestr(block.Lastlineno))
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -07003722 } else {
Robert Griesemer2faf5bc2016-03-02 11:30:29 -08003723 yyerrorl(lno, "goto %v jumps over declaration of %v at %v", from.Left.Sym, dcl, linestr(dcl.Lastlineno))
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -07003724 }
3725 }
3726}
3727
Keith Randalld2fd43a2015-04-15 15:51:25 -07003728// variable returns the value of a variable at the current location.
Keith Randall8c46aa52015-06-19 21:02:28 -07003729func (s *state) variable(name *Node, t ssa.Type) *ssa.Value {
Keith Randalld2fd43a2015-04-15 15:51:25 -07003730 v := s.vars[name]
3731 if v == nil {
Keith Randall8f22b522015-06-11 21:29:25 -07003732 v = s.newValue0A(ssa.OpFwdRef, t, name)
Keith Randallb5c5efd2016-01-14 16:02:23 -08003733 s.fwdRefs = append(s.fwdRefs, v)
Keith Randalld2fd43a2015-04-15 15:51:25 -07003734 s.vars[name] = v
Keith Randallb5c5efd2016-01-14 16:02:23 -08003735 s.addNamedValue(name, v)
Keith Randalld2fd43a2015-04-15 15:51:25 -07003736 }
3737 return v
3738}
3739
Keith Randallcfc2aa52015-05-18 16:44:20 -07003740func (s *state) mem() *ssa.Value {
Keith Randallb32217a2015-09-17 16:45:10 -07003741 return s.variable(&memVar, ssa.TypeMem)
Keith Randalld2fd43a2015-04-15 15:51:25 -07003742}
3743
David Chase6b99fb52016-04-21 13:24:58 -04003744func (s *state) linkForwardReferences(dm *sparseDefState) {
3745
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003746 // Build SSA graph. Each variable on its first use in a basic block
Keith Randalld2fd43a2015-04-15 15:51:25 -07003747 // leaves a FwdRef in that block representing the incoming value
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003748 // of that variable. This function links that ref up with possible definitions,
3749 // inserting Phi values as needed. This is essentially the algorithm
Keith Randallb5c5efd2016-01-14 16:02:23 -08003750 // described by Braun, Buchwald, Hack, Leißa, Mallon, and Zwinkau:
Keith Randalld2fd43a2015-04-15 15:51:25 -07003751 // http://pp.info.uni-karlsruhe.de/uploads/publikationen/braun13cc.pdf
Keith Randallb5c5efd2016-01-14 16:02:23 -08003752 // Differences:
3753 // - We use FwdRef nodes to postpone phi building until the CFG is
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003754 // completely built. That way we can avoid the notion of "sealed"
Keith Randallb5c5efd2016-01-14 16:02:23 -08003755 // blocks.
3756 // - Phi optimization is a separate pass (in ../ssa/phielim.go).
3757 for len(s.fwdRefs) > 0 {
3758 v := s.fwdRefs[len(s.fwdRefs)-1]
3759 s.fwdRefs = s.fwdRefs[:len(s.fwdRefs)-1]
David Chase6b99fb52016-04-21 13:24:58 -04003760 s.resolveFwdRef(v, dm)
Keith Randalld2fd43a2015-04-15 15:51:25 -07003761 }
3762}
3763
Keith Randallb5c5efd2016-01-14 16:02:23 -08003764// resolveFwdRef modifies v to be the variable's value at the start of its block.
3765// v must be a FwdRef op.
David Chase6b99fb52016-04-21 13:24:58 -04003766func (s *state) resolveFwdRef(v *ssa.Value, dm *sparseDefState) {
Keith Randallb5c5efd2016-01-14 16:02:23 -08003767 b := v.Block
3768 name := v.Aux.(*Node)
3769 v.Aux = nil
Keith Randalld2fd43a2015-04-15 15:51:25 -07003770 if b == s.f.Entry {
Keith Randallb5c5efd2016-01-14 16:02:23 -08003771 // Live variable at start of function.
Keith Randall6a8a9da2016-02-27 17:49:31 -08003772 if s.canSSA(name) {
David Chasec3b3e7b2016-04-08 13:33:43 -04003773 if strings.HasPrefix(name.Sym.Name, "autotmp_") {
3774 // It's likely that this is an uninitialized variable in the entry block.
3775 s.Fatalf("Treating auto as if it were arg, func %s, node %v, value %v", b.Func.Name, name, v)
3776 }
Keith Randallb5c5efd2016-01-14 16:02:23 -08003777 v.Op = ssa.OpArg
3778 v.Aux = name
3779 return
Keith Randall02f4d0a2015-11-02 08:10:26 -08003780 }
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003781 // Not SSAable. Load it.
Keith Randall8c46aa52015-06-19 21:02:28 -07003782 addr := s.decladdrs[name]
3783 if addr == nil {
3784 // TODO: closure args reach here.
David Chase32ffbf72015-10-08 17:14:12 -04003785 s.Unimplementedf("unhandled closure arg %s at entry to function %s", name, b.Func.Name)
Keith Randall8c46aa52015-06-19 21:02:28 -07003786 }
3787 if _, ok := addr.Aux.(*ssa.ArgSymbol); !ok {
3788 s.Fatalf("variable live at start of function %s is not an argument %s", b.Func.Name, name)
3789 }
Keith Randallb5c5efd2016-01-14 16:02:23 -08003790 v.Op = ssa.OpLoad
3791 v.AddArgs(addr, s.startmem)
3792 return
Keith Randalld2fd43a2015-04-15 15:51:25 -07003793 }
Keith Randallb5c5efd2016-01-14 16:02:23 -08003794 if len(b.Preds) == 0 {
Josh Bleecher Snyder61aa0952015-07-20 15:39:14 -07003795 // This block is dead; we have no predecessors and we're not the entry block.
Keith Randallb5c5efd2016-01-14 16:02:23 -08003796 // It doesn't matter what we use here as long as it is well-formed.
3797 v.Op = ssa.OpUnknown
3798 return
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07003799 }
Keith Randallb5c5efd2016-01-14 16:02:23 -08003800 // Find variable value on each predecessor.
3801 var argstore [4]*ssa.Value
3802 args := argstore[:0]
Keith Randall4fa05002016-04-28 16:52:47 -07003803 for _, e := range b.Preds {
3804 p := e.Block()
David Chase6b99fb52016-04-21 13:24:58 -04003805 p = dm.FindBetterDefiningBlock(name, p) // try sparse improvement on p
Keith Randallb5c5efd2016-01-14 16:02:23 -08003806 args = append(args, s.lookupVarOutgoing(p, v.Type, name, v.Line))
Keith Randalld2fd43a2015-04-15 15:51:25 -07003807 }
Keith Randallb5c5efd2016-01-14 16:02:23 -08003808
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003809 // Decide if we need a phi or not. We need a phi if there
Keith Randallb5c5efd2016-01-14 16:02:23 -08003810 // are two different args (which are both not v).
3811 var w *ssa.Value
3812 for _, a := range args {
3813 if a == v {
3814 continue // self-reference
3815 }
3816 if a == w {
3817 continue // already have this witness
3818 }
3819 if w != nil {
3820 // two witnesses, need a phi value
3821 v.Op = ssa.OpPhi
3822 v.AddArgs(args...)
3823 return
3824 }
3825 w = a // save witness
3826 }
3827 if w == nil {
3828 s.Fatalf("no witness for reachable phi %s", v)
3829 }
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00003830 // One witness. Make v a copy of w.
Keith Randallb5c5efd2016-01-14 16:02:23 -08003831 v.Op = ssa.OpCopy
3832 v.AddArg(w)
Keith Randalld2fd43a2015-04-15 15:51:25 -07003833}
3834
3835// lookupVarOutgoing finds the variable's value at the end of block b.
Keith Randallb5c5efd2016-01-14 16:02:23 -08003836func (s *state) lookupVarOutgoing(b *ssa.Block, t ssa.Type, name *Node, line int32) *ssa.Value {
Josh Bleecher Snyderbc1989f2016-05-04 14:04:03 -07003837 for {
3838 if v, ok := s.defvars[b.ID][name]; ok {
3839 return v
3840 }
3841 // The variable is not defined by b and we haven't looked it up yet.
3842 // If b has exactly one predecessor, loop to look it up there.
3843 // Otherwise, give up and insert a new FwdRef and resolve it later.
3844 if len(b.Preds) != 1 {
3845 break
3846 }
3847 b = b.Preds[0].Block()
Keith Randalld2fd43a2015-04-15 15:51:25 -07003848 }
Josh Bleecher Snyderbc1989f2016-05-04 14:04:03 -07003849 // Generate a FwdRef for the variable and return that.
Keith Randallb5c5efd2016-01-14 16:02:23 -08003850 v := b.NewValue0A(line, ssa.OpFwdRef, t, name)
3851 s.fwdRefs = append(s.fwdRefs, v)
Josh Bleecher Snyderbc1989f2016-05-04 14:04:03 -07003852 s.defvars[b.ID][name] = v
Keith Randallb5c5efd2016-01-14 16:02:23 -08003853 s.addNamedValue(name, v)
Keith Randalld2fd43a2015-04-15 15:51:25 -07003854 return v
3855}
3856
Keith Randallc24681a2015-10-22 14:22:38 -07003857func (s *state) addNamedValue(n *Node, v *ssa.Value) {
3858 if n.Class == Pxxx {
3859 // Don't track our dummy nodes (&memVar etc.).
3860 return
3861 }
Keith Randallc24681a2015-10-22 14:22:38 -07003862 if strings.HasPrefix(n.Sym.Name, "autotmp_") {
3863 // Don't track autotmp_ variables.
3864 return
3865 }
Keith Randall31d13f42016-03-08 20:09:48 -08003866 if n.Class == PPARAMOUT {
3867 // Don't track named output values. This prevents return values
3868 // from being assigned too early. See #14591 and #14762. TODO: allow this.
3869 return
3870 }
Keith Randallc24681a2015-10-22 14:22:38 -07003871 if n.Class == PAUTO && n.Xoffset != 0 {
3872 s.Fatalf("AUTO var with offset %s %d", n, n.Xoffset)
3873 }
Keith Randall02f4d0a2015-11-02 08:10:26 -08003874 loc := ssa.LocalSlot{N: n, Type: n.Type, Off: 0}
3875 values, ok := s.f.NamedValues[loc]
Keith Randallc24681a2015-10-22 14:22:38 -07003876 if !ok {
Keith Randall02f4d0a2015-11-02 08:10:26 -08003877 s.f.Names = append(s.f.Names, loc)
Keith Randallc24681a2015-10-22 14:22:38 -07003878 }
Keith Randall02f4d0a2015-11-02 08:10:26 -08003879 s.f.NamedValues[loc] = append(values, v)
Keith Randallc24681a2015-10-22 14:22:38 -07003880}
3881
Michael Pratta4e31d42016-03-12 14:07:40 -08003882// Branch is an unresolved branch.
3883type Branch struct {
3884 P *obj.Prog // branch instruction
3885 B *ssa.Block // target
Keith Randall083a6462015-05-12 11:06:44 -07003886}
3887
Michael Pratta4e31d42016-03-12 14:07:40 -08003888// SSAGenState contains state needed during Prog generation.
3889type SSAGenState struct {
3890 // Branches remembers all the branch instructions we've seen
Keith Randall9569b952015-08-28 22:51:01 -07003891 // and where they would like to go.
Michael Pratta4e31d42016-03-12 14:07:40 -08003892 Branches []Branch
Keith Randall9569b952015-08-28 22:51:01 -07003893
3894 // bstart remembers where each block starts (indexed by block ID)
3895 bstart []*obj.Prog
Keith Randall9569b952015-08-28 22:51:01 -07003896}
3897
Michael Pratta4e31d42016-03-12 14:07:40 -08003898// Pc returns the current Prog.
3899func (s *SSAGenState) Pc() *obj.Prog {
3900 return Pc
3901}
3902
3903// SetLineno sets the current source line number.
3904func (s *SSAGenState) SetLineno(l int32) {
3905 lineno = l
3906}
3907
Keith Randall083a6462015-05-12 11:06:44 -07003908// genssa appends entries to ptxt for each instruction in f.
3909// gcargs and gclocals are filled in with pointer maps for the frame.
3910func genssa(f *ssa.Func, ptxt *obj.Prog, gcargs, gclocals *Sym) {
Michael Pratta4e31d42016-03-12 14:07:40 -08003911 var s SSAGenState
Keith Randall9569b952015-08-28 22:51:01 -07003912
Josh Bleecher Snyderd2982092015-07-22 13:13:53 -07003913 e := f.Config.Frontend().(*ssaExport)
3914 // We're about to emit a bunch of Progs.
3915 // Since the only way to get here is to explicitly request it,
3916 // just fail on unimplemented instead of trying to unwind our mess.
3917 e.mustImplement = true
3918
Keith Randall083a6462015-05-12 11:06:44 -07003919 // Remember where each block starts.
Keith Randall9569b952015-08-28 22:51:01 -07003920 s.bstart = make([]*obj.Prog, f.NumBlocks())
Keith Randall083a6462015-05-12 11:06:44 -07003921
Josh Bleecher Snyderb8efee02015-07-31 14:37:15 -07003922 var valueProgs map[*obj.Prog]*ssa.Value
3923 var blockProgs map[*obj.Prog]*ssa.Block
Dave Cheneycb1f2af2016-03-17 13:46:43 +11003924 var logProgs = e.log
Josh Bleecher Snyderb8efee02015-07-31 14:37:15 -07003925 if logProgs {
3926 valueProgs = make(map[*obj.Prog]*ssa.Value, f.NumValues())
3927 blockProgs = make(map[*obj.Prog]*ssa.Block, f.NumBlocks())
3928 f.Logf("genssa %s\n", f.Name)
3929 blockProgs[Pc] = f.Blocks[0]
3930 }
3931
Keith Randall083a6462015-05-12 11:06:44 -07003932 // Emit basic blocks
3933 for i, b := range f.Blocks {
Keith Randall9569b952015-08-28 22:51:01 -07003934 s.bstart[b.ID] = Pc
Keith Randall083a6462015-05-12 11:06:44 -07003935 // Emit values in block
Michael Pratta4e31d42016-03-12 14:07:40 -08003936 Thearch.SSAMarkMoves(&s, b)
Keith Randall083a6462015-05-12 11:06:44 -07003937 for _, v := range b.Values {
Josh Bleecher Snyderb8efee02015-07-31 14:37:15 -07003938 x := Pc
Michael Pratta4e31d42016-03-12 14:07:40 -08003939 Thearch.SSAGenValue(&s, v)
Josh Bleecher Snyderb8efee02015-07-31 14:37:15 -07003940 if logProgs {
3941 for ; x != Pc; x = x.Link {
3942 valueProgs[x] = v
3943 }
3944 }
Keith Randall083a6462015-05-12 11:06:44 -07003945 }
3946 // Emit control flow instructions for block
3947 var next *ssa.Block
Keith Randall91f69c62016-02-26 16:32:01 -08003948 if i < len(f.Blocks)-1 && (Debug['N'] == 0 || b.Kind == ssa.BlockCall) {
Keith Randall8906d2a2016-02-22 23:19:00 -08003949 // If -N, leave next==nil so every block with successors
Keith Randall91f69c62016-02-26 16:32:01 -08003950 // ends in a JMP (except call blocks - plive doesn't like
3951 // select{send,recv} followed by a JMP call). Helps keep
3952 // line numbers for otherwise empty blocks.
Keith Randall083a6462015-05-12 11:06:44 -07003953 next = f.Blocks[i+1]
3954 }
Josh Bleecher Snyderb8efee02015-07-31 14:37:15 -07003955 x := Pc
Michael Pratta4e31d42016-03-12 14:07:40 -08003956 Thearch.SSAGenBlock(&s, b, next)
Josh Bleecher Snyderb8efee02015-07-31 14:37:15 -07003957 if logProgs {
3958 for ; x != Pc; x = x.Link {
3959 blockProgs[x] = b
3960 }
3961 }
Keith Randall083a6462015-05-12 11:06:44 -07003962 }
3963
3964 // Resolve branches
Michael Pratta4e31d42016-03-12 14:07:40 -08003965 for _, br := range s.Branches {
3966 br.P.To.Val = s.bstart[br.B.ID]
Keith Randall9569b952015-08-28 22:51:01 -07003967 }
Keith Randall083a6462015-05-12 11:06:44 -07003968
Josh Bleecher Snyderb8efee02015-07-31 14:37:15 -07003969 if logProgs {
3970 for p := ptxt; p != nil; p = p.Link {
3971 var s string
3972 if v, ok := valueProgs[p]; ok {
3973 s = v.String()
3974 } else if b, ok := blockProgs[p]; ok {
3975 s = b.String()
3976 } else {
3977 s = " " // most value and branch strings are 2-3 characters long
3978 }
3979 f.Logf("%s\t%s\n", s, p)
3980 }
Josh Bleecher Snyder35fb5142015-08-10 12:15:52 -07003981 if f.Config.HTML != nil {
3982 saved := ptxt.Ctxt.LineHist.PrintFilenameOnly
3983 ptxt.Ctxt.LineHist.PrintFilenameOnly = true
3984 var buf bytes.Buffer
3985 buf.WriteString("<code>")
3986 buf.WriteString("<dl class=\"ssa-gen\">")
3987 for p := ptxt; p != nil; p = p.Link {
3988 buf.WriteString("<dt class=\"ssa-prog-src\">")
3989 if v, ok := valueProgs[p]; ok {
3990 buf.WriteString(v.HTML())
3991 } else if b, ok := blockProgs[p]; ok {
3992 buf.WriteString(b.HTML())
3993 }
3994 buf.WriteString("</dt>")
3995 buf.WriteString("<dd class=\"ssa-prog\">")
3996 buf.WriteString(html.EscapeString(p.String()))
3997 buf.WriteString("</dd>")
3998 buf.WriteString("</li>")
3999 }
4000 buf.WriteString("</dl>")
4001 buf.WriteString("</code>")
4002 f.Config.HTML.WriteColumn("genssa", buf.String())
4003 ptxt.Ctxt.LineHist.PrintFilenameOnly = saved
4004 }
Josh Bleecher Snyderb8efee02015-07-31 14:37:15 -07004005 }
4006
Josh Bleecher Snyder6b416652015-07-28 10:56:39 -07004007 // Emit static data
4008 if f.StaticData != nil {
4009 for _, n := range f.StaticData.([]*Node) {
4010 if !gen_as_init(n, false) {
Keith Randallcd956572016-04-29 09:02:27 -07004011 Fatalf("non-static data marked as static: %v\n\n", n)
Josh Bleecher Snyder6b416652015-07-28 10:56:39 -07004012 }
4013 }
4014 }
4015
Keith Randalld2107fc2015-08-24 02:16:19 -07004016 // Allocate stack frame
4017 allocauto(ptxt)
Keith Randall083a6462015-05-12 11:06:44 -07004018
Keith Randalld2107fc2015-08-24 02:16:19 -07004019 // Generate gc bitmaps.
4020 liveness(Curfn, ptxt, gcargs, gclocals)
Keith Randall083a6462015-05-12 11:06:44 -07004021
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00004022 // Add frame prologue. Zero ambiguously live variables.
Keith Randalld2107fc2015-08-24 02:16:19 -07004023 Thearch.Defframe(ptxt)
4024 if Debug['f'] != 0 {
4025 frame(0)
4026 }
4027
4028 // Remove leftover instrumentation from the instruction stream.
4029 removevardef(ptxt)
Josh Bleecher Snyder35fb5142015-08-10 12:15:52 -07004030
4031 f.Config.HTML.Close()
Keith Randall083a6462015-05-12 11:06:44 -07004032}
4033
Daniel Morsing66b47812015-06-27 15:45:20 +01004034// movZero generates a register indirect move with a 0 immediate and keeps track of bytes left and next offset
Matthew Dempsky0d9258a2016-03-07 18:00:08 -08004035func movZero(as obj.As, width int64, nbytes int64, offset int64, regnum int16) (nleft int64, noff int64) {
Daniel Morsing66b47812015-06-27 15:45:20 +01004036 p := Prog(as)
4037 // TODO: use zero register on archs that support it.
4038 p.From.Type = obj.TYPE_CONST
4039 p.From.Offset = 0
4040 p.To.Type = obj.TYPE_MEM
4041 p.To.Reg = regnum
4042 p.To.Offset = offset
4043 offset += width
4044 nleft = nbytes - width
4045 return nleft, offset
4046}
4047
Michael Pratta4e31d42016-03-12 14:07:40 -08004048type FloatingEQNEJump struct {
4049 Jump obj.As
4050 Index int
David Chase8e601b22015-08-18 14:39:26 -04004051}
4052
Michael Pratta4e31d42016-03-12 14:07:40 -08004053func oneFPJump(b *ssa.Block, jumps *FloatingEQNEJump, likely ssa.BranchPrediction, branches []Branch) []Branch {
4054 p := Prog(jumps.Jump)
David Chase8e601b22015-08-18 14:39:26 -04004055 p.To.Type = obj.TYPE_BRANCH
Michael Pratta4e31d42016-03-12 14:07:40 -08004056 to := jumps.Index
Keith Randall4fa05002016-04-28 16:52:47 -07004057 branches = append(branches, Branch{p, b.Succs[to].Block()})
David Chase8e601b22015-08-18 14:39:26 -04004058 if to == 1 {
4059 likely = -likely
4060 }
4061 // liblink reorders the instruction stream as it sees fit.
4062 // Pass along what we know so liblink can make use of it.
4063 // TODO: Once we've fully switched to SSA,
4064 // make liblink leave our output alone.
4065 switch likely {
4066 case ssa.BranchUnlikely:
4067 p.From.Type = obj.TYPE_CONST
4068 p.From.Offset = 0
4069 case ssa.BranchLikely:
4070 p.From.Type = obj.TYPE_CONST
4071 p.From.Offset = 1
4072 }
4073 return branches
4074}
4075
Michael Pratta4e31d42016-03-12 14:07:40 -08004076func SSAGenFPJump(s *SSAGenState, b, next *ssa.Block, jumps *[2][2]FloatingEQNEJump) {
David Chase8e601b22015-08-18 14:39:26 -04004077 likely := b.Likely
4078 switch next {
Keith Randall4fa05002016-04-28 16:52:47 -07004079 case b.Succs[0].Block():
Michael Pratta4e31d42016-03-12 14:07:40 -08004080 s.Branches = oneFPJump(b, &jumps[0][0], likely, s.Branches)
4081 s.Branches = oneFPJump(b, &jumps[0][1], likely, s.Branches)
Keith Randall4fa05002016-04-28 16:52:47 -07004082 case b.Succs[1].Block():
Michael Pratta4e31d42016-03-12 14:07:40 -08004083 s.Branches = oneFPJump(b, &jumps[1][0], likely, s.Branches)
4084 s.Branches = oneFPJump(b, &jumps[1][1], likely, s.Branches)
David Chase8e601b22015-08-18 14:39:26 -04004085 default:
Michael Pratta4e31d42016-03-12 14:07:40 -08004086 s.Branches = oneFPJump(b, &jumps[1][0], likely, s.Branches)
4087 s.Branches = oneFPJump(b, &jumps[1][1], likely, s.Branches)
David Chase8e601b22015-08-18 14:39:26 -04004088 q := Prog(obj.AJMP)
4089 q.To.Type = obj.TYPE_BRANCH
Keith Randall4fa05002016-04-28 16:52:47 -07004090 s.Branches = append(s.Branches, Branch{q, b.Succs[1].Block()})
David Chase8e601b22015-08-18 14:39:26 -04004091 }
Josh Bleecher Snyder71b57072015-07-24 12:47:00 -07004092}
4093
Michael Pratta4e31d42016-03-12 14:07:40 -08004094// AddAux adds the offset in the aux fields (AuxInt and Aux) of v to a.
4095func AddAux(a *obj.Addr, v *ssa.Value) {
4096 AddAux2(a, v, v.AuxInt)
Keith Randall083a6462015-05-12 11:06:44 -07004097}
Michael Pratta4e31d42016-03-12 14:07:40 -08004098func AddAux2(a *obj.Addr, v *ssa.Value, offset int64) {
Keith Randall8c46aa52015-06-19 21:02:28 -07004099 if a.Type != obj.TYPE_MEM {
Keith Randallcd956572016-04-29 09:02:27 -07004100 v.Fatalf("bad AddAux addr %v", a)
Keith Randall8c46aa52015-06-19 21:02:28 -07004101 }
4102 // add integer offset
Keith Randalld43f2e32015-10-21 13:13:56 -07004103 a.Offset += offset
Keith Randall8c46aa52015-06-19 21:02:28 -07004104
4105 // If no additional symbol offset, we're done.
4106 if v.Aux == nil {
4107 return
4108 }
4109 // Add symbol's offset from its base register.
4110 switch sym := v.Aux.(type) {
4111 case *ssa.ExternSymbol:
4112 a.Name = obj.NAME_EXTERN
Ian Lance Taylor65b40202016-03-16 22:22:58 -07004113 switch s := sym.Sym.(type) {
4114 case *Sym:
4115 a.Sym = Linksym(s)
4116 case *obj.LSym:
4117 a.Sym = s
4118 default:
4119 v.Fatalf("ExternSymbol.Sym is %T", s)
4120 }
Keith Randall8c46aa52015-06-19 21:02:28 -07004121 case *ssa.ArgSymbol:
Keith Randalld2107fc2015-08-24 02:16:19 -07004122 n := sym.Node.(*Node)
4123 a.Name = obj.NAME_PARAM
4124 a.Node = n
4125 a.Sym = Linksym(n.Orig.Sym)
4126 a.Offset += n.Xoffset // TODO: why do I have to add this here? I don't for auto variables.
Keith Randall8c46aa52015-06-19 21:02:28 -07004127 case *ssa.AutoSymbol:
Keith Randalld2107fc2015-08-24 02:16:19 -07004128 n := sym.Node.(*Node)
4129 a.Name = obj.NAME_AUTO
4130 a.Node = n
4131 a.Sym = Linksym(n.Sym)
Keith Randall8c46aa52015-06-19 21:02:28 -07004132 default:
4133 v.Fatalf("aux in %s not implemented %#v", v, v.Aux)
4134 }
4135}
4136
Keith Randall582baae2015-11-02 21:28:13 -08004137// extendIndex extends v to a full int width.
Keith Randall2a5e6c42015-07-23 14:35:02 -07004138func (s *state) extendIndex(v *ssa.Value) *ssa.Value {
4139 size := v.Type.Size()
Keith Randall582baae2015-11-02 21:28:13 -08004140 if size == s.config.IntSize {
Keith Randall2a5e6c42015-07-23 14:35:02 -07004141 return v
4142 }
Keith Randall582baae2015-11-02 21:28:13 -08004143 if size > s.config.IntSize {
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00004144 // TODO: truncate 64-bit indexes on 32-bit pointer archs. We'd need to test
Keith Randall2a5e6c42015-07-23 14:35:02 -07004145 // the high word and branch to out-of-bounds failure if it is not 0.
4146 s.Unimplementedf("64->32 index truncation not implemented")
4147 return v
4148 }
4149
4150 // Extend value to the required size
4151 var op ssa.Op
4152 if v.Type.IsSigned() {
Keith Randall582baae2015-11-02 21:28:13 -08004153 switch 10*size + s.config.IntSize {
Keith Randall2a5e6c42015-07-23 14:35:02 -07004154 case 14:
4155 op = ssa.OpSignExt8to32
4156 case 18:
4157 op = ssa.OpSignExt8to64
4158 case 24:
4159 op = ssa.OpSignExt16to32
4160 case 28:
4161 op = ssa.OpSignExt16to64
4162 case 48:
4163 op = ssa.OpSignExt32to64
4164 default:
4165 s.Fatalf("bad signed index extension %s", v.Type)
4166 }
4167 } else {
Keith Randall582baae2015-11-02 21:28:13 -08004168 switch 10*size + s.config.IntSize {
Keith Randall2a5e6c42015-07-23 14:35:02 -07004169 case 14:
4170 op = ssa.OpZeroExt8to32
4171 case 18:
4172 op = ssa.OpZeroExt8to64
4173 case 24:
4174 op = ssa.OpZeroExt16to32
4175 case 28:
4176 op = ssa.OpZeroExt16to64
4177 case 48:
4178 op = ssa.OpZeroExt32to64
4179 default:
4180 s.Fatalf("bad unsigned index extension %s", v.Type)
4181 }
4182 }
Keith Randall582baae2015-11-02 21:28:13 -08004183 return s.newValue1(op, Types[TINT], v)
Keith Randall2a5e6c42015-07-23 14:35:02 -07004184}
4185
Michael Pratta4e31d42016-03-12 14:07:40 -08004186// SSARegNum returns the register (in cmd/internal/obj numbering) to
Brad Fitzpatrick5fea2cc2016-03-01 23:21:55 +00004187// which v has been allocated. Panics if v is not assigned to a
Keith Randall083a6462015-05-12 11:06:44 -07004188// register.
Josh Bleecher Snydere1395492015-08-05 16:06:39 -07004189// TODO: Make this panic again once it stops happening routinely.
Michael Pratta4e31d42016-03-12 14:07:40 -08004190func SSARegNum(v *ssa.Value) int16 {
Josh Bleecher Snydere1395492015-08-05 16:06:39 -07004191 reg := v.Block.Func.RegAlloc[v.ID]
4192 if reg == nil {
4193 v.Unimplementedf("nil regnum for value: %s\n%s\n", v.LongString(), v.Block.Func)
4194 return 0
4195 }
Michael Pratta4e31d42016-03-12 14:07:40 -08004196 return Thearch.SSARegToReg[reg.(*ssa.Register).Num]
Keith Randall083a6462015-05-12 11:06:44 -07004197}
4198
Michael Pratta4e31d42016-03-12 14:07:40 -08004199// AutoVar returns a *Node and int64 representing the auto variable and offset within it
Keith Randall02f4d0a2015-11-02 08:10:26 -08004200// where v should be spilled.
Michael Pratta4e31d42016-03-12 14:07:40 -08004201func AutoVar(v *ssa.Value) (*Node, int64) {
Keith Randall02f4d0a2015-11-02 08:10:26 -08004202 loc := v.Block.Func.RegAlloc[v.ID].(ssa.LocalSlot)
Keith Randall9094e3a2016-01-04 13:34:54 -08004203 if v.Type.Size() > loc.Type.Size() {
4204 v.Fatalf("spill/restore type %s doesn't fit in slot type %s", v.Type, loc.Type)
4205 }
Keith Randall02f4d0a2015-11-02 08:10:26 -08004206 return loc.N.(*Node), loc.Off
Keith Randall083a6462015-05-12 11:06:44 -07004207}
Keith Randallf7f604e2015-05-27 14:52:22 -07004208
Keith Randalla734bbc2016-01-11 21:05:33 -08004209// fieldIdx finds the index of the field referred to by the ODOT node n.
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07004210func fieldIdx(n *Node) int {
Keith Randalla734bbc2016-01-11 21:05:33 -08004211 t := n.Left.Type
Ian Lance Taylor5f525ca2016-03-18 16:52:30 -07004212 f := n.Sym
Matthew Dempsky3efefd92016-03-30 14:56:08 -07004213 if !t.IsStruct() {
Keith Randalla734bbc2016-01-11 21:05:33 -08004214 panic("ODOT's LHS is not a struct")
4215 }
4216
Matthew Dempsky1b9f1682016-03-14 12:45:18 -07004217 var i int
Matthew Dempskyf6bca3f2016-03-17 01:32:18 -07004218 for _, t1 := range t.Fields().Slice() {
Ian Lance Taylor5f525ca2016-03-18 16:52:30 -07004219 if t1.Sym != f {
Keith Randalla734bbc2016-01-11 21:05:33 -08004220 i++
4221 continue
4222 }
Matthew Dempsky62dddd42016-03-28 09:40:53 -07004223 if t1.Offset != n.Xoffset {
Keith Randalla734bbc2016-01-11 21:05:33 -08004224 panic("field offset doesn't match")
4225 }
4226 return i
4227 }
4228 panic(fmt.Sprintf("can't find field in expr %s\n", n))
4229
Eric Engestrom7a8caf72016-04-03 12:43:27 +01004230 // TODO: keep the result of this function somewhere in the ODOT Node
Keith Randalla734bbc2016-01-11 21:05:33 -08004231 // so we don't have to recompute it each time we need it.
4232}
4233
Keith Randallf7f604e2015-05-27 14:52:22 -07004234// ssaExport exports a bunch of compiler services for the ssa backend.
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07004235type ssaExport struct {
4236 log bool
4237 unimplemented bool
Josh Bleecher Snyderd2982092015-07-22 13:13:53 -07004238 mustImplement bool
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07004239}
Keith Randallf7f604e2015-05-27 14:52:22 -07004240
Josh Bleecher Snyder85e03292015-07-30 11:03:05 -07004241func (s *ssaExport) TypeBool() ssa.Type { return Types[TBOOL] }
4242func (s *ssaExport) TypeInt8() ssa.Type { return Types[TINT8] }
4243func (s *ssaExport) TypeInt16() ssa.Type { return Types[TINT16] }
4244func (s *ssaExport) TypeInt32() ssa.Type { return Types[TINT32] }
4245func (s *ssaExport) TypeInt64() ssa.Type { return Types[TINT64] }
4246func (s *ssaExport) TypeUInt8() ssa.Type { return Types[TUINT8] }
4247func (s *ssaExport) TypeUInt16() ssa.Type { return Types[TUINT16] }
4248func (s *ssaExport) TypeUInt32() ssa.Type { return Types[TUINT32] }
4249func (s *ssaExport) TypeUInt64() ssa.Type { return Types[TUINT64] }
David Chase52578582015-08-28 14:24:10 -04004250func (s *ssaExport) TypeFloat32() ssa.Type { return Types[TFLOAT32] }
4251func (s *ssaExport) TypeFloat64() ssa.Type { return Types[TFLOAT64] }
Josh Bleecher Snyder85e03292015-07-30 11:03:05 -07004252func (s *ssaExport) TypeInt() ssa.Type { return Types[TINT] }
4253func (s *ssaExport) TypeUintptr() ssa.Type { return Types[TUINTPTR] }
4254func (s *ssaExport) TypeString() ssa.Type { return Types[TSTRING] }
4255func (s *ssaExport) TypeBytePtr() ssa.Type { return Ptrto(Types[TUINT8]) }
4256
Josh Bleecher Snyder8d31df18a2015-07-24 11:28:12 -07004257// StringData returns a symbol (a *Sym wrapped in an interface) which
4258// is the data component of a global string constant containing s.
4259func (*ssaExport) StringData(s string) interface{} {
Keith Randall8c46aa52015-06-19 21:02:28 -07004260 // TODO: is idealstring correct? It might not matter...
Josh Bleecher Snyder8d31df18a2015-07-24 11:28:12 -07004261 _, data := stringsym(s)
4262 return &ssa.ExternSymbol{Typ: idealstring, Sym: data}
Keith Randallf7f604e2015-05-27 14:52:22 -07004263}
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07004264
Keith Randallc24681a2015-10-22 14:22:38 -07004265func (e *ssaExport) Auto(t ssa.Type) ssa.GCNode {
Keith Randalld2107fc2015-08-24 02:16:19 -07004266 n := temp(t.(*Type)) // Note: adds new auto to Curfn.Func.Dcl list
4267 e.mustImplement = true // This modifies the input to SSA, so we want to make sure we succeed from here!
4268 return n
4269}
4270
Keith Randall4a7aba72016-03-28 11:25:17 -07004271func (e *ssaExport) SplitString(name ssa.LocalSlot) (ssa.LocalSlot, ssa.LocalSlot) {
4272 n := name.N.(*Node)
4273 ptrType := Ptrto(Types[TUINT8])
4274 lenType := Types[TINT]
4275 if n.Class == PAUTO && !n.Addrtaken {
4276 // Split this string up into two separate variables.
4277 p := e.namedAuto(n.Sym.Name+".ptr", ptrType)
4278 l := e.namedAuto(n.Sym.Name+".len", lenType)
Keith Randallcd956572016-04-29 09:02:27 -07004279 return ssa.LocalSlot{N: p, Type: ptrType, Off: 0}, ssa.LocalSlot{N: l, Type: lenType, Off: 0}
Keith Randall4a7aba72016-03-28 11:25:17 -07004280 }
4281 // Return the two parts of the larger variable.
Keith Randallcd956572016-04-29 09:02:27 -07004282 return ssa.LocalSlot{N: n, Type: ptrType, Off: name.Off}, ssa.LocalSlot{N: n, Type: lenType, Off: name.Off + int64(Widthptr)}
Keith Randall4a7aba72016-03-28 11:25:17 -07004283}
4284
4285func (e *ssaExport) SplitInterface(name ssa.LocalSlot) (ssa.LocalSlot, ssa.LocalSlot) {
4286 n := name.N.(*Node)
4287 t := Ptrto(Types[TUINT8])
4288 if n.Class == PAUTO && !n.Addrtaken {
4289 // Split this interface up into two separate variables.
4290 f := ".itab"
Matthew Dempsky00e5a682016-04-01 13:36:24 -07004291 if n.Type.IsEmptyInterface() {
Keith Randall4a7aba72016-03-28 11:25:17 -07004292 f = ".type"
4293 }
4294 c := e.namedAuto(n.Sym.Name+f, t)
4295 d := e.namedAuto(n.Sym.Name+".data", t)
Keith Randallcd956572016-04-29 09:02:27 -07004296 return ssa.LocalSlot{N: c, Type: t, Off: 0}, ssa.LocalSlot{N: d, Type: t, Off: 0}
Keith Randall4a7aba72016-03-28 11:25:17 -07004297 }
4298 // Return the two parts of the larger variable.
Keith Randallcd956572016-04-29 09:02:27 -07004299 return ssa.LocalSlot{N: n, Type: t, Off: name.Off}, ssa.LocalSlot{N: n, Type: t, Off: name.Off + int64(Widthptr)}
Keith Randall4a7aba72016-03-28 11:25:17 -07004300}
4301
4302func (e *ssaExport) SplitSlice(name ssa.LocalSlot) (ssa.LocalSlot, ssa.LocalSlot, ssa.LocalSlot) {
4303 n := name.N.(*Node)
Keith Randallb04e1452016-03-31 21:24:10 -07004304 ptrType := Ptrto(name.Type.ElemType().(*Type))
Keith Randall4a7aba72016-03-28 11:25:17 -07004305 lenType := Types[TINT]
4306 if n.Class == PAUTO && !n.Addrtaken {
4307 // Split this slice up into three separate variables.
4308 p := e.namedAuto(n.Sym.Name+".ptr", ptrType)
4309 l := e.namedAuto(n.Sym.Name+".len", lenType)
4310 c := e.namedAuto(n.Sym.Name+".cap", lenType)
Keith Randallcd956572016-04-29 09:02:27 -07004311 return ssa.LocalSlot{N: p, Type: ptrType, Off: 0}, ssa.LocalSlot{N: l, Type: lenType, Off: 0}, ssa.LocalSlot{N: c, Type: lenType, Off: 0}
Keith Randall4a7aba72016-03-28 11:25:17 -07004312 }
4313 // Return the three parts of the larger variable.
Keith Randallcd956572016-04-29 09:02:27 -07004314 return ssa.LocalSlot{N: n, Type: ptrType, Off: name.Off},
4315 ssa.LocalSlot{N: n, Type: lenType, Off: name.Off + int64(Widthptr)},
4316 ssa.LocalSlot{N: n, Type: lenType, Off: name.Off + int64(2*Widthptr)}
Keith Randall4a7aba72016-03-28 11:25:17 -07004317}
4318
4319func (e *ssaExport) SplitComplex(name ssa.LocalSlot) (ssa.LocalSlot, ssa.LocalSlot) {
4320 n := name.N.(*Node)
4321 s := name.Type.Size() / 2
4322 var t *Type
4323 if s == 8 {
4324 t = Types[TFLOAT64]
4325 } else {
4326 t = Types[TFLOAT32]
4327 }
4328 if n.Class == PAUTO && !n.Addrtaken {
4329 // Split this complex up into two separate variables.
4330 c := e.namedAuto(n.Sym.Name+".real", t)
4331 d := e.namedAuto(n.Sym.Name+".imag", t)
Keith Randallcd956572016-04-29 09:02:27 -07004332 return ssa.LocalSlot{N: c, Type: t, Off: 0}, ssa.LocalSlot{N: d, Type: t, Off: 0}
Keith Randall4a7aba72016-03-28 11:25:17 -07004333 }
4334 // Return the two parts of the larger variable.
Keith Randallcd956572016-04-29 09:02:27 -07004335 return ssa.LocalSlot{N: n, Type: t, Off: name.Off}, ssa.LocalSlot{N: n, Type: t, Off: name.Off + s}
Keith Randall4a7aba72016-03-28 11:25:17 -07004336}
4337
Keith Randallb04e1452016-03-31 21:24:10 -07004338func (e *ssaExport) SplitStruct(name ssa.LocalSlot, i int) ssa.LocalSlot {
4339 n := name.N.(*Node)
4340 st := name.Type
4341 ft := st.FieldType(i)
4342 if n.Class == PAUTO && !n.Addrtaken {
4343 // Note: the _ field may appear several times. But
4344 // have no fear, identically-named but distinct Autos are
4345 // ok, albeit maybe confusing for a debugger.
4346 x := e.namedAuto(n.Sym.Name+"."+st.FieldName(i), ft)
Keith Randallcd956572016-04-29 09:02:27 -07004347 return ssa.LocalSlot{N: x, Type: ft, Off: 0}
Keith Randallb04e1452016-03-31 21:24:10 -07004348 }
Keith Randallcd956572016-04-29 09:02:27 -07004349 return ssa.LocalSlot{N: n, Type: ft, Off: name.Off + st.FieldOff(i)}
Keith Randallb04e1452016-03-31 21:24:10 -07004350}
4351
Keith Randall4a7aba72016-03-28 11:25:17 -07004352// namedAuto returns a new AUTO variable with the given name and type.
4353func (e *ssaExport) namedAuto(name string, typ ssa.Type) ssa.GCNode {
4354 t := typ.(*Type)
4355 s := Lookup(name)
4356 n := Nod(ONAME, nil, nil)
4357 s.Def = n
4358 s.Def.Used = true
4359 n.Sym = s
4360 n.Type = t
4361 n.Class = PAUTO
4362 n.Addable = true
4363 n.Ullman = 1
4364 n.Esc = EscNever
4365 n.Xoffset = 0
4366 n.Name.Curfn = Curfn
4367 Curfn.Func.Dcl = append(Curfn.Func.Dcl, n)
4368
4369 dowidth(t)
4370 e.mustImplement = true
4371
4372 return n
4373}
4374
Keith Randall7d612462015-10-22 13:07:38 -07004375func (e *ssaExport) CanSSA(t ssa.Type) bool {
Keith Randall37590bd2015-09-18 22:58:10 -07004376 return canSSAType(t.(*Type))
4377}
4378
Keith Randallb5c5efd2016-01-14 16:02:23 -08004379func (e *ssaExport) Line(line int32) string {
Robert Griesemer2faf5bc2016-03-02 11:30:29 -08004380 return linestr(line)
Keith Randallb5c5efd2016-01-14 16:02:23 -08004381}
4382
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07004383// Log logs a message from the compiler.
Josh Bleecher Snyder37ddc272015-06-24 14:03:39 -07004384func (e *ssaExport) Logf(msg string, args ...interface{}) {
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07004385 // If e was marked as unimplemented, anything could happen. Ignore.
4386 if e.log && !e.unimplemented {
4387 fmt.Printf(msg, args...)
4388 }
4389}
4390
David Chase88b230e2016-01-29 14:44:15 -05004391func (e *ssaExport) Log() bool {
4392 return e.log
4393}
4394
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07004395// Fatal reports a compiler error and exits.
Keith Randallda8af472016-01-13 11:14:57 -08004396func (e *ssaExport) Fatalf(line int32, msg string, args ...interface{}) {
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07004397 // If e was marked as unimplemented, anything could happen. Ignore.
4398 if !e.unimplemented {
Keith Randallda8af472016-01-13 11:14:57 -08004399 lineno = line
Keith Randall0ec72b62015-09-08 15:42:53 -07004400 Fatalf(msg, args...)
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07004401 }
4402}
4403
4404// Unimplemented reports that the function cannot be compiled.
4405// It will be removed once SSA work is complete.
Keith Randallda8af472016-01-13 11:14:57 -08004406func (e *ssaExport) Unimplementedf(line int32, msg string, args ...interface{}) {
Josh Bleecher Snyderd2982092015-07-22 13:13:53 -07004407 if e.mustImplement {
Keith Randallda8af472016-01-13 11:14:57 -08004408 lineno = line
Keith Randall0ec72b62015-09-08 15:42:53 -07004409 Fatalf(msg, args...)
Josh Bleecher Snyderd2982092015-07-22 13:13:53 -07004410 }
Josh Bleecher Snyder8c6abfe2015-06-12 11:01:13 -07004411 const alwaysLog = false // enable to calculate top unimplemented features
4412 if !e.unimplemented && (e.log || alwaysLog) {
4413 // first implementation failure, print explanation
4414 fmt.Printf("SSA unimplemented: "+msg+"\n", args...)
4415 }
4416 e.unimplemented = true
4417}
Keith Randallc24681a2015-10-22 14:22:38 -07004418
David Chase729abfa2015-10-26 17:34:06 -04004419// Warnl reports a "warning", which is usually flag-triggered
4420// logging output for the benefit of tests.
Todd Neal98b88de2016-03-13 23:04:31 -05004421func (e *ssaExport) Warnl(line int32, fmt_ string, args ...interface{}) {
4422 Warnl(line, fmt_, args...)
David Chase729abfa2015-10-26 17:34:06 -04004423}
4424
4425func (e *ssaExport) Debug_checknil() bool {
4426 return Debug_checknil != 0
4427}
4428
Keith Randallc24681a2015-10-22 14:22:38 -07004429func (n *Node) Typ() ssa.Type {
4430 return n.Type
4431}