Source file
src/go/types/literals.go
1
2
3
4
5
6
7
8
9
10 package types
11
12 import (
13 "go/ast"
14 "go/token"
15 . "internal/types/errors"
16 "strings"
17 )
18
19
20
21 func (check *Checker) langCompat(lit *ast.BasicLit) {
22 s := lit.Value
23 if len(s) <= 2 || check.allowVersion(go1_13) {
24 return
25 }
26
27 if strings.Contains(s, "_") {
28 check.versionErrorf(lit, go1_13, "underscore in numeric literal")
29 return
30 }
31 if s[0] != '0' {
32 return
33 }
34 radix := s[1]
35 if radix == 'b' || radix == 'B' {
36 check.versionErrorf(lit, go1_13, "binary literal")
37 return
38 }
39 if radix == 'o' || radix == 'O' {
40 check.versionErrorf(lit, go1_13, "0o/0O-style octal literal")
41 return
42 }
43 if lit.Kind != token.INT && (radix == 'x' || radix == 'X') {
44 check.versionErrorf(lit, go1_13, "hexadecimal floating-point literal")
45 }
46 }
47
48 func (check *Checker) basicLit(x *operand, e *ast.BasicLit) {
49 switch e.Kind {
50 case token.INT, token.FLOAT, token.IMAG:
51 check.langCompat(e)
52
53
54
55
56
57
58
59
60
61 const limit = 10000
62 if len(e.Value) > limit {
63 check.errorf(e, InvalidConstVal, "excessively long constant: %s... (%d chars)", e.Value[:10], len(e.Value))
64 x.invalidate()
65 return
66 }
67 }
68 x.setConst(e.Kind, e.Value)
69 if !x.isValid() {
70
71
72
73
74 check.errorf(e, InvalidConstVal, "malformed constant: %s", e.Value)
75 x.invalidate()
76 return
77 }
78
79 x.expr = e
80 check.overflow(x, opPos(x.expr))
81 }
82
83 func (check *Checker) funcLit(x *operand, e *ast.FuncLit) {
84 if sig, ok := check.typ(e.Type).(*Signature); ok {
85
86
87 sig.scope.pos = e.Pos()
88 sig.scope.end = endPos(e)
89 if !check.conf.IgnoreFuncBodies && e.Body != nil {
90
91
92
93 decl := check.decl
94 iota := check.iota
95
96
97
98
99 check.later(func() {
100 check.funcBody(decl, "<function literal>", sig, e.Body, iota)
101 }).describef(e, "func literal")
102 }
103 x.mode_ = value
104 x.typ_ = sig
105 } else {
106 check.errorf(e, InvalidSyntaxTree, "invalid function literal %v", e)
107 x.invalidate()
108 }
109 }
110
111 func (check *Checker) compositeLit(x *operand, e *ast.CompositeLit, hint Type) {
112 var typ, base Type
113 var isElem bool
114
115 switch {
116 case e.Type != nil:
117
118
119
120 if atyp, _ := e.Type.(*ast.ArrayType); atyp != nil && isdddArray(atyp) {
121
122
123
124 typ = &Array{len: -1, elem: check.varType(atyp.Elt)}
125 base = typ
126 break
127 }
128 typ = check.typ(e.Type)
129 base = typ
130
131 case hint != nil:
132
133 typ = hint
134 base = typ
135
136 u, _ := commonUnder(base, nil)
137 if b, ok := deref(u); ok {
138 base = b
139 }
140 isElem = true
141
142 default:
143
144 check.error(e, UntypedLit, "missing type in composite literal")
145
146 typ = Typ[Invalid]
147 base = typ
148 }
149
150
151 if !check.isComplete(base) {
152 x.invalidate()
153 return
154 }
155
156 switch u, _ := commonUnder(base, nil); utyp := u.(type) {
157 case *Struct:
158 if len(e.Elts) == 0 {
159 break
160 }
161
162
163
164 fields := utyp.fields
165 if _, ok := e.Elts[0].(*ast.KeyValueExpr); ok {
166
167 visited := make(trie[*Var])
168 for _, e := range e.Elts {
169 kv, _ := e.(*ast.KeyValueExpr)
170 if kv == nil {
171 check.error(e, MixedStructLit, "mixture of field:value and value elements in struct literal")
172 continue
173 }
174 key, _ := kv.Key.(*ast.Ident)
175
176
177 check.expr(nil, x, kv.Value)
178 if key == nil {
179 check.errorf(kv, InvalidLitField, "invalid field name %s in struct literal", kv.Key)
180 continue
181 }
182 obj, index, indirect := lookupFieldOrMethod(utyp, false, check.pkg, key.Name, false)
183 if obj == nil {
184 alt, _, _ := lookupFieldOrMethod(utyp, false, check.pkg, key.Name, true)
185 msg := check.lookupError(base, key.Name, alt, true)
186 check.error(kv.Key, MissingLitField, msg)
187 continue
188 }
189 fld, _ := obj.(*Var)
190 if fld == nil {
191 check.errorf(kv.Key, MissingLitField, "%s is not a field", kv.Key)
192 continue
193 }
194 if check.allowVersion(go1_27) {
195 if indirect {
196 check.errorf(kv.Key, InvalidLitField, "invalid implicit pointer indirection to reach %s", kv.Key)
197 continue
198 }
199 } else {
200 if len(index) > 1 {
201 check.errorf(kv.Key, InvalidLitField, "cannot use promoted field %s in struct literal of type %s", fieldPath(utyp, index), base)
202 continue
203 }
204 }
205 check.recordUse(key, fld)
206 etyp := fld.typ
207 check.assignment(x, etyp, "struct literal")
208 if alt, n := visited.insert(index, fld); n != 0 {
209 if fld == alt {
210 check.errorf(kv, DuplicateLitField, "duplicate field name %s in struct literal", fld.name)
211 } else if n < len(index) {
212 check.errorf(kv, DuplicateLitField, "cannot specify promoted field %s and enclosing embedded field %s", fld.name, alt.name)
213 } else {
214 check.errorf(kv, DuplicateLitField, "cannot specify embedded field %s and enclosed promoted field %s", fld.name, alt.name)
215 }
216 }
217 }
218 } else {
219
220 for i, e := range e.Elts {
221 if kv, _ := e.(*ast.KeyValueExpr); kv != nil {
222 check.error(kv, MixedStructLit, "mixture of field:value and value elements in struct literal")
223 continue
224 }
225 check.expr(nil, x, e)
226 if i >= len(fields) {
227 check.errorf(x, InvalidStructLit, "too many values in struct literal of type %s", base)
228 break
229 }
230
231 fld := fields[i]
232 if !fld.Exported() && fld.pkg != check.pkg {
233 check.errorf(x, UnexportedLitField, "implicit assignment to unexported field %s in struct literal of type %s", fld.name, base)
234 continue
235 }
236 etyp := fld.typ
237 check.assignment(x, etyp, "struct literal")
238 }
239 if len(e.Elts) < len(fields) {
240 check.errorf(inNode(e, e.Rbrace), InvalidStructLit, "too few values in struct literal of type %s", base)
241
242 }
243 }
244
245 case *Array:
246 n := check.indexedElts(e.Elts, utyp.elem, utyp.len)
247
248
249
250
251
252
253
254
255 if utyp.len < 0 {
256 utyp.len = n
257
258
259
260
261 if e.Type != nil {
262 check.recordTypeAndValue(e.Type, typexpr, utyp, nil)
263 }
264 }
265
266 case *Slice:
267 check.indexedElts(e.Elts, utyp.elem, -1)
268
269 case *Map:
270
271
272
273 keyIsInterface := isNonTypeParamInterface(utyp.key)
274 visited := make(map[any][]Type, len(e.Elts))
275 for _, e := range e.Elts {
276 kv, _ := e.(*ast.KeyValueExpr)
277 if kv == nil {
278 check.error(e, MissingLitKey, "missing key in map literal")
279 continue
280 }
281 check.exprWithHint(x, kv.Key, utyp.key)
282 check.assignment(x, utyp.key, "map literal")
283 if !x.isValid() {
284 continue
285 }
286 if x.mode() == constant_ {
287 duplicate := false
288 xkey := keyVal(x.val)
289 if keyIsInterface {
290 for _, vtyp := range visited[xkey] {
291 if Identical(vtyp, x.typ()) {
292 duplicate = true
293 break
294 }
295 }
296 visited[xkey] = append(visited[xkey], x.typ())
297 } else {
298 _, duplicate = visited[xkey]
299 visited[xkey] = nil
300 }
301 if duplicate {
302 check.errorf(x, DuplicateLitKey, "duplicate key %s in map literal", x.val)
303 continue
304 }
305 }
306 check.exprWithHint(x, kv.Value, utyp.elem)
307 check.assignment(x, utyp.elem, "map literal")
308 }
309
310 default:
311
312
313 for _, e := range e.Elts {
314 if kv, _ := e.(*ast.KeyValueExpr); kv != nil {
315
316
317
318 e = kv.Value
319 }
320 check.use(e)
321 }
322
323 if isValid(utyp) {
324 var qualifier string
325 if isElem {
326 qualifier = " element"
327 }
328 var cause string
329 if utyp == nil {
330 cause = " (no common underlying type)"
331 }
332 check.errorf(e, InvalidLit, "invalid composite literal%s type %s%s", qualifier, typ, cause)
333 x.invalidate()
334 return
335 }
336 }
337
338 x.mode_ = value
339 x.typ_ = typ
340 }
341
342
343
344
345
346 func (check *Checker) indexedElts(elts []ast.Expr, typ Type, length int64) int64 {
347 visited := make(map[int64]bool, len(elts))
348 var index, max int64
349 for _, e := range elts {
350
351 validIndex := false
352 eval := e
353 if kv, _ := e.(*ast.KeyValueExpr); kv != nil {
354 if typ, i := check.index(kv.Key, length); isValid(typ) {
355 if i >= 0 {
356 index = i
357 validIndex = true
358 } else {
359 check.errorf(e, InvalidLitIndex, "index %s must be integer constant", kv.Key)
360 }
361 }
362 eval = kv.Value
363 } else if length >= 0 && index >= length {
364 check.errorf(e, OversizeArrayLit, "index %d is out of bounds (>= %d)", index, length)
365 } else {
366 validIndex = true
367 }
368
369
370 if validIndex {
371 if visited[index] {
372 check.errorf(e, DuplicateLitKey, "duplicate index %d in array or slice literal", index)
373 }
374 visited[index] = true
375 }
376 index++
377 if index > max {
378 max = index
379 }
380
381
382 var x operand
383 check.exprWithHint(&x, eval, typ)
384 check.assignment(&x, typ, "array or slice literal")
385 }
386 return max
387 }
388
View as plain text