Solid Angle of A Cubemap Texel - 计算Cubemap的一个像素对应的立体角的大小

参考[http://www.rorydriscoll.com/2012/01/15/cubemap-texel-solid-angle/]

计算diffuse irradiance map或者求解sh系数的时候,需要对整个球面进行积分,由于cubemap不同位置的像素投影到球面上面积不同,所以不能平等的对待所有像素。

这时候就需要计算每个像素对应的solid angle。

如图,单位球位于原点。选取cubemap的一个面,假设其位于z=1的平面上。一个像素的立体角大小就是其投影到球面上的面积。

关于面积的求解分三步进行:

1. 计算平面上一点(x,y,1)投影到球面上后的坐标p'。

2. p'关于x方向和y方向的方向导数,两个方向导数叉乘的模长即为微面元的面积。

3. 对微面元积分,求得(0,0,1)-  (x,y,1)对应四边形的投影面积(立体角),然后利用这个公式就能求得一个像素的立体角。

-----------------------------------------------------------------------------------------------------------------------

1. p'就是一个放缩操作:

2. 两个方向导数:

 

这两个向量叉乘的结果:

 

然后叉乘的模长即为微面元的面积:

3. 积分,计算(0,0)到像素点(s,t)的立体角

4. 如果想计算四边形ABCD的立体角,则:S = f(A) - f(B) + f(C) - f(D)

--------------------------------------------------------------------------------------------------------------------------------------------

代码:

double sphereQuadrantArea(double x, double y) {
    return std::atan2(x*y, std::sqrt(x*x + y*y + 1));
}

double solidAngle(size_t dim, size_t u, size_t v) {
    const double iDim = 1.0f / dim;
    double s = ((u + 0.5) * 2*iDim) - 1;
    double t = ((v + 0.5) * 2*iDim) - 1;
    const double x0 = s - iDim;
    const double y0 = t - iDim;
    const double x1 = s + iDim;
    const double y1 = t + iDim;
    double solidAngle = sphereQuadrantArea(x0, y0) -
                        sphereQuadrantArea(x0, y1) -
                        sphereQuadrantArea(x1, y0) +
                        sphereQuadrantArea(x1, y1);
    return solidAngle;
}

  

 

posted @ 2019-06-04 21:03  redips  阅读(833)  评论(0编辑  收藏  举报