aboutsummaryrefslogtreecommitdiff
path: root/cljcc-compiler/src/cljcc/cljcc.clj
blob: c067b75459616c9264eeb8a47298e0887d423c7c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
(ns cljcc.cljcc
  (:require
   [clojure.tools.cli :refer [parse-opts]]
   [clojure.string :as string]
   [cljcc.util :refer [exit]]
   [cljcc.driver :as d])
  (:gen-class))

(defn run
  "Compiles source input using specified compiler options.

  Parameters:
    source - Source C file.
    options - Map of compiler configuration options.

  Returns generated AST for specified stage."
  [source & {:keys [config] :or {config {}}}]
  (let [default-config {:target {:os :linux}}]))

(set! *warn-on-reflection* true)

(defn usage [options-summary]
  (->>
   ["Usage: ./cljcc path/to/file.c [options]"
    ""
    "Options:"
    options-summary]
   (string/join \newline)))

(def cli-options
  [[nil "--lex" "Runs lexer. Does not emit any files."]
   [nil "--parse" "Runs parser. Does not emit any files."]
   [nil "--validate" "Runs semantic analyzer. Does not emit any files."]
   [nil "--tacky" "Runs tacky generation. Does not emit any files."]
   [nil "--codegen" "Runs compiler. Does not emit any files."]
   ["-c" nil "Generate object file."
    :id :generate-object-file]
   ["-h" "--help"]])

(defn validate-args [args]
  (let [{:keys [options arguments summary]} (parse-opts args cli-options)]
    (cond
      (:help options) {:exit-message (usage summary) :ok? true}
      (= 1 (count arguments)) {:file-path (first arguments)
                               :options options}
      :else {:exit-message (usage summary)})))

(defn -main
  "Main entrypoint for cljcc compiler."
  [& args]
  (let [{:keys [file-path exit-message ok? options]} (validate-args args)
        libs (filterv (fn [v] (and
                               (string? v)
                               (re-matches #"-l.+" v)))
                      args)]
    (if exit-message
      (exit (if ok? 0 1) exit-message)
      (try
        (d/run file-path (assoc options :libs libs))
        (exit 0 "Successfully executed.")
        (catch Exception e
          (exit 1 (ex-message e) e))))))

(comment

  ())