ポイントライトの減衰

提供: MonoBook
2021年8月20日 (金) 01:21時点におけるAdministrator (トーク | 投稿記録)による版 (→‎CGでの問題点)
(差分) ← 古い版 | 最新版 (差分) | 新しい版 → (差分)
ナビゲーションに移動 検索に移動

現実世界[編集 | ソースを編集]

現実世界の照明の減衰は逆二乗の法則に従っている。

attenuation = 1 / pow(x,2)

CGでの問題点[編集 | ソースを編集]

しかしコンピュータグラフィックス(CG)では少し厄介なことになります。

現実世界の照明の減衰は距離に応じて指数関数的に小さくなるため、 モデルとライトの距離がゼロ付近になると無限大(infinity)となり(浮動小数点数型オーバーフローを引き起こし)、 そのまま計算を続けるとゼロ除算を引き起こし、 だいたいの場合は真っ暗になる。

また、現実世界の豆電球は「大きさ」を持っており「点」ではなく「面」であるといえ、中心部分の一定面積は減衰がほぼない。一方でCGで使われるポイントライトは「点」で実装されていることが多く、不自然な表現になりがちである。

この問題を近似解で解決する複数の方法が考案されている。

Unity[編集 | ソースを編集]

Unityのドキュメントによると標準パイプラインでは以下の数式を使っているらしい。

r = LightRange
attenuation = 1 / ( pow( ( x / r ) * 5, 2) + 1 )

逆二乗曲線に似ているが、距離を「5」に固定し、LightRangeパラメータでスケーリングしている。 これは中央でちょうど1になるという素晴らしい特性を持っている。

Bakery[編集 | ソースを編集]

「Bakery - GPU Lightmapper」のパイプラインでは以下の数式を使っているらしい。 常に「+1」することでゼロ除算になることがないという。

attenuation = 1 / ( pow(x, 2) + 1 )

Frostbite[編集 | ソースを編集]

バトルフィールドシリーズで有名なEA DICEゲームエンジンFrostbite」のパイプラインでは以下の数式を使っているらしい。

s = LightSize = 0.01 = 1cm
attenuation = 1 / pow(max(x, s), 2)

Bakery改[編集 | ソースを編集]

Bakery + Frostbite

s = LightSize
attenuation = 1 / ( pow(x,2) + pow(s,2) )