2024-08-06 01:03:23 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2024-08-07 06:51:30 +00:00
|
|
|
"encoding/json"
|
2024-08-06 01:03:23 +00:00
|
|
|
"flag"
|
2024-08-06 02:29:12 +00:00
|
|
|
"io/ioutil"
|
2024-08-06 01:03:23 +00:00
|
|
|
"log"
|
2024-08-06 02:29:12 +00:00
|
|
|
"path/filepath"
|
2024-08-06 01:03:23 +00:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
clang := flag.String("clang", "clang", "Custom path to clang")
|
|
|
|
cflags := flag.String("cflags", `-DQT_WIDGETS_LIB -I/usr/include/x86_64-linux-gnu/qt5/QtWidgets -I/usr/include/x86_64-linux-gnu/qt5 -I/usr/include/x86_64-linux-gnu/qt5/QtCore -DQT_GUI_LIB -I/usr/include/x86_64-linux-gnu/qt5/QtGui -DQT_CORE_LIB`, "Cflags to pass to clang (e.g. `pkg-config --cflags Qt5Widgets`)")
|
2024-08-10 01:32:43 +00:00
|
|
|
outDir := flag.String("outdir", "../../qt", "Output directory for generated gen_** files")
|
|
|
|
dumpIL := flag.String("dumpil", "", "(Optional) File to dump intermediate IL JSON")
|
2024-08-06 01:03:23 +00:00
|
|
|
|
|
|
|
flag.Parse()
|
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
includeFiles := []string{
|
|
|
|
"/usr/include/x86_64-linux-gnu/qt5/QtWidgets/qwidget.h",
|
|
|
|
"/usr/include/x86_64-linux-gnu/qt5/QtWidgets/qabstractbutton.h",
|
|
|
|
"/usr/include/x86_64-linux-gnu/qt5/QtWidgets/qpushbutton.h",
|
2024-08-06 01:03:23 +00:00
|
|
|
}
|
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
for _, inputHeader := range includeFiles {
|
|
|
|
|
|
|
|
// If we have a cached clang AST, use that instead
|
|
|
|
cacheFile := filepath.Join("cachedir", strings.Replace(inputHeader, `/`, `__`, -1)+".json")
|
|
|
|
astJson, err := ioutil.ReadFile(cacheFile)
|
|
|
|
var astInner []interface{} = nil
|
2024-08-07 06:51:30 +00:00
|
|
|
if err != nil {
|
2024-08-14 05:43:54 +00:00
|
|
|
|
|
|
|
// Nonexistent cache file, regenerate from clang
|
|
|
|
log.Printf("No AST cache for file %q, running clang...", filepath.Base(inputHeader))
|
|
|
|
|
|
|
|
// Parse the file
|
|
|
|
astInner, err = clangExec(ctx, *clang, inputHeader, strings.Fields(*cflags))
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write to cache
|
|
|
|
jb, err := json.MarshalIndent(astInner, "", "\t")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = ioutil.WriteFile(cacheFile, jb, 0644)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
log.Printf("Reused cache AST for file %q", filepath.Base(inputHeader))
|
|
|
|
|
|
|
|
// Json decode
|
|
|
|
err = json.Unmarshal(astJson, &astInner)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2024-08-07 06:51:30 +00:00
|
|
|
}
|
2024-08-10 01:32:43 +00:00
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
// Convert it to our intermediate format
|
|
|
|
parsed, err := parseHeader(astInner)
|
2024-08-10 01:32:43 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-07 06:51:30 +00:00
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
if *dumpIL != "" {
|
|
|
|
jb, err := json.MarshalIndent(parsed, "", "\t")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = ioutil.WriteFile(*dumpIL, jb, 0644)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AST transforms on our IL
|
|
|
|
astTransformOptional(parsed)
|
|
|
|
astTransformOverloads(parsed)
|
2024-08-06 01:03:23 +00:00
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
// Emit 3 code files from the intermediate format
|
|
|
|
outputName := filepath.Join(*outDir, "gen_"+strings.TrimSuffix(filepath.Base(inputHeader), `.h`))
|
|
|
|
|
|
|
|
goSrc, err := emitGo(parsed, filepath.Base(inputHeader))
|
2024-08-07 06:51:30 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-10 01:32:43 +00:00
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
err = ioutil.WriteFile(outputName+".go", []byte(goSrc), 0644)
|
2024-08-10 01:32:43 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-07 06:51:30 +00:00
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
bindingCppSrc, err := emitBindingCpp(parsed, filepath.Base(inputHeader))
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-06 02:29:12 +00:00
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
err = ioutil.WriteFile(outputName+".cpp", []byte(bindingCppSrc), 0644)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-06 01:03:23 +00:00
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
bindingHSrc, err := emitBindingHeader(parsed, filepath.Base(inputHeader))
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-06 02:29:12 +00:00
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
err = ioutil.WriteFile(outputName+".h", []byte(bindingHSrc), 0644)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-06 02:29:12 +00:00
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
// Done
|
2024-08-06 02:29:12 +00:00
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
log.Printf("Processing %q completed", inputHeader)
|
2024-08-06 02:29:12 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2024-08-14 05:43:54 +00:00
|
|
|
log.Printf("Processing %d file(s) completed", len(includeFiles))
|
2024-08-06 01:03:23 +00:00
|
|
|
}
|