aboutsummaryrefslogtreecommitdiff
path: root/nexgb/xgbgen/context.go
blob: e5acb1250998639207057dd4cbc6fe97e17c465e (plain)
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
package main

import (
	"bytes"
	"encoding/xml"
	"fmt"
	"log"
	"strings"
)

type Context struct {
	xml *XML
	out *bytes.Buffer
}

func newContext() *Context {
	return &Context{
		xml: &XML{},
		out: bytes.NewBuffer([]byte{}),
	}
}

// Putln calls put and adds a new line to the end of 'format'.
func (c *Context) Putln(format string, v ...interface{}) {
	c.Put(format + "\n", v...)
}

// Put is a short alias to write to 'out'.
func (c *Context) Put(format string, v ...interface{}) {
	_, err := c.out.WriteString(fmt.Sprintf(format, v...))
	if err != nil {
		log.Fatalf("There was an error writing to context buffer: %s", err)
	}
}

// TypePrefix searches the parsed XML for a type matching 'needle'.
// It then returns the appropriate prefix to be used in source code.
// Note that the core X protocol *is* a namespace, but does not have a prefix.
// Also note that you should probably check the BaseTypeMap and TypeMap
// before calling this function.
func (c *Context) TypePrefix(needle Type) string {
	// If this is xproto, quit. No prefixes needed.
	if c.xml.Header == "xproto" {
		return ""
	}

	// First check for the type in the current namespace.
	if c.xml.HasType(needle) {
		return strings.Title(c.xml.Header)
	}

	// Now check each of the imports...
	for _, imp := range c.xml.Imports {
		if imp.xml.Header != "xproto" && imp.xml.HasType(needle) {
			return strings.Title(imp.xml.Header)
		}
	}

	return ""
}

// Translate is the big daddy of them all. It takes in an XML byte slice
// and writes Go code to the 'out' buffer.
func (c *Context) Translate(xmlBytes []byte) {
	err := xml.Unmarshal(xmlBytes, c.xml)
	if err != nil {
		log.Fatal(err)
	}

	// Parse all imports
	c.xml.Imports.Eval()

	// Make sure all top level enumerations have expressions
	// (For when there are empty items.)
	c.xml.Enums.Eval()

	// It's Morphin' Time!
	c.xml.Morph(c)

	// for _, req := range c.xml.Requests { 
		// if req.Name != "CreateContext" && req.Name != "MakeCurrent" { 
			// continue 
		// } 
		// log.Println(req.Name) 
		// for _, field := range req.Fields { 
			// log.Println("\t", field.XMLName.Local, field.Type.Morph(c)) 
		// } 
	// } 
}