Russ Cox | d3c758d | 2013-03-20 17:11:09 -0400 | [diff] [blame] | 1 | // run |
| 2 | |
| 3 | // Copyright 2013 The Go Authors. All rights reserved. |
| 4 | // Use of this source code is governed by a BSD-style |
| 5 | // license that can be found in the LICENSE file. |
| 6 | |
| 7 | package main |
| 8 | |
| 9 | // Concrete types implementing M method. |
| 10 | // Smaller than a word, word-sized, larger than a word. |
| 11 | // Value and pointer receivers. |
| 12 | |
| 13 | type Tinter interface { |
| 14 | M(int, byte) (byte, int) |
| 15 | } |
| 16 | |
| 17 | type Tsmallv byte |
| 18 | |
| 19 | func (v Tsmallv) M(x int, b byte) (byte, int) { return b, x+int(v) } |
| 20 | |
| 21 | type Tsmallp byte |
| 22 | |
| 23 | func (p *Tsmallp) M(x int, b byte) (byte, int) { return b, x+int(*p) } |
| 24 | |
| 25 | type Twordv uintptr |
| 26 | |
| 27 | func (v Twordv) M(x int, b byte) (byte, int) { return b, x+int(v) } |
| 28 | |
| 29 | type Twordp uintptr |
| 30 | |
| 31 | func (p *Twordp) M(x int, b byte) (byte, int) { return b, x+int(*p) } |
| 32 | |
| 33 | type Tbigv [2]uintptr |
| 34 | |
| 35 | func (v Tbigv) M(x int, b byte) (byte, int) { return b, x+int(v[0])+int(v[1]) } |
| 36 | |
| 37 | type Tbigp [2]uintptr |
| 38 | |
| 39 | func (p *Tbigp) M(x int, b byte) (byte, int) { return b, x+int(p[0])+int(p[1]) } |
| 40 | |
| 41 | // Again, with an unexported method. |
| 42 | |
| 43 | type tsmallv byte |
| 44 | |
| 45 | func (v tsmallv) m(x int, b byte) (byte, int) { return b, x+int(v) } |
| 46 | |
| 47 | type tsmallp byte |
| 48 | |
| 49 | func (p *tsmallp) m(x int, b byte) (byte, int) { return b, x+int(*p) } |
| 50 | |
| 51 | type twordv uintptr |
| 52 | |
| 53 | func (v twordv) m(x int, b byte) (byte, int) { return b, x+int(v) } |
| 54 | |
| 55 | type twordp uintptr |
| 56 | |
| 57 | func (p *twordp) m(x int, b byte) (byte, int) { return b, x+int(*p) } |
| 58 | |
| 59 | type tbigv [2]uintptr |
| 60 | |
| 61 | func (v tbigv) m(x int, b byte) (byte, int) { return b, x+int(v[0])+int(v[1]) } |
| 62 | |
| 63 | type tbigp [2]uintptr |
| 64 | |
| 65 | func (p *tbigp) m(x int, b byte) (byte, int) { return b, x+int(p[0])+int(p[1]) } |
| 66 | |
| 67 | type tinter interface { |
| 68 | m(int, byte) (byte, int) |
| 69 | } |
| 70 | |
| 71 | // Embedding via pointer. |
| 72 | |
| 73 | type T1 struct { |
| 74 | T2 |
| 75 | } |
| 76 | |
| 77 | type T2 struct { |
| 78 | *T3 |
| 79 | } |
| 80 | |
| 81 | type T3 struct { |
| 82 | *T4 |
| 83 | } |
| 84 | |
| 85 | type T4 struct { |
| 86 | } |
| 87 | |
| 88 | func (t4 T4) M(x int, b byte) (byte, int) { return b, x+40 } |
| 89 | |
| 90 | var failed = false |
| 91 | |
| 92 | func CheckI(name string, i Tinter, inc int) { |
| 93 | b, x := i.M(1000, 99) |
| 94 | if b != 99 || x != 1000+inc { |
| 95 | failed = true |
| 96 | print(name, ".M(1000, 99) = ", b, ", ", x, " want 99, ", 1000+inc, "\n") |
| 97 | } |
| 98 | |
| 99 | CheckF("(i="+name+")", i.M, inc) |
| 100 | } |
| 101 | |
| 102 | func CheckF(name string, f func(int, byte) (byte, int), inc int) { |
| 103 | b, x := f(1000, 99) |
| 104 | if b != 99 || x != 1000+inc { |
| 105 | failed = true |
| 106 | print(name, "(1000, 99) = ", b, ", ", x, " want 99, ", 1000+inc, "\n") |
| 107 | } |
| 108 | } |
| 109 | |
| 110 | func checkI(name string, i tinter, inc int) { |
| 111 | b, x := i.m(1000, 99) |
| 112 | if b != 99 || x != 1000+inc { |
| 113 | failed = true |
| 114 | print(name, ".m(1000, 99) = ", b, ", ", x, " want 99, ", 1000+inc, "\n") |
| 115 | } |
| 116 | |
| 117 | checkF("(i="+name+")", i.m, inc) |
| 118 | } |
| 119 | |
| 120 | func checkF(name string, f func(int, byte) (byte, int), inc int) { |
| 121 | b, x := f(1000, 99) |
| 122 | if b != 99 || x != 1000+inc { |
| 123 | failed = true |
| 124 | print(name, "(1000, 99) = ", b, ", ", x, " want 99, ", 1000+inc, "\n") |
| 125 | } |
| 126 | } |
| 127 | |
| 128 | func shouldPanic(f func()) { |
| 129 | defer func() { |
| 130 | if recover() == nil { |
| 131 | panic("not panicking") |
| 132 | } |
| 133 | }() |
| 134 | f() |
| 135 | } |
| 136 | |
| 137 | func shouldNotPanic(f func()) { |
| 138 | f() |
| 139 | } |
| 140 | |
| 141 | func main() { |
| 142 | sv := Tsmallv(1) |
| 143 | CheckI("sv", sv, 1) |
| 144 | CheckF("sv.M", sv.M, 1) |
| 145 | CheckF("(&sv).M", (&sv).M, 1) |
| 146 | psv := &sv |
| 147 | CheckI("psv", psv, 1) |
| 148 | CheckF("psv.M", psv.M, 1) |
| 149 | CheckF("(*psv).M", (*psv).M, 1) |
| 150 | |
| 151 | sp := Tsmallp(2) |
| 152 | CheckI("&sp", &sp, 2) |
| 153 | CheckF("sp.M", sp.M, 2) |
| 154 | CheckF("(&sp).M", (&sp).M, 2) |
| 155 | psp := &sp |
| 156 | CheckI("psp", psp, 2) |
| 157 | CheckF("psp.M", psp.M, 2) |
| 158 | CheckF("(*psp).M", (*psp).M, 2) |
| 159 | |
| 160 | wv := Twordv(3) |
| 161 | CheckI("wv", wv, 3) |
| 162 | CheckF("wv.M", wv.M, 3) |
| 163 | CheckF("(&wv).M", (&wv).M, 3) |
| 164 | pwv := &wv |
| 165 | CheckI("pwv", pwv, 3) |
| 166 | CheckF("pwv.M", pwv.M, 3) |
| 167 | CheckF("(*pwv).M", (*pwv).M, 3) |
| 168 | |
| 169 | wp := Twordp(4) |
| 170 | CheckI("&wp", &wp, 4) |
| 171 | CheckF("wp.M", wp.M, 4) |
| 172 | CheckF("(&wp).M", (&wp).M, 4) |
| 173 | pwp := &wp |
| 174 | CheckI("pwp", pwp, 4) |
| 175 | CheckF("pwp.M", pwp.M, 4) |
| 176 | CheckF("(*pwp).M", (*pwp).M, 4) |
| 177 | |
| 178 | bv := Tbigv([2]uintptr{5, 6}) |
| 179 | pbv := &bv |
| 180 | CheckI("bv", bv, 11) |
| 181 | CheckF("bv.M", bv.M, 11) |
| 182 | CheckF("(&bv).M", (&bv).M, 11) |
| 183 | CheckI("pbv", pbv, 11) |
| 184 | CheckF("pbv.M", pbv.M, 11) |
| 185 | CheckF("(*pbv).M", (*pbv).M, 11) |
| 186 | |
| 187 | bp := Tbigp([2]uintptr{7,8}) |
| 188 | CheckI("&bp", &bp, 15) |
| 189 | CheckF("bp.M", bp.M, 15) |
| 190 | CheckF("(&bp).M", (&bp).M, 15) |
| 191 | pbp := &bp |
| 192 | CheckI("pbp", pbp, 15) |
| 193 | CheckF("pbp.M", pbp.M, 15) |
| 194 | CheckF("(*pbp).M", (*pbp).M, 15) |
| 195 | |
| 196 | _sv := tsmallv(1) |
| 197 | checkI("_sv", _sv, 1) |
| 198 | checkF("_sv.m", _sv.m, 1) |
| 199 | checkF("(&_sv).m", (&_sv).m, 1) |
| 200 | _psv := &_sv |
| 201 | checkI("_psv", _psv, 1) |
| 202 | checkF("_psv.m", _psv.m, 1) |
| 203 | checkF("(*_psv).m", (*_psv).m, 1) |
| 204 | |
| 205 | _sp := tsmallp(2) |
| 206 | checkI("&_sp", &_sp, 2) |
| 207 | checkF("_sp.m", _sp.m, 2) |
| 208 | checkF("(&_sp).m", (&_sp).m, 2) |
| 209 | _psp := &_sp |
| 210 | checkI("_psp", _psp, 2) |
| 211 | checkF("_psp.m", _psp.m, 2) |
| 212 | checkF("(*_psp).m", (*_psp).m, 2) |
| 213 | |
| 214 | _wv := twordv(3) |
| 215 | checkI("_wv", _wv, 3) |
| 216 | checkF("_wv.m", _wv.m, 3) |
| 217 | checkF("(&_wv).m", (&_wv).m, 3) |
| 218 | _pwv := &_wv |
| 219 | checkI("_pwv", _pwv, 3) |
| 220 | checkF("_pwv.m", _pwv.m, 3) |
| 221 | checkF("(*_pwv).m", (*_pwv).m, 3) |
| 222 | |
| 223 | _wp := twordp(4) |
| 224 | checkI("&_wp", &_wp, 4) |
| 225 | checkF("_wp.m", _wp.m, 4) |
| 226 | checkF("(&_wp).m", (&_wp).m, 4) |
| 227 | _pwp := &_wp |
| 228 | checkI("_pwp", _pwp, 4) |
| 229 | checkF("_pwp.m", _pwp.m, 4) |
| 230 | checkF("(*_pwp).m", (*_pwp).m, 4) |
| 231 | |
| 232 | _bv := tbigv([2]uintptr{5, 6}) |
| 233 | _pbv := &_bv |
| 234 | checkI("_bv", _bv, 11) |
| 235 | checkF("_bv.m", _bv.m, 11) |
| 236 | checkF("(&_bv).m", (&_bv).m, 11) |
| 237 | checkI("_pbv", _pbv, 11) |
| 238 | checkF("_pbv.m", _pbv.m, 11) |
| 239 | checkF("(*_pbv).m", (*_pbv).m, 11) |
| 240 | |
| 241 | _bp := tbigp([2]uintptr{7,8}) |
| 242 | checkI("&_bp", &_bp, 15) |
| 243 | checkF("_bp.m", _bp.m, 15) |
| 244 | checkF("(&_bp).m", (&_bp).m, 15) |
| 245 | _pbp := &_bp |
| 246 | checkI("_pbp", _pbp, 15) |
| 247 | checkF("_pbp.m", _pbp.m, 15) |
| 248 | checkF("(*_pbp).m", (*_pbp).m, 15) |
| 249 | |
| 250 | t4 := T4{} |
| 251 | t3 := T3{&t4} |
| 252 | t2 := T2{&t3} |
| 253 | t1 := T1{t2} |
| 254 | CheckI("t4", t4, 40) |
| 255 | CheckI("&t4", &t4, 40) |
| 256 | CheckI("t3", t3, 40) |
| 257 | CheckI("&t3", &t3, 40) |
| 258 | CheckI("t2", t2, 40) |
| 259 | CheckI("&t2", &t2, 40) |
| 260 | CheckI("t1", t1, 40) |
| 261 | CheckI("&t1", &t1, 40) |
| 262 | |
| 263 | // x.M panics if x is an interface type and is nil, |
| 264 | // or if x.M expands to (*x).M where x is nil, |
| 265 | // or if x.M expands to x.y.z.w.M where something |
| 266 | // along the evaluation of x.y.z.w is nil. |
| 267 | var f func(int, byte) (byte, int) |
| 268 | shouldPanic(func() { psv = nil; f = psv.M }) |
| 269 | shouldPanic(func() { pwv = nil; f = pwv.M }) |
| 270 | shouldPanic(func() { pbv = nil; f = pbv.M }) |
| 271 | shouldPanic(func() { var i Tinter; f = i.M }) |
| 272 | shouldPanic(func() { _psv = nil; f = _psv.m }) |
| 273 | shouldPanic(func() { _pwv = nil; f = _pwv.m }) |
| 274 | shouldPanic(func() { _pbv = nil; f = _pbv.m }) |
| 275 | shouldPanic(func() { var _i tinter; f = _i.m }) |
| 276 | shouldPanic(func() { var t1 T1; f = t1.M }) |
| 277 | shouldPanic(func() { var t2 T2; f = t2.M }) |
| 278 | shouldPanic(func() { var t3 *T3; f = t3.M }) |
| 279 | shouldPanic(func() { var t3 T3; f = t3.M }) |
| 280 | |
| 281 | if f != nil { |
| 282 | panic("something set f") |
| 283 | } |
| 284 | |
| 285 | // x.M does not panic if x is a nil pointer and |
| 286 | // M is a method with a pointer receiver. |
| 287 | shouldNotPanic(func() { psp = nil; f = psp.M }) |
| 288 | shouldNotPanic(func() { pwp = nil; f = pwp.M }) |
| 289 | shouldNotPanic(func() { pbp = nil; f = pbp.M }) |
| 290 | shouldNotPanic(func() { _psp = nil; f = _psp.m }) |
| 291 | shouldNotPanic(func() { _pwp = nil; f = _pwp.m }) |
| 292 | shouldNotPanic(func() { _pbp = nil; f = _pbp.m }) |
| 293 | shouldNotPanic(func() { var t4 T4; f = t4.M }) |
| 294 | if f == nil { |
| 295 | panic("nothing set f") |
| 296 | } |
| 297 | } |