-
-
Notifications
You must be signed in to change notification settings - Fork 23
/
macros.go
113 lines (103 loc) · 2.2 KB
/
macros.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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
package cxgo
import (
"fmt"
"github.com/gotranspile/cxgo/types"
"sort"
"strconv"
"strings"
"modernc.org/cc/v3"
)
func (g *translator) convertValue(v cc.Value) Expr {
switch v := v.(type) {
case cc.Int64Value:
return cIntLit(int64(v))
case cc.Uint64Value:
return cUintLit(uint64(v))
case cc.Float32Value:
return FloatLit{val: float64(v)}
case cc.Float64Value:
return FloatLit{val: float64(v)}
case cc.StringValue:
e, err := g.parseCStringLit(string(v))
if err != nil {
panic(err)
}
return e
default:
panic(fmt.Errorf("unsupported value type: %T", v))
}
}
func (g *translator) convertMacros(ast *cc.AST) []CDecl {
type macro struct {
name string
m *cc.Macro
}
var arr []macro
for name, mc := range ast.Macros {
if !g.inCurFile(mc) {
continue
}
if mc.IsFnLike() {
continue // we don't support function macros yet
}
if len(mc.ReplacementTokens()) == 0 {
continue // no value
}
arr = append(arr, macro{name.String(), mc})
}
sort.Slice(arr, func(i, j int) bool {
return arr[i].m.Position().Offset < arr[j].m.Position().Offset
})
var decls []CDecl
for _, mc := range arr {
if val := g.evalMacro(mc.m, ast); val != nil {
typ := val.CType(nil)
id := types.NewIdent(mc.name, typ)
decls = append(decls, &CVarDecl{Const: true, CVarSpec: CVarSpec{
g: g, Type: typ,
Names: []*types.Ident{id},
Inits: []Expr{val},
}})
}
}
return decls
}
func (g *translator) evalMacro(m *cc.Macro, ast *cc.AST) Expr {
toks := m.ReplacementTokens()
if len(toks) != 1 {
return evalMacro2(m, ast)
}
src := strings.TrimSpace(toks[0].Src.String())
if len(src) == 0 {
return nil
}
if src[0] == '"' {
if s, err := strconv.Unquote(src); err == nil {
if l, err := g.parseCStringLit(s); err == nil {
return l
}
}
} else {
if l, err := parseCIntLit(src); err == nil {
return l
}
if l, err := parseCFloatLit(src); err == nil {
return l
}
}
return evalMacro2(m, ast)
}
func evalMacro2(m *cc.Macro, ast *cc.AST) Expr {
op, err := ast.Eval(m)
if err != nil {
return nil
}
switch x := op.Value().(type) {
case cc.Int64Value:
return cIntLit(int64(x))
case cc.Uint64Value:
return cUintLit(uint64(x))
default:
return nil
}
}