blob: 3ed7642a8de98fd71c8756df6a0a257b85a38650 [file] [log] [blame]
Alan Donovan713699d2013-08-27 18:49:13 -04001// Copyright 2013 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
Rob Pike83f21b92013-05-17 13:25:48 -07005package ssa
6
7// lvalues are the union of addressable expressions and map-index
8// expressions.
9
10import (
Alan Donovan55d678e2013-07-15 13:56:46 -040011 "go/ast"
Alan Donovan6c7ce1c2013-05-30 09:59:17 -040012 "go/token"
Alan Donovan55d678e2013-07-15 13:56:46 -040013
14 "code.google.com/p/go.tools/go/types"
Rob Pike83f21b92013-05-17 13:25:48 -070015)
16
17// An lvalue represents an assignable location that may appear on the
18// left-hand side of an assignment. This is a generalization of a
19// pointer to permit updates to elements of maps.
20//
21type lvalue interface {
22 store(fn *Function, v Value) // stores v into the location
23 load(fn *Function) Value // loads the contents of the location
Alan Donovan55d678e2013-07-15 13:56:46 -040024 address(fn *Function) Value // address of the location
Rob Pike83f21b92013-05-17 13:25:48 -070025 typ() types.Type // returns the type of the location
26}
27
28// An address is an lvalue represented by a true pointer.
29type address struct {
Alan Donovan55d678e2013-07-15 13:56:46 -040030 addr Value
Alan Donovanc8a68902013-08-22 10:13:51 -040031 starPos token.Pos // source position, if from explicit *addr
32 expr ast.Expr // source syntax [debug mode]
Rob Pike83f21b92013-05-17 13:25:48 -070033}
34
Alan Donovan69ce87a2013-07-18 16:59:06 -040035func (a *address) load(fn *Function) Value {
Alan Donovan6c7ce1c2013-05-30 09:59:17 -040036 load := emitLoad(fn, a.addr)
Alan Donovan55d678e2013-07-15 13:56:46 -040037 load.pos = a.starPos
Alan Donovan6c7ce1c2013-05-30 09:59:17 -040038 return load
Rob Pike83f21b92013-05-17 13:25:48 -070039}
40
Alan Donovan69ce87a2013-07-18 16:59:06 -040041func (a *address) store(fn *Function, v Value) {
Alan Donovan55d678e2013-07-15 13:56:46 -040042 store := emitStore(fn, a.addr, v)
43 store.pos = a.starPos
Alan Donovanc28bf6e2013-07-31 13:13:05 -040044 if a.expr != nil {
Alan Donovan55d678e2013-07-15 13:56:46 -040045 // store.Val is v converted for assignability.
Alan Donovanc28bf6e2013-07-31 13:13:05 -040046 emitDebugRef(fn, a.expr, store.Val)
Alan Donovan55d678e2013-07-15 13:56:46 -040047 }
48}
49
Alan Donovan69ce87a2013-07-18 16:59:06 -040050func (a *address) address(fn *Function) Value {
Alan Donovanc28bf6e2013-07-31 13:13:05 -040051 if a.expr != nil {
Alan Donovan55d678e2013-07-15 13:56:46 -040052 // NB: this kind of DebugRef yields the object's address.
Alan Donovanc28bf6e2013-07-31 13:13:05 -040053 emitDebugRef(fn, a.expr, a.addr)
Alan Donovan55d678e2013-07-15 13:56:46 -040054 }
55 return a.addr
Rob Pike83f21b92013-05-17 13:25:48 -070056}
57
Alan Donovan69ce87a2013-07-18 16:59:06 -040058func (a *address) typ() types.Type {
Robert Griesemerf1a88912013-07-12 21:09:33 -070059 return deref(a.addr.Type())
Rob Pike83f21b92013-05-17 13:25:48 -070060}
61
62// An element is an lvalue represented by m[k], the location of an
63// element of a map or string. These locations are not addressable
64// since pointers cannot be formed from them, but they do support
65// load(), and in the case of maps, store().
66//
67type element struct {
68 m, k Value // map or string
69 t types.Type // map element type or string byte type
Alan Donovanc8a68902013-08-22 10:13:51 -040070 pos token.Pos // source position of colon ({k:v}) or lbrack (m[k]=v)
Rob Pike83f21b92013-05-17 13:25:48 -070071}
72
73func (e *element) load(fn *Function) Value {
74 l := &Lookup{
75 X: e.m,
76 Index: e.k,
77 }
Alan Donovanc8a68902013-08-22 10:13:51 -040078 l.setPos(e.pos)
Rob Pike83f21b92013-05-17 13:25:48 -070079 l.setType(e.t)
80 return fn.emit(l)
81}
82
83func (e *element) store(fn *Function, v Value) {
Alan Donovanc8a68902013-08-22 10:13:51 -040084 up := &MapUpdate{
Rob Pike83f21b92013-05-17 13:25:48 -070085 Map: e.m,
86 Key: e.k,
87 Value: emitConv(fn, v, e.t),
Alan Donovanc8a68902013-08-22 10:13:51 -040088 }
89 up.pos = e.pos
90 fn.emit(up)
Rob Pike83f21b92013-05-17 13:25:48 -070091}
92
Alan Donovan55d678e2013-07-15 13:56:46 -040093func (e *element) address(fn *Function) Value {
94 panic("map/string elements are not addressable")
95}
96
Rob Pike83f21b92013-05-17 13:25:48 -070097func (e *element) typ() types.Type {
98 return e.t
99}
100
Alan Donovan70722532013-08-19 15:38:30 -0400101// A blank is a dummy variable whose name is "_".
Rob Pike83f21b92013-05-17 13:25:48 -0700102// It is not reified: loads are illegal and stores are ignored.
103//
104type blank struct{}
105
106func (bl blank) load(fn *Function) Value {
107 panic("blank.load is illegal")
108}
109
110func (bl blank) store(fn *Function, v Value) {
111 // no-op
112}
113
Alan Donovan55d678e2013-07-15 13:56:46 -0400114func (bl blank) address(fn *Function) Value {
115 panic("blank var is not addressable")
116}
117
Rob Pike83f21b92013-05-17 13:25:48 -0700118func (bl blank) typ() types.Type {
119 // This should be the type of the blank Ident; the typechecker
120 // doesn't provide this yet, but fortunately, we don't need it
121 // yet either.
122 panic("blank.typ is unimplemented")
123}