- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
This question询问是否可以依靠编译器不弄乱 struct
的值顺序和填充。
根据该问题的答案,
OpenGL defines, very clearly, what the byte layout of a
std140
interface block is.C++11 defines a concept called "standard layout types".
The only things C++ tells you about standard layout types with regard to layout is that empty base classes are ignored (so long as it remains standard layout) and that the first NSDM will be at the very beginning of the class. That is, there will never be padding at the front.
The other thing the standard says is that NSDMs of the same access class will be allocated in order, with later ones having larger offsets than earlier ones.
But that's it, as far as the C++ standard is concerned. [class.mem]/13 states that implementations can add padding between members for various reasons.
可能但并不总是存在的填充确实会把事情搞砸,最糟糕的是 - 这取决于编译器。
为了避免错误和噩梦,使用与编译器无关的方法不是更好吗?
例如:
class BufferData
{
private:
GLfloat data[12];
public:
GLfloat* getCameraPosition()
{
return (GLfloat*) &data[0];
}
GLfloat* getLightPosition()
{
return (GLfloat*) &data[4];
}
GLfloat* getLightDiffuse()
{
return (GLfloat*) &data[8];
}
GLfloat* getData()
{
return data;
}
};
相对于天真:
struct BufferData
{
GLfloat camera_position[4];
GLfloat light_position[4];
GLfloat light_diffuse[4];
};
还是天真的方法就足够好了?
(让我们假设类/结构不仅如此,而且可能会改变)
最佳答案
“与编译器无关”?没有这种动物。你写一个的尝试证明了这一点。考虑您的结构成员定义:
GLfloat data[12];
这需要存在 GLfloat
类型。但问题是,C++ 没有定义那种类型。 OpenGL 可以。
OpenGL 非常清楚地定义了该类型:它是一个 IEEE-754 floating-point type, using the BINARY32 format .
事实是,C++ 不要求 float
符合这一点。事实上,C++ 并不要求它的任何类型都符合这一点。如果编译器想让 float
使用 IEEE-754 以外的东西,那很好。
现在,您可能会说 OpenGL header 可以将 GLfloat
定义为一个类类型,大小为 32 位,它将从编译器的 float
类型转换为IEEE-754。当然,这可能会发生……除非没有办法获得 32 位值。
有 9 位字节的系统。或 18 位字节。这些系统有 C++ 编译器。这样的系统不能声明只有 32 位大小的类型。
但是能够传递 32 位值(更不用说 16 位和 8 位)是 OpenGL 的一个硬性要求。没有它,您将无法在缓冲区对象中传递任何数据。然而,C++ 不需要它。
说到顶点数据,半现代 OpenGL 中最基本的功能之一是 glVertexAttribPointer
。它依赖于您将字节偏移量转换为 void*
,然后它会转换回来。
C++ 不保证这有效。 C++ 标准中没有任何地方要求如果你将一个整数转换为一个指针,然后将该指针转换回一个整数,你将得到相同的整数(它确实说 ptr->int->ptr 有效,但是这并不意味着相反)。
然而 OpenGL 需要它。除非您使用单独的属性缓冲区(和 I strongly suggest you do if it's available ),否则您的代码和您调用的 OpenGL 代码都依赖于这种未定义的行为。
OpenGL 将 GLint
定义为带符号的 32 位补码。但是 C++ 不要求任何整数类型是二进制补码。
但是 OpenGL 可以。
OpenGL 无法在类型大小不正常的系统上运行。它不能在具有 9 位字节的系统上运行。它不能在使用一个补码进行有符号整数数学运算的系统上运行。我可以继续这样做,但我认为我的观点很明确。
通过选择完全使用 OpenGL(如果您想知道,也可以选择 Vulkan),您已经依赖于实现定义的行为。那么,当您已经依赖大量其他实现定义的行为时,为什么还要让您的生活更加困难来避免这种特定的实现定义行为呢?
马出了谷仓;现在关门无济于事。
关于c++ - 如何制作可移植且与编译器无关的 glBufferData?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38429628/
我的问题由两部分组成。 我注意到使用 cc 编译器的 sparc(sun) 上的 memalign(block_size,bytes) 不检查字节是否为 2 的幂,这与使用 mvsc 编译器的 int
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 6 年前。
当我尝试在我的 gwt-maven Projekt 上进行 maven-install 时,我得到了这个错误: [ERROR] Failed to execute goal org.apache.ma
gcc 有一个选项 -s 来生成汇编源代码。 csc(MS C# 编译器)或 dmcs(mono C# 编译器)是否等价?我的意思是那些编译器是否提供了一个选项来生成可以读取而不是执行二进制文件的 I
我在 matlab simulink 中有一个模型。我把matlab安装在D盘了。当我运行模型时,出现以下错误: Unable to locate a C-compiler required by S
我非常喜欢 Visual Studio 2012,因为 GUI 非常快速和灵活。问题是我需要 VS2010 的 VC++-Compiler。在 VS 2012 中设置旧的编译器、SDK 有什么可能吗?
就目前情况而言,这个问题不太适合我们的问答形式。我们希望答案得到事实、引用资料或专业知识的支持,但这个问题可能会引发辩论、争论、民意调查或扩展讨论。如果您觉得这个问题可以改进并可能重新开放,visit
我正在为类似 C 的语言开发编译器,但在语义分析和代码生成阶段遇到了一些困难。我的问题如下:1) 对于 if 语句,语法如下: if (expression) then statement1; sta
我想了解 php 编译器/解释器的工作原理。 我试图下载 php 源代码并试图了解它是如何工作的。我找不到合适的文档。如果有人可以阐明制作 php 编译器的模块以及 apache 服务器如何使用 ph
我有一些关于 python 的问题 为什么没有 python 编译器来创建本地代码?我找到了 py2exe 等,但它们只是随附了一个 python 解释器,因此,它又是执行代码的解释器。 是否无法创建
本文将是JVM 性能优化系列的第二篇文章(第一篇:传送门),Java 编译器将是本文讨论的核心内容。 本文中,作者(Eva Andreasson)首先介绍了不同种类的编译器,并对客户端编译,服务器
在 *nix 之类的系统或适当的工具包下是否有任何用于 ActionScript 3 的编译器来处理 Flash? 最佳答案 Flex SDK编译器 — mxmlc — 还将编译普通的 ActionS
我正在做一个C项目。但是其他人告诉我,由于没有C++编译器,所以无法构建它。 我不知道如何禁用C++的检测。这该怎么做? 最佳答案 检测C和C++工具链是CMake的默认行为。要禁用此行为,您需要手动
我正在寻找可以嵌入到我的程序中的 JIT 编译器或小型编译器库。我打算用它来编译动态生成的执行复数运算的代码。生成的代码在结构上非常简单:没有循环,没有条件,但它们可能很长(由 GCC 编译时只有几
多年来,我一直在 VB.NET 中使用 DEBUG 编译器常量将消息写入控制台。我也一直在以类似的方式使用 System.Diagnostics.Debug.Write。我一直认为,当 RELEASE
我了解编译器的前端和后端结构。但是,我不确定为什么编译器经常分为前端和后端。我相信有很多原因,你能给我几个吗?因为,大多数书籍/网站会告诉您它们是什么,但无法告诉您原因! 谢谢你。 最佳答案 前端处理
我有很多 JS 文件。其中一些相互依赖。其中许多依赖于 jQuery。我需要一种工具,它可以接受一个文件作为参数,传递地获取其所有依赖项,并以正确的顺序将它们编译成一个文件(基于依赖项) 依赖信息并不
我正在阅读著名的紫龙书第二版,但无法从第 65 页获取有关创建第一组的示例: 我们有以下语法(终端以粗体显示): stmt → expr; | if ( expr ) stmt | for ( opt
我正在寻找将 C# 语法编译为 native 代码(或者可能编译为 C++?)的选项。我对拥有正式成为该语言一部分的所有库不感兴趣,只是能够像编写 C++ 程序一样编写程序,但使用语言结构,例如部分类
编译器(例如:gcc)中的 -march 标志真的很重要吗? 如果我使用 -march=my_architecture 而不是 -march=i686 编译所有程序和内核,会不会更快 最佳答案 是的,
我是一名优秀的程序员,十分优秀!