Resources

Concepts

C Compiler Series by Nora Sandler

Series of posts on Writing a C Compiler by Nora Sandler

Compiler Optimizations

Compiler optimisations aim to improve the performance of generated code by transforming the source code or intermediate representations. Here are some common compiler optimizations:

  1. Constant Folding: Evaluating constant expressions at compile-time instead of runtime.
  2. Dead Code Elimination: Removing code that is never executed or has no effect on the program’s output.
  3. Common Subexpression Elimination: Identifying and eliminating redundant computations.
  4. Loop Optimization:
    • Loop-Invariant Code Motion: Moving computations outside loops if they do not depend on loop variables.
    • Loop Unrolling: Duplicating loop bodies to reduce loop overhead and increase instruction-level parallelism.
    • Loop Fusion: Combining consecutive loops to reduce memory accesses and improve cache utilization.
    • Loop Interchange: Reordering nested loops to improve spatial locality.
    • Loop Vectorization: Transforming scalar loops into SIMD (Single Instruction, Multiple Data) instructions for parallel execution on vector units.
  5. Data Flow Analysis:
    • Constant Propagation: Replacing variables with their known constant values.
    • Copy Propagation: Replacing variables with their assigned values.
    • Common Subexpression Elimination: Identifying and eliminating redundant computations.
    • Dead Code Elimination: Removing unreachable code.
    • Register Allocation: Assigning variables to processor registers for faster access.
  6. Inline Expansion: Replacing function calls with the actual function code to reduce function call overhead.
  7. Strength Reduction: Replacing expensive operations with cheaper ones (e.g., replacing multiplication with addition or shift operations).
  8. Control Flow Optimization:
    • Branch Prediction: Rearranging code to improve the accuracy of branch prediction.
    • Tail Recursion Elimination: Replacing tail-recursive function calls with loops to reduce stack usage.
  9. Loop Parallelization: Transforming loops to execute in parallel on multiple processors or cores.
  10. Function Inlining: Inlining the code of small functions directly into the calling code to avoid function call overhead.
  11. Interprocedural Optimization: Optimizing across multiple functions or translation units.
  12. Memory Optimization:
    • Loop Blocking: Partitioning arrays to fit into cache and exploit spatial locality.
    • Loop Tiling: Dividing loops into smaller blocks to improve cache utilization.
    • Memory Alias Analysis: Analyzing memory references to determine if they overlap.
    • Memory Copy Propagation: Replacing redundant memory copies with direct memory assignments.

These optimizations represent a subset of the many techniques employed by modern compilers to improve code performance. The exact set of optimizations may vary depending on the compiler and its configuration.