Files
llgo/chore/_xtool/llcppsigfetch/llcppsigfetch.go

274 lines
7.6 KiB
Go
Raw Normal View History

2024-07-25 18:46:40 +08:00
/*
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package main
import (
"fmt"
"io"
"os"
"path/filepath"
"strconv"
"strings"
2024-08-16 11:52:22 +08:00
"github.com/goplus/llgo/c"
2024-08-20 18:06:01 +08:00
"github.com/goplus/llgo/c/cjson"
"github.com/goplus/llgo/chore/_xtool/llcppsigfetch/parse"
2024-10-21 17:43:01 +08:00
"github.com/goplus/llgo/chore/_xtool/llcppsymg/args"
2024-09-24 14:43:33 +08:00
"github.com/goplus/llgo/chore/_xtool/llcppsymg/clangutils"
"github.com/goplus/llgo/chore/_xtool/llcppsymg/config"
)
2024-07-25 18:46:40 +08:00
func main() {
2024-10-21 17:43:01 +08:00
ags, remainArgs := args.ParseArgs(os.Args[1:], map[string]bool{
"--extract": true,
})
if ags.Help {
printUsage()
return
}
2024-10-21 20:51:08 +08:00
if ags.Verbose {
parse.SetDebug(parse.DbgFlagAll)
}
2024-10-21 17:43:01 +08:00
extract := false
out := false
var extractFile string
isTemp := false
isCpp := true
otherArgs := []string{}
for i := 0; i < len(remainArgs); i++ {
arg := remainArgs[i]
2024-09-29 16:18:17 +08:00
switch {
case arg == "--extract":
2024-10-21 17:43:01 +08:00
extract = true
if i+1 < len(remainArgs) && !strings.HasPrefix(remainArgs[i+1], "-") {
extractFile = remainArgs[i+1]
i++
} else {
fmt.Fprintln(os.Stderr, "Error: --extract requires a valid file argument")
2024-10-21 17:43:01 +08:00
printUsage()
os.Exit(1)
}
2024-09-29 16:18:17 +08:00
case strings.HasPrefix(arg, "-out="):
2024-10-21 17:43:01 +08:00
out = parseBoolArg(arg, "out", false)
case strings.HasPrefix(arg, "-temp="):
isTemp = parseBoolArg(arg, "temp", false)
case strings.HasPrefix(arg, "-cpp="):
isCpp = parseBoolArg(arg, "cpp", true)
default:
otherArgs = append(otherArgs, arg)
2024-09-23 15:24:03 +08:00
}
2024-09-05 12:18:37 +08:00
}
2024-10-21 17:43:01 +08:00
if extract {
2024-10-21 20:51:08 +08:00
if ags.Verbose {
fmt.Fprintln(os.Stderr, "runExtract: extractFile:", extractFile)
fmt.Fprintln(os.Stderr, "isTemp:", isTemp)
fmt.Fprintln(os.Stderr, "isCpp:", isCpp)
fmt.Fprintln(os.Stderr, "out:", out)
fmt.Fprintln(os.Stderr, "otherArgs:", otherArgs)
2024-10-21 20:51:08 +08:00
}
runExtract(extractFile, isTemp, isCpp, out, otherArgs, ags.Verbose)
2024-10-21 17:43:01 +08:00
} else {
2024-10-21 20:51:08 +08:00
if ags.Verbose {
fmt.Fprintln(os.Stderr, "runFromConfig: config file:", ags.CfgFile)
fmt.Fprintln(os.Stderr, "use stdin:", ags.UseStdin)
fmt.Fprintln(os.Stderr, "output to file:", out)
2024-10-21 20:51:08 +08:00
}
runFromConfig(ags.CfgFile, ags.UseStdin, out, ags.Verbose)
2024-10-21 17:43:01 +08:00
}
2024-09-05 12:18:37 +08:00
}
func printUsage() {
fmt.Println("Usage:")
2024-09-23 15:24:03 +08:00
fmt.Println(" llcppsigfetch [<config_file>] [-out=<bool>]")
2024-09-05 12:18:37 +08:00
fmt.Println(" OR")
2024-09-23 15:24:03 +08:00
fmt.Println(" llcppsigfetch --extract <file> [-out=<bool>] [-temp=<bool>] [-cpp=<bool>] [args...]")
2024-09-05 12:18:37 +08:00
fmt.Println("")
fmt.Println("Options:")
fmt.Println(" [<config_file>]: Path to the configuration file (use '-' for stdin)")
fmt.Println(" If not provided, uses default 'llcppg.cfg'")
2024-09-23 15:24:03 +08:00
fmt.Println(" -out=<bool>: Optional. Set to 'true' to output results to a file,")
fmt.Println(" 'false' (default) to output to stdout")
fmt.Println(" This option can be used with both modes")
2024-09-05 12:18:37 +08:00
fmt.Println("")
fmt.Println(" --extract: Extract information from a single file")
fmt.Println(" <file>: Path to the file to process, or file content if -temp=true")
fmt.Println(" -temp=<bool>: Optional. Set to 'true' if <file> contains file content,")
fmt.Println(" 'false' (default) if it's a file path")
fmt.Println(" -cpp=<bool>: Optional. Set to 'true' if the language is C++ (default: true)")
fmt.Println(" If not present, <file> is a file path")
fmt.Println(" [args]: Optional additional arguments")
fmt.Println(" Default for C++: -x c++")
fmt.Println(" Default for C: -x c")
2024-09-05 12:18:37 +08:00
fmt.Println("")
fmt.Println(" --help, -h: Show this help message")
fmt.Println("")
fmt.Println("Note: The two usage modes are mutually exclusive. Use either [<config_file>] OR --extract, not both.")
}
2024-10-21 20:51:08 +08:00
func runFromConfig(cfgFile string, useStdin bool, outputToFile bool, verbose bool) {
var data []byte
var err error
2024-09-29 16:18:17 +08:00
if useStdin {
data, err = io.ReadAll(os.Stdin)
} else {
data, err = os.ReadFile(cfgFile)
}
2024-10-21 20:51:08 +08:00
if verbose {
if useStdin {
fmt.Fprintln(os.Stderr, "runFromConfig: read from stdin")
2024-10-21 20:51:08 +08:00
} else {
fmt.Fprintln(os.Stderr, "runFromConfig: read from file", cfgFile)
2024-10-21 20:51:08 +08:00
}
}
check(err)
conf, err := config.GetConf(data)
check(err)
defer conf.Delete()
if err != nil {
fmt.Fprintln(os.Stderr, "Failed to parse config file:", cfgFile)
2024-10-21 20:51:08 +08:00
os.Exit(1)
}
2024-10-21 20:51:08 +08:00
//todo(zzy): reuse the llcppsymg's cflags parse
cflag := ParseCFlags(conf.CFlags)
files, notFounds, err := cflag.GenHeaderFilePaths(conf.Include)
check(err)
if verbose {
fmt.Fprintln(os.Stderr, "runFromConfig: header file paths", files)
2024-10-21 20:51:08 +08:00
if len(notFounds) > 0 {
fmt.Fprintln(os.Stderr, "runFromConfig: not found header files", notFounds)
2024-10-21 20:51:08 +08:00
}
}
context := parse.NewContext(conf.Cplusplus)
err = context.ProcessFiles(files)
check(err)
2024-09-23 15:24:03 +08:00
outputInfo(context, outputToFile)
}
2024-10-21 20:51:08 +08:00
func runExtract(file string, isTemp bool, isCpp bool, outToFile bool, otherArgs []string, verbose bool) {
2024-09-24 14:43:33 +08:00
cfg := &clangutils.Config{
2024-10-21 17:43:01 +08:00
File: file,
Args: otherArgs,
IsCpp: isCpp,
Temp: isTemp,
2024-09-05 12:18:37 +08:00
}
converter, err := parse.NewConverter(cfg)
check(err)
_, err = converter.Convert()
check(err)
2024-09-19 15:35:42 +08:00
result := converter.MarshalOutputASTFiles()
2024-09-05 12:18:37 +08:00
cstr := result.Print()
2024-10-21 17:43:01 +08:00
outputResult(cstr, outToFile)
2024-09-05 12:18:37 +08:00
cjson.FreeCStr(cstr)
result.Delete()
converter.Dispose()
}
func check(err error) {
if err != nil {
panic(err)
}
}
2024-09-23 15:24:03 +08:00
func outputResult(result *c.Char, outputToFile bool) {
if outputToFile {
outputFile := "llcppg.sigfetch.json"
err := os.WriteFile(outputFile, []byte(c.GoString(result)), 0644)
if err != nil {
fmt.Fprintf(os.Stderr, "Error writing to output file: %v\n", err)
os.Exit(1)
}
fmt.Fprintf(os.Stderr, "Results saved to %s\n", outputFile)
2024-09-23 15:24:03 +08:00
} else {
c.Printf(result)
}
}
2024-10-21 20:51:08 +08:00
// todo(zzy): reuse the llcppsymg's cflags parse https://github.com/goplus/llgo/pull/788
type CFlags struct {
Paths []string // Include Path
}
func ParseCFlags(cflags string) *CFlags {
parts := strings.Fields(cflags)
cf := &CFlags{}
for _, part := range parts {
if strings.HasPrefix(part, "-I") {
cf.Paths = append(cf.Paths, part[2:])
}
}
return cf
}
func (cf *CFlags) GenHeaderFilePaths(files []string) ([]string, []string, error) {
var foundPaths []string
var notFound []string
for _, file := range files {
var found bool
for _, path := range cf.Paths {
fullPath := filepath.Join(path, file)
if _, err := os.Stat(fullPath); err == nil {
foundPaths = append(foundPaths, fullPath)
found = true
break
}
}
if !found {
notFound = append(notFound, file)
}
}
2024-10-21 20:51:08 +08:00
if len(foundPaths) == 0 {
return nil, notFound, fmt.Errorf("failed to find any header files")
}
return foundPaths, notFound, nil
}
2024-09-23 15:24:03 +08:00
func outputInfo(context *parse.Context, outputToFile bool) {
2024-08-20 18:06:01 +08:00
info := context.Output()
str := info.Print()
defer cjson.FreeCStr(str)
defer info.Delete()
2024-09-23 15:24:03 +08:00
outputResult(str, outputToFile)
2024-07-25 18:46:40 +08:00
}
func parseBoolArg(arg, name string, defaultValue bool) bool {
parts := strings.SplitN(arg, "=", 2)
if len(parts) != 2 {
fmt.Fprintf(os.Stderr, "Warning: Invalid -%s= argument, defaulting to %v\n", name, defaultValue)
return defaultValue
}
value, err := strconv.ParseBool(parts[1])
if err != nil {
fmt.Fprintf(os.Stderr, "Warning: Invalid -%s= value '%s', defaulting to %v\n", name, parts[1], defaultValue)
return defaultValue
}
return value
}