1 // Copyright 2009 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 5 // “Abstract” syntax representation. 6 7 package ir 8 9 import ( 10 "fmt" 11 "go/constant" 12 "sort" 13 14 "cmd/compile/internal/base" 15 "cmd/compile/internal/types" 16 "cmd/internal/src" 17 ) 18 19 // A Node is the abstract interface to an IR node. 20 type Node interface { 21 // Formatting 22 Format(s fmt.State, verb rune) 23 24 // Source position. 25 Pos() src.XPos 26 SetPos(x src.XPos) 27 28 // For making copies. For Copy and SepCopy. 29 copy() Node 30 31 doChildren(func(Node) bool) bool 32 editChildren(func(Node) Node) 33 editChildrenWithHidden(func(Node) Node) 34 35 // Abstract graph structure, for generic traversals. 36 Op() Op 37 Init() Nodes 38 39 // Fields specific to certain Ops only. 40 Type() *types.Type 41 SetType(t *types.Type) 42 Name() *Name 43 Sym() *types.Sym 44 Val() constant.Value 45 SetVal(v constant.Value) 46 47 // Storage for analysis passes. 48 Esc() uint16 49 SetEsc(x uint16) 50 51 // Typecheck values: 52 // 0 means the node is not typechecked 53 // 1 means the node is completely typechecked 54 // 2 means typechecking of the node is in progress 55 Typecheck() uint8 56 SetTypecheck(x uint8) 57 NonNil() bool 58 MarkNonNil() 59 } 60 61 // Line returns n's position as a string. If n has been inlined, 62 // it uses the outermost position where n has been inlined. 63 func Line(n Node) string { 64 return base.FmtPos(n.Pos()) 65 } 66 67 func IsSynthetic(n Node) bool { 68 name := n.Sym().Name 69 return name[0] == '.' || name[0] == '~' 70 } 71 72 // IsAutoTmp indicates if n was created by the compiler as a temporary, 73 // based on the setting of the .AutoTemp flag in n's Name. 74 func IsAutoTmp(n Node) bool { 75 if n == nil || n.Op() != ONAME { 76 return false 77 } 78 return n.Name().AutoTemp() 79 } 80 81 // MayBeShared reports whether n may occur in multiple places in the AST. 82 // Extra care must be taken when mutating such a node. 83 func MayBeShared(n Node) bool { 84 switch n.Op() { 85 case ONAME, OLITERAL, ONIL, OTYPE: 86 return true 87 } 88 return false 89 } 90 91 type InitNode interface { 92 Node 93 PtrInit() *Nodes 94 SetInit(x Nodes) 95 } 96 97 func TakeInit(n Node) Nodes { 98 init := n.Init() 99 if len(init) != 0 { 100 n.(InitNode).SetInit(nil) 101 } 102 return init 103 } 104 105 //go:generate stringer -type=Op -trimprefix=O node.go 106 107 type Op uint8 108 109 // Node ops. 110 const ( 111 OXXX Op = iota 112 113 // names 114 ONAME // var or func name 115 // Unnamed arg or return value: f(int, string) (int, error) { etc } 116 // Also used for a qualified package identifier that hasn't been resolved yet. 117 ONONAME 118 OTYPE // type name 119 OLITERAL // literal 120 ONIL // nil 121 122 // expressions 123 OADD // X + Y 124 OSUB // X - Y 125 OOR // X | Y 126 OXOR // X ^ Y 127 OADDSTR // +{List} (string addition, list elements are strings) 128 OADDR // &X 129 OANDAND // X && Y 130 OAPPEND // append(Args); after walk, X may contain elem type descriptor 131 OBYTES2STR // Type(X) (Type is string, X is a []byte) 132 OBYTES2STRTMP // Type(X) (Type is string, X is a []byte, ephemeral) 133 ORUNES2STR // Type(X) (Type is string, X is a []rune) 134 OSTR2BYTES // Type(X) (Type is []byte, X is a string) 135 OSTR2BYTESTMP // Type(X) (Type is []byte, X is a string, ephemeral) 136 OSTR2RUNES // Type(X) (Type is []rune, X is a string) 137 OSLICE2ARR // Type(X) (Type is [N]T, X is a []T) 138 OSLICE2ARRPTR // Type(X) (Type is *[N]T, X is a []T) 139 // X = Y or (if Def=true) X := Y 140 // If Def, then Init includes a DCL node for X. 141 OAS 142 // Lhs = Rhs (x, y, z = a, b, c) or (if Def=true) Lhs := Rhs 143 // If Def, then Init includes DCL nodes for Lhs 144 OAS2 145 OAS2DOTTYPE // Lhs = Rhs (x, ok = I.(int)) 146 OAS2FUNC // Lhs = Rhs (x, y = f()) 147 OAS2MAPR // Lhs = Rhs (x, ok = m["foo"]) 148 OAS2RECV // Lhs = Rhs (x, ok = <-c) 149 OASOP // X AsOp= Y (x += y) 150 OCALL // X(Args) (function call, method call or type conversion) 151 152 // OCALLFUNC, OCALLMETH, and OCALLINTER have the same structure. 153 // Prior to walk, they are: X(Args), where Args is all regular arguments. 154 // After walk, if any argument whose evaluation might requires temporary variable, 155 // that temporary variable will be pushed to Init, Args will contains an updated 156 // set of arguments. 157 OCALLFUNC // X(Args) (function call f(args)) 158 OCALLMETH // X(Args) (direct method call x.Method(args)) 159 OCALLINTER // X(Args) (interface method call x.Method(args)) 160 OCAP // cap(X) 161 OCLEAR // clear(X) 162 OCLOSE // close(X) 163 OCLOSURE // func Type { Func.Closure.Body } (func literal) 164 OCOMPLIT // Type{List} (composite literal, not yet lowered to specific form) 165 OMAPLIT // Type{List} (composite literal, Type is map) 166 OSTRUCTLIT // Type{List} (composite literal, Type is struct) 167 OARRAYLIT // Type{List} (composite literal, Type is array) 168 OSLICELIT // Type{List} (composite literal, Type is slice), Len is slice length. 169 OPTRLIT // &X (X is composite literal) 170 OCONV // Type(X) (type conversion) 171 OCONVIFACE // Type(X) (type conversion, to interface) 172 OCONVNOP // Type(X) (type conversion, no effect) 173 OCOPY // copy(X, Y) 174 ODCL // var X (declares X of type X.Type) 175 176 // Used during parsing but don't last. 177 ODCLFUNC // func f() or func (r) f() 178 179 ODELETE // delete(Args) 180 ODOT // X.Sel (X is of struct type) 181 ODOTPTR // X.Sel (X is of pointer to struct type) 182 ODOTMETH // X.Sel (X is non-interface, Sel is method name) 183 ODOTINTER // X.Sel (X is interface, Sel is method name) 184 OXDOT // X.Sel (before rewrite to one of the preceding) 185 ODOTTYPE // X.Ntype or X.Type (.Ntype during parsing, .Type once resolved); after walk, Itab contains address of interface type descriptor and Itab.X contains address of concrete type descriptor 186 ODOTTYPE2 // X.Ntype or X.Type (.Ntype during parsing, .Type once resolved; on rhs of OAS2DOTTYPE); after walk, Itab contains address of interface type descriptor 187 OEQ // X == Y 188 ONE // X != Y 189 OLT // X < Y 190 OLE // X <= Y 191 OGE // X >= Y 192 OGT // X > Y 193 ODEREF // *X 194 OINDEX // X[Index] (index of array or slice) 195 OINDEXMAP // X[Index] (index of map) 196 OKEY // Key:Value (key:value in struct/array/map literal) 197 OSTRUCTKEY // Field:Value (key:value in struct literal, after type checking) 198 OLEN // len(X) 199 OMAKE // make(Args) (before type checking converts to one of the following) 200 OMAKECHAN // make(Type[, Len]) (type is chan) 201 OMAKEMAP // make(Type[, Len]) (type is map) 202 OMAKESLICE // make(Type[, Len[, Cap]]) (type is slice) 203 OMAKESLICECOPY // makeslicecopy(Type, Len, Cap) (type is slice; Len is length and Cap is the copied from slice) 204 // OMAKESLICECOPY is created by the order pass and corresponds to: 205 // s = make(Type, Len); copy(s, Cap) 206 // 207 // Bounded can be set on the node when Len == len(Cap) is known at compile time. 208 // 209 // This node is created so the walk pass can optimize this pattern which would 210 // otherwise be hard to detect after the order pass. 211 OMUL // X * Y 212 ODIV // X / Y 213 OMOD // X % Y 214 OLSH // X << Y 215 ORSH // X >> Y 216 OAND // X & Y 217 OANDNOT // X &^ Y 218 ONEW // new(X); corresponds to calls to new in source code 219 ONOT // !X 220 OBITNOT // ^X 221 OPLUS // +X 222 ONEG // -X 223 OOROR // X || Y 224 OPANIC // panic(X) 225 OPRINT // print(List) 226 OPRINTLN // println(List) 227 OPAREN // (X) 228 OSEND // Chan <- Value 229 OSLICE // X[Low : High] (X is untypechecked or slice) 230 OSLICEARR // X[Low : High] (X is pointer to array) 231 OSLICESTR // X[Low : High] (X is string) 232 OSLICE3 // X[Low : High : Max] (X is untypedchecked or slice) 233 OSLICE3ARR // X[Low : High : Max] (X is pointer to array) 234 OSLICEHEADER // sliceheader{Ptr, Len, Cap} (Ptr is unsafe.Pointer, Len is length, Cap is capacity) 235 OSTRINGHEADER // stringheader{Ptr, Len} (Ptr is unsafe.Pointer, Len is length) 236 ORECOVER // recover() 237 ORECOVERFP // recover(Args) w/ explicit FP argument 238 ORECV // <-X 239 ORUNESTR // Type(X) (Type is string, X is rune) 240 OSELRECV2 // like OAS2: Lhs = Rhs where len(Lhs)=2, len(Rhs)=1, Rhs[0].Op = ORECV (appears as .Var of OCASE) 241 OMIN // min(List) 242 OMAX // max(List) 243 OREAL // real(X) 244 OIMAG // imag(X) 245 OCOMPLEX // complex(X, Y) 246 OUNSAFEADD // unsafe.Add(X, Y) 247 OUNSAFESLICE // unsafe.Slice(X, Y) 248 OUNSAFESLICEDATA // unsafe.SliceData(X) 249 OUNSAFESTRING // unsafe.String(X, Y) 250 OUNSAFESTRINGDATA // unsafe.StringData(X) 251 OMETHEXPR // X(Args) (method expression T.Method(args), first argument is the method receiver) 252 OMETHVALUE // X.Sel (method expression t.Method, not called) 253 254 // statements 255 OBLOCK // { List } (block of code) 256 OBREAK // break [Label] 257 // OCASE: case List: Body (List==nil means default) 258 // For OTYPESW, List is a OTYPE node for the specified type (or OLITERAL 259 // for nil) or an ODYNAMICTYPE indicating a runtime type for generics. 260 // If a type-switch variable is specified, Var is an 261 // ONAME for the version of the type-switch variable with the specified 262 // type. 263 OCASE 264 OCONTINUE // continue [Label] 265 ODEFER // defer Call 266 OFALL // fallthrough 267 OFOR // for Init; Cond; Post { Body } 268 OGOTO // goto Label 269 OIF // if Init; Cond { Then } else { Else } 270 OLABEL // Label: 271 OGO // go Call 272 ORANGE // for Key, Value = range X { Body } 273 ORETURN // return Results 274 OSELECT // select { Cases } 275 OSWITCH // switch Init; Expr { Cases } 276 // OTYPESW: X := Y.(type) (appears as .Tag of OSWITCH) 277 // X is nil if there is no type-switch variable 278 OTYPESW 279 280 // misc 281 // intermediate representation of an inlined call. Uses Init (assignments 282 // for the captured variables, parameters, retvars, & INLMARK op), 283 // Body (body of the inlined function), and ReturnVars (list of 284 // return values) 285 OINLCALL // intermediary representation of an inlined call. 286 OMAKEFACE // construct an interface value from rtype/itab and data pointers 287 OITAB // rtype/itab pointer of an interface value 288 OIDATA // data pointer of an interface value 289 OSPTR // base pointer of a slice or string. Bounded==1 means known non-nil. 290 OCFUNC // reference to c function pointer (not go func value) 291 OCHECKNIL // emit code to ensure pointer/interface not nil 292 ORESULT // result of a function call; Xoffset is stack offset 293 OINLMARK // start of an inlined body, with file/line of caller. Xoffset is an index into the inline tree. 294 OLINKSYMOFFSET // offset within a name 295 OJUMPTABLE // A jump table structure for implementing dense expression switches 296 OINTERFACESWITCH // A type switch with interface cases 297 298 // opcodes for generics 299 ODYNAMICDOTTYPE // x = i.(T) where T is a type parameter (or derived from a type parameter) 300 ODYNAMICDOTTYPE2 // x, ok = i.(T) where T is a type parameter (or derived from a type parameter) 301 ODYNAMICTYPE // a type node for type switches (represents a dynamic target type for a type switch) 302 303 // arch-specific opcodes 304 OTAILCALL // tail call to another function 305 OGETG // runtime.getg() (read g pointer) 306 OGETCALLERPC // runtime.getcallerpc() (continuation PC in caller frame) 307 OGETCALLERSP // runtime.getcallersp() (stack pointer in caller frame) 308 309 OEND 310 ) 311 312 // IsCmp reports whether op is a comparison operation (==, !=, <, <=, 313 // >, or >=). 314 func (op Op) IsCmp() bool { 315 switch op { 316 case OEQ, ONE, OLT, OLE, OGT, OGE: 317 return true 318 } 319 return false 320 } 321 322 // Nodes is a slice of Node. 323 type Nodes []Node 324 325 // ToNodes returns s as a slice of Nodes. 326 func ToNodes[T Node](s []T) Nodes { 327 res := make(Nodes, len(s)) 328 for i, n := range s { 329 res[i] = n 330 } 331 return res 332 } 333 334 // Append appends entries to Nodes. 335 func (n *Nodes) Append(a ...Node) { 336 if len(a) == 0 { 337 return 338 } 339 *n = append(*n, a...) 340 } 341 342 // Prepend prepends entries to Nodes. 343 // If a slice is passed in, this will take ownership of it. 344 func (n *Nodes) Prepend(a ...Node) { 345 if len(a) == 0 { 346 return 347 } 348 *n = append(a, *n...) 349 } 350 351 // Take clears n, returning its former contents. 352 func (n *Nodes) Take() []Node { 353 ret := *n 354 *n = nil 355 return ret 356 } 357 358 // Copy returns a copy of the content of the slice. 359 func (n Nodes) Copy() Nodes { 360 if n == nil { 361 return nil 362 } 363 c := make(Nodes, len(n)) 364 copy(c, n) 365 return c 366 } 367 368 // NameQueue is a FIFO queue of *Name. The zero value of NameQueue is 369 // a ready-to-use empty queue. 370 type NameQueue struct { 371 ring []*Name 372 head, tail int 373 } 374 375 // Empty reports whether q contains no Names. 376 func (q *NameQueue) Empty() bool { 377 return q.head == q.tail 378 } 379 380 // PushRight appends n to the right of the queue. 381 func (q *NameQueue) PushRight(n *Name) { 382 if len(q.ring) == 0 { 383 q.ring = make([]*Name, 16) 384 } else if q.head+len(q.ring) == q.tail { 385 // Grow the ring. 386 nring := make([]*Name, len(q.ring)*2) 387 // Copy the old elements. 388 part := q.ring[q.head%len(q.ring):] 389 if q.tail-q.head <= len(part) { 390 part = part[:q.tail-q.head] 391 copy(nring, part) 392 } else { 393 pos := copy(nring, part) 394 copy(nring[pos:], q.ring[:q.tail%len(q.ring)]) 395 } 396 q.ring, q.head, q.tail = nring, 0, q.tail-q.head 397 } 398 399 q.ring[q.tail%len(q.ring)] = n 400 q.tail++ 401 } 402 403 // PopLeft pops a Name from the left of the queue. It panics if q is 404 // empty. 405 func (q *NameQueue) PopLeft() *Name { 406 if q.Empty() { 407 panic("dequeue empty") 408 } 409 n := q.ring[q.head%len(q.ring)] 410 q.head++ 411 return n 412 } 413 414 // NameSet is a set of Names. 415 type NameSet map[*Name]struct{} 416 417 // Has reports whether s contains n. 418 func (s NameSet) Has(n *Name) bool { 419 _, isPresent := s[n] 420 return isPresent 421 } 422 423 // Add adds n to s. 424 func (s *NameSet) Add(n *Name) { 425 if *s == nil { 426 *s = make(map[*Name]struct{}) 427 } 428 (*s)[n] = struct{}{} 429 } 430 431 // Sorted returns s sorted according to less. 432 func (s NameSet) Sorted(less func(*Name, *Name) bool) []*Name { 433 var res []*Name 434 for n := range s { 435 res = append(res, n) 436 } 437 sort.Slice(res, func(i, j int) bool { return less(res[i], res[j]) }) 438 return res 439 } 440 441 type PragmaFlag uint16 442 443 const ( 444 // Func pragmas. 445 Nointerface PragmaFlag = 1 << iota 446 Noescape // func parameters don't escape 447 Norace // func must not have race detector annotations 448 Nosplit // func should not execute on separate stack 449 Noinline // func should not be inlined 450 NoCheckPtr // func should not be instrumented by checkptr 451 CgoUnsafeArgs // treat a pointer to one arg as a pointer to them all 452 UintptrKeepAlive // pointers converted to uintptr must be kept alive 453 UintptrEscapes // pointers converted to uintptr escape 454 455 // Runtime-only func pragmas. 456 // See ../../../../runtime/HACKING.md for detailed descriptions. 457 Systemstack // func must run on system stack 458 Nowritebarrier // emit compiler error instead of write barrier 459 Nowritebarrierrec // error on write barrier in this or recursive callees 460 Yeswritebarrierrec // cancels Nowritebarrierrec in this function and callees 461 462 // Go command pragmas 463 GoBuildPragma 464 465 RegisterParams // TODO(register args) remove after register abi is working 466 467 ) 468 469 var BlankNode *Name 470 471 func IsConst(n Node, ct constant.Kind) bool { 472 return ConstType(n) == ct 473 } 474 475 // IsNil reports whether n represents the universal untyped zero value "nil". 476 func IsNil(n Node) bool { 477 return n != nil && n.Op() == ONIL 478 } 479 480 func IsBlank(n Node) bool { 481 if n == nil { 482 return false 483 } 484 return n.Sym().IsBlank() 485 } 486 487 // IsMethod reports whether n is a method. 488 // n must be a function or a method. 489 func IsMethod(n Node) bool { 490 return n.Type().Recv() != nil 491 } 492 493 // HasUniquePos reports whether n has a unique position that can be 494 // used for reporting error messages. 495 // 496 // It's primarily used to distinguish references to named objects, 497 // whose Pos will point back to their declaration position rather than 498 // their usage position. 499 func HasUniquePos(n Node) bool { 500 switch n.Op() { 501 case ONAME: 502 return false 503 case OLITERAL, ONIL, OTYPE: 504 if n.Sym() != nil { 505 return false 506 } 507 } 508 509 if !n.Pos().IsKnown() { 510 if base.Flag.K != 0 { 511 base.Warn("setlineno: unknown position (line 0)") 512 } 513 return false 514 } 515 516 return true 517 } 518 519 func SetPos(n Node) src.XPos { 520 lno := base.Pos 521 if n != nil && HasUniquePos(n) { 522 base.Pos = n.Pos() 523 } 524 return lno 525 } 526 527 // The result of InitExpr MUST be assigned back to n, e.g. 528 // 529 // n.X = InitExpr(init, n.X) 530 func InitExpr(init []Node, expr Node) Node { 531 if len(init) == 0 { 532 return expr 533 } 534 535 n, ok := expr.(InitNode) 536 if !ok || MayBeShared(n) { 537 // Introduce OCONVNOP to hold init list. 538 n = NewConvExpr(base.Pos, OCONVNOP, nil, expr) 539 n.SetType(expr.Type()) 540 n.SetTypecheck(1) 541 } 542 543 n.PtrInit().Prepend(init...) 544 return n 545 } 546 547 // what's the outer value that a write to n affects? 548 // outer value means containing struct or array. 549 func OuterValue(n Node) Node { 550 for { 551 switch nn := n; nn.Op() { 552 case OXDOT: 553 base.FatalfAt(n.Pos(), "OXDOT in OuterValue: %v", n) 554 case ODOT: 555 nn := nn.(*SelectorExpr) 556 n = nn.X 557 continue 558 case OPAREN: 559 nn := nn.(*ParenExpr) 560 n = nn.X 561 continue 562 case OCONVNOP: 563 nn := nn.(*ConvExpr) 564 n = nn.X 565 continue 566 case OINDEX: 567 nn := nn.(*IndexExpr) 568 if nn.X.Type() == nil { 569 base.Fatalf("OuterValue needs type for %v", nn.X) 570 } 571 if nn.X.Type().IsArray() { 572 n = nn.X 573 continue 574 } 575 } 576 577 return n 578 } 579 } 580 581 const ( 582 EscUnknown = iota 583 EscNone // Does not escape to heap, result, or parameters. 584 EscHeap // Reachable from the heap 585 EscNever // By construction will not escape. 586 ) 587