zoukankan      html  css  js  c++  java
  • Solid Angle of A Cubemap Texel

    参考[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;
    }
    

      

  • 相关阅读:
    docker 学习
    grpc 学习
    ubuntu 完全干净的卸载docker
    numpy学习
    2020年假期sql excel文件 获取
    (a2b_hex)binascii.Error: Non-hexadecimal digit found
    数据库索引学习
    网络基础之网络协议
    Day11 进程相关
    基于socket套接字的网络通讯
  • 原文地址:https://www.cnblogs.com/redips-l/p/10976173.html
Copyright © 2011-2022 走看看