llvm/libcxx/test/std/time/time.cal/time.cal.ymwdlast/time.cal.ymwdlast.nonmembers/plus.pass.cpp

//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
// UNSUPPORTED: c++03, c++11, c++14, c++17

// <chrono>
// class year_month_weekday_last;

// constexpr year_month_weekday_last operator+(const year_month_weekday_last& ymwdl, const months& dm) noexcept;
//   Returns: (ymwdl.year() / ymwdl.month() + dm) / ymwdl.weekday_last().
//
// constexpr year_month_weekday_last operator+(const months& dm, const year_month_weekday_last& ymwdl) noexcept;
//   Returns: ymwdl + dm.
//
// constexpr year_month_weekday_last operator+(const year_month_weekday_last& ymwdl, const years& dy) noexcept;
//   Returns: {ymwdl.year()+dy, ymwdl.month(), ymwdl.weekday_last()}.
//
// constexpr year_month_weekday_last operator+(const years& dy, const year_month_weekday_last& ymwdl) noexcept;
//   Returns: ymwdl + dy.

#include <chrono>
#include <cassert>
#include <type_traits>
#include <utility>

#include "test_macros.h"

using year                    = std::chrono::year;
using month                   = std::chrono::month;
using weekday                 = std::chrono::weekday;
using weekday_last            = std::chrono::weekday_last;
using year_month_weekday_last = std::chrono::year_month_weekday_last;
using years                   = std::chrono::years;
using months                  = std::chrono::months;

constexpr bool test() {
  constexpr weekday Tuesday = std::chrono::Tuesday;
  constexpr month January   = std::chrono::January;

  { // year_month_weekday_last + months
    year_month_weekday_last ymwdl{year{1234}, January, weekday_last{Tuesday}};
    for (int i = 0; i <= 10; ++i) {
      year_month_weekday_last ymwdl1 = ymwdl + months{i};
      year_month_weekday_last ymwdl2 = months{i} + ymwdl;
      assert(ymwdl1.year() == year{1234});
      assert(ymwdl2.year() == year{1234});
      assert(ymwdl1.month() == month(1 + i));
      assert(ymwdl2.month() == month(1 + i));
      assert(ymwdl1.weekday() == Tuesday);
      assert(ymwdl2.weekday() == Tuesday);
      assert(ymwdl1.weekday_last() == weekday_last{Tuesday});
      assert(ymwdl2.weekday_last() == weekday_last{Tuesday});
      assert(ymwdl1 == ymwdl2);
    }
    // Test the year wraps around.
    for (int i = 12; i <= 15; ++i) {
      year_month_weekday_last ymwdl1 = ymwdl + months{i};
      year_month_weekday_last ymwdl2 = months{i} + ymwdl;
      assert(ymwdl1.year() == year{1235});
      assert(ymwdl2.year() == year{1235});
      assert(ymwdl1.month() == month(1 + i - 12));
      assert(ymwdl2.month() == month(1 + i - 12));
      assert(ymwdl1.weekday() == Tuesday);
      assert(ymwdl2.weekday() == Tuesday);
      assert(ymwdl1.weekday_last() == weekday_last{Tuesday});
      assert(ymwdl2.weekday_last() == weekday_last{Tuesday});
      assert(ymwdl1 == ymwdl2);
    }
  }

  { // year_month_weekday_last + years
    year_month_weekday_last ymwdl{year{1234}, std::chrono::January, weekday_last{Tuesday}};
    for (int i = 0; i <= 10; ++i) {
      year_month_weekday_last ymwdl1 = ymwdl + years{i};
      year_month_weekday_last ymwdl2 = years{i} + ymwdl;
      assert(ymwdl1.year() == year(1234 + i));
      assert(ymwdl2.year() == year(1234 + i));
      assert(ymwdl1.month() == January);
      assert(ymwdl2.month() == January);
      assert(ymwdl1.weekday() == Tuesday);
      assert(ymwdl2.weekday() == Tuesday);
      assert(ymwdl1.weekday_last() == weekday_last{Tuesday});
      assert(ymwdl2.weekday_last() == weekday_last{Tuesday});
      assert(ymwdl1 == ymwdl2);
    }
  }

  return true;
}

int main(int, char**) {
  // year_month_weekday_last + months
  ASSERT_NOEXCEPT(std::declval<year_month_weekday_last>() + std::declval<months>());
  ASSERT_NOEXCEPT(std::declval<months>() + std::declval<year_month_weekday_last>());

  ASSERT_SAME_TYPE(year_month_weekday_last, decltype(std::declval<year_month_weekday_last>() + std::declval<months>()));
  ASSERT_SAME_TYPE(year_month_weekday_last, decltype(std::declval<months>() + std::declval<year_month_weekday_last>()));

  // year_month_weekday_last + years
  ASSERT_NOEXCEPT(std::declval<year_month_weekday_last>() + std::declval<years>());
  ASSERT_NOEXCEPT(std::declval<years>() + std::declval<year_month_weekday_last>());

  ASSERT_SAME_TYPE(year_month_weekday_last, decltype(std::declval<year_month_weekday_last>() + std::declval<years>()));
  ASSERT_SAME_TYPE(year_month_weekday_last, decltype(std::declval<years>() + std::declval<year_month_weekday_last>()));

  test();
  static_assert(test());

  return 0;
}