c++ - 成员结构位域元素的初始化列表初始化导致 IAR ARM 中的错误

标签 c++ arm bit-fields vtable iar

我在 IAR 中有以下类结构:

class A
{
public:
    A(){}
    virtual ~A() {};
    virtual void load() {};
};


class C
{
public:
    C()
    {
        //C does other stuff, not relevant
    }
};

class D;

class B : public A
{
public:
    B() : invert(false) {};
    virtual ~B() {};
    void load()
    {
        //Irrelevant stuff done here
    }
private:
    C member_c;
    std::vector<D*> vector_of_d;
    struct {
        bool var_1:1;
        bool var_2:1;
        bool var_3:1;
        bool var_4:1;
        bool invert:1;
    };
};

我遇到了为初始化 B 而生成的程序集的错误,它似乎对 VTable 指针的位置与匿名结构位域的位置感到“困惑”。当它将反转位设置为假时,它会转到对象的第一个字(即 VTable 指针)并翻转地址中的一位。当我稍后调用 load() 时,它跟随无效的 VTable 指针并最终找到一个空指针,然后盲目地跟随它。事情显然从那里分崩离析。

下面是一个会引发这个问题的代码示例:

void load_A(A* to_be_loaded){
    if(to_be_loaded) to_be_loaded->load();
}

int main(){
   load_A(new B());
}

现在最大的问题是,我是否不小心在某处引入了一些未定义的行为?这是从 GCC-ARM 移植过来的代码,它在那里运行良好,但现在突然在使用 IAR 编译时导致硬故障。我的两个理论是:

  • 这是编译器错误(我知道,这绝不是编译器错误)
  • GCC 作为扩展处理的非标准行为

据我所知,使用初始化列表来初始化匿名结构中的字段应该没有任何问题。我确实认识到匿名结构编译器扩展,但它们在 IAR 和 GCC 中都有记录。无论哪种方式,IAR 都没有给我任何警告或错误,并且正在生成明显损坏的程序集。

这是它为 B 构造函数制作的程序集

1 |    B() : invert(false) {};
2 |B::B():
3 |_ZN6BC1Ev:
4 |    0x80645e8: 0xb510         PUSH      {R4, LR}
5 |    0x80645ea: 0x4604         MOV       R4, R0
6 |    B() : invert(false) {};
7 |    0x80645ec: 0xf007 0xfb20  BL        A::subobject A() ; 0x806bc30
8 |    0x80645f0: 0x4807         LDR.N     R0, [PC, #0x1c]         ; 0x8088808 (134776840)
9 |    0x80645f2: 0x6020         STR       R0, [R4]
10|    0x80645f4: 0xf104 0x0018  ADD.W     R0, R4, #24             ; 0x18
11|    0x80645f8: 0xf00a 0xfadd  BL        C::C()              ; 0x806ebb6
12|    0x80645fc: 0xf104 0x001c  ADD.W     R0, R4, #28             ; 0x1c
13|    0x8064600: 0xf00e 0xff2e  BL        std::vector<D *>::vector() ; 0x8073460
14|    0x8064604: 0x7820         LDRB      R0, [R4]
15|    0x8064606: 0xf000 0x00ef  AND.W     R0, R0, #239            ; 0xef
16|    0x806460a: 0x7020         STRB      R0, [R4]
17|    B() : invert(false) {};
18|    0x806460c: 0x4620         MOV       R0, R4
19|    0x806460e: 0xbd10         POP       {R4, PC}
20|    0x8064610: 0x08088808     DC32      0x8088808 (134776840)

在第 14 行,我们加载 R4 指向的值,这是我们对象的基地址。它不对其应用任何偏移量,这意味着它指向对象中的第一个东西,即 VTable 指针。然后继续假定它具有位域并在第 15 行取消设置一位,然后将其放回从第 16 行获取它的对象中。

作为引用,如果我们将 B 的构造函数更改为使用初始化列表(如下所示),它将按预期工作:

class B : public A
{
public:
    B(){ invert = false; };
    virtual ~B() {};
    void load()
    {
        //Irrelevant stuff done here
    }
private:
    C member_c;
    std::vector<D*> vector_of_d;
    struct {
        bool var_1:1;
        bool var_2:1;
        bool var_3:1;
        bool var_4:1;
        bool invert:1;
    }
};

生成的程序集如下,注意第14行和第16行LDRBSTRB指令中使用的偏移量。这是访问正确的偏移量对象中的位域。

1 |    B(){ invert = false; };
2 |B::B():
3 |_ZN6BC1Ev:
4 |    0x80645e8: 0xb510         PUSH      {R4, LR}
5 |    0x80645ea: 0x4604         MOV       R4, R0
6 |    B(){ invert = false; };
7 |    0x80645ec: 0xf007 0xfb20  BL        A::subobject A() ; 0x806bc30
8 |    0x80645f0: 0x4807         LDR.N     R0, [PC, #0x20]         ; 0x8088808 (134776840)
9 |    0x80645f2: 0x6020         STR       R0, [R4]
10|    0x80645f4: 0xf104 0x0018  ADD.W     R0, R4, #24             ; 0x18
11|    0x80645f8: 0xf00a 0xfadd  BL        C::C()              ; 0x806ebb6
12|    0x80645fc: 0xf104 0x001c  ADD.W     R0, R4, #28             ; 0x1c
13|    0x8064600: 0xf00e 0xff2e  BL        std::vector<D *>::vector() ; 0x8073460
14|    0x8064604: 0x7820         LDRB      R0, [R4, #0x2c]
15|    0x8064606: 0xf000 0x00ef  AND.W     R0, R0, #239            ; 0xef
16|    0x806460a: 0x7020         STRB      R0, [R4, #0x2c]
17|    B(){ invert = false; };
18|    0x806460c: 0x4620         MOV       R0, R4
19|    0x806460e: 0xbd10         POP       {R4, PC}
20|    0x8064610: 0x08088808     DC32      0x8088808 (134776840)

旁注,第 8 行略有变化,但这可能是由于某些偏移量变化所致。

有没有人知道可能导致这种情况的原因?

最佳答案

这是一个编译器错误,根据我的调查,它至少会在 EWARM 7.80.1 和 8.11.2 中触发。它不会在 EWARM 8.20.1 中触发。该错误会在所有优化级别上触发,除了问题中提到的那个,我想不出其他解决方法。

关于c++ - 成员结构位域元素的初始化列表初始化导致 IAR ARM 中的错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47315289/

相关文章:

c++ - char vs wchar_t vs char16_t vs char32_t (c++11)

c++ - 使用 Cmake 在 Eclipse IDE 项目中索引 Qt5

c++ - 在 C++ 中显示为 "expression must have integral or enum type"的错误消息

c - 是否可以在单字节对齐结构中对齐特定结构成员?

embedded - Neon VLD 消耗的周期比预期多?

c - ARM 汇编程序 - 如何使用 CMP、BLT 和 BGT?

c++ - 结构位域最大大小(C99,C++)

c++ - 如何循环写入 vector ?

c - C 中的位域 - 对齐与初始化

c - 如何使用 union (位字段)处理自定义浮点?