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

[arcilator] Add JIT runtime environment library and stdio hooks #7445

Merged
merged 11 commits into from
Aug 7, 2024

Conversation

fzi-hielscher
Copy link
Contributor

@fzi-hielscher fzi-hielscher commented Aug 6, 2024

This PR adds a dynamically linked runtime environment library to arcilator, which makes functions declared in the arcilator-runtime.h available for JIT compiled runs.

The runtime is intended to provide support functions that can be called directly from the generated IR, similar to the CRunnerUtils in LLVM. I went for a dynamic instead of a static library because I think there is some (limited) use in having the option to swap the runtime environment without having to recompile arcilator itself.

Notably, it provides a well defined interface for certain methods of the C standard library (currently fprintf, fputs, fputc). While it is technically possible to invoke them directly from the IR, going through the standard headers and a C preprocessor should avoid trouble caused by different libc implementations (glibc, musl, msvcrt, ucrt, ...).

PS: Just noticed that I switched from camel case to snake case. _arcLibcFprintf doesn't look right to me...

@fzi-hielscher fzi-hielscher marked this pull request as ready for review August 6, 2024 01:14
@@ -444,6 +450,8 @@ static LogicalResult processBuffer(
llvm::errs() << "failed to create execution engine: "
<< info.message() << "\n";
});

arc_jit_runtime_env_deinit();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: you can use llvm::make_scope_exit to automatically call deinit.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice. Thanks!

// --- Exports to the IR ---

#ifdef _WIN32
#define ARCEXPORT(rtype) extern "C" __declspec(dllexport) rtype __cdecl
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you make this consistent with MLIR_RUNNERUTILS_EXPORT macro?

Copy link
Contributor Author

@fzi-hielscher fzi-hielscher Aug 6, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

MLIR_RUNNERUTILS_EXPORT does not include extern "C" yet it looks like it is prefixed with extern "C" every time it is used. That seems odd to me.
The __cdecl is probably not required. It don't have access to my Windows PC right now, I'll check later today. Basically, I've just used every magic incantation here that I could come up with to increase the likelihood of it linking. 😅

%ascii_em = llvm.mlir.constant(33 : i32) : i32
%ascii_lf = llvm.mlir.constant(10 : i32) : i32
%ascii_l = llvm.mlir.constant(108 : i32) : i32
%in_clk = arc.root_input "clk", %arg0 {offset = 0 : i32} : (!arc.storage<1>) -> !arc.state<i1>
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This seems to be unused. Was there some issue with the storage lowering when removing this?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No, tbh I didn't even try without at least one port as I was unsure whether this is a supported use-case.
Looks like it is working just fine. 👍

Comment on lines +53 to +56
ARC_ENV_DECL_GET_PRINT_STREAM(id) {
(void)id;
return stderr;
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm wondering if it would make sense to let the id influence whether stderr or stdout is returned?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, I've added this in anticipation of #7111. The problem is, AFAIK we don't have any lowering path that would support this, let alone an agreed on concept for file descriptors / output streams in the IR. I had a short exchange with @seldridge on this in the discord channel some weeks ago. For now we could at best add a command line flag to control this globally.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah I see. I was just thinking about adding an if-statement here to allow the LLVM call operation to choose between stderr and stdout when writing MLIR integration tests and examples by hand. But this is not important at all. We can also just leave it as is until we have a proper story for file descriptors.

Comment on lines 23 to 24
#ifdef _WIN32
#define ARCEXPORT(rtype) extern "C" __declspec(dllexport) rtype __cdecl
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've no clue about those windows specific things, but thanks for making this work with Windows as well! 🚀

@@ -0,0 +1,23 @@
// Arcilator JIT runtime environment API facing the arcilator.cpp
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should probably also add the standard header comments here.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, nice catch. I'm a little surprised the CI did not complain about it. The arcilator-runtime.h and arcilator-header-cpp.py should probably also at least get a license identifier, shouldn't they?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, seems like it slipped through there.

@@ -0,0 +1,7 @@
#include "../arcilator-runtime.h"
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should probably add the standard header comments here.

Copy link
Member

@maerhart maerhart left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM! Thanks!

Copy link
Contributor

@fabianschuiki fabianschuiki left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM! Thanks for adding this 🥳

Can't wait for us to finally get the VCD writing out of the runtime header and built into the model a bit more thoroughly. That will make the JIT pretty much equivalent to a separately-compiled model 😄

@fzi-hielscher
Copy link
Contributor Author

That does make me wonder how to proceed from here. When writing this PR's code I didn't want to mess up existing build flows. So, the "arcilator-runtime" remains a header library and we simply provide a compiled binary of it to the JIT runner. That comes at the cost of mixing declarations and definitions in the header.

If our goal is to move stuff out of the header, maybe it would be better if we had an arcilator-runtime.cpp to put static (i.e., model independent) logic in? The compiled library would then be usable for both JIT and conventional flows. I'm specifically thinking of the parsing logic for .mem files. A header file is not a great place to put that. But I think dynamically building it into the MLIR model makes even less sense.

What do you think?

@fabianschuiki
Copy link
Contributor

Hmmm yeah I think it would make sense to actually compile a runtime library, link against it from the JIT, and also install it as a standalone library alongside the rest of CIRCT. If you use arcilator to build an object file, you can link against that and optionally the runtime in whatever build flow you're using. And if you use arcilator in JIT mode, you also have the library available.

@fzi-hielscher
Copy link
Contributor Author

I've brought the export macro closer to MLIR_RUNNERUTILS_EXPORT as @uenoku suggested. On my local runs the test produces the expected output for Windows, MacOS, and Linux (glibc). So, I'm about 50% sure this will not break in the nightly tests. 😬

@fzi-hielscher fzi-hielscher merged commit 30ed753 into llvm:main Aug 7, 2024
4 checks passed
@uenoku
Copy link
Member

uenoku commented Aug 7, 2024

Thank you @fzi-hielscher!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants