Alan Donovan | 713699d | 2013-08-27 18:49:13 -0400 | [diff] [blame^] | 1 | // 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 Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 5 | package ssa |
| 6 | |
| 7 | // lvalues are the union of addressable expressions and map-index |
| 8 | // expressions. |
| 9 | |
| 10 | import ( |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 11 | "go/ast" |
Alan Donovan | 6c7ce1c | 2013-05-30 09:59:17 -0400 | [diff] [blame] | 12 | "go/token" |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 13 | |
| 14 | "code.google.com/p/go.tools/go/types" |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 15 | ) |
| 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 | // |
| 21 | type 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 Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 24 | address(fn *Function) Value // address of the location |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 25 | typ() types.Type // returns the type of the location |
| 26 | } |
| 27 | |
| 28 | // An address is an lvalue represented by a true pointer. |
| 29 | type address struct { |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 30 | addr Value |
Alan Donovan | c8a6890 | 2013-08-22 10:13:51 -0400 | [diff] [blame] | 31 | starPos token.Pos // source position, if from explicit *addr |
| 32 | expr ast.Expr // source syntax [debug mode] |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 33 | } |
| 34 | |
Alan Donovan | 69ce87a | 2013-07-18 16:59:06 -0400 | [diff] [blame] | 35 | func (a *address) load(fn *Function) Value { |
Alan Donovan | 6c7ce1c | 2013-05-30 09:59:17 -0400 | [diff] [blame] | 36 | load := emitLoad(fn, a.addr) |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 37 | load.pos = a.starPos |
Alan Donovan | 6c7ce1c | 2013-05-30 09:59:17 -0400 | [diff] [blame] | 38 | return load |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 39 | } |
| 40 | |
Alan Donovan | 69ce87a | 2013-07-18 16:59:06 -0400 | [diff] [blame] | 41 | func (a *address) store(fn *Function, v Value) { |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 42 | store := emitStore(fn, a.addr, v) |
| 43 | store.pos = a.starPos |
Alan Donovan | c28bf6e | 2013-07-31 13:13:05 -0400 | [diff] [blame] | 44 | if a.expr != nil { |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 45 | // store.Val is v converted for assignability. |
Alan Donovan | c28bf6e | 2013-07-31 13:13:05 -0400 | [diff] [blame] | 46 | emitDebugRef(fn, a.expr, store.Val) |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 47 | } |
| 48 | } |
| 49 | |
Alan Donovan | 69ce87a | 2013-07-18 16:59:06 -0400 | [diff] [blame] | 50 | func (a *address) address(fn *Function) Value { |
Alan Donovan | c28bf6e | 2013-07-31 13:13:05 -0400 | [diff] [blame] | 51 | if a.expr != nil { |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 52 | // NB: this kind of DebugRef yields the object's address. |
Alan Donovan | c28bf6e | 2013-07-31 13:13:05 -0400 | [diff] [blame] | 53 | emitDebugRef(fn, a.expr, a.addr) |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 54 | } |
| 55 | return a.addr |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 56 | } |
| 57 | |
Alan Donovan | 69ce87a | 2013-07-18 16:59:06 -0400 | [diff] [blame] | 58 | func (a *address) typ() types.Type { |
Robert Griesemer | f1a8891 | 2013-07-12 21:09:33 -0700 | [diff] [blame] | 59 | return deref(a.addr.Type()) |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 60 | } |
| 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 | // |
| 67 | type element struct { |
| 68 | m, k Value // map or string |
| 69 | t types.Type // map element type or string byte type |
Alan Donovan | c8a6890 | 2013-08-22 10:13:51 -0400 | [diff] [blame] | 70 | pos token.Pos // source position of colon ({k:v}) or lbrack (m[k]=v) |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 71 | } |
| 72 | |
| 73 | func (e *element) load(fn *Function) Value { |
| 74 | l := &Lookup{ |
| 75 | X: e.m, |
| 76 | Index: e.k, |
| 77 | } |
Alan Donovan | c8a6890 | 2013-08-22 10:13:51 -0400 | [diff] [blame] | 78 | l.setPos(e.pos) |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 79 | l.setType(e.t) |
| 80 | return fn.emit(l) |
| 81 | } |
| 82 | |
| 83 | func (e *element) store(fn *Function, v Value) { |
Alan Donovan | c8a6890 | 2013-08-22 10:13:51 -0400 | [diff] [blame] | 84 | up := &MapUpdate{ |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 85 | Map: e.m, |
| 86 | Key: e.k, |
| 87 | Value: emitConv(fn, v, e.t), |
Alan Donovan | c8a6890 | 2013-08-22 10:13:51 -0400 | [diff] [blame] | 88 | } |
| 89 | up.pos = e.pos |
| 90 | fn.emit(up) |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 91 | } |
| 92 | |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 93 | func (e *element) address(fn *Function) Value { |
| 94 | panic("map/string elements are not addressable") |
| 95 | } |
| 96 | |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 97 | func (e *element) typ() types.Type { |
| 98 | return e.t |
| 99 | } |
| 100 | |
Alan Donovan | 7072253 | 2013-08-19 15:38:30 -0400 | [diff] [blame] | 101 | // A blank is a dummy variable whose name is "_". |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 102 | // It is not reified: loads are illegal and stores are ignored. |
| 103 | // |
| 104 | type blank struct{} |
| 105 | |
| 106 | func (bl blank) load(fn *Function) Value { |
| 107 | panic("blank.load is illegal") |
| 108 | } |
| 109 | |
| 110 | func (bl blank) store(fn *Function, v Value) { |
| 111 | // no-op |
| 112 | } |
| 113 | |
Alan Donovan | 55d678e | 2013-07-15 13:56:46 -0400 | [diff] [blame] | 114 | func (bl blank) address(fn *Function) Value { |
| 115 | panic("blank var is not addressable") |
| 116 | } |
| 117 | |
Rob Pike | 83f21b9 | 2013-05-17 13:25:48 -0700 | [diff] [blame] | 118 | func (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 | } |