Awesome
<p align="center"> <img alt="polanalyser logo" src="documents/polanalyser_logo.png" height="90em"> </p>Polanalyser is polarization image analysis tool.
Key Features
- Demosaicing for polarimetric image sensor
- Both Monochrome/Color Polarization image sensors (e.g., IMX250MZR / MYR) are supported.
- Analysis of Stokes vector
- Obtain Stokes vector from images captured with a polarization camera or custom setup.
- Convert Stokes vector to meaningful parameters, such as DoLP, AoLP.
- Analysis of Mueller matrix
- Provide basic Mueller matrix elements, such as polarizer, retarder, and rotator.
- Obtain Mueller matrix from images captured under a variety of polarimetric conditions by using a least-squares method.
- Visualizing polarimetric images
- Apply colormap to polarization images, such as DoLP, AoLP, ToP, and CoP.
- Visualize the Mueller matrix image in grid form.
- Symbolic Stokes-Mueller computation
- Symbolic calculation of the Stokes vector and Mueller matrix to understand complex combinations of optical elements.
Dependencies and Installation
- Numpy
- OpenCV
- matplotlib
- SymPy (optional)
pip install polanalyser
Polarization Image Dataset
Dataset of images captured by a polarization camera (FLIR, BFS-U3-51S5P-C) is available. You can use these images to learn how to analyze polarization images.
[Click here to download the dataset (Google Drive)]
Usage
Polarization demosaicing
Demosaic raw polarization image captured with a polarization sensor (e.g., IMX250MZR / MYR).
<picture> <source media="(prefers-color-scheme: dark)" srcset="documents/demosaicing_dark.png"> <source media="(prefers-color-scheme: light)" srcset="documents/demosaicing_light.png"> <img alt="demosaicing" src="documents/demosaicing_light.png"> </picture>import cv2
import polanalyser as pa
img_raw = cv2.imread("dataset/dragon.png", 0)
img_000, img_045, img_090, img_135 = pa.demosaicing(img_raw, pa.COLOR_PolarMono)
# img_000, img_045, img_090, img_135 = pa.demosaicing(img_raw, pa.COLOR_PolarRGB)
Analysis of Stokes vector
Stokes vector describes the polarization states. We can measure these values by using a linear polarizer (To measure the circular polarization $s_3$, we also need to use a retarder).
<picture> <source media="(prefers-color-scheme: dark)" srcset="documents/stokes_setup_dark.png"> <source media="(prefers-color-scheme: light)" srcset="documents/stokes_setup_light.png"> <img alt="Stokes setup" src="documents/stokes_setup_light.png"> </picture>Stokes vector can be converted to meaningful values. Degree of Linear Polarization (DoLP) represents how much the light is polarized. The value is 1 for perfectly polarized light and 0 for unpolarized light. Angle of Linear Polarization (AoLP) represents the polarization angle of the incident light relative to the camera sensor axis. The value ranges from 0 to 180 degrees.
import cv2
import numpy as np
import polanalyser as pa
# Read image and demosaicing
img_raw = cv2.imread("dataset/dragon.png", 0)
img_000, img_045, img_090, img_135 = pa.demosaicing(img_raw, pa.COLOR_PolarMono)
# Calculate the Stokes vector per-pixel
image_list = [img_000, img_045, img_090, img_135]
angles = np.deg2rad([0, 45, 90, 135])
img_stokes = pa.calcStokes(image_list, angles)
# Decompose the Stokes vector into its components
img_s0, img_s1, img_s2 = cv2.split(img_stokes)
# Convert the Stokes vector to Intensity, DoLP and AoLP
img_intensity = pa.cvtStokesToIntensity(img_stokes)
img_dolp = pa.cvtStokesToDoLP(img_stokes)
img_aolp = pa.cvtStokesToAoLP(img_stokes)
Example of results | ||
---|---|---|
Intensity (s0) | DoLP | AoLP |
Analysis of Mueller matrix
Mueller matrix represents the change of the polarization state of light. The matrix size is 4x4 (When we consider only linear polarization, the size is 3x3).
Polanalyzer provides basic Mueller matrix elements in numpy array. For example, the following code shows the Mueller matrix of a linear polarizer and a quarter-wave plate.
import polanalyser as pa
# Linear polarizer
M_LP = pa.polarizer(0)
print(M_LP)
# [[0.5 0.5 0. 0. ]
# [0.5 0.5 0. 0. ]
# [0. 0. 0. 0. ]
# [0. 0. 0. 0. ]]
# Quarter-wave plate
M_QWP = pa.qwp(0)
print(M_QWP)
# [[ 1. 0. 0. 0.]
# [ 0. 1. 0. 0.]
# [ 0. 0. 0. 1.]
# [ 0. 0. -1. 0.]]
We can measure the unknown Mueller matrix by changing the polarization state of both the light and the detector (a.k.a. ellipsometry). The following figure shows a schematic diagram to measure the unknown Mueller matrix $\mathbf{M}
$
$I$ denotes the intensity of the unpolarized light source. $\mathbf{M}_\textrm{PSG}
$ and $\mathbf{M}_\textrm{PSA}
$ represent the Polarization state generator and analyzer (PSG and PSA) in Mueller matrix form. PSG and PSA are commonly composed of the basic optical elements (i.e., linear polarizer and retarder).
The detector measures the intensity $f$ expressed by $f = [ \mathbf{M}_\textrm{PSA} \mathbf{M} \mathbf{M}_\textrm{PSG} I]_{00}
$. $[...]_{00}$ extracts the (0, 0) component of the matrix.
Measuring $f$ by changing many combinations of $\mathbf{M}_\textrm{PSG}
$ and $\mathbf{M}_\textrm{PSA}
$ can estimate the unknown Mueller matrix $\mathbf{M}
$ with a linear least-squares method.
The following code shows the example to estimate the 3x3 Mueller matrix image.
import polanalyser as pa
# Read 16 images and Mueller matrices of PSG and PSA
filepath = "dataset/toy_example_3x3_pc"
images, props = pa.imreadMultiple(filepath)
print(images.shape) # (16, 2048, 2448)
mm_psg = props["mueller_psg"] # (16, 3, 3)
mm_psa = props["mueller_psa"] # (16, 3, 3)
# Calculate Mueller matrix image
img_mueller = pa.calcMueller(images, mm_psg, mm_psa)
print(img_mueller.shape) # (2048, 2448, 3, 3)
Visualizing polarimetric images
Stokes vector visualization
Polanalyser provides functions to visualize Stokes vector images, such as AoLP, DoLP, ToP (Type of Polarization), and CoP (Chirality of Polarization). The color mapping is designed based on the relevant papers [Wilkie and Weidlich, SCCG2010], [Baek+, SIGGRAPH2020], [Jeon+, CVPR2024]. Note that this mapping is slightly different from the original papers.
# Example of visualization functions
img_aolp_vis = pa.applyColorToAoLP(img_aolp)
img_dolp_vis = pa.applyColorToDoP(img_dolp)
img_top_vis = pa.applyColorToToP(img_ellipticity_angle, img_dop)
img_cop_vis = pa.applyColorToCoP(img_ellipticity_angle)
Here is an example of visualizing the Stokes vector images. The stokes image is borrowed from the spectro-polarimetric dataset [Jeon+, CVPR2024].
s0 | s1 | s2 | s3 |
DoLP | AoLP | AoLP (light) | AoLP (dark) |
DoP | DoCP | ToP | CoP |
In AoLP visualization, Polanalyser provides three types of AoLP visualization: AoLP, AoLP (light), and AoLP (dark). For more details, see the wiki page.
Mueller matrix visualization
Polanalyser provides functions to apply a colormap and make a 3x3 or 4x4 grid to visualize the Mueller matrix image.
Before visualizing the Mueller matrix image, we need to normalize the Mueller matrix. Here are three possible options, each with pros and cons. You need to choose the appropriate normalization method according to the purpose of the visualization and the chosen colormap.
# Normalize Mueller matrix image
# img_mueller: (H, W, 3, 3)
# Option 1: Normalize by the maximum value
# Pros: Show values linearly
# Cons: The small values may not be visible
img_mueller_maxnorm = img_mueller / np.abs(img_mueller).max()
# Option 2: Gamma correction
# Pros: Enhance the small values
# Cons: The large values become saturated
img_mueller_gamma = pa.gammaCorrection(img_mueller_maxnorm)
# Option 3: m00 norm (scale by m00 value of each pixel)
# Pros: Visualizes polarization components ratio independently of the intensity
# Cons: m00 becomes 1, and cannot represent the absolute intensity
img_mueller_m00norm = img_mueller / img_mueller[..., 0, 0][..., None, None]
After normalizing the Mueller matrix image, we can apply a colormap and make a grid to visualize the Mueller matrix image.
# Apply colormap and make grid
img_mueller_norm_vis = pa.applyColorMap(img_mueller_maxnorm, "RdBu", -1, 1) # (H, W, 3, 3, 3)
img_mueller_norm_vis_grid = pa.makeGridMueller(img_mueller_maxnorm_vis) # (H*3, W*3, 3)
Max norm | Max norm + Gamma | m00 norm |
---|---|---|
Symbolic Stokes-Mueller computation
This feature supports the symbolic computation of the Stokes vector and Mueller matrix powered by SymPy. This feature is particularly useful for understanding the effects of complex combinations of optical elements.
Here are examples of Malus's law and an ellipsometer. We can symbolically obtain the intensity of light passing through the sequence of optical elements without tedious calculations by hand.
from sympy import symbols, simplify
import polanalyser.sympy as pas
theta = symbols("theta", real=True)
# Example 1: Malus's law
M_L1 = pas.polarizer(0)
M_L2 = pas.polarizer(theta)
f = (M_L2 @ M_L1)[0, 0]
print(simplify(f))
# 0.5*cos(theta)**2
# Example 2: Ellipsometer [Azzam+, 1978][Baek+, 2020]
M = pas.mueller() # Symbolic Mueller matrix
I = 1.0
M_PSG = pas.qwp(theta) @ pas.polarizer(0)
M_PSA = pas.polarizer(0) @ pas.qwp(5 * theta)
f = (M_PSA @ M @ M_PSG * I)[0, 0]
print(simplify(f))
# 0.25*m00 + 0.25*m10*cos(10*theta)**2 + 0.125*m20*sin(20*theta) - 0.25*m30*sin(10*theta) + 0.25*(m01 + m11*cos(10*theta)**2 + m21*sin(20*theta)/2 - m31*sin(10*theta))*cos(2*theta)**2 + 0.25*(m02 + m12*cos(10*theta)**2 + m22*sin(20*theta)/2 - m32*sin(10*theta))*sin(2*theta)*cos(2*theta) + 0.25*(m03 + m13*cos(10*theta)**2 + m23*sin(20*theta)/2 - m33*sin(10*theta))*sin(2*theta)
Notations
Please refer to the notations.md for the definition of the Stokes vector and Mueller matrix used in Polanalyser.
Citation
If you find Polanalyser useful, please consider citing as follows:
@software{maeda2019polanalyser,
author = {Ryota Maeda},
title = {Polanalyser: Polarization Image Analysis Tool},
url = {https://github.com/elerac/polanalyser},
year = {2019},
}
Related Project
If you like Polanalyser, you are also interested in the following paper.
- Ryota Maeda, Shinsaku Hiura, Polarimetric Light Transport Analysis for Specular Inter-reflection, IEEE Transactions on Computational Imaging, 2024. [Project Page]