// bbldc_terminatedisda30360eom.h                                     -*-C++-*-
#ifndef INCLUDED_BBLDC_TERMINATEDISDA30360EOM
#define INCLUDED_BBLDC_TERMINATEDISDA30360EOM

#include <bsls_ident.h>
BSLS_IDENT("$Id: $")

//@PURPOSE: Provide stateless functions for ISDA 30/360 eom convention.
//
//@CLASSES:
//  bbldc::TerminatedIsda30360Eom: ISDA 30/360 eom stateless functions
//
//@DESCRIPTION: This component provides a 'struct',
// 'bbldc::TerminatedIsda30360Eom', that serves as a namespace for defining a
// suite of date-related functions used to compute the day count and year
// fraction between two dates as prescribed by the International Swaps and
// Derivatives Association (ISDA) 30/360 day-count convention with end-of-month
// (eom) adjustments.  In this day-count convention (also known as "30E/360
// ISDA"), each year is assumed to have 12 months and 360 days, with each month
// consisting of exactly 30 days.  Special end-of-month rule adjustments *are*
// made to account for the last day of February.  Note that in this day-count
// convention, the second date may or may not be adjusted depending on the
// termination date (e.g., maturity date).
//
///ISDA 30/360 eom Day Count Algorithm
///-----------------------------------
// Given 'beginDate' and 'endDate', let:
//..
// Ye =  year of earlier date           Yl =  year of later date
// Me = month of earlier date           Ml = month of later date
// De =   day of earlier date           Dl =   day of later date
//
//   o If Dl is the last day of the month, unless Dl is the termination date
//     and is in February, change Dl to 30.
//
//   o If De is the last day of the month, change De to 30.
//
// daysDiff ::= sign(endDate - beginDate) *
//                                   (Yl - Ye) * 360 + (Ml - Me) * 30 + Dl - De
//..
// Reference: http://www.isda.org/c_and_a/docs/30-360-2006ISDADefs.xls
//            (tab labeled "30E-360 ISDA")
//
// The year fraction is simply the day count divided by 360.
//
///Usage
///-----
// This section illustrates intended use of this component.
//
///Example 1: Computing Day Count and Year Fraction
///- - - - - - - - - - - - - - - - - - - - - - - -
// The following snippets of code illustrate how to use
// 'bbldc::TerminatedIsda30360Eom' methods.  First, create two 'bdlt::Date'
// variables, 'd1' and 'd2':
//..
//  const bdlt::Date d1(2004,  9, 30);
//  const bdlt::Date d2(2004, 12, 31);
//..
// Then, compute the day count between the two dates:
//..
//  const int daysDiff = bbldc::TerminatedIsda30360Eom::daysDiff(d1, d2);
//  assert(90 == daysDiff);
//..
// Finally, compute the year fraction between the two dates:
//..
//  const double yearsDiff = bbldc::TerminatedIsda30360Eom::yearsDiff(d1, d2);
//  assert(0.25 == yearsDiff);
//..

#include <bblscm_version.h>

#include <bdlt_date.h>

#include <bsls_platform.h>

namespace BloombergLP {
namespace bbldc {

                      // =============================
                      // struct TerminatedIsda30360Eom
                      // =============================

struct TerminatedIsda30360Eom {
    // This 'struct' provides a namespace for a suite of pure functions that
    // compute values based on dates according to the ISDA 30/360 end-of-month
    // day-count convention.

    // CLASS METHODS
    static int daysDiff(const bdlt::Date& beginDate,
                        const bdlt::Date& endDate,
                        const bdlt::Date& terminationDate = bdlt::Date());
        // Return the (signed) number of days between the specified 'beginDate'
        // and 'endDate', with the optionally specified 'terminationDate',
        // according to the ISDA 30/360 end-of-month day-count convention.  If
        // 'beginDate <= endDate', then the result is non-negative.  Note that
        // reversing the order of 'beginDate' and 'endDate' negates the result.
        // Also note that, in accordance with the convention definition, there
        // are no constraints upon the supplied dates.

    static double yearsDiff(const bdlt::Date& beginDate,
                            const bdlt::Date& endDate,
                            const bdlt::Date& terminationDate
                                                   = bdlt::Date(9999, 12, 31));
        // Return the (signed fractional) number of years between the specified
        // 'beginDate' and 'endDate', with the optionally specified
        // 'terminationDate', according to the ISDA 30/360 end-of-month
        // day-count convention.  If 'beginDate <= endDate', then the result is
        // non-negative.  Note that reversing the order of 'beginDate' and
        // 'endDate' negates the result; specifically,
        // '|yearsDiff(b, e) + yearsDiff(e, b)| <= 1.0e-15' for all dates 'b'
        // and 'e'.  Also note that, in accordance with the convention
        // definition, there are no constraints upon the supplied dates.
};

// ============================================================================
//                             INLINE DEFINITIONS
// ============================================================================

                      // -----------------------------
                      // struct TerminatedIsda30360Eom
                      // -----------------------------

// CLASS METHODS
inline
double TerminatedIsda30360Eom::yearsDiff(const bdlt::Date& beginDate,
                                         const bdlt::Date& endDate,
                                         const bdlt::Date& terminationDate)
{
#if defined(BSLS_PLATFORM_CMP_GNU) && (BSLS_PLATFORM_CMP_VERSION >= 50301)
    // Storing the result value in a 'volatile double' removes extra-precision
    // available in floating-point registers.

    const volatile double rv =
#else
    const double rv =
#endif
                      static_cast<double>(daysDiff(beginDate,
                                                   endDate,
                                                   terminationDate)) / 360.0;

    return rv;
}

}  // close package namespace
}  // close enterprise namespace

#endif

// ----------------------------------------------------------------------------
// Copyright 2017 Bloomberg Finance L.P.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// ----------------------------- END-OF-FILE ----------------------------------