diff options
author | Eduardo Julian | 2022-02-06 22:07:51 -0400 |
---|---|---|
committer | Eduardo Julian | 2022-02-06 22:07:51 -0400 |
commit | 7065801a9ad1724c6a82e9803c218b2981bc59b3 (patch) | |
tree | 696b90821666a9477a2e7a953d9016c95a61ad1e | |
parent | 290de8ebcb7edc92877f2ccc333171214e5eae23 (diff) |
Fixes for JVM interop.
13 files changed, 531 insertions, 193 deletions
diff --git a/documentation/bookmark/back_end/c++.md b/documentation/bookmark/back_end/c++.md index 10ba1fcbb..186993319 100644 --- a/documentation/bookmark/back_end/c++.md +++ b/documentation/bookmark/back_end/c++.md @@ -8,3 +8,7 @@ 0. [Type-and-resource Safety in Modern C++ - Bjarne Stroustrup - CppCon 2021](https://www.youtube.com/watch?v=l3rvjWfBzZI) 0. [Exceptional C++ - Victor Ciura - CppCon 2021](https://www.youtube.com/watch?v=SjlfhyZn2yA) +# Undefined behavior + +0. [Back To Basics: Undefined Behavior - Ansel Sermersheim & Barbara Geller - CppCon 2021](https://www.youtube.com/watch?v=NpL9YnxnOqM) + diff --git a/stdlib/source/library/lux/ffi.jvm.lux b/stdlib/source/library/lux/ffi.jvm.lux index f13818a4a..59d2b2374 100644 --- a/stdlib/source/library/lux/ffi.jvm.lux +++ b/stdlib/source/library/lux/ffi.jvm.lux @@ -1135,8 +1135,9 @@ (~ (code.text (product.left (parser.read_class super_class)))) (~ (code.text name)) [(~+ (list#each (|>> ..signature code.text) type_vars))] - (~ (code.local_symbol self_name)) + ("jvm object cast" (~ (code.local_symbol self_name))) (~+ (|> args + (list#each (|>> ~ "jvm object cast" `)) (list.zipped/2 (list#each product.right arguments)) (list#each ..decorate_input)))))))))] (` ("override" diff --git a/stdlib/source/library/lux/tool/compiler/language/lux/analysis/scope.lux b/stdlib/source/library/lux/tool/compiler/language/lux/analysis/scope.lux index 838c2c362..d3187458a 100644 --- a/stdlib/source/library/lux/tool/compiler/language/lux/analysis/scope.lux +++ b/stdlib/source/library/lux/tool/compiler/language/lux/analysis/scope.lux @@ -76,7 +76,7 @@ _ (..captured name scope))) -(def: .public (find name) +(def: .public (variable name) (-> Text (Operation (Maybe [Type Variable]))) (extension.lifted (function (_ state) diff --git a/stdlib/source/library/lux/tool/compiler/language/lux/phase/analysis/reference.lux b/stdlib/source/library/lux/tool/compiler/language/lux/phase/analysis/reference.lux index 8fdf78aa8..5bedbd7bf 100644 --- a/stdlib/source/library/lux/tool/compiler/language/lux/phase/analysis/reference.lux +++ b/stdlib/source/library/lux/tool/compiler/language/lux/phase/analysis/reference.lux @@ -84,7 +84,7 @@ (def: (variable var_name) (-> Text (Operation (Maybe Analysis))) (do [! ///.monad] - [?var (/scope.find var_name)] + [?var (/scope.variable var_name)] (case ?var {.#Some [actualT ref]} (do ! diff --git a/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/analysis/jvm.lux b/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/analysis/jvm.lux index 21980f491..fa1a73e1e 100644 --- a/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/analysis/jvm.lux +++ b/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/analysis/jvm.lux @@ -15,31 +15,45 @@ ["<[0]>" code {"+" Parser}] ["<[0]>" text]]] [data + [binary {"+" Binary}] ["[0]" product] ["[0]" text ("[1]#[0]" equivalence) ["%" format {"+" format}]] [collection ["[0]" list ("[1]#[0]" mix monad monoid)] ["[0]" array] - ["[0]" dictionary {"+" Dictionary}]]] + ["[0]" dictionary {"+" Dictionary}] + ["[0]" sequence]] + ["[0]" format "_" + ["[1]" binary]]] [macro ["[0]" template]] [math [number - ["n" nat]]] + ["n" nat] + ["[0]" i32]]] [target ["[0]" jvm "_" ["[0]!" reflection] + ["_" bytecode {"+" Bytecode} ("[1]#[0]" monad)] + ["[0]" modifier {"+" Modifier} ("[1]#[0]" monoid)] + ["[0]" attribute] + ["[0]" field] + ["[0]" version] + ["[0]" method] + ["[0]" class] + ["[0]" constant + ["[0]" pool {"+" Resource}]] [encoding - [name {"+" External}]] + ["[0]" name {"+" External}]] ["[1]" type {"+" Type Argument Typed} ("[1]#[0]" equivalence) ["[0]" category {"+" Void Value' Value Return' Return Primitive Object Class Array Var Parameter Method}] ["[0]" box] ["[0]" reflection] ["[0]" descriptor] ["[0]" signature] - ["[1]_[0]" parser] - ["[1]_[0]" alias {"+" Aliasing}] + ["[0]" parser] + ["[0]" alias {"+" Aliasing}] ["[0]T" lux {"+" Mapping}]]]] ["[0]" type ["[0]" check {"+" Check} ("[1]#[0]" monad)]]]] @@ -47,20 +61,25 @@ ["[1][0]" lux {"+" custom}] ["/[1]" // ["[1][0]" bundle] - ["//[1]" /// "_" - ["[1][0]" synthesis] - ["[1][0]" analysis {"+" Analysis Operation Phase Handler Bundle} - ["[1]/[0]" complex] - ["[1]/[0]" pattern] - ["[0]A" type] - ["[0]A" inference] - ["[0]" scope]] - [/// - ["[0]" phase ("[1]#[0]" monad)] - [meta - [archive {"+" Archive} - [module - [descriptor {"+" Module}]]]]]]]]) + ["/[1]" // "_" + [generation + [jvm + ["[0]" runtime]]] + ["/[1]" // "_" + ["[0]" generation] + ["[0]" directive] + ["[1][0]" analysis {"+" Analysis Operation Phase Handler Bundle} + ["[0]" complex] + ["[0]" pattern] + ["[0]" inference] + ["[0]A" type] + ["[0]" scope]] + [/// + ["[0]" phase ("[1]#[0]" monad)] + [meta + [archive {"+" Archive} + [module + [descriptor {"+" Module}]]]]]]]]]) (import: java/lang/ClassLoader) @@ -159,8 +178,7 @@ not)))) (def: reflection - (All (_ category) - (-> (Type (<| Return' Value' category)) Text)) + (All (_ category) (-> (Type (<| Return' Value' category)) Text)) (|>> jvm.reflection reflection.reflection)) (def: signature (|>> jvm.signature signature.signature)) @@ -169,12 +187,6 @@ External "java.lang.Object") -(def: inheritance_relationship_type_name "_jvm_inheritance") -(def: .public (inheritance_relationship_type class super_class super_interfaces) - (-> .Type .Type (List .Type) .Type) - {.#Primitive ..inheritance_relationship_type_name - (list& class super_class super_interfaces)}) - ... TODO: Get rid of this template block and use the definition in ... lux/ffi.jvm.lux ASAP (template [<name> <class>] @@ -399,7 +411,7 @@ (function (_ parameterT) (do phase.monad [parameterJT (jvm_type parameterT)] - (case (jvm_parser.parameter? parameterJT) + (case (parser.parameter? parameterJT) {.#Some parameterJT} (in parameterJT) @@ -420,7 +432,7 @@ [objectJ (jvm_type objectT)] (|> objectJ ..signature - (<text>.result jvm_parser.array) + (<text>.result parser.array) phase.lifted))) (def: (primitive_array_length_handler primitive_type) @@ -482,7 +494,7 @@ (analyse archive lengthC)) expectedT (///.lifted meta.expected_type) expectedJT (jvm_array_type expectedT) - elementJT (case (jvm_parser.array? expectedJT) + elementJT (case (parser.array? expectedJT) {.#Some elementJT} (in elementJT) @@ -952,20 +964,16 @@ _ (phase.assertion ..primitives_are_not_objects [target_name] (not (dictionary.key? ..boxes target_name))) target_class (phase.lifted (reflection!.load class_loader target_name)) - _ (if (text#= ..inheritance_relationship_type_name source_name) - (in []) - (do ! - [source_class (phase.lifted (reflection!.load class_loader source_name))] - (phase.assertion ..cannot_cast [fromT toT fromC] - (java/lang/Class::isAssignableFrom source_class target_class))))] + _ (do ! + [source_class (phase.lifted (reflection!.load class_loader source_name))] + (phase.assertion ..cannot_cast [fromT toT fromC] + (java/lang/Class::isAssignableFrom source_class target_class)))] (loop [[current_name currentT] [source_name fromT]] (if (text#= target_name current_name) (in true) (do ! [candidate_parents (: (Operation (List [[Text .Type] Bit])) - (if (text#= ..inheritance_relationship_type_name current_name) - (inheritance_candidate_parents class_loader currentT target_class toT fromC) - (class_candidate_parents class_loader current_name currentT target_name target_class)))] + (class_candidate_parents class_loader current_name currentT target_name target_class))] (case (|> candidate_parents (list.only product.right) (list#each product.left)) @@ -1131,7 +1139,7 @@ (list#mix (function (_ [expectedJC actualJC] prev) (and prev (jvm#= expectedJC (: (Type Value) - (case (jvm_parser.var? actualJC) + (case (parser.var? actualJC) {.#Some name} (|> aliasing (dictionary.value name) @@ -1160,7 +1168,7 @@ (n.= (list.size inputsJT) (list.size parameters)) (list.every? (function (_ [expectedJC actualJC]) (jvm#= expectedJC (: (Type Value) - (case (jvm_parser.var? actualJC) + (case (parser.var? actualJC) {.#Some name} (|> aliasing (dictionary.value name) @@ -1327,8 +1335,8 @@ (def: (aliasing expected actual) (-> (List (Type Var)) (List (Type Var)) Aliasing) - (|> (list.zipped/2 (list#each jvm_parser.name actual) - (list#each jvm_parser.name expected)) + (|> (list.zipped/2 (list#each parser.name actual) + (list#each parser.name expected)) (dictionary.of_list text.hash))) (def: (method_candidate class_loader actual_class_tvars class_name actual_method_tvars method_name method_style inputsJT) @@ -1398,10 +1406,10 @@ (Parser (Type <category>)) (<text>.then <parser> <code>.text))] - [var Var jvm_parser.var] - [class Class jvm_parser.class] - [type Value jvm_parser.value] - [return Return jvm_parser.return] + [var Var parser.var] + [class Class parser.class] + [type Value parser.value] + [return Return parser.return] ) (def: input @@ -1429,7 +1437,7 @@ [methodT deprecated? exceptionsT] (..method_candidate class_loader class_tvars class method_tvars method {#Static} argsT) _ (phase.assertion ..deprecated_method [class method methodT] (not deprecated?)) - [outputT argsA] (inferenceA.general archive analyse methodT (list#each product.right argsTC)) + [outputT argsA] (inference.general archive analyse methodT (list#each product.right argsTC)) outputJT (check_return outputT)] (in {/////analysis.#Extension extension_name (list& (/////analysis.text (..signature (jvm.class class (list)))) (/////analysis.text method) @@ -1447,7 +1455,7 @@ [methodT deprecated? exceptionsT] (..method_candidate class_loader class_tvars class method_tvars method {#Virtual} argsT) _ (phase.assertion ..deprecated_method [class method methodT] (not deprecated?)) - [outputT allA] (inferenceA.general archive analyse methodT (list& objectC (list#each product.right argsTC))) + [outputT allA] (inference.general archive analyse methodT (list& objectC (list#each product.right argsTC))) .let [[objectA argsA] (case allA {.#Item objectA argsA} [objectA argsA] @@ -1472,11 +1480,18 @@ [methodT deprecated? exceptionsT] (..method_candidate class_loader class_tvars class method_tvars method {#Special} argsT) _ (phase.assertion ..deprecated_method [class method methodT] (not deprecated?)) - [outputT argsA] (inferenceA.general archive analyse methodT (list& objectC (list#each product.right argsTC))) + [outputT allA] (inference.general archive analyse methodT (list& objectC (list#each product.right argsTC))) + .let [[objectA argsA] (case allA + {.#Item objectA argsA} + [objectA argsA] + + _ + (undefined))] outputJT (check_return outputT)] (in {/////analysis.#Extension extension_name (list& (/////analysis.text (..signature (jvm.class class (list)))) (/////analysis.text method) (/////analysis.text (..signature outputJT)) + objectA (decorate_inputs argsT argsA))})))])) (def: (invoke::interface class_loader) @@ -1493,7 +1508,7 @@ [methodT deprecated? exceptionsT] (..method_candidate class_loader class_tvars class_name method_tvars method {#Interface} argsT) _ (phase.assertion ..deprecated_method [class_name method methodT] (not deprecated?)) - [outputT allA] (inferenceA.general archive analyse methodT (list& objectC (list#each product.right argsTC))) + [outputT allA] (inference.general archive analyse methodT (list& objectC (list#each product.right argsTC))) .let [[objectA argsA] (case allA {.#Item objectA argsA} [objectA argsA] @@ -1519,7 +1534,7 @@ [methodT deprecated? exceptionsT] (..constructor_candidate class_loader class_tvars class method_tvars argsT) _ (phase.assertion ..deprecated_method [class ..constructor_method methodT] (not deprecated?)) - [outputT argsA] (inferenceA.general archive analyse methodT (list#each product.right argsTC))] + [outputT argsA] (inference.general archive analyse methodT (list#each product.right argsTC))] (in {/////analysis.#Extension extension_name (list& (/////analysis.text (..signature (jvm.class class (list)))) (decorate_inputs argsT argsA))})))])) @@ -2030,7 +2045,7 @@ supers (List (Type Class))]) (exception.report ["Name" (%.text name)] - ["Available" (exception.listing (|>> jvm_parser.read_class product.left) supers)])) + ["Available" (exception.listing (|>> parser.read_class product.left) supers)])) (exception: .public (mismatched_super_parameters [name Text expected Nat @@ -2042,9 +2057,9 @@ (def: (override_mapping mapping supers parent_type) (-> Mapping (List (Type Class)) (Type Class) (Operation (List [Text .Type]))) - (let [[parent_name parent_parameters] (jvm_parser.read_class parent_type)] + (let [[parent_name parent_parameters] (parser.read_class parent_type)] (case (list.one (function (_ super) - (let [[super_name super_parameters] (jvm_parser.read_class super)] + (let [[super_name super_parameters] (parser.read_class super)] (if (text#= parent_name super_name) {.#Some super_parameters} {.#None}))) @@ -2055,7 +2070,7 @@ (if (n.= expected_count actual_count) (do [! phase.monad] [parent_parameters (|> parent_parameters - (monad.each maybe.monad jvm_parser.var?) + (monad.each maybe.monad parser.var?) try.of_maybe phase.lifted)] (|> super_parameters @@ -2075,7 +2090,7 @@ (in [var exT]))) vars)] (in (list#mix (function (_ [varJ varT] mapping) - (dictionary.has (jvm_parser.name varJ) varT mapping)) + (dictionary.has (parser.name varJ) varT mapping)) mapping pairings)))) @@ -2099,7 +2114,7 @@ 2 {/////analysis.#Case (/////analysis.unit) [[/////analysis.#when - {/////analysis/pattern.#Bind 2} + {pattern.#Bind 2} /////analysis.#then bodyA] @@ -2108,11 +2123,11 @@ _ {/////analysis.#Case (/////analysis.unit) [[/////analysis.#when - {/////analysis/pattern.#Complex - {/////analysis/complex.#Tuple + {pattern.#Complex + {complex.#Tuple (|> arity list.indices - (list#each (|>> (n.+ 2) {/////analysis/pattern.#Bind})))}} + (list#each (|>> (n.+ 2) {pattern.#Bind})))}} /////analysis.#then bodyA] @@ -2166,10 +2181,6 @@ (..hidden_method_body (list.size arguments) bodyA)} )))))) -(type: .public (Method_Definition a) - (Variant - {#Overriden_Method (Overriden_Method a)})) - (def: .public parameter_types (-> (List (Type Var)) (Check (List [(Type Var) .Type]))) (monad.each check.monad @@ -2207,7 +2218,7 @@ (def: (super_aliasing class_loader class) (-> java/lang/ClassLoader (Type Class) (Operation Aliasing)) (do phase.monad - [.let [[name actual_parameters] (jvm_parser.read_class class)] + [.let [[name actual_parameters] (parser.read_class class)] jvm_class (phase.lifted (reflection!.load class_loader name)) .let [expected_parameters (|> (java/lang/Class::getTypeParameters jvm_class) (array.list {.#None}) @@ -2217,13 +2228,13 @@ (list.size actual_parameters)))] (in (|> (list.zipped/2 expected_parameters actual_parameters) (list#mix (function (_ [expected actual] mapping) - (case (jvm_parser.var? actual) + (case (parser.var? actual) {.#Some actual} (dictionary.has actual expected mapping) {.#None} mapping)) - jvm_alias.fresh))))) + alias.fresh))))) (def: (anonymous_class_name module id) (-> Module Nat Text) @@ -2246,7 +2257,7 @@ (list#each product.right arguments) return exceptions]) - (jvm_alias.method aliasing) + (alias.method aliasing) [parent_type method_name])))) methods) .let [missing_abstract_methods (mismatched_methods overriden_methods required_abstract_methods) @@ -2257,8 +2268,237 @@ (list.empty? invalid_overriden_methods))] (in []))) -(def: (class::anonymous class_loader) - (-> java/lang/ClassLoader Handler) +(type: Declaration + [Text (List (Type Var))]) + +(type: Constant + [Text (List Annotation) (Type Value) Code]) + +(type: Variable + [Text (Modifier field.Field) (Modifier field.Field) (List Annotation) (Type Value)]) + +(type: Field + (Variant + {#Constant Constant} + {#Variable Variable})) + +(type: (Method_Definition a) + (Variant + {#Constructor (..Constructor a)} + {#Virtual_Method (..Virtual_Method a)} + {#Static_Method (..Static_Method a)} + {#Overriden_Method (..Overriden_Method a)} + {#Abstract_Method (..Abstract_Method a)})) + +(def: class_name + (|>> parser.read_class product.left name.internal)) + +(def: (mock_class [name parameters] super interfaces fields methods modifier) + (-> Declaration (Type Class) (List (Type Class)) + (List (Resource field.Field)) (List (Resource method.Method)) (Modifier class.Class) + (Try [External Binary])) + (let [signature (signature.inheritance (list#each jvm.signature parameters) + (jvm.signature super) + (list#each jvm.signature interfaces))] + (try#each (|>> (format.result class.writer) + [name]) + (class.class version.v6_0 + ($_ modifier#composite + class.public + modifier) + (name.internal name) + {.#Some signature} + (..class_name super) + (list#each ..class_name interfaces) + fields + methods + sequence.empty)))) + +(def: constant::modifier + (Modifier field.Field) + ($_ modifier#composite + field.public + field.static + field.final + )) + +(def: (field_definition field) + (-> Field (Resource field.Field)) + (case field + ... TODO: Handle annotations. + {#Constant [name annotations type value]} + (case value + (^template [<tag> <type> <constant>] + [[_ {<tag> value}] + (do pool.monad + [constant (`` (|> value (~~ (template.spliced <constant>)))) + attribute (attribute.constant constant)] + (field.field ..constant::modifier name <type> true (sequence.sequence attribute)))]) + ([.#Bit jvm.boolean [(case> #0 +0 #1 +1) .i64 i32.i32 constant.integer pool.integer]] + [.#Int jvm.byte [.i64 i32.i32 constant.integer pool.integer]] + [.#Int jvm.short [.i64 i32.i32 constant.integer pool.integer]] + [.#Int jvm.int [.i64 i32.i32 constant.integer pool.integer]] + [.#Int jvm.long [constant.long pool.long]] + [.#Frac jvm.float [ffi.double_to_float constant.float pool.float]] + [.#Frac jvm.double [constant.double pool.double]] + [.#Nat jvm.char [.i64 i32.i32 constant.integer pool.integer]] + [.#Text (jvm.class "java.lang.String" (list)) [pool.string]] + ) + + ... TODO: Tighten this pattern-matching so this catch-all clause isn't necessary. + _ + (undefined)) + + ... TODO: Handle annotations. + {#Variable [name visibility state annotations type]} + (field.field (modifier#composite visibility state) + name type true sequence.empty))) + +(def: method_privacy + (-> ffi.Privacy (Modifier method.Method)) + (|>> (case> {ffi.#PublicP} method.public + {ffi.#PrivateP} method.private + {ffi.#ProtectedP} method.protected + {ffi.#DefaultP} modifier.empty))) + +(def: constructor_name + "<init>") + +(def: (mock_value valueT) + (-> (Type Value) (Bytecode Any)) + (case (jvm.primitive? valueT) + {.#Left classT} + _.aconst_null + + {.#Right primitiveT} + (cond (# jvm.equivalence = jvm.long primitiveT) + _.lconst_0 + + (# jvm.equivalence = jvm.float primitiveT) + _.fconst_0 + + (# jvm.equivalence = jvm.double primitiveT) + _.dconst_0 + + ... jvm.boolean jvm.byte jvm.short jvm.int jvm.char + _.iconst_0))) + +(def: (mock_return returnT) + (-> (Type Return) (Bytecode Any)) + (case (jvm.void? returnT) + {.#Right returnT} + _.return + + {.#Left valueT} + ($_ _.composite + (mock_value valueT) + (case (jvm.primitive? valueT) + {.#Left classT} + _.areturn + + {.#Right primitiveT} + (cond (# jvm.equivalence = jvm.long primitiveT) + _.lreturn + + (# jvm.equivalence = jvm.float primitiveT) + _.freturn + + (# jvm.equivalence = jvm.double primitiveT) + _.dreturn + + ... jvm.boolean jvm.byte jvm.short jvm.int jvm.char + _.ireturn))))) + +(def: (mock_method super method) + (-> (Type Class) (Method_Definition Code) (Resource method.Method)) + (case method + {#Constructor [privacy strict_floating_point? annotations variables exceptions + self arguments constructor_arguments + body]} + (method.method ($_ modifier#composite + (..method_privacy privacy) + (if strict_floating_point? + method.strict + modifier.empty)) + ..constructor_name + (jvm.method [variables (list#each product.right arguments) jvm.void exceptions]) + (list) + {.#Some ($_ _.composite + (_.aload 0) + (|> constructor_arguments + (list#each (|>> product.left ..mock_value)) + (monad.all _.monad)) + (|> (jvm.method [(list) (list#each product.left constructor_arguments) jvm.void (list)]) + (_.invokespecial super ..constructor_name)) + _.return + )}) + + {#Overriden_Method [super name strict_floating_point? annotations variables + self arguments return exceptions + body]} + (method.method ($_ modifier#composite + method.public + (if strict_floating_point? + method.strict + modifier.empty)) + name + (jvm.method [variables (list#each product.right arguments) return exceptions]) + (list) + {.#Some (..mock_return return)}) + + {#Virtual_Method [name privacy final? strict_floating_point? annotations variables + self arguments return exceptions + body]} + (method.method ($_ modifier#composite + (..method_privacy privacy) + (if strict_floating_point? + method.strict + modifier.empty) + (if final? + method.final + modifier.empty)) + name + (jvm.method [variables (list#each product.right arguments) return exceptions]) + (list) + {.#Some (..mock_return return)}) + + {#Static_Method [name privacy strict_floating_point? annotations + variables arguments return exceptions + body]} + (method.method ($_ modifier#composite + method.static + (..method_privacy privacy) + (if strict_floating_point? + method.strict + modifier.empty)) + name + (jvm.method [variables (list#each product.right arguments) return exceptions]) + (list) + {.#Some (..mock_return return)}) + + {#Abstract_Method [name privacy annotations + variables arguments return exceptions]} + (method.method ($_ modifier#composite + method.abstract + (..method_privacy privacy)) + name + (jvm.method [variables (list#each product.right arguments) return exceptions]) + (list) + {.#None}) + )) + +(def: (mock declaration super interfaces inheritance fields methods) + (-> Declaration + (Type Class) (List (Type Class)) + (Modifier class.Class) (List ..Field) (List (Method_Definition Code)) + (Try [External Binary])) + (mock_class declaration super interfaces + (list#each ..field_definition fields) + (list#each (..mock_method super) methods) + inheritance)) + +(def: (class::anonymous class_loader host) + (-> java/lang/ClassLoader runtime.Host Handler) (..custom [($_ <>.and (<code>.tuple (<>.some ..var)) @@ -2274,9 +2514,25 @@ (do [! phase.monad] [_ (..ensure_fresh_class! class_loader (..reflection super_class)) _ (monad.each ! (|>> ..reflection (..ensure_fresh_class! class_loader)) super_interfaces) + + self_name (///.lifted (do meta.monad + [where meta.current_module_name + id meta.seed] + (in (..anonymous_class_name where id)))) + .let [selfT {.#Primitive self_name (list)}] + mock (<| phase.lifted + (..mock [self_name parameters] + super_class + super_interfaces + class.final + (list) + (list#each (|>> {#Overriden_Method}) methods))) + ... Necessary for reflection to work properly during analysis. + _ (phase.lifted (# host execute mock)) + parameters (typeA.check (..parameter_types parameters)) .let [mapping (list#mix (function (_ [parameterJ parameterT] mapping) - (dictionary.has (jvm_parser.name parameterJ) + (dictionary.has (parser.name parameterJ) parameterT mapping)) luxT.fresh @@ -2285,12 +2541,6 @@ super_interfaceT+ (typeA.check (monad.each check.monad (|>> ..signature (luxT.check (luxT.class mapping))) super_interfaces)) - selfT (///.lifted (do meta.monad - [where meta.current_module_name - id meta.seed] - (in (inheritance_relationship_type {.#Primitive (..anonymous_class_name where id) (list)} - super_classT - super_interfaceT+)))) _ (typeA.inference selfT) constructor_argsA+ (monad.each ! (function (_ [type term]) (do ! @@ -2308,15 +2558,15 @@ (/////analysis.tuple (list#each typed_analysis constructor_argsA+)) (/////analysis.tuple methodsA))})))])) -(def: (bundle::class class_loader) - (-> java/lang/ClassLoader Bundle) +(def: (bundle::class class_loader host) + (-> java/lang/ClassLoader runtime.Host Bundle) (<| (///bundle.prefix "class") (|> ///bundle.empty - (///bundle.install "anonymous" (class::anonymous class_loader)) + (///bundle.install "anonymous" (class::anonymous class_loader host)) ))) -(def: .public (bundle class_loader) - (-> java/lang/ClassLoader Bundle) +(def: .public (bundle class_loader host) + (-> java/lang/ClassLoader runtime.Host Bundle) (<| (///bundle.prefix "jvm") (|> ///bundle.empty (dictionary.merged bundle::conversion) @@ -2328,5 +2578,5 @@ (dictionary.merged bundle::array) (dictionary.merged (bundle::object class_loader)) (dictionary.merged (bundle::member class_loader)) - (dictionary.merged (bundle::class class_loader)) + (dictionary.merged (bundle::class class_loader host)) ))) diff --git a/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/directive/jvm.lux b/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/directive/jvm.lux index 5641140a4..49cb5d6f0 100644 --- a/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/directive/jvm.lux +++ b/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/directive/jvm.lux @@ -28,8 +28,8 @@ ["[0]" template]] [math [number - ["[0]" i32] - ["n" nat]]] + ["n" nat] + ["[0]" i32]]] [target [jvm ["_" bytecode {"+" Bytecode} ("[1]#[0]" monad)] diff --git a/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/generation/jvm/host.lux b/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/generation/jvm/host.lux index ffa8e8b03..6f35d182a 100644 --- a/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/generation/jvm/host.lux +++ b/stdlib/source/library/lux/tool/compiler/language/lux/phase/extension/generation/jvm/host.lux @@ -776,7 +776,7 @@ (_.invokestatic class method (type.method [(list) (list#each product.left inputsTG) outputT (list)])) (prepare_output outputT)))))])) -(template [<name> <invoke>] +(template [<check_cast?> <name> <invoke>] [(def: <name> Handler (..custom @@ -787,14 +787,16 @@ inputsTG (monad.each ! (generate_input generate archive) inputsTS)] (in ($_ _.composite objectG - (_.checkcast class) + (if <check_cast?> + (_.checkcast class) + (_#in [])) (monad.each _.monad product.right inputsTG) (<invoke> class method (type.method [(list) (list#each product.left inputsTG) outputT (list)])) (prepare_output outputT)))))]))] - [invoke::virtual _.invokevirtual] - [invoke::special _.invokespecial] - [invoke::interface _.invokeinterface] + [#1 invoke::virtual _.invokevirtual] + [#0 invoke::special _.invokespecial] + [#1 invoke::interface _.invokeinterface] ) (def: invoke::constructor diff --git a/stdlib/source/library/lux/tool/compiler/meta/cli.lux b/stdlib/source/library/lux/tool/compiler/meta/cli.lux index eee8d719c..056652661 100644 --- a/stdlib/source/library/lux/tool/compiler/meta/cli.lux +++ b/stdlib/source/library/lux/tool/compiler/meta/cli.lux @@ -26,7 +26,9 @@ [module ["[0]" descriptor]]]]]] [world - [file {"+" Path}]]]]) + [file {"+" Path}]]]] + ["[0]" / "_" + ["[1][0]" compiler {"+" Compiler}]]) (type: .public Host_Dependency Path) @@ -34,48 +36,6 @@ (type: .public Library Path) -(type: .public Compiler - (Record - [#definition Symbol - #parameters (List Text)])) - -(def: .public compiler_equivalence - (Equivalence Compiler) - ($_ product.equivalence - symbol.equivalence - (list.equivalence text.equivalence) - )) - -(template [<ascii> <name>] - [(def: <name> - Text - (text.of_char (hex <ascii>)))] - - ["02" parameter_start] - ["03" parameter_end] - ) - -(def: compiler_parameter - (-> Text Text) - (text.enclosed [..parameter_start ..parameter_end])) - -(def: .public (compiler_format [[module short] parameters]) - (%.Format Compiler) - (%.format (..compiler_parameter module) (..compiler_parameter short) - (text.together (list#each ..compiler_parameter parameters)))) - -(def: compiler_parser' - (<text>.Parser Compiler) - (let [parameter (: (<text>.Parser Text) - (<| (<>.after (<text>.this ..parameter_start)) - (<>.before (<text>.this ..parameter_end)) - (<text>.slice (<text>.many! (<text>.none_of! ..parameter_end)))))] - (do <>.monad - [module parameter - short parameter - parameters (<>.some parameter)] - (in [[module short] parameters])))) - (type: .public Source Path) @@ -113,7 +73,7 @@ [host_dependency_parser "--host_dependency" Host_Dependency <cli>.any] [library_parser "--library" Library <cli>.any] - [compiler_parser "--compiler" Compiler (<text>.then ..compiler_parser' <cli>.any)] + [compiler_parser "--compiler" Compiler (<text>.then /compiler.parser <cli>.any)] [source_parser "--source" Source <cli>.any] [target_parser "--target" Target <cli>.any] [module_parser "--module" Module <cli>.any] diff --git a/stdlib/source/library/lux/tool/compiler/meta/cli/compiler.lux b/stdlib/source/library/lux/tool/compiler/meta/cli/compiler.lux new file mode 100644 index 000000000..9bc446b4d --- /dev/null +++ b/stdlib/source/library/lux/tool/compiler/meta/cli/compiler.lux @@ -0,0 +1,61 @@ +(.using + [library + [lux "*" + [abstract + [monad {"+" do}] + [equivalence {"+" Equivalence}]] + [control + ["<>" parser + ["<[0]>" text {"+" Parser}]]] + [data + ["[0]" product] + ["[0]" text + ["%" format]] + [collection + ["[0]" list ("[1]#[0]" functor)]]] + [math + [number {"+" hex}]] + [meta + ["[0]" symbol]]]]) + +(type: .public Compiler + (Record + [#definition Symbol + #parameters (List Text)])) + +(def: .public equivalence + (Equivalence Compiler) + ($_ product.equivalence + symbol.equivalence + (list.equivalence text.equivalence) + )) + +(template [<ascii> <name>] + [(def: <name> + Text + (text.of_char (hex <ascii>)))] + + ["02" start] + ["03" end] + ) + +(def: parameter + (-> Text Text) + (text.enclosed [..start ..end])) + +(def: .public (format [[module short] parameters]) + (%.Format Compiler) + (%.format (..parameter module) (..parameter short) + (text.together (list#each ..parameter parameters)))) + +(def: .public parser + (Parser Compiler) + (let [parameter (: (Parser Text) + (<| (<>.after (<text>.this ..start)) + (<>.before (<text>.this ..end)) + (<text>.slice (<text>.many! (<text>.none_of! ..end)))))] + (do <>.monad + [module parameter + short parameter + parameters (<>.some parameter)] + (in [[module short] parameters])))) diff --git a/stdlib/source/test/lux/control/parser/code.lux b/stdlib/source/test/lux/control/parser/code.lux index d4fac4fa6..d851a79d1 100644 --- a/stdlib/source/test/lux/control/parser/code.lux +++ b/stdlib/source/test/lux/control/parser/code.lux @@ -1,31 +1,31 @@ (.using - [library - [lux "*" - ["_" test {"+" Test}] - [abstract - [monad {"+" do}]] - [control - ["[0]" function] - ["[0]" try] - ["<>" parser]] - [data - ["[0]" bit] - ["[0]" text] - [collection - ["[0]" list]]] - [macro - ["[0]" code]] - [math - ["[0]" random {"+" Random}] - [number - ["[0]" nat] - ["[0]" int] - ["[0]" rev] - ["[0]" frac]]] - [meta - ["[0]" symbol]]]] - [\\library - ["[0]" /]]) + [library + [lux "*" + ["_" test {"+" Test}] + [abstract + [monad {"+" do}]] + [control + ["[0]" function] + ["[0]" try] + ["<>" parser]] + [data + ["[0]" bit] + ["[0]" text] + [collection + ["[0]" list]]] + [macro + ["[0]" code]] + [math + ["[0]" random {"+" Random} ("[1]#[0]" functor)] + [number + ["[0]" nat] + ["[0]" int] + ["[0]" rev] + ["[0]" frac]]] + [meta + ["[0]" symbol]]]] + [\\library + ["[0]" /]]) (template: (!expect <pattern> <value>) [(case <value> @@ -35,10 +35,24 @@ _ false)]) -(def: random_symbol +(def: local_symbol + (Random Text) + (random.ascii/lower 1)) + +(def: global_symbol + (Random Symbol) + ($_ random.and + (random.ascii/lower 1) + (random.ascii/lower 1) + )) + +(def: any_symbol (Random Symbol) - (random.and (random.unicode 1) - (random.unicode 1))) + ($_ random.either + (random#each (|>> [""]) + ..local_symbol) + ..global_symbol + )) (def: .public test Test @@ -75,8 +89,9 @@ [/.rev /.rev! random.rev code.rev rev.equivalence] [/.frac /.frac! random.safe_frac code.frac frac.equivalence] [/.text /.text! (random.unicode 1) code.text text.equivalence] - [/.symbol /.symbol! ..random_symbol code.symbol symbol.equivalence] - [/.local_symbol /.local_symbol! (random.unicode 1) code.local_symbol text.equivalence] + [/.local_symbol /.local_symbol! ..local_symbol code.local_symbol text.equivalence] + [/.global_symbol /.global_symbol! ..global_symbol code.symbol symbol.equivalence] + [/.symbol /.symbol! ..any_symbol code.symbol symbol.equivalence] )) (~~ (template [<query> <code>] [(do [! random.monad] diff --git a/stdlib/source/test/lux/tool/compiler/language/lux/analysis/scope.lux b/stdlib/source/test/lux/tool/compiler/language/lux/analysis/scope.lux index dbd1f83de..3338cc9a2 100644 --- a/stdlib/source/test/lux/tool/compiler/language/lux/analysis/scope.lux +++ b/stdlib/source/test/lux/tool/compiler/language/lux/analysis/scope.lux @@ -59,8 +59,8 @@ type/0 ($type.random 0) type/1 ($type.random 0)] ($_ _.and - (_.cover [/.find] - (|> (/.find name/0) + (_.cover [/.variable] + (|> (/.variable name/0) /.with (//phase.result state) (try#each (|>> product.right @@ -69,7 +69,7 @@ (try.else false))) (_.cover [/.with_local] (|> (/.with_local [name/0 type/0] - (/.find name/0)) + (/.variable name/0)) /.with (//phase.result state) (try#each (|>> product.right @@ -81,12 +81,12 @@ [register/0 /.next]) (/.with_local [name/0 type/0]) (do ! - [var/0 (/.find name/0)]) + [var/0 (/.variable name/0)]) (do ! [register/1 /.next]) (/.with_local [name/1 type/1]) (do ! - [var/1 (/.find name/1)]) + [var/1 (/.variable name/1)]) (in (do maybe.monad [var/0 var/0 var/1 var/1] @@ -138,8 +138,8 @@ (|> (<| /.with (/.with_local [name/0 type/0]) (do //phase.monad - [var/0' (/.find name/0) - [scope/1 var/0''] (/.with (/.find name/0))] + [var/0' (/.variable name/0) + [scope/1 var/0''] (/.with (/.variable name/0))] (<| //phase.lifted try.of_maybe (do maybe.monad @@ -174,7 +174,7 @@ (/.with_local [name/0 type/0]) (/.with_local [name/1 type/1]) (do ! - [[scope/1 _] (/.with (/.find name/0))] + [[scope/1 _] (/.with (/.variable name/0))] (in [register/0 (/.environment scope/1)]))) (//phase.result state) (try#each (function (_ [_ [register/0 environment]]) @@ -190,8 +190,8 @@ (/.with_local [name/1 type/1]) (do [! //phase.monad] [[scope/1 _] (/.with (do ! - [_ (/.find name/1) - _ (/.find name/0)] + [_ (/.variable name/1) + _ (/.variable name/0)] (in [])))] (in [register/0 register/1 (/.environment scope/1)]))) (//phase.result state) diff --git a/stdlib/source/test/lux/tool/compiler/meta/cli.lux b/stdlib/source/test/lux/tool/compiler/meta/cli.lux index 5a128b0ff..15441533e 100644 --- a/stdlib/source/test/lux/tool/compiler/meta/cli.lux +++ b/stdlib/source/test/lux/tool/compiler/meta/cli.lux @@ -22,15 +22,10 @@ ["[0]" symbol "_" ["$[1]" \\test]]]]] [\\library - ["[0]" /]]) - -(def: random_compiler - (Random /.Compiler) - (do [! random.monad] - [definition ($symbol.random 1 1) - amount (# ! each (n.% 5) random.nat) - parameters (random.list amount (random.ascii/lower 2))] - (in [definition parameters]))) + ["[0]" / + ["[1][0]" compiler {"+" Compiler}]]] + ["$[0]" / "_" + ["[1][0]" compiler]]) (def: .public test Test @@ -44,11 +39,11 @@ libraries (random.list amount (random.ascii/lower 3)) target (random.ascii/lower 4) module (random.ascii/lower 5) - compilers (random.list amount ..random_compiler) + compilers (random.list amount $/compiler.random) .let [compilation' ($_ list#composite (list#conjoint (list#each (|>> (list "--host_dependency")) host_dependencies)) (list#conjoint (list#each (|>> (list "--library")) libraries)) - (list#conjoint (list#each (|>> /.compiler_format (list "--compiler")) compilers)) + (list#conjoint (list#each (|>> /compiler.format (list "--compiler")) compilers)) (list#conjoint (list#each (|>> (list "--source")) sources)) (list "--target" target) (list "--module" module)) @@ -73,7 +68,7 @@ [/.Host_Dependency /.#host_dependencies (list#= host_dependencies)] [/.Library /.#libraries (list#= libraries)] - [/.Compiler /.#compilers (# (list.equivalence /.compiler_equivalence) = compilers)] + [/compiler.Compiler /.#compilers (# (list.equivalence /compiler.equivalence) = compilers)] [/.Source /.#sources (list#= sources)] [/.Target /.#target (same? target)] [/.Module /.#module (same? module)] @@ -94,7 +89,7 @@ [/.#host_dependencies (list#= host_dependencies)] [/.#libraries (list#= libraries)] - [/.#compilers (# (list.equivalence /.compiler_equivalence) = compilers)] + [/.#compilers (# (list.equivalence /compiler.equivalence) = compilers)] [/.#sources (list#= sources)] [/.#target (same? target)] [/.#module (same? module)] @@ -133,4 +128,6 @@ /.#module module]}] [{/.#Export [sources target]}] ))))) + + $/compiler.test )))) diff --git a/stdlib/source/test/lux/tool/compiler/meta/cli/compiler.lux b/stdlib/source/test/lux/tool/compiler/meta/cli/compiler.lux new file mode 100644 index 000000000..69a9db048 --- /dev/null +++ b/stdlib/source/test/lux/tool/compiler/meta/cli/compiler.lux @@ -0,0 +1,48 @@ +(.using + [library + [lux "*" + ["_" test {"+" Test}] + [abstract + [monad {"+" do}] + [\\specification + ["$[0]" equivalence]]] + [control + ["[0]" try ("[1]#[0]" functor)] + ["<>" parser + ["<[0]>" text]]] + [math + ["[0]" random {"+" Random}] + [number + ["n" nat]]] + [meta + ["[0]" symbol "_" + ["$[1]" \\test]]]]] + [\\library + ["[0]" /]]) + +(def: .public random + (Random /.Compiler) + (do [! random.monad] + [definition ($symbol.random 1 1) + amount (# ! each (n.% 5) random.nat) + parameters (random.list amount (random.ascii/lower 2))] + (in [/.#definition definition + /.#parameters parameters]))) + +(def: .public test + Test + (<| (_.covering /._) + (_.for [/.Compiler]) + (do [! random.monad] + [expected ..random] + ($_ _.and + (_.for [/.equivalence] + ($equivalence.spec /.equivalence ..random)) + + (_.cover [/.format /.parser] + (|> expected + /.format + (<text>.result /.parser) + (try#each (# /.equivalence = expected)) + (try.else false))) + )))) |