package main import( "fmt" "regexp" "runtime" "path" "path/filepath" "strings" "crypto/sha256" "encoding/hex" ) type ParserResult struct { InputList []string InputFiles []string ObjectFiles []string OutputFilename string CompileArgs []string LinkArgs []string IsVerbose bool IsDependencyOnly bool IsPreprocessOnly bool IsAssembleOnly bool IsAssembly bool IsCompileOnly bool IsEmitLLVM bool } type FlagInfo struct { arity int handler func(string, []string) } func parse(argList []string) ParserResult { var pr = ParserResult{} pr.InputList = argList var argsExactMatches = map[string]FlagInfo{ "-o": FlagInfo{1, pr.outputFileCallback}, "-c": FlagInfo{0, pr.compileOnlyCallback}, "-E": FlagInfo{0, pr.preprocessOnlyCallback}, "-S": FlagInfo{0, pr.assembleOnlyCallback}, "--verbose": FlagInfo{0, pr.verboseFlagCallback}, "--param": FlagInfo{1, pr.defaultBinaryCallback}, "-aux-info": FlagInfo{1, pr.defaultBinaryCallback}, "--version": FlagInfo{0, pr.compileOnlyCallback}, "-v": FlagInfo{0, pr.compileOnlyCallback}, "-w": FlagInfo{0, pr.compileOnlyCallback}, "-W": FlagInfo{0, pr.compileOnlyCallback}, "-emit-llvm": FlagInfo{0, pr.emitLLVMCallback}, "-pipe": FlagInfo{0, pr.compileUnaryCallback}, "-undef": FlagInfo{0, pr.compileUnaryCallback}, "-nostdinc": FlagInfo{0, pr.compileUnaryCallback}, "-nostdinc++": FlagInfo{0, pr.compileUnaryCallback}, "-Qunused-arguments": FlagInfo{0, pr.compileUnaryCallback}, "-no-integrated-as": FlagInfo{0, pr.compileUnaryCallback}, "-integrated-as": FlagInfo{0, pr.compileUnaryCallback}, "-pthread": FlagInfo{0, pr.compileUnaryCallback}, "-nostdlibinc": FlagInfo{0, pr.compileUnaryCallback}, "-mno-omit-leaf-frame-pointer": FlagInfo{0, pr.compileUnaryCallback}, "-maes": FlagInfo{0, pr.compileUnaryCallback}, "-mno-aes": FlagInfo{0, pr.compileUnaryCallback}, "-mavx": FlagInfo{0, pr.compileUnaryCallback}, "-mno-avx": FlagInfo{0, pr.compileUnaryCallback}, "-mcmodel=kernel": FlagInfo{0, pr.compileUnaryCallback}, "-mno-red-zone": FlagInfo{0, pr.compileUnaryCallback}, "-mmmx": FlagInfo{0, pr.compileUnaryCallback}, "-mno-mmx": FlagInfo{0, pr.compileUnaryCallback}, "-msse": FlagInfo{0, pr.compileUnaryCallback}, "-mno-sse2": FlagInfo{0, pr.compileUnaryCallback}, "-msse2": FlagInfo{0, pr.compileUnaryCallback}, "-mno-sse3": FlagInfo{0, pr.compileUnaryCallback}, "-msse3": FlagInfo{0, pr.compileUnaryCallback}, "-mno-sse": FlagInfo{0, pr.compileUnaryCallback}, "-msoft-float": FlagInfo{0, pr.compileUnaryCallback}, "-m3dnow": FlagInfo{0, pr.compileUnaryCallback}, "-mno-3dnow": FlagInfo{0, pr.compileUnaryCallback}, "-m32": FlagInfo{0, pr.compileUnaryCallback}, "-m64": FlagInfo{0, pr.compileUnaryCallback}, "-mstackrealign": FlagInfo{0, pr.compileUnaryCallback}, "-A": FlagInfo{1, pr.compileBinaryCallback}, "-D": FlagInfo{1, pr.compileBinaryCallback}, "-U": FlagInfo{1, pr.compileBinaryCallback}, "-M" : FlagInfo{0, pr.dependencyOnlyCallback}, "-MM": FlagInfo{0, pr.dependencyOnlyCallback}, "-MF": FlagInfo{1, pr.dependencyBinaryCallback}, "-MG": FlagInfo{0, pr.dependencyOnlyCallback}, "-MP": FlagInfo{0, pr.dependencyOnlyCallback}, "-MT": FlagInfo{1, pr.dependencyBinaryCallback}, "-MQ": FlagInfo{1, pr.dependencyBinaryCallback}, "-MD": FlagInfo{0, pr.dependencyOnlyCallback}, "-MMD": FlagInfo{0, pr.dependencyOnlyCallback}, "-I": FlagInfo{1, pr.compileBinaryCallback}, "-idirafter": FlagInfo{1, pr.compileBinaryCallback}, "-include": FlagInfo{1, pr.compileBinaryCallback}, "-imacros": FlagInfo{1, pr.compileBinaryCallback}, "-iprefix": FlagInfo{1, pr.compileBinaryCallback}, "-iwithprefix": FlagInfo{1, pr.compileBinaryCallback}, "-iwithprefixbefore": FlagInfo{1, pr.compileBinaryCallback}, "-isystem": FlagInfo{1, pr.compileBinaryCallback}, "-isysroot": FlagInfo{1, pr.compileBinaryCallback}, "-iquote": FlagInfo{1, pr.compileBinaryCallback}, "-imultilib": FlagInfo{1, pr.compileBinaryCallback}, "-ansi": FlagInfo{0, pr.compileUnaryCallback}, "-pedantic": FlagInfo{0, pr.compileUnaryCallback}, "-x": FlagInfo{1, pr.compileBinaryCallback}, "-g": FlagInfo{0, pr.compileUnaryCallback}, "-g0": FlagInfo{0, pr.compileUnaryCallback}, "-ggdb": FlagInfo{0, pr.compileUnaryCallback}, "-ggdb3": FlagInfo{0, pr.compileUnaryCallback}, "-gdwarf-2": FlagInfo{0, pr.compileUnaryCallback}, "-gdwarf-3": FlagInfo{0, pr.compileUnaryCallback}, "-gline-tables-only": FlagInfo{0, pr.compileUnaryCallback}, "-p": FlagInfo{0, pr.compileUnaryCallback}, "-pg": FlagInfo{0, pr.compileUnaryCallback}, "-O": FlagInfo{0, pr.compileUnaryCallback}, "-O0": FlagInfo{0, pr.compileUnaryCallback}, "-O1": FlagInfo{0, pr.compileUnaryCallback}, "-O2": FlagInfo{0, pr.compileUnaryCallback}, "-O3": FlagInfo{0, pr.compileUnaryCallback}, "-Os": FlagInfo{0, pr.compileUnaryCallback}, "-Ofast": FlagInfo{0, pr.compileUnaryCallback}, "-Og": FlagInfo{0, pr.compileUnaryCallback}, "-Xclang": FlagInfo{1, pr.compileBinaryCallback}, "-Xpreprocessor": FlagInfo{1, pr.defaultBinaryCallback}, "-Xassembler": FlagInfo{1, pr.defaultBinaryCallback}, "-Xlinker": FlagInfo{1, pr.defaultBinaryCallback}, "-l": FlagInfo{1, pr.linkBinaryCallback}, "-L": FlagInfo{1, pr.linkBinaryCallback}, "-T": FlagInfo{1, pr.linkBinaryCallback}, "-u": FlagInfo{1, pr.linkBinaryCallback}, "-e": FlagInfo{1, pr.linkBinaryCallback}, "-rpath": FlagInfo{1, pr.linkBinaryCallback}, "-shared": FlagInfo{0, pr.linkUnaryCallback}, "-static": FlagInfo{0, pr.linkUnaryCallback}, "-pie": FlagInfo{0, pr.linkUnaryCallback}, "-nostdlib": FlagInfo{0, pr.linkUnaryCallback}, "-nodefaultlibs": FlagInfo{0, pr.linkUnaryCallback}, "-rdynamic": FlagInfo{0, pr.linkUnaryCallback}, "-dynamiclib": FlagInfo{0, pr.linkUnaryCallback}, "-current_version": FlagInfo{1, pr.linkBinaryCallback}, "-compatibility_version": FlagInfo{1, pr.linkBinaryCallback}, "-print-multi-directory": FlagInfo{0, pr.compileUnaryCallback}, "-print-multi-lib": FlagInfo{0, pr.compileUnaryCallback}, "-print-libgcc-file-name": FlagInfo{0, pr.compileUnaryCallback}, "-fprofile-arcs": FlagInfo{0, pr.compileLinkUnaryCallback}, "-coverage": FlagInfo{0, pr.compileLinkUnaryCallback}, "--coverage": FlagInfo{0, pr.compileLinkUnaryCallback}, "-Wl,-dead_strip": FlagInfo{0, pr.darwinWarningLinkUnaryCallback}, } var argPatterns = map[string]FlagInfo{ `^.+\.(c|cc|cpp|C|cxx|i|s|S|bc)$`: FlagInfo{0, pr.inputFileCallback}, `^.+\.([fF](|[0-9][0-9]|or|OR|pp|PP))$`: FlagInfo{0, pr.inputFileCallback}, `^.+\.(o|lo|So|so|po|a|dylib)$`: FlagInfo{0, pr.objectFileCallback}, `^.+\.dylib(\.\d)+$`: FlagInfo{0, pr.objectFileCallback}, `^.+\.(So|so)(\.\d)+$`: FlagInfo{0, pr.objectFileCallback}, `^-(l|L).+$`: FlagInfo{0, pr.linkUnaryCallback}, `^-I.+$`: FlagInfo{0, pr.compileUnaryCallback}, `^-D.+$`: FlagInfo{0, pr.compileUnaryCallback}, `^-U.+$`: FlagInfo{0, pr.compileUnaryCallback}, `^-Wl,.+$`: FlagInfo{0, pr.linkUnaryCallback}, `^-W.*$`: FlagInfo{0, pr.compileUnaryCallback}, `^-f.+$`: FlagInfo{0, pr.compileUnaryCallback}, `^-rtlib=.+$`: FlagInfo{0, pr.linkUnaryCallback}, `^-std=.+$`: FlagInfo{0, pr.compileUnaryCallback}, `^-stdlib=.+$`: FlagInfo{0, pr.compileLinkUnaryCallback}, `^-mtune=.+$`: FlagInfo{0, pr.compileUnaryCallback}, `^--sysroot=.+$`: FlagInfo{0, pr.compileUnaryCallback}, `^-print-prog-name=.*$`: FlagInfo{0, pr.compileUnaryCallback}, `^-print-file-name=.*$`: FlagInfo{0, pr.compileUnaryCallback}, } for len(argList) > 0 && !(pr.IsAssembly || pr.IsAssembleOnly || pr.IsPreprocessOnly) { var elem = argList[0] // Try to match the flag exactly if fi, ok := argsExactMatches[elem]; ok { fi.handler(elem, argList[1:1+fi.arity]) argList = argList[1+fi.arity:] // Else try to match a pattern } else { var listShift = 0 for pattern, fi := range argPatterns { var regExp = regexp.MustCompile(pattern) if regExp.MatchString(elem) { fi.handler(elem, argList[1:1+fi.arity]) listShift = fi.arity break } } argList = argList[1+listShift:] } } return pr } // Return the object and bc filenames that correspond to the i-th source file func getArtifactNames(pr ParserResult, srcFileIndex int, hidden bool) (objBase string, bcBase string) { if len(pr.InputFiles) == 1 && pr.IsCompileOnly && len(pr.OutputFilename) > 0 { objBase = pr.OutputFilename dir, baseName := path.Split(objBase) bcBaseName := fmt.Sprintf(".%s.bc", baseName) bcBase = path.Join(dir, bcBaseName) } else { srcFile := pr.InputFiles[srcFileIndex] var dir, baseNameWithExt = path.Split(srcFile) var baseName = strings.TrimSuffix(baseNameWithExt, filepath.Ext(baseNameWithExt)) bcBase = fmt.Sprintf(".%s.o.bc", baseName) if hidden { objBase = path.Join(dir, fmt.Sprintf(".%s.o", baseName)) } else { objBase = path.Join(dir, fmt.Sprintf("%s.o", baseName)) } } return } // Return a hash for the absolute object path func getHashedPath(path string) string { inputBytes := []byte(path) hasher := sha256.New() hasher.Write(inputBytes) hash := hex.EncodeToString(hasher.Sum(nil)) return hash } func (pr *ParserResult) inputFileCallback(flag string, _ []string) { var regExp = regexp.MustCompile(`\\.(s|S)$`) pr.InputFiles = append(pr.InputFiles, flag) if regExp.MatchString(flag) { pr.IsAssembly = true } } func (pr *ParserResult) outputFileCallback(_ string, args []string) { pr.OutputFilename = args[0] } func (pr *ParserResult) objectFileCallback(flag string, _ []string) { pr.ObjectFiles = append(pr.ObjectFiles, flag) } func (pr *ParserResult) preprocessOnlyCallback(_ string, _ []string) { pr.IsPreprocessOnly = true } func (pr *ParserResult) dependencyOnlyCallback(flag string, _ []string) { pr.IsDependencyOnly = true pr.CompileArgs = append(pr.CompileArgs, flag) } func (pr *ParserResult) assembleOnlyCallback(_ string, _ []string) { pr.IsAssembleOnly = true } func (pr *ParserResult) verboseFlagCallback(_ string, _ []string) { pr.IsVerbose = true } func (pr *ParserResult) compileOnlyCallback(_ string, _ []string) { pr.IsCompileOnly = true } func (pr *ParserResult) emitLLVMCallback(_ string, _ []string) { pr.IsCompileOnly = true pr.IsEmitLLVM = true } func (pr *ParserResult) linkUnaryCallback(flag string, _ []string) { pr.LinkArgs = append(pr.LinkArgs, flag) } func (pr *ParserResult) compileUnaryCallback(flag string, _ []string) { pr.CompileArgs = append(pr.CompileArgs, flag) } func (pr *ParserResult) darwinWarningLinkUnaryCallback(flag string, _ []string) { if runtime.GOOS == "darwin" { fmt.Println("The flag", flag, "cannot be used with this tool.") } else { pr.LinkArgs = append(pr.LinkArgs, flag) } } func (_ *ParserResult) defaultBinaryCallback(_ string, _ []string) { // Do nothing } func (pr *ParserResult) dependencyBinaryCallback(flag string, args []string) { pr.CompileArgs = append(pr.CompileArgs, flag, args[0]) pr.IsDependencyOnly = true } func (pr *ParserResult) compileBinaryCallback(flag string, args []string) { pr.CompileArgs = append(pr.CompileArgs, flag, args[0]) } func (pr *ParserResult) linkBinaryCallback(flag string, args []string) { pr.LinkArgs = append(pr.LinkArgs, flag, args[0]) } func (pr *ParserResult) compileLinkUnaryCallback(flag string, _ []string) { pr.LinkArgs = append(pr.LinkArgs, flag) pr.CompileArgs = append(pr.CompileArgs, flag) }