Skip to content

Commit ee5491a

Browse files
committed
PassManager: simplify the options to print intermediate SIL.
* rename -sil-print-only-function to -sil-print-function and -sil-print-only-functions to -sil-print-functions * to print single functions, don't require -Xllvm -sil-print-all. It's now sufficient to use e.g. -Xllvm -sil-print-function=<f> But it's still possible to select functions with -sil-print-function(s) for other print options, -sil-print-after.
1 parent e4afa28 commit ee5491a

6 files changed

+60
-68
lines changed

docs/DebuggingTheCompiler.md

+22-38
Original file line numberDiff line numberDiff line change
@@ -212,47 +212,31 @@ Often it is not sufficient to dump the SIL at the beginning or end of
212212
the optimization pipeline. The SILPassManager supports useful options
213213
to dump the SIL also between pass runs.
214214

215-
The SILPassManager's SIL dumping options vary along two orthogonal
216-
functional axes:
215+
A short (non-exhaustive) list of SIL printing options:
217216

218-
1. Options that control if functions/modules are printed.
219-
2. Options that filter what is printed at those points.
217+
* `-Xllvm '-sil-print-function=SWIFT_MANGLED_NAME'`: Print the specified
218+
function after each pass which modifies the function. Note that for module
219+
passes, the function is printed if the pass changed _any_ function (the
220+
pass manager doesn't know which functions a module pass has changed).
221+
Multiple functions can be specified as a comma separated list.
220222

221-
One generally always specifies an option of type 1 and optionally adds
222-
an option of type 2 to filter the output.
223+
* `-Xllvm '-sil-print-functions=NAME'`: Like `-sil-print-function`, except that
224+
functions are selected if NAME is _contained_ in their mangled names.
223225

224-
A short (non-exhaustive) list of type 1 options:
226+
* `-Xllvm -sil-print-all`: Print all functions when ever a function pass
227+
modifies a function and print the entire module if a module pass modifies
228+
the SILModule.
225229

226-
* `-Xllvm -sil-print-all`: Print functions/modules when ever a
227-
function pass modifies a function and Print the entire module
228-
(modulo filtering) if a module pass modifies a SILModule.
230+
* `-Xllvm -sil-print-around=$PASS_NAME`: Print the SIL before and after a pass
231+
with name `$PASS_NAME` runs on a function or module.
232+
By default it prints the whole module. To print only specific functions, add
233+
`-sil-print-function` and/or `-sil-print-functions`.
229234

230-
A short (non-exhaustive) list of type 2 options:
235+
* `-Xllvm -sil-print-before=$PASS_NAME`: Like `-sil-print-around`, but prints
236+
the SIL only _before_ the specfied pass runs.
231237

232-
* `-Xllvm -sil-print-around=$PASS_NAME`: Print a function/module
233-
before and after a function pass with name `$PASS_NAME` runs on a
234-
function/module or dump a module before a module pass with name
235-
`$PASS_NAME` runs on a module.
236-
237-
* `-Xllvm -sil-print-before=$PASS_NAME`: Print a function/module
238-
before a function pass with name `$PASS_NAME` runs on a
239-
function/module or dump a module before a module pass with name
240-
`$PASS_NAME` runs on a module. NOTE: This happens even without
241-
sil-print-all set!
242-
243-
* `-Xllvm -sil-print-after=$PASS_NAME`: Print a function/module
244-
after a function pass with name `$PASS_NAME` runs on a
245-
function/module or dump a module before a module pass with name
246-
`$PASS_NAME` runs on a module.
247-
248-
* `-Xllvm '-sil-print-only-function=SWIFT_MANGLED_NAME'`: When ever
249-
one would print a function/module, only print the given function.
250-
251-
These options together allow one to visualize how a
252-
SILFunction/SILModule is optimized by the optimizer as each
253-
optimization pass runs easily via formulations like:
254-
255-
swiftc -Xllvm '-sil-print-only-function=$myMainFunction' -Xllvm -sil-print-all
238+
* `-Xllvm -sil-print-after=$PASS_NAME`: Like `-sil-print-around`, but prints
239+
the SIL only _after_ the specfied pass did run.
256240

257241
NOTE: This may emit a lot of text to stderr, so be sure to pipe the
258242
output to a file.
@@ -586,11 +570,11 @@ it's quite easy to do this manually:
586570

587571
2. Get the SIL before and after the bad optimization.
588572

589-
a. Add the compiler options
590-
`-Xllvm -sil-print-all -Xllvm -sil-print-only-function='<function>'`
573+
a. Add the compiler option
574+
`-Xllvm -sil-print-function='<function>'`
591575
where `<function>` is the function name (including the preceding `$`).
592576
For example:
593-
`-Xllvm -sil-print-all -Xllvm -sil-print-only-function='$s4test6testityS2iF'`.
577+
`-Xllvm -sil-print-function='$s4test6testityS2iF'`.
594578
Again, the output can be large, so it's best to redirect stderr to a file.
595579
b. From the output, copy the SIL of the function *before* the bad
596580
run into a separate file and the SIL *after* the bad run into a file.

lib/SILOptimizer/PassManager/PassManager.cpp

+33-25
Original file line numberDiff line numberDiff line change
@@ -60,11 +60,11 @@ llvm::cl::opt<std::string> SILBreakOnPass(
6060
llvm::cl::desc("Break before running a particular function pass"));
6161

6262
llvm::cl::list<std::string>
63-
SILPrintOnlyFun("sil-print-only-function", llvm::cl::CommaSeparated,
63+
SILPrintFunction("sil-print-function", llvm::cl::CommaSeparated,
6464
llvm::cl::desc("Only print out the sil for this function"));
6565

6666
llvm::cl::opt<std::string>
67-
SILPrintOnlyFuns("sil-print-only-functions", llvm::cl::init(""),
67+
SILPrintFunctions("sil-print-functions", llvm::cl::init(""),
6868
llvm::cl::desc("Only print out the sil for the functions "
6969
"whose name contains this substring"));
7070

@@ -157,13 +157,23 @@ static llvm::cl::opt<DebugOnlyPassNumberOpt, true,
157157
llvm::cl::location(DebugOnlyPassNumberOptLoc),
158158
llvm::cl::ValueRequired);
159159

160-
static bool doPrintBefore(SILTransform *T, SILFunction *F) {
161-
if (!SILPrintOnlyFun.empty() && F && SILPrintOnlyFun.end() ==
162-
std::find(SILPrintOnlyFun.begin(), SILPrintOnlyFun.end(), F->getName()))
160+
static bool isFunctionSelectedForPrinting(SILFunction *F) {
161+
if (!SILPrintFunction.empty() && SILPrintFunction.end() ==
162+
std::find(SILPrintFunction.begin(), SILPrintFunction.end(), F->getName()))
163+
return false;
164+
165+
if (!F->getName().contains(SILPrintFunctions))
163166
return false;
164167

165-
if (!SILPrintOnlyFuns.empty() && F &&
166-
!F->getName().contains(SILPrintOnlyFuns))
168+
return true;
169+
}
170+
171+
static bool functionSelectionEmpty() {
172+
return SILPrintFunction.empty() && SILPrintFunctions.empty();
173+
}
174+
175+
static bool doPrintBefore(SILTransform *T, SILFunction *F) {
176+
if (F && !isFunctionSelectedForPrinting(F))
167177
return false;
168178

169179
auto MatchFun = [&](const std::string &Str) -> bool {
@@ -173,21 +183,20 @@ static bool doPrintBefore(SILTransform *T, SILFunction *F) {
173183
if (SILPrintBefore.end() !=
174184
std::find_if(SILPrintBefore.begin(), SILPrintBefore.end(), MatchFun))
175185
return true;
186+
if (!SILPrintBefore.empty())
187+
return false;
176188

177189
if (SILPrintAround.end() !=
178190
std::find_if(SILPrintAround.begin(), SILPrintAround.end(), MatchFun))
179191
return true;
192+
if (!SILPrintAround.empty())
193+
return false;
180194

181195
return false;
182196
}
183197

184-
static bool doPrintAfter(SILTransform *T, SILFunction *F, bool Default) {
185-
if (!SILPrintOnlyFun.empty() && F && SILPrintOnlyFun.end() ==
186-
std::find(SILPrintOnlyFun.begin(), SILPrintOnlyFun.end(), F->getName()))
187-
return false;
188-
189-
if (!SILPrintOnlyFuns.empty() && F &&
190-
!F->getName().contains(SILPrintOnlyFuns))
198+
static bool doPrintAfter(SILTransform *T, SILFunction *F, bool PassChangedSIL) {
199+
if (F && !isFunctionSelectedForPrinting(F))
191200
return false;
192201

193202
auto MatchFun = [&](const std::string &Str) -> bool {
@@ -197,12 +206,16 @@ static bool doPrintAfter(SILTransform *T, SILFunction *F, bool Default) {
197206
if (SILPrintAfter.end() !=
198207
std::find_if(SILPrintAfter.begin(), SILPrintAfter.end(), MatchFun))
199208
return true;
209+
if (!SILPrintAfter.empty())
210+
return false;
200211

201212
if (SILPrintAround.end() !=
202213
std::find_if(SILPrintAround.begin(), SILPrintAround.end(), MatchFun))
203214
return true;
215+
if (!SILPrintAround.empty())
216+
return false;
204217

205-
return Default;
218+
return PassChangedSIL && (SILPrintAll || !functionSelectionEmpty());
206219
}
207220

208221
static bool isDisabled(SILTransform *T, SILFunction *F = nullptr) {
@@ -221,16 +234,12 @@ static bool isDisabled(SILTransform *T, SILFunction *F = nullptr) {
221234
}
222235

223236
static void printModule(SILModule *Mod, bool EmitVerboseSIL) {
224-
if (SILPrintOnlyFun.empty() && SILPrintOnlyFuns.empty()) {
237+
if (functionSelectionEmpty()) {
225238
Mod->dump();
226239
return;
227240
}
228241
for (auto &F : *Mod) {
229-
if (!SILPrintOnlyFun.empty() && SILPrintOnlyFun.end() !=
230-
std::find(SILPrintOnlyFun.begin(), SILPrintOnlyFun.end(), F.getName()))
231-
F.dump(EmitVerboseSIL);
232-
233-
if (!SILPrintOnlyFuns.empty() && F.getName().contains(SILPrintOnlyFuns))
242+
if (isFunctionSelectedForPrinting(&F))
234243
F.dump(EmitVerboseSIL);
235244
}
236245
}
@@ -472,7 +481,7 @@ void SILPassManager::runPassOnFunction(unsigned TransIdx, SILFunction *F) {
472481
}
473482

474483
// If this pass invalidated anything, print and verify.
475-
if (doPrintAfter(SFT, F, CurrentPassHasInvalidated && SILPrintAll)) {
484+
if (doPrintAfter(SFT, F, CurrentPassHasInvalidated)) {
476485
dumpPassInfo("*** SIL function after ", TransIdx);
477486
F->dump(getOptions().EmitVerboseSIL);
478487
}
@@ -617,8 +626,7 @@ void SILPassManager::runModulePass(unsigned TransIdx) {
617626
}
618627

619628
// If this pass invalidated anything, print and verify.
620-
if (doPrintAfter(SMT, nullptr,
621-
CurrentPassHasInvalidated && SILPrintAll)) {
629+
if (doPrintAfter(SMT, nullptr, CurrentPassHasInvalidated)) {
622630
dumpPassInfo("*** SIL module after", TransIdx);
623631
printModule(Mod, Options.EmitVerboseSIL);
624632
}
@@ -731,7 +739,7 @@ SILPassManager::~SILPassManager() {
731739
}
732740

733741
void SILPassManager::notifyOfNewFunction(SILFunction *F, SILTransform *T) {
734-
if (doPrintAfter(T, F, SILPrintAll)) {
742+
if (doPrintAfter(T, F, /*PassChangedSIL*/ true)) {
735743
dumpPassInfo("*** New SIL function in ", T, F);
736744
F->dump(getOptions().EmitVerboseSIL);
737745
}

test/SILOptimizer/definite-init-wrongscope.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
// RUN: %target-swift-frontend -primary-file %s -Onone -emit-sil -Xllvm \
22
// RUN: -sil-print-after=raw-sil-inst-lowering -Xllvm \
3-
// RUN: -sil-print-only-functions=$s3del1MC4fromAcA12WithDelegate_p_tKcfc \
3+
// RUN: -sil-print-functions=$s3del1MC4fromAcA12WithDelegate_p_tKcfc \
44
// RUN: -Xllvm -sil-print-debuginfo -o /dev/null -module-name del 2>&1 | %FileCheck %s
55

66
public protocol DelegateA {}

test/SILOptimizer/di-conditional-destroy-scope.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
// RUN: %target-swift-frontend -emit-sil %s -Onone -Xllvm \
22
// RUN: -sil-print-after=raw-sil-inst-lowering -Xllvm \
3-
// RUN: -sil-print-only-functions=$s2fs36RecursibleDirectoryContentsGeneratorC4path10fileSystemAcA12AbsolutePathV_AA04FileH0_ptKc33_F8B132991B28208F48606E87DC165393Llfc \
3+
// RUN: -sil-print-functions=$s2fs36RecursibleDirectoryContentsGeneratorC4path10fileSystemAcA12AbsolutePathV_AA04FileH0_ptKc33_F8B132991B28208F48606E87DC165393Llfc \
44
// RUN: -Xllvm -sil-print-debuginfo -o /dev/null 2>&1 | %FileCheck %s
55

66
// REQUIRES: objc_interop

test/SILOptimizer/stack-nesting-wrong-scope.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
// RUN: %target-swift-frontend -emit-sil %s -Onone -Xllvm \
22
// RUN: -sil-print-after=allocbox-to-stack -Xllvm \
3-
// RUN: -sil-print-only-functions=$s3red19ThrowAddrOnlyStructV016throwsOptionalToG0ACyxGSgSi_tcfC \
3+
// RUN: -sil-print-functions=$s3red19ThrowAddrOnlyStructV016throwsOptionalToG0ACyxGSgSi_tcfC \
44
// RUN: -Xllvm -sil-print-debuginfo -o %t -module-name red 2>&1 | %FileCheck %s
55

66
// CHECK: bb{{[0-9]+}}(%{{[0-9]+}} : @owned $Error):

utils/swift-autocomplete.bash

+2-2
Original file line numberDiff line numberDiff line change
@@ -58,8 +58,8 @@ _swift_complete()
5858
-sil-print-pass-name \
5959
-sil-print-pass-time \
6060
-sil-opt-pass-count \
61-
-sil-print-only-function \
62-
-sil-print-only-functions \
61+
-sil-print-function \
62+
-sil-print-functions \
6363
-sil-print-before \
6464
-sil-print-after \
6565
-sil-print-around \

0 commit comments

Comments
 (0)