diff options
Diffstat (limited to '')
-rw-r--r-- | src/lang.clj | 5 | ||||
-rw-r--r-- | src/lang/compiler.clj | 32 | ||||
-rw-r--r-- | test2.lang | 4 |
3 files changed, 37 insertions, 4 deletions
diff --git a/src/lang.clj b/src/lang.clj index 97465a7a8..eb26b9cc8 100644 --- a/src/lang.clj +++ b/src/lang.clj @@ -27,7 +27,6 @@ ;; TODO: Add tuples. ;; TODO: Add let. ;; TODO: Add pattern-matching. - ;; TODO: Add tagged values. ;; TODO: Do tail-call optimization. ;; TODO: Add macros. ;; TODO: Add type-level computations. @@ -39,4 +38,8 @@ ;; TODO: Allow using other modules. ;; TODO: ;; TODO: + + ;; jar cvf test2.jar test2 test2.class + ;; java -cp "test2.jar" test2 + ;; jar cvf test2.jar test2 test2.class && java -cp "test2.jar" test2 ) diff --git a/src/lang/compiler.clj b/src/lang/compiler.clj index d2b24eb02..dbc088668 100644 --- a/src/lang/compiler.clj +++ b/src/lang/compiler.clj @@ -141,6 +141,13 @@ (doseq [[class field] ?fields] (doto (.visitField =class Opcodes/ACC_PUBLIC field (->type-signature class) nil nil) (.visitEnd))) + (doto (.visitMethod =class Opcodes/ACC_PUBLIC "<init>" "()V" nil nil) + (.visitCode) + (.visitVarInsn Opcodes/ALOAD 0) + (.visitMethodInsn Opcodes/INVOKESPECIAL "java/lang/Object" "<init>" "()V") + (.visitInsn Opcodes/RETURN) + (.visitMaxs 0 0) + (.visitEnd)) (.visitEnd =class) (let [parent-dir (->class *name*)] (.mkdirs (java.io.File. parent-dir)) @@ -161,6 +168,23 @@ (with-open [stream (java.io.BufferedOutputStream. (java.io.FileOutputStream. (str parent-dir "/" ?name ".class")))] (.write stream (.toByteArray =interface)))))) +(let [+tagged+ "test2/Tagged"] + (defcompiler ^:private compile-tagged + [::&parser/tagged ?tag ?value] + (do (prn 'compile-tagged ?tag ?value) + (doto *writer* + (.visitTypeInsn Opcodes/NEW +tagged+) + (.visitInsn Opcodes/DUP) + (.visitMethodInsn Opcodes/INVOKESPECIAL +tagged+ "<init>" "()V") + (.visitInsn Opcodes/DUP) + (.visitLdcInsn ?tag) + (.visitFieldInsn Opcodes/PUTFIELD +tagged+ "tag" "Ljava/lang/String;") + (.visitInsn Opcodes/DUP)) + (compile-form (assoc *state* :form ?value)) + (doto *writer* + (.visitFieldInsn Opcodes/PUTFIELD +tagged+ "value" "Ljava/lang/Object;")) + ))) + (let [+compilers+ [compile-boolean compile-string compile-ident @@ -172,7 +196,8 @@ compile-def compile-module compile-defclass - compile-definterface]] + compile-definterface + compile-tagged]] (defn ^:private compile-form [state] (prn 'compile-form/state state) (some #(% state) +compilers+))) @@ -194,7 +219,10 @@ ;; (.visitInsn Opcodes/RETURN) ;; (.visitMaxs 0 0) ;; (.visitEnd)) - (doall (map #(compile-form (assoc state :form %)) inputs)) + (doseq [input inputs] + (when (not (compile-form (assoc state :form input))) + (assert false input))) + ;; (doall (map #(compile-form (assoc state :form %)) inputs)) (when-let [constants (seq (for [input inputs :let [payload (match input [::&parser/def [::&parser/ident ?name] ?body] diff --git a/test2.lang b/test2.lang index 8e878641f..0dd69acb9 100644 --- a/test2.lang +++ b/test2.lang @@ -10,13 +10,15 @@ fields (: out java.io.PrintStream)) -(defclass Tagged [[java.lang.String tag] [java.lang.Object data]]) +(defclass Tagged [[java.lang.String tag] [java.lang.Object value]]) (definterface Function (: apply (-> [java.lang.Object] java.lang.Object))) (def const "IDENTITY") +(def sample (#Tag "value")) + ## (ann id #type (All [x] (-> [x] x))) (def (id x) x) |