@@ -20,6 +20,17 @@ type arrayDesc struct {
20
20
elementContains [][]placeholderID
21
21
}
22
22
23
+ func (a * arrayDesc ) copy () * arrayDesc {
24
+ elementContains := make ([][]placeholderID , len (a .elementContains ))
25
+ for i , placeholders := range a .elementContains {
26
+ elementContains [i ] = append ([]placeholderID {}, placeholders ... )
27
+ }
28
+ return & arrayDesc {
29
+ furtherContain : append ([]placeholderID {}, a .furtherContain ... ),
30
+ elementContains : elementContains ,
31
+ }
32
+ }
33
+
23
34
func (a * arrayDesc ) widen (other * arrayDesc ) {
24
35
if other == nil {
25
36
return
@@ -49,6 +60,18 @@ type objectDesc struct {
49
60
allFieldsKnown bool
50
61
}
51
62
63
+ func (o * objectDesc ) copy () * objectDesc {
64
+ fieldContains := make (map [string ][]placeholderID )
65
+ for k , v := range o .fieldContains {
66
+ fieldContains [k ] = append ([]placeholderID (nil ), v ... )
67
+ }
68
+ return & objectDesc {
69
+ unknownContain : append ([]placeholderID (nil ), o .unknownContain ... ),
70
+ fieldContains : fieldContains ,
71
+ allFieldsKnown : o .allFieldsKnown ,
72
+ }
73
+ }
74
+
52
75
func (o * objectDesc ) widen (other * objectDesc ) {
53
76
if other == nil {
54
77
return
@@ -77,6 +100,7 @@ func (o *objectDesc) normalize() {
77
100
type functionDesc struct {
78
101
resultContains []placeholderID
79
102
103
+ // Read-only
80
104
// TODO(sbarzowski) instead of keeping "real" parameters here,
81
105
// maybe keep only what we care about in the linter desc
82
106
// (names and required-or-not).
@@ -85,6 +109,15 @@ type functionDesc struct {
85
109
minArity , maxArity int
86
110
}
87
111
112
+ func (f * functionDesc ) copy () * functionDesc {
113
+ return & functionDesc {
114
+ resultContains : append ([]placeholderID {}, f .resultContains ... ),
115
+ params : f .params , // this is read-only, so we can just copy the pointer
116
+ minArity : f .minArity ,
117
+ maxArity : f .maxArity ,
118
+ }
119
+ }
120
+
88
121
func sameParameters (a , b []ast.Parameter ) bool {
89
122
if a == nil || b == nil {
90
123
return false
@@ -255,7 +288,44 @@ func Describe(t *TypeDesc) string {
255
288
return strings .Join (parts , " or " )
256
289
}
257
290
291
+ // An intuitive notion of the size of type description. For performance tuning.
292
+ func (t * TypeDesc ) size () int64 {
293
+ res := 0
294
+ if t .Bool {
295
+ res += 1
296
+ }
297
+ if t .Number {
298
+ res += 1
299
+ }
300
+ if t .String {
301
+ res += 1
302
+ }
303
+ if t .Null {
304
+ res += 1
305
+ }
306
+ if t .Function () {
307
+ res += len (t .FunctionDesc .resultContains )
308
+ res += len (t .FunctionDesc .params )
309
+ }
310
+ if t .Array () {
311
+ res += len (t .ArrayDesc .furtherContain )
312
+ for _ , elem := range t .ArrayDesc .elementContains {
313
+ res += len (elem )
314
+ }
315
+ }
316
+ if t .Object () {
317
+ res += len (t .ObjectDesc .unknownContain )
318
+ for _ , elem := range t .ObjectDesc .fieldContains {
319
+ res += len (elem )
320
+ }
321
+ }
322
+ return int64 (res )
323
+ }
324
+
258
325
func (t * TypeDesc ) widen (b * TypeDesc ) {
326
+ if t == b {
327
+ panic ("BUG: widening the type with itself" )
328
+ }
259
329
t .Bool = t .Bool || b .Bool
260
330
t .Number = t .Number || b .Number
261
331
t .String = t .String || b .String
@@ -264,22 +334,24 @@ func (t *TypeDesc) widen(b *TypeDesc) {
264
334
if t .FunctionDesc != nil {
265
335
t .FunctionDesc .widen (b .FunctionDesc )
266
336
} else if t .FunctionDesc == nil && b .FunctionDesc != nil {
267
- copy := * b .FunctionDesc
268
- t .FunctionDesc = & copy
337
+ // copy := *b.FunctionDesc
338
+ // t.FunctionDesc = ©
339
+ t .FunctionDesc = b .FunctionDesc .copy ()
269
340
}
270
341
271
342
if t .ObjectDesc != nil {
272
343
t .ObjectDesc .widen (b .ObjectDesc )
273
344
} else if t .ObjectDesc == nil && b .ObjectDesc != nil {
274
- copy := * b .ObjectDesc
275
- t .ObjectDesc = & copy
345
+ // TODO(sbarzowski) Maybe we want copy on write?
346
+ // So that it actually aliases underneath (with a bool marker)
347
+ // which is resolved when widened.
348
+ t .ObjectDesc = b .ObjectDesc .copy ()
276
349
}
277
350
278
351
if t .ArrayDesc != nil {
279
352
t .ArrayDesc .widen (b .ArrayDesc )
280
353
} else if t .ArrayDesc == nil && b .ArrayDesc != nil {
281
- copy := * b .ArrayDesc
282
- t .ArrayDesc = & copy
354
+ t .ArrayDesc = b .ArrayDesc .copy ()
283
355
}
284
356
}
285
357
0 commit comments