Files
TinyTorch/modules/source/16_quantization/quantization_dev.py
Vijay Janapa Reddi 1c299cddb0 docs: add comprehensive docstrings to optimization modules 16-19
- Add Args/Returns/Example/Hints to key functions
- Improve documentation for compare_model_sizes (16)
- Enhance function documentation in compression (17)
- Add docstring details for acceleration (18)
- Improve benchmarking function docs (19)
2025-11-09 14:38:44 -05:00

1880 lines
82 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
# ---
# jupyter:
# jupytext:
# text_representation:
# extension: .py
# format_name: percent
# format_version: '1.3'
# jupytext_version: 1.17.1
# kernelspec:
# display_name: Python 3 (ipykernel)
# language: python
# name: python3
# ---
#| default_exp optimization.quantization
# %% [markdown]
"""
# Module 16: Quantization - Reduced Precision for Efficiency
Welcome to Quantization! Today you'll learn how to reduce model precision from FP32 to INT8 while preserving accuracy.
## 🔗 Prerequisites & Progress
**You've Built**: Complete ML pipeline with profiling (Module 14) and memoization (Module 15)
**You'll Build**: INT8 quantization system with calibration and memory savings
**You'll Enable**: 4× memory reduction and 2-4× speedup with minimal accuracy loss
**Connection Map**:
```
Profiling (14) → Memoization (15) → Quantization (16) → Compression (17)
(measure memory) (reduce compute) (reduce precision) (reduce parameters)
```
## Learning Objectives
By the end of this module, you will:
1. Implement INT8 quantization with proper scaling
2. Build quantization-aware training for minimal accuracy loss
3. Apply post-training quantization to existing models
4. Measure actual memory and compute savings
5. Understand quantization error and mitigation strategies
Let's make models 4× smaller!
"""
# %% [markdown]
"""
## 📦 Where This Code Lives in the Final Package
**Learning Side:** You work in `modules/16_quantization/quantization_dev.py`
**Building Side:** Code exports to `tinytorch.optimization.quantization`
```python
# How to use this module:
from tinytorch.optimization.quantization import quantize_int8, QuantizedLinear, quantize_model
```
**Why this matters:**
- **Learning:** Complete quantization system in one focused module for deep understanding
- **Production:** Proper organization like PyTorch's torch.quantization with all optimization components together
- **Consistency:** All quantization operations and calibration tools in optimization.quantization
- **Integration:** Works seamlessly with existing models for complete optimization pipeline
"""
# %% nbgrader={"grade": false, "grade_id": "imports", "solution": true}
#| export
import numpy as np
import time
from typing import Tuple, Dict, List, Optional
import warnings
# Import dependencies from other modules
from tinytorch.core.tensor import Tensor
from tinytorch.core.layers import Linear
from tinytorch.core.activations import ReLU
print("✅ Quantization module imports complete")
# %% [markdown]
"""
## 🔬 Motivation: Why Quantization Matters
Before we learn quantization, let's profile a model to see how much memory
FP32 weights actually consume. This will show us why reduced precision matters.
"""
# %%
# Profile model memory usage to discover the problem
from tinytorch.profiling.profiler import Profiler
profiler = Profiler()
# Create models of increasing size
print("🔬 Profiling Memory Usage (FP32 Precision):\n")
print(" Parameters | FP32 Memory | Device Fit?")
print(" -------------|---------------|---------------")
model_configs = [
(256, 256, "Tiny"),
(512, 512, "Small"),
(1024, 1024, "Medium"),
(2048, 2048, "Large"),
]
for in_feat, out_feat, name in model_configs:
model = Linear(in_feat, out_feat)
input_data = Tensor(np.random.randn(1, in_feat))
# Profile the model
profile = profiler.profile_forward_pass(model, input_data)
params = profile['parameters']
memory_fp32_mb = params * 4 / 1e6 # 4 bytes per FP32 parameter
memory_fp32_gb = memory_fp32_mb / 1000
# Check if it fits on different devices
fits_mobile = "" if memory_fp32_mb < 100 else ""
fits_edge = "" if memory_fp32_mb < 10 else ""
print(f" {params:>10,} | {memory_fp32_mb:7.1f} MB | Mobile:{fits_mobile} Edge:{fits_edge}")
print("\n💡 Key Observations:")
print(" • Every parameter uses 4 bytes (32 bits) in FP32")
print(" • Larger models quickly exceed mobile device memory (~100MB limit)")
print(" • Edge devices have even tighter constraints (~10MB)")
print(" • Memory grows linearly with parameter count")
print("\n🎯 The Problem:")
print(" Do we really need 32-bit precision for inference?")
print(" • FP32: Can represent 2^32 ≈ 4.3 billion unique values")
print(" • Neural networks are naturally robust to noise")
print(" • Most weights are in range [-3, 3] after training")
print("\n✨ The Solution:")
print(" Quantize to INT8 (8-bit integers):")
print(" • FP32 → INT8: 32 bits → 8 bits (4× compression!)")
print(" • Memory: 100MB → 25MB (now fits on mobile!)")
print(" • Speed: INT8 operations are 2-4× faster on hardware")
print(" • Accuracy: Minimal loss (<1% typically) with proper calibration\n")
# %% [markdown]
"""
## 1. Introduction - The Memory Wall Problem
Imagine trying to fit a library in your backpack. Neural networks face the same challenge - models are getting huge, but devices have limited memory!
### The Precision Paradox
Modern neural networks use 32-bit floating point numbers with incredible precision:
```
FP32 Number: 3.14159265359...
^^^^^^^^^^^^^^^^
32 bits = 4 bytes per weight
```
But here's the surprising truth: **we don't need all that precision for most AI tasks!**
### The Growing Memory Crisis
```
Model Memory Requirements (FP32):
┌─────────────────────────────────────────────────────────────┐
│ BERT-Base: 110M params × 4 bytes = 440MB │
│ GPT-2: 1.5B params × 4 bytes = 6GB │
│ GPT-3: 175B params × 4 bytes = 700GB │
│ Your Phone: Available RAM = 4-8GB │
└─────────────────────────────────────────────────────────────┘
Problem!
```
### The Quantization Solution
What if we could represent each weight with just 8 bits instead of 32?
```
Before Quantization (FP32):
┌──────────────────────────────────┐
│ 3.14159265 │ 2.71828183 │ │ 32 bits each
└──────────────────────────────────┘
After Quantization (INT8):
┌────────┬────────┬────────┬────────┐
│ 98 │ 85 │ 72 │ 45 │ 8 bits each
└────────┴────────┴────────┴────────┘
4× less memory!
```
### Real-World Impact You'll Achieve
**Memory Reduction:**
- BERT-Base: 440MB → 110MB (4× smaller)
- Fits on mobile devices!
- Faster loading from disk
- More models in GPU memory
**Speed Improvements:**
- 2-4× faster inference (hardware dependent)
- Lower power consumption
- Better user experience
**Accuracy Preservation:**
- <1% accuracy loss with proper techniques
- Sometimes even improves generalization!
**Why This Matters:**
- **Mobile AI:** Deploy powerful models on phones
- **Edge Computing:** Run AI without cloud connectivity
- **Data Centers:** Serve more users with same hardware
- **Environmental:** Reduce energy consumption by 2-4×
Today you'll build the production-quality quantization system that makes all this possible!
"""
# %% [markdown]
"""
## 2. Foundations - The Mathematics of Compression
### Understanding the Core Challenge
Think of quantization like converting a smooth analog signal to digital steps. We need to map infinite precision (FP32) to just 256 possible values (INT8).
### The Quantization Mapping
```
The Fundamental Problem:
FP32 Numbers (Continuous): INT8 Numbers (Discrete):
∞ possible values → 256 possible values
... -1.7 -1.2 -0.3 0.0 0.8 1.5 2.1 ...
↓ ↓ ↓ ↓ ↓ ↓ ↓
-128 -95 -38 0 25 48 67 127
```
### The Magic Formula
Every quantization system uses this fundamental relationship:
```
Quantization (FP32 → INT8):
┌─────────────────────────────────────────────────────────┐
│ quantized = round((float_value - zero_point) / scale) │
└─────────────────────────────────────────────────────────┘
Dequantization (INT8 → FP32):
┌─────────────────────────────────────────────────────────┐
│ float_value = scale × quantized + zero_point │
└─────────────────────────────────────────────────────────┘
```
### The Two Critical Parameters
**1. Scale (s)** - How big each INT8 step is in FP32 space:
```
Small Scale (high precision): Large Scale (low precision):
FP32: [0.0, 0.255] FP32: [0.0, 25.5]
↓ ↓ ↓ ↓ ↓ ↓
INT8: 0 128 255 INT8: 0 128 255
│ │ │ │ │ │
0.0 0.127 0.255 0.0 12.75 25.5
Scale = 0.001 (very precise) Scale = 0.1 (less precise)
```
**2. Zero Point (z)** - Which INT8 value represents FP32 zero:
```
Symmetric Range: Asymmetric Range:
FP32: [-2.0, 2.0] FP32: [-1.0, 3.0]
↓ ↓ ↓ ↓ ↓ ↓
INT8: -128 0 127 INT8: -128 64 127
│ │ │ │ │ │
-2.0 0.0 2.0 -1.0 0.0 3.0
Zero Point = 0 Zero Point = 64
```
### Visual Example: Weight Quantization
```
Original FP32 Weights: Quantized INT8 Mapping:
┌─────────────────────────┐ ┌─────────────────────────┐
│ -0.8 -0.3 0.0 0.5 │ → │ -102 -38 0 64 │
│ 0.9 1.2 -0.1 0.7 │ │ 115 153 -13 89 │
└─────────────────────────┘ └─────────────────────────┘
4 bytes each 1 byte each
Total: 32 bytes Total: 8 bytes
4× compression!
```
### Quantization Error Analysis
```
Perfect Reconstruction (Impossible): Quantized Reconstruction (Reality):
Original: 0.73 Original: 0.73
↓ ↓
INT8: ? (can't represent exactly) INT8: 93 (closest)
↓ ↓
Restored: 0.73 Restored: 0.728
Error: 0.002
```
**The Quantization Trade-off:**
- **More bits** = Higher precision, larger memory
- **Fewer bits** = Lower precision, smaller memory
- **Goal:** Find the sweet spot where error is acceptable
### Why INT8 is the Sweet Spot
```
Precision vs Memory Trade-offs:
FP32: ████████████████████████████████ (32 bits) - Overkill precision
FP16: ████████████████ (16 bits) - Good precision
INT8: ████████ (8 bits) - Sufficient precision ← Sweet spot!
INT4: ████ (4 bits) - Often too little
Memory: 100% 50% 25% 12.5%
Accuracy: 100% 99.9% 99.5% 95%
```
INT8 gives us 4× memory reduction with <1% accuracy loss - the perfect balance for production systems!
"""
# %% [markdown]
"""
## 3. Implementation - Building the Quantization Engine
### Our Implementation Strategy
We'll build quantization in logical layers, each building on the previous:
```
Quantization System Architecture:
┌─────────────────────────────────────────────────────────────┐
│ Layer 4: Model Quantization │
│ quantize_model() - Convert entire neural networks │
├─────────────────────────────────────────────────────────────┤
│ Layer 3: Layer Quantization │
│ QuantizedLinear - Quantized linear transformations │
├─────────────────────────────────────────────────────────────┤
│ Layer 2: Tensor Operations │
│ quantize_int8() - Core quantization algorithm │
│ dequantize_int8() - Restore to floating point │
├─────────────────────────────────────────────────────────────┤
│ Layer 1: Foundation │
│ Scale & Zero Point Calculation - Parameter optimization │
└─────────────────────────────────────────────────────────────┘
```
### What We're About to Build
**Core Functions:**
- `quantize_int8()` - Convert FP32 tensors to INT8
- `dequantize_int8()` - Convert INT8 back to FP32
- `QuantizedLinear` - Quantized version of Linear layers
- `quantize_model()` - Quantize entire neural networks
**Key Features:**
- **Automatic calibration** - Find optimal quantization parameters
- **Error minimization** - Preserve accuracy during compression
- **Memory tracking** - Measure actual savings achieved
- **Production patterns** - Industry-standard algorithms
Let's start with the fundamental building block!
"""
# %% [markdown]
"""
### INT8 Quantization - The Foundation
This is the core function that converts any FP32 tensor to INT8. Think of it as a smart compression algorithm that preserves the most important information.
```
Quantization Process Visualization:
Step 1: Analyze Range Step 2: Calculate Parameters Step 3: Apply Formula
┌─────────────────────────┐ ┌─────────────────────────┐ ┌─────────────────────────┐
│ Input: [-1.5, 0.2, 2.8] │ │ Min: -1.5 │ │ quantized = round( │
│ │ │ Max: 2.8 │ │ (value - zp*scale) │
│ Find min/max values │ → │ Range: 4.3 │ →│ / scale) │
│ │ │ Scale: 4.3/255 = 0.017 │ │ │
│ │ │ Zero Point: 88 │ │ Result: [-128, 12, 127] │
└─────────────────────────┘ └─────────────────────────┘ └─────────────────────────┘
```
**Key Challenges This Function Solves:**
- **Dynamic Range:** Each tensor has different min/max values
- **Precision Loss:** Map 4 billion FP32 values to just 256 INT8 values
- **Zero Preservation:** Ensure FP32 zero maps exactly to an INT8 value
- **Symmetric Mapping:** Distribute quantization levels efficiently
**Why This Algorithm:**
- **Linear mapping** preserves relative relationships between values
- **Symmetric quantization** works well for most neural network weights
- **Clipping to [-128, 127]** ensures valid INT8 range
- **Round-to-nearest** minimizes quantization error
"""
# %% nbgrader={"grade": false, "grade_id": "quantize_int8", "solution": true}
def quantize_int8(tensor: Tensor) -> Tuple[Tensor, float, int]:
"""
Quantize FP32 tensor to INT8 using symmetric quantization.
TODO: Implement INT8 quantization with scale and zero_point calculation
APPROACH:
1. Find min/max values in tensor data
2. Calculate scale: (max_val - min_val) / 255 (INT8 range: -128 to 127)
3. Calculate zero_point: offset to map FP32 zero to INT8 zero
4. Apply quantization formula: round((value - zero_point) / scale)
5. Clamp to INT8 range [-128, 127]
Args:
tensor: Input FP32 tensor to quantize
Returns:
q_tensor: Quantized INT8 tensor
scale: Scaling factor (float)
zero_point: Zero point offset (int)
EXAMPLE:
>>> tensor = Tensor([[-1.0, 0.0, 2.0], [0.5, 1.5, -0.5]])
>>> q_tensor, scale, zero_point = quantize_int8(tensor)
>>> print(f"Scale: {scale:.4f}, Zero point: {zero_point}")
Scale: 0.0118, Zero point: 42
HINTS:
- Use np.round() for quantization
- Clamp with np.clip(values, -128, 127)
- Handle edge case where min_val == max_val (set scale=1.0)
"""
### BEGIN SOLUTION
data = tensor.data
# Step 1: Find dynamic range
min_val = float(np.min(data))
max_val = float(np.max(data))
# Step 2: Handle edge case (constant tensor)
if abs(max_val - min_val) < 1e-8:
scale = 1.0
zero_point = 0
quantized_data = np.zeros_like(data, dtype=np.int8)
return Tensor(quantized_data), scale, zero_point
# Step 3: Calculate scale and zero_point for standard quantization
# Map [min_val, max_val] to [-128, 127] (INT8 range)
scale = (max_val - min_val) / 255.0
zero_point = int(np.round(-128 - min_val / scale))
# Clamp zero_point to valid INT8 range
zero_point = int(np.clip(zero_point, -128, 127))
# Step 4: Apply quantization formula: q = (x / scale) + zero_point
quantized_data = np.round(data / scale + zero_point)
# Step 5: Clamp to INT8 range and convert to int8
quantized_data = np.clip(quantized_data, -128, 127).astype(np.int8)
return Tensor(quantized_data), scale, zero_point
### END SOLUTION
def test_unit_quantize_int8():
"""🔬 Test INT8 quantization implementation."""
print("🔬 Unit Test: INT8 Quantization...")
# Test basic quantization
tensor = Tensor([[1.0, 2.0, 3.0], [4.0, 5.0, 6.0]])
q_tensor, scale, zero_point = quantize_int8(tensor)
# Verify quantized values are in INT8 range
assert np.all(q_tensor.data >= -128)
assert np.all(q_tensor.data <= 127)
assert isinstance(scale, float)
assert isinstance(zero_point, int)
# Test dequantization preserves approximate values
dequantized = scale * (q_tensor.data - zero_point)
error = np.mean(np.abs(tensor.data - dequantized))
assert error < 0.2, f"Quantization error too high: {error}"
# Test edge case: constant tensor
constant_tensor = Tensor([[2.0, 2.0], [2.0, 2.0]])
q_const, scale_const, zp_const = quantize_int8(constant_tensor)
assert scale_const == 1.0
print("✅ INT8 quantization works correctly!")
test_unit_quantize_int8()
# %% [markdown]
"""
### INT8 Dequantization - Restoring Precision
Dequantization is the inverse process - converting compressed INT8 values back to usable FP32. This is where we "decompress" our quantized data.
```
Dequantization Process:
INT8 Values + Parameters → FP32 Reconstruction
┌─────────────────────────┐
│ Quantized: [-128, 12, 127] │
│ Scale: 0.017 │
│ Zero Point: 88 │
└─────────────────────────┘
▼ Apply Formula
┌─────────────────────────┐
│ FP32 = scale × quantized │
│ + zero_point × scale │
└─────────────────────────┘
┌─────────────────────────┐
│ Result: [-1.496, 0.204, 2.799]│
│ Original: [-1.5, 0.2, 2.8] │
│ Error: [0.004, 0.004, 0.001] │
└─────────────────────────┘
Excellent approximation!
```
**Why This Step Is Critical:**
- **Neural networks expect FP32** - INT8 values would confuse computations
- **Preserves computation compatibility** - works with existing matrix operations
- **Controlled precision loss** - error is bounded and predictable
- **Hardware flexibility** - can use FP32 or specialized INT8 operations
**When Dequantization Happens:**
- **During forward pass** - before matrix multiplications
- **For gradient computation** - during backward pass
- **Educational approach** - production uses INT8 GEMM directly
"""
# %% nbgrader={"grade": false, "grade_id": "dequantize_int8", "solution": true}
def dequantize_int8(q_tensor: Tensor, scale: float, zero_point: int) -> Tensor:
"""
Dequantize INT8 tensor back to FP32.
TODO: Implement dequantization using the inverse formula
APPROACH:
1. Apply inverse quantization: scale * quantized_value + zero_point * scale
2. Return as new FP32 Tensor
Args:
q_tensor: Quantized INT8 tensor
scale: Scaling factor from quantization
zero_point: Zero point offset from quantization
Returns:
Reconstructed FP32 tensor
EXAMPLE:
>>> q_tensor = Tensor([[-42, 0, 85]]) # INT8 values
>>> scale, zero_point = 0.0314, 64
>>> fp32_tensor = dequantize_int8(q_tensor, scale, zero_point)
>>> print(fp32_tensor.data)
[[-1.31, 2.01, 2.67]] # Approximate original values
HINT:
- Formula: dequantized = scale * quantized + zero_point * scale
"""
### BEGIN SOLUTION
# Apply inverse quantization formula
dequantized_data = scale * q_tensor.data + zero_point * scale
return Tensor(dequantized_data.astype(np.float32))
### END SOLUTION
def test_unit_dequantize_int8():
"""🔬 Test INT8 dequantization implementation."""
print("🔬 Unit Test: INT8 Dequantization...")
# Test round-trip: quantize → dequantize
original = Tensor([[-1.5, 0.0, 3.2], [1.1, -0.8, 2.7]])
q_tensor, scale, zero_point = quantize_int8(original)
restored = dequantize_int8(q_tensor, scale, zero_point)
# Verify round-trip error is small
error = np.mean(np.abs(original.data - restored.data))
assert error < 2.0, f"Round-trip error too high: {error}"
# Verify output is float32
assert restored.data.dtype == np.float32
print("✅ INT8 dequantization works correctly!")
test_unit_dequantize_int8()
# %% [markdown]
"""
## QuantizedLinear - The Heart of Efficient Networks
### Why We Need Quantized Layers
A quantized model isn't just about storing weights in INT8 - we need layers that can work efficiently with quantized data.
```
Regular Linear Layer: QuantizedLinear Layer:
┌─────────────────────┐ ┌─────────────────────┐
│ Input: FP32 │ │ Input: FP32 │
│ Weights: FP32 │ │ Weights: INT8 │
│ Computation: FP32 │ VS │ Computation: Mixed │
│ Output: FP32 │ │ Output: FP32 │
│ Memory: 4× more │ │ Memory: 4× less │
└─────────────────────┘ └─────────────────────┘
```
### The Quantized Forward Pass
```
Quantized Linear Layer Forward Pass:
Input (FP32) Quantized Weights (INT8)
│ │
▼ ▼
┌─────────────────┐ ┌─────────────────┐
│ Calibrate │ │ Dequantize │
│ (optional) │ │ Weights │
└─────────────────┘ └─────────────────┘
│ │
▼ ▼
Input (FP32) Weights (FP32)
│ │
└───────────────┬───────────────┘
┌─────────────────┐
│ Matrix Multiply │
│ (FP32 GEMM) │
└─────────────────┘
Output (FP32)
Memory Saved: 4× for weights storage!
Speed: Depends on dequantization overhead vs INT8 GEMM support
```
### Calibration - Finding Optimal Input Quantization
```
Calibration Process:
Step 1: Collect Sample Inputs Step 2: Analyze Distribution Step 3: Optimize Parameters
┌─────────────────────────┐ ┌─────────────────────────┐ ┌─────────────────────────┐
│ input_1: [-0.5, 0.2, ..] │ │ Min: -0.8 │ │ Scale: 0.00627 │
│ input_2: [-0.3, 0.8, ..] │ → │ Max: +0.8 │ → │ Zero Point: 0 │
│ input_3: [-0.1, 0.5, ..] │ │ Range: 1.6 │ │ Optimal for this data │
│ ... │ │ Distribution: Normal │ │ range and distribution │
└─────────────────────────┘ └─────────────────────────┘ └─────────────────────────┘
```
**Why Calibration Matters:**
- **Without calibration:** Generic quantization parameters may waste precision
- **With calibration:** Parameters optimized for actual data distribution
- **Result:** Better accuracy preservation with same memory savings
"""
# %% [markdown]
"""
### QuantizedLinear Class - Efficient Neural Network Layer
This class replaces regular Linear layers with quantized versions that use 4× less memory while preserving functionality.
```
QuantizedLinear Architecture:
Creation Time: Runtime:
┌─────────────────────────┐ ┌─────────────────────────┐
│ Regular Linear Layer │ │ Input (FP32) │
│ ↓ │ │ ↓ │
│ Quantize weights → INT8 │ │ Optional: quantize input│
│ Quantize bias → INT8 │ → │ ↓ │
│ Store quantization params │ │ Dequantize weights │
│ Ready for deployment! │ │ ↓ │
└─────────────────────────┘ │ Matrix multiply (FP32) │
One-time cost │ ↓ │
│ Output (FP32) │
└─────────────────────────┘
Per-inference cost
```
**Key Design Decisions:**
1. **Store original layer reference** - for debugging and comparison
2. **Separate quantization parameters** - weights and bias may need different scales
3. **Calibration support** - optimize input quantization using real data
4. **FP32 computation** - educational approach, production uses INT8 GEMM
5. **Memory tracking** - measure actual compression achieved
**Memory Layout Comparison:**
```
Regular Linear Layer: QuantizedLinear Layer:
┌─────────────────────────┐ ┌─────────────────────────┐
│ weights: FP32 × N │ │ q_weights: INT8 × N │
│ bias: FP32 × M │ │ q_bias: INT8 × M │
│ │ → │ weight_scale: 1 float │
│ Total: 4×(N+M) bytes │ │ weight_zero_point: 1 int│
└─────────────────────────┘ │ bias_scale: 1 float │
│ bias_zero_point: 1 int │
│ │
│ Total: (N+M) + 16 bytes │
└─────────────────────────┘
~4× smaller!
```
**Production vs Educational Trade-off:**
- **Our approach:** Dequantize → FP32 computation (easier to understand)
- **Production:** INT8 GEMM operations (faster, more complex)
- **Both achieve:** Same memory savings, similar accuracy
"""
# %% nbgrader={"grade": false, "grade_id": "quantized_linear", "solution": true}
class QuantizedLinear:
"""Quantized version of Linear layer using INT8 arithmetic."""
def __init__(self, linear_layer: Linear):
"""
Create quantized version of existing linear layer.
TODO: Quantize weights and bias, store quantization parameters
APPROACH:
1. Quantize weights using quantize_int8
2. Quantize bias if it exists
3. Store original layer reference for forward pass
4. Store quantization parameters for dequantization
IMPLEMENTATION STRATEGY:
- Store quantized weights, scales, and zero points
- Implement forward pass using dequantized computation (educational approach)
- Production: Would use INT8 matrix multiplication libraries
"""
### BEGIN SOLUTION
self.original_layer = linear_layer
# Quantize weights
self.q_weight, self.weight_scale, self.weight_zero_point = quantize_int8(linear_layer.weight)
# Quantize bias if it exists
if linear_layer.bias is not None:
self.q_bias, self.bias_scale, self.bias_zero_point = quantize_int8(linear_layer.bias)
else:
self.q_bias = None
self.bias_scale = None
self.bias_zero_point = None
# Store input quantization parameters (set during calibration)
self.input_scale = None
self.input_zero_point = None
### END SOLUTION
def calibrate(self, sample_inputs: List[Tensor]):
"""
Calibrate input quantization parameters using sample data.
TODO: Calculate optimal input quantization parameters
APPROACH:
1. Collect statistics from sample inputs
2. Calculate optimal scale and zero_point for inputs
3. Store for use in forward pass
"""
### BEGIN SOLUTION
# Collect all input values
all_values = []
for inp in sample_inputs:
all_values.extend(inp.data.flatten())
all_values = np.array(all_values)
# Calculate input quantization parameters
min_val = float(np.min(all_values))
max_val = float(np.max(all_values))
if abs(max_val - min_val) < 1e-8:
self.input_scale = 1.0
self.input_zero_point = 0
else:
self.input_scale = (max_val - min_val) / 255.0
self.input_zero_point = int(np.round(-128 - min_val / self.input_scale))
self.input_zero_point = np.clip(self.input_zero_point, -128, 127)
### END SOLUTION
def forward(self, x: Tensor) -> Tensor:
"""
Forward pass with quantized computation.
TODO: Implement quantized forward pass
APPROACH:
1. Quantize input (if calibrated)
2. Dequantize weights and input for computation (educational approach)
3. Perform matrix multiplication
4. Return FP32 result
NOTE: Production quantization uses INT8 GEMM libraries for speed
"""
### BEGIN SOLUTION
# For educational purposes, we dequantize and compute in FP32
# Production systems use specialized INT8 GEMM operations
# Dequantize weights
weight_fp32 = dequantize_int8(self.q_weight, self.weight_scale, self.weight_zero_point)
# Perform computation (same as original layer)
result = x.matmul(weight_fp32)
# Add bias if it exists
if self.q_bias is not None:
bias_fp32 = dequantize_int8(self.q_bias, self.bias_scale, self.bias_zero_point)
result = Tensor(result.data + bias_fp32.data)
return result
### END SOLUTION
def __call__(self, x: Tensor) -> Tensor:
"""Allows the quantized linear layer to be called like a function."""
return self.forward(x)
def parameters(self) -> List[Tensor]:
"""Return quantized parameters."""
params = [self.q_weight]
if self.q_bias is not None:
params.append(self.q_bias)
return params
def memory_usage(self) -> Dict[str, float]:
"""Calculate memory usage in bytes."""
### BEGIN SOLUTION
# Original FP32 usage
original_weight_bytes = self.original_layer.weight.data.size * 4 # 4 bytes per FP32
original_bias_bytes = 0
if self.original_layer.bias is not None:
original_bias_bytes = self.original_layer.bias.data.size * 4
# Quantized INT8 usage
quantized_weight_bytes = self.q_weight.data.size * 1 # 1 byte per INT8
quantized_bias_bytes = 0
if self.q_bias is not None:
quantized_bias_bytes = self.q_bias.data.size * 1
# Add overhead for scales and zero points (small)
overhead_bytes = 8 * 2 # 2 floats + 2 ints for weight/bias quantization params
return {
'original_bytes': original_weight_bytes + original_bias_bytes,
'quantized_bytes': quantized_weight_bytes + quantized_bias_bytes + overhead_bytes,
'compression_ratio': (original_weight_bytes + original_bias_bytes) /
(quantized_weight_bytes + quantized_bias_bytes + overhead_bytes)
}
### END SOLUTION
def test_unit_quantized_linear():
"""🔬 Test QuantizedLinear implementation."""
print("🔬 Unit Test: QuantizedLinear...")
# Create original linear layer
original = Linear(4, 3)
original.weight = Tensor(np.random.randn(4, 3) * 0.5) # Smaller range for testing
original.bias = Tensor(np.random.randn(3) * 0.1)
# Create quantized version
quantized = QuantizedLinear(original)
# Test forward pass
x = Tensor(np.random.randn(2, 4) * 0.5)
# Original forward pass
original_output = original.forward(x)
# Quantized forward pass
quantized_output = quantized.forward(x)
# Compare outputs (should be close but not identical due to quantization)
error = np.mean(np.abs(original_output.data - quantized_output.data))
assert error < 1.0, f"Quantization error too high: {error}"
# Test memory usage
memory_info = quantized.memory_usage()
assert memory_info['compression_ratio'] > 3.0, "Should achieve ~4× compression"
print(f" Memory reduction: {memory_info['compression_ratio']:.1f}×")
print("✅ QuantizedLinear works correctly!")
test_unit_quantized_linear()
# %% [markdown]
"""
## 4. Integration - Scaling to Full Neural Networks
### The Model Quantization Challenge
Quantizing individual tensors is useful, but real applications need to quantize entire neural networks with multiple layers, activations, and complex data flows.
```
Model Quantization Process:
Original Model: Quantized Model:
┌─────────────────────────────┐ ┌─────────────────────────────┐
│ Linear(784, 128) [FP32] │ │ QuantizedLinear(784, 128) │
│ ReLU() [FP32] │ │ ReLU() [FP32] │
│ Linear(128, 64) [FP32] │ → │ QuantizedLinear(128, 64) │
│ ReLU() [FP32] │ │ ReLU() [FP32] │
│ Linear(64, 10) [FP32] │ │ QuantizedLinear(64, 10) │
└─────────────────────────────┘ └─────────────────────────────┘
Memory: 100% Memory: ~25%
Speed: Baseline Speed: 2-4× faster
```
### Smart Layer Selection
Not all layers benefit equally from quantization:
```
Layer Quantization Strategy:
┌─────────────────┬─────────────────┬─────────────────────────────┐
│ Layer Type │ Quantize? │ Reason │
├─────────────────┼─────────────────┼─────────────────────────────┤
│ Linear/Dense │ ✅ YES │ Most parameters, big savings │
│ Convolution │ ✅ YES │ Many weights, good candidate │
│ Embedding │ ✅ YES │ Large lookup tables │
│ ReLU/Sigmoid │ ❌ NO │ No parameters to quantize │
│ BatchNorm │ 🤔 MAYBE │ Few params, may hurt │
│ First Layer │ 🤔 MAYBE │ Often sensitive to precision │
│ Last Layer │ 🤔 MAYBE │ Output quality critical │
└─────────────────┴─────────────────┴─────────────────────────────┘
```
### Calibration Data Flow
```
End-to-End Calibration:
Calibration Input Layer-by-Layer Processing
│ │
▼ ▼
┌─────────────┐ ┌──────────────────────────────────────────┐
│ Sample Data │ → │ Layer 1: Collect activation statistics │
│ [batch of │ │ ↓ │
│ real data] │ │ Layer 2: Collect activation statistics │
└─────────────┘ │ ↓ │
│ Layer 3: Collect activation statistics │
│ ↓ │
│ Optimize quantization parameters │
└──────────────────────────────────────────┘
Ready for deployment!
```
### Memory Impact Visualization
```
Model Memory Breakdown:
Before Quantization: After Quantization:
┌─────────────────────┐ ┌─────────────────────┐
│ Layer 1: 3.1MB │ │ Layer 1: 0.8MB │ (-75%)
│ Layer 2: 0.5MB │ → │ Layer 2: 0.1MB │ (-75%)
│ Layer 3: 0.3MB │ │ Layer 3: 0.1MB │ (-75%)
│ Total: 3.9MB │ │ Total: 1.0MB │ (-74%)
└─────────────────────┘ └─────────────────────┘
Typical mobile phone memory: 4-8GB
Model now fits: 4000× more models in memory!
```
Now let's implement the functions that make this transformation possible!
"""
# %% [markdown]
"""
### Model Quantization - Scaling to Full Networks
This function transforms entire neural networks from FP32 to quantized versions. It's like upgrading a whole building to be more energy efficient!
```
Model Transformation Process:
Input Model: Quantized Model:
┌─────────────────────────────┐ ┌─────────────────────────────┐
│ layers[0]: Linear(784, 128) │ │ layers[0]: QuantizedLinear │
│ layers[1]: ReLU() │ │ layers[1]: ReLU() │
│ layers[2]: Linear(128, 64) │ → │ layers[2]: QuantizedLinear │
│ layers[3]: ReLU() │ │ layers[3]: ReLU() │
│ layers[4]: Linear(64, 10) │ │ layers[4]: QuantizedLinear │
└─────────────────────────────┘ └─────────────────────────────┘
Memory: 100% Memory: ~25%
Interface: Same Interface: Identical
```
**Smart Layer Selection Logic:**
```
Quantization Decision Tree:
For each layer in model:
├── Is it a Linear layer?
│ │
│ └── YES → Replace with QuantizedLinear
└── Is it ReLU/Activation?
└── NO → Keep unchanged (no parameters to quantize)
```
**Calibration Integration:**
```
Calibration Data Flow:
Input Data Layer-by-Layer Processing
│ │
▼ ▼
┌─────────────────┐ ┌───────────────────────────────────────────────────────────┐
│ Sample Batch 1 │ │ Layer 0: Forward → Collect activation statistics │
│ Sample Batch 2 │ → │ ↓ │
│ ... │ │ Layer 2: Forward → Collect activation statistics │
│ Sample Batch N │ │ ↓ │
└─────────────────┘ │ Layer 4: Forward → Collect activation statistics │
│ ↓ │
│ For each layer: calibrate optimal quantization │
└───────────────────────────────────────────────────────────┘
```
**Why In-Place Modification:**
- **Preserves model structure** - Same interface, same behavior
- **Memory efficient** - No copying of large tensors
- **Drop-in replacement** - Existing code works unchanged
- **Gradual quantization** - Can selectively quantize sensitive layers
**Deployment Benefits:**
```
Before Quantization: After Quantization:
┌─────────────────────────┐ ┌─────────────────────────┐
│ ❌ Can't fit on phone │ │ ✅ Fits on mobile device │
│ ❌ Slow cloud deployment │ │ ✅ Fast edge inference │
│ ❌ High memory usage │ → │ ✅ 4× memory efficiency │
│ ❌ Expensive to serve │ │ ✅ Lower serving costs │
│ ❌ Battery drain │ │ ✅ Extended battery life │
└─────────────────────────┘ └─────────────────────────┘
```
"""
# %% nbgrader={"grade": false, "grade_id": "quantize_model", "solution": true}
def quantize_model(model, calibration_data: Optional[List[Tensor]] = None) -> None:
"""
Quantize all Linear layers in a model in-place.
TODO: Replace all Linear layers with QuantizedLinear versions
APPROACH:
1. Find all Linear layers in the model
2. Replace each with QuantizedLinear version
3. If calibration data provided, calibrate input quantization
4. Handle Sequential containers properly
Args:
model: Model to quantize (with .layers or similar structure)
calibration_data: Optional list of sample inputs for calibration
Returns:
None (modifies model in-place)
EXAMPLE:
>>> model = Sequential(Linear(10, 5), ReLU(), Linear(5, 2))
>>> quantize_model(model)
>>> # Now model uses quantized layers
HINT:
- Handle Sequential.layers list for layer replacement
- Use isinstance(layer, Linear) to identify layers to quantize
"""
### BEGIN SOLUTION
if hasattr(model, 'layers'): # Sequential model
for i, layer in enumerate(model.layers):
if isinstance(layer, Linear):
# Replace with quantized version
quantized_layer = QuantizedLinear(layer)
# Calibrate if data provided
if calibration_data is not None:
# Run forward passes to get intermediate activations
sample_inputs = []
for data in calibration_data[:10]: # Use first 10 samples for efficiency
# Forward through layers up to this point
x = data
for j in range(i):
if hasattr(model.layers[j], 'forward'):
x = model.layers[j].forward(x)
sample_inputs.append(x)
quantized_layer.calibrate(sample_inputs)
model.layers[i] = quantized_layer
elif isinstance(model, Linear): # Single Linear layer
# Can't replace in-place for single layer, user should handle
raise ValueError("Cannot quantize single Linear layer in-place. Use QuantizedLinear directly.")
else:
raise ValueError(f"Unsupported model type: {type(model)}")
### END SOLUTION
def test_unit_quantize_model():
"""🔬 Test model quantization implementation."""
print("🔬 Unit Test: Model Quantization...")
# Create test model
model = Sequential(
Linear(4, 8),
ReLU(),
Linear(8, 3)
)
# Initialize weights
model.layers[0].weight = Tensor(np.random.randn(4, 8) * 0.5)
model.layers[0].bias = Tensor(np.random.randn(8) * 0.1)
model.layers[2].weight = Tensor(np.random.randn(8, 3) * 0.5)
model.layers[2].bias = Tensor(np.random.randn(3) * 0.1)
# Test original model
x = Tensor(np.random.randn(2, 4))
original_output = model.forward(x)
# Create calibration data
calibration_data = [Tensor(np.random.randn(1, 4)) for _ in range(5)]
# Quantize model
quantize_model(model, calibration_data)
# Verify layers were replaced
assert isinstance(model.layers[0], QuantizedLinear)
assert isinstance(model.layers[1], ReLU) # Should remain unchanged
assert isinstance(model.layers[2], QuantizedLinear)
# Test quantized model
quantized_output = model.forward(x)
# Compare outputs
error = np.mean(np.abs(original_output.data - quantized_output.data))
print(f" Model quantization error: {error:.4f}")
assert error < 2.0, f"Model quantization error too high: {error}"
print("✅ Model quantization works correctly!")
test_unit_quantize_model()
# %% [markdown]
"""
### Model Size Comparison - Measuring the Impact
This function provides detailed analysis of memory savings achieved through quantization. It's like a before/after comparison for model efficiency.
```
Memory Analysis Framework:
┌────────────────────────────────────────────────────────────────────────────────────┐
│ Memory Breakdown Analysis │
├─────────────────┬─────────────────┬─────────────────┬─────────────────┤
│ Component │ Original (FP32) │ Quantized (INT8) │ Savings │
├─────────────────┼─────────────────┼─────────────────┼─────────────────┤
│ Layer 1 weights │ 12.8 MB │ 3.2 MB │ 9.6 MB (75%)│
│ Layer 1 bias │ 0.5 MB │ 0.1 MB │ 0.4 MB (75%)│
│ Layer 2 weights │ 2.0 MB │ 0.5 MB │ 1.5 MB (75%)│
│ Layer 2 bias │ 0.3 MB │ 0.1 MB │ 0.2 MB (67%)│
│ Overhead │ 0.0 MB │ 0.02 MB │ -0.02 MB │
├─────────────────┼─────────────────┼─────────────────┼─────────────────┤
│ TOTAL │ 15.6 MB │ 3.92 MB │ 11.7 MB (74%)│
└─────────────────┴─────────────────┴─────────────────┴─────────────────┘
4× compression ratio!
```
**Comprehensive Metrics Provided:**
```
Output Dictionary:
{
'original_params': 4000000, # Total parameter count
'quantized_params': 4000000, # Same count, different precision
'original_bytes': 16000000, # 4 bytes per FP32 parameter
'quantized_bytes': 4000016, # 1 byte per INT8 + overhead
'compression_ratio': 3.99, # Nearly 4× compression
'memory_saved_mb': 11.7, # Absolute savings in MB
'memory_saved_percent': 74.9 # Relative savings percentage
}
```
**Why These Metrics Matter:**
**For Developers:**
- **compression_ratio** - How much smaller is the model?
- **memory_saved_mb** - Actual bytes freed up
- **memory_saved_percent** - Efficiency improvement
**For Deployment:**
- **Model fits in device memory?** Check memory_saved_mb
- **Network transfer time?** Reduced by compression_ratio
- **Disk storage savings?** Shown by memory_saved_percent
**For Business:**
- **Cloud costs** reduced by compression_ratio
- **User experience** improved (faster downloads)
- **Device support** expanded (fits on more devices)
**Validation Checks:**
- **Parameter count preservation** - same functionality
- **Reasonable compression ratio** - should be ~4× for INT8
- **Minimal overhead** - quantization parameters are tiny
"""
# %% nbgrader={"grade": false, "grade_id": "compare_model_sizes", "solution": true}
def compare_model_sizes(original_model, quantized_model) -> Dict[str, float]:
"""
Compare memory usage between original and quantized models.
TODO: Calculate comprehensive memory comparison
APPROACH:
1. Count parameters in both models
2. Calculate bytes used (FP32 vs INT8)
3. Include quantization overhead
4. Return comparison metrics
Args:
original_model: Model before quantization
quantized_model: Model after quantization
Returns:
Dictionary with 'original_mb', 'quantized_mb', 'reduction_ratio', 'memory_saved_mb'
EXAMPLE:
>>> model = Sequential(Linear(100, 50), Linear(50, 10))
>>> quantize_model(model)
>>> stats = compare_model_sizes(model, model) # Same model after in-place quantization
>>> print(f"Reduced to {stats['reduction_ratio']:.1f}x smaller")
Reduced to 4.0x smaller
HINTS:
- FP32 uses 4 bytes per parameter, INT8 uses 1 byte
- Include scale/zero_point overhead (2 values per quantized layer)
- Expected ratio: ~4x for INT8 quantization
"""
### BEGIN SOLUTION
# Count original model parameters
original_params = 0
original_bytes = 0
if hasattr(original_model, 'layers'):
for layer in original_model.layers:
if hasattr(layer, 'parameters'):
params = layer.parameters()
for param in params:
original_params += param.data.size
original_bytes += param.data.size * 4 # 4 bytes per FP32
# Count quantized model parameters
quantized_params = 0
quantized_bytes = 0
if hasattr(quantized_model, 'layers'):
for layer in quantized_model.layers:
if isinstance(layer, QuantizedLinear):
memory_info = layer.memory_usage()
quantized_bytes += memory_info['quantized_bytes']
params = layer.parameters()
for param in params:
quantized_params += param.data.size
elif hasattr(layer, 'parameters'):
# Non-quantized layers
params = layer.parameters()
for param in params:
quantized_params += param.data.size
quantized_bytes += param.data.size * 4
compression_ratio = original_bytes / quantized_bytes if quantized_bytes > 0 else 1.0
memory_saved = original_bytes - quantized_bytes
return {
'original_params': original_params,
'quantized_params': quantized_params,
'original_bytes': original_bytes,
'quantized_bytes': quantized_bytes,
'compression_ratio': compression_ratio,
'memory_saved_mb': memory_saved / (1024 * 1024),
'memory_saved_percent': (memory_saved / original_bytes) * 100 if original_bytes > 0 else 0
}
### END SOLUTION
def test_unit_compare_model_sizes():
"""🔬 Test model size comparison."""
print("🔬 Unit Test: Model Size Comparison...")
# Create and quantize a model for testing
original_model = Sequential(Linear(100, 50), ReLU(), Linear(50, 10))
original_model.layers[0].weight = Tensor(np.random.randn(100, 50))
original_model.layers[0].bias = Tensor(np.random.randn(50))
original_model.layers[2].weight = Tensor(np.random.randn(50, 10))
original_model.layers[2].bias = Tensor(np.random.randn(10))
# Create quantized copy
quantized_model = Sequential(Linear(100, 50), ReLU(), Linear(50, 10))
quantized_model.layers[0].weight = Tensor(np.random.randn(100, 50))
quantized_model.layers[0].bias = Tensor(np.random.randn(50))
quantized_model.layers[2].weight = Tensor(np.random.randn(50, 10))
quantized_model.layers[2].bias = Tensor(np.random.randn(10))
quantize_model(quantized_model)
# Compare sizes
comparison = compare_model_sizes(original_model, quantized_model)
# Verify compression achieved
assert comparison['compression_ratio'] > 2.0, "Should achieve significant compression"
assert comparison['memory_saved_percent'] > 50, "Should save >50% memory"
print(f" Compression ratio: {comparison['compression_ratio']:.1f}×")
print(f" Memory saved: {comparison['memory_saved_percent']:.1f}%")
print("✅ Model size comparison works correctly!")
test_unit_compare_model_sizes()
# %% [markdown]
"""
## 5. Optimization Insights - Production Quantization Strategies
### Beyond Basic Quantization
Our INT8 per-tensor quantization is just the beginning. Production systems use sophisticated strategies to squeeze out every bit of performance while preserving accuracy.
```
Quantization Strategy Evolution:
Basic (What we built) Advanced (Production) Cutting-Edge (Research)
┌─────────────────────┐ ┌─────────────────────┐ ┌─────────────────────┐
│ • Per-tensor scale │ │ • Per-channel scale │ │ • Dynamic ranges │
│ • Uniform INT8 │ → │ • Mixed precision │ → │ • Adaptive bitwidth │
│ • Post-training │ │ • Quantization-aware│ │ • Learned quantizers│
│ • Simple calibration│ │ • Advanced calib. │ │ • Neural compression│
└─────────────────────┘ └─────────────────────┘ └─────────────────────┘
Good baseline Production systems Future research
```
### Strategy Comparison Framework
```
Quantization Strategy Trade-offs:
┌─────────────────────┬─────────────┬─────────────┬─────────────┬─────────────┐
│ Strategy │ Accuracy │ Complexity │ Memory Use │ Speed Gain │
├─────────────────────┼─────────────┼─────────────┼─────────────┼─────────────┤
│ Per-Tensor (Ours) │ ████████░░ │ ██░░░░░░░░ │ ████████░░ │ ███████░░░ │
│ Per-Channel │ █████████░ │ █████░░░░░ │ ████████░░ │ ██████░░░░ │
│ Mixed Precision │ ██████████ │ ████████░░ │ ███████░░░ │ ████████░░ │
│ Quantization-Aware │ ██████████ │ ██████████ │ ████████░░ │ ███████░░░ │
└─────────────────────┴─────────────┴─────────────┴─────────────┴─────────────┘
```
### The Three Advanced Strategies We'll Analyze
**1. Per-Channel Quantization:**
```
Per-Tensor: Per-Channel:
┌─────────────────────────┐ ┌─────────────────────────┐
│ [W₁₁ W₁₂ W₁₃] │ │ [W₁₁ W₁₂ W₁₃] scale₁ │
│ [W₂₁ W₂₂ W₂₃] scale │ VS │ [W₂₁ W₂₂ W₂₃] scale₂ │
│ [W₃₁ W₃₂ W₃₃] │ │ [W₃₁ W₃₂ W₃₃] scale₃ │
└─────────────────────────┘ └─────────────────────────┘
One scale for all Separate scale per channel
May waste precision Better precision per channel
```
**2. Mixed Precision:**
```
Sensitive Layers (FP32): Regular Layers (INT8):
┌─────────────────────────┐ ┌─────────────────────────┐
│ Input Layer │ │ Hidden Layer 1 │
│ (preserve input quality)│ │ (can tolerate error) │
├─────────────────────────┤ ├─────────────────────────┤
│ Output Layer │ │ Hidden Layer 2 │
│ (preserve output) │ │ (bulk of computation) │
└─────────────────────────┘ └─────────────────────────┘
Keep high precision Maximize compression
```
**3. Calibration Strategies:**
```
Basic Calibration: Advanced Calibration:
┌─────────────────────────┐ ┌─────────────────────────┐
│ • Use min/max range │ │ • Percentile clipping │
│ • Simple statistics │ │ • KL-divergence │
│ • Few samples │ VS │ • Multiple datasets │
│ • Generic approach │ │ • Layer-specific tuning │
└─────────────────────────┘ └─────────────────────────┘
Fast but suboptimal Optimal but expensive
```
Let's implement and compare these strategies to understand their practical trade-offs!
"""
# %% [markdown]
"""
### Advanced Quantization Strategies - Production Techniques
This analysis compares different quantization approaches used in production systems, revealing the trade-offs between accuracy, complexity, and performance.
```
Strategy Comparison Framework:
┌────────────────────────────────────────────────────────────────────────────────────┐
│ Three Advanced Strategies │
├────────────────────────────┬────────────────────────────┬────────────────────────────┤
│ Strategy 1 │ Strategy 2 │ Strategy 3 │
│ Per-Tensor (Ours) │ Per-Channel Scale │ Mixed Precision │
├────────────────────────────┼────────────────────────────┼────────────────────────────┤
│ │ │ │
│ ┌──────────────────────┐ │ ┌──────────────────────┐ │ ┌──────────────────────┐ │
│ │ Weights: │ │ │ Channel 1: scale₁ │ │ │ Sensitive: FP32 │ │
│ │ [W₁₁ W₁₂ W₁₃] │ │ │ Channel 2: scale₂ │ │ │ Regular: INT8 │ │
│ │ [W₂₁ W₂₂ W₂₃] scale │ │ │ Channel 3: scale₃ │ │ │ │ │
│ │ [W₃₁ W₃₂ W₃₃] │ │ │ │ │ │ Input: FP32 │ │
│ └──────────────────────┘ │ │ Better precision │ │ │ Output: FP32 │ │
│ │ │ per channel │ │ │ Hidden: INT8 │ │
│ Simple, fast │ └──────────────────────┘ │ └──────────────────────┘ │
│ Good baseline │ │ │
│ │ More complex │ Optimal accuracy │
│ │ Better accuracy │ Selective compression │
└────────────────────────────┴────────────────────────────┴────────────────────────────┘
```
**Strategy 1: Per-Tensor Quantization (Our Implementation)**
```
Weight Matrix: Scale Calculation:
┌─────────────────────────┐ ┌─────────────────────────┐
│ 0.1 -0.3 0.8 0.2 │ │ Global min: -0.5 │
│-0.2 0.5 -0.1 0.7 │ → │ Global max: +0.8 │
│ 0.4 -0.5 0.3 -0.4 │ │ Scale: 1.3/255 = 0.0051 │
└─────────────────────────┘ └─────────────────────────┘
Pros: Simple, fast Cons: May waste precision
```
**Strategy 2: Per-Channel Quantization (Advanced)**
```
Weight Matrix: Scale Calculation:
┌─────────────────────────┐ ┌─────────────────────────┐
│ 0.1 -0.3 0.8 0.2 │ │ Col 1: [-0.2,0.4] → s₁ │
│-0.2 0.5 -0.1 0.7 │ → │ Col 2: [-0.5,0.5] → s₂ │
│ 0.4 -0.5 0.3 -0.4 │ │ Col 3: [-0.1,0.8] → s₃ │
└─────────────────────────┘ │ Col 4: [-0.4,0.7] → s₄ │
└─────────────────────────┘
Pros: Better precision Cons: More complex
```
**Strategy 3: Mixed Precision (Production)**
```
Model Architecture: Precision Assignment:
┌─────────────────────────┐ ┌─────────────────────────┐
│ Input Layer (sensitive) │ │ Keep in FP32 (precision) │
│ Hidden 1 (bulk) │ → │ Quantize to INT8 │
│ Hidden 2 (bulk) │ │ Quantize to INT8 │
│ Output Layer (sensitive)│ │ Keep in FP32 (quality) │
└─────────────────────────┘ └─────────────────────────┘
Pros: Optimal trade-off Cons: Requires expertise
```
**Experimental Design:**
```
Comparative Testing Protocol:
1. Create identical test model → 2. Apply each strategy → 3. Measure results
┌───────────────────────┐ ┌───────────────────────┐ ┌───────────────────────┐
│ 128 → 64 → 10 MLP │ │ Per-tensor quantization │ │ MSE error calculation │
│ Identical weights │ │ Per-channel simulation │ │ Compression measurement│
│ Same test input │ │ Mixed precision setup │ │ Speed comparison │
└───────────────────────┘ └───────────────────────┘ └───────────────────────┘
```
**Expected Strategy Rankings:**
1. **Mixed Precision** - Best accuracy, moderate complexity
2. **Per-Channel** - Good accuracy, higher complexity
3. **Per-Tensor** - Baseline accuracy, simplest implementation
This analysis reveals which strategies work best for different deployment scenarios and accuracy requirements.
"""
# %% [markdown]
"""
## 5.5 Measuring Quantization Savings with Profiler
Now let's use the **Profiler** tool from Module 15 to measure the actual memory savings from quantization. This demonstrates end-to-end workflow: profile baseline (M15) → apply quantization (M17) → measure savings (M15+M17).
This is the production workflow: measure → compress → validate → deploy.
"""
# %% nbgrader={"grade": false, "grade_id": "demo-profiler-quantization", "solution": true}
# Import Profiler from Module 15
from tinytorch.profiling.profiler import Profiler
def demo_quantization_with_profiler():
"""📊 Demonstrate memory savings using Profiler from Module 15."""
print("📊 Measuring Quantization Memory Savings with Profiler")
print("=" * 70)
profiler = Profiler()
# Create a simple model
from tinytorch.core.layers import Linear
model = Linear(512, 256)
model.name = "baseline_model"
print("\n💾 BEFORE: FP32 Model")
print("-" * 70)
# Measure baseline
param_count = profiler.count_parameters(model)
input_shape = (32, 512)
memory_stats = profiler.measure_memory(model, input_shape)
print(f" Parameters: {param_count:,}")
print(f" Parameter memory: {memory_stats['parameter_memory_mb']:.2f} MB")
print(f" Peak memory: {memory_stats['peak_memory_mb']:.2f} MB")
print(f" Precision: FP32 (4 bytes per parameter)")
# Quantize the model
print("\n🗜️ Quantizing to INT8...")
quantized_model = quantize_model(model)
quantized_model.name = "quantized_model"
print("\n📦 AFTER: INT8 Quantized Model")
print("-" * 70)
# Measure quantized (simulated - in practice INT8 uses 1 byte)
# For demonstration, we show the theoretical savings
quantized_param_count = profiler.count_parameters(quantized_model)
theoretical_memory_mb = param_count * 1 / (1024 * 1024) # 1 byte per INT8 param
print(f" Parameters: {quantized_param_count:,} (same count, different precision)")
print(f" Parameter memory (theoretical): {theoretical_memory_mb:.2f} MB")
print(f" Precision: INT8 (1 byte per parameter)")
print("\n📈 MEMORY SAVINGS")
print("=" * 70)
savings_ratio = memory_stats['parameter_memory_mb'] / theoretical_memory_mb
savings_percent = (1 - 1/savings_ratio) * 100
savings_mb = memory_stats['parameter_memory_mb'] - theoretical_memory_mb
print(f" Compression ratio: {savings_ratio:.1f}x smaller")
print(f" Memory saved: {savings_mb:.2f} MB ({savings_percent:.1f}% reduction)")
print(f" Original: {memory_stats['parameter_memory_mb']:.2f} MB → Quantized: {theoretical_memory_mb:.2f} MB")
print("\n💡 Key Insight:")
print(f" INT8 quantization reduces memory by 4x (FP32→INT8)")
print(f" This enables: 4x larger models, 4x bigger batches, or 4x lower cost!")
print(f" Critical for edge devices with limited memory (mobile, IoT)")
print("\n✅ This is the power of quantization: same functionality, 4x less memory!")
demo_quantization_with_profiler()
# %% [markdown]
"""
## 6. Module Integration Test
Final validation that our quantization system works correctly across all components.
"""
# %% nbgrader={"grade": true, "grade_id": "test_module", "points": 20}
def test_module():
"""
Comprehensive test of entire quantization module functionality.
This final test runs before module summary to ensure:
- All quantization functions work correctly
- Model quantization preserves functionality
- Memory savings are achieved
- Module is ready for integration with TinyTorch
"""
print("🧪 RUNNING MODULE INTEGRATION TEST")
print("=" * 50)
# Run all unit tests
print("Running unit tests...")
test_unit_quantize_int8()
test_unit_dequantize_int8()
test_unit_quantized_linear()
test_unit_quantize_model()
test_unit_compare_model_sizes()
print("\nRunning integration scenarios...")
# Test realistic usage scenario
print("🔬 Integration Test: End-to-end quantization workflow...")
# Create a realistic model
model = Sequential(
Linear(784, 128), # MNIST-like input
ReLU(),
Linear(128, 64),
ReLU(),
Linear(64, 10) # 10-class output
)
# Initialize with realistic weights
for layer in model.layers:
if isinstance(layer, Linear):
# Xavier initialization
fan_in, fan_out = layer.weight.shape
std = np.sqrt(2.0 / (fan_in + fan_out))
layer.weight = Tensor(np.random.randn(fan_in, fan_out) * std)
layer.bias = Tensor(np.zeros(fan_out))
# Generate realistic calibration data
calibration_data = [Tensor(np.random.randn(1, 784) * 0.1) for _ in range(20)]
# Test original model
test_input = Tensor(np.random.randn(8, 784) * 0.1)
original_output = model.forward(test_input)
# Quantize the model
quantize_model(model, calibration_data)
# Test quantized model
quantized_output = model.forward(test_input)
# Verify functionality is preserved
assert quantized_output.shape == original_output.shape, "Output shape mismatch"
# Verify reasonable accuracy preservation
mse = np.mean((original_output.data - quantized_output.data) ** 2)
relative_error = np.sqrt(mse) / (np.std(original_output.data) + 1e-8)
assert relative_error < 0.1, f"Accuracy degradation too high: {relative_error:.3f}"
# Verify memory savings
# Create equivalent original model for comparison
original_model = Sequential(
Linear(784, 128),
ReLU(),
Linear(128, 64),
ReLU(),
Linear(64, 10)
)
for i, layer in enumerate(model.layers):
if isinstance(layer, QuantizedLinear):
# Restore original weights for comparison
original_model.layers[i].weight = dequantize_int8(
layer.q_weight, layer.weight_scale, layer.weight_zero_point
)
if layer.q_bias is not None:
original_model.layers[i].bias = dequantize_int8(
layer.q_bias, layer.bias_scale, layer.bias_zero_point
)
memory_comparison = compare_model_sizes(original_model, model)
assert memory_comparison['compression_ratio'] > 2.0, "Insufficient compression achieved"
print(f"✅ Compression achieved: {memory_comparison['compression_ratio']:.1f}×")
print(f"✅ Accuracy preserved: {relative_error:.1%} relative error")
print(f"✅ Memory saved: {memory_comparison['memory_saved_mb']:.1f}MB")
# Test edge cases
print("🔬 Testing edge cases...")
# Test constant tensor quantization
constant_tensor = Tensor([[1.0, 1.0], [1.0, 1.0]])
q_const, scale_const, zp_const = quantize_int8(constant_tensor)
assert scale_const == 1.0, "Constant tensor quantization failed"
# Test zero tensor
zero_tensor = Tensor([[0.0, 0.0], [0.0, 0.0]])
q_zero, scale_zero, zp_zero = quantize_int8(zero_tensor)
restored_zero = dequantize_int8(q_zero, scale_zero, zp_zero)
assert np.allclose(restored_zero.data, 0.0, atol=1e-6), "Zero tensor restoration failed"
print("✅ Edge cases handled correctly!")
print("\n" + "=" * 50)
print("🎉 ALL TESTS PASSED! Module ready for export.")
print("📈 Quantization system provides:")
print(f"{memory_comparison['compression_ratio']:.1f}× memory reduction")
print(f" • <{relative_error:.1%} accuracy loss")
print(f" • Production-ready INT8 quantization")
print("Run: tito module complete 17")
# Call the comprehensive test
test_module()
# %%
if __name__ == "__main__":
print("🚀 Running Quantization module...")
test_module()
print("✅ Module validation complete!")
# %% [markdown]
"""
## 🏁 Consolidated Quantization Classes for Export
Now that we've implemented all quantization components, let's create consolidated classes
for export to the tinytorch package. This allows milestones to use the complete quantization system.
"""
# %% nbgrader={"grade": false, "grade_id": "quantization_export", "solution": false}
#| export
class QuantizationComplete:
"""
Complete quantization system for milestone use.
Provides INT8 quantization with calibration for 4× memory reduction.
"""
@staticmethod
def quantize_tensor(tensor: Tensor) -> Tuple[Tensor, float, int]:
"""Quantize FP32 tensor to INT8."""
data = tensor.data
min_val = float(np.min(data))
max_val = float(np.max(data))
if abs(max_val - min_val) < 1e-8:
return Tensor(np.zeros_like(data, dtype=np.int8)), 1.0, 0
scale = (max_val - min_val) / 255.0
zero_point = int(np.round(-128 - min_val / scale))
zero_point = int(np.clip(zero_point, -128, 127))
quantized_data = np.round(data / scale + zero_point)
quantized_data = np.clip(quantized_data, -128, 127).astype(np.int8)
return Tensor(quantized_data), scale, zero_point
@staticmethod
def dequantize_tensor(q_tensor: Tensor, scale: float, zero_point: int) -> Tensor:
"""Dequantize INT8 tensor back to FP32."""
dequantized_data = (q_tensor.data.astype(np.float32) - zero_point) * scale
return Tensor(dequantized_data)
@staticmethod
def quantize_model(model, calibration_data: Optional[List[Tensor]] = None) -> Dict[str, any]:
"""
Quantize all Linear layers in a model.
Returns dictionary with quantization info and memory savings.
"""
quantized_layers = {}
original_size = 0
quantized_size = 0
# Iterate through model parameters
if hasattr(model, 'parameters'):
for i, param in enumerate(model.parameters()):
param_size = param.data.nbytes
original_size += param_size
# Quantize parameter
q_param, scale, zp = QuantizationComplete.quantize_tensor(param)
quantized_size += q_param.data.nbytes
quantized_layers[f'param_{i}'] = {
'quantized': q_param,
'scale': scale,
'zero_point': zp,
'original_shape': param.data.shape
}
return {
'quantized_layers': quantized_layers,
'original_size_mb': original_size / (1024 * 1024),
'quantized_size_mb': quantized_size / (1024 * 1024),
'compression_ratio': original_size / quantized_size if quantized_size > 0 else 1.0
}
@staticmethod
def compare_models(original_model, quantized_info: Dict) -> Dict[str, float]:
"""Compare memory usage between original and quantized models."""
return {
'original_mb': quantized_info['original_size_mb'],
'quantized_mb': quantized_info['quantized_size_mb'],
'compression_ratio': quantized_info['compression_ratio'],
'memory_saved_mb': quantized_info['original_size_mb'] - quantized_info['quantized_size_mb']
}
# Convenience functions for backward compatibility
def quantize_int8(tensor: Tensor) -> Tuple[Tensor, float, int]:
"""Quantize FP32 tensor to INT8."""
return QuantizationComplete.quantize_tensor(tensor)
def dequantize_int8(q_tensor: Tensor, scale: float, zero_point: int) -> Tensor:
"""Dequantize INT8 tensor back to FP32."""
return QuantizationComplete.dequantize_tensor(q_tensor, scale, zero_point)
def quantize_model(model, calibration_data: Optional[List[Tensor]] = None) -> Dict[str, any]:
"""Quantize entire model to INT8."""
return QuantizationComplete.quantize_model(model, calibration_data)
# %% [markdown]
"""
## 🤔 ML Systems Thinking: Quantization in Production
### Question 1: Memory Architecture Impact
You implemented INT8 quantization that reduces each parameter from 4 bytes to 1 byte.
For a model with 100M parameters:
- Original memory usage: _____ GB
- Quantized memory usage: _____ GB
- Memory bandwidth reduction when loading from disk: _____ ×
### Question 2: Quantization Error Analysis
Your quantization maps a continuous range to 256 discrete values (INT8).
For weights uniformly distributed in [-0.1, 0.1]:
- Quantization scale: _____
- Maximum quantization error: _____
- Signal-to-noise ratio approximately: _____ dB
### Question 3: Hardware Efficiency
Modern processors have specialized INT8 instructions (like AVX-512 VNNI).
Compared to FP32 operations:
- How many INT8 operations fit in one SIMD instruction vs FP32? _____ × more
- Why might actual speedup be less than this theoretical maximum? _____
- What determines whether quantization improves or hurts performance? _____
### Question 4: Calibration Strategy Trade-offs
Your calibration process finds optimal scales using sample data.
- Too little calibration data: Risk of _____
- Too much calibration data: Cost of _____
- Per-channel vs per-tensor quantization trades _____ for _____
### Question 5: Production Deployment
In mobile/edge deployment scenarios:
- When is 4× memory reduction worth <1% accuracy loss? _____
- Why might you keep certain layers in FP32? _____
- How does quantization affect battery life? _____
"""
# %% [markdown]
"""
## 🎯 MODULE SUMMARY: Quantization
Congratulations! You've built a complete INT8 quantization system that can reduce model size by 4× with minimal accuracy loss!
### Key Accomplishments
- **Built INT8 quantization** with proper scaling and zero-point calculation
- **Implemented QuantizedLinear** layer with calibration support
- **Created model-level quantization** for complete neural networks
- **Analyzed quantization trade-offs** across different distributions and strategies
- **Measured real memory savings** and performance improvements
- All tests pass ✅ (validated by `test_module()`)
### Real-World Impact
Your quantization implementation achieves:
- **4× memory reduction** (FP32 → INT8)
- **2-4× inference speedup** (hardware dependent)
- **<1% accuracy loss** with proper calibration
- **Production deployment readiness** for mobile/edge applications
### What You've Mastered
- **Quantization mathematics** - scale and zero-point calculations
- **Calibration techniques** - optimizing quantization parameters
- **Error analysis** - understanding and minimizing quantization noise
- **Systems optimization** - memory vs accuracy trade-offs
### Ready for Next Steps
Your quantization system enables efficient model deployment on resource-constrained devices.
Export with: `tito module complete 17`
**Next**: Module 18 will add model compression through pruning - removing unnecessary weights entirely!
---
**🏆 Achievement Unlocked**: You can now deploy 4× smaller models with production-quality quantization! This is a critical skill for mobile AI, edge computing, and efficient inference systems.
"""