Every programming language aims to be performant in its niche and achieving superior performance requires a lot of compiler level optimizations. One famous optimization technique is Constant Folding where during compile time the engine tries to recognize constant expressions, evaluate them, and replaces the expression with this newly evaluated value, making the runtime leaner.
In this essay, we dive deep and find what exactly is Constant Folding, understand the scope of it in the world of Python and finally go through Python's source code - CPython - and find out how elegantly Python actually implements it.
Constant Folding
In Constant Folding, the engine finds and evaluates constant expressions at compile time rather than computing them at runtime, making the runtime leaner and faster.
When the compiler encounters a constant expression, like above, it evaluates the expression and replaces it with the evaluated value. The expression is usually replaced by the evaluated value in the Abstract Syntax Tree, but the implementation is totally up to the language. Hence the above expression is effectively executed as
Constant Folding in Python
In Python, we could use the Disassembler module to get the CPython bytecode giving us a good peek at how things will be executed. When we disassemble the above constant expression using the dis
module, we get the following bytecode
We see that the bytecode, instead of having two binary multiply operations followed by one LOAD_CONST
, is having just one LOAD_CONST
with the already evaluated value of 86400
. This indicates that the CPython interpreter during parsing and building of Abstract Syntax Tree folded the constant expression, 24 * 60 * 60
and replaced it with the evaluated value 86400
.
Scope of Constant Folding
Python tries to fold every single constant expression present but there are some cases where even though the expression is constant, but Python chooses not to fold it. For example, Python does not fold x = 4 ** 64
while it does fold x = 2 ** 64
.
Apart from the arithmetic expressions, Python also folds expressions involving Strings and Tuples, where constant string expressions till the length 4096
are folded.
Internals of Constant Folding
Now we shift our focus to the internals and find exactly where and how CPython implements Constant Folding. All AST optimizations, including Constant Folding, can be found in file ast_opt.c. The base function starting it all is astfold_expr
which folds any and every expression that Python source has. The function recursively goes through the AST and tries to fold every constant expression, as seen in the snippet below.
The astfold_expr
before folding the expression at hand, tries to fold its child expressions (operands) and then delegates the folding to the corresponding specific expression folding function. The operation-specific folding function evaluates the expression and returns the evaluated constant value, which is then put into the AST.
For example, whenever astfold_expr
encounters a binary operation, it recursively folds the two child operands (expressions) before evaluating the expression at hand using fold_binop
. The function fold_binop
returns the evaluated constant value as seen in the snippet below.
fold_binop
function folds the binary operation by checking the kind of operator at hand and then invoking the corresponding evaluation function on them. For example, if the operation at hand is an addition then, to evaluate the final value, it invokes PyNumber_Add
on both its left and right operands.
What makes this elegant?
Instead of writing special logic to handle certain patterns or types to fold constant expressions efficiently, CPython invokes the same general code. For example, it invokes the same usual PyNumber_Add
function while folding that it does to perform the usual addition operation.
CPython has thus eradicated the need to write special functions to handle constant folding by making sure its code and evaluation process is structured in such a way that the general-purpose code itself can handle the evaluation of constant expressions.
References
Other essays you might like
If you liked what you read, consider subscribing to my weekly newsletter at arpitbhayani.me/newsletter where, once a week, I write an essay about programming languages internals, or a deep dive on some super-clever algorithm, or just a few tips on building highly scalable distributed systems.
You can always find me browsing through Twitter @arpit_bhayani.
If my work adds value, consider supporting me
A subtle point: because constant folding is applied recursively on the already-generated AST, its ability to fold is impacted by how that tree was constructed by the parser. For example, it cannot reorder commutative operations:
```
dis.dis("foo = bar * 2 * foo * 3")
1 0 LOAD_NAME 0 (bar)
3 LOAD_CONST 0 (2)
6 BINARY_MULTIPLY
7 LOAD_NAME 1 (foo)
10 BINARY_MULTIPLY
11 LOAD_CONST 1 (3)
14 BINARY_MULTIPLY
15 STORE_NAME 1 (foo)
18 LOAD_CONST 2 (None)
21 RETURN_VALUE
```
If you manually reorder the expression so the constants appear next to each other, the parser will drop them in the AST as children of the same binop node. In that case, constant folding will work.
```
dis.dis("foo = 2 * 3 * bar * foo")
1 0 LOAD_CONST 3 (6)
3 LOAD_NAME 0 (bar)
6 BINARY_MULTIPLY
7 LOAD_NAME 1 (foo)
10 BINARY_MULTIPLY
11 STORE_NAME 1 (foo)
14 LOAD_CONST 2 (None)
17 RETURN_VALUE
```