<div dir="ltr">Hi Fedor,<div><br></div><div>Thanks for replying my questions about porting the OptBisecting to new PM.</div><div><br></div><div>This thread looks like a great improvement on what we currently have.</div><div>Though we are also trying to make opt-bisect more granular.</div><div><br></div><div>In particular, we think it would be helpful if we could have opt-bisect work on a per-optimization level rather than per-pass level.</div><div>I believe this will be a more invasive change and we would like to do this as a follow-up to this thread.</div><div><br></div><div>How difficult do you think it would be to support this use-case with your design?</div><div><br></div><div>Thank you!</div><div>Zhizhou</div><div> </div></div><br><div class="gmail_quote"><div dir="ltr">On Wed, Jun 6, 2018 at 5:00 PM Fedor Sergeev via llvm-dev <<a href="mailto:llvm-dev@lists.llvm.org" target="_blank">llvm-dev@lists.llvm.org</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">TL;DR<br>
====<br>
<br>
This RFC suggests an API to enable customizable instrumentation of pass <br>
execution.<br>
The intent is to have a common machinery to implement all the <br>
pass-execution-debugging<br>
features mentioned here.<br>
<br>
Prime target of the interface is the new pass manager.<br>
The overall approach and most of the implementation details should be <br>
equially applicable<br>
to the legacy one though.<br>
<br>
<br>
Background<br>
==========<br>
<br>
There are quite a few important debugging facilities in LLVM that affect<br>
pass execution sequence:<br>
<br>
-print-after/-print-before[-all]<br>
execute IR-print pass before or after a particularly <br>
insteresting pass<br>
(or all passes)<br>
<br>
-verify-each<br>
execute verifier pass after each<br>
<br>
-opt-bisect-limit<br>
execute passes only up to a selected "pass-counter"<br>
<br>
-time-passes<br>
track execution time for each pass<br>
<br>
There are also quite a few similar ideas floating around, i.e:<br>
-git-commit-after-all<br>
-debugify-each<br>
<br>
All these facilities essentially require instrumentation of pass <br>
execution points<br>
in the pass manager, each being implemented in a legacy pass manager <br>
through their<br>
own custom way, e.g:<br>
* -time-passes has a bunch of dedicated code in each of the pass managers<br>
<br>
* -print-before/after/verify-each insert additional passes before/after<br>
the passes in the pipeline<br>
<br>
And there is no implementation of any of these features for the new pass <br>
manager,<br>
which obviously is a problem for new pass manager transition.<br>
<br>
Proposal<br>
========<br>
<br>
Main idea:<br>
- introduce an API that allows to instrument points of pass execution<br>
- access through LLVM Context (allows to control life-time and scope <br>
in multi-context execution)<br>
- wrap it into an analysis for easier access from pass managers<br>
<br>
<br>
Details:<br>
1. introduce llvm::PassInstrumentation<br>
<br>
This is the main interface that handles the customization and <br>
provides instrumentation calls<br>
<br>
- resides in IR<br>
- is accessible through LLVMContext::getPassInstrumentation()<br>
(with context owning this object).<br>
<br>
2. every single point of Pass execution in the (new) PassManager(s) <br>
will query<br>
this analysis and run instrumentation call specific to a <br>
particular point.<br>
<br>
Instrumentation points:<br>
<br>
bool BeforePass (PassID, PassExecutionCounter);<br>
void AfterPass (PassID, PassExecutionCounter);<br>
<br>
Run before/after a particular pass execution<br>
BeforePass instrumentation call returns true if this <br>
execution is allowed to run.<br>
<br>
'PassID'<br>
certain unique identifier for a pass (pass name?).<br>
<br>
'PassExecutionCounter'<br>
a number that uniquely identifies this particular pass <br>
execution<br>
in current pipeline, as tracked by Pass Manager.<br>
<br>
void StartPipeline()<br>
void EndPipeline()<br>
<br>
Run at the start/end of a pass pipeline execution.<br>
(useful for initialization/finalization purposes)<br>
<br>
<br>
3. custom callbacks are registered with <br>
PassInstrumentation::register* interfaces<br>
<br>
A sequence of registered callbacks is called at each <br>
instrumentation point as appropriate.<br>
<br>
4. introduce llvm::ExecutionCounter to track execution of passes<br>
<br>
(akin to DebugCounter, yet enabled in Release mode as well?)<br>
<br>
Note: it is somewhat nontrivial to uniquely track pass executions <br>
with counters in new pass<br>
manager as its pipeline schedule can be dynamic. Ideas are welcome <br>
on how to efficiently<br>
implement unique execution tracking that does not break in <br>
presence of fixed-point iteration<br>
passes like RepeatedPass/DevirtSCCRepeatedPass<br>
<br>
Also, the intent is for execution counters to be able provide <br>
thread-safety in multi-threaded<br>
pipeline execution (though no work planned for it yet).<br>
<br>
5. introduce a new analysis llvm::PassInstrumentationAnalysis<br>
<br>
This is a convenience wrapper to provide an access to <br>
PassInstrumentation via analysis framework.<br>
If using analysis is not convenient (?legacy) then <br>
PassInstrumentation can be queried<br>
directly from LLVMContext.<br>
<br>
<br>
Additional goals<br>
================<br>
<br>
- layering problem<br>
Currently OptBisect/OptPassGate has layering issue - interface <br>
dependencies on all the "IR units",<br>
even those that are analyses - Loop, CallGraphSCC.<br>
<br>
Generic PassInstrumentation facilitiy allows to inject arbitrary <br>
call-backs in run-time,<br>
removing any compile-time interface dependencies on internals of <br>
those callbacks,<br>
effectively solving this layering issue.<br>
<br>
- life-time/scope control for multi-context execution<br>
<br>
Currently there are issues with multi-context execution of, say, <br>
-time-passes which store<br>
their data in global maps.<br>
<br>
With LLVMContext owning PassInstrumentation there should be no <br>
problem with multi-context execution<br>
(callbacks can be made owning the instrumentation data).<br>
<br>
Open Questions<br>
==============<br>
<br>
- whats the best way to handle ownership of PassInstrumentation<br>
<br>
Any problems with owning by LLVMContext?<br>
Something similar to TargetLibraryInfo (owned by <br>
TargetLibraryAnalysis/TargetLibraryInfoWrapperPass)?<br>
<br>
- using PassInstrumentationAnalysis or directly querying LLVMContext<br>
<br>
PassInstrumentationAnalysis appeared to be a nice idea, only until <br>
I tried querying it<br>
in new pass manager framework, and amount of hooplas to jump over <br>
makes me shiver a bit...<br>
<br>
Querying LLVMContext is plain and straightforward, but we do not <br>
have a generic way to access LLVMContext<br>
from a PassManager template (need to introduce generic <br>
IRUnit::getContext?)<br>
<br>
Implementation<br>
==============<br>
<br>
PassInstrumentationAnalysis proof-of-concept unfinished prototype <br>
implementation:<br>
(Heavily under construction, do not enter without wearing a hard hat...)<br>
<br>
<a href="https://reviews.llvm.org/D47858" rel="noreferrer" target="_blank">https://reviews.llvm.org/D47858</a><br>
<br>
_______________________________________________<br>
LLVM Developers mailing list<br>
<a href="mailto:llvm-dev@lists.llvm.org" target="_blank">llvm-dev@lists.llvm.org</a><br>
<a href="http://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev" rel="noreferrer" target="_blank">http://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev</a><br>
</blockquote></div>