xref: /llvm-project/clang/test/SemaCXX/warn-bitfield-enum-conversion.cpp (revision 329f24d6f6e733fcadfd1be7cd3b430d63047c2e)
1*329f24d6SReid Kleckner // RUN: %clang_cc1 -std=c++11 -triple x86_64-windows-msvc -verify %s -Wbitfield-enum-conversion
2*329f24d6SReid Kleckner // RUN: %clang_cc1 -std=c++11 -triple x86_64-linux -verify %s -Wbitfield-enum-conversion
3*329f24d6SReid Kleckner 
4*329f24d6SReid Kleckner enum TwoBits { Hi1 = 3 } two_bits;
5*329f24d6SReid Kleckner enum TwoBitsSigned { Lo2 = -2, Hi2 = 1 } two_bits_signed;
6*329f24d6SReid Kleckner enum ThreeBits { Hi3 = 7 } three_bits;
7*329f24d6SReid Kleckner enum ThreeBitsSigned { Lo4 = -4, Hi4 = 3 } three_bits_signed;
8*329f24d6SReid Kleckner enum TwoBitsFixed : unsigned { Hi5 = 3 } two_bits_fixed;
9*329f24d6SReid Kleckner 
10*329f24d6SReid Kleckner struct Foo {
11*329f24d6SReid Kleckner   unsigned two_bits : 2;        // expected-note 2 {{widen this field to 3 bits}} expected-note 2 {{type signed}}
12*329f24d6SReid Kleckner   int two_bits_signed : 2;      // expected-note 2 {{widen this field to 3 bits}} expected-note 1 {{type unsigned}}
13*329f24d6SReid Kleckner   unsigned three_bits : 3;      // expected-note 2 {{type signed}}
14*329f24d6SReid Kleckner   int three_bits_signed : 3;    // expected-note 1 {{type unsigned}}
15*329f24d6SReid Kleckner 
16*329f24d6SReid Kleckner #ifdef _WIN32
17*329f24d6SReid Kleckner   // expected-note@+2 {{type unsigned}}
18*329f24d6SReid Kleckner #endif
19*329f24d6SReid Kleckner   ThreeBits three_bits_enum : 3;
20*329f24d6SReid Kleckner   ThreeBits four_bits_enum : 4;
21*329f24d6SReid Kleckner };
22*329f24d6SReid Kleckner 
f()23*329f24d6SReid Kleckner void f() {
24*329f24d6SReid Kleckner   Foo f;
25*329f24d6SReid Kleckner 
26*329f24d6SReid Kleckner   f.two_bits = two_bits;
27*329f24d6SReid Kleckner   f.two_bits = two_bits_signed;            // expected-warning {{negative enumerators}}
28*329f24d6SReid Kleckner   f.two_bits = three_bits;                 // expected-warning {{not wide enough}}
29*329f24d6SReid Kleckner   f.two_bits = three_bits_signed;          // expected-warning {{negative enumerators}} expected-warning {{not wide enough}}
30*329f24d6SReid Kleckner   f.two_bits = two_bits_fixed;
31*329f24d6SReid Kleckner 
32*329f24d6SReid Kleckner   f.two_bits_signed = two_bits;            // expected-warning {{needs an extra bit}}
33*329f24d6SReid Kleckner   f.two_bits_signed = two_bits_signed;
34*329f24d6SReid Kleckner   f.two_bits_signed = three_bits;          // expected-warning {{not wide enough}}
35*329f24d6SReid Kleckner   f.two_bits_signed = three_bits_signed;   // expected-warning {{not wide enough}}
36*329f24d6SReid Kleckner 
37*329f24d6SReid Kleckner   f.three_bits = two_bits;
38*329f24d6SReid Kleckner   f.three_bits = two_bits_signed;          // expected-warning {{negative enumerators}}
39*329f24d6SReid Kleckner   f.three_bits = three_bits;
40*329f24d6SReid Kleckner   f.three_bits = three_bits_signed;        // expected-warning {{negative enumerators}}
41*329f24d6SReid Kleckner 
42*329f24d6SReid Kleckner   f.three_bits_signed = two_bits;
43*329f24d6SReid Kleckner   f.three_bits_signed = two_bits_signed;
44*329f24d6SReid Kleckner   f.three_bits_signed = three_bits;        // expected-warning {{needs an extra bit}}
45*329f24d6SReid Kleckner   f.three_bits_signed = three_bits_signed;
46*329f24d6SReid Kleckner 
47*329f24d6SReid Kleckner #ifdef _WIN32
48*329f24d6SReid Kleckner   // Enums on Windows are always implicitly 'int', which is signed, so you need
49*329f24d6SReid Kleckner   // an extra bit to store values that set the MSB. This is not true on SysV
50*329f24d6SReid Kleckner   // platforms like Linux.
51*329f24d6SReid Kleckner   // expected-warning@+2 {{needs an extra bit}}
52*329f24d6SReid Kleckner #endif
53*329f24d6SReid Kleckner   f.three_bits_enum = three_bits;
54*329f24d6SReid Kleckner   f.four_bits_enum = three_bits;
55*329f24d6SReid Kleckner 
56*329f24d6SReid Kleckner   // Explicit casts suppress the warning.
57*329f24d6SReid Kleckner   f.two_bits = (unsigned)three_bits_signed;
58*329f24d6SReid Kleckner   f.two_bits = static_cast<unsigned>(three_bits_signed);
59*329f24d6SReid Kleckner }
60