osg中相机矩阵到vsg相机矩阵的转换

前言

在OSG和VSG中,相机的核心功能主要包括视图变换与投影变换:视图变换负责将世界坐标系中的物体转换到相机坐标系,而投影变换则进一步将相机坐标系中的物体映射到投影空间。由于OpenGL与Vulkan在底层实现上存在差异,Vulkan的深度范围定义为[0, 1],且y轴方向由上向下,这与OpenGL有所不同。因此,本章将重点讨论从OSG相机矩阵到VSG相机矩阵的转换方法,并进一步讨论投影矩阵之间的转换。


目录

  • 1 vsg相机矩阵到CesiumNative间的矩阵转换
  • 2 osg中的相机矩阵到vsg相机矩阵的转换

1 vsg相机矩阵到CesiumNative间的矩阵转换

vsgCs(https://github.com/timoore/vsgCs/ v1.1.1)中,CesiumNative采用glm::dmat4数据结构来表达矩阵,可通过数值拷贝的方式可实现vsg中view矩阵和投影矩阵间的转换。如下所示:

cpp 复制代码
    std::optional<Cesium3DTilesSelection::ViewState>
    createViewState(const vsg::ref_ptr<vsg::View>& view, const vsg::ref_ptr<vsg::RenderGraph>& renderGraph)
    {
        auto* viewData = dynamic_cast<ViewData*>(view->getObject("vsgCsViewData"));
        if (!viewData)
        {
            return {};
        }
        vsg::dmat4 Pw = vsg::computeTransform(viewData->tilesetPath);
        vsg::dmat4 PcsInv = TilesetNode::yUp2zUp * view->camera->viewMatrix->transform() * Pw;
        vsg::dmat4 Pcs = vsg::inverse(PcsInv);
        glm::dvec3 position(Pcs[3][0], Pcs[3][1], Pcs[3][2]);
        glm::dvec3 direction(Pcs[1][0], Pcs[1][1], Pcs[1][2]);
        glm::dvec3 up(Pcs[2][0], Pcs[2][1], Pcs[2][2]);
        vsg::ref_ptr<vsg::ProjectionMatrix> projMat = view->camera->projectionMatrix;
        glm::dvec2 viewportSize;
        if (view->camera->viewportState)
        {
            VkViewport viewport = view->camera->viewportState->getViewport();
            viewportSize[0] = viewport.width;
            viewportSize[1] = viewport.height;
        }
        else
        {
            viewportSize[0] = renderGraph->renderArea.extent.width;
            viewportSize[1] = renderGraph->renderArea.extent.height;
        }
        Cesium3DTilesSelection::ViewState result =
            Cesium3DTilesSelection::ViewState(vsg2glm(view->camera->viewMatrix->transform()), vsg2glm(projMat->transform()), viewportSize);
        return {result};
    }

2 osg中的相机矩阵到vsg相机矩阵的转换

osg中采用行优先形式存储矩阵,而vsg中采用列优先形式存储矩阵,以投影矩阵为例,透视投影下两者的计算公式分别如下:

|----------------------------------------------------------------------------|
| |
| osg投影矩阵 |
| |
| vsg投影矩阵 |

从公式上看,值-1.0,两者在同一个位置,但却表达的含义不同,在osg中-1.0所在位置为第三行第四列,而在vsg中为第四行第散列。

osg矩阵到vsg矩阵的转换代码如下所示:

cpp 复制代码
vsg::dmat4 osg2vsg(const osg::Matrix& osgMat)
{
    vsg::dmat4 result;
    std::memcpy(result.data(), osgMat.ptr(), sizeof(double) * 16);
    return result;
}

考虑到vsg中投影空间深度范围为0-1,且采用了reversed-z,Y轴从上到下,因此osg的投影矩阵到vsg的投影矩阵转换函数如下:

cpp 复制代码
vsg::dmat4 osg2vsg(const osg::Matrix& osgMat)
{
    osg::Matrix oToV(
        1.0, 0.0, 0.0, 0.0,
        0.0, -1.0, 0.0, 0.0,
        0.0, 0.0, -0.5, 0,
        0.0, 0.0, 0.5, 1.0
    );
    vsg::dmat4 result;
    std::memcpy(result.data(), (osgMat* oToV).ptr(), sizeof(double) * 16);
    return result;
}

文末:本章借助vsgCs中的代码分析了vsg矩阵到CesiumNative矩阵的转换方法,可直接通过值拷贝实现转换。接着讨论了osg到vsg的矩阵转换方法,考虑到了osg中矩阵为行优先,而vsg中矩阵为列优先,在此基础上进一步讨论了两者投影矩阵的转换方法,考虑到了vsg投影空间的投影深度为[1,0],且y轴由上向下的特点。

相关推荐
千里马-horse14 小时前
When and Why to use Extensions -- VK_KHR_descriptor_update_template
vulkan
zhooyu1 天前
利用叉乘判断OpenGL中的左右关系
c++·3d·opengl
火柴-人1 天前
我用 C++ 写了个 MCP ,让 AI 看懂了每一帧 GPU 在画什么
图形渲染·claude·codex·skill·vulkan·mcp·renderdoc
千里马-horse7 天前
Using Vulkan -- Atomics
vulkan
zhooyu13 天前
GLM中lerp实现线性插值
c++·opengl
智算菩萨14 天前
【OpenGL】10 完整游戏开发实战:基于OpenGL的2D/3D游戏框架、物理引擎集成与AI辅助编程指南
人工智能·python·游戏·3d·矩阵·pygame·opengl
智算菩萨14 天前
【OpenGL】6 真实感光照渲染实战:Phong模型、材质系统与PBR基础
开发语言·python·游戏引擎·游戏程序·pygame·材质·opengl
千里马-horse17 天前
Using Vulkan -- Mapping Data to Shaders -- Storage Image and Texel Buffers
着色器·vulkan·图像存储·纹理元素缓存区
梵尔纳多18 天前
视角的移动以及模型的平移,旋转,缩放
c++·图形渲染·opengl
千里马-horse20 天前
Linux 安装Vulkan, 在终端输入:vulkaninfo
vulkan·vulkaninfo