Let's see why this is with a counter-example : MSBuild.
MSBuild works by running a list of Targets in dependency order. Each Target executes one or more tasks in sequence. So overall, all Task executions are completely serialized within a single project's execution. A fancy C++ project sequence will go in phases like this:
- Generate source file gen1.cpp from gen1.pl
- Generate source file gen2.cpp from gen2.py
- Compile all C++ sources. (gen1.cpp, gen2.cpp, a.cpp, b.cpp, c.cpp)
- Link object files.
- Copy linker output to final destination.
According to MSBuild's programming model, each step must fully complete before the next step may begin. In this case, the opportunity to parallelize the code-generation from #1 and #2 are lost. Additionally, #1 and #2 could be parallelized with the compilation of a.cpp, b.cpp, and c.cpp; this opportunity is lost as well.
With max parallelization of 3 set on the CL task, you might see this play out:
Time | 0 | 1 | 2 | 3 | 4 |
Proc #2 | . | . | c.cpp | . | . |
Proc #1 | . | . | b.cpp | gen2.cpp | . |
Proc #0 | gen1.pl | gen2.py | a.cpp | gen1.cpp | prog.exe |
A build system that supports file-level dependencies overcomes this kind of wasteful serialization. The code-generation can occur while independent compilation occurs. For example, a Makefile can achieve the following:
Time | 0 | 1 | 2 | 3 | 4 |
Proc #2 | a.cpp | gen1.cpp | . | . | . |
Proc #1 | gen2.py | c.cpp | . | . | . |
Proc #0 | gen1.pl | b.cpp | gen2.cpp | prog.exe | . |
Of course, both of these examples are over-simplified -- build steps are quantized into equal chunks of time. In a real build, variable times for each build step will cause things to overlap in the most efficient way possible, resulting in optimal behavior under varying conditions.
No comments:
Post a Comment