/
zero_value.go
79 lines (66 loc) · 1.62 KB
/
zero_value.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
package sexpconv
import (
"exn"
"go/types"
"magic_pkg/emacs/lisp"
"sexp"
)
func basicTypeZeroValue(typ *types.Basic) sexp.Form {
switch typ.Kind() {
case types.String:
return sexp.Str("")
case types.Bool:
return sexp.Bool(false)
default:
info := typ.Info()
if info&types.IsFloat != 0 {
return sexp.Float(0)
}
if info&types.IsInteger != 0 {
return sexp.Int(0)
}
}
panic(exn.NoImpl("can not provide zero value for %#v", typ))
}
func ZeroValue(typ types.Type) sexp.Form {
if lisp.Package != nil && types.Identical(typ, lisp.TypSymbol) {
return sexp.Nil
}
switch typ := typ.(type) {
case *types.Basic:
return basicTypeZeroValue(typ)
case *types.Array:
zv := ZeroValue(typ.Elem())
return &sexp.SparseArrayLit{
Ctor: sexp.NewLispCall(lisp.FnMakeVector, sexp.Int(typ.Len()), zv),
Typ: typ,
}
case *types.Map:
return nilMap
case *types.Named:
utyp := typ.Underlying()
if structTyp, ok := utyp.(*types.Struct); ok {
vals := make([]sexp.Form, structTyp.NumFields())
for i := range vals {
vals[i] = ZeroValue(structTyp.Field(i).Type())
}
return &sexp.StructLit{Vals: vals, Typ: typ}
}
if _, ok := utyp.(*types.Interface); ok {
return nilInterface
}
return basicTypeZeroValue(utyp.(*types.Basic))
}
panic(exn.NoImpl("can not provide zero value for %#v", typ))
}
// Nil values
var (
// #REFS: #74.
nilMap = sexp.Var{
Name: "goism-rt.NilMap",
Typ: types.NewMap(lisp.TypObject, lisp.TypObject),
}
nilFunc = sexp.Symbol{Val: "goism-rt.NilFunction"}
nilInterface = sexp.Symbol{Val: "goism-rt.NilInterface"}
nilSlice = sexp.Symbol{Val: "goism-rt.NilSlice"}
)