/src/start/main.go

https://code.google.com/p/godag/ · Go · 545 lines · 410 code · 78 blank · 57 comment · 79 complexity · 0467ffb74789ef8129bceb9ee9f7c124 MD5 · raw file

  1. // Copyright Š 2009 bjarneh
  2. //
  3. // This program is free software: you can redistribute it and/or modify
  4. // it under the terms of the GNU General Public License as published by
  5. // the Free Software Foundation, either version 3 of the License, or
  6. // (at your option) any later version.
  7. //
  8. // This program is distributed in the hope that it will be useful,
  9. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. // GNU General Public License for more details.
  12. //
  13. // You should have received a copy of the GNU General Public License
  14. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  15. package main
  16. import (
  17. "cmplr/compiler"
  18. "cmplr/dag"
  19. "cmplr/gdmake"
  20. "fmt"
  21. "log"
  22. "os"
  23. "parse/gopt"
  24. "path/filepath"
  25. "runtime"
  26. "strings"
  27. "utilz/global"
  28. "utilz/handy"
  29. "utilz/say"
  30. "utilz/timer"
  31. "utilz/walker"
  32. )
  33. // option parser object (struct)
  34. var getopt *gopt.GetOpt
  35. // list of files to compile
  36. var files []string
  37. // libraries other than $GOROOT/pkg/PLATFORM
  38. var includes []string = make([]string, 0)
  39. // source root
  40. var srcdir string = "src"
  41. // keys for the bool options
  42. var bools = []string{
  43. "-help",
  44. "-clean",
  45. "-static",
  46. "-version",
  47. "-sort",
  48. "-print",
  49. "-dryrun",
  50. "-test",
  51. "-list",
  52. "-verbose",
  53. "-fmt",
  54. "-all",
  55. "-quiet",
  56. "-tab",
  57. "-external",
  58. "-update-external",
  59. // add missing test options + alias
  60. "-test.short",
  61. "-test.v",
  62. "-strip",
  63. }
  64. // keys for the string options
  65. // note: -I is handled seperately
  66. var strs = []string{
  67. "-dot",
  68. "-tabwidth",
  69. "-rewrite",
  70. "-output",
  71. "-bench",
  72. "-match",
  73. "-test-bin",
  74. "-lib",
  75. "-main",
  76. "-backend",
  77. "-gdmk",
  78. "-mkcomplete",
  79. // add missing test options + alias
  80. "-test.bench",
  81. "-test.benchtime",
  82. "-test.cpu",
  83. "-test.cpuprofile",
  84. "-test.memprofile",
  85. "-test.memprofilerate",
  86. "-test.timeout",
  87. "-test.parallel",
  88. }
  89. func init() {
  90. // initialize option parser
  91. getopt = gopt.New()
  92. // add all options (bool/string)
  93. getopt.BoolOption("-h -help --help help")
  94. getopt.BoolOption("-c -clean --clean clean")
  95. getopt.BoolOption("-S -static --static")
  96. getopt.BoolOption("-v -version --version version")
  97. getopt.BoolOption("-s -sort --sort sort")
  98. getopt.BoolOption("-p -print --print print")
  99. getopt.BoolOption("-d -dryrun --dryrun dryrun")
  100. getopt.BoolOption("-t -test --test test")
  101. getopt.BoolOption("-l -list --list list")
  102. getopt.BoolOption("-q -quiet --quiet")
  103. getopt.BoolOption("-V -verbose --verbose")
  104. getopt.BoolOption("-f -fmt --fmt fmt")
  105. getopt.BoolOption("-T -tab --tab")
  106. getopt.BoolOption("-a -all --all")
  107. getopt.BoolOption("-y -strip --strip strip")
  108. getopt.BoolOption("-e -external --external")
  109. getopt.BoolOption("-u -updatex --updatex "+
  110. "-update-external --update-external")
  111. getopt.StringOption("-I -I=")
  112. getopt.StringOption("-mkcomplete")
  113. getopt.StringOptionFancy("-D --dot")
  114. getopt.StringOptionFancy("-L --lib")
  115. getopt.StringOptionFancy("-g --gdmk")
  116. getopt.StringOptionFancy("-w --tabwidth")
  117. getopt.StringOptionFancy("-r --rewrite")
  118. getopt.StringOptionFancy("-o --output")
  119. getopt.StringOptionFancy("-M --main")
  120. getopt.StringOptionFancy("-b --bench")
  121. getopt.StringOptionFancy("-m --match")
  122. getopt.StringOptionFancy("--test-bin")
  123. getopt.StringOptionFancy("-B --backend")
  124. // new test options and aliases
  125. getopt.BoolOption("-test.short --test.short")
  126. getopt.BoolOption("-test.v --test.v")
  127. getopt.StringOptionFancy("--test.bench")
  128. getopt.StringOptionFancy("--test.benchtime")
  129. getopt.StringOptionFancy("--test.cpu")
  130. getopt.StringOptionFancy("--test.cpuprofile")
  131. getopt.StringOptionFancy("--test.memprofile")
  132. getopt.StringOptionFancy("--test.memprofilerate")
  133. getopt.StringOptionFancy("--test.timeout")
  134. getopt.StringOptionFancy("--test.parallel")
  135. // override IncludeFile to make walker pick up only .go files
  136. walker.IncludeFile = noTestFilesFilter
  137. // override IncludeDir to make walker ignore 'hidden' directories
  138. walker.IncludeDir = func(s string) bool {
  139. _, dirname := filepath.Split(s)
  140. return dirname[0] != '.'
  141. }
  142. for _, bkey := range bools {
  143. global.SetBool(bkey, false)
  144. }
  145. for _, skey := range strs {
  146. global.SetString(skey, "")
  147. }
  148. // Testing on Windows requires .exe ending
  149. goos := handy.GOOS()
  150. if goos == "windows" {
  151. global.SetString("-test-bin", "gdtest.exe")
  152. } else {
  153. global.SetString("-test-bin", "gdtest")
  154. }
  155. global.SetString("-backend", runtime.Compiler)
  156. global.SetString("-I", "")
  157. }
  158. // utility func for walker: *.go unless start = '_' || end = _test.go
  159. func noTestFilesFilter(s string) bool {
  160. return strings.HasSuffix(s, ".go") &&
  161. !strings.HasSuffix(s, "_test.go") &&
  162. !strings.HasPrefix(filepath.Base(s), "_")
  163. }
  164. // utility func for walker: *.go unless start = '_'
  165. func allGoFilesFilter(s string) bool {
  166. return strings.HasSuffix(s, ".go") &&
  167. !strings.HasPrefix(filepath.Base(s), "_")
  168. }
  169. func reportTime() {
  170. timer.Stop("everything")
  171. delta, _ := timer.Delta("everything")
  172. say.Printf("time used: %s\n", timer.Nano2Time(delta))
  173. }
  174. func main() {
  175. var (
  176. ok, up2date bool
  177. e error
  178. argv, args []string
  179. config [4]string
  180. )
  181. timer.Start("everything")
  182. defer reportTime()
  183. // possible config locations
  184. config[0] = filepath.Join(os.Getenv("XDG_CONFIG_HOME"), "godag", "gdrc")
  185. config[1] = filepath.Join(os.Getenv("HOME"), ".config", "godag", "gdrc")
  186. config[2] = filepath.Join(os.Getenv("HOME"), ".gdrc")
  187. config[3] = filepath.Join(os.Getenv("PWD"), ".gdrc")
  188. for _, conf := range config {
  189. argv, ok = handy.ConfigToArgv(conf)
  190. if ok {
  191. args = parseArgv(argv)
  192. if len(args) > 0 {
  193. log.Print("[WARNING] non-option arguments in config file\n")
  194. }
  195. }
  196. }
  197. // small gorun version if single go-file is given
  198. if len(os.Args) > 1 && strings.HasSuffix(os.Args[1], ".go") && handy.IsFile(os.Args[1]) {
  199. say.Mute() // be silent unless error here
  200. single, name := dag.ParseSingle(os.Args[1])
  201. compiler.InitBackend()
  202. compiler.CreateArgv(single)
  203. up2date = compiler.Compile(single)
  204. if handy.GOOS() == "windows" {
  205. name = name + ".exe"
  206. }
  207. compiler.ForkLink(name, single, nil, up2date)
  208. args = os.Args[1:]
  209. args[0] = name
  210. handy.StdExecve(args, true)
  211. os.Exit(0)
  212. }
  213. // command line arguments overrides/appends config
  214. args = parseArgv(os.Args[1:])
  215. mkcomplete := global.GetString("-mkcomplete")
  216. if mkcomplete != "" {
  217. targets := dag.GetMakeTargets(mkcomplete)
  218. for _, t := range targets {
  219. fmt.Println(t)
  220. }
  221. os.Exit(0)
  222. }
  223. if len(args) > 0 {
  224. if len(args) > 1 {
  225. log.Print("[WARNING] len(input directories) > 1\n")
  226. }
  227. srcdir = args[0]
  228. if srcdir == "." {
  229. srcdir, e = os.Getwd()
  230. if e != nil {
  231. log.Fatal("[ERROR] can't find working directory\n")
  232. }
  233. }
  234. }
  235. // expand variables in includes
  236. for i := 0; i < len(includes); i++ {
  237. includes[i] = os.ExpandEnv(includes[i])
  238. }
  239. // expand variables in -lib
  240. global.SetString("-lib", os.ExpandEnv(global.GetString("-lib")))
  241. // expand variables in -output
  242. global.SetString("-output", os.ExpandEnv(global.GetString("-output")))
  243. if global.GetBool("-list") {
  244. printListing()
  245. os.Exit(0)
  246. }
  247. if global.GetBool("-help") {
  248. printHelp()
  249. os.Exit(0)
  250. }
  251. if global.GetBool("-version") {
  252. printVersion()
  253. os.Exit(0)
  254. }
  255. if len(args) == 0 {
  256. // give nice feedback if missing input dir
  257. if !handy.IsDir("src") {
  258. fmt.Printf("usage: gd [OPTIONS] src-directory\n")
  259. os.Exit(1)
  260. }
  261. }
  262. if global.GetBool("-quiet") {
  263. say.Mute()
  264. }
  265. handy.DirOrExit(srcdir)
  266. files = walker.PathWalk(filepath.Clean(srcdir))
  267. // gofmt on all files gathered
  268. if global.GetBool("-fmt") {
  269. compiler.FormatFiles(files)
  270. os.Exit(0)
  271. }
  272. // parse the source code, look for dependencies
  273. dgrph := dag.New()
  274. dgrph.Parse(srcdir, files)
  275. // print collected dependency info
  276. if global.GetBool("-print") {
  277. dgrph.PrintInfo()
  278. os.Exit(0)
  279. }
  280. // draw graphviz dot graph
  281. if global.GetString("-dot") != "" {
  282. dgrph.MakeDotGraph(global.GetString("-dot"))
  283. os.Exit(0)
  284. }
  285. // build all external dependencies
  286. if global.GetBool("-external") {
  287. // update external dependencies
  288. if global.GetBool("-update-external") {
  289. dgrph.External(true)
  290. } else {
  291. dgrph.External(false)
  292. }
  293. os.Exit(0)
  294. }
  295. // sort graph based on dependencies
  296. dgrph.GraphBuilder()
  297. sorted := dgrph.Topsort()
  298. // clean only what we possibly could have generated…
  299. if global.GetBool("-clean") {
  300. compiler.DeleteObjects(srcdir, sorted)
  301. os.Exit(0)
  302. }
  303. // print packages sorted
  304. if global.GetBool("-sort") {
  305. for i := 0; i < len(sorted); i++ {
  306. fmt.Printf("%s\n", sorted[i].Name)
  307. }
  308. os.Exit(0)
  309. }
  310. // compile argv
  311. compiler.Init(srcdir, includes)
  312. if global.GetString("-lib") != "" {
  313. compiler.CreateLibArgv(sorted)
  314. } else {
  315. compiler.CreateArgv(sorted)
  316. }
  317. // gdmk
  318. if global.GetString("-gdmk") != "" {
  319. gdmake.Make(global.GetString("-gdmk"), sorted, dgrph.Alien().Slice())
  320. os.Exit(0)
  321. }
  322. // compile; up2date == true => 0 packages modified
  323. if global.GetBool("-dryrun") {
  324. compiler.Dryrun(sorted)
  325. } else {
  326. up2date = compiler.Compile(sorted) // updated parallel
  327. }
  328. // test
  329. if global.GetBool("-test") {
  330. os.Setenv("SRCROOT", srcdir)
  331. testMain, testDir, testLib := dgrph.MakeMainTest(srcdir)
  332. if global.GetString("-lib") != "" {
  333. compiler.CreateLibArgv(testMain)
  334. } else {
  335. compiler.CreateArgv(testMain)
  336. }
  337. if !global.GetBool("-dryrun") {
  338. compiler.Compile(testMain)
  339. }
  340. switch global.GetString("-backend") {
  341. case "gc", "express":
  342. compiler.ForkLink(global.GetString("-test-bin"), testMain, nil, false)
  343. case "gccgo", "gcc":
  344. compiler.ForkLink(global.GetString("-test-bin"), testMain, sorted, false)
  345. default:
  346. log.Fatalf("[ERROR] '%s' unknown back-end\n", global.GetString("-backend"))
  347. }
  348. compiler.DeletePackages(testMain)
  349. handy.Delete(testDir, false)
  350. if testLib != "" {
  351. handy.Delete(testLib, false)
  352. }
  353. testArgv := compiler.CreateTestArgv()
  354. if global.GetBool("-dryrun") {
  355. testArgv[0] = filepath.Base(testArgv[0])
  356. say.Printf("%s\n", strings.Join(testArgv, " "))
  357. } else {
  358. say.Printf("testing : ")
  359. if global.GetBool("-verbose") || global.GetBool("-test.v") {
  360. say.Printf("\n")
  361. }
  362. ok = handy.StdExecve(testArgv, false)
  363. handy.Delete(global.GetString("-test-bin"), false)
  364. if !ok {
  365. os.Exit(1)
  366. }
  367. }
  368. // if packages contain both test-files and regular files
  369. // test-files should not be part of the objects, i.e. init
  370. // functions in test-packages can cause unexpected behaviour
  371. if compiler.ReCompile(sorted) {
  372. say.Printf("recompile: --tests\n")
  373. compiler.Compile(sorted)
  374. }
  375. }
  376. // link if ! up2date
  377. if global.GetString("-output") != "" {
  378. compiler.ForkLink(global.GetString("-output"), sorted, nil, up2date)
  379. } else if global.GetBool("-all") {
  380. compiler.ForkLinkAll(sorted, up2date)
  381. }
  382. }
  383. func parseArgv(argv []string) (args []string) {
  384. args = getopt.Parse(argv)
  385. for _, bkey := range bools {
  386. if getopt.IsSet(bkey) {
  387. global.SetBool(bkey, true)
  388. }
  389. }
  390. for _, skey := range strs {
  391. if getopt.IsSet(skey) {
  392. global.SetString(skey, getopt.Get(skey))
  393. }
  394. }
  395. if getopt.IsSet("-test") || getopt.IsSet("-fmt") || getopt.IsSet("-clean") {
  396. // override IncludeFile to make walker pick _test.go files
  397. walker.IncludeFile = allGoFilesFilter
  398. }
  399. if getopt.IsSet("-gdmk") {
  400. global.SetString("-lib", "_obj")
  401. // gdmk does not support testing
  402. walker.IncludeFile = noTestFilesFilter
  403. }
  404. if getopt.IsSet("-I") {
  405. includes = append(includes, getopt.GetMultiple("-I")...)
  406. }
  407. getopt.Reset()
  408. return args
  409. }
  410. func printHelp() {
  411. var helpMSG string = `
  412. Godag is a compiler front-end for golang,
  413. its main purpose is to help build projects
  414. which are pure Go-code without Makefiles.
  415. Hopefully it simplifies testing as well.
  416. usage: gd [OPTIONS] src-directory
  417. options:
  418. -h --help print this message and quit
  419. -v --version print version and quit
  420. -l --list list option values and quit
  421. -p --print print package info collected
  422. -s --sort print legal compile order
  423. -o --output link main package -> output
  424. -S --static statically link binary
  425. -y --strip strip symbols from executable
  426. -g --gdmk create a go makefile for project
  427. -d --dryrun print what gd would do (stdout)
  428. -c --clean delete generated object code
  429. -q --quiet silent, print only errors
  430. -L --lib write objects to other dir (!src)
  431. -M --main regex to select main package
  432. -a --all link main pkgs to bin/nameOfMainDir
  433. -D --dot create a graphviz dot file
  434. -I import package directories
  435. -t --test run all unit-tests
  436. -m --match regex to select unit-tests
  437. -b --bench regex to select benchmarks
  438. -V --verbose verbose unit-test and go install
  439. --test-bin name of test-binary (default: gdtest)
  440. --test.* any valid gotest option
  441. -f --fmt run gofmt on src and exit
  442. -r --rewrite pass rewrite rule to gofmt
  443. -T --tab pass -tabs=true to gofmt
  444. -w --tabwidth pass -tabwidth to gofmt (default: 4)
  445. -e --external go install all external dependencies
  446. -u --updatex go install -u all external dependencies
  447. -B --backend [gc,gccgo,express] (default: gc)
  448. `
  449. fmt.Println(helpMSG)
  450. }
  451. func printVersion() {
  452. fmt.Println("godag 0.4.0 (release-branch.go1.1)")
  453. }
  454. func printListing() {
  455. fmt.Println("\n Listing of options and their content:\n")
  456. defer fmt.Println("")
  457. for i := 0; i < len(bools); i++ {
  458. fmt.Printf(" %-20s => %v\n", bools[i], global.GetBool(bools[i]))
  459. }
  460. for i := 0; i < len(strs); i++ {
  461. fmt.Printf(" %-20s => %v\n", strs[i], global.GetString(strs[i]))
  462. }
  463. fmt.Printf(" %-20s => %v\n", "-lib", includes)
  464. }