2024-08-06 13:03:23 +12:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2024-08-07 18:51:30 +12:00
|
|
|
"encoding/json"
|
2024-08-06 13:03:23 +12:00
|
|
|
"flag"
|
2024-08-06 14:29:12 +12:00
|
|
|
"io/ioutil"
|
2024-08-06 13:03:23 +12:00
|
|
|
"log"
|
2024-08-19 19:14:47 +12:00
|
|
|
"os"
|
2024-08-06 14:29:12 +12:00
|
|
|
"path/filepath"
|
2024-10-07 19:25:27 +13:00
|
|
|
"runtime"
|
2024-08-06 13:03:23 +12:00
|
|
|
"strings"
|
2024-10-07 19:25:27 +13:00
|
|
|
"sync"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
ClangSubprocessCount = 3
|
2024-08-06 13:03:23 +12:00
|
|
|
)
|
|
|
|
|
2024-08-26 22:45:11 +12:00
|
|
|
func cacheFilePath(inputHeader string) string {
|
|
|
|
return filepath.Join("cachedir", strings.Replace(inputHeader, `/`, `__`, -1)+".json")
|
|
|
|
}
|
|
|
|
|
2024-08-06 13:03:23 +12:00
|
|
|
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 13:32:43 +12:00
|
|
|
outDir := flag.String("outdir", "../../qt", "Output directory for generated gen_** files")
|
2024-08-06 13:03:23 +12:00
|
|
|
|
|
|
|
flag.Parse()
|
|
|
|
|
2024-08-19 19:14:47 +12:00
|
|
|
var includeFiles []string
|
|
|
|
|
|
|
|
for _, srcDir := range []string{
|
|
|
|
"/usr/include/x86_64-linux-gnu/qt5/QtCore",
|
|
|
|
"/usr/include/x86_64-linux-gnu/qt5/QtGui",
|
|
|
|
"/usr/include/x86_64-linux-gnu/qt5/QtWidgets",
|
|
|
|
} {
|
|
|
|
content, err := os.ReadDir(srcDir)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, includeFile := range content {
|
|
|
|
if includeFile.IsDir() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if !strings.HasSuffix(includeFile.Name(), `.h`) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
fullPath := filepath.Join(srcDir, includeFile.Name())
|
|
|
|
if !AllowHeader(fullPath) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
includeFiles = append(includeFiles, fullPath)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("Found %d header files to process.", len(includeFiles))
|
|
|
|
|
|
|
|
{
|
|
|
|
log.Printf("Cleaning up output directory %q...", *outDir)
|
|
|
|
|
|
|
|
existing, err := os.ReadDir(*outDir)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
cleaned := 0
|
|
|
|
for _, e := range existing {
|
|
|
|
if e.IsDir() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if !strings.HasPrefix(e.Name(), `gen_`) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// One of ours, clean up
|
|
|
|
err := os.Remove(filepath.Join(*outDir, e.Name()))
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("WARNING: Failed to remove existing file %q", e.Name())
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
cleaned++
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("Removed %d file(s).", cleaned)
|
2024-08-06 13:03:23 +12:00
|
|
|
}
|
|
|
|
|
2024-08-26 22:45:11 +12:00
|
|
|
var processHeaders []*CppParsedHeader
|
2024-09-16 19:03:45 +12:00
|
|
|
atr := astTransformRedundant{
|
|
|
|
preserve: make(map[string]*CppEnum),
|
|
|
|
}
|
2024-08-26 22:45:11 +12:00
|
|
|
|
2024-09-19 19:39:05 +12:00
|
|
|
InsertTypedefs()
|
|
|
|
|
2024-10-07 19:25:27 +13:00
|
|
|
//
|
|
|
|
// PASS 0 (Fill clang cache)
|
|
|
|
//
|
|
|
|
|
|
|
|
var clangChan = make(chan string, 0)
|
|
|
|
var clangWg sync.WaitGroup
|
|
|
|
|
|
|
|
for i := 0; i < ClangSubprocessCount; i++ {
|
|
|
|
clangWg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer clangWg.Done()
|
|
|
|
log.Printf("Clang worker: starting")
|
|
|
|
|
|
|
|
for {
|
|
|
|
inputHeader, ok := <-clangChan
|
|
|
|
if !ok {
|
|
|
|
return // Done
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("Clang worker got message for file %q", inputHeader)
|
|
|
|
|
|
|
|
// Parse the file
|
|
|
|
// This seems to intermittently fail, so allow retrying
|
|
|
|
astInner := mustClangExec(ctx, *clang, inputHeader, strings.Fields(*cflags))
|
|
|
|
|
|
|
|
// Write to cache
|
|
|
|
jb, err := json.MarshalIndent(astInner, "", "\t")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = ioutil.WriteFile(cacheFilePath(inputHeader), jb, 0644)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
astInner = nil
|
|
|
|
jb = nil
|
|
|
|
runtime.GC()
|
|
|
|
|
|
|
|
}
|
|
|
|
log.Printf("Clang worker: exiting")
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2024-08-14 17:43:54 +12:00
|
|
|
for _, inputHeader := range includeFiles {
|
|
|
|
|
2024-10-07 19:25:27 +13:00
|
|
|
// Check if there is a matching cache hit
|
2024-08-26 22:45:11 +12:00
|
|
|
cacheFile := cacheFilePath(inputHeader)
|
2024-10-07 19:25:27 +13:00
|
|
|
|
|
|
|
if _, err := os.Stat(cacheFile); err != nil && os.IsNotExist(err) {
|
2024-08-14 17:43:54 +12:00
|
|
|
|
|
|
|
// Nonexistent cache file, regenerate from clang
|
|
|
|
log.Printf("No AST cache for file %q, running clang...", filepath.Base(inputHeader))
|
2024-10-07 19:25:27 +13:00
|
|
|
clangChan <- inputHeader
|
|
|
|
}
|
|
|
|
}
|
2024-08-14 17:43:54 +12:00
|
|
|
|
2024-10-07 19:25:27 +13:00
|
|
|
// Done with all clang workers
|
|
|
|
close(clangChan)
|
|
|
|
clangWg.Wait()
|
2024-09-11 17:44:36 +12:00
|
|
|
|
2024-10-07 19:25:27 +13:00
|
|
|
// The cache should now be fully populated.
|
2024-08-14 17:43:54 +12:00
|
|
|
|
2024-10-07 19:25:27 +13:00
|
|
|
//
|
|
|
|
// PASS 1 (clang2il)
|
|
|
|
//
|
2024-08-14 17:43:54 +12:00
|
|
|
|
2024-10-07 19:25:27 +13:00
|
|
|
for _, inputHeader := range includeFiles {
|
2024-08-14 17:43:54 +12:00
|
|
|
|
2024-10-07 19:25:27 +13:00
|
|
|
cacheFile := cacheFilePath(inputHeader)
|
2024-08-14 17:43:54 +12:00
|
|
|
|
2024-10-07 19:25:27 +13:00
|
|
|
astJson, err := ioutil.ReadFile(cacheFile)
|
|
|
|
if err != nil {
|
|
|
|
panic("Expected cache to be created for " + inputHeader + ", but got error " + err.Error())
|
|
|
|
}
|
2024-08-14 17:43:54 +12:00
|
|
|
|
2024-10-07 19:25:27 +13:00
|
|
|
// Json decode
|
|
|
|
var astInner []interface{} = nil
|
|
|
|
err = json.Unmarshal(astJson, &astInner)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
2024-08-07 18:51:30 +12:00
|
|
|
}
|
2024-08-10 13:32:43 +12:00
|
|
|
|
2024-08-14 17:43:54 +12:00
|
|
|
// Convert it to our intermediate format
|
2024-08-27 19:12:08 +12:00
|
|
|
parsed, err := parseHeader(astInner, "")
|
2024-08-10 13:32:43 +12:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-07 18:51:30 +12:00
|
|
|
|
2024-08-26 22:45:11 +12:00
|
|
|
parsed.Filename = inputHeader // Stash
|
|
|
|
|
2024-08-15 19:51:49 +12:00
|
|
|
// AST transforms on our IL
|
2024-08-25 19:08:28 +12:00
|
|
|
astTransformChildClasses(parsed) // must be first
|
2024-08-15 19:51:49 +12:00
|
|
|
astTransformOptional(parsed)
|
|
|
|
astTransformOverloads(parsed)
|
2024-09-16 19:03:45 +12:00
|
|
|
atr.Process(parsed)
|
2024-08-15 19:51:49 +12:00
|
|
|
|
2024-08-26 22:48:37 +12:00
|
|
|
// Update global state tracker (AFTER astTransformChildClasses)
|
|
|
|
// Currently, this is only used for inner classes
|
|
|
|
for _, c := range parsed.Classes {
|
|
|
|
KnownClassnames[c.ClassName] = struct{}{}
|
|
|
|
}
|
2024-08-26 22:49:33 +12:00
|
|
|
for _, td := range parsed.Typedefs {
|
|
|
|
KnownTypedefs[td.Alias] = td // copy
|
|
|
|
}
|
2024-09-04 18:54:10 +12:00
|
|
|
for _, en := range parsed.Enums {
|
|
|
|
KnownEnums[en.EnumName] = en // copy
|
|
|
|
}
|
2024-08-26 22:49:33 +12:00
|
|
|
|
2024-08-26 22:45:11 +12:00
|
|
|
processHeaders = append(processHeaders, parsed)
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// PASS 2
|
|
|
|
//
|
|
|
|
|
|
|
|
for _, parsed := range processHeaders {
|
|
|
|
|
|
|
|
log.Printf("Processing %q...", parsed.Filename)
|
2024-08-26 22:49:33 +12:00
|
|
|
|
|
|
|
// More AST transforms on our IL
|
|
|
|
astTransformTypedefs(parsed)
|
2024-08-27 18:44:10 +12:00
|
|
|
astTransformBlocklist(parsed) // Must happen after typedef transformation
|
2024-08-26 22:49:33 +12:00
|
|
|
|
2024-08-15 19:51:49 +12:00
|
|
|
{
|
|
|
|
// Save the IL file for debug inspection
|
2024-08-14 17:43:54 +12:00
|
|
|
jb, err := json.MarshalIndent(parsed, "", "\t")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2024-08-26 22:45:11 +12:00
|
|
|
err = ioutil.WriteFile(cacheFilePath(parsed.Filename)+".ours.json", jb, 0644)
|
2024-08-14 17:43:54 +12:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-19 19:11:36 +12:00
|
|
|
// Breakout if there is nothing bindable
|
|
|
|
if parsed.Empty() {
|
|
|
|
log.Printf("Nothing in this header was bindable.")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2024-08-14 17:43:54 +12:00
|
|
|
// Emit 3 code files from the intermediate format
|
2024-08-26 22:45:11 +12:00
|
|
|
outputName := filepath.Join(*outDir, "gen_"+strings.TrimSuffix(filepath.Base(parsed.Filename), `.h`))
|
2024-08-14 17:43:54 +12:00
|
|
|
|
2024-08-26 22:45:11 +12:00
|
|
|
goSrc, err := emitGo(parsed, filepath.Base(parsed.Filename))
|
2024-08-07 18:51:30 +12:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-10 13:32:43 +12:00
|
|
|
|
2024-08-14 17:43:54 +12:00
|
|
|
err = ioutil.WriteFile(outputName+".go", []byte(goSrc), 0644)
|
2024-08-10 13:32:43 +12:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-07 18:51:30 +12:00
|
|
|
|
2024-08-26 22:45:11 +12:00
|
|
|
bindingCppSrc, err := emitBindingCpp(parsed, filepath.Base(parsed.Filename))
|
2024-08-14 17:43:54 +12:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-06 14:29:12 +12:00
|
|
|
|
2024-08-14 17:43:54 +12:00
|
|
|
err = ioutil.WriteFile(outputName+".cpp", []byte(bindingCppSrc), 0644)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-06 13:03:23 +12:00
|
|
|
|
2024-08-26 22:45:11 +12:00
|
|
|
bindingHSrc, err := emitBindingHeader(parsed, filepath.Base(parsed.Filename))
|
2024-08-14 17:43:54 +12:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-06 14:29:12 +12:00
|
|
|
|
2024-08-14 17:43:54 +12:00
|
|
|
err = ioutil.WriteFile(outputName+".h", []byte(bindingHSrc), 0644)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-08-06 14:29:12 +12:00
|
|
|
|
2024-08-14 17:43:54 +12:00
|
|
|
// Done
|
2024-08-06 14:29:12 +12:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2024-08-14 17:43:54 +12:00
|
|
|
log.Printf("Processing %d file(s) completed", len(includeFiles))
|
2024-08-06 13:03:23 +12:00
|
|
|
}
|