134 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			C++
		
	
	
		
		
			
		
	
	
			134 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			C++
		
	
	
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								    tests/test_enums.cpp -- enumerations
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    Copyright (c) 2016 Wenzel Jakob <wenzel.jakob@epfl.ch>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    All rights reserved. Use of this source code is governed by a
							 | 
						||
| 
								 | 
							
								    BSD-style license that can be found in the LICENSE file.
							 | 
						||
| 
								 | 
							
								*/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include "pybind11_tests.h"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								TEST_SUBMODULE(enums, m) {
							 | 
						||
| 
								 | 
							
								    // test_unscoped_enum
							 | 
						||
| 
								 | 
							
								    enum UnscopedEnum { EOne = 1, ETwo, EThree };
							 | 
						||
| 
								 | 
							
								    py::enum_<UnscopedEnum>(m, "UnscopedEnum", py::arithmetic(), "An unscoped enumeration")
							 | 
						||
| 
								 | 
							
								        .value("EOne", EOne, "Docstring for EOne")
							 | 
						||
| 
								 | 
							
								        .value("ETwo", ETwo, "Docstring for ETwo")
							 | 
						||
| 
								 | 
							
								        .value("EThree", EThree, "Docstring for EThree")
							 | 
						||
| 
								 | 
							
								        .export_values();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // test_scoped_enum
							 | 
						||
| 
								 | 
							
								    enum class ScopedEnum { Two = 2, Three };
							 | 
						||
| 
								 | 
							
								    py::enum_<ScopedEnum>(m, "ScopedEnum", py::arithmetic())
							 | 
						||
| 
								 | 
							
								        .value("Two", ScopedEnum::Two)
							 | 
						||
| 
								 | 
							
								        .value("Three", ScopedEnum::Three);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    m.def("test_scoped_enum", [](ScopedEnum z) {
							 | 
						||
| 
								 | 
							
								        return "ScopedEnum::" + std::string(z == ScopedEnum::Two ? "Two" : "Three");
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // test_binary_operators
							 | 
						||
| 
								 | 
							
								    enum Flags { Read = 4, Write = 2, Execute = 1 };
							 | 
						||
| 
								 | 
							
								    py::enum_<Flags>(m, "Flags", py::arithmetic())
							 | 
						||
| 
								 | 
							
								        .value("Read", Flags::Read)
							 | 
						||
| 
								 | 
							
								        .value("Write", Flags::Write)
							 | 
						||
| 
								 | 
							
								        .value("Execute", Flags::Execute)
							 | 
						||
| 
								 | 
							
								        .export_values();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // test_implicit_conversion
							 | 
						||
| 
								 | 
							
								    class ClassWithUnscopedEnum {
							 | 
						||
| 
								 | 
							
								    public:
							 | 
						||
| 
								 | 
							
								        enum EMode { EFirstMode = 1, ESecondMode };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        static EMode test_function(EMode mode) { return mode; }
							 | 
						||
| 
								 | 
							
								    };
							 | 
						||
| 
								 | 
							
								    py::class_<ClassWithUnscopedEnum> exenum_class(m, "ClassWithUnscopedEnum");
							 | 
						||
| 
								 | 
							
								    exenum_class.def_static("test_function", &ClassWithUnscopedEnum::test_function);
							 | 
						||
| 
								 | 
							
								    py::enum_<ClassWithUnscopedEnum::EMode>(exenum_class, "EMode")
							 | 
						||
| 
								 | 
							
								        .value("EFirstMode", ClassWithUnscopedEnum::EFirstMode)
							 | 
						||
| 
								 | 
							
								        .value("ESecondMode", ClassWithUnscopedEnum::ESecondMode)
							 | 
						||
| 
								 | 
							
								        .export_values();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // test_enum_to_int
							 | 
						||
| 
								 | 
							
								    m.def("test_enum_to_int", [](int) {});
							 | 
						||
| 
								 | 
							
								    m.def("test_enum_to_uint", [](uint32_t) {});
							 | 
						||
| 
								 | 
							
								    m.def("test_enum_to_long_long", [](long long) {});
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // test_duplicate_enum_name
							 | 
						||
| 
								 | 
							
								    enum SimpleEnum { ONE, TWO, THREE };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    m.def("register_bad_enum", [m]() {
							 | 
						||
| 
								 | 
							
								        py::enum_<SimpleEnum>(m, "SimpleEnum")
							 | 
						||
| 
								 | 
							
								            .value("ONE", SimpleEnum::ONE) // NOTE: all value function calls are called with the
							 | 
						||
| 
								 | 
							
								                                           // same first parameter value
							 | 
						||
| 
								 | 
							
								            .value("ONE", SimpleEnum::TWO)
							 | 
						||
| 
								 | 
							
								            .value("ONE", SimpleEnum::THREE)
							 | 
						||
| 
								 | 
							
								            .export_values();
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // test_enum_scalar
							 | 
						||
| 
								 | 
							
								    enum UnscopedUCharEnum : unsigned char {};
							 | 
						||
| 
								 | 
							
								    enum class ScopedShortEnum : short {};
							 | 
						||
| 
								 | 
							
								    enum class ScopedLongEnum : long {};
							 | 
						||
| 
								 | 
							
								    enum UnscopedUInt64Enum : std::uint64_t {};
							 | 
						||
| 
								 | 
							
								    static_assert(
							 | 
						||
| 
								 | 
							
								        py::detail::all_of<
							 | 
						||
| 
								 | 
							
								            std::is_same<py::enum_<UnscopedUCharEnum>::Scalar, unsigned char>,
							 | 
						||
| 
								 | 
							
								            std::is_same<py::enum_<ScopedShortEnum>::Scalar, short>,
							 | 
						||
| 
								 | 
							
								            std::is_same<py::enum_<ScopedLongEnum>::Scalar, long>,
							 | 
						||
| 
								 | 
							
								            std::is_same<py::enum_<UnscopedUInt64Enum>::Scalar, std::uint64_t>>::value,
							 | 
						||
| 
								 | 
							
								        "Error during the deduction of enum's scalar type with normal integer underlying");
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // test_enum_scalar_with_char_underlying
							 | 
						||
| 
								 | 
							
								    enum class ScopedCharEnum : char { Zero, Positive };
							 | 
						||
| 
								 | 
							
								    enum class ScopedWCharEnum : wchar_t { Zero, Positive };
							 | 
						||
| 
								 | 
							
								    enum class ScopedChar32Enum : char32_t { Zero, Positive };
							 | 
						||
| 
								 | 
							
								    enum class ScopedChar16Enum : char16_t { Zero, Positive };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // test the scalar of char type enums according to chapter 'Character types'
							 | 
						||
| 
								 | 
							
								    // from https://en.cppreference.com/w/cpp/language/types
							 | 
						||
| 
								 | 
							
								    static_assert(
							 | 
						||
| 
								 | 
							
								        py::detail::any_of<
							 | 
						||
| 
								 | 
							
								            std::is_same<py::enum_<ScopedCharEnum>::Scalar, signed char>,  // e.g. gcc on x86
							 | 
						||
| 
								 | 
							
								            std::is_same<py::enum_<ScopedCharEnum>::Scalar, unsigned char> // e.g. arm linux
							 | 
						||
| 
								 | 
							
								            >::value,
							 | 
						||
| 
								 | 
							
								        "char should be cast to either signed char or unsigned char");
							 | 
						||
| 
								 | 
							
								    static_assert(sizeof(py::enum_<ScopedWCharEnum>::Scalar) == 2
							 | 
						||
| 
								 | 
							
								                      || sizeof(py::enum_<ScopedWCharEnum>::Scalar) == 4,
							 | 
						||
| 
								 | 
							
								                  "wchar_t should be either 16 bits (Windows) or 32 (everywhere else)");
							 | 
						||
| 
								 | 
							
								    static_assert(
							 | 
						||
| 
								 | 
							
								        py::detail::all_of<
							 | 
						||
| 
								 | 
							
								            std::is_same<py::enum_<ScopedChar32Enum>::Scalar, std::uint_least32_t>,
							 | 
						||
| 
								 | 
							
								            std::is_same<py::enum_<ScopedChar16Enum>::Scalar, std::uint_least16_t>>::value,
							 | 
						||
| 
								 | 
							
								        "char32_t, char16_t (and char8_t)'s size, signedness, and alignment is determined");
							 | 
						||
| 
								 | 
							
								#if defined(PYBIND11_HAS_U8STRING)
							 | 
						||
| 
								 | 
							
								    enum class ScopedChar8Enum : char8_t { Zero, Positive };
							 | 
						||
| 
								 | 
							
								    static_assert(std::is_same<py::enum_<ScopedChar8Enum>::Scalar, unsigned char>::value);
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // test_char_underlying_enum
							 | 
						||
| 
								 | 
							
								    py::enum_<ScopedCharEnum>(m, "ScopedCharEnum")
							 | 
						||
| 
								 | 
							
								        .value("Zero", ScopedCharEnum::Zero)
							 | 
						||
| 
								 | 
							
								        .value("Positive", ScopedCharEnum::Positive);
							 | 
						||
| 
								 | 
							
								    py::enum_<ScopedWCharEnum>(m, "ScopedWCharEnum")
							 | 
						||
| 
								 | 
							
								        .value("Zero", ScopedWCharEnum::Zero)
							 | 
						||
| 
								 | 
							
								        .value("Positive", ScopedWCharEnum::Positive);
							 | 
						||
| 
								 | 
							
								    py::enum_<ScopedChar32Enum>(m, "ScopedChar32Enum")
							 | 
						||
| 
								 | 
							
								        .value("Zero", ScopedChar32Enum::Zero)
							 | 
						||
| 
								 | 
							
								        .value("Positive", ScopedChar32Enum::Positive);
							 | 
						||
| 
								 | 
							
								    py::enum_<ScopedChar16Enum>(m, "ScopedChar16Enum")
							 | 
						||
| 
								 | 
							
								        .value("Zero", ScopedChar16Enum::Zero)
							 | 
						||
| 
								 | 
							
								        .value("Positive", ScopedChar16Enum::Positive);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // test_bool_underlying_enum
							 | 
						||
| 
								 | 
							
								    enum class ScopedBoolEnum : bool { FALSE, TRUE };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // bool is unsigned (std::is_signed returns false) and 1-byte long, so represented with u8
							 | 
						||
| 
								 | 
							
								    static_assert(std::is_same<py::enum_<ScopedBoolEnum>::Scalar, std::uint8_t>::value, "");
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    py::enum_<ScopedBoolEnum>(m, "ScopedBoolEnum")
							 | 
						||
| 
								 | 
							
								        .value("FALSE", ScopedBoolEnum::FALSE)
							 | 
						||
| 
								 | 
							
								        .value("TRUE", ScopedBoolEnum::TRUE);
							 | 
						||
| 
								 | 
							
								}
							 |