Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Move Builtin Types and Methods to stdlib #3363

Merged
merged 80 commits into from
May 5, 2022
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
80 commits
Select commit Hold shift + click to select a range
38a2da7
Dynamically replace builtin methods with functions
hubertp Mar 25, 2022
374a94c
Deal with Polyglot split by selectively importing
hubertp Mar 29, 2022
6afa443
Move Array type out of Builtins
hubertp Apr 4, 2022
02ab5b1
Move Ref type + basic test
hubertp Apr 4, 2022
6938b2d
Move Boolean out of Builtins to std library
hubertp Apr 5, 2022
75cd43d
Drop reflection requirement for builtin methods
hubertp Apr 6, 2022
0f89f44
Use constant values to avoid typos
hubertp Apr 6, 2022
91c6432
Move Meta out of builtins
hubertp Apr 7, 2022
43840f2
Revert "Move Boolean out of Builtins to std library"
hubertp Apr 8, 2022
9822d76
wip
hubertp Apr 9, 2022
e561565
Move Any to stdlib
hubertp Apr 11, 2022
9dbfa06
Move Boolean to stdlib
hubertp Apr 11, 2022
c42b092
Missing Booleans.enso from last commit
hubertp Apr 11, 2022
fb10830
Simplify builtins metadata generation
hubertp Apr 11, 2022
fb666c1
Cleanup
hubertp Apr 12, 2022
1dc9594
Simplify resource reading
hubertp Apr 12, 2022
8ca36e8
Merge branch 'develop' into wip/hubert/shadow-definition-builtins-181…
hubertp Apr 12, 2022
a588c80
Move List to stdlib
hubertp Apr 12, 2022
d771a7f
WIP: remove hardcoded builtin types in Builtins
hubertp Apr 13, 2022
d529ad8
Workaround separate compilation problems
hubertp Apr 13, 2022
3cfdedd
Address PR review
hubertp Apr 14, 2022
8ee6654
CamelCase to Snake_Case for builtin types
hubertp Apr 14, 2022
c04f11e
Move DataflowError to stdlib
hubertp Apr 15, 2022
20f8be4
typo
hubertp Apr 15, 2022
fd094e1
Move all error types + text
hubertp Apr 19, 2022
1a2a80a
Move IO stuff
hubertp Apr 19, 2022
8e76ba8
Add missing shared test class
hubertp Apr 19, 2022
d261608
Improve runtime of tests
hubertp Apr 19, 2022
d3e5fa8
Add IO to stdlib
hubertp Apr 19, 2022
75ba97c
DRY
hubertp Apr 19, 2022
870aacd
nit
hubertp Apr 19, 2022
67dafaa
Final AtomConstructor
hubertp Apr 19, 2022
f75be8a
Move Nothing and Function to stdlib
hubertp Apr 20, 2022
f0b45a7
Nits
hubertp Apr 21, 2022
8e7e15f
Cons/Nil are no longer Builtin_Types
hubertp Apr 21, 2022
b6ae3f0
Adapt benchmark fixtures
hubertp Apr 21, 2022
870ef1a
Remove unused imports
hubertp Apr 21, 2022
70d558d
Rename local 'reverse' method in List benchmarks
hubertp Apr 21, 2022
4f1fe32
Amend benchmark with missing import
hubertp Apr 21, 2022
6cfeca0
Make benchmarks happy again
hubertp Apr 21, 2022
b6fe9cc
fix typo
hubertp Apr 21, 2022
f17d3cb
Move runtime builtin types to stdlib
hubertp Apr 22, 2022
e4037b4
Move Prim_Warning to stdlib
hubertp Apr 22, 2022
1a586d9
Builtins is dead, long live builtins
hubertp Apr 25, 2022
da040f8
Missed some test adaptation
hubertp Apr 25, 2022
4e4535e
Getting rid of Standard.Builtins mentions
hubertp Apr 25, 2022
10d1a52
Remove empty Builtins.enso references
hubertp Apr 25, 2022
5ae3289
Automatic formatting
hubertp Apr 25, 2022
69ad669
Remove extension method for unimplemented error
hubertp Apr 25, 2022
966c2f1
Merge branch 'develop' into wip/hubert/shadow-definition-builtins-181…
hubertp Apr 25, 2022
1372387
Minor adjustments to expectations
hubertp Apr 25, 2022
9a703a3
Update Changelog
hubertp Apr 25, 2022
2ddcf9f
More docs for annotation processors
hubertp Apr 25, 2022
c927582
More Standard.Builtins purge
hubertp Apr 25, 2022
f7145e8
Merge branch 'develop' into wip/hubert/shadow-definition-builtins-181…
hubertp Apr 28, 2022
1ecd317
Follow up on merge with develop
hubertp Apr 28, 2022
a384ed0
PR review re IO and Prim_Io modules
hubertp Apr 28, 2022
40ad2c7
PR review Numbers
hubertp Apr 28, 2022
23e9298
Remove builtin Ordering type
hubertp Apr 28, 2022
a33b45d
Move FIXME comment on Ref to pivotal ticket
hubertp Apr 28, 2022
0c5c19c
Remove leftover comment
hubertp Apr 28, 2022
3417b66
More cleanups, as per PR comments
hubertp Apr 28, 2022
7dc9a34
Formatting
hubertp Apr 28, 2022
3adc0ef
Remove logic to handle Any/Nothing methods
hubertp Apr 29, 2022
dc11b5a
Auto-generate Builtins Constants
hubertp Apr 29, 2022
9414fce
Test adjustments to the previous commit
hubertp May 4, 2022
fa4ca5b
Update distribution/lib/Standard/Base/0.0.0-dev/src/Error/Common.enso
hubertp May 4, 2022
cba36f3
Make formatter happy again
hubertp May 4, 2022
7bf6261
DRY
hubertp May 4, 2022
fe89bc1
Add a requested compiler assert
hubertp May 4, 2022
1c40d25
Merge branch 'develop' into wip/hubert/shadow-definition-builtins-181…
hubertp May 4, 2022
6ba1bc4
Fix tests
hubertp May 4, 2022
4fde6ba
Increase timeout to avoid FPs
hubertp May 4, 2022
b10bb8c
Turn on IR caches for tests
hubertp May 5, 2022
6f27d5e
Merge branch 'develop' into wip/hubert/shadow-definition-builtins-181…
hubertp May 5, 2022
aeeef17
Fix tests
hubertp May 5, 2022
670cad6
Merge branch 'develop' into wip/hubert/shadow-definition-builtins-181…
hubertp May 5, 2022
6c06032
Make disabling IR cache in tests configurable
hubertp May 5, 2022
c90836d
More formatting
hubertp May 5, 2022
884a7d1
Merge branch 'develop' into wip/hubert/shadow-definition-builtins-181…
hubertp May 5, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 1 addition & 2 deletions build.sbt
Original file line number Diff line number Diff line change
Expand Up @@ -1147,8 +1147,7 @@ lazy val runtime = (project in file("engine/runtime"))
"org.scalatest" %% "scalatest" % scalatestVersion % Test,
"org.graalvm.truffle" % "truffle-api" % graalVersion % Benchmark,
"org.typelevel" %% "cats-core" % catsVersion,
"eu.timepit" %% "refined" % refinedVersion,
"org.reflections" % "reflections" % "0.10.2"
"eu.timepit" %% "refined" % refinedVersion
),
// Note [Unmanaged Classpath]
Compile / unmanagedClasspath += (`core-definition` / Compile / packageBin).value,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
import com.oracle.truffle.api.nodes.Node;
import org.enso.interpreter.dsl.BuiltinMethod;

@BuiltinMethod(type = "", name = "<join_thread>")
@BuiltinMethod(type = "Special", name = "<join_thread>")
public class JoinThreadNode extends Node {
public Object execute(Object thread) {
try {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
import org.enso.interpreter.dsl.BuiltinMethod;
import org.enso.interpreter.runtime.data.Ref;

@BuiltinMethod(type = "", name = "<new_ref>")
@BuiltinMethod(type = "Special", name = "<new_ref>")
public class NewRefNode extends Node {
public Ref execute() {
return new Ref(null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
import org.enso.interpreter.dsl.BuiltinMethod;
import org.enso.interpreter.runtime.data.Ref;

@BuiltinMethod(type = "", name = "<read_ref>")
@BuiltinMethod(type = "Special", name = "<read_ref>")
public class ReadRefNode extends Node {
public Object execute(Ref ref) {
return ref.getValue();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
import org.enso.interpreter.node.callable.thunk.ThunkExecutorNodeGen;
import org.enso.interpreter.runtime.Context;

@BuiltinMethod(type = "", name = "<run_thread>")
@BuiltinMethod(type = "Special", name = "<run_thread>")
public abstract class RunThreadNode extends Node {
static RunThreadNode build() {
return RunThreadNodeGen.create();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
import org.enso.interpreter.dsl.BuiltinMethod;
import org.enso.interpreter.runtime.data.Ref;

@BuiltinMethod(type = "", name = "<write_ref>")
@BuiltinMethod(type = "Special", name = "<write_ref>")
public class WriteRefNode extends Node {
public Object execute(Ref ref, Object value) {
ref.setValue(value);
Expand Down
Original file line number Diff line number Diff line change
@@ -1,17 +1,21 @@
package org.enso.interpreter.runtime.builtin;

import com.oracle.truffle.api.CompilerDirectives;

import java.io.IOException;
import java.lang.reflect.Method;
import java.net.URI;
import java.nio.charset.StandardCharsets;
import java.nio.file.*;
import java.util.*;
import java.util.stream.Stream;

import com.oracle.truffle.api.nodes.NodeInfo;
import org.enso.compiler.Passes;
import org.enso.compiler.context.FreshNameSupply;
import org.enso.compiler.exception.CompilerError;
import org.enso.compiler.phase.BuiltinsIrBuilder;
import org.enso.interpreter.Language;
import org.enso.interpreter.dsl.model.MethodDefinition;
import org.enso.interpreter.node.expression.builtin.BuiltinRootNode;
import org.enso.interpreter.node.expression.builtin.debug.DebugBreakpointMethodGen;
import org.enso.interpreter.node.expression.builtin.debug.DebugEvalMethodGen;
Expand All @@ -21,10 +25,6 @@
import org.enso.interpreter.node.expression.builtin.error.GetAttachedStackTraceMethodGen;
import org.enso.interpreter.node.expression.builtin.error.ThrowPanicMethodGen;
import org.enso.interpreter.node.expression.builtin.function.ExplicitCallFunctionMethodGen;
import org.enso.interpreter.node.expression.builtin.interop.generic.GetMemberMethodGen;
import org.enso.interpreter.node.expression.builtin.interop.generic.IsLanguageInstalledMethodGen;
import org.enso.interpreter.node.expression.builtin.interop.java.AddToClassPathMethodGen;
import org.enso.interpreter.node.expression.builtin.interop.java.LookupClassMethodGen;
import org.enso.interpreter.node.expression.builtin.io.GetCwdMethodGen;
import org.enso.interpreter.node.expression.builtin.io.GetFileMethodGen;
import org.enso.interpreter.node.expression.builtin.io.GetUserHomeMethodGen;
Expand Down Expand Up @@ -52,9 +52,6 @@
import org.enso.interpreter.runtime.type.Constants;
import org.enso.pkg.QualifiedName;

import org.reflections.Reflections;
import org.reflections.scanners.Scanners;

/** Container class for static predefined atoms, methods, and their containing scope. */
public class Builtins {
public static final String PACKAGE_NAME = "Builtins";
Expand Down Expand Up @@ -209,7 +206,7 @@ public Builtins(Context context) {
thread, "with_interrupt_handler", WithInterruptHandlerMethodGen.makeFunction(language));

scope.registerMethod(unsafe, "set_atom_field", SetAtomFieldMethodGen.makeFunction(language));
readAllBuiltinNodes(scope);
readBuiltinsMetadata(scope);
}

/** @return {@code true} if the IR has been initialized, otherwise {@code false} */
Expand Down Expand Up @@ -250,39 +247,98 @@ public void initializeBuiltinsIr(FreshNameSupply freshNameSupply, Passes passes)
}
}

@CompilerDirectives.TruffleBoundary
public void readAllBuiltinNodes(ModuleScope scope) {
Reflections reflections = new Reflections("org.enso.interpreter.node.expression.builtin");
Set<String> subtypes = reflections.get(Scanners.SubTypes.of(Scanners.TypesAnnotated.with(NodeInfo.class)));
subtypes.stream().forEach(className -> {
private void readBuiltinsMetadata(ModuleScope scope) {
ClassLoader classLoader = getClass().getClassLoader();
FileSystem fs = null;
Stream<Path> builtinMetaPath;
try {
URI resource = classLoader.getResource(MethodDefinition.META_PATH).toURI();
hubertp marked this conversation as resolved.
Show resolved Hide resolved
fs = initFileSystem(resource);
builtinMetaPath = Files.walk(Paths.get(resource)).flatMap(p -> acceptFiles(p));
} catch (Exception ioe) {
ioe.printStackTrace();
builtinMetaPath = Stream.empty();
}
builtinMetaPath.forEach(metaPath -> {
List<String> lines;
try {
Class<BuiltinRootNode> clazz = (Class<BuiltinRootNode>) Class.forName(className);
NodeInfo info = clazz.getDeclaredAnnotation(NodeInfo.class);
String[] name = info.shortName().split("\\.");
if (name.length == 2 && !name[0].isEmpty()) {
scope.getLocalConstructor(name[0]).ifPresentOrElse(constr -> {
Map<String, Class<BuiltinRootNode>> atomNodes = builtinNodes.get(name[0]);
if (atomNodes == null) {
atomNodes = new HashMap<>();
// TODO: move away from String Map once Builtins are gone
builtinNodes.put(constr.getName(), atomNodes);
}
atomNodes.put(name[1], clazz);
}, () -> {
Map<String, Class<BuiltinRootNode>> atomNodes = builtinNodes.get(name[0]);
if (atomNodes == null) {
atomNodes = new HashMap<>();
// TODO: move away from String Map once Builtins are gone
builtinNodes.put(name[0], atomNodes);
}
atomNodes.put(name[1], clazz);
});
}
} catch (Exception e) {
throw new CompilerError(e.getMessage());
lines = Files.readAllLines(metaPath, StandardCharsets.UTF_8);
} catch (IOException e) {
e.printStackTrace();
lines = new ArrayList<>();
}
// todo: handle gracefully other cases
lines.forEach(line -> {
String[] builtinMeta = line.split(":");
if (builtinMeta.length != 2) {
throw new CompilerError("Invalid builtin metadata in " + metaPath + ": " + line);
}
String[] builtinName = builtinMeta[0].split("\\.");
if (builtinName.length != 2) {
throw new CompilerError("Invalid builtin metadata in " + metaPath + ": " + line);
}
try {
Class<BuiltinRootNode> clazz = (Class<BuiltinRootNode>) Class.forName(builtinMeta[1]);
String builtinMethodOwner = builtinName[0];
String builtinMethodName = builtinName[1];
scope.getLocalConstructor(builtinMethodOwner).ifPresentOrElse(constr -> {
Map<String, Class<BuiltinRootNode>> atomNodes = builtinNodes.get(builtinMethodOwner);
if (atomNodes == null) {
atomNodes = new HashMap<>();
// TODO: move away from String Map once Builtins are gone
builtinNodes.put(constr.getName(), atomNodes);
}
atomNodes.put(builtinMethodName, clazz);
}, () -> {
Map<String, Class<BuiltinRootNode>> atomNodes = builtinNodes.get(builtinMethodOwner);
if (atomNodes == null) {
atomNodes = new HashMap<>();
// TODO: move away from String Map once Builtins are gone
builtinNodes.put(builtinMethodOwner, atomNodes);
}
atomNodes.put(builtinMethodName, clazz);
});
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
});
});
if (fs != null) {
try {
fs.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}

private FileSystem initFileSystem(URI uri) throws IOException
{
// Returning null ensures that we use the default one and at the same time we don't attempt
// to close it.
try {
FileSystems.getFileSystem(uri);
return null;
} catch (IllegalArgumentException iae) {
// file: schema doesn't like non-/ path but that's fine, it means the default file system is already setup
return null;
} catch (FileSystemNotFoundException e) {
Map<String, String> env = new HashMap<>();
env.put("create", "true");
return FileSystems.newFileSystem(uri, env);
}
}

private Stream<Path> acceptFiles(Path path) {
if (Files.isRegularFile(path) && path.getFileName().toString().endsWith(".builtin")) {
return Stream.of(path);
} else if (Files.isDirectory(path)) {
try {
return Files.walk(path).flatMap(p -> {if (p != path) return acceptFiles(p); else return Stream.empty();});
} catch (IOException ioe) {
ioe.printStackTrace();
}
}
return Stream.empty();
}

public Optional<Function> getBuiltinFunction(AtomConstructor atom, String methodName, Language language) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,8 @@
import javax.lang.model.element.TypeElement;
import javax.tools.Diagnostic;
import javax.tools.JavaFileObject;
import javax.tools.FileObject;
import javax.tools.StandardLocation;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.*;
Expand Down Expand Up @@ -61,6 +63,7 @@ public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment
}
try {
generateCode(def);
registerBuiltinMethod(def, element);
} catch (IOException e) {
e.printStackTrace();
}
Expand Down Expand Up @@ -91,6 +94,19 @@ public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment
"org.enso.interpreter.runtime.state.Stateful",
"org.enso.interpreter.runtime.type.TypesGen");

private void registerBuiltinMethod(MethodDefinition methodDefinition, Element element) throws IOException {
String tpe = methodDefinition.getType().toLowerCase();
if (tpe.isEmpty()) {
throw new InternalError("Type of the BuiltinMethod cannot be empty");
}
FileObject res = processingEnv.getFiler().createResource(
StandardLocation.CLASS_OUTPUT, "",
MethodDefinition.META_PATH + "/" + tpe + "/" + methodDefinition.getClassName() + ".builtin", element
);
hubertp marked this conversation as resolved.
Show resolved Hide resolved
String fullClassName = methodDefinition.getPackageName() + "." + methodDefinition.getClassName();
res.openWriter().append(methodDefinition.getDeclaredName() + ":" + fullClassName+"\n").close();
}

private void generateCode(MethodDefinition methodDefinition) throws IOException {
JavaFileObject gen =
processingEnv.getFiler().createSourceFile(methodDefinition.getQualifiedName());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,8 @@
/** A domain-specific representation of a builtin method. */
public class MethodDefinition {
private static final String STATEFUL = "org.enso.interpreter.runtime.state.Stateful";
public static final String NODE_PKG = "org.enso.interpreter.node.expression.builtin";
public static final String META_PATH = "META-INF" + "/" + NODE_PKG.replace('.', '/');

private final String packageName;
private final String originalClassName;
Expand Down Expand Up @@ -138,6 +140,11 @@ public String getDeclaredName() {
return annotation.type() + "." + annotation.name();
}

/** @return the language-level owner type of this method. */
public String getType() {
return annotation.type();
}

/** @return get the description of this method. */
public String getDescription() {
return annotation.description();
Expand Down