From f1149a94031a4470c97992581eaa74217322816a Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Sun, 7 Jul 2024 18:24:30 -0500 Subject: [PATCH 01/18] Add a first tutorial --- docs/Project.toml | 2 + docs/src/index.md | 107 +++----------- docs/src/tutorials/invalidations.md | 208 ++++++++++++++++++++++++++++ 3 files changed, 229 insertions(+), 88 deletions(-) create mode 100644 docs/src/tutorials/invalidations.md diff --git a/docs/Project.toml b/docs/Project.toml index c3c736ff..cb4daa2d 100644 --- a/docs/Project.toml +++ b/docs/Project.toml @@ -1,5 +1,6 @@ [deps] AbstractTrees = "1520ce14-60c1-5f80-bbc7-55ef81b5835c" +Cthulhu = "f68482b8-f384-11e8-15f7-abe071a5a75f" Documenter = "e30172f5-a6a5-5a46-863b-614d45cd2de4" JET = "c3a54625-cd67-489e-a8e7-0a5a0ff4e31b" MethodAnalysis = "85b6ec6f-f7df-4429-9514-a64bcd9ee824" @@ -8,6 +9,7 @@ SnoopCompile = "aa65fe97-06da-5843-b5b1-d5d13cad87d2" [compat] AbstractTrees = "0.4" +Cthulhu = "2" Documenter = "1" JET = "0.9" MethodAnalysis = "0.4" diff --git a/docs/src/index.md b/docs/src/index.md index baf2e9e8..94e97362 100644 --- a/docs/src/index.md +++ b/docs/src/index.md @@ -1,101 +1,32 @@ # SnoopCompile.jl -SnoopCompile "snoops" on the Julia compiler, causing it to record the -functions and argument types it's compiling. From these lists of methods, -you can generate lists of `precompile` directives that may reduce the latency between -loading packages and using them to do "real work." +Julia is fast, but its execution speed depends on optimizing code through *compilation*. Code must be compiled before you can use it, and unfortunately compilation is slow. This can cause *latency* the first time you use code: this latency is often called *time-to-first-plot* (TTFP) or more generally *time-to-first-execution* (TTFX). If something feels slow the first time you use it, and fast thereafter, you're probably experiencing the latency of compilation. -SnoopCompile can also detect and analyze *method cache invalidations*, -which occur when new method definitions alter dispatch in a way that forces Julia to discard previously-compiled code. -Any later usage of invalidated methods requires recompilation. -Invalidation can trigger a domino effect, in which all users of invalidated code also become invalidated, propagating all the way back to the top-level call. -When a source of invalidation can be identified and either eliminated or mitigated, -you can reduce the amount of work that the compiler needs to repeat and take better advantage of precompilation. +Modern versions of Julia can store compiled code to disk (*precompilation*) to reduce or eliminate latency. Users and developers who are interested in reducing TTFX should first head to [PrecompileTools](https://github.com/JuliaLang/PrecompileTools.jl), read its documentation thoroughly, and try using it to solve latency problems. -Finally, SnoopCompile interacts with other important diagnostics and debugging tools in the Julia ecosystem. -For example, the combination of SnoopCompile and [JET](https://github.com/aviatesk/JET.jl) allows you to analyze an entire call-chain for -potential errors; see the page on [JET integration](@ref JET) for more information. +This package, **SnoopCompile**, should be considered when: -## Background +- precompilation doesn't reduce latency as much as you wish +- precompilation "works," but only in isolation: as soon as you load (certain) additional packages, TTFX is bad again +- you're wondering if you can reduce the amount of time needed to precompile your package and/or the size of the shared library files -Julia uses -[Just-in-time (JIT) compilation](https://en.wikipedia.org/wiki/Just-in-time_compilation) to -generate the code that runs on your CPU. -Broadly speaking, there are two major steps: *inference* and *code generation*. -Inference is the process of determining the type of each object, which in turn -determines which specific methods get called; once type inference is complete, -code generation performs optimizations and ultimately generates the assembly -language (native code) used on CPUs. -Some aspects of this process are documented [here](https://docs.julialang.org/en/v1/devdocs/eval/). +In other words, SnoopCompile is a diagonostic package that helps reveal the causes of latency. Historically, it proceeded PrecompileTools, and indeed PrecompileTools was split out from SnoopCompile. Today, SnoopCompile is generally needed only when PrecompileTools fails to deliver the desired benefits. -Every time you load a package in a fresh Julia session, the methods you use need -to be JIT-compiled, and this contributes to the latency of using the package. -In some circumstances, you can cache the results of compilation to files to -reduce the latency when your package is used. These files are the the `*.ji` and -`*.so` files that live in the `compiled` directory of the Julia depot, usually -located at `~/.julia/compiled`. However, if these files become large, loading -them can be another source for latency. Julia needs time both to load and -validate the cached compiled code. Minimizing the latency of using a package -involves focusing on caching the compilation of code that is both commonly used -and takes time to compile. +## SnoopCompile analysis modes -This is called *precompilation*. Julia is able to save inference results in the -`*.ji` files and ([since Julia -1.9](https://julialang.org/blog/2023/04/julia-1.9-highlights/#caching_of_native_code)) -native code in the `*.so` files, and thus precompilation can eliminate the time -needed for type inference and native code compilation (though what does get -saved can sometimes be invalidated by loading other packages). +SnoopCompile "snoops" on the Julia compiler, collecting information that may be useful to developers. Here are some of the things you can do with SnoopCompile: -SnoopCompile is designed to try to allow you to analyze the costs of JIT-compilation, identify -key bottlenecks that contribute to latency, and set up `precompile` directives to see whether -it produces measurable benefits. +- diagnose *invalidations*, cases where Julia must throw away previously-compiled code (see [Tutorial on `@snoop_invalidations`](@ref)) +- trace *inference*, to learn what code is being newly (or freshly) analyzed in an early stage of the compilation pipeline ([Tutorial on `@snoop_inference`](@ref)) +- trace *code generation by LLVM*, a late stage in the compilation pipeline ([Tutorial on `@snoop_llvm`](@ref)) +- reveal methods with excessive numbers of compiler-generated specializations, a.k.a.*profile-guided despecialization* ([Tutorial on PGDS](@ref)) +- integrate with tools like [JET](https://github.com/aviatesk/JET.jl) to help reduce the risk that your lovingly-precompiled code will be invalidated by loading other packages ([Tutorial on JET integration](@ref)) -## Who should use this package - -SnoopCompile is intended primarily for package *developers* who want to improve the -experience for their users. -Because the results of SnoopCompile are typically stored in the `*.ji` precompile files, -users automatically get the benefit of any latency reductions achieved by adding -`precompile` directives to the source code of your package. - -[PackageCompiler](https://github.com/JuliaLang/PackageCompiler.jl) is an alternative -that *non-developer users* may want to consider for their own workflow. -It builds an entire system image (Julia + a set of selected packages) and caches both the -results of type inference and the native code. -Typically, PackageCompiler reduces latency more than just "plain" `precompile` directives. -However, PackageCompiler does have significant downsides, of which the largest is that -it is incompatible with package updates--any packages built into your system image -cannot be updated without rebuilding the entire system. -Particularly for people who develop or frequently update their packages, the downsides of -PackageCompiler may outweigh its benefits. - -Finally, another alternative for reducing latency without any modifications -to package files is [Revise](https://github.com/timholy/Revise.jl). -It can be used in conjunction with SnoopCompile. - -## [A note on Julia versions and the recommended workflow](@id workflow) +## Background information -SnoopCompile is closely intertwined with Julia's own internals. -Some "data collection" and analysis features are available only on newer versions of Julia. -In particular, some of the most powerful tools were made possible through several additions made in Julia 1.6; -SnoopCompile just exposes these tools in convenient form. +Confused? Some of the core concepts are explained in [Understanding SnoopCompile and Julia's compilation pipeline](@ref). -If you're a developer looking to reduce the latency of your packages, you are *strongly* -encouraged to use SnoopCompile on Julia 1.6 or later. The fruits of your labors will often -reduce latency even for users of earlier Julia versions, but your ability to understand -what changes need to be made will be considerably enhanced by using the latest tools. - -For developers who can use Julia 1.6+, the recommended sequence is: - -1. Check for [invalidations](@ref), and if egregious make fixes before proceeding further -2. Record inference data with [`@snoop_inference`](@ref). Analyze the data to: - + adjust method specialization in your package or its dependencies (see [pgds](@ref)) - + fix problems in [inferrability](@ref) - + add [precompilation](@ref) - -Under 2, the first two sub-points can often be done at the same time; the last item is best done as a final step, because the specific -precompile directives needed depend on the state of your code, and a few fixes in specialization -and/or type inference can alter or even decrease the number of necessary precompile directives. +## Who should use this package -Although there are other tools within SnoopCompile available, most developers can probably stop after the steps above. -The documentation will describe the tools in this order, followed by descriptions of additional and older tools. +SnoopCompile is intended primarily for package *developers* who want to improve the +experience for their users. It is also recommended for users who are willing to "dig deep" and understand why packages they depend on have high latency. **Your experience with latency may be personal, as it can depend on the specific combination of packages you load.** If latency troubles you, don't make the assumption that it must be unfixable: you might be the first person affected by that specific cause of latency. diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md new file mode 100644 index 00000000..8cd63c75 --- /dev/null +++ b/docs/src/tutorials/invalidations.md @@ -0,0 +1,208 @@ +# Tutorial on `@snoop_invalidations` + +Invalidations are a product of unexpected interactions between packages, or between packages and Base Julia. We'll illustrate invalidations by creating two packages, where loading the second package invalidates some methods in the first one. This example is inspired by the [darts problem](https://exercism.org/tracks/julia/exercises/darts) on Exercism. + +While [PkgTemplates](https://github.com/JuliaCI/PkgTemplates.jl) is recommended for creating packages, to keep dependencies minimal here we'll just use the basic capabilities in `Pkg`. +While you would generally use an editor to create the package code, here we'll do it programmatically so that you can just copy/paste the code below and expect to get the same outcome. + +## Add SnoopCompileCore, SnoopCompile, and helper packages to your environment + +Here, we'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/). You can, of course, perform this analysis in a different environment if you prefer. + +```@repl +using Pkg +Pkg.add(["SnoopCompileCore", "SnoopCompile", "AbstractTrees", "Cthulhu"]) +``` + +SnoopCompileCore is a tiny package with no dependencies; it's used for collecting data, and it has been designed in such a way that it cannot cause any invalidations of its own. +SnoopCompile is a much larger package that performs analysis on the data collected by SnoopCompileCore; loading SnoopCompile can (and does) trigger invalidations. +Consequently, you're urged to always collect data with just SnoopCompileCore loaded, +and wait to load SnoopCompile until after you've finished collecting the data. + +[Cthulhu](https://github.com/JuliaDebug/Cthulhu.jl) is a companion package you'll want for any analysis of invalidations. Here we'll also use the [AbstractTrees](https://github.com/JuliaCollections/AbstractTrees.jl) package for simple printing. + +## Creating the demonstration packages + +Here are the steps executed by the code below: +- navigate to a temporary directory and create both packages +- make the first package (`ScoreDarts`) depend on [PrecompileTools](https://github.com/JuliaLang/PrecompileTools.jl) (we're interested in reducing latency!) +- make the second package (`MoreDarts`) depend on the first one (`ScoreDarts`) + +```@repl tutorial-invalidations +cd(mktempdir()) +using Pkg +Pkg.generate("ScoreDarts"); +Pkg.activate("ScoreDarts") +Pkg.add("PrecompileTools") +Pkg.generate("MoreDarts"); +Pkg.activate("MoreDarts") +Pkg.develop(PackageSpec(path=joinpath(pwd(), "ScoreDarts"))); +``` + +Now it's time to create the code for `ScoreDarts`. The code below does the following: +- creates different `TargetCircle`s that one can hit +- assigns the correct score for each target circle +- creates a convenience function that parses a sequence of characters, converts them into target circles, and then adds the scores +- precompiles the code to reduce latency on first use + +```@repl tutorial-invalidations +write(joinpath("ScoreDarts", "src", "ScoreDarts.jl"), """ + module ScoreDarts + + using PrecompileTools + + @enum TargetCircle outer middle inner + + score(circle::TargetCircle) = circle == outer ? 1 : + circle == middle ? 5 : 10 + + # Add up the scores (this is much simpler than Julia's own `sum(score, turns)`) + function tallyscores(turns) + s = 0 + for t in turns + s += score(t) + end + return s + end + + # Create a dictionary that maps characters to circles + const chardict = Dict{Char,Any}('o' => outer, + 'm' => middle, + 'i' => inner) + + # Create a function that translates 'o', 'm', and 'i' into their different circles + # and then adds the scores + function scoresequence(chars) + turns = [] + for c in chars + push!(turns, chardict[c]) + end + return tallyscores(turns) + end + + # Precompile `scoresequence`: + @compile_workload begin + scoresequence("omi") + end + + end + """) +``` + +`MoreDarts` will extend `ScoreDarts` in an important way: we acknowledge that maybe the target wasn't hit at all, in which case the score is zero. Because we can't add to the `TargetCircle` `@enum`, let's just create a new type: + +```@repl tutorial-invalidations +write(joinpath("MoreDarts", "src", "MoreDarts.jl"), """ + module MoreDarts + + using ScoreDarts + + struct MissedTarget end + const miss = MissedTarget() + + # Add a new `score` method: + ScoreDarts.score(::MissedTarget) = 0 + + # Use `'x'` to indicate a miss + push!(ScoreDarts.chardict, 'x' => miss) + + end + """) +``` + +Now we're ready! As a + +## Recording invalidations + +Here are the steps executed by the code below +- load `SnoopCompileCore` +- load `MoreDarts` while recording invalidations +- load `SnoopCompile` and `AbstractTrees` for analysis + +```@repl tutorial-invalidations +using SnoopCompileCore +invs = @snoop_invalidations using MoreDarts; +using SnoopCompile, AbstractTrees +``` + +## Analyzing invalidations + +Now we're ready to see what, if anything, got invalidated: + +```@repl tutorial-invalidations +trees = invalidation_trees(invs) +``` + +This has only one "tree" of invalidations, from a single cause described in the top line: +adding the new method `score(::MoreDarts.MissedTarget)`. + +`trees` is a `Vector` so we can index it; we can also extract the list of "root" targets for invalidation, and then see what happened using `@ascend`: + +```@repl tutorial-invalidations +sig, root = trees[1].mt_backedges[end]; +sig +``` + +!!! note + `mt_backedges` stands for "method-table backedges." SnoopCompile has a second type of invalidation, just called `backedges`. With these, there is no `sig`, and so you'll use just `root = trees[i].backedges[j]`. + +Here you can see the problematic `sig`nature: `score` was called on an object of (inferred) type `Any`. Because Julia couldn't infer the actual type, this code was vulnerable, and insertion of the new `score` method "exploited" this vulnerability. + +`root` itself indicates that `tallyscores` is the "victim" of invalidation, but this is not the full extent of what got invalidated: + +```@repl tutorial-invalidations +print_tree(root) +``` + +We can see that the invalidation propagated all the way up to `scoresequence(::String)`. +In general, roots with lots of "children" deserve the greatest attention. + +In most cases, Cthulhu's `ascend` provides a more useful tool for identifying the cause of invalidations: + +```julia +julia> using Cthulhu + +julia> ascend(root) +Choose a call for analysis (q to quit): + > tallyscores(::Vector{Any}) + scoresequence(::String) +``` + +This is an interactive REPL-menu, described more completely (including a Youtube video) at [ascend](https://github.com/JuliaDebug/Cthulhu.jl?tab=readme-ov-file#usage-ascend). + +## Why the invalidations occur + +Evidently, `turns` is a `Vector{Any}`, and this means that `tallyscores` can't guess what kind of objects will come out of it, and thus it can't guess what kind of objects are passed into `score`. However, Julia notices that `score` only supports `TargetCircle`s, and thus it *speculatively* guesses that all the objects in `turns` will likely be `TargetCircle`s. It thus compiles the code to (1) check that indeed the object really is a `TargetCircle`, and (2) call `score` for a `TargetCircle`. + +This speculative compilation based on what methods and types are "in the world" at the time of compilation (sometimes called *world-splitting*) can be a useful performance optimization. Looking up applicable methods while the code is running can be slow, and any work that can be done ahead of time can lead to significant savings. + +Unfortunately, when the "world" changes--through the addition of a new `score` method for a new type--the old compiled code is no longer valid. The code must be recompiled, accounting for this new possibility, before it can be safely run again. Thus, invalidations are a product of compilation happening in a "smaller" world than the one in the user's running session. + +## Fixing invalidations + +In broad strokes, there are two ways to prevent invalidation. The first is to improve the quality of type-inference. For example, in + +```julia + function scoresequence(chars) + turns = [] +``` +that unqualified `[]` is what makes `turns` a `Vector{Any}`. If we didn't need to allow extensions of `scoresequence`, we could have said `turns = TargetCircle[]` and that would have required that all elements of `turns` be `TargetCircles`. This would "bullet-proof" `scoresequence` from invalidation, at the cost of preventing extensions such as `MoreDarts`. (The package developer(s) get to decide whether this is a good or bad thing--in this case, one could argue that the failure to account for `miss` is a bug in `ScoreDarts` and is better fixed there.) + +!!! tip + Many vulnerabilities can be fixed by improving inference. In complex code, it's easy to unwittingly write things in ways that defeat Julia's type inference. Not only does fixing the inference problem prevent invalidation, it typically makes your code run faster too. When fixing invalidations, your first thought should always be, "can I improve inference in this code without making compromises in its functionality?" + +The second option is to prevent Julia's speculative optimization: one could replace the direct call to `score` with an `invokelatest(score, turn)`: + +```julia +function tallyscores(turns) + s = 0 + for t in turns + s += invokelatest(score, t) + end + return s + end +``` + +This forces Julia to always look up the appropriate method of `score` while the code is running, and thus prevents the speculative optimizations that leave the code vulnerable to invalidation. If you re-run the above demonstration, using `invokelatest` as shown, then `trees` will be empty--no invalidations! However, the cost is that your code may run somewhat more slowly. + +More details about fixing invalidations can be found in [Techniques for fixing inference problems](@ref). From c62b31775c41fd0989f2e31e6814359485530348 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Fri, 12 Jul 2024 08:51:29 -0500 Subject: [PATCH 02/18] Polish the invalidations tutorial Switches from darts to blackjack. There were several features about the darts example that made for poor teaching, including the gap between `@enum` and a `struct` and the artificiality of not extending the base package. Also polishes the explanations and improves how things are named. --- docs/src/tutorials/invalidations.md | 242 ++++++++++++++++------------ 1 file changed, 143 insertions(+), 99 deletions(-) diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md index 8cd63c75..71cbbb95 100644 --- a/docs/src/tutorials/invalidations.md +++ b/docs/src/tutorials/invalidations.md @@ -1,131 +1,133 @@ # Tutorial on `@snoop_invalidations` -Invalidations are a product of unexpected interactions between packages, or between packages and Base Julia. We'll illustrate invalidations by creating two packages, where loading the second package invalidates some methods in the first one. This example is inspired by the [darts problem](https://exercism.org/tracks/julia/exercises/darts) on Exercism. +## What are invalidations? -While [PkgTemplates](https://github.com/JuliaCI/PkgTemplates.jl) is recommended for creating packages, to keep dependencies minimal here we'll just use the basic capabilities in `Pkg`. -While you would generally use an editor to create the package code, here we'll do it programmatically so that you can just copy/paste the code below and expect to get the same outcome. +Invalidations result from unanticipated interactions between packages, or between packages and Base Julia. They are essential for ensuring that Julia runs fast and behaves correctly, but they make latency worse by "throwing away" precompiled code. +A good developer can often design packages to minimize the number and/or impact of invalidations. -## Add SnoopCompileCore, SnoopCompile, and helper packages to your environment +## Learning to observe, diagnose, and fix invalidations -Here, we'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/). You can, of course, perform this analysis in a different environment if you prefer. +We'll illustrate invalidations by creating two packages, where loading the second package invalidates some code that was compiled in the first one. We'll then go over approaches for "fixing" invalidations (i.e., preventing them from occuring). + +!!! tip + Since SnoopCompile's tools are interactive, you are strongly encouraged to try these examples yourself as you read along. + +### Add SnoopCompileCore, SnoopCompile, and helper packages to your environment + +Here, we'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/). (With the exception of `AbstractTrees`, these "developer tool" packages should not be added to the Project file of any real packages unless you're extending the tool itself.) ```@repl using Pkg Pkg.add(["SnoopCompileCore", "SnoopCompile", "AbstractTrees", "Cthulhu"]) ``` -SnoopCompileCore is a tiny package with no dependencies; it's used for collecting data, and it has been designed in such a way that it cannot cause any invalidations of its own. -SnoopCompile is a much larger package that performs analysis on the data collected by SnoopCompileCore; loading SnoopCompile can (and does) trigger invalidations. -Consequently, you're urged to always collect data with just SnoopCompileCore loaded, -and wait to load SnoopCompile until after you've finished collecting the data. - -[Cthulhu](https://github.com/JuliaDebug/Cthulhu.jl) is a companion package you'll want for any analysis of invalidations. Here we'll also use the [AbstractTrees](https://github.com/JuliaCollections/AbstractTrees.jl) package for simple printing. +### Create the demonstration packages -## Creating the demonstration packages +We're going to implement a toy version of the card game [blackjack](), where players take cards with the aim of collecting 21 points. The higher you go the better, *unless* you go over 21 points, in which case you "go bust" (i.e., you lose). Because our real goal is to illustrate invalidations, we'll create a "blackjack ecosystem" that involves an interaction between two packages. -Here are the steps executed by the code below: +While [PkgTemplates](https://github.com/JuliaCI/PkgTemplates.jl) is recommended for creating packages, here we'll just use the basic capabilities in `Pkg`. +To create the (empty) packages, the code below executes the following steps: - navigate to a temporary directory and create both packages -- make the first package (`ScoreDarts`) depend on [PrecompileTools](https://github.com/JuliaLang/PrecompileTools.jl) (we're interested in reducing latency!) -- make the second package (`MoreDarts`) depend on the first one (`ScoreDarts`) +- make the first package (`Blackjack`) depend on [PrecompileTools](https://github.com/JuliaLang/PrecompileTools.jl) (we're interested in reducing latency!) +- make the second package (`BlackjackFacecards`) depend on the first one (`Blackjack`) ```@repl tutorial-invalidations cd(mktempdir()) using Pkg -Pkg.generate("ScoreDarts"); -Pkg.activate("ScoreDarts") +Pkg.generate("Blackjack"); +Pkg.activate("Blackjack") Pkg.add("PrecompileTools") -Pkg.generate("MoreDarts"); -Pkg.activate("MoreDarts") -Pkg.develop(PackageSpec(path=joinpath(pwd(), "ScoreDarts"))); +Pkg.generate("BlackjackFacecards"); +Pkg.activate("BlackjackFacecards") +Pkg.develop(PackageSpec(path=joinpath(pwd(), "Blackjack"))); ``` -Now it's time to create the code for `ScoreDarts`. The code below does the following: -- creates different `TargetCircle`s that one can hit -- assigns the correct score for each target circle -- creates a convenience function that parses a sequence of characters, converts them into target circles, and then adds the scores -- precompiles the code to reduce latency on first use +Now it's time to create the code for `Blackjack`. Normally, you'd do this with an editor, but to make it reproducible here we'll use code to create these packages. The package code we'll create below defines the following: +- a `score` function to assign a numeric value to a card +- `tallyscore` which adds the total score for a hand of cards +- `playgame` which uses a simple strategy to decide whether to take another card from the deck and add it to the hand + +To reduce latency on first use, we then precompile `playgame`. In a real application, we'd also want a function to manage the `deck` of cards, but for brevity we'll omit this and do it manually. ```@repl tutorial-invalidations -write(joinpath("ScoreDarts", "src", "ScoreDarts.jl"), """ - module ScoreDarts +write(joinpath("Blackjack", "src", "Blackjack.jl"), """ + module Blackjack using PrecompileTools - @enum TargetCircle outer middle inner + export playgame - score(circle::TargetCircle) = circle == outer ? 1 : - circle == middle ? 5 : 10 + const deck = [] # the deck of cards that can be dealt - # Add up the scores (this is much simpler than Julia's own `sum(score, turns)`) - function tallyscores(turns) + # Compute the score of one card + score(card::Int) = card + + # Add up the score in a hand of cards + function tallyscores(cards) s = 0 - for t in turns - s += score(t) + for card in cards + s += score(card) end return s end - # Create a dictionary that maps characters to circles - const chardict = Dict{Char,Any}('o' => outer, - 'm' => middle, - 'i' => inner) - - # Create a function that translates 'o', 'm', and 'i' into their different circles - # and then adds the scores - function scoresequence(chars) - turns = [] - for c in chars - push!(turns, chardict[c]) + # Play the game! We use a simple strategy to decide whether to draw another card. + function playgame() + myhand = [] + while tallyscores(myhand) <= 14 && !isempty(deck) + push!(myhand, pop!(deck)) # "Hit me!" end - return tallyscores(turns) + myscore = tallyscores(myhand) + return myscore <= 21 ? myscore : "Busted" end - # Precompile `scoresequence`: - @compile_workload begin - scoresequence("omi") + # Precompile `playgame`: + @setup_workload begin + push!(deck, 8, 10) # initialize the deck + @compile_workload begin + playgame() + end end end """) ``` -`MoreDarts` will extend `ScoreDarts` in an important way: we acknowledge that maybe the target wasn't hit at all, in which case the score is zero. Because we can't add to the `TargetCircle` `@enum`, let's just create a new type: +Suppose you use `Blackjack` and like it, but you notice it doesn't support face cards. Perhaps you're nervous about contributing to the `Blackjack` package (you shouldn't be!), and so you decide to start your own package that extends its functionality. You create `BlackjackFacecards` to add scoring of the jack, queen, king, and ace (for simplicity we'll make the ace always worth 11): ```@repl tutorial-invalidations -write(joinpath("MoreDarts", "src", "MoreDarts.jl"), """ - module MoreDarts - - using ScoreDarts +write(joinpath("BlackjackFacecards", "src", "BlackjackFacecards.jl"), """ + module BlackjackFacecards - struct MissedTarget end - const miss = MissedTarget() + using Blackjack # Add a new `score` method: - ScoreDarts.score(::MissedTarget) = 0 - - # Use `'x'` to indicate a miss - push!(ScoreDarts.chardict, 'x' => miss) + Blackjack.score(card::Char) = card ∈ ('J', 'Q', 'K') ? 10 : + card == 'A' ? 11 : error(card, " not known") end """) ``` -Now we're ready! As a +!!! warning + Because `BlackjackFacecards` "owns" neither `Char` nor `score`, this is [piracy]((https://docs.julialang.org/en/v1/manual/style-guide/#Avoid-type-piracy-1)) and should generally be avoided. Piracy is one way to cause invalidations, but it's not the only one. `BlackjackFacecards` could avoid committing piracy by defining a `struct Facecard ... end` and defining `score(card::Facecard)` instead of `score(card::Char)`, but this would *not* fix the invalidations--all the factors described below are unchanged. + +Now we're ready! -## Recording invalidations +### Recording invalidations Here are the steps executed by the code below - load `SnoopCompileCore` -- load `MoreDarts` while recording invalidations +- load `Blackjack` and `BlackjackFacecards` while *recording invalidations* with the `@snoop_invalidations` macro. - load `SnoopCompile` and `AbstractTrees` for analysis ```@repl tutorial-invalidations using SnoopCompileCore -invs = @snoop_invalidations using MoreDarts; +invs = @snoop_invalidations using Blackjack, BlackjackFacecards; using SnoopCompile, AbstractTrees ``` -## Analyzing invalidations +### Analyzing invalidations Now we're ready to see what, if anything, got invalidated: @@ -133,76 +135,118 @@ Now we're ready to see what, if anything, got invalidated: trees = invalidation_trees(invs) ``` -This has only one "tree" of invalidations, from a single cause described in the top line: -adding the new method `score(::MoreDarts.MissedTarget)`. +This has only one "tree" of invalidations. `trees` is a `Vector` so we can index it: -`trees` is a `Vector` so we can index it; we can also extract the list of "root" targets for invalidation, and then see what happened using `@ascend`: +```@repl tutorial-invalidations +tree = trees[1] +``` + +Each tree stems from a single *cause* described in the top line. For this tree, the cause was adding the new method `score(::Char)` in `BlackjackFacecards`. + +Each *cause* is associated with one or more *victims* of invalidation, a list here named `mt_backedges`. Let's extract the final (and in this case, only) victim: ```@repl tutorial-invalidations -sig, root = trees[1].mt_backedges[end]; -sig +sig, victim = tree.mt_backedges[end]; ``` !!! note - `mt_backedges` stands for "method-table backedges." SnoopCompile has a second type of invalidation, just called `backedges`. With these, there is no `sig`, and so you'll use just `root = trees[i].backedges[j]`. + `mt_backedges` stands for "MethodTable backedges." In other cases you may see a second type of invalidation, just called `backedges`. With these, there is no `sig`, and so you'll use just `victim = tree.backedges[i]`. + +First let's look at the the problematic method `sig`nature: + +```@repl tutorial-invalidations +sig +``` -Here you can see the problematic `sig`nature: `score` was called on an object of (inferred) type `Any`. Because Julia couldn't infer the actual type, this code was vulnerable, and insertion of the new `score` method "exploited" this vulnerability. +This is a type-tuple, i.e., `Tuple{typeof(f), typesof(args)...}`. We see that `score` was called on an object of (inferred) type `Any`. Calling a function with unknown argument types makes code vulnerable to invalidation, and insertion of the new `score` method "exploited" this vulnerability. -`root` itself indicates that `tallyscores` is the "victim" of invalidation, but this is not the full extent of what got invalidated: +`victim` shows which compiled code got invalidated: ```@repl tutorial-invalidations -print_tree(root) +victim ``` -We can see that the invalidation propagated all the way up to `scoresequence(::String)`. -In general, roots with lots of "children" deserve the greatest attention. +But this is not the full extent of what got invalidated: -In most cases, Cthulhu's `ascend` provides a more useful tool for identifying the cause of invalidations: +```@repl tutorial-invalidations +print_tree(victim) +``` + +Invalidations propagate throughout entire call trees, here up to `playgame()`: anything that calls code that may no longer be correct is itself at risk for being incorrect. +In general, victims with lots of "children" deserve the greatest attention. + +While `print_tree` can be useful, Cthulhu's `ascend` is a far more powerful tool for gaining deeper insight: ```julia julia> using Cthulhu -julia> ascend(root) +julia> ascend(victim) Choose a call for analysis (q to quit): > tallyscores(::Vector{Any}) - scoresequence(::String) + playgame() ``` -This is an interactive REPL-menu, described more completely (including a Youtube video) at [ascend](https://github.com/JuliaDebug/Cthulhu.jl?tab=readme-ov-file#usage-ascend). +This is an interactive REPL-menu, described more completely (via text and video) at [ascend](https://github.com/JuliaDebug/Cthulhu.jl?tab=readme-ov-file#usage-ascend). + +### Why the invalidations occur -## Why the invalidations occur +`tallyscores` and `playgame` were compiled in `Blackjack`, a "world" where the `score` method defined in `BlackjackFacecards` does not yet exist. When you load the `BlackjackFacecards` package, Julia must ask itself: now that this new `score` method exists, am I certain that I would compile `tallyscores` the same way? If the answer is "no," Julia invalidates the old compiled code, and compiles a fresh version with full awareness of the new `score` method in `BlackjackFacecards`. -Evidently, `turns` is a `Vector{Any}`, and this means that `tallyscores` can't guess what kind of objects will come out of it, and thus it can't guess what kind of objects are passed into `score`. However, Julia notices that `score` only supports `TargetCircle`s, and thus it *speculatively* guesses that all the objects in `turns` will likely be `TargetCircle`s. It thus compiles the code to (1) check that indeed the object really is a `TargetCircle`, and (2) call `score` for a `TargetCircle`. +Why would the compilation of `tallyscores` change? Evidently, `cards` is a `Vector{Any}`, and this means that `tallyscores` can't guess what kind of object `card` might be, and thus it can't guess what kind of objects are passed into `score`. The crux of the invalidation is thus: +- when `Blackjack` is compiled, inference does not know which `score` method will be called. However, at the time of compilation the only `score` method is for `Int`. Thus Julia will reason that anything that isn't an `Int` is going to trigger an error anyway, and so you might as well optimize `tallyscore` expecting all cards to be `Int`s. (More information about how `tallyscores` gets optimized can be found in [World-splitting](@ref).) +- however, when `BlackjackFacecards` is loaded, suddenly there are two `score` methods supporting both `Int` and `Char`. Now Julia's guess that all `cards` will probably be `Int`s doesn't seem so likely to be true, and thus `tallyscores` should be recompiled. -This speculative compilation based on what methods and types are "in the world" at the time of compilation (sometimes called *world-splitting*) can be a useful performance optimization. Looking up applicable methods while the code is running can be slow, and any work that can be done ahead of time can lead to significant savings. +Thus, invalidations arise from optimization based on what methods and types are "in the world" at the time of compilation (sometimes called *world-splitting*). This form of optimization can have performance benefits, but it also leaves your code vulnerable to invalidation. -Unfortunately, when the "world" changes--through the addition of a new `score` method for a new type--the old compiled code is no longer valid. The code must be recompiled, accounting for this new possibility, before it can be safely run again. Thus, invalidations are a product of compilation happening in a "smaller" world than the one in the user's running session. +### Fixing invalidations -## Fixing invalidations +In broad strokes, there are three ways to prevent invalidation. The first is to ensure that +the full range of possibilties (the entire "world of code") is present before any compilation occurs. In this case, probably the best approach would be to merge the `BlackjackFacecards` package into `Blackjack` itself. Or, if you are a maintainer of the "Blackjack ecosystem" and have reasons for thinking that keeping the packages separate makes sense, you could alternatively move the `PrecompileTools` workload to `BlackjackFacecards`. Either approach should prevent the invalidations from occuring. -In broad strokes, there are two ways to prevent invalidation. The first is to improve the quality of type-inference. For example, in +The second way to prevent invalidations is to improve the inferability of the victim(s). If `Int` and `Char` really are the only possible kinds of cards, then in `playgame` it would be better to declare ```julia - function scoresequence(chars) - turns = [] +myhand = Union{Int,Char}[] ``` -that unqualified `[]` is what makes `turns` a `Vector{Any}`. If we didn't need to allow extensions of `scoresequence`, we could have said `turns = TargetCircle[]` and that would have required that all elements of `turns` be `TargetCircles`. This would "bullet-proof" `scoresequence` from invalidation, at the cost of preventing extensions such as `MoreDarts`. (The package developer(s) get to decide whether this is a good or bad thing--in this case, one could argue that the failure to account for `miss` is a bug in `ScoreDarts` and is better fixed there.) +and similarly for `deck` itself. That untyped `[]` is what makes `myhand` (and thus `cards`, when passed to `tallyscore`) a `Vector{Any}`, and the possibilities for `card` are endless. By constraining the possible types, we allow inference to know more clearly what methods might be called. More tips on fixing invalidations through improving inference can be found in [Techniques for fixing inference problems](@ref). + +In this particular case, just annotating `Union{Int,Char}[]` isn't sufficient on its own, because the `score` method for `Char` doesn't yet exist, so Julia doesn't know what to call. However, in most real-world cases this change alone would be sufficient: usually all the needed methods exist, it's just a question of Julia knowing that no other options are possible. + +!!! note + This fix exploits [union-splitting](), which is closely related to "world-splitting." However, union-splitting is far more effective at fixing inference problems, as it guarantees that no other possibilities will *ever* exist, no matter how many other methods get defined. !!! tip - Many vulnerabilities can be fixed by improving inference. In complex code, it's easy to unwittingly write things in ways that defeat Julia's type inference. Not only does fixing the inference problem prevent invalidation, it typically makes your code run faster too. When fixing invalidations, your first thought should always be, "can I improve inference in this code without making compromises in its functionality?" + Many vulnerabilities can be fixed by improving inference. In complex code, it's easy to unwittingly write things in ways that defeat Julia's type inference. Not only is well-inferred code resistant to invalidation, it typically runs faster too. + + While in real life it's usually a bad idea to "blame the victim," it's typically the right attitude for fixing invalidations. Keep in mind, though, that the source of the problem may not be the immediate victim: in this case, it was a poor container choice in `playgame` that put `tallyscore` in the bad position of having to operate on a `Vector{Any}`. Redesigning the call chain to be more inferable is often an ideal fix. However, there are cases where there is no good way to make the code inferable, in which case other strategies are needed. -The second option is to prevent Julia's speculative optimization: one could replace the direct call to `score` with an `invokelatest(score, turn)`: +The third option is to prevent Julia's speculative optimization: one could replace `score(card)` with `invokelatest(score, card)`: ```julia -function tallyscores(turns) - s = 0 - for t in turns - s += invokelatest(score, t) - end - return s +function tallyscores(cards) + s = 0 + for card in cards + s += invokelatest(score, card) end + return s +end +``` + +This forces Julia to always look up the appropriate method of `score` while the code is running, and thus prevents the speculative optimizations that leave the code vulnerable to invalidation. However, the cost is that your code may run somewhat more slowly. + +If you plan to define at least two `score` methods, another way to turn off this optimization would be to declare + +```julia +Base.Experimental.@max_methods 1 function score end ``` -This forces Julia to always look up the appropriate method of `score` while the code is running, and thus prevents the speculative optimizations that leave the code vulnerable to invalidation. If you re-run the above demonstration, using `invokelatest` as shown, then `trees` will be empty--no invalidations! However, the cost is that your code may run somewhat more slowly. +before defining any `score` methods. You can read the documentation on `@max_methods` to learn more about how it works. + +!!! tip + Most of us learn best by doing. Try at least one of these methods of fixing the invalidation, and use SnoopCompile to verify that it works. + +### Undoing the damage from invalidations + +If you can't prevent the invalidation, an alternative approach is to recompile the invalidated code. For example, one could repeat the precompile workload from `Blackjack` in `BlackjackFacecards`. While this will mean that the whole "stack" will be compiled twice and cached twice (which is wasteful), it should be effective in reducing latency for users. -More details about fixing invalidations can be found in [Techniques for fixing inference problems](@ref). +PrecompileTools also has a `@recompile_invalidations`. This isn't generally recommended for use in package (you can end up with long compile times for things you don't need), but it can be useful in personal "Startup packages." See the PrecompileTools documentation for details. From a299eded03460caa04517ede9cf38c3bfa225cdd Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Fri, 12 Jul 2024 13:27:31 -0500 Subject: [PATCH 03/18] Add docs for `@snoop_inference` --- docs/make.jl | 12 +- docs/src/tutorials/fixing_inference.md | 154 ++++++++++++++++++++ docs/src/tutorials/invalidations.md | 5 +- docs/src/{ => tutorials}/snoop_inference.md | 64 ++++---- 4 files changed, 196 insertions(+), 39 deletions(-) create mode 100644 docs/src/tutorials/fixing_inference.md rename docs/src/{ => tutorials}/snoop_inference.md (64%) diff --git a/docs/make.jl b/docs/make.jl index 1b2b338b..a73800b5 100644 --- a/docs/make.jl +++ b/docs/make.jl @@ -8,12 +8,16 @@ makedocs( prettyurls = get(ENV, "CI", nothing) == "true" ), modules = [SnoopCompile.SnoopCompileCore, SnoopCompile], - linkcheck = true, + linkcheck = false, # FIXME make true # doctest = :fix, + warnonly=true, pages = ["index.md", - "tutorial.md", - "Modern tools" => ["snoop_invalidations.md", "snoop_inference.md", "pgdsgui.md", "snoop_inference_analysis.md", "snoop_inference_parcel.md", "jet.md"], - "reference.md"], + "Basic tutorials" => ["tutorials/invalidations.md", "tutorials/snoop_inference.md"], + "Advanced tutorials" => ["tutorials/fixing_inference.md",], + # "tutorial.md", + # "Modern tools" => ["snoop_invalidations.md", "snoop_inference.md", "pgdsgui.md", "snoop_inference_analysis.md", "snoop_inference_parcel.md", "jet.md"], + # "reference.md"], + ] ) deploydocs( diff --git a/docs/src/tutorials/fixing_inference.md b/docs/src/tutorials/fixing_inference.md new file mode 100644 index 00000000..9767dd8d --- /dev/null +++ b/docs/src/tutorials/fixing_inference.md @@ -0,0 +1,154 @@ +# Techniques for fixing inference problems + +Here we assume you've dug into your code with a tool like Cthulhu, and want to know how to fix some of the problems that you discover. Below is a collection of specific cases and some tricks for handling them. + +## Fixing `Core.Box` + +[Julia issue 15276](https://github.com/JuliaLang/julia/issues/15276) is one of the more surprising forms of inference failure; it is the most common cause of a `Core.Box` annotation. +If other variables depend on the `Box`ed variable, then a single `Core.Box` can lead to widespread inference problems. +For this reason, these are also among the first inference problems you should tackle. + +Read [this explanation of why this happens and what you can do to fix it](https://docs.julialang.org/en/v1/manual/performance-tips/#man-performance-captured). +If you are directed to find `Core.Box` inference triggers via [`suggest`](@ref), you may need to explore around the call site a bit-- +the inference trigger may be in the closure itself, but the fix needs to go in the method that creates the closure. + +Use of `ascend` is highly recommended for fixing `Core.Box` inference failures. + +## Adding type annotations + +In cases where invalidations occur, but you can't use concrete types (there are indeed many valid uses of `Vector{Any}`), +you can often prevent the invalidation using some additional knowledge. +One common example is extracting information from an [`IOContext`](https://docs.julialang.org/en/v1/manual/networking-and-streams/#IO-Output-Contextual-Properties-1) structure, which is roughly defined as + +```julia +struct IOContext{IO_t <: IO} <: AbstractPipe + io::IO_t + dict::ImmutableDict{Symbol, Any} +end +``` + +There are good reasons to use a value-type of `Any`, but that makes it impossible for the compiler to infer the type of any object looked up in an `IOContext`. +Fortunately, you can help! +For example, the documentation specifies that the `:color` setting should be a `Bool`, and since it appears in documentation it's something we can safely enforce. +Changing + +``` +iscolor = get(io, :color, false) +``` + +to + +``` +iscolor = get(io, :color, false)::Bool # assert that the rhs is Bool-valued +``` + +will throw an error if it isn't a `Bool`, and this allows the compiler to take advantage of the type being known in subsequent operations. + +We've already seen another relevant example above, where `getaddrinfo(::AbstractString)` was inferred to return an `IPAddr`, which is an abstract type. +Since there are only two such types supported by the Sockets library, +one potential fix is to annotate the returned value from `getaddrinfo` to be `Union{IPv4,IPv6}`. +This will allow Julia to [union-split](https://julialang.org/blog/2018/08/union-splitting/) future operations made using the returned value. + +Before turning to a more complex example, it's worth noting that this trick applied to field accesses of abstract types is often one of the simplest ways to fix widespread inference problems. +This is such an important case that it is described in the section below. + +As a more detailed example, suppose you're writing code that parses Julia's `Expr` type: + +```julia +julia> ex = :(Array{Float32,3}) +:(Array{Float32, 3}) + +julia> dump(ex) +Expr + head: Symbol curly + args: Vector{Any(3,)) + 1: Symbol Array + 2: Symbol Float32 + 3: Int64 3 +``` + +`ex.args` is a `Vector{Any}`. +However, for a `:curly` expression only certain types will be found among the arguments; you could write key portions of your code as + +```julia +a = ex.args[2] +if a isa Symbol + # inside this block, Julia knows `a` is a Symbol, and so methods called on `a` will be resistant to invalidation + foo(a) +elseif a isa Expr && length((a::Expr).args) > 2 + a::Expr # sometimes you have to help inference by adding a type-assert + x = bar(a) # `bar` is now resistant to invalidation +elseif a isa Integer + # even though you've not made this fully-inferrable, you've at least reduced the scope for invalidations + # by limiting the subset of `foobar` methods that might be called + y = foobar(a) +end +``` + +Other tricks include replacing broadcasting on `v::Vector{Any}` with `Base.mapany(f, v)`--`mapany` avoids trying to narrow the type of `f(v[i])` and just assumes it will be `Any`, thereby avoiding invalidations of many `convert` methods. + +Adding type-assertions and fixing inference problems are the most common approaches for fixing invalidations. +You can discover these manually, but using Cthulhu is highly recommended. + +## Inferrable field access for abstract types + +When invalidations happen for methods that manipulate fields of abstract types, often there is a simple solution: create an "interface" for the abstract type specifying that certain fields must have certain types. +Here's an example: + +``` +abstract type AbstractDisplay end + +struct Monitor <: AbstractDisplay + height::Int + width::Int + maker::String +end + +struct Phone <: AbstractDisplay + height::Int + width::Int + maker::Symbol +end + +function Base.show(@nospecialize(d::AbstractDisplay), x) + str = string(x) + w = d.width + if length(str) > w # do we have to truncate to fit the display width? + ... +``` + +In this `show` method, we've deliberately chosen to prevent specialization on the specific type of `AbstractDisplay` (to reduce the total number of times we have to compile this method). +As a consequence, Julia's inference generally will not realize that `d.width` returns an `Int`--it might be able to discover that by exhaustively checking all subtypes, but if there are a lot of such subtypes then such checks would slow compilation considerably. + +Fortunately, you can help by defining an interface for generic `AbstractDisplay` objects: + +``` +function Base.getproperty(d::AbstractDisplay, name::Symbol) + if name === :height + return getfield(d, :height)::Int + elseif name === :width + return getfield(d, :width)::Int + elseif name === :maker + return getfield(d, :maker)::Union{String,Symbol} + end + return getfield(d, name) +end +``` + +Julia's [constant propagation](https://en.wikipedia.org/wiki/Constant_folding) will ensure that most accesses of those fields will be determined at compile-time, so this simple change robustly fixes many inference problems. + +## Handling edge cases + +You can sometimes get invalidations from failing to handle "formal" possibilities. +For example, operations with regular expressions might return a `Union{Nothing, RegexMatch}`. +You can sometimes get poor type inference by writing code that fails to take account of the possibility that `nothing` might be returned. +For example, a comprehension + +```julia +ms = [m.match for m in match.((rex,), my_strings)] +``` +might be replaced with +```julia +ms = [m.match for m in match.((rex,), my_strings) if m !== nothing] +``` +and return a better-typed result. diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md index 71cbbb95..ff7069ac 100644 --- a/docs/src/tutorials/invalidations.md +++ b/docs/src/tutorials/invalidations.md @@ -2,7 +2,8 @@ ## What are invalidations? -Invalidations result from unanticipated interactions between packages, or between packages and Base Julia. They are essential for ensuring that Julia runs fast and behaves correctly, but they make latency worse by "throwing away" precompiled code. +Invalidations result from interactions between different "chunks" of code. They are essential for ensuring that Julia runs fast and behaves correctly, but they make latency worse by "throwing away" compiled code. +Invalidations are triggered by defining new methods, which most commonly happens from *loading* packages. A good developer can often design packages to minimize the number and/or impact of invalidations. ## Learning to observe, diagnose, and fix invalidations @@ -188,6 +189,8 @@ Choose a call for analysis (q to quit): This is an interactive REPL-menu, described more completely (via text and video) at [ascend](https://github.com/JuliaDebug/Cthulhu.jl?tab=readme-ov-file#usage-ascend). +There are quite a few other tools for working with `invs` and `trees`, see the [Invalidations reference](@ref). If your list of invalidations is dauntingly large, you may be interested in [Combining data streams to focus on important invalidations](@ref). + ### Why the invalidations occur `tallyscores` and `playgame` were compiled in `Blackjack`, a "world" where the `score` method defined in `BlackjackFacecards` does not yet exist. When you load the `BlackjackFacecards` package, Julia must ask itself: now that this new `score` method exists, am I certain that I would compile `tallyscores` the same way? If the answer is "no," Julia invalidates the old compiled code, and compiles a fresh version with full awareness of the new `score` method in `BlackjackFacecards`. diff --git a/docs/src/snoop_inference.md b/docs/src/tutorials/snoop_inference.md similarity index 64% rename from docs/src/snoop_inference.md rename to docs/src/tutorials/snoop_inference.md index ce493880..8dcf90a2 100644 --- a/docs/src/snoop_inference.md +++ b/docs/src/tutorials/snoop_inference.md @@ -1,23 +1,23 @@ -# Snooping on inference: `@snoop_inference` +# Tutorial on `@snoop_inference` -!!! compat - `@snoop_inference` is available on `Julia 1.6.0-DEV.1190` or above, but the results can be relevant for all Julia versions. +Inference may occur when you *run* code. Inference is the first step of *type-specialized* compilation. `@snoop_inference` collects data on what inference is doing, giving you greater insight into what is being inferred and how long it takes. -Currently, `precompile` only caches results for type-inference, not other stages in code generation. -For that reason, efforts at reducing latency should be informed by measuring the amount of time spent on type-inference. -Moreover, because all code needs to be type-inferred before undergoing later stages of code generation, monitoring this "entry point" can give you an overview of the entire compile chain. +Compilation is needed only for "fresh" code; running the demos below on code you've already used will yield misleading results. When analyzing inference, you're advised to always start from a fresh session. -The rich data collected by `@snoop_inference` are useful for several different purposes; -on this page, we'll describe the basic tool and show how it can be used to profile inference. -On later pages we'll show other ways to use the data to reduce the amount of type-inference or cache its results. +### Add SnoopCompileCore, SnoopCompile, and helper packages to your environment -## Collecting the data +Here, we'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/). (With the exception of `AbstractTrees`, these "developer tool" packages should not be added to the Project file of any real packages unless you're extending the tool itself.) -Like [`@snoop_invalidations`](@ref), `@snoop_inference` is exported by both `SnoopCompileCore` and `SnoopCompile`, but in this case there is not as much reason to do the data collection by a very minimal package. Consequently here we'll just load `SnoopCompile` at the outset. +```@repl +using Pkg +Pkg.add(["SnoopCompileCore", "SnoopCompile", "AbstractTrees", "ProfileView"]) +``` + +## Setting up the demo To see `@snoop_inference` in action, we'll use the following demo: -```jldoctest flatten-demo +```jldoctest flatten-demo; filter=r"Main\.var\"Main\"\." module FlattenDemo struct MyType{T} x::T end @@ -42,23 +42,29 @@ FlattenDemo ``` The main call, `packintype`, stores the input in a `struct`, and then calls functions that extract the field value and performs arithmetic on the result. -To profile inference on this call, we simply do the following: -```jldoctest flatten-demo; setup=:(using SnoopCompile), filter=r"([0-9]*\.?[0-9]+([eE][-+]?[0-9]+)?|WARNING: replacing module FlattenDemo\.\n)" -julia> tinf = @snoop_inference FlattenDemo.packintype(1) +## [Collecting the data](@id sccshow) + +To profile inference on this call, do the following: + +```jldoctest flatten-demo; filter=r"([0-9]*\.?[0-9]+([eE][-+]?[0-9]+)?|WARNING: replacing module FlattenDemo\.\n)" +julia> using SnoopCompileCore + +julia> tinf = @snoop_inference FlattenDemo.packintype(1); + +julia> using SnoopCompile + +julia> tinf InferenceTimingNode: 0.002712/0.003278 on Core.Compiler.Timings.ROOT() with 1 direct children ``` !!! tip - Inference gets called only on the *first* invocation of a method with those specific types. You have to redefine the `FlattenDemo` module (by just re-executing the command we used to define it) if you want to collect data with `@snoop_inference` on the same code a second time. - - To make it easier to perform these demonstrations and use them for documentation purposes, `SnoopCompile` includes a function [`SnoopCompile.flatten_demo()`](@ref) that redefines the module and returns `tinf`. + Don't omit the semicolon on the `tinf = @snoop_inference ...` line, or you may get an enormous amount of output. The compact display on the final line is possible only because `SnoopCompile` defines nice `Base.show` methods for the data returned by `@snoop_inference`. These methods cannot be defined in `SnoopCompileCore` because it has a fundamental design constraint: loading `SnoopCompileCore` is not allowed to invalidate any code. Moving those `Base.show` methods to `SnoopCompileCore` would violate that guarantee. This may not look like much, but there's a wealth of information hidden inside `tinf`. ## A quick check for potential invalidations - After running `@snoop_inference`, it's generally recommended to check the output of [`staleinstances`](@ref): ```julia julia> staleinstances(tinf) @@ -66,16 +72,7 @@ SnoopCompileCore.InferenceTiming[] ``` If you see this, all's well. -A non-empty list might indicate method invalidations, which can be checked (in a fresh session) by running the identical workload with [`@snoop_invalidations`](@ref). - -!!! warning - Rampant invalidation can make the process of analyzing `tinf` more confusing: "why am I getting reinference of this `MethodInstance` when I `precompile`d it?" Routine use of `staleinstances` at the beginning can save you some head-scratching later. - -!!! tip - Your workload may load packages and/or (re)define methods; these can be sources of invalidation and therefore non-empty output - from `staleinstances`. - One trick that may circumvent some invalidation is to load the packages and make the method definitions before launching `@snoop_inference`, because it ensures the methods are in place - before your workload triggers compilation. +A non-empty list might indicate method invalidations, which can be checked (in a fresh session) using the tools described in [Tutorial on `@snoop_invalidations`](@ref). If you do have a lot of invalidations, [`precompile_blockers`](@ref) may be an effective way to reveal those invalidations that affect your particular package and workload. @@ -89,7 +86,7 @@ You may have noticed that this `ROOT` node prints with two numbers. It will be easier to understand their meaning if we first display the whole tree. We can do that with the [AbstractTrees](https://github.com/JuliaCollections/AbstractTrees.jl) package: -```jldoctest flatten-demo; filter=r"[0-9]*\.?[0-9]+([eE][-+]?[0-9]+)?" +```jldoctest flatten-demo; filter=[r"[0-9]*\.?[0-9]+([eE][-+]?[0-9]+)?", r"Main\.var\"Main\"\."] julia> using AbstractTrees julia> print_tree(tinf) @@ -105,7 +102,7 @@ InferenceTimingNode: 0.002712/0.003278 on Core.Compiler.Timings.ROOT() with 1 di This tree structure reveals the caller-callee relationships, showing the specific types that were used for each `MethodInstance`. Indeed, as the calls to `getproperty` reveal, it goes beyond the types and even shows the results of [constant propagation](https://en.wikipedia.org/wiki/Constant_folding); -the `getproperty(::MyType{Int64}, x::Symbol)` (note `x::Symbol` instead of just plain `::Symbol`) means that the call was `getproperty(y, :x)`, which corresponds to `y.x` in the definition of `extract`. +the `getproperty(::MyType{Int64}, x::Symbol)` corresponds to `y.x` in the definition of `extract`. !!! note Generally we speak of [call graphs](https://en.wikipedia.org/wiki/Call_graph) rather than call trees. @@ -147,7 +144,7 @@ julia> ProfileView.view(fg) You should see something like this: -![flamegraph](assets/flamegraph-flatten-demo.png) +![flamegraph](../assets/flamegraph-flatten-demo.png) Users are encouraged to read the ProfileView documentation to understand how to interpret this, but briefly: @@ -165,5 +162,4 @@ You can explore this flamegraph and compare it to the output from `print_tree`. Finally, [`flatten`](@ref), on its own or together with [`accumulate_by_source`](@ref), allows you to get an sense for the cost of individual `MethodInstance`s or `Method`s. The tools here allow you to get an overview of where inference is spending its time. -Sometimes, this information alone is enough to show you how to change your code to reduce latency: perhaps your code is spending a lot of time inferring cases that are not needed in practice and could be simplified. -However, most efforts at latency reduction will probably leverage additional tools (described next) that help identify the main opportunities for intervention. +This gives you insight into the major contributors to latency. From 5bf39cade04569e9de7f84e7763cb6d998132757 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Fri, 12 Jul 2024 13:28:07 -0500 Subject: [PATCH 04/18] Don't reexort SnoopCompileCore This is intended to encourage users to collect data with just SnoopCompileCore. --- src/SnoopCompile.jl | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/SnoopCompile.jl b/src/SnoopCompile.jl index 84783375..b004ffa8 100644 --- a/src/SnoopCompile.jl +++ b/src/SnoopCompile.jl @@ -72,15 +72,15 @@ end include("parcel_snoop_inference.jl") include("inference_demos.jl") -export @snoop_inference, exclusive, inclusive, flamegraph, flatten, accumulate_by_source, collect_for, runtime_inferencetime, staleinstances +export exclusive, inclusive, flamegraph, flatten, accumulate_by_source, collect_for, runtime_inferencetime, staleinstances export InferenceTrigger, inference_triggers, callerinstance, callingframe, skiphigherorder, trigger_tree, suggest, isignorable export report_callee, report_caller, report_callees include("parcel_snoop_llvm.jl") -export read_snoop_llvm, @snoop_llvm +export read_snoop_llvm include("invalidations.jl") -export @snoop_invalidations, uinvalidated, invalidation_trees, filtermod, findcaller +export uinvalidated, invalidation_trees, filtermod, findcaller include("invalidation_and_inference.jl") export precompile_blockers From 4d3e57324133e587ba17045574222f64cda7eafb Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 04:07:32 -0500 Subject: [PATCH 05/18] semicomplete --- docs/Project.toml | 2 + docs/make.jl | 9 +- docs/src/explanations/basic.md | 40 +++++ .../fixing_inference.md | 47 +++--- docs/src/explanations/gotchas.md | 26 +++ docs/src/explanations/tools.md | 29 ++++ docs/src/index.md | 6 +- docs/src/reference.md | 4 +- docs/src/snoop_invalidations.md | 150 ------------------ docs/src/tutorials/invalidations.md | 45 ++++-- docs/src/{ => tutorials}/jet.md | 2 +- docs/src/{ => tutorials}/pgdsgui.md | 0 docs/src/tutorials/snoop_inference.md | 4 +- .../snoop_inference_analysis.md | 0 .../{ => tutorials}/snoop_inference_parcel.md | 93 ++--------- docs/src/tutorials/snoop_llvm.md | 37 +++++ src/utils.jl | 11 +- test/snoop_inference.jl | 2 +- 18 files changed, 222 insertions(+), 285 deletions(-) create mode 100644 docs/src/explanations/basic.md rename docs/src/{tutorials => explanations}/fixing_inference.md (78%) create mode 100644 docs/src/explanations/gotchas.md create mode 100644 docs/src/explanations/tools.md rename docs/src/{ => tutorials}/jet.md (99%) rename docs/src/{ => tutorials}/pgdsgui.md (100%) rename docs/src/{ => tutorials}/snoop_inference_analysis.md (100%) rename docs/src/{ => tutorials}/snoop_inference_parcel.md (62%) create mode 100644 docs/src/tutorials/snoop_llvm.md diff --git a/docs/Project.toml b/docs/Project.toml index cb4daa2d..f82615e3 100644 --- a/docs/Project.toml +++ b/docs/Project.toml @@ -6,6 +6,7 @@ JET = "c3a54625-cd67-489e-a8e7-0a5a0ff4e31b" MethodAnalysis = "85b6ec6f-f7df-4429-9514-a64bcd9ee824" PyPlot = "d330b81b-6aea-500a-939a-2ce795aea3ee" SnoopCompile = "aa65fe97-06da-5843-b5b1-d5d13cad87d2" +SnoopCompileCore = "e2b509da-e806-4183-be48-004708413034" [compat] AbstractTrees = "0.4" @@ -15,3 +16,4 @@ JET = "0.9" MethodAnalysis = "0.4" PyPlot = "2" SnoopCompile = "3" +SnoopCompileCore = "3" diff --git a/docs/make.jl b/docs/make.jl index a73800b5..72d1b80d 100644 --- a/docs/make.jl +++ b/docs/make.jl @@ -12,11 +12,10 @@ makedocs( # doctest = :fix, warnonly=true, pages = ["index.md", - "Basic tutorials" => ["tutorials/invalidations.md", "tutorials/snoop_inference.md"], - "Advanced tutorials" => ["tutorials/fixing_inference.md",], - # "tutorial.md", - # "Modern tools" => ["snoop_invalidations.md", "snoop_inference.md", "pgdsgui.md", "snoop_inference_analysis.md", "snoop_inference_parcel.md", "jet.md"], - # "reference.md"], + "Basic tutorials" => ["tutorials/invalidations.md", "tutorials/snoop_inference.md", "tutorials/snoop_llvm.md", "tutorials/pgdsgui.md", "tutorials/jet.md"], + "Advanced tutorials" => ["tutorials/snoop_inference_analysis.md", "tutorials/snoop_inference_parcel.md"], + "Explanations" => ["tools.md", "gotchas.md", "explanations/fixing_inference.md"], + "reference.md", ] ) diff --git a/docs/src/explanations/basic.md b/docs/src/explanations/basic.md new file mode 100644 index 00000000..04f205e7 --- /dev/null +++ b/docs/src/explanations/basic.md @@ -0,0 +1,40 @@ +# Understanding SnoopCompile and Julia's compilation pipeline + +Julia uses +[Just-in-time (JIT) compilation](https://en.wikipedia.org/wiki/Just-in-time_compilation) to +generate the code that runs on your CPU. +Broadly speaking, there are two major compilation steps: *inference* and *code generation*. +Inference is the process of determining the type of each object, which in turn +determines which specific methods get called; once type inference is complete, +code generation performs optimizations and ultimately generates the assembly +language (native code) used on CPUs. +Some aspects of this process are documented [here](https://docs.julialang.org/en/v1/devdocs/eval/). + +Using code that has never been compiled requires that it first be JIT-compiled, and this contributes to the latency of using the package. +In some circumstances, you can cache (store) the results of compilation to files to +reduce the latency when your package is used. These files are the the `*.ji` and +`*.so` files that live in the `compiled` directory of your Julia depot, usually +located at `~/.julia/compiled`. However, if these files become large, loading +them can be another source for latency. Julia needs time both to load and +validate the cached compiled code. Minimizing the latency of using a package +involves focusing on caching the compilation of code that is both commonly used +and takes time to compile. + +Caching code for later use is called *precompilation*. Julia has had some forms of precompilation almost since the very first packages. However, it was [Julia +1.9](https://julialang.org/blog/2023/04/julia-1.9-highlights/#caching_of_native_code) that first supported "complete" precompilation, including the ability to store native code in shared-library cache files. + +SnoopCompile is designed to try to allow you to analyze the costs of JIT-compilation, identify +key bottlenecks that contribute to latency, and set up `precompile` directives to see whether +it produces measurable benefits. + +## Package precompilation + +When a package is precompiled, here's what happens under the hood: + +- Julia loads all of the package's dependencies (the ones in the `[deps]` section of the `Project.toml` file), typically from precompile cache files +- Julia evaluates the source code (text files) that define the package module(s). Evaluating `function foo(args...) ... end` creates a new method `foo`. Note that: + + the source code might also contain statements that create "data" (e.g., `const`s). In some cases this can lead to some subtle precompilation ["gotchas"](@ref running-during-pc) + + the source code might also contain a precompile workload, which forces compilation and tracking of package methods. +- Julia iterates over the module contents and writes the *result* to disk. Note that the module contents might include compiled code, and if so it is written along with everything else to the cache file. + +When Julia loads your package, it just loads the "snapshot" stored in the cache file: it does not re-evaluate the source-text files that defined your package! It is appropriate to think of the source files of your package as "build scripts" that create your module; once the "build scripts" are executed, it's the module itself that gets cached, and the job of the build scripts is done. diff --git a/docs/src/tutorials/fixing_inference.md b/docs/src/explanations/fixing_inference.md similarity index 78% rename from docs/src/tutorials/fixing_inference.md rename to docs/src/explanations/fixing_inference.md index 9767dd8d..de12db66 100644 --- a/docs/src/tutorials/fixing_inference.md +++ b/docs/src/explanations/fixing_inference.md @@ -2,19 +2,24 @@ Here we assume you've dug into your code with a tool like Cthulhu, and want to know how to fix some of the problems that you discover. Below is a collection of specific cases and some tricks for handling them. -## Fixing `Core.Box` +Note that there is also a [tutorial on fixing inference](@ref inferrability) that delves into advanced topics. -[Julia issue 15276](https://github.com/JuliaLang/julia/issues/15276) is one of the more surprising forms of inference failure; it is the most common cause of a `Core.Box` annotation. -If other variables depend on the `Box`ed variable, then a single `Core.Box` can lead to widespread inference problems. -For this reason, these are also among the first inference problems you should tackle. +## Adding type annotations -Read [this explanation of why this happens and what you can do to fix it](https://docs.julialang.org/en/v1/manual/performance-tips/#man-performance-captured). -If you are directed to find `Core.Box` inference triggers via [`suggest`](@ref), you may need to explore around the call site a bit-- -the inference trigger may be in the closure itself, but the fix needs to go in the method that creates the closure. +### Using concrete types -Use of `ascend` is highly recommended for fixing `Core.Box` inference failures. +Defining variables like `list = []` can be convenient, but it creates a `list` of type `Vector{Any}`. This prevents inference from knowing the type of items extracted from `list`. Using `list = String[]` for a container of strings, etc., is an excellent fix. When in doubt, check the type with `isconcretetype`: a common mistake is to think that `list_of_lists = Array{Int}[]` gives you a vector-of-vectors, but -## Adding type annotations +```jldoctest +julia> isconcretetype(Array{Int}) +false +``` + +reminds you that `Array` requires a second parameter indicating the dimensionality of the array. (Or use `list_of_lists = Vector{Int}[]` instead, as `Vector{Int} === Array{Int, 1}`.) + +Many valuable tips can be found among [Julia's performance tips](https://docs.julialang.org/en/v1/manual/performance-tips/), and readers are encouraged to consult that page. + +### Working with non-concrete types In cases where invalidations occur, but you can't use concrete types (there are indeed many valid uses of `Vector{Any}`), you can often prevent the invalidation using some additional knowledge. @@ -27,7 +32,7 @@ struct IOContext{IO_t <: IO} <: AbstractPipe end ``` -There are good reasons to use a value-type of `Any`, but that makes it impossible for the compiler to infer the type of any object looked up in an `IOContext`. +There are good reasons that `dict` uses a value-type of `Any`, but that makes it impossible for the compiler to infer the type of any object looked up in an `IOContext`. Fortunately, you can help! For example, the documentation specifies that the `:color` setting should be a `Bool`, and since it appears in documentation it's something we can safely enforce. Changing @@ -44,13 +49,7 @@ iscolor = get(io, :color, false)::Bool # assert that the rhs is Bool-valued will throw an error if it isn't a `Bool`, and this allows the compiler to take advantage of the type being known in subsequent operations. -We've already seen another relevant example above, where `getaddrinfo(::AbstractString)` was inferred to return an `IPAddr`, which is an abstract type. -Since there are only two such types supported by the Sockets library, -one potential fix is to annotate the returned value from `getaddrinfo` to be `Union{IPv4,IPv6}`. -This will allow Julia to [union-split](https://julialang.org/blog/2018/08/union-splitting/) future operations made using the returned value. - -Before turning to a more complex example, it's worth noting that this trick applied to field accesses of abstract types is often one of the simplest ways to fix widespread inference problems. -This is such an important case that it is described in the section below. +If the return type is one of a small number of possibilities (generally three or fewer), you can annotate the return type with `Union{...}`. This is generally advantageous only when the intersection of what inference already knows about the types of a variable and the types in the `Union` results in an concrete type. As a more detailed example, suppose you're writing code that parses Julia's `Expr` type: @@ -118,7 +117,7 @@ function Base.show(@nospecialize(d::AbstractDisplay), x) ``` In this `show` method, we've deliberately chosen to prevent specialization on the specific type of `AbstractDisplay` (to reduce the total number of times we have to compile this method). -As a consequence, Julia's inference generally will not realize that `d.width` returns an `Int`--it might be able to discover that by exhaustively checking all subtypes, but if there are a lot of such subtypes then such checks would slow compilation considerably. +As a consequence, Julia's inference may not realize that `d.width` returns an `Int`. Fortunately, you can help by defining an interface for generic `AbstractDisplay` objects: @@ -137,6 +136,18 @@ end Julia's [constant propagation](https://en.wikipedia.org/wiki/Constant_folding) will ensure that most accesses of those fields will be determined at compile-time, so this simple change robustly fixes many inference problems. +## Fixing `Core.Box` + +[Julia issue 15276](https://github.com/JuliaLang/julia/issues/15276) is one of the more surprising forms of inference failure; it is the most common cause of a `Core.Box` annotation. +If other variables depend on the `Box`ed variable, then a single `Core.Box` can lead to widespread inference problems. +For this reason, these are also among the first inference problems you should tackle. + +Read [this explanation of why this happens and what you can do to fix it](https://docs.julialang.org/en/v1/manual/performance-tips/#man-performance-captured). +If you are directed to find `Core.Box` inference triggers via [`suggest`](@ref), you may need to explore around the call site a bit-- +the inference trigger may be in the closure itself, but the fix needs to go in the method that creates the closure. + +Use of `ascend` is highly recommended for fixing `Core.Box` inference failures. + ## Handling edge cases You can sometimes get invalidations from failing to handle "formal" possibilities. diff --git a/docs/src/explanations/gotchas.md b/docs/src/explanations/gotchas.md new file mode 100644 index 00000000..d728e12d --- /dev/null +++ b/docs/src/explanations/gotchas.md @@ -0,0 +1,26 @@ +# Precompilation "gotcha"s + +## [Running code during module definition](@ref running-during-pc) + +Suppose you're working on an astronomy package and your source code has a line + +``` +const planets = map(makeplanet, ["Mercury", ...]) +``` + +Julia will dutifully create `planets` and store it in the package's precompile cache file. This also runs `makeplanet`, and if this is the first time it gets run, it will compile `makeplanet`. Assuming that `makeplanet` is a method defined in the package, the compiled code for `makeplanet` will be stored in the cache file. + +However, two circumstances can lead to puzzling omissions from the cache files: +- if `makeplanet` is a method defined in a dependency of your package, it will *not* be cached in your package. You'd want to add precompilation of `makeplanet` to the package that creates that method. +- if `makeplanet` is poorly-infered and uses runtime dispatch, any such callees that are not owned by your package will not be cached. For example, suppose `makeplanet` ends up calling methods in Base Julia or its standard libraries that are not precompiled into Julia itself: the compiled code for those methods will not be added to the cache file. + +One option to ensure this dependent code gets cached is to create `planets` inside `@compile_workload`: + +``` +@compile_workload begin + global planets + const planet = map(makeplanet, ["Mercury", ...]) +end +``` + +Note that your package definition can have multiple `@compile_workload` blocks. diff --git a/docs/src/explanations/tools.md b/docs/src/explanations/tools.md new file mode 100644 index 00000000..54bdff88 --- /dev/null +++ b/docs/src/explanations/tools.md @@ -0,0 +1,29 @@ +# Package roles and alternatives + +## SnoopCompile + +SnoopCompileCore is a tiny package with no dependencies; it's used for collecting data, and it has been designed in such a way that it cannot cause any invalidations of its own. Collecting data on invalidations and inference with SnoopCompileCore is the only way you can be sure you are observing the "native state" of your code. + +## SnoopCompile + +SnoopCompile is a much larger package that performs analysis on the data collected by SnoopCompileCore; loading SnoopCompile can (and does) trigger invalidations. +Consequently, you're urged to always collect data with just SnoopCompileCore loaded, +and wait to load SnoopCompile until after you've finished collecting the data. + +## Cthulhu + +[Cthulhu](https://github.com/JuliaDebug/Cthulhu.jl) is a companion package that gives deep insights into the origin of invalidations or inference failures. + +## AbstractTrees + +[AbstractTrees](https://github.com/JuliaCollections/AbstractTrees.jl) is the one package in this list that can be both a "workhorse" and a developer tool. SnoopCompile uses it mostly for pretty-printing. + +## JET + +[JET](https://github.com/aviatesk/JET.jl) is perhaps the main alternative to SnoopCompile. The packages have some overlap in what they can tell you about your code, but their mechanisms of action are fundamentally different: + +- JET is a "static analyzer," which means that it analyzes the code itself. JET can tell you about inference failures (runtime dispatch) much like SnoopCompile, with one major advantage: SnoopCompileCore requires you to use `@snoop_inference` in a fresh session and omits information about any callees that have been compiled previously, whereas JET's `@report_opt` provides exhaustive information about the entire *inferable* callgraph (i.e., the part of the callgraph that inference can predict from the initial call). + +- SnoopCompileCore collects data by watching normal inference at work. On code that hasn't been compiled previously, this can yield results similar to JET's, with a different major advantage: SnoopCompileCore can "see through" runtime dispatch, and provide insights about callees that are invisible to JET. + +Perhaps surprisingly, combining JET and SnoopCompile can provide insights that are difficult to obtain with either package in isolation. See the [Tutorial on JET integration](@ref). \ No newline at end of file diff --git a/docs/src/index.md b/docs/src/index.md index 94e97362..21e11cdb 100644 --- a/docs/src/index.md +++ b/docs/src/index.md @@ -24,7 +24,11 @@ SnoopCompile "snoops" on the Julia compiler, collecting information that may be ## Background information -Confused? Some of the core concepts are explained in [Understanding SnoopCompile and Julia's compilation pipeline](@ref). +If nothing else, you should know this: +- invalidations occur when you *load* code (e.g., `using MyPkg`) or otherwise define new methods +- inference and other stages of compilation occur the first time you *run* code for a particular combination of input types + +The individual tutorials briefly explain core concepts. More detail can be found in [Understanding SnoopCompile and Julia's compilation pipeline](@ref). ## Who should use this package diff --git a/docs/src/reference.md b/docs/src/reference.md index ea058dbb..44551760 100644 --- a/docs/src/reference.md +++ b/docs/src/reference.md @@ -50,12 +50,10 @@ report_callees report_caller ``` -## Other utilities +## Analysis of LLVM ```@docs -SnoopCompile.read SnoopCompile.read_snoop_llvm -SnoopCompile.format_userimg ``` ## Demos diff --git a/docs/src/snoop_invalidations.md b/docs/src/snoop_invalidations.md index 5daa32f7..a1c19d6d 100644 --- a/docs/src/snoop_invalidations.md +++ b/docs/src/snoop_invalidations.md @@ -497,153 +497,3 @@ Since these tips also improve performance and allow programs to behave more pred these guidelines are not intrusive. Indeed, searching for and eliminating invalidations can help you improve the quality of your code. -#### Fixing `Core.Box` - -[Julia issue 15276](https://github.com/JuliaLang/julia/issues/15276) is one of the more surprising forms of inference failure; it is the most common cause of a `Core.Box` annotation. -If other variables depend on the `Box`ed variable, then a single `Core.Box` can lead to widespread inference problems. -For this reason, these are also among the first inference problems you should tackle. - -Read [this explanation of why this happens and what you can do to fix it](https://docs.julialang.org/en/v1/manual/performance-tips/#man-performance-captured). -If you are directed to find `Core.Box` inference triggers via [`suggest`](@ref), you may need to explore around the call site a bit-- -the inference trigger may be in the closure itself, but the fix needs to go in the method that creates the closure. - -Use of `ascend` is highly recommended for fixing `Core.Box` inference failures. - -#### Adding type annotations - -In cases where invalidations occur, but you can't use concrete types (there are indeed many valid uses of `Vector{Any}`), -you can often prevent the invalidation using some additional knowledge. -One common example is extracting information from an [`IOContext`](https://docs.julialang.org/en/v1/manual/networking-and-streams/#IO-Output-Contextual-Properties-1) structure, which is roughly defined as - -```julia -struct IOContext{IO_t <: IO} <: AbstractPipe - io::IO_t - dict::ImmutableDict{Symbol, Any} -end -``` - -There are good reasons to use a value-type of `Any`, but that makes it impossible for the compiler to infer the type of any object looked up in an `IOContext`. -Fortunately, you can help! -For example, the documentation specifies that the `:color` setting should be a `Bool`, and since it appears in documentation it's something we can safely enforce. -Changing - -``` -iscolor = get(io, :color, false) -``` - -to - -``` -iscolor = get(io, :color, false)::Bool # assert that the rhs is Bool-valued -``` - -will throw an error if it isn't a `Bool`, and this allows the compiler to take advantage of the type being known in subsequent operations. - -We've already seen another relevant example above, where `getaddrinfo(::AbstractString)` was inferred to return an `IPAddr`, which is an abstract type. -Since there are only two such types supported by the Sockets library, -one potential fix is to annotate the returned value from `getaddrinfo` to be `Union{IPv4,IPv6}`. -This will allow Julia to [union-split](https://julialang.org/blog/2018/08/union-splitting/) future operations made using the returned value. - -Before turning to a more complex example, it's worth noting that this trick applied to field accesses of abstract types is often one of the simplest ways to fix widespread inference problems. -This is such an important case that it is described in the section below. - -As a more detailed example, suppose you're writing code that parses Julia's `Expr` type: - -```julia -julia> ex = :(Array{Float32,3}) -:(Array{Float32, 3}) - -julia> dump(ex) -Expr - head: Symbol curly - args: Vector{Any(3,)) - 1: Symbol Array - 2: Symbol Float32 - 3: Int64 3 -``` - -`ex.args` is a `Vector{Any}`. -However, for a `:curly` expression only certain types will be found among the arguments; you could write key portions of your code as - -```julia -a = ex.args[2] -if a isa Symbol - # inside this block, Julia knows `a` is a Symbol, and so methods called on `a` will be resistant to invalidation - foo(a) -elseif a isa Expr && length((a::Expr).args) > 2 - a::Expr # sometimes you have to help inference by adding a type-assert - x = bar(a) # `bar` is now resistant to invalidation -elseif a isa Integer - # even though you've not made this fully-inferrable, you've at least reduced the scope for invalidations - # by limiting the subset of `foobar` methods that might be called - y = foobar(a) -end -``` - -Other tricks include replacing broadcasting on `v::Vector{Any}` with `Base.mapany(f, v)`--`mapany` avoids trying to narrow the type of `f(v[i])` and just assumes it will be `Any`, thereby avoiding invalidations of many `convert` methods. - -Adding type-assertions and fixing inference problems are the most common approaches for fixing invalidations. -You can discover these manually, but using Cthulhu is highly recommended. - -#### Inferrable field access for abstract types - -When invalidations happen for methods that manipulate fields of abstract types, often there is a simple solution: create an "interface" for the abstract type specifying that certain fields must have certain types. -Here's an example: - -``` -abstract type AbstractDisplay end - -struct Monitor <: AbstractDisplay - height::Int - width::Int - maker::String -end - -struct Phone <: AbstractDisplay - height::Int - width::Int - maker::Symbol -end - -function Base.show(@nospecialize(d::AbstractDisplay), x) - str = string(x) - w = d.width - if length(str) > w # do we have to truncate to fit the display width? - ... -``` - -In this `show` method, we've deliberately chosen to prevent specialization on the specific type of `AbstractDisplay` (to reduce the total number of times we have to compile this method). -As a consequence, Julia's inference generally will not realize that `d.width` returns an `Int`--it might be able to discover that by exhaustively checking all subtypes, but if there are a lot of such subtypes then such checks would slow compilation considerably. - -Fortunately, you can help by defining an interface for generic `AbstractDisplay` objects: - -``` -function Base.getproperty(d::AbstractDisplay, name::Symbol) - if name === :height - return getfield(d, :height)::Int - elseif name === :width - return getfield(d, :width)::Int - elseif name === :maker - return getfield(d, :maker)::Union{String,Symbol} - end - return getfield(d, name) -end -``` - -Julia's [constant propagation](https://en.wikipedia.org/wiki/Constant_folding) will ensure that most accesses of those fields will be determined at compile-time, so this simple change robustly fixes many inference problems. - -#### Handling edge cases - -You can sometimes get invalidations from failing to handle "formal" possibilities. -For example, operations with regular expressions might return a `Union{Nothing, RegexMatch}`. -You can sometimes get poor type inference by writing code that fails to take account of the possibility that `nothing` might be returned. -For example, a comprehension - -```julia -ms = [m.match for m in match.((rex,), my_strings)] -``` -might be replaced with -```julia -ms = [m.match for m in match.((rex,), my_strings) if m !== nothing] -``` -and return a better-typed result. diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md index ff7069ac..10b390c6 100644 --- a/docs/src/tutorials/invalidations.md +++ b/docs/src/tutorials/invalidations.md @@ -2,9 +2,15 @@ ## What are invalidations? -Invalidations result from interactions between different "chunks" of code. They are essential for ensuring that Julia runs fast and behaves correctly, but they make latency worse by "throwing away" compiled code. -Invalidations are triggered by defining new methods, which most commonly happens from *loading* packages. -A good developer can often design packages to minimize the number and/or impact of invalidations. +Invalidations result from interactions between different pieces of code. Invalidations are essential to make Julia fast, interactive, and correct: you *need* invalidations if you want to be able to define some methods, run (compile) some code, and then in the same session define new methods and get answers that account for the new methods. + +Invalidations can happen just from loading packages. Packages are precompiled in isolation, but you can load many packages into a single interactive session. It's impossible for the individual packages to anticipate the full "world of methods" in your interactive session, so sometimes Julia has to discard code that was compiled in a smaller world because it's at risk for being incorrect in the larger world. + +The downside of invalidations is that they make latency worse, as code must be recompiled when you first run it. The benefits of precompilation are partially lost, and the work done during precompilation is partially wasted. + +While some invalidations are unavoidable, in practice a good developer can often design packages to minimize the number and/or impact of invalidations. Invalidation-resistant code is often faster, with smaller binary size, than code that is vulnerable to invalidation. + +A good first step is to measure what's being invalidated, and why. ## Learning to observe, diagnose, and fix invalidations @@ -24,7 +30,7 @@ Pkg.add(["SnoopCompileCore", "SnoopCompile", "AbstractTrees", "Cthulhu"]) ### Create the demonstration packages -We're going to implement a toy version of the card game [blackjack](), where players take cards with the aim of collecting 21 points. The higher you go the better, *unless* you go over 21 points, in which case you "go bust" (i.e., you lose). Because our real goal is to illustrate invalidations, we'll create a "blackjack ecosystem" that involves an interaction between two packages. +We're going to implement a toy version of the card game [blackjack](https://www.wikihow.com/Play-Blackjack), where players take cards with the aim of collecting 21 points. The higher you go the better, *unless* you go over 21 points, in which case you "go bust" (i.e., you lose). Because our real goal is to illustrate invalidations, we'll create a "blackjack ecosystem" that involves an interaction between two packages. While [PkgTemplates](https://github.com/JuliaCI/PkgTemplates.jl) is recommended for creating packages, here we'll just use the basic capabilities in `Pkg`. To create the (empty) packages, the code below executes the following steps: @@ -111,7 +117,7 @@ write(joinpath("BlackjackFacecards", "src", "BlackjackFacecards.jl"), """ ``` !!! warning - Because `BlackjackFacecards` "owns" neither `Char` nor `score`, this is [piracy]((https://docs.julialang.org/en/v1/manual/style-guide/#Avoid-type-piracy-1)) and should generally be avoided. Piracy is one way to cause invalidations, but it's not the only one. `BlackjackFacecards` could avoid committing piracy by defining a `struct Facecard ... end` and defining `score(card::Facecard)` instead of `score(card::Char)`, but this would *not* fix the invalidations--all the factors described below are unchanged. + Because `BlackjackFacecards` "owns" neither `Char` nor `score`, this is [piracy](https://docs.julialang.org/en/v1/manual/style-guide/#Avoid-type-piracy-1) and should generally be avoided. Piracy is one way to cause invalidations, but it's not the only one. `BlackjackFacecards` could avoid committing piracy by defining a `struct Facecard ... end` and defining `score(card::Facecard)` instead of `score(card::Char)`. However, this would *not* fix the invalidations--all the factors described below are unchanged. Now we're ready! @@ -159,7 +165,7 @@ First let's look at the the problematic method `sig`nature: sig ``` -This is a type-tuple, i.e., `Tuple{typeof(f), typesof(args)...}`. We see that `score` was called on an object of (inferred) type `Any`. Calling a function with unknown argument types makes code vulnerable to invalidation, and insertion of the new `score` method "exploited" this vulnerability. +This is a type-tuple, i.e., `Tuple{typeof(f), typesof(args)...}`. We see that `score` was called on an object of (inferred) type `Any`. **Calling a function with unknown argument types makes code vulnerable to invalidation, and insertion of the new `score` method "exploited" this vulnerability.** `victim` shows which compiled code got invalidated: @@ -203,8 +209,13 @@ Thus, invalidations arise from optimization based on what methods and types are ### Fixing invalidations -In broad strokes, there are three ways to prevent invalidation. The first is to ensure that -the full range of possibilties (the entire "world of code") is present before any compilation occurs. In this case, probably the best approach would be to merge the `BlackjackFacecards` package into `Blackjack` itself. Or, if you are a maintainer of the "Blackjack ecosystem" and have reasons for thinking that keeping the packages separate makes sense, you could alternatively move the `PrecompileTools` workload to `BlackjackFacecards`. Either approach should prevent the invalidations from occuring. +In broad strokes, there are three ways to prevent invalidation. + +#### Method 1: defer compilation until the full world is known + +The first and simplest technique is to ensure that the full range of possibilties (the entire "world of code") is present before any compilation occurs. In this case, probably the best approach would be to merge the `BlackjackFacecards` package into `Blackjack` itself. Or, if you are a maintainer of the "Blackjack ecosystem" and have reasons for thinking that keeping the packages separate makes sense, you could alternatively move the `PrecompileTools` workload to `BlackjackFacecards`. Either approach should prevent the invalidations from occuring. + +#### Method 2: improve inferability The second way to prevent invalidations is to improve the inferability of the victim(s). If `Int` and `Char` really are the only possible kinds of cards, then in `playgame` it would be better to declare @@ -213,15 +224,21 @@ myhand = Union{Int,Char}[] ``` and similarly for `deck` itself. That untyped `[]` is what makes `myhand` (and thus `cards`, when passed to `tallyscore`) a `Vector{Any}`, and the possibilities for `card` are endless. By constraining the possible types, we allow inference to know more clearly what methods might be called. More tips on fixing invalidations through improving inference can be found in [Techniques for fixing inference problems](@ref). -In this particular case, just annotating `Union{Int,Char}[]` isn't sufficient on its own, because the `score` method for `Char` doesn't yet exist, so Julia doesn't know what to call. However, in most real-world cases this change alone would be sufficient: usually all the needed methods exist, it's just a question of Julia knowing that no other options are possible. +In this particular case, just annotating `Union{Int,Char}[]` isn't sufficient on its own, because the `score` method for `Char` doesn't yet exist, so Julia doesn't know what to call. However, in most real-world cases this change alone would be sufficient: usually all the needed methods exist, it's just a question of reassuring Julia that no other options are even possible. !!! note - This fix exploits [union-splitting](), which is closely related to "world-splitting." However, union-splitting is far more effective at fixing inference problems, as it guarantees that no other possibilities will *ever* exist, no matter how many other methods get defined. + This fix leverages [union-splitting](https://julialang.org/blog/2018/08/union-splitting/), which is conceptually related to "world-splitting." However, union-splitting is far more effective at fixing inference problems, as it guarantees that no other possibilities will *ever* exist, no matter how many other methods get defined. !!! tip - Many vulnerabilities can be fixed by improving inference. In complex code, it's easy to unwittingly write things in ways that defeat Julia's type inference. Not only is well-inferred code resistant to invalidation, it typically runs faster too. + Many vulnerabilities can be fixed by improving inference. In complex code, it's easy to unwittingly write things in ways that defeat Julia's type inference. Tools that help you discover inference problems, like SnoopCompile and [JET](@ref), help you discover these unwitting "mistakes." + +While in real life it's usually a bad idea to "blame the victim," it's typically the right attitude for fixing invalidations. Keep in mind, though, that the source of the problem may not be the immediate victim: in this case, it was a poor container choice in `playgame` that put `tallyscore` in the bad position of having to operate on a `Vector{Any}`. + +Improving inferability is probably the most broadly-applicable technique, and when applicable it usually gives the best outcomes: not only is your code more resistant to invalidation, but it's likely faster and compiles to smaller binaries. However, of the three approaches it is also the one that requires the deepest understanding of Julia's type system, and thus may be difficult for some coders to use. + +There are cases where there is no good way to make the code inferable, in which case other strategies are needed. - While in real life it's usually a bad idea to "blame the victim," it's typically the right attitude for fixing invalidations. Keep in mind, though, that the source of the problem may not be the immediate victim: in this case, it was a poor container choice in `playgame` that put `tallyscore` in the bad position of having to operate on a `Vector{Any}`. Redesigning the call chain to be more inferable is often an ideal fix. However, there are cases where there is no good way to make the code inferable, in which case other strategies are needed. +#### Method 3: disable Julia's speculative optimization The third option is to prevent Julia's speculative optimization: one could replace `score(card)` with `invokelatest(score, card)`: @@ -235,7 +252,7 @@ function tallyscores(cards) end ``` -This forces Julia to always look up the appropriate method of `score` while the code is running, and thus prevents the speculative optimizations that leave the code vulnerable to invalidation. However, the cost is that your code may run somewhat more slowly. +This forces Julia to always look up the appropriate method of `score` while the code is running, and thus prevents the speculative optimizations that leave the code vulnerable to invalidation. However, the cost is that your code may run somewhat more slowly, particularly here where the call is inside a loop. If you plan to define at least two `score` methods, another way to turn off this optimization would be to declare @@ -252,4 +269,4 @@ before defining any `score` methods. You can read the documentation on `@max_met If you can't prevent the invalidation, an alternative approach is to recompile the invalidated code. For example, one could repeat the precompile workload from `Blackjack` in `BlackjackFacecards`. While this will mean that the whole "stack" will be compiled twice and cached twice (which is wasteful), it should be effective in reducing latency for users. -PrecompileTools also has a `@recompile_invalidations`. This isn't generally recommended for use in package (you can end up with long compile times for things you don't need), but it can be useful in personal "Startup packages." See the PrecompileTools documentation for details. +PrecompileTools also has a `@recompile_invalidations`. This isn't generally recommended for use in package (you can end up with long compile times for things you don't need), but it can be useful in personal "Startup packages" where you want to reduce latency for a particular project you're working on. See the PrecompileTools documentation for details. diff --git a/docs/src/jet.md b/docs/src/tutorials/jet.md similarity index 99% rename from docs/src/jet.md rename to docs/src/tutorials/jet.md index 5d403b34..6443a366 100644 --- a/docs/src/jet.md +++ b/docs/src/tutorials/jet.md @@ -1,4 +1,4 @@ -# [JET integration](@id JET) +# Tutorial on JET integration [JET](https://github.com/aviatesk/JET.jl) is a powerful tool for analyzing call graphs. Some of its functionality overlaps that of SnoopCompile's, that is, JET also provides mechanisms to detect potential errors. diff --git a/docs/src/pgdsgui.md b/docs/src/tutorials/pgdsgui.md similarity index 100% rename from docs/src/pgdsgui.md rename to docs/src/tutorials/pgdsgui.md diff --git a/docs/src/tutorials/snoop_inference.md b/docs/src/tutorials/snoop_inference.md index 8dcf90a2..0e9beb23 100644 --- a/docs/src/tutorials/snoop_inference.md +++ b/docs/src/tutorials/snoop_inference.md @@ -2,7 +2,7 @@ Inference may occur when you *run* code. Inference is the first step of *type-specialized* compilation. `@snoop_inference` collects data on what inference is doing, giving you greater insight into what is being inferred and how long it takes. -Compilation is needed only for "fresh" code; running the demos below on code you've already used will yield misleading results. When analyzing inference, you're advised to always start from a fresh session. +Compilation is needed only for "fresh" code; running the demos below on code you've already used will yield misleading results. When analyzing inference, you're advised to always start from a fresh session. See also the [comparison between SnoopCompile and JET](@ref jet). ### Add SnoopCompileCore, SnoopCompile, and helper packages to your environment @@ -76,7 +76,7 @@ A non-empty list might indicate method invalidations, which can be checked (in a If you do have a lot of invalidations, [`precompile_blockers`](@ref) may be an effective way to reveal those invalidations that affect your particular package and workload. -## Viewing the results +## [Viewing the results](@id flamegraph) Let's start unpacking the output of `@snoop_inference` and see how to get more insight. First, notice that the output is an `InferenceTimingNode`: it's the root element of a tree of such nodes, all connected by caller-callee relationships. diff --git a/docs/src/snoop_inference_analysis.md b/docs/src/tutorials/snoop_inference_analysis.md similarity index 100% rename from docs/src/snoop_inference_analysis.md rename to docs/src/tutorials/snoop_inference_analysis.md diff --git a/docs/src/snoop_inference_parcel.md b/docs/src/tutorials/snoop_inference_parcel.md similarity index 62% rename from docs/src/snoop_inference_parcel.md rename to docs/src/tutorials/snoop_inference_parcel.md index 164b83d1..744e9a01 100644 --- a/docs/src/snoop_inference_parcel.md +++ b/docs/src/tutorials/snoop_inference_parcel.md @@ -1,65 +1,21 @@ -# [Using `@snoop_inference` results for precompilation](@id precompilation) +# [Using `@snoop_inference` to emit manual precompile directives](@id precompilation) -Improving inferrability, specialization, and precompilability may sometimes feel like "eating your vegetables": really good for you, but it sometimes feels like work. (Depending on tastes, of course; I love vegetables.) -While we've already gotten some payoff, now we're going to collect an additional reward for our hard work: the "dessert" of adding `precompile` directives. -It's worth emphasing that if we hadn't done the analysis of inference triggers and made improvements to our package, the benefit of adding `precompile` directives would have been substantially smaller. +In a few cases, it may be inconvenient or impossible to precompile using a [workload](https://julialang.github.io/PrecompileTools.jl/stable/#Tutorial:-forcing-precompilation-with-workloads). Some examples might be: +- an application that opens graphical windows +- an application that connects to a database +- an application that creates, deletes, or rewrites files on disk -## Running work +In such cases, one alternative is to create a manual list of precompile directives using Julia's `precompile(f, argtypes)` function. -One of the simplest ways to force precompilation is to execute code. This has several advantages: +!!! warning + Manual precompile directives are much more likely to "go stale" as the package is developed---`precompile` does not throw an error if a method for the given `argtypes` cannot be found. They are also more likely to be dependent on the Julia version, operating system, or CPU architecture. Whenever possible, it's safer to use a workload. -- It is typically more robust across Julia versions -- It automatically handles architecture differences like 32- vs 64-bit machines -- It precompiles even the runtime-dispatch dependencies of a command - if the dependent methods are in the same package. This typically - results in much shorter precompile files than those that explicitly - use `precompile`. - -This approach looks like the following: - -``` -module MyPkg - -# All of your code that defines `MyPkg` goes here - -# precompile as the final step of the module definition: -if ccall(:jl_generating_output, Cint, ()) == 1 # if we're precompiling the package - let - x = rand(Int, 5) - my_function(x) # this will force precompilation `my_function(::Vector{Int}`) - end -end - -end # module MyPkg -``` - -When your module is being precompiled (`[ Info: Precompiling MyPkg [...]`), just before the module "closes" your block of work will be executed. This forces compilation, and these compiled MethodInstances will be cached. - -After adding such directives, it's recommended to check the flamegraph again and see if there are any major omissions. You may need to add similar directives to some of the packages you depend on: precompilation is only effective if performed from the module that owns the method. (One advantage of `parcel` is that it automatically assigns `precompile` directives to the appropriate package.) - -!!! note - The work done inside this block is only executed when the package is - being precompiled, not when it is loaded with `using - MyPkg`. Precompilation essentially takes a "snapshot" of the - module; `using` just reloads that snapshot, it does not re-execute - all the commands used to produce that snapshot. - - The only role for the `ccall` is to prevent this work from being done - if you've started Julia with `--compiled-modules=no`. - -!!! warn - This style of precompilation may be undesirable or impossible if - your statements have side effects like opening new windows. In such - cases, you may be able to use it for lower-level calls. - -## Parcel - -`precompile` directives have to be emitted by the module that owns the method. +`precompile` directives have to be emitted by the module that owns the method and/or types. SnoopCompile comes with a tool, `parcel`, that splits out the "root-most" precompilable MethodInstances into their constituent modules. -In our case, since we've made almost every call precompilable, this will typically correspond to the bottom row of boxes in the flame graph. +This will typically correspond to the bottom row of boxes in the [flame graph](@ref flamegraph). In cases where you have some non-precompilable MethodInstances, they will include MethodInstances from higher up in the call tree. -Let's use `SnoopCompile.parcel` on `OptimizeMeFixed` in its current state: +Let's use `SnoopCompile.parcel` on [`OptimizeMeFixed`](@ref inferrability): ```julia julia> ttot, pcs = SnoopCompile.parcel(tinf); @@ -111,11 +67,6 @@ Base.Ryu: precompiled 0.15733664599999997 out of 0.15733664599999997 Main.OptimizeMeFixed: precompiled 0.4204474180000001 out of 0.4204474180000001 ``` -!!! tip - For packages that support just Julia 1.6 and higher, you may be able to slim down the precompile file by - adding `has_bodyfunction=true` to the arguments for `write`. - This setting applies for all packges in `pcs`, so you may need to call `write` twice (with both `false` and `true`) and select the appropriate precompile file for each package. - You'll now find a directory `/tmp/precompiles_OptimizeMe`, and inside you'll find three files, for `Base`, `Base.Ryu`, and `OptimizeMeFixed`, respectively. The contents of the last of these should be recognizable: @@ -132,27 +83,6 @@ The first `ccall` line ensures we only pay the cost of running these `precompile This file is ready to be moved into the `OptimizeMe` repository and `include`d into your module definition. Since we added `warmup` manually, you could consider moving `precompile(warmup, ())` into this function. -In general, it's recommended to run precompilation from inside a block - -```julia -if Base.VERSION >= v"1.4.2" - include("precompile.jl") - _precompile_() -end -``` - -because earlier versions of Julia occasionally crashed on certain precompile directives. -It's also perfectly fine to omit the function call, and use - -```julia -if Base.VERSION >= v"1.4.2" - Base.precompile(Tuple{typeof(main)}) # time: 0.4204474 - precompile(warmup, ()) -end -``` - -directly in the `OptimizeMeFixed` module, usually as the last block of the module definition. - You might also consider submitting some of the other files (or their `precompile` directives) to the packages you depend on. In some cases, the specific argument type combinations may be too "niche" to be worth specializing; one such case is found here, a `show` method for `Tuple{String, Int64}` for `Base`. But in other cases, these may be very worthy additions to the package. @@ -212,4 +142,3 @@ It's also worth appreciating how much we have succeeded in reducing latency, wit `@snoop_inference` collects enough data to learn which methods are triggering inference, how heavily methods are being specialized, and so on. Examining your code from the standpoint of inference and specialization may be unfamiliar at first, but like other aspects of package development (testing, documentation, and release compatibility management) it can lead to significant improvements in the quality-of-life for you and your users. -By optimizing your packages and then adding `precompile` directives, you can often cut down substantially on latency. diff --git a/docs/src/tutorials/snoop_llvm.md b/docs/src/tutorials/snoop_llvm.md new file mode 100644 index 00000000..3e16c063 --- /dev/null +++ b/docs/src/tutorials/snoop_llvm.md @@ -0,0 +1,37 @@ +# Tutorial on `@snoop_llvm` + +Julia uses the [LLVM compiler](https://llvm.org/) to generate machine code. Typically, the two main contributors to the overall compile time are inference and LLVM, and thus together `@snoop_inference` and `@snoop_llvm` collect fairly comprehensive data on the compiler. + +`@snoop_llvm` has a somewhat different design than `@snoop_inference`: while `@snoop_inference` runs in the same session that you'll be using for analysis (and thus requires that you remember to do the data gathering in a fresh session), `@snoop_llvm` spawns a fresh process to collect the data. The downside is that you get less interactivity, as the data have to be written out in intermediate forms as a text file. + +### Add SnoopCompileCore and SnoopCompile to your environment + +Here, we'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/). + +```@repl +using Pkg +Pkg.add(["SnoopCompileCore", "SnoopCompile"]) +``` + +## Collecting the data + +Here's a simple demonstration of usage: + +```@repl tutorial-llvm +using SnoopCompileCore +@snoop_llvm "func_names.csv" "llvm_timings.yaml" begin + using InteractiveUtils + @eval InteractiveUtils.peakflops() +end + +using SnoopCompile +times, info = SnoopCompile.read_snoop_llvm("func_names.csv", "llvm_timings.yaml", tmin_secs = 0.025); +``` + +This will write two files, `"func_names.csv"` and `"llvm_timings.yaml"`, in your current working directory. Let's look at what was read from these files: + +```@repl tutorial-llvm +times +info +``` + diff --git a/src/utils.jl b/src/utils.jl index 8b2d9e1b..e2335474 100644 --- a/src/utils.jl +++ b/src/utils.jl @@ -128,7 +128,7 @@ function add_repr!(list, modgens::Dict{Module, Vector{Method}}, mi::MethodInstan return add_if_evals!(list, topmod, reprcontext(topmod, p), paramrepr, tt, check_eval = check_eval, time=time) end -function handle_kwbody(topmod::Module, m::Method, paramrepr, tt, fstr="fbody"; check_eval = true, has_bodyfunction::Bool=false) +function handle_kwbody(topmod::Module, m::Method, paramrepr, tt, fstr="fbody"; check_eval = true) nameparent = Symbol(match(r"^#([^#]*)#", String(m.name)).captures[1]) if !isdefined(m.module, nameparent) @debug "Module $topmod: skipping $m due to inability to look up kwbody parent" # see example related to issue #237 @@ -140,17 +140,12 @@ function handle_kwbody(topmod::Module, m::Method, paramrepr, tt, fstr="fbody"; c can1, exc1 = can_eval(topmod, whichstr, check_eval) if can1 ttstr = tuplestring(paramrepr) - pcstr = has_bodyfunction ? """ + pcstr = """ let fbody = try Base.bodyfunction($whichstr) catch missing end if !ismissing(fbody) precompile($fstr, $ttstr) end - end""" : """ - let fbody = try __lookup_kwbody__($whichstr) catch missing end - if !ismissing(fbody) - precompile($fstr, $ttstr) - end - end""" # extra indentation because `write` will indent 1st line + end""" can2, exc2 = can_eval(topmod, pcstr, check_eval) if can2 return pcstr diff --git a/test/snoop_inference.jl b/test/snoop_inference.jl index 85e1fd04..58136c85 100644 --- a/test/snoop_inference.jl +++ b/test/snoop_inference.jl @@ -730,7 +730,7 @@ include("testmodules/SnoopBench.jl") SnoopCompile.write(io, tmis; tmin=0.0) str = String(take!(io)) @test occursin("__lookup_kwbody__", str) - SnoopCompile.write(io, tmis; tmin=0.0, has_bodyfunction=true) + SnoopCompile.write(io, tmis; tmin=0.0) str = String(take!(io)) @test !occursin("__lookup_kwbody__", str) From 3e1f57e0e376f57a11b382c4c8fef8ee9cc99175 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 07:25:39 -0500 Subject: [PATCH 06/18] fixes --- SnoopCompileCore/src/snoop_inference.jl | 44 +++++++++-------- SnoopCompileCore/src/snoop_invalidations.jl | 14 +++--- SnoopCompileCore/src/snoop_llvm.jl | 9 ++-- docs/make.jl | 7 +-- docs/src/explanations/gotchas.md | 4 +- docs/src/explanations/tools.md | 8 ++-- docs/src/index.md | 2 +- docs/src/reference.md | 7 +-- docs/src/tutorials/invalidations.md | 8 ++-- docs/src/tutorials/pgdsgui.md | 48 +++++++++++++++---- docs/src/tutorials/snoop_inference.md | 4 +- .../src/tutorials/snoop_inference_analysis.md | 21 ++------ docs/src/tutorials/snoop_llvm.md | 2 +- ext/SCPyPlotExt.jl | 18 ------- src/SnoopCompile.jl | 20 +++++++- 15 files changed, 118 insertions(+), 98 deletions(-) diff --git a/SnoopCompileCore/src/snoop_inference.jl b/SnoopCompileCore/src/snoop_inference.jl index 76c9f0e8..08398f4c 100644 --- a/SnoopCompileCore/src/snoop_inference.jl +++ b/SnoopCompileCore/src/snoop_inference.jl @@ -98,36 +98,40 @@ function _snoop_inference(cmd::Expr) end """ - tinf = @snoop_inference commands + tinf = @snoop_inference commands; -Produce a profile of julia's type inference, recording the amount of time spent inferring -every `MethodInstance` processed while executing `commands`. Each fresh entrance to -type inference (whether executed directly in `commands` or because a call was made -by runtime-dispatch) also collects a backtrace so the caller can be identified. +Produce a profile of julia's type inference, recording the amount of time spent +inferring every `MethodInstance` processed while executing `commands`. Each +fresh entrance to type inference (whether executed directly in `commands` or +because a call was made by runtime-dispatch) also collects a backtrace so the +caller can be identified. -`tinf` is a tree, each node containing data on a particular inference "frame" (the method, -argument-type specializations, parameters, and even any constant-propagated values). -Each reports the [`exclusive`](@ref)/[`inclusive`](@ref) times, where the exclusive -time corresponds to the time spent inferring this frame in and of itself, whereas -the inclusive time includes the time needed to infer all the callees of this frame. +`tinf` is a tree, each node containing data on a particular inference "frame" +(the method, argument-type specializations, parameters, and even any +constant-propagated values). Each reports the +[`exclusive`](@ref)/[`inclusive`](@ref) times, where the exclusive time +corresponds to the time spent inferring this frame in and of itself, whereas the +inclusive time includes the time needed to infer all the callees of this frame. The top-level node in this profile tree is `ROOT`. Uniquely, its exclusive time -corresponds to the time spent _not_ in julia's type inference (codegen, llvm_opt, runtime, etc). +corresponds to the time spent _not_ in julia's type inference (codegen, +llvm_opt, runtime, etc). -There are many different ways of inspecting and using the data stored in `tinf`. -The simplest is to load the `AbstracTrees` package and display the tree with -`AbstractTrees.print_tree(tinf)`. -See also: `flamegraph`, `flatten`, `inference_triggers`, `SnoopCompile.parcel`, -`runtime_inferencetime`. +Working with `tinf` effectively requires loading `SnoopCompile`. + +!!! warning + Note the semicolon `;` at the end of the `@snoop_inference` macro call. + Because `SnoopCompileCore` is not permitted to invalidate any code, it cannot define + the `Base.show` methods that pretty-print `tinf`. Defer inspection of `tinf` + until `SnoopCompile` has been loaded. # Example -```jldoctest; setup=:(using SnoopCompile), filter=r"([0-9]*\\.?[0-9]+([eE][-+]?[0-9]+)?/[0-9]*\\.?[0-9]+([eE][-+]?[0-9]+)?|\\d direct)" + +```jldoctest; setup=:(using SnoopCompileCore), filter=r"([0-9]*\\.?[0-9]+([eE][-+]?[0-9]+)?/[0-9]*\\.?[0-9]+([eE][-+]?[0-9]+)?|\\d direct)" julia> tinf = @snoop_inference begin sort(rand(100)) # Evaluate some code and profile julia's type inference - end -InferenceTimingNode: 0.110018224/0.131464476 on Core.Compiler.Timings.ROOT() with 2 direct children + end; ``` - """ macro snoop_inference(cmd) return _snoop_inference(cmd) diff --git a/SnoopCompileCore/src/snoop_invalidations.jl b/SnoopCompileCore/src/snoop_invalidations.jl index 5119e678..25fe5899 100644 --- a/SnoopCompileCore/src/snoop_invalidations.jl +++ b/SnoopCompileCore/src/snoop_invalidations.jl @@ -1,18 +1,18 @@ export @snoop_invalidations """ - list = @snoop_invalidations expr + invs = @snoop_invalidations expr Capture method cache invalidations triggered by evaluating `expr`. -`list` is a sequence of invalidated `Core.MethodInstance`s together with "explanations," consisting +`invs` is a sequence of invalidated `Core.MethodInstance`s together with "explanations," consisting of integers (encoding depth) and strings (documenting the source of an invalidation). -Unless you are working at a low level, you essentially always want to pass `list` +Unless you are working at a low level, you essentially always want to pass `invs` directly to [`SnoopCompile.invalidation_trees`](@ref). # Extended help -`list` is in a format where the "reason" comes after the items. +`invs` is in a format where the "reason" comes after the items. Method deletion results in the sequence [zero or more (mi, "invalidate_mt_cache") pairs..., zero or more (depth1 tree, loctag) pairs..., method, loctag] with loctag = "jl_method_table_disable" @@ -22,14 +22,16 @@ where `mi` means a `MethodInstance`. `depth1` means a sequence starting at `dept Method insertion results in the sequence [zero or more (depth0 tree, sig) pairs..., same info as with delete_method except loctag = "jl_method_table_insert"] + +The authoritative reference is Julia's own `src/gf.c` file. """ macro snoop_invalidations(expr) quote - local list = ccall(:jl_debug_method_invalidation, Any, (Cint,), 1) + local invs = ccall(:jl_debug_method_invalidation, Any, (Cint,), 1) Expr(:tryfinally, $(esc(expr)), ccall(:jl_debug_method_invalidation, Any, (Cint,), 0) ) - list + invs end end diff --git a/SnoopCompileCore/src/snoop_llvm.jl b/SnoopCompileCore/src/snoop_llvm.jl index 8e2bc0fa..602cf3de 100644 --- a/SnoopCompileCore/src/snoop_llvm.jl +++ b/SnoopCompileCore/src/snoop_llvm.jl @@ -3,11 +3,10 @@ export @snoop_llvm using Serialization """ -``` -@snoop_llvm "func_names.csv" "llvm_timings.yaml" begin - # Commands to execute, in a new process -end -``` + @snoop_llvm "func_names.csv" "llvm_timings.yaml" begin + # Commands to execute, in a new process + end + causes the julia compiler to log timing information for LLVM optimization during the provided commands to the files "func_names.csv" and "llvm_timings.yaml". These files can be used for the input to `SnoopCompile.read_snoop_llvm("func_names.csv", "llvm_timings.yaml")`. diff --git a/docs/make.jl b/docs/make.jl index 72d1b80d..141a4131 100644 --- a/docs/make.jl +++ b/docs/make.jl @@ -1,4 +1,5 @@ using Documenter +using SnoopCompileCore using SnoopCompile import PyPlot # so that the visualizations.jl file is loaded @@ -8,13 +9,13 @@ makedocs( prettyurls = get(ENV, "CI", nothing) == "true" ), modules = [SnoopCompile.SnoopCompileCore, SnoopCompile], - linkcheck = false, # FIXME make true + linkcheck = true, # the link check is slow, set to false if you're building frequently # doctest = :fix, - warnonly=true, + warnonly=true, # delete when https://github.com/JuliaDocs/Documenter.jl/issues/2541 is fixed pages = ["index.md", "Basic tutorials" => ["tutorials/invalidations.md", "tutorials/snoop_inference.md", "tutorials/snoop_llvm.md", "tutorials/pgdsgui.md", "tutorials/jet.md"], "Advanced tutorials" => ["tutorials/snoop_inference_analysis.md", "tutorials/snoop_inference_parcel.md"], - "Explanations" => ["tools.md", "gotchas.md", "explanations/fixing_inference.md"], + "Explanations" => ["explanations/tools.md", "explanations/gotchas.md", "explanations/fixing_inference.md"], "reference.md", ] ) diff --git a/docs/src/explanations/gotchas.md b/docs/src/explanations/gotchas.md index d728e12d..d49a0b02 100644 --- a/docs/src/explanations/gotchas.md +++ b/docs/src/explanations/gotchas.md @@ -1,6 +1,6 @@ # Precompilation "gotcha"s -## [Running code during module definition](@ref running-during-pc) +## [Running code during module definition](@id running-during-pc) Suppose you're working on an astronomy package and your source code has a line @@ -14,7 +14,7 @@ However, two circumstances can lead to puzzling omissions from the cache files: - if `makeplanet` is a method defined in a dependency of your package, it will *not* be cached in your package. You'd want to add precompilation of `makeplanet` to the package that creates that method. - if `makeplanet` is poorly-infered and uses runtime dispatch, any such callees that are not owned by your package will not be cached. For example, suppose `makeplanet` ends up calling methods in Base Julia or its standard libraries that are not precompiled into Julia itself: the compiled code for those methods will not be added to the cache file. -One option to ensure this dependent code gets cached is to create `planets` inside `@compile_workload`: +One option to ensure this dependent code gets cached is to create `planets` inside `PrecompileTools.@compile_workload`: ``` @compile_workload begin diff --git a/docs/src/explanations/tools.md b/docs/src/explanations/tools.md index 54bdff88..cdb1dcaa 100644 --- a/docs/src/explanations/tools.md +++ b/docs/src/explanations/tools.md @@ -20,10 +20,10 @@ and wait to load SnoopCompile until after you've finished collecting the data. ## JET -[JET](https://github.com/aviatesk/JET.jl) is perhaps the main alternative to SnoopCompile. The packages have some overlap in what they can tell you about your code, but their mechanisms of action are fundamentally different: +[JET](https://github.com/aviatesk/JET.jl) is a powerful developer tool that in some ways is an alternative to SnoopCompile. While the two have different goals, the packages have some overlap in what they can tell you about your code. However, their mechanisms of action are fundamentally different: -- JET is a "static analyzer," which means that it analyzes the code itself. JET can tell you about inference failures (runtime dispatch) much like SnoopCompile, with one major advantage: SnoopCompileCore requires you to use `@snoop_inference` in a fresh session and omits information about any callees that have been compiled previously, whereas JET's `@report_opt` provides exhaustive information about the entire *inferable* callgraph (i.e., the part of the callgraph that inference can predict from the initial call). +- JET is a "static analyzer," which means that it analyzes the code itself. JET can tell you about inference failures (runtime dispatch) much like SnoopCompile, with a major advantage: SnoopCompileCore omits information about any callees that are already compiled, but JET's `@report_opt` provides *exhaustive* information about the entire *inferable* callgraph (i.e., the part of the callgraph that inference can predict from the initial call) regardless of whether it has been previously compiled. With JET, you don't have to remember to run each analysis in a fresh session. -- SnoopCompileCore collects data by watching normal inference at work. On code that hasn't been compiled previously, this can yield results similar to JET's, with a different major advantage: SnoopCompileCore can "see through" runtime dispatch, and provide insights about callees that are invisible to JET. +- SnoopCompileCore collects data by watching normal inference at work. On code that hasn't been compiled previously, this can yield results similar to JET's, with a different major advantage: JET can't "see through" runtime dispatch, but SnoopCompileCore can. With SnoopCompile, you can immediately get a wholistic view of your entire callgraph. -Perhaps surprisingly, combining JET and SnoopCompile can provide insights that are difficult to obtain with either package in isolation. See the [Tutorial on JET integration](@ref). \ No newline at end of file +Combining JET and SnoopCompile can provide insights that are difficult to obtain with either package in isolation. See the [Tutorial on JET integration](@ref). diff --git a/docs/src/index.md b/docs/src/index.md index 21e11cdb..2c5833dd 100644 --- a/docs/src/index.md +++ b/docs/src/index.md @@ -19,7 +19,7 @@ SnoopCompile "snoops" on the Julia compiler, collecting information that may be - diagnose *invalidations*, cases where Julia must throw away previously-compiled code (see [Tutorial on `@snoop_invalidations`](@ref)) - trace *inference*, to learn what code is being newly (or freshly) analyzed in an early stage of the compilation pipeline ([Tutorial on `@snoop_inference`](@ref)) - trace *code generation by LLVM*, a late stage in the compilation pipeline ([Tutorial on `@snoop_llvm`](@ref)) -- reveal methods with excessive numbers of compiler-generated specializations, a.k.a.*profile-guided despecialization* ([Tutorial on PGDS](@ref)) +- reveal methods with excessive numbers of compiler-generated specializations, a.k.a.*profile-guided despecialization* ([Tutorial on PGDS](@ref pgds)) - integrate with tools like [JET](https://github.com/aviatesk/JET.jl) to help reduce the risk that your lovingly-precompiled code will be invalidated by loading other packages ([Tutorial on JET integration](@ref)) ## Background information diff --git a/docs/src/reference.md b/docs/src/reference.md index 44551760..860f4e2a 100644 --- a/docs/src/reference.md +++ b/docs/src/reference.md @@ -3,9 +3,9 @@ ## Data collection ```@docs -@snoop_invalidations -@snoop_inference -@snoop_llvm +SnoopCompileCore.@snoop_invalidations +SnoopCompileCore.@snoop_inference +SnoopCompileCore.@snoop_llvm ``` ## GUIs @@ -23,6 +23,7 @@ invalidation_trees precompile_blockers filtermod findcaller +report_invalidations ``` ## Analysis of `@snoop_inference` diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md index 10b390c6..5596ebd8 100644 --- a/docs/src/tutorials/invalidations.md +++ b/docs/src/tutorials/invalidations.md @@ -2,7 +2,7 @@ ## What are invalidations? -Invalidations result from interactions between different pieces of code. Invalidations are essential to make Julia fast, interactive, and correct: you *need* invalidations if you want to be able to define some methods, run (compile) some code, and then in the same session define new methods and get answers that account for the new methods. +In this context, *invalidation* means discarding previously-compiled code. Invalidations occur because of interactions between independent pieces of code. Invalidations are essential to make Julia fast, interactive, and correct: you *need* invalidations if you want to be able to define some methods, run (compile) some code, and then in the same session define new methods that might lead to different answers if you were to recompile the code in the presence of the new methods. Invalidations can happen just from loading packages. Packages are precompiled in isolation, but you can load many packages into a single interactive session. It's impossible for the individual packages to anticipate the full "world of methods" in your interactive session, so sometimes Julia has to discard code that was compiled in a smaller world because it's at risk for being incorrect in the larger world. @@ -25,7 +25,7 @@ Here, we'll add these packages to your [default environment](https://pkgdocs.jul ```@repl using Pkg -Pkg.add(["SnoopCompileCore", "SnoopCompile", "AbstractTrees", "Cthulhu"]) +Pkg.add(["SnoopCompileCore", "SnoopCompile", "AbstractTrees", "Cthulhu"]); ``` ### Create the demonstration packages @@ -43,7 +43,7 @@ cd(mktempdir()) using Pkg Pkg.generate("Blackjack"); Pkg.activate("Blackjack") -Pkg.add("PrecompileTools") +Pkg.add("PrecompileTools"); Pkg.generate("BlackjackFacecards"); Pkg.activate("BlackjackFacecards") Pkg.develop(PackageSpec(path=joinpath(pwd(), "Blackjack"))); @@ -195,7 +195,7 @@ Choose a call for analysis (q to quit): This is an interactive REPL-menu, described more completely (via text and video) at [ascend](https://github.com/JuliaDebug/Cthulhu.jl?tab=readme-ov-file#usage-ascend). -There are quite a few other tools for working with `invs` and `trees`, see the [Invalidations reference](@ref). If your list of invalidations is dauntingly large, you may be interested in [Combining data streams to focus on important invalidations](@ref). +There are quite a few other tools for working with `invs` and `trees`, see the [Reference](@ref). If your list of invalidations is dauntingly large, you may be interested in [Combining data streams to focus on important invalidations](@ref). ### Why the invalidations occur diff --git a/docs/src/tutorials/pgdsgui.md b/docs/src/tutorials/pgdsgui.md index 8b8f94a5..9e10b385 100644 --- a/docs/src/tutorials/pgdsgui.md +++ b/docs/src/tutorials/pgdsgui.md @@ -1,6 +1,14 @@ # [Profile-guided despecialization](@id pgds) -As indicated in the [workflow](@ref), one of the important early steps is to evaluate and potentially adjust method specialization. +Julia's multiple dispatch allows developers to create methods for specific argument types. On top of this, the Julia compiler performs *automatic specialization*: + +``` +function countnonzeros(A::AbstractArray) + ... +end +``` + +will be compiled separately for `Vector{Int}`, `Matrix{Float64}`, `SubArray{...}`, and so on, if it gets called for each of these types. Each specialization (each `MethodInstance` with different argument types) costs extra inference and code-generation time, so while specialization often improves runtime performance, that has to be weighed against the cost in latency. There are also cases in which [overspecialization can hurt both run-time and compile-time performance](https://docs.julialang.org/en/v1/manual/performance-tips/#The-dangers-of-abusing-multiple-dispatch-(aka,-more-on-types-with-values-as-parameters)). @@ -11,7 +19,7 @@ The name is a reference to a related technique, [profile-guided optimization](ht Both PGO and PGDS use runtime profiling to help guide decisions about code optimization. PGO is often used in languages whose default mode is to avoid specialization, whereas PGDS seems more appropriate for a language like Julia which specializes by default. -While PGO is sometimes an automatic part of the compiler that optimizes code midstream during execution, PGDS is a tool for making static changes in code. +While PGO is sometimes an automatic part of the compiler that optimizes code midstream during execution, SnoopCompile's PGDS is a tool for making static changes (edits) to code. Again, this seems appropriate for a language where specialization typically happens prior to the first execution of the code. ## Using the PGDS graphical user interface @@ -20,7 +28,7 @@ To illustrate the use of PGDS, we'll examine an example in which some methods ge To keep this example short, we'll create functions that operate on types themselves. !!! note - For a `DataType` `T`, `T.name` returns a `Core.TypeName`, and `T.name.name` returns the name as a `Symbol`. + As background to this example, for a `DataType` `T`, `T.name` returns a `Core.TypeName`, and `T.name.name` returns the name as a `Symbol`. `Base.unwrap_unionall(T)` preserves `DataType`s as-is, but converts a `UnionAll` type into a `DataType`. ```julia @@ -56,12 +64,11 @@ mappushes(f, src) = mappushes!(f, [], src) There are two stages to PGDS: first (and preferrably starting in a fresh Julia session), we profile type-inference: ```julia -julia> using SnoopCompile +julia> using SnoopCompileCore julia> Ts = subtypes(Any); # get a long list of different types -julia> tinf = @snoop_inference mappushes(spelltype, Ts) -InferenceTimingNode: 4.476700/5.591207 on InferenceFrameInfo for Core.Compiler.Timings.ROOT() with 587 direct children +julia> tinf = @snoop_inference mappushes(spelltype, Ts); ``` Then, *in the same session*, profile the runtime: @@ -78,6 +85,8 @@ get a realistic view of where your code spends its time during actual use. Now let's launch the PDGS GUI: ```julia +julia> using SnoopCompile + julia> import PyPlot # the GUI is dependent on PyPlot, must load it before the next line julia> mref, ax = pgdsgui(tinf); @@ -85,7 +94,7 @@ julia> mref, ax = pgdsgui(tinf); You should see something like this: -![pgdsgui](assets/pgds_spec.png) +![pgdsgui](../assets/pgds_spec.png) In this graph, each dot corresponds to a single method; for this method, we plot inference time (vertical axis) against the run time (horizontal axis). The coloration of each dot encodes the number of specializations (the number of distinct `MethodInstance`s) for that method; @@ -151,14 +160,14 @@ end ``` !!! warning - `where` type-parameters force specialization, regardless of `@nospecialize`: in `spelltype(@nospecialize(::Type{T})) where T`, the `@nospecialize` has no impact and you'll get full specialization on `T`. + `where` type-parameters force specialization: regardless of `@nospecialize`: in `spelltype(@nospecialize(::Type{T})) where T`, the `@nospecialize` has no impact and you'll get full specialization on `T`. Instead, use `@nospecialize(T::Type)` as shown. If we now rerun that demo, you should see a plot of the same kind as shown above, but with different costs for each dot. The differences are best appreciated comparing them side-by-side ([`pgdsgui`](@ref) allows you to specify a particular axis into which to plot): -![pgdsgui-compare](assets/pgds_compareplots.png) +![pgdsgui-compare](../assets/pgds_compareplots.png) The results with `@nospecialize` are shown on the right. You can see that: @@ -173,11 +182,30 @@ Reducing specialization, when appropriate, can often yield your biggest reductio !!! tip When you add `@nospecialize`, sometimes it's beneficial to compensate for the loss of inferrability by adding some type assertions. This topic will be discussed in greater detail in the next section, but for the example above we can improve runtime performance by annotating the return type of `Base.unwrap_unionall(T)`: `name = (Base.unwrap_unionall(T)::DataType).name.name`. - Then, later lines in `spell` know that `name` is a `Symbol`. + Then, later lines in `spelltype` know that `name` is a `Symbol`. With this change, the unspecialized variant outperforms the specialized variant in *both compile-time and run-time*. The reason is that the specialized variant of `spell` needs to be called by runtime dispatch, whereas for the unspecialized variant there's only one `MethodInstance`, so its dispatch is handled at compile time. +### Blocking inference: `Base.@nospecializeinfer` + +Perhaps surprisingly, `@nospecialize` doesn't prevent Julia's type-inference from inspecting a method. The reason is that it's sometimes useful if the *caller* knows what type will be returned, even if the *callee* doesn't exploit this information. In our `mappushes` example, this isn't an issue, because `Ts` is a `Vector{Any}` and this already defeats inference. But in other cases, the caller may be inferable but (to save inference time) you'd prefer to block inference from inspecting the method. + +Beginning with Julia 1.10, you can prevent even inference from "looking at" `@nospecialize`d arguments with `Base.@nospecializeinfer`: + +``` +Base.@nospecializeinfer function spelltype(@nospecialize(T::Type)) + name = (Base.unwrap_unionall(T)::DataType).name.name + str = "" + for c in string(name) + str *= c + end + return str +end +``` + +Note that the `::DataType` annotation described in the tip above is still effective and recommended. `@nospecializeinfer` directly affects only arguments that are marked with `@nospecialize`, and in this case the type-assertion prevents type uncertainty from propagating to the remainder of the function. + ### Argument standardization While not immediately relevant to the example above, a very important technique that falls within the domain of reducing specialization is *argument standardization*: instead of diff --git a/docs/src/tutorials/snoop_inference.md b/docs/src/tutorials/snoop_inference.md index 0e9beb23..1d306e24 100644 --- a/docs/src/tutorials/snoop_inference.md +++ b/docs/src/tutorials/snoop_inference.md @@ -2,7 +2,7 @@ Inference may occur when you *run* code. Inference is the first step of *type-specialized* compilation. `@snoop_inference` collects data on what inference is doing, giving you greater insight into what is being inferred and how long it takes. -Compilation is needed only for "fresh" code; running the demos below on code you've already used will yield misleading results. When analyzing inference, you're advised to always start from a fresh session. See also the [comparison between SnoopCompile and JET](@ref jet). +Compilation is needed only for "fresh" code; running the demos below on code you've already used will yield misleading results. When analyzing inference, you're advised to always start from a fresh session. See also the [comparison between SnoopCompile and JET](@ref). ### Add SnoopCompileCore, SnoopCompile, and helper packages to your environment @@ -10,7 +10,7 @@ Here, we'll add these packages to your [default environment](https://pkgdocs.jul ```@repl using Pkg -Pkg.add(["SnoopCompileCore", "SnoopCompile", "AbstractTrees", "ProfileView"]) +Pkg.add(["SnoopCompileCore", "SnoopCompile", "AbstractTrees", "ProfileView"]); ``` ## Setting up the demo diff --git a/docs/src/tutorials/snoop_inference_analysis.md b/docs/src/tutorials/snoop_inference_analysis.md index 36cda6c8..637e8574 100644 --- a/docs/src/tutorials/snoop_inference_analysis.md +++ b/docs/src/tutorials/snoop_inference_analysis.md @@ -1,21 +1,5 @@ # [Using `@snoop_inference` results to improve inferrability](@id inferrability) -As indicated in the [workflow](@ref), the recommended steps to reduce latency are: - -- check for invalidations -- adjust method specialization in your package or its dependencies -- fix problems in type inference -- add `precompile` directives - -The importance of fixing "problems" in type-inference was indicated in the [tutorial](@ref): successful precompilation requires a chain of ownership, but runtime dispatch (when inference cannot predict the callee) results in breaks in this chain. By improving inferrability, you can convert short, unconnected call-trees into a smaller number of large call-trees that all link back to your package(s). - -In practice, it also turns out that opportunities to adjust specialization are often revealed by analyzing inference failures, so this page is complementary to the previous one. - -Finally, improving inference may also yield improvements in runtime performance, itself an excellent outcome. - -!!! note - [JET also detects inference failures](https://aviatesk.github.io/JET.jl/dev/optanalysis/), but JET and SnoopCompile use different mechanisms: JET performs *static* analysis of a particular call, while SnoopCompile performs *dynamic* analysis of new inference. As a consequence, JET's detection of inference failures is reproducible (you can run the same analysis repeatedly and get the same result) but terminates at any non-inferrable node of the call graph: you will miss runtime dispatch in any non-inferrable callees. Conversely, SnoopCompile's detection of inference failures can explore the entire callgraph, but only for those portions that have not been previously inferred, and the analysis cannot be repeated in the same session. - Throughout this page, we'll use the `OptimizeMe` demo, which ships with `SnoopCompile`. !!! note @@ -56,7 +40,7 @@ Node(FlameGraphs.NodeData(ROOT() at typeinfer.jl:75, 0x00, 0:2713559552)) If you visualize `fg` with ProfileView, you'll see something like this: -![flamegraph-OptimizeMe](assets/flamegraph-OptimizeMe.png) +![flamegraph-OptimizeMe](../assets/flamegraph-OptimizeMe.png) From the standpoint of precompilation, this has some obvious problems: @@ -630,6 +614,7 @@ The call to `show` comes from `show(io, mime, x[])`. This implementation uses a clever trick, wrapping `x` in a `Ref{Any}(x)`, to prevent specialization of the method defined by the `do` block on the specific type of `x`. This trick is designed to limit the number of `MethodInstance`s inferred for this `display` method. + Unfortunately, from the standpoint of precompilation we have something of a conundrum. It turns out that this trigger corresponds to the first of the big red flames in the flame graph. `show(::IOContext{Base.TTY}, ::MIME{Symbol("text/plain")}, ::Vector{Main.OptimizeMe.Container{Any}})` is not precompilable because `Base` owns the `show` method for `Vector`; @@ -664,7 +649,7 @@ precompile(warmup, ()) We handled not just `Vector{Container{Any}}` but also `Vector{Object}`, since that turns out to correspond to the other wide block of red bars. If you make this change, start a fresh session, and recreate the flame graph, you'll see that the wide red flames are gone: -![flamegraph-OptimizeMeFixed](assets/flamegraph-OptimizeMeFixed.png) +![flamegraph-OptimizeMeFixed](../assets/flamegraph-OptimizeMeFixed.png) !!! info diff --git a/docs/src/tutorials/snoop_llvm.md b/docs/src/tutorials/snoop_llvm.md index 3e16c063..448f5225 100644 --- a/docs/src/tutorials/snoop_llvm.md +++ b/docs/src/tutorials/snoop_llvm.md @@ -10,7 +10,7 @@ Here, we'll add these packages to your [default environment](https://pkgdocs.jul ```@repl using Pkg -Pkg.add(["SnoopCompileCore", "SnoopCompile"]) +Pkg.add(["SnoopCompileCore", "SnoopCompile"]); ``` ## Collecting the data diff --git a/ext/SCPyPlotExt.jl b/ext/SCPyPlotExt.jl index 9c924735..fca26cca 100644 --- a/ext/SCPyPlotExt.jl +++ b/ext/SCPyPlotExt.jl @@ -7,24 +7,6 @@ using PyPlot: PyPlot, plt, PyCall get_bystr(@nospecialize(by)) = by === inclusive ? "Inclusive" : by === exclusive ? "Exclusive" : error("unknown ", by) -""" - methodref, ax = pgdsgui(tinf::InferenceTimingNode; consts::Bool=true, by=inclusive) - methodref = pgdsgui(ax, tinf::InferenceTimingNode; kwargs...) - -Create a scatter plot comparing: - - (vertical axis) the inference time for all instances of each Method, as captured by `tinf`; - - (horizontal axis) the run time cost, as estimated by capturing a `@profile` before calling this function. - -Each dot corresponds to a single method. The face color encodes the number of times that method was inferred, -and the edge color corresponds to the fraction of the runtime spent on runtime dispatch (black is 0%, bright red is 100%). -Clicking on a dot prints the method (or location, if inlined) to the REPL, and sets `methodref[]` to -that method. - -`ax` is the pyplot axis of the scatterplot. - -!!! compat - `pgdsgui` depends on PyPlot via the Requires.jl package. You must load both SnoopCompile and PyPlot for this function to be defined. -""" function pgdsgui(ax::PyCall.PyObject, ridata::AbstractVector{Pair{Union{Method,MethodLoc},PGDSData}}; bystr, consts, markersz=25, linewidth=0.5, t0 = 0.001, interactive::Bool=true, kwargs...) methodref = Ref{Union{Method,MethodLoc}}() # returned to the user for inspection of clicked methods function onclick(event) diff --git a/src/SnoopCompile.jl b/src/SnoopCompile.jl index b004ffa8..29878794 100644 --- a/src/SnoopCompile.jl +++ b/src/SnoopCompile.jl @@ -86,9 +86,27 @@ include("invalidation_and_inference.jl") export precompile_blockers # Write -# include("write.jl") +include("write.jl") # For PyPlot extension +""" + methodref, ax = pgdsgui(tinf::InferenceTimingNode; consts::Bool=true, by=inclusive) + methodref = pgdsgui(ax, tinf::InferenceTimingNode; kwargs...) + +Create a scatter plot comparing: + - (vertical axis) the inference time for all instances of each Method, as captured by `tinf`; + - (horizontal axis) the run time cost, as estimated by capturing a `@profile` before calling this function. + +Each dot corresponds to a single method. The face color encodes the number of times that method was inferred, +and the edge color corresponds to the fraction of the runtime spent on runtime dispatch (black is 0%, bright red is 100%). +Clicking on a dot prints the method (or location, if inlined) to the REPL, and sets `methodref[]` to +that method. + +`ax` is the pyplot axis of the scatterplot. + +!!! compat + `pgdsgui` depends on PyPlot via the Requires.jl package. You must load both SnoopCompile and PyPlot for this function to be defined. +""" function pgdsgui end export pgdsgui # For PrettyTables extension From 6bad8064039de97324ee700756e9ec3093804d32 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 07:32:56 -0500 Subject: [PATCH 07/18] fix test --- test/snoop_inference.jl | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/test/snoop_inference.jl b/test/snoop_inference.jl index 58136c85..356534f8 100644 --- a/test/snoop_inference.jl +++ b/test/snoop_inference.jl @@ -729,10 +729,7 @@ include("testmodules/SnoopBench.jl") io = IOBuffer() SnoopCompile.write(io, tmis; tmin=0.0) str = String(take!(io)) - @test occursin("__lookup_kwbody__", str) - SnoopCompile.write(io, tmis; tmin=0.0) - str = String(take!(io)) - @test !occursin("__lookup_kwbody__", str) + @test occursin("bodyfunction", str) A = [a] tinf = @snoop_inference SnoopBench.mappushes(identity, A) From c3da9bf0334f4afafb88c8d59670d138e9fd506c Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 08:01:04 -0500 Subject: [PATCH 08/18] Don't run `Pkg.add` --- docs/src/tutorials/invalidations.md | 2 +- docs/src/tutorials/snoop_inference.md | 4 ++-- docs/src/tutorials/snoop_llvm.md | 2 +- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md index 5596ebd8..5a804dc4 100644 --- a/docs/src/tutorials/invalidations.md +++ b/docs/src/tutorials/invalidations.md @@ -23,7 +23,7 @@ We'll illustrate invalidations by creating two packages, where loading the secon Here, we'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/). (With the exception of `AbstractTrees`, these "developer tool" packages should not be added to the Project file of any real packages unless you're extending the tool itself.) -```@repl +``` using Pkg Pkg.add(["SnoopCompileCore", "SnoopCompile", "AbstractTrees", "Cthulhu"]); ``` diff --git a/docs/src/tutorials/snoop_inference.md b/docs/src/tutorials/snoop_inference.md index 1d306e24..adad66a9 100644 --- a/docs/src/tutorials/snoop_inference.md +++ b/docs/src/tutorials/snoop_inference.md @@ -2,13 +2,13 @@ Inference may occur when you *run* code. Inference is the first step of *type-specialized* compilation. `@snoop_inference` collects data on what inference is doing, giving you greater insight into what is being inferred and how long it takes. -Compilation is needed only for "fresh" code; running the demos below on code you've already used will yield misleading results. When analyzing inference, you're advised to always start from a fresh session. See also the [comparison between SnoopCompile and JET](@ref). +Compilation is needed only for "fresh" code; running the demos below on code you've already used will yield misleading results. When analyzing inference, you're advised to always start from a fresh session. See also the [comparison between SnoopCompile and JET](@ref JET). ### Add SnoopCompileCore, SnoopCompile, and helper packages to your environment Here, we'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/). (With the exception of `AbstractTrees`, these "developer tool" packages should not be added to the Project file of any real packages unless you're extending the tool itself.) -```@repl +``` using Pkg Pkg.add(["SnoopCompileCore", "SnoopCompile", "AbstractTrees", "ProfileView"]); ``` diff --git a/docs/src/tutorials/snoop_llvm.md b/docs/src/tutorials/snoop_llvm.md index 448f5225..a1b2eacb 100644 --- a/docs/src/tutorials/snoop_llvm.md +++ b/docs/src/tutorials/snoop_llvm.md @@ -8,7 +8,7 @@ Julia uses the [LLVM compiler](https://llvm.org/) to generate machine code. Typi Here, we'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/). -```@repl +``` using Pkg Pkg.add(["SnoopCompileCore", "SnoopCompile"]); ``` From 735b7bd6bffdf2ad8e869ec468245c326792bcd3 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 10:23:22 -0500 Subject: [PATCH 09/18] Fix path on CI --- .github/workflows/Documenter.yml | 3 +++ docs/make.jl | 4 ++-- docs/src/tutorials/invalidations.md | 7 +++++++ 3 files changed, 12 insertions(+), 2 deletions(-) diff --git a/.github/workflows/Documenter.yml b/.github/workflows/Documenter.yml index 8bbf8bc7..ab189d12 100644 --- a/.github/workflows/Documenter.yml +++ b/.github/workflows/Documenter.yml @@ -26,6 +26,9 @@ jobs: version: '1' - run: julia --project -e 'using Pkg; Pkg.develop([PackageSpec(path=joinpath(pwd(), "SnoopCompileCore"))])' - uses: julia-actions/julia-buildpkg@latest + # To access the developer tools from within a package's environment, they should be in the default environment + - run: julia -e 'using Pkg; Pkg.develop([PackageSpec(path=joinpath(pwd(), "SnoopCompileCore")), PackageSpec(path=joinpath(pwd()))]); Pkg.instantiate()' + # Documenter wants them to be in the local environment - run: julia --project=docs/ -e 'using Pkg; Pkg.develop([PackageSpec(path=joinpath(pwd(), "SnoopCompileCore")), PackageSpec(path=joinpath(pwd()))]); Pkg.instantiate()' - uses: julia-actions/julia-docdeploy@releases/v1 env: diff --git a/docs/make.jl b/docs/make.jl index 141a4131..7fc7f401 100644 --- a/docs/make.jl +++ b/docs/make.jl @@ -6,9 +6,9 @@ import PyPlot # so that the visualizations.jl file is loaded makedocs( sitename = "SnoopCompile", format = Documenter.HTML( - prettyurls = get(ENV, "CI", nothing) == "true" + prettyurls = true, ), - modules = [SnoopCompile.SnoopCompileCore, SnoopCompile], + modules = [SnoopCompileCore, SnoopCompile], linkcheck = true, # the link check is slow, set to false if you're building frequently # doctest = :fix, warnonly=true, # delete when https://github.com/JuliaDocs/Documenter.jl/issues/2541 is fixed diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md index 5a804dc4..afb6b201 100644 --- a/docs/src/tutorials/invalidations.md +++ b/docs/src/tutorials/invalidations.md @@ -39,6 +39,7 @@ To create the (empty) packages, the code below executes the following steps: - make the second package (`BlackjackFacecards`) depend on the first one (`Blackjack`) ```@repl tutorial-invalidations +oldproj = Base.active_project() # hide cd(mktempdir()) using Pkg Pkg.generate("Blackjack"); @@ -132,8 +133,14 @@ Here are the steps executed by the code below using SnoopCompileCore invs = @snoop_invalidations using Blackjack, BlackjackFacecards; using SnoopCompile, AbstractTrees +Pkg.activate(oldproj) # hide ``` +!!! tip + If you get errors like `Package SnoopCompileCore not found in current path`, a likely explanation is that + you didn't add it to your default environment. In the example above, we're in the `BlackjackFacecards` environment + so we can develop the package, but you also need access to `SnoopCompile` and `SnoopCompileCore`. Having these in your [default environment](https://docs.julialang.org/en/v1/manual/code-loading/#Environment-stacks) lets them be found even if they aren't part of the current environment. + ### Analyzing invalidations Now we're ready to see what, if anything, got invalidated: From 74bf3801846bb7f4e025192ef30b7e45bdf0a701 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 10:25:45 -0500 Subject: [PATCH 10/18] Fix precompile_blockers refs --- docs/src/tutorials/invalidations.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md index afb6b201..06f807ae 100644 --- a/docs/src/tutorials/invalidations.md +++ b/docs/src/tutorials/invalidations.md @@ -202,7 +202,7 @@ Choose a call for analysis (q to quit): This is an interactive REPL-menu, described more completely (via text and video) at [ascend](https://github.com/JuliaDebug/Cthulhu.jl?tab=readme-ov-file#usage-ascend). -There are quite a few other tools for working with `invs` and `trees`, see the [Reference](@ref). If your list of invalidations is dauntingly large, you may be interested in [Combining data streams to focus on important invalidations](@ref). +There are quite a few other tools for working with `invs` and `trees`, see the [Reference](@ref). If your list of invalidations is dauntingly large, you may be interested in [precompile_blockers](@ref). ### Why the invalidations occur From 44a7721adc52781a82d4d36e55fe4da56d9858ff Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 10:53:12 -0500 Subject: [PATCH 11/18] More fix --- .github/workflows/Documenter.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/.github/workflows/Documenter.yml b/.github/workflows/Documenter.yml index ab189d12..59da4eaf 100644 --- a/.github/workflows/Documenter.yml +++ b/.github/workflows/Documenter.yml @@ -28,6 +28,8 @@ jobs: - uses: julia-actions/julia-buildpkg@latest # To access the developer tools from within a package's environment, they should be in the default environment - run: julia -e 'using Pkg; Pkg.develop([PackageSpec(path=joinpath(pwd(), "SnoopCompileCore")), PackageSpec(path=joinpath(pwd()))]); Pkg.instantiate()' + # Additional packages we'll need + - run: julia --project=docs/ -e 'using Pkg; Pkg.add(["AbstractTrees", "Cthulhu"]) # pyplot would be nice but it often errors # Documenter wants them to be in the local environment - run: julia --project=docs/ -e 'using Pkg; Pkg.develop([PackageSpec(path=joinpath(pwd(), "SnoopCompileCore")), PackageSpec(path=joinpath(pwd()))]); Pkg.instantiate()' - uses: julia-actions/julia-docdeploy@releases/v1 From 7ef63cdf8c9a6e8db75a4885386c82baf2e1a52e Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 10:53:30 -0500 Subject: [PATCH 12/18] Explicitly add needed packages for PGDS --- docs/src/tutorials/pgdsgui.md | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/docs/src/tutorials/pgdsgui.md b/docs/src/tutorials/pgdsgui.md index 9e10b385..a0341f2b 100644 --- a/docs/src/tutorials/pgdsgui.md +++ b/docs/src/tutorials/pgdsgui.md @@ -22,6 +22,17 @@ a language like Julia which specializes by default. While PGO is sometimes an automatic part of the compiler that optimizes code midstream during execution, SnoopCompile's PGDS is a tool for making static changes (edits) to code. Again, this seems appropriate for a language where specialization typically happens prior to the first execution of the code. +### Add SnoopCompileCore, SnoopCompile, and helper packages to your environment + +We'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/) so you can use them while in the package environment: + +``` +using Pkg +Pkg.add(["SnoopCompileCore", "SnoopCompile", "PyPlot"]); +``` + +PyPLot is used for the PGDS interface in part to reduce interference with native-Julia plotting packages like Makie--it's a little awkward to depend on a package that you might be simultaneously modifying! + ## Using the PGDS graphical user interface To illustrate the use of PGDS, we'll examine an example in which some methods get specialized for hundreds of types. From 3ba31c8358522544274b1b2f02b8a987188394a5 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 11:04:30 -0500 Subject: [PATCH 13/18] fix --- .github/workflows/Documenter.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/Documenter.yml b/.github/workflows/Documenter.yml index 59da4eaf..218dc550 100644 --- a/.github/workflows/Documenter.yml +++ b/.github/workflows/Documenter.yml @@ -29,7 +29,7 @@ jobs: # To access the developer tools from within a package's environment, they should be in the default environment - run: julia -e 'using Pkg; Pkg.develop([PackageSpec(path=joinpath(pwd(), "SnoopCompileCore")), PackageSpec(path=joinpath(pwd()))]); Pkg.instantiate()' # Additional packages we'll need - - run: julia --project=docs/ -e 'using Pkg; Pkg.add(["AbstractTrees", "Cthulhu"]) # pyplot would be nice but it often errors + - run: julia -e 'using Pkg; Pkg.add(["AbstractTrees", "Cthulhu"])' # pyplot would be nice but it often errors # Documenter wants them to be in the local environment - run: julia --project=docs/ -e 'using Pkg; Pkg.develop([PackageSpec(path=joinpath(pwd(), "SnoopCompileCore")), PackageSpec(path=joinpath(pwd()))]); Pkg.instantiate()' - uses: julia-actions/julia-docdeploy@releases/v1 From 373caaae9f684acd089c4d5360e87b8c0c016077 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 11:17:34 -0500 Subject: [PATCH 14/18] Move --- docs/src/tutorials/invalidations.md | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md index 06f807ae..235f49c6 100644 --- a/docs/src/tutorials/invalidations.md +++ b/docs/src/tutorials/invalidations.md @@ -133,7 +133,6 @@ Here are the steps executed by the code below using SnoopCompileCore invs = @snoop_invalidations using Blackjack, BlackjackFacecards; using SnoopCompile, AbstractTrees -Pkg.activate(oldproj) # hide ``` !!! tip @@ -277,3 +276,7 @@ before defining any `score` methods. You can read the documentation on `@max_met If you can't prevent the invalidation, an alternative approach is to recompile the invalidated code. For example, one could repeat the precompile workload from `Blackjack` in `BlackjackFacecards`. While this will mean that the whole "stack" will be compiled twice and cached twice (which is wasteful), it should be effective in reducing latency for users. PrecompileTools also has a `@recompile_invalidations`. This isn't generally recommended for use in package (you can end up with long compile times for things you don't need), but it can be useful in personal "Startup packages" where you want to reduce latency for a particular project you're working on. See the PrecompileTools documentation for details. + +```@repl tutorial-invalidations +Pkg.activate(oldproj) # hide +``` From 315e4c3c038887b6767efe6f84931f73732b63aa Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 23 Jul 2024 15:46:04 -0500 Subject: [PATCH 15/18] Polish --- docs/src/index.md | 8 +- docs/src/snoop_invalidations.md | 499 ------------------ docs/src/snoopi.md | 375 ------------- docs/src/tutorial.md | 257 --------- docs/src/tutorials/invalidations.md | 2 +- docs/src/tutorials/jet.md | 30 +- docs/src/tutorials/pgdsgui.md | 4 +- .../src/tutorials/snoop_inference_analysis.md | 1 - 8 files changed, 19 insertions(+), 1157 deletions(-) delete mode 100644 docs/src/snoop_invalidations.md delete mode 100644 docs/src/snoopi.md delete mode 100644 docs/src/tutorial.md diff --git a/docs/src/index.md b/docs/src/index.md index 2c5833dd..30971ce1 100644 --- a/docs/src/index.md +++ b/docs/src/index.md @@ -1,14 +1,14 @@ # SnoopCompile.jl -Julia is fast, but its execution speed depends on optimizing code through *compilation*. Code must be compiled before you can use it, and unfortunately compilation is slow. This can cause *latency* the first time you use code: this latency is often called *time-to-first-plot* (TTFP) or more generally *time-to-first-execution* (TTFX). If something feels slow the first time you use it, and fast thereafter, you're probably experiencing the latency of compilation. +Julia is fast, but its execution speed depends on optimizing code through *compilation*. Code must be compiled before you can use it, and unfortunately compilation is slow. This can cause *latency* the first time you use code: this latency is often called *time-to-first-plot* (TTFP) or more generally *time-to-first-execution* (TTFX). If something feels slow the first time you use it, and fast thereafter, you're probably experiencing the latency of compilation. Note that TTFX is distinct from time-to-load (TTL, which refers to the time you spend waiting for `using MyPkg` to finish), even though both contribute to latency. Modern versions of Julia can store compiled code to disk (*precompilation*) to reduce or eliminate latency. Users and developers who are interested in reducing TTFX should first head to [PrecompileTools](https://github.com/JuliaLang/PrecompileTools.jl), read its documentation thoroughly, and try using it to solve latency problems. This package, **SnoopCompile**, should be considered when: -- precompilation doesn't reduce latency as much as you wish +- precompilation doesn't reduce TTFX as much as you wish - precompilation "works," but only in isolation: as soon as you load (certain) additional packages, TTFX is bad again -- you're wondering if you can reduce the amount of time needed to precompile your package and/or the size of the shared library files +- you're wondering if you can reduce the amount of time needed to precompile your package and/or the size of the precompilation cache files In other words, SnoopCompile is a diagonostic package that helps reveal the causes of latency. Historically, it proceeded PrecompileTools, and indeed PrecompileTools was split out from SnoopCompile. Today, SnoopCompile is generally needed only when PrecompileTools fails to deliver the desired benefits. @@ -20,7 +20,7 @@ SnoopCompile "snoops" on the Julia compiler, collecting information that may be - trace *inference*, to learn what code is being newly (or freshly) analyzed in an early stage of the compilation pipeline ([Tutorial on `@snoop_inference`](@ref)) - trace *code generation by LLVM*, a late stage in the compilation pipeline ([Tutorial on `@snoop_llvm`](@ref)) - reveal methods with excessive numbers of compiler-generated specializations, a.k.a.*profile-guided despecialization* ([Tutorial on PGDS](@ref pgds)) -- integrate with tools like [JET](https://github.com/aviatesk/JET.jl) to help reduce the risk that your lovingly-precompiled code will be invalidated by loading other packages ([Tutorial on JET integration](@ref)) +- integrate with tools like [JET](https://github.com/aviatesk/JET.jl) to further reduce the risk that your lovingly-precompiled code will be invalidated by loading other packages ([Tutorial on JET integration](@ref)) ## Background information diff --git a/docs/src/snoop_invalidations.md b/docs/src/snoop_invalidations.md deleted file mode 100644 index a1c19d6d..00000000 --- a/docs/src/snoop_invalidations.md +++ /dev/null @@ -1,499 +0,0 @@ -# [Snooping on and fixing invalidations: `@snoop_invalidations`](@id invalidations) - -!!! compat - `@snoop_invalidations` is available on `Julia 1.6.0-DEV.154` or above, but the results can be relevant for all Julia versions. - -Invalidations occur when there is a danger that new methods would supersede older methods in previously-compiled code. -For safety, Julia's compiler *invalidates* that old code, marking it as unsuitable for use; the next time you call -that method, it will have to be compiled again from scratch. (If no one ever needs that method again, there is no -major loss.) - -Some packages define new methods that force invalidation of previously-compiled code. If your package, or any of your -dependencies, triggers many invalidations, it has several bad effects: - -- any invalidated methods you need for the functionality in your package will have to be recompiled. - This will lead to a direct (and occasionally large) slowdown for your package. -- invalidations by your dependencies (packages you rely on) can block precompilation of methods in your package, - preventing you from taking advantage of some of the other features of SnoopCompile. -- even if you don't need the invalidated code for your package, any invalidations triggered by your package - might harm packages that depend on yours. - -For these reasons, it's advisable to begin by analyzing invalidations. -On recent Julia versions, most packages do not trigger a large number of invalidations; often, all that is needed is a quick glance at invalidations before moving on to the next step. -Occasionally, checking for invalidations can save you a lot of confusion and frustration at later steps, so it is well worth taking a look. - -Readers who want more background and context are encouraged to read [this blog post](https://julialang.org/blog/2020/08/invalidations/). - -!!! note - Invalidatons occur only for compiled code; method definitions themselves cannot be invalidated. - As a consequence, it's possible to have latent invalidation risk; this risk can become exposed - if you use some intermediate functionality before loading your package, or if your dependencies someday add `precompile` - directives. So even if you've checked for invalidations previously, sometimes it's worth taking a fresh look. - -## Recording invalidations - -```@meta -DocTestFilters = r"(in|@) [a-zA-Z0-9]* (REPL\[\d+\]|none):\d+" -DocTestSetup = quote - using SnoopCompile -end -``` - -To record the invalidations caused by defining new methods, use [`@snoop_invalidations`](@ref). -`@snoop_invalidations` is exported by `SnoopCompile`, but the recommended approach is to record invalidations using the minimalistic `SnoopCompileCore` package, and then load `SnoopCompile` to do the analysis. -_**Remember**_ to run julia with the `--startup-file="no"` flag set, if you load packages such as [`Revise`](https://github.com/timholy/Revise.jl) in your startup file. -Otherwise invalidations relating to those packages will also show up. - -```julia -using SnoopCompileCore -invalidations = @snoop_invalidations begin - # package loads and/or method definitions that might invalidate other code -end -using SnoopCompile # now that we've collected the data, load the complete package to analyze the results -``` - -!!! note - `SnoopCompileCore` was split out from `SnoopCompile` to reduce the risk of invalidations from loading `SnoopCompile` itself. - Once a `MethodInstance` gets invalidated, it doesn't show up in future `@snoop_invalidations` results, so anything that - gets invalidated in order to provide `@snoop_invalidations` would be omitted from the results. - `SnoopCompileCore` is a very small package with no dependencies and which avoids extending any of Julia's own functions, - so it cannot invalidate any other code. - -## Analyzing invalidations - -### A first example - -We'll walk through this process with the following example: - -```jldoctest invalidations -julia> f(::Real) = 1; - -julia> callf(container) = f(container[1]); - -julia> call2f(container) = callf(container); -``` - -Because code doesn't get compiled until it gets run, and invalidations only affect compiled code, let's run this with three different container types: - -```jldoctest invalidations -julia> c64 = [1.0]; c32 = [1.0f0]; cabs = AbstractFloat[1.0]; # Vector{Float64}, Vector{Float32}, and Vector{AbstractFloat}, respectively - -julia> call2f(c64) -1 - -julia> call2f(c32) -1 - -julia> call2f(cabs) -1 -``` - -!!! warning - If you're following along, be sure you actually execute these methods, or you won't obtain the results below. - -Now we'll define a new `f` method, one specialized for `Float64`. -So we can see the consequences for the compiled code, we'll make this definition while snooping on the compiler with `@snoop_invalidations`: - -```jldoctest invalidations -julia> using SnoopCompileCore - -julia> invalidations = @snoop_invalidations f(::Float64) = 2; -``` - -As should be apparent, running `call2f` on `c64` should produce a different result than formerly, so Julia certainly -needs to invalidate that code. Let's see what that looks like. The simplest thing we can do is list or count invalidations: - -```jldoctest invalidations -julia> using SnoopCompile - -julia> length(uinvalidated(invalidations)) # collect the unique MethodInstances & count them -6 -``` - -The length of this set is your simplest insight into the extent of invalidations triggered by this method definition. - -If you want to fix invalidations, it's crucial to know *why* certain `MethodInstance`s were invalidated. -For that, it's best to use a tree structure, in which children are invalidated because their parents get invalidated: - -```jldoctest invalidations; filter=[r"(in Main at|@ Main) (REPL\[\d+\]|none)"] -julia> trees = invalidation_trees(invalidations) -1-element Vector{SnoopCompile.MethodInvalidations}: - inserting f(::Float64) @ Main none:1 invalidated: - backedges: 1: superseding f(::Real) @ Main none:1 with MethodInstance for f(::Float64) (2 children) - 2: superseding f(::Real) @ Main none:1 with MethodInstance for f(::AbstractFloat) (2 children) - 1 mt_cache -``` - -The output, `trees`, is a vector of `MethodInvalidations`, a data type defined in `SnoopCompile`; each of these is the set of invalidations triggered by a particular method definition. -In this case, we only defined one method, so we can get at most one `MethodInvalidation`. -`@snoop_invalidations using SomePkg` might result in a list of such objects, each connected to a particular method defined in a particular package (either `SomePkg` itself or one of its dependencies). - -In this case, "`inserting f(::Float64)`" indicates that we added a method with signature `f(::Float64)`, and that this method triggered invalidations. -(Invalidations can also be triggered by method deletion, although this should not happen in typical usage.) -Next, notice the `backedges` line, and the fact that there are two items listed for it. -This indicates that there were two proximal triggers for the invalidation, both of which superseded the method `f(::Real)`. -One of these had been compiled specifically for `Float64`, due to our `call2f(c64)`. -The other had been compiled specifically for `AbstractFloat`, due to our `call2f(cabs)`. - -You can look at these invalidation trees in greater detail: - -```jldoctest invalidations -julia> method_invalidations = trees[1]; # invalidations stemming from a single method - -julia> root = method_invalidations.backedges[1] # get the first triggered invalidation -MethodInstance for f(::Float64) at depth 0 with 2 children - -julia> show(root) -MethodInstance for f(::Float64) (2 children) - MethodInstance for callf(::Vector{Float64}) (1 children) - ⋮ - -julia> show(root; minchildren=0) -MethodInstance for f(::Float64) (2 children) - MethodInstance for callf(::Vector{Float64}) (1 children) - MethodInstance for call2f(::Vector{Float64}) (0 children) -``` - -The indentation here reveals that `call2f` called `callf` which called `f`, -and shows the entire "chain" of invalidations triggered by this method definition. -Examining `root2 = method_invalidations.backedges[2]` yields similar results, but for `Vector{AbstractFloat}`. - -### `mt_backedges` invalidations - -`MethodInvalidations` can have a second field, `mt_backedges`. -These are invalidations triggered via the `MethodTable` for a particular function. -When extracting `mt_backedges`, in addition to a root `MethodInstance` these also indicate a particular signature that triggered the invalidation. -We can illustrate this by returning to the `call2f` example above: - -```jldoctest invalidations; filter=[r"(in Main at|@ Main) (REPL\[\d+\]|none)"] -julia> call2f(["hello"]) -ERROR: MethodError: no method matching f(::String) -[...] - -julia> invalidations = @snoop_invalidations f(::AbstractString) = 2; - -julia> trees = invalidation_trees(invalidations) -1-element Vector{SnoopCompile.MethodInvalidations}: - inserting f(::AbstractString) @ Main none:1 invalidated: - mt_backedges: 1: signature Tuple{typeof(f), String} triggered MethodInstance for callf(::Vector{String}) (1 children) - - -julia> sig, root = trees[1].mt_backedges[end]; - -julia> sig -Tuple{typeof(f), String} - -julia> root -MethodInstance for callf(::Vector{String}) at depth 0 with 1 children -``` - -You can see that the invalidating signature, `f(::String)`, is more specific than the signature of the defined method, but that it is what was minimally needed by `callf(::Vector{String})`. - -`mt_backedges` invalidations often reflect "unhandled" conditions in methods that have already been compiled. - -### A more complex example - -The structure of these trees can be considerably more complicated. For example, if `callf` -also got called by some other method, and that method had also been executed (forcing it to be compiled), -then `callf` would have multiple children. -This is often seen with more complex, real-world tests. -As a medium-complexity example, try the following: - -!!! info - Any demonstration involving real-world packages might be altered from what is shown here by new releases of the relevant packages. - -```julia -julia> using Revise - -julia> using SnoopCompileCore - -julia> invalidations = @snoop_invalidations using FillArrays; - -julia> using SnoopCompile - -julia> trees = invalidation_trees(invalidations) -3-element Vector{SnoopCompile.MethodInvalidations}: - inserting all(f::Function, x::FillArrays.AbstractFill) in FillArrays at /pathto/.julia/packages/FillArrays/NjFh2/src/FillArrays.jl:556 invalidated: - backedges: 1: superseding all(f::Function, a::AbstractArray; dims) in Base at reducedim.jl:880 with MethodInstance for all(::Base.var"#388#389"{_A} where _A, ::AbstractArray) (3 children) - 2: superseding all(f, itr) in Base at reduce.jl:918 with MethodInstance for all(::Base.var"#388#389"{_A} where _A, ::Any) (3 children) - - inserting any(f::Function, x::FillArrays.AbstractFill) in FillArrays at /pathto/.julia/packages/FillArrays/NjFh2/src/FillArrays.jl:555 invalidated: - backedges: 1: superseding any(f::Function, a::AbstractArray; dims) in Base at reducedim.jl:877 with MethodInstance for any(::typeof(ismissing), ::AbstractArray) (1 children) - 2: superseding any(f, itr) in Base at reduce.jl:871 with MethodInstance for any(::typeof(ismissing), ::Any) (1 children) - 3: superseding any(f, itr) in Base at reduce.jl:871 with MethodInstance for any(::LoweredCodeUtils.var"#11#12"{_A} where _A, ::Any) (2 children) - 4: superseding any(f::Function, a::AbstractArray; dims) in Base at reducedim.jl:877 with MethodInstance for any(::LoweredCodeUtils.var"#11#12"{_A} where _A, ::AbstractArray) (4 children) - - inserting broadcasted(::Base.Broadcast.DefaultArrayStyle{N}, op, r::FillArrays.AbstractFill{T,N,Axes} where Axes) where {T, N} in FillArrays at /pathto/.julia/packages/FillArrays/NjFh2/src/fillbroadcast.jl:8 invalidated: - backedges: 1: superseding broadcasted(::S, f, args...) where S<:Base.Broadcast.BroadcastStyle in Base.Broadcast at broadcast.jl:1265 with MethodInstance for broadcasted(::Base.Broadcast.BroadcastStyle, ::typeof(JuliaInterpreter._Typeof), ::Any) (1 children) - 2: superseding broadcasted(::S, f, args...) where S<:Base.Broadcast.BroadcastStyle in Base.Broadcast at broadcast.jl:1265 with MethodInstance for broadcasted(::Base.Broadcast.BroadcastStyle, ::typeof(string), ::AbstractArray) (177 children) -``` - -Your specific results may differ from this, depending on which version of Julia and of packages you are using. -In this case, you can see that three methods (one for `all`, one for `any`, and one for `broadcasted`) triggered invalidations. -Perusing this list, you can see that methods in `Base`, `LoweredCodeUtils`, and `JuliaInterpreter` (the latter two were loaded by `Revise`) got invalidated by methods defined in `FillArrays`. - -The most consequential ones (the ones with the most children) are listed last, and should be where you direct your attention first. -That last entry looks particularly problematic, so let's extract it: - -```julia -julia> methinvs = trees[end]; - -julia> root = methinvs.backedges[end] -MethodInstance for broadcasted(::Base.Broadcast.BroadcastStyle, ::typeof(string), ::AbstractArray) at depth 0 with 177 children - -julia> show(root; maxdepth=10) -MethodInstance for broadcasted(::Base.Broadcast.BroadcastStyle, ::typeof(string), ::AbstractArray) (177 children) - MethodInstance for broadcasted(::typeof(string), ::AbstractArray) (176 children) - MethodInstance for #unpack#104(::Bool, ::typeof(Pkg.PlatformEngines.unpack), ::String, ::String) (175 children) - MethodInstance for (::Pkg.PlatformEngines.var"#unpack##kw")(::NamedTuple{(:verbose,),Tuple{Bool}}, ::typeof(Pkg.PlatformEngines.unpack), ::String, ::String) (174 children) - MethodInstance for #download_verify_unpack#109(::Nothing, ::Bool, ::Bool, ::Bool, ::Bool, ::typeof(Pkg.PlatformEngines.download_verify_unpack), ::String, ::Nothing, ::String) (165 children) - MethodInstance for (::Pkg.PlatformEngines.var"#download_verify_unpack##kw")(::NamedTuple{(:ignore_existence, :verbose),Tuple{Bool,Bool}}, ::typeof(Pkg.PlatformEngines.download_verify_unpack), ::String, ::Nothing, ::String) (33 children) - MethodInstance for (::Pkg.Artifacts.var"#39#40"{Bool,String,Nothing})(::String) (32 children) - MethodInstance for create_artifact(::Pkg.Artifacts.var"#39#40"{Bool,String,Nothing}) (31 children) - MethodInstance for #download_artifact#38(::Bool, ::Bool, ::typeof(Pkg.Artifacts.download_artifact), ::Base.SHA1, ::String, ::Nothing) (30 children) - MethodInstance for (::Pkg.Artifacts.var"#download_artifact##kw")(::NamedTuple{(:verbose, :quiet_download),Tuple{Bool,Bool}}, ::typeof(Pkg.Artifacts.download_artifact), ::Base.SHA1, ::String, ::Nothing) (23 children) - MethodInstance for (::Pkg.Artifacts.var"#download_artifact##kw")(::NamedTuple{(:verbose, :quiet_download),Tuple{Bool,Bool}}, ::typeof(Pkg.Artifacts.download_artifact), ::Base.SHA1, ::String) (22 children) - ⋮ - ⋮ - MethodInstance for (::Pkg.PlatformEngines.var"#download_verify_unpack##kw")(::NamedTuple{(:ignore_existence,),Tuple{Bool}}, ::typeof(Pkg.PlatformEngines.download_verify_unpack), ::String, ::Nothing, ::String) (130 children) - MethodInstance for (::Pkg.Types.var"#94#97"{Pkg.Types.Context,String,Pkg.Types.RegistrySpec})(::String) (116 children) - MethodInstance for #mktempdir#21(::String, ::typeof(mktempdir), ::Pkg.Types.var"#94#97"{Pkg.Types.Context,String,Pkg.Types.RegistrySpec}, ::String) (115 children) - MethodInstance for mktempdir(::Pkg.Types.var"#94#97"{Pkg.Types.Context,String,Pkg.Types.RegistrySpec}, ::String) (114 children) - MethodInstance for mktempdir(::Pkg.Types.var"#94#97"{Pkg.Types.Context,String,Pkg.Types.RegistrySpec}) (113 children) - MethodInstance for clone_or_cp_registries(::Pkg.Types.Context, ::Vector{Pkg.Types.RegistrySpec}, ::String) (112 children) - ⋮ - ⋮ - ⋮ -``` - -Here you can see a much more complex branching structure. -From this, you can see that methods in `Pkg` are the most significantly affected; -you could expect that loading `FillArrays` might slow down your next `Pkg` operation (perhaps depending on which operation you choose) executed in this same session. - -Again, if you're following along, it's possible that you'll see something quite different, if subsequent development has protected `Pkg` against this form of invalidation. - -## Filtering invalidations - -!!! info - The experimental tool [`SnoopCompile.precompile_blockers`](@ref) may be - able to help you identify just the invalidations you need to fix - for your use-case. - -Some method definitions trigger widespread invalidation. -If you don't have time to fix all of them, you might want to focus on a specific set of invalidations. -For instance, you might be the author of `PkgA` and you've noted that loading `PkgB` invalidates a lot of `PkgA`'s code. -In that case, you might want to find just those invalidations triggered in your package. -You can find them with [`filtermod`](@ref): - -```julia -trees = invalidation_trees(@snoop_invalidations using PkgB) -ftrees = filtermod(PkgA, trees) -``` - -By default, `filtermod` only selects trees where the root method was defined in the specified module. -`filtermod(PkgA, trees; recursive=true)` will return all trees that lead to any method defined -in `PkgA`. - -A more selective yet exhaustive tool is [`findcaller`](@ref), which allows you to find the path through the trees to a particular method: - -```julia -m = @which f(data) # look for the "path" that invalidates this method -f(data) # run once to force compilation -using SnoopCompile -trees = invalidation_trees(@snoop_invalidations using SomePkg) -invs = findcaller(m, trees) # select the branch that invalidated a compiled instance of `m` -``` - -When you don't know which method to choose, but know an operation that got slowed down by loading `SomePkg`, you can use `@snoopi` to find methods that needed to be recompiled. See [`findcaller`](@ref) for further details. - - -## Fixing invalidations - -In addition to the text below, there is a -[video](https://www.youtube.com/watch?v=7VbXbI6OmYo) illustrating many -of the same package features. The video also walks through a real-world -example fixing invalidations that stemmed from inference problems in -some of `Pkg`'s code. - -### `ascend` - -SnoopCompile, partnering with the remarkable [Cthulhu.jl](https://github.com/JuliaDebug/Cthulhu.jl), -provides a tool called `ascend` to simplify diagnosing and fixing invalidations. -To demonstrate this tool, let's use it on our test methods defined above. -For best results, you'll want to copy those method definitions into a file: - -```julia -f(::Real) = 1 -callf(container) = f(container[1]) -call2f(container) = callf(container) - -c64 = [1.0]; c32 = [1.0f0]; cabs = AbstractFloat[1.0]; -call2f(c64) -call2f(c32) -call2f(cabs) - -using SnoopCompileCore -invalidations = @snoop_invalidations f(::Float64) = 2 -using SnoopCompile -trees = invalidation_trees(invalidations) -method_invalidations = trees[1] -``` - -and `include` it into a fresh session. (The full functionality of `ascend` doesn't work for methods defined at the REPL, but does if the methods are defined in a file.) -In this demo, I called that file `/tmp/snoop_invalidations.jl`. - - -We start with - -```julia -julia> root = method_invalidations.backedges[end] -MethodInstance for f(::AbstractFloat) at depth 0 with 2 children -``` - -(It's common to start from the last element of `backedges` or `mt_backedges` since these have the largest number of children and are therefore most consequential.) -Then: - -```julia -julia> ascend(root) -Choose a call for analysis (q to quit): - > f(::AbstractFloat) - callf(::Vector{AbstractFloat}) - call2f(::Vector{AbstractFloat}) -``` - -This is an interactive menu: press the down arrow to go down, the up arrow to go up, and `Enter` to select an item for more detailed analysis. -In large trees, you may also want to "fold" nodes of the tree (collapsing it so that the children are no longer displayed), particularly if you are working your way through a long series of invalidations and want to hide ones you've already dealt with. You toggle folding using the space bar, and folded nodes are printed with a `+` in front of them. - -For example, if we press the down arrow once, we get - -```julia -julia> ascend(root) -Choose a call for analysis (q to quit): - f(::AbstractFloat) - > callf(::Vector{AbstractFloat}) - call2f(::Vector{AbstractFloat}) -``` - -Now hit `Enter` to select it: - -```julia -Choose caller of MethodInstance for f(::AbstractFloat) or proceed to typed code: - > "/tmp/snoop_invalidations.jl", callf: lines [2] - Browse typed code -``` - -This is showing you another menu, with only two options (a third is to go back by hitting `q`). -The first entry shows you the option to open the "offending" source file in `callf` at the position of the call to the parent node of `callf`, which in this case is `f`. -(Sometimes there will be more than one call to the parent within the method, in which case instead of showing `[1]` it might show `[1, 17, 39]` indicating each separate location.) -Selecting this option, when available, is typically the best way to start because you can sometimes resolve the problem just by inspection of the source. - -If you hit the down arrow - -```julia -Choose caller of MethodInstance for f(::AbstractFloat) or proceed to typed code: - "/tmp/snoop_invalidations.jl", callf: lines [2] - > Browse typed code -``` - -and then hit `Enter`, this is what you see: - -```julia -│ ─ %-1 = invoke callf(::Vector{AbstractFloat})::Int64 -Variables - #self#::Core.Const(callf, false) - container::Vector{AbstractFloat} - -Body::Int64 - @ /tmp/snoop_invalidations.jl:2 within `callf' -1 ─ %1 = Base.getindex(container, 1)::AbstractFloat -│ %2 = Main.f(%1)::Int64 -└── return %2 - -Select a call to descend into or ↩ to ascend. [q]uit. [b]ookmark. -Toggles: [o]ptimize, [w]arn, [d]ebuginfo, [s]yntax highlight for Source/LLVM/Native. -Show: [S]ource code, [A]ST, [L]LVM IR, [N]ative code -Advanced: dump [P]arams cache. - - • %1 = invoke getindex(::Vector{AbstractFloat},::Int64)::AbstractFloat - %2 = call #f(::AbstractFloat)::Int64 - ↩ -``` - -This is output from Cthulhu, and you should see its documentation for more information. -(See also [this video](https://www.youtube.com/watch?v=qf9oA09wxXY).) -While it takes a bit of time to master Cthulhu, it is an exceptionally powerful tool for diagnosing and fixing inference issues. - -### "Dead ends": finding runtime callers with MethodAnalysis - -When a call is made by runtime dispatch and the world of available methods to handle the call does not narrow the types -beyond what is known to the caller, the call-chain terminates. -Here is a real-world example (one that may already be "fixed" by the time you read this) from analyzing invalidations triggered by specializing `Base.unsafe_convert(::Type{Ptr{T}}, ::Base.RefValue{S})` for specific types `S` and `T`: - -``` -julia> ascend(root) -Choose a call for analysis (q to quit): - > unsafe_convert(::Type{Ptr{Nothing}}, ::Base.RefValue{_A} where _A) - _show_default(::IOBuffer, ::Any) - show_default(::IOBuffer, ::Function) - show_function(::IOBuffer, ::Function, ::Bool) - print(::IOBuffer, ::Function) - show_default(::IOBuffer, ::ProcessFailedException) - show(::IOBuffer, ::ProcessFailedException) - print(::IOBuffer, ::ProcessFailedException) - show_default(::IOBuffer, ::Sockets.IPAddr) - show(::IOBuffer, ::Sockets.IPAddr) -``` - -Unfortunately for our investigations, none of these "top level" callers have defined backedges. (Overall, it's very fortunate that they don't, in that runtime dispatch without backedges avoids any need to invalidate the caller; the alternative would be extremely long chains of completely unnecessary invalidation, which would have many undesirable consequences.) - -If you want to fix such "short chains" of invalidation, one strategy is to identify callers by brute force search enabled by the [MethodAnalysis.jl](https://github.com/timholy/MethodAnalysis.jl) package. -For example, one can discover the caller of `show(::IOBuffer, ::Sockets.IPAddr)` with - -```julia -using MethodAnalysis # best from a fresh Julia session -mis = methodinstances(); # collect all *existing* MethodInstances (any future compilation will be ignored) -# Create a predicate that finds these argument types -using Sockets -argmatch(typs) = length(typs) >= 2 && typs[1] === IOBuffer && typs[2] === Sockets.IPAddr -# Find any callers -callers = findcallers(show, argmatch, mis) -``` - -which yields a single hit in `print(::IOBuffer, ::IPAddr)`. -This too lacks any backedges, so a second application `findcallers(print, argmatch, mis)` links to `print_to_string(::IPAddr)`. -This MethodInstance has a backedge to `string(::IPAddr)`, which has backedges to the method `Distributed.connect_to_worker(host::AbstractString, port::Integer)`. -A bit of digging shows that this calls `Sockets.getaddrinfo` to look up an IP address, and this is inferred to return an `IPAddr` but the concrete type is unknown. -A potential fix for this situation is described below. - -This does not always work; for example, trying something similar for `ProcessExitedException` fails, likely because the call was made with even less type information. -We might be able to find it with a more general predicate, for example - -``` -argmatch(typs) = length(typs) >= 2 && typs[1] === IOBuffer && ProcessExitedException <: typs[2] -``` - -but this returns a lot of candidates and it is difficult to guess which of these might be the culprit(s). -Finally, `findcallers` only detects method calls that are "hard-wired" into type-inferred code; if the call we're seeking was made from toplevel, or if the function itself was a runtime variable, there is no hope that `findcallers` will detect it. - -### Tips for fixing invalidations - -Invalidations occur in situations like our `call2f(c64)` example, where we changed our mind about what value `f` should return for `Float64`. -Julia could not have returned the newly-correct answer without recompiling the call chain. - -Aside from cases like these, most invalidations occur whenever new types are introduced, -and some methods were previously compiled for abstract types. -In some cases, this is inevitable, and the resulting invalidations simply need to be accepted as a consequence of a dynamic, updatable language. -(As recommended above, you can often minimize invalidations by loading all your code at the beginning of your session, before triggering the compilation of more methods.) -However, in many circumstances an invalidation indicates an opportunity to improve code. -In our first example, note that the call `call2f(c32)` did not get invalidated: this is because the compiler -knew all the specific types, and new methods did not affect any of those types. -The main tips for writing invalidation-resistant code are: - -- use [concrete types](https://docs.julialang.org/en/v1/manual/performance-tips/#man-performance-abstract-container-1) wherever possible -- write inferrable code -- don't engage in [type-piracy](https://docs.julialang.org/en/v1/manual/style-guide/#Avoid-type-piracy-1) (our `c64` example is essentially like type-piracy, where we redefined behavior for a pre-existing type) - -Since these tips also improve performance and allow programs to behave more predictably, -these guidelines are not intrusive. -Indeed, searching for and eliminating invalidations can help you improve the quality of your code. - diff --git a/docs/src/snoopi.md b/docs/src/snoopi.md deleted file mode 100644 index b59b9f28..00000000 --- a/docs/src/snoopi.md +++ /dev/null @@ -1,375 +0,0 @@ -# [Snooping on inference: `@snoopi`](@id macro-snoopi) - -If you can't use `@snoop_inference` due to Julia version constraints, the most useful choice is `@snoopi`, which is available on Julia 1.2 or higher. - -Julia can cache inference results, so you can use `@snoopi` to generate `precompile` -directives for your package. Executing these directives when the package is compiled -may reduce compilation (inference) time when the package is used. - -Here's a quick demo: - -```julia -using SnoopCompile - -a = rand(Float16, 5) - -julia> inf_timing = @snoopi sum(a) -1-element Array{Tuple{Float64,Core.MethodInstance},1}: - (0.011293888092041016, MethodInstance for sum(::Array{Float16,1})) -``` - -We defined the argument `a`, and then called `sum(a)` while "snooping" on inference. -(The `i` in `@snoopi` means "inference.") -The return is a list of "top level" methods that got compiled, together with the amount of -time spent on inference. -In this case it was just a single method, which required approximately 11ms of inference time. -(Inferring `sum` required inferring all the methods that it calls, but these are -subsumed into the top level inference of `sum` itself.) -Note that the method that got called, - -```julia -julia> @which sum(a) -sum(a::AbstractArray) in Base at reducedim.jl:652 -``` - -is much more general (i.e., defined for `AbstractArray`) than the `MethodInstance` -(defined for `Array{Float16,1}`). This is because precompilation requires the -types of the arguments to specialize the code appropriately. - -The information obtained from `@snoopi` can be used in several ways, primarily to reduce latency during usage of your package: - -- to help you understand which calls take the most inference time -- to help you write `precompile` directives that run inference on specific calls during package precompilation, so that you don't pay this cost repeatedly each time you use the package -- to help you identify inference problems that prevent successful or comprehensive precompilation - -If you're starting a project to try to reduce latency in your package, broadly speaking there are two paths you can take: - -1. you can use SnoopCompile, perhaps together with [CompileBot](https://github.com/aminya/CompileBot.jl), - to automatically generate lists of precompile directives that may reduce latency; -2. you can use SnoopCompile primarily as an analysis tool, and then intervene manually to reduce latency. - -Beginners often leap at option 1, but experience shows there are good reasons to consider option 2. -To avoid introducing too much complexity early on, we'll defer this discussion to the end of this page, but readers who are serious about reducing latency should be sure to read [Understanding precompilation and its limitations](@ref). - -!!! note - Because invalidations can prevent effective precompilation, developers analyzing their - packages with `@snoopi` are encouraged to use Julia versions (1.6 and higher) that have a lower risk - of invalidations in Base and the standard library. - -## [Precompile scripts](@id pcscripts) - -You can use `@snoopi` to come up with a list of precompile-worthy functions. -A recommended approach is to write a script that "exercises" the functionality -you'd like to precompile. -One option is to use your package's `"runtests.jl"` file, or you can write a custom -script for this purpose. -Here's an example for the -[FixedPointNumbers package](https://github.com/JuliaMath/FixedPointNumbers.jl): - -``` -using FixedPointNumbers - -x = N0f8(0.2) -y = x + x -y = x - x -y = x*x -y = x/x -y = Float32(x) -y = Float64(x) -y = 0.3*x -y = x*0.3 -y = 2*x -y = x*2 -y = x/15 -y = x/8.0 -``` - -Save this as a file `"snoopfpn.jl"` and navigate at the Julia REPL to that directory, -and then do - -```julia -julia> using SnoopCompile - -julia> inf_timing = @snoopi tmin=0.01 include("snoopfpn.jl") -2-element Array{Tuple{Float64,Core.MethodInstance},1}: - (0.03108978271484375, MethodInstance for *(::Normed{UInt8,8}, ::Normed{UInt8,8})) - (0.04189491271972656, MethodInstance for Normed{UInt8,8}(::Float64)) -``` - -Here, note the `tmin=0.01`, which causes any methods that take less than 10ms of inference -time to be discarded. - -!!! note - If you're testing this, you might get different results depending on - the speed of your machine. Moreover, if FixedPointNumbers has - already precompiled these method and type combinations---perhaps - by incorporating a precompile file produced by SnoopCompile---then - those methods will be absent. For packages whose precompile - directives are executed only when `ccall(:jl_generating_output, - Cint, ()) == 1`, you can start Julia with `--compiled-modules=no` - to disable them. Alternatively, you can `dev` the package and - comment them out. - -You can inspect these results and write your own precompile file, or use the automated -tools provided by SnoopCompile. - -## [Producing precompile directives automatically](@id auto) - -You can take the output of `@snoopi` and "parcel" it into packages: - -```julia -julia> pc = SnoopCompile.parcel(inf_timing) -Dict{Symbol,Array{String,1}} with 1 entry: - :FixedPointNumbers => ["precompile(Tuple{typeof(*),Normed{UInt8,8},Normed{UInt8,8}})", "precompile(Tuple{Type{Normed{UInt8,8}},Float64})"] -``` - -This splits the calls up into a dictionary, `pc`, indexed by the package which "owns" -each call. -(In this case there is only one, `FixedPointNumbers`, but in more complex cases there may -be several.) You can then write the results to files: - -```julia -julia> SnoopCompile.write("/tmp/precompile", pc) -``` - -If you look in the `/tmp/precompile` directory, you'll see one or more -files, named by their parent package, that may be suitable for `include`ing into the package. -In this case: - -``` -/tmp/precompile$ cat precompile_FixedPointNumbers.jl -function _precompile_() - ccall(:jl_generating_output, Cint, ()) == 1 || return nothing - precompile(Tuple{typeof(*),Normed{UInt8,8},Normed{UInt8,8}}) - precompile(Tuple{Type{Normed{UInt8,8}},Float64}) -end -``` - -If you copy this file to a `precompile.jl` file in the `src` directory, -you can incorporate it into the package like this: - -```julia -module FixedPointNumbers - -# All the usual commands that define the module go here - -# ... followed by: - -include("precompile.jl") -_precompile_() - -end # module FixedPointNumbers -``` - -The listed method/type combinations should have their inference results cached. -Load the package once to precompile it, and then in a fresh Julia session try this: - -```julia -julia> using SnoopCompile - -julia> inf_timing = @snoopi tmin=0.01 include("snoopfpn.jl") -0-element Array{Tuple{Float64,Core.MethodInstance},1} -``` - -The fact that no methods were returned is a sign of success: Julia didn't need to call -inference on those methods, because it used the inference results from the cache file. - -!!! note - Sometimes, `@snoopi` will show method & type combinations that you precompiled. - This is a sign that despite your attempts, Julia declined to cache the inference - results for those methods. - You can either delete those directives from the precompile file, or hope that - they will become useful in a future version of Julia. - Note that having many "useless" precompile directives can slow down precompilation. - -!!! note - As you develop your package, it's possible you'll modify or delete some of the - methods that appear in your `"precompile.jl"` file. - This will *not* result in an error; by default `precompile` fails silently. - If you want to be certain that your precompile directives don't go stale, - you can check that `precompile` returns `true` and otherwise issue a warning. - By default, [`SnoopCompile.write`](@ref) generates - a macro, `@warnpcfail`, and you can use it by - changing `precompile(args...)` to `@warnpcfail precompile(args...)`. - - -If you find that some precompile directives are -ineffective (they appear in a new `@snoopi` despite being precompiled) and their -inference time is substantial, sometimes a bit of manual investigation of the callees -can lead to insights. For example, you might be able to introduce a precompile in a -dependent package that can mitigate the total time. -(`@snoop_inference` makes the analysis and resolution of these issues more straightforward.) - -!!! tip - For packages that support just Julia 1.6 and higher, you may be able to slim down the precompile file by - adding `has_bodyfunction=true` to the arguments for `parcel`. - This setting applies for all packges in `inf_timing`, so you may need to call `parcel` twice (with both `false` and `true`) and select the appropriate precompile file for each package. - -## Producing precompile directives manually - -While this "automated" approach is often useful, sometimes it makes more sense to -inspect the results and write your own precompile directives. -For example, for FixedPointNumbers a more elegant and comprehensive precompile file -might be - -```julia -function _precompile_() - ccall(:jl_generating_output, Cint, ()) == 1 || return nothing - for T in (N0f8, N0f16) # Normed types we want to support - for f in (+, -, *, /) # operations we want to support - precompile(Tuple{typeof(f),T,T}) - for S in (Float32, Float64, Int) # other number types we want to support - precompile(Tuple{typeof(f),T,S}) - precompile(Tuple{typeof(f),S,T}) - end - end - for S in (Float32, Float64) - precompile(Tuple{Type{T},S}) - precompile(Tuple{Type{S},T}) - end - end -end -``` - -This covers `+`, `-`, `*`, `/`, and conversion for various combinations of types. -The results from `@snoopi` can suggest method/type combinations that might be useful to -precompile, but often you can generalize its suggestions in useful ways. - -## Analyzing omitted methods - -There are some method signatures that cannot be precompiled. -For example, suppose you have two packages, `A` and `B`, that are independent of one another. -Then `A.f([B.Object(1)])` cannot be precompiled, because `A` does not know about `B.Object`, -and `B` does not know about `A.f`, unless both `A` and `B` get included into a third package. - -Such problematic method signatures are removed automatically. -If you want to be informed about these removals, you can use Julia's logging framework -while running `parcel`: - -``` -julia> using Base.CoreLogging - -julia> logger = SimpleLogger(IOBuffer(), CoreLogging.Debug); - -julia> pc = with_logger(logger) do - SnoopCompile.parcel(inf_timing) - end - -julia> msgs = String(take!(logger.stream)) -``` - -The omitted method signatures will be logged to the string `msgs`. - - -## Understanding precompilation and its limitations - -Suppose your package includes the following method: - -```julia -""" - idx = index_midsum(a) - -Return the index of the first item more than "halfway to the cumulative sum," -meaning the smallest integer so that `sum(a[begin:idx]) >= sum(a)/2`. -""" -function index_midsum(a::AbstractVector) - ca = cumsum(vcat(0, a)) # cumulative sum of items in a, starting from 0 - s = ca[end] # the sum of all elements - return findfirst(x->x >= s/2, ca) - 1 # compensate for inserting 0 -end -``` -Now, suppose that you'd like to reduce latency in using this method, and you know that an important use case is when `a` is a `Vector{Int}`. -Therefore, you might precompile it: - -```julia -julia> precompile(index_midsum, (Vector{Int},)) -true -``` -This will cause Julia to infer this method for the given argument types. If you add such statements to your package, it potentially saves your users from having to wait for it to be inferred each time they use your package. - -!!! note - The `true` indicates that Julia was successfully able to find a method supporting this signature and precompile it. - See the note about `@warnpcfail` above for ways to exploit this in your package. - - -But if you execute these lines in the REPL, and then check how well it worked, you might see something like the following: -```julia -julia> using SnoopCompile - -julia> tinf = @snoopi index_midsum([1,2,3,4,100]) -3-element Vector{Tuple{Float64, Core.MethodInstance}}: - (0.00048613548278808594, MethodInstance for cat_similar(::Int64, ::Type, ::Tuple{Int64})) - (0.010090827941894531, MethodInstance for (::Base.var"#cat_t##kw")(::NamedTuple{(:dims,), Tuple{Val{1}}}, ::typeof(Base.cat_t), ::Type{Int64}, ::Int64, ::Vararg{Any, N} where N)) - (0.016659975051879883, MethodInstance for __cat(::Vector{Int64}, ::Tuple{Int64}, ::Tuple{Bool}, ::Int64, ::Vararg{Any, N} where N)) -``` -Even though we'd already said `precompile(index_midsum, (Vector{Int},))` in this session, somehow we needed *more* inference of various concatenation methods. -Why does this happen? -A detailed investigation (e.g., using [Cthulhu](https://github.com/JuliaDebug/Cthulhu.jl) or `@code_warntype`) would reveal that `vcat(0, a)` is not inferrable "all the way down," and hence the `precompile` directive couldn't predict everything that was going to be needed. - -No problem, you say: let's just precompile those methods too. The most expensive is the last one. You might not know where `__cat` is defined, but you can find out with -```julia -julia> mi = tinf[end][2] # get the MethodInstance -MethodInstance for __cat(::Vector{Int64}, ::Tuple{Int64}, ::Tuple{Bool}, ::Int64, ::Vararg{Any, N} where N) - -julia> mi.def # get the Method -__cat(A, shape::Tuple{Vararg{Int64, M}}, catdims, X...) where M in Base at abstractarray.jl:1599 - -julia> mi.def.module # which module was this method defined in? -Base -``` - -!!! note - When using `@snoopi` you might sometimes see entries like - `MethodInstance for (::SomeModule.var"#10#12"{SomeType})(::AnotherModule.AnotherType)`. - These typically correspond to closures/anonymous functions defined with `->` or `do` blocks, - but it may not be immediately obvious where these come from. - `mi.def` will show you the file/line number that these are defined on. - You can either convert them into named functions to make them easier to precompile, - or you can fix inference problems that prevent automatic precompilation (as illustrated below). - -Armed with this knowledge, let's start a fresh session (so that nothing is precompiled yet), and in addition to defining `index_midsum` and precompiling it, we also execute - -```julia -julia> precompile(Base.__cat, (Vector{Int64}, Tuple{Int64}, Tuple{Bool}, Int, Vararg{Any, N} where N)) -true -``` - -Now if you try that `tinf = @snoopi index_midsum([1,2,3,4,100])` line, you'll see that the `__cat` call is omitted, suggesting success. - -However, if you copy both `precompile` directives into your package source files and then check it with `@snoopi` again, -you may be in for a rude surprise: the `__cat` precompile directive doesn't "work." -That turns out to be because your package doesn't "own" that `__cat` method--the module is `Base` rather than `YourPackage`--and because inference cannot determine that it's needed by by `index_midsum(::Vector{Int})`, Julia doesn't know which `*.ji` file to use to -store its precompiled form. - -How to fix this? -Fundamentally, the problem is that `vcat` call: if we can write `index_midsum` in a way so that inference succeeds, then all these problems go away. -(You can use `ascend(mi)`, with Cthulhu.jl, where `mi` was obtained above, to discover that `__cat` gets called from `vcat`. See [`Cthulhu.ascend`](@ref ascend-itrig) for more information.) -It turns out that `vcat` is inferrable if all the arguments have the same type, so just changing `vcat(0, a)` to `vcat([zero(eltype(a))], a)` fixes the problem. -(Alternatively, you could make a copy and then use `pushfirst!`.) -In a fresh Julia session: - -```julia -function index_midsum(a::AbstractVector) - ca = cumsum(vcat([zero(eltype(a))], a)) # cumulative sum of items in a, starting from 0 - s = ca[end] # the sum of all elements - return findfirst(x->x >= s/2, ca) - 1 # compensate for inserting 0 -end - -julia> precompile(index_midsum, (Vector{Int},)) -true - -julia> using SnoopCompile - -julia> tinf = @snoopi index_midsum([1,2,3,4,100]) -Tuple{Float64, Core.MethodInstance}[] -``` - -Tada! No additional inference was needed, ensuring that your users will not suffer any latency due to type-inference of this particular method/argument combination. -In addition to identifing a call deserving of precompilation, `@snoopi` helped us identify a weakness in its implementation. -Fixing that weakness reduced latency, made the code more resistant to invalidation, and may improve runtime performance. - -In other cases, manual inspection of the results from `@snoopi` may lead you in a different direction: you may discover that a huge number of specializations are being created for a method that doesn't need them. -Typical examples are methods that take types or functions as inputs: for example, there is no reason to recompile `methods(f)` for each separate `f`. -In such cases, by far your best option is to add `@nospecialize` annotations to one or more of the arguments of that method. Such changes can have dramatic impact on the latency of your package. - -The ability to make interventions like these--which can both reduce latency and improve runtime speed--is a major reason to consider `@snoopi` primarily as an analysis tool rather than just a utility to blindly generate lists of precompile directives. diff --git a/docs/src/tutorial.md b/docs/src/tutorial.md deleted file mode 100644 index 7f6a9577..00000000 --- a/docs/src/tutorial.md +++ /dev/null @@ -1,257 +0,0 @@ -# [Tutorial on the foundations](@id tutorial) - -Certain concepts and types will appear repeatedly, so it's worth -spending a little time to familiarize yourself at the outset. -You can find a more expansive version of this page in [this blog post](https://julialang.org/blog/2021/01/precompile_tutorial/). - -## Cut to the Chase: A copy-paste analysis of invalidations - -The following is a quick "grab and go" script for analyzing invalidations. -Insert package loads (`using` or `import` statements) and/or method definitions into the `@snoop_invalidations` block, -and put the workload you want to be fast in the `@snoop_inference` block. -The resulting plot shows the distribution of the invalidations sorted by the number of children affected. -Generally, invalidations with many children matter more than those -with few children, and thus this shows how many "bad actors" need to be investigated. `show(trees[end])` show the method which leads to the most -invalidations, with `show(trees[end-1])` being the second most, and so forth. -While the plot shows total invalidations (`trees`), only the ones in `staletrees` affect the workload in `@snoop_inference`. - -```julia -using SnoopCompileCore -invalidations = @snoop_invalidations using PkgA, PkgB; -tinf = @snoop_inference begin - some_workload() -end -using SnoopCompile -trees = invalidation_trees(invalidations) -staletrees = precompile_blockers(trees, tinf) - -@show length(uinvalidated(invalidations)) # show total invalidations - -show(trees[end]) # show the most invalidating method - -# Count number of children (number of invalidations per invalidated method) -n_invalidations = map(SnoopCompile.countchildren, trees) - -# (optional) plot the number of children per method invalidations -import Plots -Plots.plot( - 1:length(trees), - n_invalidations; - markershape=:circle, - xlabel="i-th method invalidation", - label="Number of children per method invalidations" -) - -# (optional) report invalidations summary -using PrettyTables # needed for `report_invalidations` to be defined -SnoopCompile.report_invalidations(; - invalidations, - process_filename = x -> last(split(x, ".julia/packages/")), - n_rows = 0, # no-limit (show all invalidations) - ) -``` - -## `MethodInstance`s, type-inference, and backedges - -Our first goal is to understand how code connects together. -We'll try some experiments using the following: - -```julia -double(x::Real) = 2x -calldouble(container) = double(container[1]) -calldouble2(container) = calldouble(container) -``` - -```@meta -DocTestSetup = quote - double(x::Real) = 2x - calldouble(container) = double(container[1]) - calldouble2(container) = calldouble(container) -end -``` - -Let's create a `container` and run this code: - -```jldoctest tutorial -julia> c64 = [1.0] -1-element Vector{Float64}: - 1.0 - -julia> calldouble2(c64) -2.0 -``` - -Using the [MethodAnalysis](https://github.com/timholy/MethodAnalysis.jl) package, we can get some insights into how Julia represents this code and its compilation dependencies: - -```jldoctest tutorial; setup=:(calldouble2(c64)) -julia> using MethodAnalysis - -julia> mi = methodinstance(double, (Float64,)) -MethodInstance for double(::Float64) - -julia> using AbstractTrees - -julia> print_tree(mi) -MethodInstance for double(::Float64) -└─ MethodInstance for calldouble(::Vector{Float64}) - └─ MethodInstance for calldouble2(::Vector{Float64}) -``` - -This indicates that the result for type-inference on `calldouble2(::Vector{Float64})` depended on the result for `calldouble(::Vector{Float64})`, which in turn depended on `double(::Float64)`. - -Now let's create a new container, one with abstract element type, so that Julia's type-inference cannot accurately predict the type of elements in the container: - -```jldoctest tutorial -julia> cabs = AbstractFloat[1.0f0] # put a Float32 in a Vector{AbstractFloat} -1-element Vector{AbstractFloat}: - 1.0f0 - -julia> calldouble2(cabs) -2.0f0 -``` - -Now let's look at the available instances: - -```jldoctest tutorial; setup=:(calldouble2(c64); calldouble2(cabs)) -julia> mis = methodinstances(double) -3-element Vector{Core.MethodInstance}: - MethodInstance for double(::Float64) - MethodInstance for double(::AbstractFloat) - MethodInstance for double(::Float32) - -julia> print_tree(mis[1]) -MethodInstance for double(::Float64) -└─ MethodInstance for calldouble(::Vector{Float64}) - └─ MethodInstance for calldouble2(::Vector{Float64}) - -julia> print_tree(mis[2]) -MethodInstance for double(::AbstractFloat) - -julia> print_tree(mis[3]) -MethodInstance for double(::Float32) -``` - -`double(::Float64)` has backedges to `calldouble` and `calldouble2`, but the second two do not because `double` was only called via runtime dispatch. However, `calldouble` has backedges to `calldouble2` - -```julia -julia> mis = methodinstances(calldouble) -2-element Vector{Core.MethodInstance}: - MethodInstance for calldouble(::Vector{Float64}) - MethodInstance for calldouble(::Vector{AbstractFloat}) - -julia> print_tree(mis[1]) -MethodInstance for calldouble(::Vector{Float64}) -└─ MethodInstance for calldouble2(::Vector{Float64}) - -julia> print_tree(mis[2]) -MethodInstance for calldouble(::Vector{AbstractFloat}) -└─ MethodInstance for calldouble2(::Vector{AbstractFloat}) -``` - -because `Vector{AbstractFloat}` is a concrete type, whereas `AbstractFloat` is not. - -If we create `c32 = [1.0f0]` and then `calldouble2(c32)`, we would also see backedges from `double(::Float32)` all the way back to `calldouble2(::Vector{Float32})`. - -## Precompilation - -During *package precompilation*, Julia creates a `*.ji` file typically stored in `.julia/compiled/v1.x/`, where `1.x` is your version of Julia. -Your `*.ji` file might just have definitions of constants, types, and methods, but optionally you can also include the results of type-inference. -This happens automatically if you run code while your package is being built, but generally the recommended procedure is to add *precompile directives*. - -Let's turn the example above into a package. In a fresh session, - -```julia -(@v1.6) pkg> generate SnoopCompileDemo - Generating project SnoopCompileDemo: - SnoopCompileDemo/Project.toml - SnoopCompileDemo/src/SnoopCompileDemo.jl - -julia> open("SnoopCompileDemo/src/SnoopCompileDemo.jl", "w") do io - write(io, """ - module SnoopCompileDemo - - double(x::Real) = 2x - calldouble(container) = double(container[1]) - calldouble2(container) = calldouble(container) - - precompile(calldouble2, (Vector{Float32},)) - precompile(calldouble2, (Vector{Float64},)) - precompile(calldouble2, (Vector{AbstractFloat},)) - - end - """) - end -282 - -julia> push!(LOAD_PATH, "SnoopCompileDemo/") -4-element Vector{String}: - "@" - "@v#.#" - "@stdlib" - "SnoopCompileDemo/" - -julia> using SnoopCompileDemo -[ Info: Precompiling SnoopCompileDemo [44c70eed-03a3-46c0-8383-afc033fb6a27] - -julia> using MethodAnalysis - -julia> methodinstances(SnoopCompileDemo.double) -3-element Vector{Core.MethodInstance}: - MethodInstance for double(::Float32) - MethodInstance for double(::Float64) - MethodInstance for double(::AbstractFloat) -``` - -Because of those `precompile` statements, the `MethodInstance`s exist after loading the package even though we haven't run the code in this session--not because it precompiled them when the package loaded, but because they were precompiled during the `Precompiling SnoopCompileDemo...` phase, stored to `*.ji` file, and then reloaded whenever we use the package. -You can also verify that the same backedges get created as when we ran this code interactively above. - -By having these `MethodInstance`s "pre-loaded" we can save some of the time needed to run type-inference: not much time in this case because the code is so simple, but for more complex methods the savings can be substantial. - -This code got cached in `SnoopCompileDemo.ji`. It's worth noting that even though the `precompile` directive got issued from this package, it might save `MethodInstances` for methods defined in other packages. -For example, Julia does not come pre-built with the inferred code for `Int * Float32`: in a fresh session, - -```julia -julia> using MethodAnalysis - -julia> mi = methodinstance(*, (Int, Float32)) - -``` -returns `nothing` (the `MethodInstance` doesn't exist), whereas if we've loaded `SnoopCompileDemo` then - -```julia -julia> mi = methodinstance(*, (Int, Float32)) -MethodInstance for *(::Int64, ::Float32) - -julia> mi.def -*(x::Number, y::Number) in Base at promotion.jl:322 -``` - -So even though the method is defined in `Base`, because `SnoopCompileDemo` needed this code it got stashed in `SnoopCompileDemo.ji`. - -*The ability to cache `MethodInstance`s from code defined in other packages or libraries is fundamental to latency reduction; however, it has significant limitations.* Most crucially, `*.ji` files can only hold code they "own," either: - -- to a method defined in the package -- through a chain of backedges to methods owned by the package - -If we add - -```julia -precompile(*, (Int, Float16)) -``` - -to the definition of `SnoopCompileDemo.jl`, nothing happens: - -```julia -julia> mi = methodinstance(*, (Int, Float16)) - # nothing -``` - -because there is no "chain of ownership" to `SnoopCompileDemo`. -Consequently, we can't precompile methods defined in other modules in and of themselves; we can only do it if those methods are linked by backedges to this package. - -Because backedges are created during successful type-inference, the consequence is that *precompilation works better when type inference succeeds.* -For some packages, time invested in improving inferrability can make your `precompile` directives work better. - -```@meta -DocTestSetup = nothing -``` diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md index 235f49c6..99925f4b 100644 --- a/docs/src/tutorials/invalidations.md +++ b/docs/src/tutorials/invalidations.md @@ -21,7 +21,7 @@ We'll illustrate invalidations by creating two packages, where loading the secon ### Add SnoopCompileCore, SnoopCompile, and helper packages to your environment -Here, we'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/). (With the exception of `AbstractTrees`, these "developer tool" packages should not be added to the Project file of any real packages unless you're extending the tool itself.) +Here, we'll add these packages to your [default environment](https://pkgdocs.julialang.org/v1/environments/). (With the exception of `AbstractTrees`, these "developer tool" packages should not be added to the Project file of any real packages unless you're extending the tool itself.) From your default environment (i.e., in package mode you should see something like `(@v1.10) pkg>`), do ``` using Pkg diff --git a/docs/src/tutorials/jet.md b/docs/src/tutorials/jet.md index 6443a366..af220d35 100644 --- a/docs/src/tutorials/jet.md +++ b/docs/src/tutorials/jet.md @@ -1,22 +1,17 @@ # Tutorial on JET integration -[JET](https://github.com/aviatesk/JET.jl) is a powerful tool for analyzing call graphs. -Some of its functionality overlaps that of SnoopCompile's, that is, JET also provides mechanisms to detect potential errors. -Conversely, JET is a purely static-analysis tool and lacks SnoopCompile's ability to "bridge" across runtime dispatch. -In summary, JET doesn't need Julia to restart to find inference failures, but JET will only find the first inference failure. -SnoopCompile has to run in a fresh session, but finds all inference failures. - -For this reason, the combination of the tools provides capabilities that neither package has on its own. -Specifically, one can use SnoopCompile to collect data on the callgraph and JET to perform the error analysis. +[JET](https://github.com/aviatesk/JET.jl) is a powerful tool for analyzing your code. +As described [elsewhere](@ref JET), some of its functionality overlaps SnoopCompile, but its mechanism of action is very different. The combination JET and SnoopCompile provides capabilities that neither package has on its own. +Specifically, one can use SnoopCompile to collect data on the full callgraph and JET to perform the exhaustive analysis of individual nodes. The integration between the two packages is bundled into SnoopCompile, specifically [`report_callee`](@ref), -[`report_callees`](@ref), and [`report_caller`](@ref). These take [`InferenceTrigger`](@ref) (see the page on [inference failures](@ref inferrability)) and use them to generate JET reports. +[`report_callees`](@ref), and [`report_caller`](@ref). These take [`InferenceTrigger`](@ref) (see the page on [inference failures](@ref inferrability)) and use them to generate JET reports. These tools focus on error-analysis rather than optimization, as SnoopCompile can already identify runtime dispatch. We can demonstrate both the need and use of these tools with a simple extended example. ## JET usage -JET provides a useful report for the following call: +As a basic introduction to JET, let's analyze the following call from JET's own documentation: ```jldoctest jet; filter=[r"@ reduce.*", r"(in|@)", r"(REPL\[\d+\]|none)"] julia> using JET @@ -62,7 +57,7 @@ ERROR: MethodError: no method matching zero(::Type{Any}) (This can be circumvented with `sum(Any[]; init=0)`.) -This is the kind of bug that can "lurk" undetected for a long time, and JET excels at exposing them. +This is the kind of bug that can lurk undetected for a long time, and JET excels at exposing them. ## JET limitations @@ -85,15 +80,13 @@ Because we "hid" the type of `list` from inference, JET couldn't tell what speci ## JET/SnoopCompile integration -The resolution to this problem is to use SnoopCompile to do the "data collection" and JET to do the analysis. +A resolution to this problem is to use SnoopCompile to do the "data collection" and JET to do the analysis. The key reason is that SnoopCompile is a dynamic analyzer, and is capable of bridging across runtime dispatch. As always, you need to do the data collection in a fresh session where the calls have not previously been inferred. After restarting Julia, we can do this: ```julia -julia> using SnoopCompile, JET, Cthulhu - -julia> using JET # this is necessary to enable the integration +julia> using SnoopCompileCore julia> list = Any[1,2,3]; @@ -101,8 +94,9 @@ julia> lc = Any[list]; # "hide" `list` inside a Vector{Any} julia> callsum(listcontainer) = sum(listcontainer[1]); -julia> tinf = @snoop_inference callsum(lc) -InferenceTimingNode: 0.039239/0.046793 on Core.Compiler.Timings.ROOT() with 2 direct children +julia> tinf = @snoop_inference callsum(lc); + +julia> using SnoopCompile, JET, Cthulhu julia> tinf.children 2-element Vector{SnoopCompileCore.InferenceTimingNode}: @@ -133,7 +127,7 @@ julia> report_callees(inference_triggers(tinf)) ││││││││││││││││└──────────────────── ``` -Because SnoopCompile collected the runtime-dispatched `sum` call, we can pass it to JET. +Because SnoopCompileCore collected the runtime-dispatched `sum` call, we can pass it to JET. `report_callees` filters those calls which generate JET reports, allowing you to focus on potential errors. !!! note diff --git a/docs/src/tutorials/pgdsgui.md b/docs/src/tutorials/pgdsgui.md index a0341f2b..2f4c3202 100644 --- a/docs/src/tutorials/pgdsgui.md +++ b/docs/src/tutorials/pgdsgui.md @@ -171,8 +171,8 @@ end ``` !!! warning - `where` type-parameters force specialization: regardless of `@nospecialize`: in `spelltype(@nospecialize(::Type{T})) where T`, the `@nospecialize` has no impact and you'll get full specialization on `T`. - Instead, use `@nospecialize(T::Type)` as shown. + `where` type-parameters force specialization: in `spelltype(@nospecialize(::Type{T})) where T`, the `@nospecialize` has no impact and you'll get full specialization on `T`. + Instead, use `@nospecialize(T::Type)` (without the `where` statement) as shown. If we now rerun that demo, you should see a plot of the same kind as shown above, but with different costs for each dot. The differences are best appreciated comparing them side-by-side ([`pgdsgui`](@ref) allows you to specify a particular axis into diff --git a/docs/src/tutorials/snoop_inference_analysis.md b/docs/src/tutorials/snoop_inference_analysis.md index 637e8574..d6024f58 100644 --- a/docs/src/tutorials/snoop_inference_analysis.md +++ b/docs/src/tutorials/snoop_inference_analysis.md @@ -729,7 +729,6 @@ v display(::REPL.REPLDisplay, ::MIME{Symbol("text/plain")}, ``` -`ascend` was covered in much greater detail in [fixing invalidations](@ref invalidations), and you can read about using it on that page. Here, one twist is that some lines contain content like ``` From d6323db5df94e05c173d67bcbecbb9f933a0c365 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Wed, 24 Jul 2024 02:43:55 -0500 Subject: [PATCH 16/18] WIP OptimizeMe demo --- docs/src/tutorials/snoop_inference.md | 2 +- .../src/tutorials/snoop_inference_analysis.md | 108 ++++-------------- examples/OptimizeMe.jl | 13 +-- examples/OptimizeMeFixed.jl | 19 +-- 4 files changed, 29 insertions(+), 113 deletions(-) diff --git a/docs/src/tutorials/snoop_inference.md b/docs/src/tutorials/snoop_inference.md index adad66a9..ebd1389a 100644 --- a/docs/src/tutorials/snoop_inference.md +++ b/docs/src/tutorials/snoop_inference.md @@ -154,7 +154,7 @@ Users are encouraged to read the ProfileView documentation to understand how to - right-clicking on a box opens the corresponding method in your editor - ctrl-click can be used to zoom in - empty horizontal spaces correspond to activities other than type-inference -- any boxes colored red (there are none in this particular example, but you'll see some later) correspond to *non-precompilable* `MethodInstance`s, in which the method is owned by one module but the types are from another unrelated module. +- any boxes colored red (there are none in this particular example, but you'll see some later) correspond to *naively non-precompilable* `MethodInstance`s, in which the method is owned by one module but the types are from another unrelated module. Such `MethodInstance`s are omitted from the precompile cache file unless they've been "marked" by `PrecompileTools.@compile_workload` or an explicit `precompile` directive. - any boxes colored orange-yellow (there is one in this demo) correspond to methods inferred for specific constants (constant propagation) You can explore this flamegraph and compare it to the output from `print_tree`. diff --git a/docs/src/tutorials/snoop_inference_analysis.md b/docs/src/tutorials/snoop_inference_analysis.md index d6024f58..201c3e78 100644 --- a/docs/src/tutorials/snoop_inference_analysis.md +++ b/docs/src/tutorials/snoop_inference_analysis.md @@ -5,37 +5,12 @@ Throughout this page, we'll use the `OptimizeMe` demo, which ships with `SnoopCo !!! note To understand what follows, it's essential to refer to [`OptimizeMe` source code](https://github.com/timholy/SnoopCompile.jl/blob/master/examples/OptimizeMe.jl) as you follow along. -```julia -julia> using SnoopCompile - -julia> cd(joinpath(pkgdir(SnoopCompile), "examples")) - -julia> include("OptimizeMe.jl") -Main.OptimizeMe - -julia> tinf = @snoop_inference OptimizeMe.main() -lotsa containers: -7-element Vector{Main.OptimizeMe.Container}: - Main.OptimizeMe.Container{Int64}(1) - Main.OptimizeMe.Container{UInt8}(0x01) - Main.OptimizeMe.Container{UInt16}(0xffff) - Main.OptimizeMe.Container{Float32}(2.0f0) - Main.OptimizeMe.Container{Char}('a') - Main.OptimizeMe.Container{Vector{Int64}}([0]) - Main.OptimizeMe.Container{Tuple{String, Int64}}(("key", 42)) -3.14 is great -2.718 is jealous -6-element Vector{Main.OptimizeMe.Object}: - Main.OptimizeMe.Object(1) - Main.OptimizeMe.Object(2) - Main.OptimizeMe.Object(3) - Main.OptimizeMe.Object(4) - Main.OptimizeMe.Object(5) - Main.OptimizeMe.Object(7) -InferenceTimingNode: 1.423913/2.713560 on InferenceFrameInfo for Core.Compiler.Timings.ROOT() with 77 direct children - -julia> fg = flamegraph(tinf) -Node(FlameGraphs.NodeData(ROOT() at typeinfer.jl:75, 0x00, 0:2713559552)) +```@repl fix-inference +using SnoopCompileCore, SnoopCompile # here we need the SnoopCompile path for the next line (normally you should wait until after data collection is complete) +cd(joinpath(pkgdir(SnoopCompile), "examples")) +include("OptimizeMe.jl") +tinf = @snoop_inference OptimizeMe.main(); +fg = flamegraph(tinf) ``` If you visualize `fg` with ProfileView, you'll see something like this: @@ -45,68 +20,39 @@ If you visualize `fg` with ProfileView, you'll see something like this: From the standpoint of precompilation, this has some obvious problems: - even though we called a single method, `OptimizeMe.main()`, there are many distinct flames separated by blank spaces. This indicates that many calls are being made by runtime dispatch: each separate flame is a fresh entrance into inference. -- several of the flames are marked in red, indicating that they are not precompilable. While SnoopCompile does have the capability to automatically emit `precompile` directives for the non-red bars that sit on top of the red ones, in some cases the red extends to the highest part of the flame. In such cases there is no available precompile directive, and therefore no way to avoid the cost of type-inference. +- several of the flames are marked in red, indicating that they are not naively precompilable (see the [Tutorial on `@snoop_inference`](@ref)). While `@compile_workload` can handle these flames, an even more robust solution is to eliminate them altogether. -Our goal will be to improve the design of `OptimizeMe` to make it more precompilable. +Our goal will be to improve the design of `OptimizeMe` to make it more readily precompilable. ## Analyzing inference triggers We'll first extract the "triggers" of inference, which is just a repackaging of part of the information contained within `tinf`. Specifically an [`InferenceTrigger`](@ref) captures callee/caller relationships that straddle a fresh entrance to type-inference, allowing you to identify which calls were made by runtime dispatch and what `MethodInstance` they called. -```julia -julia> itrigs = inference_triggers(tinf) -76-element Vector{InferenceTrigger}: - Inference triggered to call MethodInstance for vect(::Int64, ::Vararg{Any, N} where N) from lotsa_containers (/pathto/SnoopCompile/examples/OptimizeMe.jl:13) with specialization MethodInstance for lotsa_containers() - Inference triggered to call MethodInstance for promote_typeof(::Int64, ::UInt8, ::Vararg{Any, N} where N) from vect (./array.jl:126) with specialization MethodInstance for vect(::Int64, ::Vararg{Any, N} where N) - Inference triggered to call MethodInstance for promote_typeof(::UInt8, ::UInt16, ::Vararg{Any, N} where N) from promote_typeof (./promotion.jl:272) with specialization MethodInstance for promote_typeof(::Int64, ::UInt8, ::Vararg{Any, N} where N) - ⋮ +```@repl fix-inference +itrigs = inference_triggers(tinf) ``` -This indicates that a whopping 76 calls were (1) made by runtime dispatch and (2) the callee had not previously been inferred. -(There was a 77th call that had to be inferred, the original call to `main()`, but by default [`inference_triggers`](@ref) excludes calls made directly from top-level. You can change that through keyword arguments.) +The number of elements in this `Vector{InferenceTrigger}` tells you how many calls were (1) made by runtime dispatch and (2) the callee had not previously been inferred. !!! tip In the REPL, `SnoopCompile` displays `InferenceTrigger`s with yellow coloration for the callee, red for the caller method, and blue for the caller specialization. This makes it easier to quickly identify the most important information. -In some cases, this might indicate that you'll need to fix 76 separate callers; fortunately, in many cases fixing the origin of inference problems can fix a number of later callees. +In some cases, this might indicate that you'll need to fix each case separately; fortunately, in many cases fixing one problem addresses many other. ### [Method triggers](@id methtrigs) Most often, it's most convenient to organize them by the method triggering the need for inference: -```julia -julia> mtrigs = accumulate_by_source(Method, itrigs) -18-element Vector{SnoopCompile.TaggedTriggers{Method}}: - print_matrix_row(io::IO, X::AbstractVecOrMat{T} where T, A::Vector{T} where T, i::Integer, cols::AbstractVector{T} where T, sep::AbstractString) in Base at arrayshow.jl:96 (1 callees from 1 callers) - show(io::IO, x::T, forceuntyped::Bool, fromprint::Bool) where T<:Union{Float16, Float32, Float64} in Base.Ryu at ryu/Ryu.jl:111 (1 callees from 1 callers) - Pair(a, b) in Base at pair.jl:15 (1 callees from 1 callers) - vect(X...) in Base at array.jl:125 (1 callees from 1 callers) - makeobjects() in Main.OptimizeMe at /pathto/SnoopCompile/examples/OptimizeMe.jl:36 (1 callees from 1 callers) - show_delim_array(io::IO, itr, op, delim, cl, delim_one, i1, n) in Base at show.jl:1058 (1 callees from 1 callers) - typeinfo_prefix(io::IO, X) in Base at arrayshow.jl:515 (2 callees from 1 callers) - (::REPL.var"#38#39")(io) in REPL at /home/tim/src/julia-master/usr/share/julia/stdlib/v1.6/REPL/src/REPL.jl:214 (2 callees from 1 callers) - _cat_t(dims, ::Type{T}, X...) where T in Base at abstractarray.jl:1633 (2 callees from 1 callers) - contain_list(list) in Main.OptimizeMe at /pathto/SnoopCompile/examples/OptimizeMe.jl:27 (4 callees from 1 callers) - promote_typeof(x, xs...) in Base at promotion.jl:272 (4 callees from 4 callers) - combine_eltypes(f, args::Tuple) in Base.Broadcast at broadcast.jl:740 (5 callees from 1 callers) - lotsa_containers() in Main.OptimizeMe at /pathto/SnoopCompile/examples/OptimizeMe.jl:12 (7 callees from 1 callers) - alignment(io::IO, x) in Base at show.jl:2528 (7 callees from 7 callers) - var"#sprint#386"(context, sizehint::Integer, ::typeof(sprint), f::Function, args...) in Base at strings/io.jl:100 (8 callees from 2 callers) - alignment(io::IO, X::AbstractVecOrMat{T} where T, rows::AbstractVector{T} where T, cols::AbstractVector{T} where T, cols_if_complete::Integer, cols_otherwise::Integer, sep::Integer) in Base at arrayshow.jl:60 (8 callees from 2 callers) - copyto_nonleaf!(dest, bc::Base.Broadcast.Broadcasted, iter, state, count) in Base.Broadcast at broadcast.jl:1070 (9 callees from 3 callers) - _show_default(io::IO, x) in Base at show.jl:397 (12 callees from 1 callers) +```@repl fix-inference +mtrigs = accumulate_by_source(Method, itrigs) ``` The methods triggering the largest number of inference runs are shown at the bottom. -You can select methods from a particular module: +You can also select methods from a particular module: -```julia -julia> modtrigs = filtermod(OptimizeMe, mtrigs) -3-element Vector{SnoopCompile.TaggedTriggers{Method}}: - makeobjects() in Main.OptimizeMe at /home/tim/.julia/dev/SnoopCompile/examples/OptimizeMe.jl:36 (1 callees from 1 callers) - contain_list(list) in Main.OptimizeMe at /home/tim/.julia/dev/SnoopCompile/examples/OptimizeMe.jl:27 (4 callees from 1 callers) - lotsa_containers() in Main.OptimizeMe at /home/tim/.julia/dev/SnoopCompile/examples/OptimizeMe.jl:12 (7 callees from 1 callers) +```@repl fix-inference +modtrigs = filtermod(OptimizeMe, mtrigs) ``` Rather than filter by a single module, you can alternatively call `SnoopCompile.parcel(mtrigs)` to split them out by module. @@ -115,28 +61,20 @@ However, many of the failures in `Base` were nevertheless indirectly due to `Opt Fortunately, we'll see that using more careful design in `OptimizeMe` can avoid many of those problems. !!! tip - If you have a longer list of inference triggers than you feel comfortable tackling, filtering by your package's module is probably the best way to start. + If you have a longer list of inference triggers than you feel comfortable tackling, filtering by your package's module or using [`precompile_blockers`](@ref) can be a good way to start. Fixing issues in the package itself can end up resolving many of the "indirect" triggers too. Also be sure to note the ability to filter out likely "noise" from [test suites](@ref test-suites). -If you're hoping to fix inference problems, one of the most efficient things you can do is call `summary`: +If you're hoping to fix inference problems, one tool that's sometimes useful is `summary`: -```julia -julia> mtrig = modtrigs[1] -makeobjects() in Main.OptimizeMe at /home/tim/.julia/dev/SnoopCompile/examples/OptimizeMe.jl:36 (1 callees from 1 callers) - -julia> summary(mtrig) -makeobjects() in Main.OptimizeMe at /home/tim/.julia/dev/SnoopCompile/examples/OptimizeMe.jl:36 had 1 specializations -Triggering calls: -Inlined _cat at ./abstractarray.jl:1630: calling cat_t##kw (1 instances) +```@repl fix-inference +mtrig = modtrigs[1] +summary(mtrig) ``` Sometimes from these hints alone you can figure out how to fix the problem. -(`Inlined _cat` means that the inference trigger did not come directly from a source line of `makeobjects` but from a call, `_cat`, that got inlined into the compiled version. -Below we'll see more concretely how to interpret this hint.) - You can also say `edit(mtrig)` and be taken directly to the method you're analyzing in your editor. -Finally, you can recover the individual triggers: +But often, you have to "dig deep" into individual triggers: ```julia julia> mtrig.itrigs[1] diff --git a/examples/OptimizeMe.jl b/examples/OptimizeMe.jl index 20952407..ba1c4196 100644 --- a/examples/OptimizeMe.jl +++ b/examples/OptimizeMe.jl @@ -1,7 +1,8 @@ """ -OptimizeMe is a demonstration module used in illustrating how to improve code and generate effective `precompile` directives. -It has deliberate weaknesses in its design, and the analysis of these weaknesses via `@snoop_inference` is discussed -in the documentation. +OptimizeMe is a module used to demonstrate how to make code more precompilable +and more resistant to invalidation. It has deliberate weaknesses in its design, +and the analysis and resolution of these weaknesses via `@snoop_inference` is +discussed in the documentation. """ module OptimizeMe @@ -33,17 +34,11 @@ struct Object x::Int end -function makeobjects() - xs = [1:5; 7] - return Object.(xs) -end - function main() lotsa_containers() println(contain_concrete(3.14, "is great")) list = [2.718, "is jealous"] println(contain_list(list)) - display(makeobjects()) end end diff --git a/examples/OptimizeMeFixed.jl b/examples/OptimizeMeFixed.jl index 0c158fbf..9e5841ba 100644 --- a/examples/OptimizeMeFixed.jl +++ b/examples/OptimizeMeFixed.jl @@ -1,7 +1,5 @@ """ -OptimizeMe is a demonstration module used in illustrating how to improve code and generate effective `precompile` directives. -It has deliberate weaknesses in its design, and the analysis of these weaknesses via `@snoop_inference` is discussed -in the documentation. +OptimizeMeFixed is the "improved" version of OptimizeMe. See the file in this same directory for details. """ module OptimizeMeFixed @@ -41,19 +39,6 @@ function makeobjects() return Object.(xs) end -# "Stub" callers for precompilability -function warmup() - mime = MIME("text/plain") - io = Base.stdout::Base.TTY - v = [Container{Any}(0)] - show(io, mime, v) - show(IOContext(io), mime, v) - v = [Object(0)] - show(io, mime, v) - show(IOContext(io), mime, v) - return nothing -end - function main() lotsa_containers() println(contain_concrete(3.14, "is great")) @@ -62,7 +47,5 @@ function main() display(makeobjects()) end -precompile(Tuple{typeof(main)}) # time: 0.4204474 -precompile(Tuple{typeof(warmup)}) end From 3c328f17d0701bcc3dc5f7bb5bb000a2ef85c9e9 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Wed, 24 Jul 2024 07:36:07 -0500 Subject: [PATCH 17/18] Rewrite the `OptimizeMe` example --- docs/src/assets/ascend_optimizeme1.png | Bin 0 -> 237919 bytes .../src/tutorials/snoop_inference_analysis.md | 625 +++--------------- examples/OptimizeMe.jl | 29 +- examples/OptimizeMeFixed.jl | 58 +- src/parcel_snoop_inference.jl | 13 +- 5 files changed, 132 insertions(+), 593 deletions(-) create mode 100644 docs/src/assets/ascend_optimizeme1.png diff --git a/docs/src/assets/ascend_optimizeme1.png b/docs/src/assets/ascend_optimizeme1.png new file mode 100644 index 0000000000000000000000000000000000000000..aa595dd935b940b6bc28d47903a7b77a3e1c6a22 GIT binary patch literal 237919 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 l(z7Prdgd$Mg>^BYddaRu($)q^(r#1eXw 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 z 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 IHJiKt0~4@aKmY&$ literal 0 HcmV?d00001 diff --git a/docs/src/tutorials/snoop_inference_analysis.md b/docs/src/tutorials/snoop_inference_analysis.md index 201c3e78..e9b66027 100644 --- a/docs/src/tutorials/snoop_inference_analysis.md +++ b/docs/src/tutorials/snoop_inference_analysis.md @@ -13,7 +13,7 @@ tinf = @snoop_inference OptimizeMe.main(); fg = flamegraph(tinf) ``` -If you visualize `fg` with ProfileView, you'll see something like this: +If you visualize `fg` with ProfileView, you may see something like this: ![flamegraph-OptimizeMe](../assets/flamegraph-OptimizeMe.png) @@ -65,485 +65,132 @@ Fortunately, we'll see that using more careful design in `OptimizeMe` can avoid Fixing issues in the package itself can end up resolving many of the "indirect" triggers too. Also be sure to note the ability to filter out likely "noise" from [test suites](@ref test-suites). -If you're hoping to fix inference problems, one tool that's sometimes useful is `summary`: +You can get an overview of each Method trigger with `summary`: ```@repl fix-inference mtrig = modtrigs[1] summary(mtrig) ``` -Sometimes from these hints alone you can figure out how to fix the problem. You can also say `edit(mtrig)` and be taken directly to the method you're analyzing in your editor. -But often, you have to "dig deep" into individual triggers: +You can still "dig deep" into individual triggers: -```julia -julia> mtrig.itrigs[1] -Inference triggered to call MethodInstance for (::Base.var"#cat_t##kw")(::NamedTuple{(:dims,), Tuple{Val{1}}}, ::typeof(Base.cat_t), ::Type{Int64}, ::UnitRange{Int64}, ::Vararg{Any, N} where N) from _cat (./abstractarray.jl:1630) inlined into MethodInstance for makeobjects() (/home/tim/.julia/dev/SnoopCompile/examples/OptimizeMe.jl:37) +```@repl fix-inference +itrig = mtrig.itrigs[1] ``` -This is useful if you want to analyze a method via [`Cthulhu.ascend`](@ref ascend-itrig). -`Method`-based triggers, which may aggregate many different individual triggers, are particularly useful mostly because tools like [Cthulhu.jl](https://github.com/JuliaDebug/Cthulhu.jl) show you the inference results for the entire `MethodInstance`, allowing you to fix many different inference problems at once. +This is useful if you want to analyze with [`Cthulhu.ascend`](@ref ascend-itrig). +`Method`-based triggers, which may aggregate many different individual triggers, can be useful because tools like [Cthulhu.jl](https://github.com/JuliaDebug/Cthulhu.jl) show you the inference results for the entire `MethodInstance`, allowing you to fix many different inference problems at once. ### Trigger trees While method triggers are probably the most useful way of organizing these inference triggers, for learning purposes here we'll use a more detailed scheme, which organizes inference triggers in a tree: -```julia -julia> itree = trigger_tree(itrigs) -TriggerNode for root with 14 direct children - -julia> using AbstractTrees - -julia> print_tree(itree) -root -├─ MethodInstance for vect(::Int64, ::Vararg{Any, N} where N) -│ └─ MethodInstance for promote_typeof(::Int64, ::UInt8, ::Vararg{Any, N} where N) -│ └─ MethodInstance for promote_typeof(::UInt8, ::UInt16, ::Vararg{Any, N} where N) -│ └─ MethodInstance for promote_typeof(::UInt16, ::Float32, ::Vararg{Any, N} where N) -│ └─ MethodInstance for promote_typeof(::Float32, ::Char, ::Vararg{Any, N} where N) -│ ⋮ -│ -├─ MethodInstance for combine_eltypes(::Type, ::Tuple{Vector{Any}}) -│ ├─ MethodInstance for return_type(::Any, ::Any) -│ ├─ MethodInstance for return_type(::Any, ::Any, ::UInt64) -│ ├─ MethodInstance for return_type(::Core.Compiler.NativeInterpreter, ::Any, ::Any) -│ ├─ MethodInstance for contains_is(::Core.SimpleVector, ::Any) -│ └─ MethodInstance for promote_typejoin_union(::Type{Main.OptimizeMe.Container}) -├─ MethodInstance for Main.OptimizeMe.Container(::Int64) -⋮ +```@repl fix-inference +itree = trigger_tree(itrigs) +using AbstractTrees +print_tree(itree) ``` +This gives you a big-picture overview of how the inference failures arose. The parent-child relationships are based on the backtraces at the entrance to inference, and the nodes are organized in the order in which inference occurred. +Inspection of these trees can be informative; for example, here we notice a lot of method specializations for `Container{T}` for different `T`. -We're going to march through these systematically. Let's start with the first of these. - -### `suggest` and a fix involving manual `eltype` specification - -Because the analysis of inference failures is somewhat complex, `SnoopCompile` attempts to [`suggest`](@ref) an interpretation and/or remedy for each trigger: - -``` -julia> suggest(itree.children[1]) -/pathto/SnoopCompile/examples/OptimizeMe.jl:13: invoked callee is varargs (ignore this one, homogenize the arguments, declare an umbrella type, or force-specialize the callee MethodInstance for vect(::Int64, ::Vararg{Any, N} where N)) -immediate caller(s): -1-element Vector{Base.StackTraces.StackFrame}: - main() at OptimizeMe.jl:42 -└─ ./array.jl:126: caller is varargs (ignore this one, specialize the caller vect(::Int64, ::Vararg{Any, N} where N) at array.jl:126, or improve inferrability of its caller) - immediate caller(s): - 1-element Vector{Base.StackTraces.StackFrame}: - lotsa_containers() at OptimizeMe.jl:13 - └─ ./promotion.jl:272: caller is varargs (ignore this one, specialize the caller promote_typeof(::Int64, ::UInt8, ::Vararg{Any, N} where N) at promotion.jl:272, or improve inferrability of its caller) - immediate caller(s): - 1-element Vector{Base.StackTraces.StackFrame}: - vect(::Int64, ::Vararg{Any, N} where N) at array.jl:126 - └─ ./promotion.jl:272: caller is varargs (ignore this one, specialize the caller promote_typeof(::UInt8, ::UInt16, ::Vararg{Any, N} where N) at promotion.jl:272, or improve inferrability of its caller) - immediate caller(s): - 1-element Vector{Base.StackTraces.StackFrame}: - promote_typeof(::Int64, ::UInt8, ::Vararg{Any, N} where N) at promotion.jl:272 - └─ ./promotion.jl:272: caller is varargs (ignore this one, specialize the caller promote_typeof(::UInt16, ::Float32, ::Vararg{Any, N} where N) at promotion.jl:272, or improve inferrability of its caller) - immediate caller(s): - 1-element Vector{Base.StackTraces.StackFrame}: - promote_typeof(::UInt8, ::UInt16, ::Vararg{Any, N} where N) at promotion.jl:272 - └─ ./promotion.jl:272: caller is varargs (ignore this one, specialize the caller promote_typeof(::Float32, ::Char, ::Vararg{Any, N} where N) at promotion.jl:272, or improve inferrability of its caller) - immediate caller(s): - 1-element Vector{Base.StackTraces.StackFrame}: - promote_typeof(::UInt16, ::Float32, ::Vararg{Any, N} where N) at promotion.jl:272 - ⋮ -``` - -!!! tip - In the REPL, interpretations are highlighted in color to help distinguish individual suggestions. - -In this case, the interpretation for the first node is "invoked callee is varargs" and suggestions are to choose one of "ignore...homogenize...umbrella type...force-specialize". -Initially, this may seem pretty opaque. -It helps if we look at the referenced line `OptimizeMe.jl:13`: - -```julia -list = [1, 0x01, 0xffff, 2.0f0, 'a', [0], ("key", 42)] -``` - -You'll notice above that the callee for the first node is `vect`; that's what handles the creation of the vector `[1, ...]`. -If you look back up at the `itree`, you can see that a lot of `promote_typeof` calls follow, and you can see that the types listed in the arguments match the elements in `list`. -The problem, here, is that `vect` has never been inferred for this particular combination of argument types, and the fact that the types are diverse means that Julia has decided not to specialize it for this combination. -(If Julia had specialized it, it would have been inferred when `lotsa_containers` was inferred; the fact that it is showing up as a trigger means it wasn't.) - -Let's see what kind of object this line creates: - -```julia -julia> typeof(list) -Vector{Any} (alias for Array{Any, 1}) -``` - -Since it creates a `Vector{Any}`, perhaps we should just tell Julia to create such an object directly: we modify `[1, 0x01, ...]` to `Any[1, 0x01, ...]` (note the `Any` in front of `[`), so that Julia doesn't have to deduce the container type on its own. -This follows the "declare an umbrella type" suggestion. - -!!! note - "Force-specialize" means to encourage Julia to violate its heuristics and specialize the callee. - Often this can be achieved by supplying a "spurious" type parameter. - Examples include replacing `higherorderfunction(f::Function, args...)` with `function higherorderfunction(f::F, args...) where F<:Function`, - or `function getindex(A::MyArrayType{T,N}, idxs::Vararg{Int,N}) where {T,N}` instead of just `getindex(A::MyArrayType, idxs::Int...)`. - (In the latter case, the `N` parameter is the crucial one: it forces specialization for a particular number of `Int` arguments.) - - This technique is not useful for the particular case we analyzed here, but it can be in other settings. - -Making this simple 3-character fix eliminates that entire branch of the tree (a savings of 6 inference triggers). - -### `eltype`s and reducing specialization in `broadcast` - -Let's move on to the next entry: - -``` -julia> print_tree(itree.children[2]) -MethodInstance for combine_eltypes(::Type, ::Tuple{Vector{Any}}) -├─ MethodInstance for return_type(::Any, ::Any) -├─ MethodInstance for return_type(::Any, ::Any, ::UInt64) -├─ MethodInstance for return_type(::Core.Compiler.NativeInterpreter, ::Any, ::Any) -├─ MethodInstance for contains_is(::Core.SimpleVector, ::Any) -└─ MethodInstance for promote_typejoin_union(::Type{Main.OptimizeMe.Container}) - -julia> suggest(itree.children[2]) -./broadcast.jl:905: regular invoke (perhaps precompile lotsa_containers() at OptimizeMe.jl:14) -├─ ./broadcast.jl:740: I've got nothing to say for MethodInstance for return_type(::Any, ::Any) consider `stacktrace(itrig)` or `Cthulhu.ascend(itrig)` -├─ ./broadcast.jl:740: I've got nothing to say for MethodInstance for return_type(::Any, ::Any, ::UInt64) consider `stacktrace(itrig)` or `Cthulhu.ascend(itrig)` -├─ ./broadcast.jl:740: I've got nothing to say for MethodInstance for return_type(::Core.Compiler.NativeInterpreter, ::Any, ::Any) consider `stacktrace(itrig)` or `Cthulhu.ascend(itrig)` -├─ ./broadcast.jl:740: I've got nothing to say for MethodInstance for contains_is(::Core.SimpleVector, ::Any) consider `stacktrace(itrig)` or `Cthulhu.ascend(itrig)` -└─ ./broadcast.jl:740: non-inferrable call, perhaps annotate combine_eltypes(f, args::Tuple) in Base.Broadcast at broadcast.jl:740 with type MethodInstance for promote_typejoin_union(::Type{Main.OptimizeMe.Container}) - If a noninferrable argument is a type or function, Julia's specialization heuristics may be responsible. - immediate caller(s): - 3-element Vector{Base.StackTraces.StackFrame}: - copy at broadcast.jl:905 [inlined] - materialize at broadcast.jl:883 [inlined] - lotsa_containers() at OptimizeMe.jl:14 -``` - -While this tree is attributed to `broadcast`, you can see several references here to `OptimizeMe.jl:14`, which contains: - -```julia -cs = Container.(list) -``` +We're going to march through these systematically. -`Container.(list)` is a broadcasting operation, and once again we find that this has inferrability problems. -In this case, the initial suggestion "perhaps precompile `lotsa_containers`" is *not* helpful. -(The "regular invoke" just means that the initial call was one where inference knew all the argument types, and hence in principle might be precompilable, but from this tree we see that this broke down in some of its callees.) -Several children have no interpretation ("I've got nothing to say..."). -Only the last one, "non-inferrable call", is (marginally) useful, it means that a call was made with arguments whose types could not be inferred. +### `suggest` and fixing `Core.Box` -!!! warning - You should always view these suggestions skeptically. - Often, they flag downstream issues that are better addressed at the source; frequently the best fix may be at a line a bit before the one identified in a trigger, or even in a dependent callee of a line prior to the flagged one. - This is a product of the fact that *returning* a non-inferrable argument is not the thing that forces a new round of inference; - it's *doing something* (making a specialization-worthy call) with the object of non-inferrable type that triggers a fresh entrance into inference. +You may have noticed above that `summary(mtrig)` generated a red `has Core.Box` message. Assuming that `itrig` is still the first (and it turns out, only) trigger from this method, let's look at this again, explicitly using [`suggest`](@ref), the tool that generated this hint: -How might we go about fixing this? -One hint is to notice that `itree.children[3]` through `itree.children[7]` also ultimiately derive from this one line of `OptimizeMe`, -but from a later line within `broadcast.jl` which explains why they are not bundled together with `itree.children[2]`. -May of these correspond to creating different `Container` types, for example: - -``` -└─ MethodInstance for restart_copyto_nonleaf!(::Vector{Main.OptimizeMe.Container}, ::Vector{Main.OptimizeMe.Container{Int64}}, ::Base.Broadcast.Broadcasted{Base.Broadcast.DefaultArrayStyle{1}, Tuple{Base.OneTo{Int64}}, Type{Main.OptimizeMe.Container}, Tuple{Base.Broadcast.Extruded{Vector{Any}, Tuple{Bool}, Tuple{Int64}}}}, ::Main.OptimizeMe.Container{UInt8}, ::Int64, ::Base.OneTo{Int64}, ::Int64, ::Int64) - ├─ MethodInstance for Main.OptimizeMe.Container(::UInt16) - ├─ MethodInstance for Main.OptimizeMe.Container(::Float32) - ├─ MethodInstance for Main.OptimizeMe.Container(::Char) - ├─ MethodInstance for Main.OptimizeMe.Container(::Vector{Int64}) - └─ MethodInstance for Main.OptimizeMe.Container(::Tuple{String, Int64}) -``` - -We've created a `Container{T}` for each specific `T` of the objects in `list`. -In some cases, there may be good reasons for such specialization, and in such cases we just have to live with these inference failures. -However, in other cases the specialization might be detrimental to compile-time and/or runtime performance. -In such cases, we might decide to create them all as `Container{Any}`: - -```julia -cs = Container{Any}.(list) -``` - -This 5-character change ends up eliminating 45 of our original 76 triggers. -Not only did we eliminate the triggers from broadcasting, but we limited the number of different `show(::IO, ::Container{T})`-`MethodInstance`s we need from later calls in `main`. - -When the `Container` constructor does more complex operations, in some cases you may find that `Container{Any}(args...)` still gets specialized for different types of `args...`. -In such cases, you can create a special constructor that instructs Julia to avoid specialization in specific instances, e.g., - -```julia -struct Container{T} - field1::T - morefields... - - # This constructor permits specialization on `args` - Container{T}(args...) where {T} = new{T}(args...) - - # For Container{Any}, we prevent specialization - Container{Any}(@nospecialize(args...)) = new{Any}(args...) -end +```@repl fix-inference +suggest(itrig) ``` -If you're following along, the best option is to make these fixes and go back to the beginning, re-collecting `tinf` and processing the triggers. -We're down to 32 inference triggers. - -### [Adding type-assertions](@id typeasserts) +You can see that SnoopCompile recommends tackling this first; depending on how much additional code is affected, fixing a `Core.Box` allows inference to work better and may resolve other triggers. -If you've made the fixes above, the first child of `itree` is one for `show(::IOContext{Base.TTY}, ::MIME{Symbol("text/plain")}, ::Vector{Main.OptimizeMe.Container{Any}})`; -we'll skip that one for now, because it's a bit more sophisticated. -Right below it, we see +This message also directs readers to a section of [this documentation](@ref Fixing-Core.Box) that links to a page of the Julia manual describing the underlying problem. The Julia documentation suggests a couple of fixes, of which the best (in this case) is to use the `let` statement to rebind the variable and end any "conflict" with the closure: ``` -├─ MethodInstance for combine_eltypes(::Type, ::Tuple{Vector{Any}}) -│ ├─ MethodInstance for return_type(::Any, ::Any) -│ ├─ MethodInstance for return_type(::Any, ::Any, ::UInt64) -``` - -and related nodes for `similar`, `copyto_nonleaf!`, etc., just as we saw above, so this looks like another case of broadcasting failure. -In this case, `suggest` quickly indicates that it's the broadcasting in - -```julia -function contain_list(list) - cs = Container.(list) - return concat_string(cs...) -end -``` - -Now we know the problem: `main` creates `list = [2.718, "is jealous"]`, -a vector with different object types, and this leads to inference failures in broadcasting. -But wait, you might notice, `contain_concrete` gets called before `contain_list`, why doesn't it have a problem? -The reason is that `contain_concrete` and its callee, `concat_string`, provide opportunities for inference to handle each object in a separate argument; -the problems arise from bundling objects of different types into the same container. - -There are several ways we could go about fixig this example: - -- we could delete `contain_list` altogether and use `contain_concrete` for everything. -- we could try creating `list` as a tuple rather than a `Vector{Any}`; (small) tuples sometimes allow inference to succeed even when each element has a different type. This is as simple as changing `list = [2.718, "is jealous"]` to `list = (2.718, "is jealous")`, but whether it works to solve all your inference problems depends on the particular case. -- we could use external knowledge to annotate the types of the items in `list::Vector{Any}`. - -Here we'll illustrate the last of these, since it's the only one that's nontrivial. -(It's also often a useful pattern in many real-world contexts, such as cases where you have a `Dict{String,Any}` but know something about the kinds of value-types associated with particular string keys.) -We could rewrite `contain_list` so it looks like this: - -```julia -function contain_list(list) - length(list) == 2 || throw(DimensionMismatch("list must have length 2")) - item1 = list[1]::Float64 - item2 = list[2]::String - return contain_concrete(item1, item2) # or we could repeat the body of contain_concrete +function abmult(r::Int, ys) + if r < 0 + r = -r + end + let r = r # Julia #15276 + return map(x -> howbig(r * x), ys) + end end ``` -The type-assertions tell inference that the corresponding items have the given types, and assist inference in cases where it has no mechanism to deduce the answer on its own. -Julia will throw an error if the type-assertion fails. -In some cases, a more forgiving option might be - -```julia -item1 = convert(Float64, list[1])::Float64 -``` -which will attempt to convert `list[1]` to a `Float64`, and therefore handle a wider range of number types stored in the first element of `list`. -Believe it or not, both the `convert()` and the `::Float64` type-assertion are necessary: -since `list[1]` is of type `Any`, Julia will not be able to deduce which `convert` method will be used to perform the conversion, and it's always possible that someone has written a sloppy `convert` that doesn't return a value of the requested type. -Without that final `::Float64`, inference cannot simply assume that the result is a `Float64`. -The type-assert `::Float64` enforces the fact that you're expecting that `convert` call to actually return a `Float64`--it will error if it fails to do so, and it's this error that allows inference to be certain that for the purposes of any later code it must be a `Float64`. - -Of course, this just trades one form of inference failure for another--the call to `convert` will be made by runtime dispatch--but this can nevertheless be a big win for two reasons: - -- even though the `convert` call will be made by runtime dispatch, in this particular case `convert(Float64, ::Float64)` is already compiled in Julia itself. Consequently it doesn't require a fresh run of inference. -- even in cases where the types are such that `convert` might need to be inferred & compiled, the type-assertion allows Julia to assume that `item1` is henceforth a `Float64`. This makes it possible for inference to succeed for any code that follows. When that's a large amount of code, the savings can be considerable. -Let's make that fix and also annotate the container type from `main`, `list = Any[2.718, "is jealous"]`. -Just to see how we're progressing, we start a fresh session and discover we're down to 20 triggers with just three direct branches. -### Vararg homogenization +### `suggest` and a fix involving manual `eltype` specification -We'll again skip over the `show` branches (they are two of the remaining three), and focus on this one: +Let's look at the other Method-trigger rooted in `OptimizeMe`: -```julia -julia> node = itree.children[2] -TriggerNode for MethodInstance for (::Base.var"#cat_t##kw")(::NamedTuple{(:dims,), Tuple{Val{1}}}, ::typeof(Base.cat_t), ::Type{Int64}, ::UnitRange{Int64}, ::Vararg{Any, N} where N) with 2 direct children - -julia> print_tree(node) -MethodInstance for (::Base.var"#cat_t##kw")(::NamedTuple{(:dims,), Tuple{Val{1}}}, ::typeof(Base.cat_t), ::Type{Int64}, ::UnitRange{Int64}, ::Vararg{Any, N} where N) -├─ MethodInstance for cat_similar(::UnitRange{Int64}, ::Type, ::Tuple{Int64}) -└─ MethodInstance for __cat(::Vector{Int64}, ::Tuple{Int64}, ::Tuple{Bool}, ::UnitRange{Int64}, ::Vararg{Any, N} where N) - -julia> suggest(node) -./abstractarray.jl:1630: invoked callee is varargs (ignore this one, force-specialize the callee MethodInstance for (::Base.var"#cat_t##kw")(::NamedTuple{(:dims,), Tuple{Val{1}}}, ::typeof(Base.cat_t), ::Type{Int64}, ::UnitRange{Int64}, ::Vararg{Any, N} where N), or declare an umbrella type) -immediate caller(s): -1-element Vector{Base.StackTraces.StackFrame}: - main() at OptimizeMe.jl:48 -├─ ./abstractarray.jl:1636: caller is varargs (ignore this one, specialize the caller _cat_t(::Val{1}, ::Type{Int64}, ::UnitRange{Int64}, ::Vararg{Any, N} where N) at abstractarray.jl:1636, or improve inferrability of its caller) -│ immediate caller(s): -│ 1-element Vector{Base.StackTraces.StackFrame}: -│ cat_t(::Type{Int64}, ::UnitRange{Int64}, ::Vararg{Any, N} where N; dims::Val{1}) at abstractarray.jl:1632 -└─ ./abstractarray.jl:1640: caller is varargs (ignore this one, specialize the caller _cat_t(::Val{1}, ::Type{Int64}, ::UnitRange{Int64}, ::Vararg{Any, N} where N) at abstractarray.jl:1640, or improve inferrability of its caller) - immediate caller(s): - 1-element Vector{Base.StackTraces.StackFrame}: - cat_t(::Type{Int64}, ::UnitRange{Int64}, ::Vararg{Any, N} where N; dims::Val{1}) at abstractarray.jl:1632 +```@repl fix-inference +mtrig = modtrigs[2] +summary(mtrig) +itrig = mtrig.itrigs[1] ``` -Due to Julia's optimization and inlining, it's sometimes a bit hard to tell from these shortened displays where a particular trigger comes from. -(It turns out that this is finally the trigger we looked at in greatest detail in [method-based triggers](@ref methtrigs).) -In this case we extract the specific trigger and show the stacktrace: +If you use Cthulhu's `ascend(itrig)` you might see something like this: -```julia -julia> itrig = node.itrig -Inference triggered to call MethodInstance for (::Base.var"#cat_t##kw")(::NamedTuple{(:dims,), Tuple{Val{1}}}, ::typeof(Base.cat_t), ::Type{Int64}, ::UnitRange{Int64}, ::Vararg{Any, N} where N) from _cat (./abstractarray.jl:1630) inlined into MethodInstance for makeobjects() (/tmp/OptimizeMe.jl:39) - -julia> stacktrace(itrig) -24-element Vector{Base.StackTraces.StackFrame}: - exit_current_timer at typeinfer.jl:166 [inlined] - typeinf(interp::Core.Compiler.NativeInterpreter, frame::Core.Compiler.InferenceState) at typeinfer.jl:208 - typeinf_ext(interp::Core.Compiler.NativeInterpreter, mi::Core.MethodInstance) at typeinfer.jl:835 - typeinf_ext_toplevel(interp::Core.Compiler.NativeInterpreter, linfo::Core.MethodInstance) at typeinfer.jl:868 - typeinf_ext_toplevel(mi::Core.MethodInstance, world::UInt64) at typeinfer.jl:864 - _cat at abstractarray.jl:1630 [inlined] - #cat#127 at abstractarray.jl:1769 [inlined] - cat at abstractarray.jl:1769 [inlined] - vcat at abstractarray.jl:1698 [inlined] - makeobjects() at OptimizeMe.jl:39 - main() at OptimizeMe.jl:48 - top-level scope at snoop_inference.jl:53 - eval(m::Module, e::Any) at boot.jl:360 - eval_user_input(ast::Any, backend::REPL.REPLBackend) at REPL.jl:139 - repl_backend_loop(backend::REPL.REPLBackend) at REPL.jl:200 - start_repl_backend(backend::REPL.REPLBackend, consumer::Any) at REPL.jl:185 - run_repl(repl::REPL.AbstractREPL, consumer::Any; backend_on_current_task::Bool) at REPL.jl:317 - run_repl(repl::REPL.AbstractREPL, consumer::Any) at REPL.jl:305 - (::Base.var"#872#874"{Bool, Bool, Bool})(REPL::Module) at client.jl:387 - #invokelatest#2 at essentials.jl:707 [inlined] - invokelatest at essentials.jl:706 [inlined] - run_main_repl(interactive::Bool, quiet::Bool, banner::Bool, history_file::Bool, color_set::Bool) at client.jl:372 - exec_options(opts::Base.JLOptions) at client.jl:302 - _start() at client.jl:485 -``` +![ascend-lotsa](../assets/ascend_optimizeme1.png) -(You can also call `stacktrace` directly on `node`.) -It's the lines immediately following `typeinf_ext_toplevel` that need concern us: -you can see that the "last stop" on code we wrote here was `makeobjects() at OptimizeMe.jl:39`, after which it goes fairly deep into the concatenation pipeline before suffering an inference trigger at `_cat at abstractarray.jl:1630`. +The first thing to note here is that `cs` is inferred as an `AbstractVector`--fixing this to make it a concrete type should be our next goal. There's a second, more subtle hint: in the call menu at the bottom, the selected call is marked `< semi-concrete eval >`. This is a hint that a method is being called with a non-concrete type. -In this case, the first hint is quite useful, if we know how to interpret it. -The `invoked callee is varargs` reassures us that the immediate caller, `_cat`, knows exactly which method it is calling (that's the meaning of the `invoked`). -The real problem is that it doesn't know how to specialize it. -The suggestion to `homogenize the arguments` is the crucial hint: -the problem comes from the fact that in +What might that non-concrete type be? -```julia -xs = [1:5; 7] +```@repl fix-inference +isconcretetype(OptimizeMe.Container) ``` -`1:5` is a `UnitRange{Int}` whereas `7` is an `Int`, and the fact that these are two different types prevents Julia from knowing how to specialize that varargs call. -But this is easy to fix, because the result will be identical if we write this as +The statement `Container.(list)` is thus creating an `AbstractVector` with a non-concrete element type. +You can seem in greater detail what happens, inference-wise, in this snippet from `print_tree(itree)`: -```julia -xs = [1:5; 7:7] ``` - -in which case both arguments are `UnitRange{Int}`, and this allows Julia to specialize the varargs call. - -!!! note - It's generally a good thing that Julia doesn't specialize each and every varargs call, because the lack of specialization reduces latency. - However, when you can homogenize the argument types and make it inferrable, you make it more worthy of precompilation, which is a different and ultimately more impactful approach to latency reduction. - -### Defining `show` methods for custom types - -Finally we are left with nodes that are related to `show`. -We'll temporarily skip the first of these and examine - -```julia -julia> print_tree(node) -MethodInstance for show(::IOContext{Base.TTY}, ::MIME{Symbol("text/plain")}, ::Vector{Main.OptimizeMe.Object}) -└─ MethodInstance for var"#sprint#386"(::IOContext{Base.TTY}, ::Int64, ::typeof(sprint), ::Function, ::Main.OptimizeMe.Object) - └─ MethodInstance for sizeof(::Main.OptimizeMe.Object) + ├─ similar(::Base.Broadcast.Broadcasted{Base.Broadcast.DefaultArrayStyle{1}, Tuple{Base.OneTo{Int64}}, Type{Main.OptimizeMe.Container}, Tuple{Base.Broadcast.Extruded{Vector{Any}, Tuple{Bool}, Tuple{Int64}}}}, ::Type{Main.OptimizeMe.Container{Int64}}) + ├─ setindex!(::Vector{Main.OptimizeMe.Container{Int64}}, ::Main.OptimizeMe.Container{Int64}, ::Int64) + ├─ Base.Broadcast.copyto_nonleaf!(::Vector{Main.OptimizeMe.Container{Int64}}, ::Base.Broadcast.Broadcasted{Base.Broadcast.DefaultArrayStyle{1}, Tuple{Base.OneTo{Int64}}, Type{Main.OptimizeMe.Container}, Tuple{Base.Broadcast.Extruded{Vector{Any}, Tuple{Bool}, Tuple{Int64}}}}, ::Base.OneTo{Int64}, ::Int64, ::Int64) + │ ├─ similar(::Base.Broadcast.Broadcasted{Base.Broadcast.DefaultArrayStyle{1}, Tuple{Base.OneTo{Int64}}, Type{Main.OptimizeMe.Container}, Tuple{Base.Broadcast.Extruded{Vector{Any}, Tuple{Bool}, Tuple{Int64}}}}, ::Type{Main.OptimizeMe.Container}) + │ └─ Base.Broadcast.restart_copyto_nonleaf!(::Vector{Main.OptimizeMe.Container}, ::Vector{Main.OptimizeMe.Container{Int64}}, ::Base.Broadcast.Broadcasted ``` -We'll use this as an excuse to point out that if you don't know how to deal with the root node of this (sub)tree, you can tackle later nodes: +In rough terms, what this means is the following: +- since the first item in `list` is an `Int`, the output initially gets created as a `Vector{Container{Int}}` +- however, `copyto_nonleaf!` runs into trouble when it goes to copy the second item, which is a `Container{UInt8}` +- hence, `copyto_nonleaf!` re-allocates the output array to be a generic `Vector{Container}` and then calls `restart_copyto_nonleaf!`. -```julia -julia> itrigsnode = flatten(node) -3-element Vector{InferenceTrigger}: - Inference triggered to call MethodInstance for show(::IOContext{Base.TTY}, ::MIME{Symbol("text/plain")}, ::Vector{Main.OptimizeMe.Object}) from #38 (/home/tim/src/julia-master/usr/share/julia/stdlib/v1.6/REPL/src/REPL.jl:220) with specialization MethodInstance for (::REPL.var"#38#39"{REPL.REPLDisplay{REPL.LineEditREPL}, MIME{Symbol("text/plain")}, Base.RefValue{Any}})(::Any) - Inference triggered to call MethodInstance for var"#sprint#386"(::IOContext{Base.TTY}, ::Int64, ::typeof(sprint), ::Function, ::Main.OptimizeMe.Object) from sprint##kw (./strings/io.jl:101) inlined into MethodInstance for alignment(::IOContext{Base.TTY}, ::Vector{Main.OptimizeMe.Object}, ::UnitRange{Int64}, ::UnitRange{Int64}, ::Int64, ::Int64, ::Int64) (./arrayshow.jl:68) - Inference triggered to call MethodInstance for sizeof(::Main.OptimizeMe.Object) from _show_default (./show.jl:402) with specialization MethodInstance for _show_default(::IOContext{IOBuffer}, ::Any) - -julia> itrig = itrigsnode[end] -Inference triggered to call MethodInstance for sizeof(::Main.OptimizeMe.Object) from _show_default (./show.jl:402) with specialization MethodInstance for _show_default(::IOContext{IOBuffer}, ::Any) -``` - -The stacktrace begins +We can prevent all this hassle with one simple change: rewrite that line as -```julia -julia> stacktrace(itrig) -35-element Vector{Base.StackTraces.StackFrame}: - exit_current_timer at typeinfer.jl:166 [inlined] - typeinf(interp::Core.Compiler.NativeInterpreter, frame::Core.Compiler.InferenceState) at typeinfer.jl:208 - typeinf_ext(interp::Core.Compiler.NativeInterpreter, mi::Core.MethodInstance) at typeinfer.jl:835 - typeinf_ext_toplevel(interp::Core.Compiler.NativeInterpreter, linfo::Core.MethodInstance) at typeinfer.jl:868 - typeinf_ext_toplevel(mi::Core.MethodInstance, world::UInt64) at typeinfer.jl:864 - _show_default(io::IOContext{IOBuffer}, x::Any) at show.jl:402 - show_default at show.jl:395 [inlined] - show(io::IOContext{IOBuffer}, x::Any) at show.jl:390 - sprint(f::Function, args::Main.OptimizeMe.Object; context::IOContext{Base.TTY}, sizehint::Int64) at io.jl:103 -⋮ ``` - -You can see that `sprint` called `show` which called `_show_default`; -`_show_default` clearly needed to call `sizeof`. -The hint, in this case, suggests the impossible: - -``` -julia> suggest(itrig) -./show.jl:402: non-inferrable call, perhaps annotate _show_default(io::IO, x) in Base at show.jl:397 with type MethodInstance for sizeof(::Main.OptimizeMe.Object) -If a noninferrable argument is a type or function, Julia's specialization heuristics may be responsible. -immediate caller(s): -2-element Vector{Base.StackTraces.StackFrame}: - show_default at show.jl:395 [inlined] - show(io::IOContext{IOBuffer}, x::Any) at show.jl:390 -``` - -Because `Base` doesn't know about `OptimizeMe.Object`, you could not add such an annotation, and it wouldn't be correct in the vast majority of cases. - -As the name implies, `_show_default` is the fallback `show` method. -We can fix this by adding our own `show` method - -```julia -Base.show(io::IO, o::Object) = print(io, "Object x: ", o.x) +cs = Container{Any}.(list) ``` -to the module definition. -`Object` is so simple that this is slightly silly, but in more complex cases adding good `show` methods improves usability of your packages tremendously. -(SnoopCompile has many `show` specializations, and without them it would be practically unusable.) +We use `Container{Any}` here because there is no more specific element type--other than an unreasonably-large `Union`--that can hold all the items in `list`. -When you do define a custom `show` method, you own it, so of course it will be precompilable. -So we've circumvented this particular issue. +If you make these edits manually, you'll see that we've gone from dozens of `itrigs` (38 on Julia 1.10, you may get a different number on other Julia versions) down to about a dozen (13 on Julia 1.10). Real progress! -### Creating "warmup" methods +### Replacing hard-to-infer calls with lower-level APIs -Finally, it is time to deal with those long-delayed `show(::IOContext{Base.TTY}, ::MIME{Symbol("text/plain")}, ::T)` triggers and the triggers they inspire. -We have two of them, one for `T = Vector{Main.OptimizeMe.Container{Any}}` and one for `T = Vector{Main.OptimizeMe.Object}`. -Let's look at just the trigger associated with the first: +We note that many of the remaining triggers are somehow related to `show`, for example: ``` -julia> itrig -Inference triggered to call MethodInstance for show(::IOContext{Base.TTY}, ::MIME{Symbol("text/plain")}, ::Vector{Main.OptimizeMeFixed.Container{Any}}) from #38 (/pathto/julia/usr/share/julia/stdlib/v1.6/REPL/src/REPL.jl:220) with specialization MethodInstance for (::REPL.var"#38#39"{REPL.REPLDisplay{REPL.LineEditREPL}, MIME{Symbol("text/plain")}, Base.RefValue{Any}})(::Any) +Inference triggered to call show(::IOContext{Base.TTY}, ::MIME{Symbol("text/plain")}, ::Vector{Main.OptimizeMe.Container{Any}}) from #55 (/cache/build/builder-amdci4-0/julialang/julia-release-1-dot-10/usr/share/julia/stdlib/v1.10/REPL/src/REPL.jl:273) with specialization (::REPL.var"#55#56"{REPL.REPLDisplay{REPL.LineEditREPL}, MIME{Symbol("text/plain")}, Base.RefValue{Any}})(::Any) ``` -In this case we see that the method is `#38`. This is a `gensym`, or generated symbol, indicating that the method was generated during Julia's lowering pass, and might indicate a macro, a `do` block or other anonymous function, the generator for a `@generated` function, etc. +In this case we see that the calling method is `#55`. This is a `gensym`, or generated symbol, indicating that the method was generated during Julia's lowering pass, and might indicate a macro, a `do` block or other anonymous function, the generator for a `@generated` function, etc. -!!! warning - It's particularly worthwhile to improve inferrability for gensym-methods. The number assiged to a gensymmed-method may change as you or other developers modify the package (possibly due to changes at very difference source-code locations), and so any explicit `precompile` directives involving gensyms may not have a long useful life. - - But not all methods with `#` in their name are problematic: methods ending in `##kw` or that look like `##funcname#39` are *keyword* and *body* methods, respectively, for methods that accept keywords. They can be obtained from the main method, and so `precompile` directives for such methods will not be outdated by incidental changes to the package. - -`edit(itrig)` (or equivalently, `edit(node)` where `node` is a child of `itree`) takes us to this method in `Base`: +`edit(itrig)` (or equivalently, `edit(node)` where `node` is a child of `itree`) takes us to this method in `Base`, for which key lines are ```julia function display(d::REPLDisplay, mime::MIME"text/plain", x) x = Ref{Any}(x) with_repl_linfo(d.repl) do io - io = IOContext(io, :limit => true, :module => Main::Module) - get(io, :color, false) && write(io, answer_color(d.repl)) - if isdefined(d.repl, :options) && isdefined(d.repl.options, :iocontext) - # this can override the :limit property set initially - io = foldl(IOContext, d.repl.options.iocontext, init=io) - end + ⋮ show(io, mime, x[]) - println(io) - end - return nothing + ⋮ end ``` @@ -552,130 +199,42 @@ The call to `show` comes from `show(io, mime, x[])`. This implementation uses a clever trick, wrapping `x` in a `Ref{Any}(x)`, to prevent specialization of the method defined by the `do` block on the specific type of `x`. This trick is designed to limit the number of `MethodInstance`s inferred for this `display` method. +A great option is to replace the call to `display` with an explicit -Unfortunately, from the standpoint of precompilation we have something of a conundrum. -It turns out that this trigger corresponds to the first of the big red flames in the flame graph. -`show(::IOContext{Base.TTY}, ::MIME{Symbol("text/plain")}, ::Vector{Main.OptimizeMe.Container{Any}})` is not precompilable because `Base` owns the `show` method for `Vector`; -we might own the element type, but we're leveraging the generic machinery in `Base` and consequently it owns the method. -If these were all packages, you might request its developers to add a `precompile` directive, but that will work only if the package that owns the method knows about the relevant type. -In this situation, Julia's `Base` module doesn't know about `OptimizeMe.Container{Any}`, so we're stuck. - -There are a couple of ways one might go about improving matters. -First, one option is that this should be changed in Julia itself: since the caller, `display`, has gone to some lengths to reduce specialization, it would be worth contemplating whether `show(io::IO, ::MIME"text/plain", X::AbstractArray)` should have a `@nospecialize` around `X`. -Here, we'll pursue a simple "cheat," one that allows us to directly precompile this method. -The trick is to link it, via a chain of backedges, to a method that our package owns: - -```julia -# "Stub" callers for precompilability (we don't use this function for any real work) -function warmup() - mime = MIME("text/plain") - io = Base.stdout::Base.TTY - # Container{Any} - v = [Container{Any}(0)] - show(io, mime, v) - show(IOContext(io), mime, v) - # Object - v = [Object(0)] - show(io, mime, v) - show(IOContext(io), mime, v) - return nothing -end - -precompile(warmup, ()) ``` - -We handled not just `Vector{Container{Any}}` but also `Vector{Object}`, since that turns out to correspond to the other wide block of red bars. -If you make this change, start a fresh session, and recreate the flame graph, you'll see that the wide red flames are gone: - -![flamegraph-OptimizeMeFixed](../assets/flamegraph-OptimizeMeFixed.png) - - -!!! info - It's worth noting that this `warmup` method needed to be carefully written to succeed in its mission. `stdout` is not inferrable (it's a global that can be replaced by `redirect_stdout`), so we needed to annotate its type. We also might have been tempted to use a loop, `for io in (stdout, IOContext(stdout)) ... end`, but inference needs a dedicated call-site where it knows all the types. ([Union-splitting](https://julialang.org/blog/2018/08/union-splitting/) can sometimes come to the rescue, but not if the list is long or elements non-inferrable.) The safest option is to make each call from a separate site in the code. - -The next trigger, a call to `sprint` from inside `Base.alignment(io::IO, x::Any)`, could also be handled using this `warmup` trick, but the flamegraph says this call (also marked in red) isn't an expensive method to infer. In such cases, it's fine to choose to leave it be. - -### Implementing or requesting `precompile` directives in upstream packages - -Of the remaining triggers (now numbering 14), the flamegraph indicates that the most expensive inference run is - -``` -Inference triggered to call MethodInstance for show(::IOContext{IOBuffer}, ::Float32) from _show_default (./show.jl:412) with specialization MethodInstance for _show_default(::IOContext{IOBuffer}, ::Any) +show(stdout, MIME("text/plain"), cs) ``` -You can check that by listing the children of `ROOT` in order of `inclusive` time: +There's one extra detail: the type of `stdout` is not fixed (and therefore not known), because one can use a terminal, a file, `devnull`, etc., as `stdout`. If you want to prevent all runtime dispatch from this call, you'd need to supply an `io::IO` object of known type as the first argument. It could, for example, be passed in to `lotsa_containers` from `main`: -```julia -julia> nodes = sort(tinf.children; by=inclusive) -14-element Vector{SnoopCompileCore.InferenceTimingNode}: - InferenceTimingNode: 0.000053/0.000053 on InferenceFrameInfo for ==(::Type, nothing::Nothing) with 0 direct children - InferenceTimingNode: 0.000054/0.000054 on InferenceFrameInfo for sizeof(::Main.OptimizeMeFixed.Container{Any}) with 0 direct children - InferenceTimingNode: 0.000061/0.000061 on InferenceFrameInfo for Base.typeinfo_eltype(::Type) with 0 direct children - InferenceTimingNode: 0.000075/0.000380 on InferenceFrameInfo for show(::IOContext{IOBuffer}, ::Any) with 1 direct children - InferenceTimingNode: 0.000445/0.000445 on InferenceFrameInfo for Pair{Symbol, DataType}(::Any, ::Any) with 0 direct children - InferenceTimingNode: 0.000663/0.000663 on InferenceFrameInfo for print(::IOContext{Base.TTY}, ::String, ::String, ::Vararg{String, N} where N) with 0 direct children - InferenceTimingNode: 0.000560/0.001049 on InferenceFrameInfo for Base.var"#sprint#386"(::IOContext{Base.TTY}, ::Int64, sprint::typeof(sprint), ::Function, ::Main.OptimizeMeFixed.Object) with 4 direct children - InferenceTimingNode: 0.000441/0.001051 on InferenceFrameInfo for Pair(::Symbol, ::Type) with 1 direct children - InferenceTimingNode: 0.000627/0.001140 on InferenceFrameInfo for Base.var"#sprint#386"(::IOContext{Base.TTY}, ::Int64, sprint::typeof(sprint), ::Function, ::Main.OptimizeMeFixed.Container{Any}) with 4 direct children - InferenceTimingNode: 0.000321/0.001598 on InferenceFrameInfo for show(::IOContext{IOBuffer}, ::UInt16) with 4 direct children - InferenceTimingNode: 0.000190/0.012516 on InferenceFrameInfo for show(::IOContext{IOBuffer}, ::Vector{Int64}) with 3 direct children - InferenceTimingNode: 0.021179/0.033940 on InferenceFrameInfo for Base.Ryu.writeshortest(::Vector{UInt8}, ::Int64, ::Float32, ::Bool, ::Bool, ::Bool, ::Int64, ::UInt8, ::Bool, ::UInt8, ::Bool, ::Bool) with 29 direct children - InferenceTimingNode: 0.000083/0.035496 on InferenceFrameInfo for show(::IOContext{IOBuffer}, ::Tuple{String, Int64}) with 1 direct children - InferenceTimingNode: 0.000188/0.092555 on InferenceFrameInfo for show(::IOContext{IOBuffer}, ::Float32) with 1 direct children ``` - -You can see it's the most expensive remaining root, weighing in at nearly 100ms. -This method is defined in the `Base.Ryu` module, - -```julia -julia> node = nodes[end] -InferenceTimingNode: 0.000188/0.092555 on InferenceFrameInfo for show(::IOContext{IOBuffer}, ::Float32) with 1 direct children - -julia> Method(node) -show(io::IO, x::T) where T<:Union{Float16, Float32, Float64} in Base.Ryu at ryu/Ryu.jl:111 +function lotsa_containers(io::IO) + ⋮ + println(io, "lotsa containers:") + show(io, MIME("text/plain"), cs) +end ``` -Now, we could add this to `warmup` and at least solve the inference problem. -However, on the flamegraph you might note that this is followed shortly by a couple of calls to `Ryu.writeshortest` (the third-most expensive to infer), followed by a long gap. -That hints that other steps, like native code generation, may be expensive. -Since these are base Julia methods, and `Float32` is a common type, it would make sense to file an issue or pull request that Julia should come shipped with these precompiled--that would cache not only the type-inference but also the native code, and thus represents a far more complete solution. - -Later, we'll see how `parcel` can generate such precompile directives automatically, so this is not a step you need to implement entirely on your own. +However, if you want it to go to `stdout`--and to allow users to redirect `stdout` to a target of their choosing--then an `io` argument may have to be of unknown type when called from `main`. -Another `show` `MethodInstance`, `show(::IOContext{IOBuffer}, ::Tuple{String, Int64})`, seems too specific to be worth worrying about, so we call it quits here. +### When you need to rely on `@compile_workload` -### [Advanced analysis: `Cthulhu.ascend`](@id ascend-itrig) - -One thing that hasn't yet been covered is that when you really need more insight, you can use `ascend`: - -```julia -julia> itrig = itrigs[5] -Inference triggered to call MethodInstance for show(::IOContext{IOBuffer}, ::Float32) from _show_default (./show.jl:412) with specialization MethodInstance for _show_default(::IOContext{IOBuffer}, ::Any) - -julia> ascend(itrig) -Choose a call for analysis (q to quit): - > show(::IOContext{IOBuffer}, ::Float32) - _show_default(::IOContext{IOBuffer}, ::Any) at ./show.jl:412 - show_default at ./show.jl:395 => show(::IOContext{IOBuffer}, ::Any) at ./show.jl:390 - #sprint#386(::IOContext{Base.TTY}, ::Int64, ::typeof(sprint), ::Function, ::Main.OptimizeMeFixed.Container{Any}) at ./strings/io.jl:103 - sprint##kw at ./strings/io.jl:101 => alignment at ./show.jl:2528 => alignment(::IOContext{Base.TTY}, ::Vector{Main.OptimizeMeFixed.Container{Any}}, ::UnitRange{Int64}, ::UnitRange{Int64}, :: - print_matrix(::IOContext{Base.TTY}, ::AbstractVecOrMat{T} where T, ::String, ::String, ::String, ::String, ::String, ::String, ::Int64, ::Int64) at ./arrayshow.jl:197 - print_matrix at ./arrayshow.jl:169 => print_array at ./arrayshow.jl:323 => show(::IOContext{Base.TTY}, ::MIME{Symbol("text/plain")}, ::Vector{Main.OptimizeMeFixed.Container{Any}}) at ./a - (::REPL.var"#38#39"{REPL.REPLDisplay{REPL.LineEditREPL}, MIME{Symbol("text/plain")}, Base.RefValue{Any}})(::Any) at /home/tim/src/julia-master/usr/share/julia/stdlib/v1.6/REPL/src/REPL - with_repl_linfo(::Any, ::REPL.LineEditREPL) at /home/tim/src/julia-master/usr/share/julia/stdlib/v1.6/REPL/src/REPL.jl:462 -v display(::REPL.REPLDisplay, ::MIME{Symbol("text/plain")}, ::Any) at /home/tim/src/julia-master/usr/share/julia/stdlib/v1.6/REPL/src/REPL.jl:213 +Most of the remaining triggers are difficult to fix because they occur in deliberately-`@nospecialize`d portions of Julia's internal code for displaying arrays. In such cases, adding a `PrecompileTools.@compile_workload` is your best option. Here we use an interesting trick: +``` +@compile_workload begin + lotsa_containers(devnull) # use `devnull` to suppress output + abmult(rand(-5:5), rand(3)) +end +precompile(lotsa_containers, (Base.TTY,)) ``` -Here, one twist is that some lines contain content like +During the workload, we pass `devnull` as the `io` object to `lotsa_containers`: this suppresses the output so you don't see anything during precompilation. However, `devnull` is not a `Base.TTY`, the standard type of `stdout`. Nevertheless, this is effective because we can see that many of the callees in the remaining inference-triggers do not depend on the `io` object. -``` -show_default at ./show.jl:395 => show(::IOContext{IOBuffer}, ::Any) at ./show.jl:390 -``` +To really ice the cake, we also add a manual `precompile` directive. (`precompile` doesn't execute the method, it just compiles it.) This doesn't "step through" runtime dispatch, but at least it precompiles the entry point. +Thus, at least `lotsa_containers` will be precompiled for the most likely `IO` type encountered in practice. -This indicates that `show_default` was inlined into `show`. -`ascend` needs the full non-inlined `MethodInstance` to descend into, so the tree only includes such nodes. -However, within Cthulhu you can toggle optimization and thereby descend into some of these inlined method, or see the full consequence of their inlining into the caller. +With these changes, we've fixed nearly all the latency problems in `OptimizeMe`, and made it much less vulnerable to invalidation as well. You can see the final code in the [`OptimizeMeFixed` source code](https://github.com/timholy/SnoopCompile.jl/blob/master/examples/OptimizeMeFixed.jl). Note that this would have to be turned into a real package for the `@compile_workload` to have any effect. ## [A note on analyzing test suites](@id test-suites) @@ -704,23 +263,3 @@ julia> length(itrigsel) ``` While there is some risk of discarding triggers that provide clues about the origin of other triggers (e.g., they would have shown up in the same branch of the `trigger_tree`), the shorter list may help direct your attention to the "real" issues. - -## Results from the improvements - -An improved version of `OptimizeMe` can be found in `OptimizeMeFixed.jl` in the same directory. -Let's see where we stand: - -```julia -julia> tinf = @snoop_inference OptimizeMeFixed.main() -3.14 is great -2.718 is jealous -... - Object x: 7 -InferenceTimingNode: 0.888522055/1.496965222 on InferenceFrameInfo for Core.Compiler.Timings.ROOT() with 15 direct children -``` - -We've substantially shrunk the overall inclusive time from 2.68s to about 1.5s. -Some of this came from our single `precompile` directive, for `warmup`. -But even more of it came from limiting specialization (using `Container{Any}` instead of `Container`) and by making some results easier on type-inference (e.g., our changes for the `vcat` pipeline). - -On the next page, we'll wrap all this up with more explicit `precompile` directives. diff --git a/examples/OptimizeMe.jl b/examples/OptimizeMe.jl index ba1c4196..1a7a2504 100644 --- a/examples/OptimizeMe.jl +++ b/examples/OptimizeMe.jl @@ -17,28 +17,21 @@ function lotsa_containers() display(cs) end -concat_string(c1::Container, c2::Container) = string(c1.value) * ' ' * string(c2.value) - -function contain_concrete(item1, item2) - c1 = Container(item1) - c2 = Container(item2) - return concat_string(c1, c2) -end - -function contain_list(list) - cs = Container.(list) - return concat_string(cs...) -end - -struct Object - x::Int +howbig(str::AbstractString) = length(str) +howbig(x::Char) = 1 +howbig(x::Unsigned) = x +howbig(x::Real) = abs(x) + +function abmult(r::Int, ys) + if r < 0 + r = -r + end + return map(x -> howbig(r * x), ys) end function main() lotsa_containers() - println(contain_concrete(3.14, "is great")) - list = [2.718, "is jealous"] - println(contain_list(list)) + return abmult(rand(-5:5), rand(3)) end end diff --git a/examples/OptimizeMeFixed.jl b/examples/OptimizeMeFixed.jl index 9e5841ba..af0e76ad 100644 --- a/examples/OptimizeMeFixed.jl +++ b/examples/OptimizeMeFixed.jl @@ -3,49 +3,47 @@ OptimizeMeFixed is the "improved" version of OptimizeMe. See the file in this sa """ module OptimizeMeFixed +using PrecompileTools + struct Container{T} value::T end +Base.show(io::IO, c::Container) = print(io, "Container(", c.value, ")") -function lotsa_containers() - list = Any[1, 0x01, 0xffff, 2.0f0, 'a', [0], ("key", 42)] +function lotsa_containers(io::IO) + list = [1, 0x01, 0xffff, 2.0f0, 'a', [0], ("key", 42)] cs = Container{Any}.(list) - println("lotsa containers:") - display(cs) + println(io, "lotsa containers:") + show(io, MIME("text/plain"), cs) end -concat_string(c1::Container, c2::Container) = string(c1.value) * ' ' * string(c2.value) +howbig(str::AbstractString) = length(str) +howbig(x::Char) = 1 +howbig(x::Unsigned) = x +howbig(x::Real) = abs(x) -function contain_concrete(item1, item2) - c1 = Container(item1) - c2 = Container(item2) - return concat_string(c1, c2) +function abmult(r::Int, ys) + if r < 0 + r = -r + end + let r = r # Julia #15276 + return map(x -> howbig(r * x), ys) + end end -function contain_list(list) - length(list) == 2 || throw(DimensionMismatch("list must have length 2")) - item1 = convert(Float64, list[1])::Float64 - item2 = list[2]::String - return contain_concrete(item1, item2) +function main() + lotsa_containers(stdout) + return abmult(rand(-5:5), rand(3)) end -struct Object - x::Int -end -Base.show(io::IO, o::Object) = print(io, "Object x: ", o.x) -function makeobjects() - xs = [1:5; 7:7] - return Object.(xs) +@compile_workload begin + lotsa_containers(devnull) # use `devnull` to suppress output + abmult(rand(-5:5), rand(3)) end - -function main() - lotsa_containers() - println(contain_concrete(3.14, "is great")) - list = [2.718, "is jealous"] - println(contain_list(list)) - display(makeobjects()) -end - +# since `devnull` is not a `Base.TTY`--the standard type of `stdout`--let's also +# use an explicit `precompile` directive. (Note this does not trigger any visible output). +# This doesn't "follow" runtime dispatch but at least it precompiles the entry point. +precompile(lotsa_containers, (Base.TTY,)) end diff --git a/src/parcel_snoop_inference.jl b/src/parcel_snoop_inference.jl index 56a22861..6862ef4d 100644 --- a/src/parcel_snoop_inference.jl +++ b/src/parcel_snoop_inference.jl @@ -1053,7 +1053,7 @@ function Location(itrig::InferenceTrigger) end Base.show(io::IO, loc::Location) = print(io, loc.func, " at ", loc.file, ':', loc.line) -InteractiveUtils.edit(loc::Location) = edit(string(loc.file), loc.line) +InteractiveUtils.edit(loc::Location) = edit(Base.fixup_stdlib_path(string(loc.file)), loc.line) const LocationTriggers = TaggedTriggers{Location} @@ -1169,6 +1169,8 @@ function Base.show(io::IO, s::Suggested) show_suggest(io, s.categories, rtcallee, sf) end +Base.haskey(s::Suggested, k::Suggestion) = k in s.categories + function show_suggest(io::IO, categories, rtcallee, sf) showcaller = true showvahint = showannotate = false @@ -1660,7 +1662,14 @@ function suggest!(stree, node) return stree end -Base.show(io::IO, node::SuggestNode) = print_tree(io, node) +function Base.show(io::IO, node::SuggestNode) + if node.s === nothing + print(io, "no inference trigger") + else + show(io, node.s) + end + print(" (", string(length(node.children)), " children)") +end function strip_prefix(io::IO, obj, prefix) print(io, obj) From 354f18df1f9a2d2884694f6fb23874a86dd2e659 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Wed, 24 Jul 2024 08:43:40 -0500 Subject: [PATCH 18/18] More OptimizeMe updates; delete broken refs --- docs/src/tutorials/invalidations.md | 2 +- docs/src/tutorials/pgdsgui.md | 2 +- .../src/tutorials/snoop_inference_analysis.md | 5 +- docs/src/tutorials/snoop_inference_parcel.md | 115 +++--------------- 4 files changed, 24 insertions(+), 100 deletions(-) diff --git a/docs/src/tutorials/invalidations.md b/docs/src/tutorials/invalidations.md index 99925f4b..8d4d1f9d 100644 --- a/docs/src/tutorials/invalidations.md +++ b/docs/src/tutorials/invalidations.md @@ -208,7 +208,7 @@ There are quite a few other tools for working with `invs` and `trees`, see the [ `tallyscores` and `playgame` were compiled in `Blackjack`, a "world" where the `score` method defined in `BlackjackFacecards` does not yet exist. When you load the `BlackjackFacecards` package, Julia must ask itself: now that this new `score` method exists, am I certain that I would compile `tallyscores` the same way? If the answer is "no," Julia invalidates the old compiled code, and compiles a fresh version with full awareness of the new `score` method in `BlackjackFacecards`. Why would the compilation of `tallyscores` change? Evidently, `cards` is a `Vector{Any}`, and this means that `tallyscores` can't guess what kind of object `card` might be, and thus it can't guess what kind of objects are passed into `score`. The crux of the invalidation is thus: -- when `Blackjack` is compiled, inference does not know which `score` method will be called. However, at the time of compilation the only `score` method is for `Int`. Thus Julia will reason that anything that isn't an `Int` is going to trigger an error anyway, and so you might as well optimize `tallyscore` expecting all cards to be `Int`s. (More information about how `tallyscores` gets optimized can be found in [World-splitting](@ref).) +- when `Blackjack` is compiled, inference does not know which `score` method will be called. However, at the time of compilation the only `score` method is for `Int`. Thus Julia will reason that anything that isn't an `Int` is going to trigger an error anyway, and so you might as well optimize `tallyscore` expecting all cards to be `Int`s. - however, when `BlackjackFacecards` is loaded, suddenly there are two `score` methods supporting both `Int` and `Char`. Now Julia's guess that all `cards` will probably be `Int`s doesn't seem so likely to be true, and thus `tallyscores` should be recompiled. Thus, invalidations arise from optimization based on what methods and types are "in the world" at the time of compilation (sometimes called *world-splitting*). This form of optimization can have performance benefits, but it also leaves your code vulnerable to invalidation. diff --git a/docs/src/tutorials/pgdsgui.md b/docs/src/tutorials/pgdsgui.md index 2f4c3202..17d0dc61 100644 --- a/docs/src/tutorials/pgdsgui.md +++ b/docs/src/tutorials/pgdsgui.md @@ -243,7 +243,7 @@ The "standardizing method" `foo(x, y)` is short and therefore quick to compile, Without it, `foo(x, y)` might call itself in an infinite loop, ultimately triggering a StackOverflowError. StackOverflowErrors are a particularly nasty form of error, and the typeassert ensures that you get a simple `TypeError` instead. - In other contexts, such typeasserts would also have the effect of fixing inference problems even if the type of `x` is not well-inferred (this will be discussed in more detail [later](@ref typeasserts)), but in this case dispatch to `foo(x::X, y::Y)` would have ensured the same outcome. + In other contexts, such typeasserts would also have the effect of fixing inference problems even if the type of `x` is not well-inferred, but in this case dispatch to `foo(x::X, y::Y)` would have ensured the same outcome. There are of course cases where you can't implement your code in this way: after all, part of the power of Julia is the ability of generic methods to "do the right thing" for a wide variety of types. But in cases where you're doing a standard task, e.g., writing some data to a file, there's really no good reason to recompile your `save` method for a filename encoded as a `String` and again for a `SubString{String}` and again for a `SubstitutionString` and again for an `AbstractString` and ...: after all, the core of the `save` method probably isn't sensitive to the precise encoding of the filename. In such cases, it should be safe to convert all filenames to `String`, thereby reducing the diversity of input arguments for expensive-to-compile methods. diff --git a/docs/src/tutorials/snoop_inference_analysis.md b/docs/src/tutorials/snoop_inference_analysis.md index e9b66027..34dee1cb 100644 --- a/docs/src/tutorials/snoop_inference_analysis.md +++ b/docs/src/tutorials/snoop_inference_analysis.md @@ -7,8 +7,7 @@ Throughout this page, we'll use the `OptimizeMe` demo, which ships with `SnoopCo ```@repl fix-inference using SnoopCompileCore, SnoopCompile # here we need the SnoopCompile path for the next line (normally you should wait until after data collection is complete) -cd(joinpath(pkgdir(SnoopCompile), "examples")) -include("OptimizeMe.jl") +include(joinpath(pkgdir(SnoopCompile), "examples", "OptimizeMe.jl")) tinf = @snoop_inference OptimizeMe.main(); fg = flamegraph(tinf) ``` @@ -79,7 +78,7 @@ You can still "dig deep" into individual triggers: itrig = mtrig.itrigs[1] ``` -This is useful if you want to analyze with [`Cthulhu.ascend`](@ref ascend-itrig). +This is useful if you want to analyze with `Cthulhu.ascend`. `Method`-based triggers, which may aggregate many different individual triggers, can be useful because tools like [Cthulhu.jl](https://github.com/JuliaDebug/Cthulhu.jl) show you the inference results for the entire `MethodInstance`, allowing you to fix many different inference problems at once. ### Trigger trees diff --git a/docs/src/tutorials/snoop_inference_parcel.md b/docs/src/tutorials/snoop_inference_parcel.md index 744e9a01..f072cc4c 100644 --- a/docs/src/tutorials/snoop_inference_parcel.md +++ b/docs/src/tutorials/snoop_inference_parcel.md @@ -13,45 +13,33 @@ In such cases, one alternative is to create a manual list of precompile directiv `precompile` directives have to be emitted by the module that owns the method and/or types. SnoopCompile comes with a tool, `parcel`, that splits out the "root-most" precompilable MethodInstances into their constituent modules. This will typically correspond to the bottom row of boxes in the [flame graph](@ref flamegraph). -In cases where you have some non-precompilable MethodInstances, they will include MethodInstances from higher up in the call tree. +In cases where you have some that are not naively precompilable, they will include MethodInstances from higher up in the call tree. -Let's use `SnoopCompile.parcel` on [`OptimizeMeFixed`](@ref inferrability): +Let's use `SnoopCompile.parcel` on our [`OptimizeMe`](@ref inferrability) demo: -```julia -julia> ttot, pcs = SnoopCompile.parcel(tinf); - -julia> ttot -0.6084431670000001 - -julia> pcs -4-element Vector{Pair{Module, Tuple{Float64, Vector{Tuple{Float64, Core.MethodInstance}}}}}: - Core => (0.000135179, [(0.000135179, MethodInstance for (NamedTuple{(:sizehint,), T} where T<:Tuple)(::Tuple{Int64}))]) - Base => (0.028383533000000002, [(3.2456e-5, MethodInstance for getproperty(::IOBuffer, ::Symbol)), (4.7474e-5, MethodInstance for ==(::Type, ::Nothing)), (5.7944e-5, MethodInstance for typeinfo_eltype(::Type)), (0.00039092299999999994, MethodInstance for show(::IOContext{IOBuffer}, ::Any)), (0.000433143, MethodInstance for IOContext(::IOBuffer, ::IOContext{Base.TTY})), (0.000484984, MethodInstance for Pair{Symbol, DataType}(::Any, ::Any)), (0.000742383, MethodInstance for print(::IOContext{Base.TTY}, ::String, ::String, ::Vararg{String, N} where N)), (0.001293705, MethodInstance for Pair(::Symbol, ::Type)), (0.0018914350000000003, MethodInstance for show(::IOContext{IOBuffer}, ::UInt16)), (0.010604793000000001, MethodInstance for show(::IOContext{IOBuffer}, ::Tuple{String, Int64})), (0.012404293, MethodInstance for show(::IOContext{IOBuffer}, ::Vector{Int64}))]) - Base.Ryu => (0.15733664599999997, [(0.05721630600000001, MethodInstance for writeshortest(::Vector{UInt8}, ::Int64, ::Float32, ::Bool, ::Bool, ::Bool, ::Int64, ::UInt8, ::Bool, ::UInt8, ::Bool, ::Bool)), (0.10012033999999997, MethodInstance for show(::IOContext{IOBuffer}, ::Float32))]) - Main.OptimizeMeFixed => (0.4204474180000001, [(0.4204474180000001, MethodInstance for main())]) +```@repl parcel-inference +using SnoopCompileCore, SnoopCompile # here we need the SnoopCompile path for the next line (normally you should wait until after data collection is complete) +include(joinpath(pkgdir(SnoopCompile), "examples", "OptimizeMe.jl")) +tinf = @snoop_inference OptimizeMe.main(); +ttot, pcs = SnoopCompile.parcel(tinf); +ttot +pcs ``` -This tells us that a total of ~0.6s were spent on inference. -`parcel` discovered precompilable MethodInstances for four modules, `Core`, `Base`, `Base.Ryu`, and `OptimizeMeFixed`. +`ttot` shows the total amount of time spent on type-inference. +`parcel` discovered precompilable MethodInstances for four modules, `Core`, `Base.Multimedia`, `Base`, and `OptimizeMe` that might benefit from precompile directives. These are listed in increasing order of inference time. Let's look specifically at `OptimizeMeFixed`, since that's under our control: -```julia -julia> pcmod = pcs[end] -Main.OptimizeMeFixed => (0.4204474180000001, Tuple{Float64, Core.MethodInstance}[(0.4204474180000001, MethodInstance for main())]) - -julia> tmod, tpcs = pcmod.second; - -julia> tmod -0.4204474180000001 - -julia> tpcs -1-element Vector{Tuple{Float64, Core.MethodInstance}}: - (0.4204474180000001, MethodInstance for main()) +```@repl parcel-inference +pcmod = pcs[end] +tmod, tpcs = pcmod.second; +tmod +tpcs ``` -0.42s of that time is due to `OptimizeMeFixed`, and `parcel` discovered a single MethodInstances to precompile, `main()`. +This indicates the amount of time spent specifically on `OptimizeMe`, plus the list of calls that could be precompiled in that module. We could look at the other modules (packages) similarly. @@ -59,15 +47,11 @@ We could look at the other modules (packages) similarly. You can generate files that contain ready-to-use `precompile` directives using `SnoopCompile.write`: -```julia -julia> SnoopCompile.write("/tmp/precompiles_OptimizeMe", pcs) -Core: no precompile statements out of 0.000135179 -Base: precompiled 0.026194226 out of 0.028383533000000002 -Base.Ryu: precompiled 0.15733664599999997 out of 0.15733664599999997 -Main.OptimizeMeFixed: precompiled 0.4204474180000001 out of 0.4204474180000001 +```@repl parcel-inference +SnoopCompile.write("/tmp/precompiles_OptimizeMe", pcs) ``` -You'll now find a directory `/tmp/precompiles_OptimizeMe`, and inside you'll find three files, for `Base`, `Base.Ryu`, and `OptimizeMeFixed`, respectively. +You'll now find a directory `/tmp/precompiles_OptimizeMe`, and inside you'll find files for modules that could have precompile directives added manually. The contents of the last of these should be recognizable: ```julia @@ -81,64 +65,5 @@ The first `ccall` line ensures we only pay the cost of running these `precompile (It would also matter if you've set `__precompile__(false)` at the top of your module, but if so why are you reading this?) This file is ready to be moved into the `OptimizeMe` repository and `include`d into your module definition. -Since we added `warmup` manually, you could consider moving `precompile(warmup, ())` into this function. You might also consider submitting some of the other files (or their `precompile` directives) to the packages you depend on. -In some cases, the specific argument type combinations may be too "niche" to be worth specializing; one such case is found here, a `show` method for `Tuple{String, Int64}` for `Base`. -But in other cases, these may be very worthy additions to the package. - -## Final results - -Let's check out the final results of adding these `precompile` directives to `OptimizeMeFixed`. -First, let's build both modules as precompiled packages: - -```julia -ulia> push!(LOAD_PATH, ".") -4-element Vector{String}: - "@" - "@v#.#" - "@stdlib" - "." - -julia> using OptimizeMe -[ Info: Precompiling OptimizeMe [top-level] - -julia> using OptimizeMeFixed -[ Info: Precompiling OptimizeMeFixed [top-level] -``` - -Now in fresh sessions, - -```julia -julia> @time (using OptimizeMe; OptimizeMe.main()) -3.14 is great -2.718 is jealous -⋮ -Object x: 7 - 3.159908 seconds (10.63 M allocations: 582.091 MiB, 5.19% gc time, 99.67% compilation time) -``` - -versus - -```julia -julia> @time (using OptimizeMeFixed; OptimizeMeFixed.main()) -3.14 is great -2.718 is jealous -⋮ - Object x: 7 - 1.840034 seconds (5.38 M allocations: 289.402 MiB, 5.03% gc time, 96.70% compilation time) -``` - -We've cut down on the latency by nearly a factor of two. -Moreover, if Julia someday caches generated code, we're well-prepared to capitalize on the benefits, because the same improvements in "code ownership" are almost certain to pay dividends there too. - -If you inspect the results, you may sometimes suffer a few disappointments: some methods that we expected to precompile don't "take." -At the moment there appears to be a small subset of methods that fail to precompile, and the reasons are not yet widely understood. -At present, the best advice seems to be to comment-out any precompile directives that don't "take," since otherwise they increase the build time for the package without material benefit. -These failures may be addressed in future versions of Julia. -It's also worth appreciating how much we have succeeded in reducing latency, with the awareness that we may be able to get even greater benefit in the future. - -## Summary - -`@snoop_inference` collects enough data to learn which methods are triggering inference, how heavily methods are being specialized, and so on. -Examining your code from the standpoint of inference and specialization may be unfamiliar at first, but like other aspects of package development (testing, documentation, and release compatibility management) it can lead to significant improvements in the quality-of-life for you and your users.