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 Klecknervoid 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