// { dg-options "-std=gnu++17" } // { dg-do run { target c++17 } } // Copyright (C) 2013-2019 Free Software Foundation, Inc. // // This file is part of the GNU ISO C++ Library. This library is free // software; you can redistribute it and/or modify it under the // terms of the GNU General Public License as published by the // Free Software Foundation; either version 3, or (at your option) // any later version. // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // You should have received a copy of the GNU General Public License along // with this library; see the file COPYING3. If not see // . #include #include #include #include struct tracker { tracker(int value) : value(value) { ++count; } ~tracker() { --count; } tracker(tracker const& other) : value(other.value) { ++count; } tracker(tracker&& other) : value(other.value) { other.value = -1; ++count; } tracker& operator=(tracker const&) = default; tracker& operator=(tracker&&) = default; int value; static int count; }; int tracker::count = 0; struct exception { }; struct throwing_construction { explicit throwing_construction(bool propagate) : propagate(propagate) { } throwing_construction(throwing_construction const& other) : propagate(other.propagate) { if(propagate) throw exception {}; } bool propagate; }; int main() { // [20.5.4.1] Constructors { auto i = 0x1234ABCD; std::optional o { i }; VERIFY( o ); VERIFY( *o == 0x1234ABCD ); VERIFY( i == 0x1234ABCD ); } { auto i = 0x1234ABCD; std::optional o = i; VERIFY( o ); VERIFY( *o == 0x1234ABCD ); VERIFY( i == 0x1234ABCD ); } { auto i = 0x1234ABCD; std::optional o = { i }; VERIFY( o ); VERIFY( *o == 0x1234ABCD ); VERIFY( i == 0x1234ABCD ); } { auto i = 0x1234ABCD; std::optional o { std::move(i) }; VERIFY( o ); VERIFY( *o == 0x1234ABCD ); VERIFY( i == 0x1234ABCD ); } { auto i = 0x1234ABCD; std::optional o = std::move(i); VERIFY( o ); VERIFY( *o == 0x1234ABCD ); VERIFY( i == 0x1234ABCD ); } { auto i = 0x1234ABCD; std::optional o = { std::move(i) }; VERIFY( o ); VERIFY( *o == 0x1234ABCD ); VERIFY( i == 0x1234ABCD ); } { std::vector v = { 0, 1, 2, 3, 4, 5 }; std::optional> o { v }; VERIFY( !v.empty() ); VERIFY( o->size() == 6 ); } { std::vector v = { 0, 1, 2, 3, 4, 5 }; std::optional> o = v; VERIFY( !v.empty() ); VERIFY( o->size() == 6 ); } { std::vector v = { 0, 1, 2, 3, 4, 5 }; std::optional> o { v }; VERIFY( !v.empty() ); VERIFY( o->size() == 6 ); } { std::vector v = { 0, 1, 2, 3, 4, 5 }; std::optional> o { std::move(v) }; VERIFY( v.empty() ); VERIFY( o->size() == 6 ); } { std::vector v = { 0, 1, 2, 3, 4, 5 }; std::optional> o = std::move(v); VERIFY( v.empty() ); VERIFY( o->size() == 6 ); } { std::vector v = { 0, 1, 2, 3, 4, 5 }; std::optional> o { std::move(v) }; VERIFY( v.empty() ); VERIFY( o->size() == 6 ); } { tracker t { 333 }; std::optional o = t; VERIFY( o->value == 333 ); VERIFY( tracker::count == 2 ); VERIFY( t.value == 333 ); } { tracker t { 333 }; std::optional o = std::move(t); VERIFY( o->value == 333 ); VERIFY( tracker::count == 2 ); VERIFY( t.value == -1 ); } enum outcome { nothrow, caught, bad_catch }; { outcome result = nothrow; throwing_construction t { false }; try { std::optional o { t }; } catch(exception const&) { result = caught; } catch(...) { result = bad_catch; } VERIFY( result == nothrow ); } { outcome result = nothrow; throwing_construction t { true }; try { std::optional o { t }; } catch(exception const&) { result = caught; } catch(...) { result = bad_catch; } VERIFY( result == caught ); } { outcome result = nothrow; throwing_construction t { false }; try { std::optional o { std::move(t) }; } catch(exception const&) { result = caught; } catch(...) { result = bad_catch; } VERIFY( result == nothrow ); } { outcome result = nothrow; throwing_construction t { true }; try { std::optional o { std::move(t) }; } catch(exception const&) { result = caught; } catch(...) { result = bad_catch; } VERIFY( result == caught ); } { std::optional os = "foo"; struct X { explicit X(int) {} X& operator=(int) {return *this;} }; std::optional ox{42}; std::optional oi{42}; std::optional ox2{oi}; std::optional os2; os2 = "foo"; std::optional ox3; ox3 = 42; std::optional ox4; ox4 = oi; } { std::optional oi = std::optional(); VERIFY(!bool(oi)); std::optional os = std::optional(); VERIFY(!bool(os)); std::optional> ooi = std::optional(); VERIFY(bool(ooi)); ooi = std::optional(); VERIFY(bool(ooi)); ooi = std::optional(42); VERIFY(bool(ooi)); VERIFY(bool(*ooi)); std::optional> ooi2 = std::optional(); VERIFY(bool(ooi2)); ooi2 = std::optional(); VERIFY(bool(ooi2)); ooi2 = std::optional(6); VERIFY(bool(ooi2)); VERIFY(bool(*ooi2)); std::optional> ooi3 = std::optional(42); VERIFY(bool(ooi3)); VERIFY(bool(*ooi3)); std::optional> ooi4 = std::optional(6); VERIFY(bool(ooi4)); VERIFY(bool(*ooi4)); } }