/* Copyright 2012. Bloomberg Finance L.P.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to
 * deal in the Software without restriction, including without limitation the
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
 * sell copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:  The above
 * copyright notice and this permission notice shall be included in all copies
 * or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
 * IN THE SOFTWARE.
 */
// blpapi_eventdispatcher.h                                           -*-C++-*-
#ifndef INCLUDED_BLPAPI_EVENTDISPATCHER
#define INCLUDED_BLPAPI_EVENTDISPATCHER

//@PURPOSE: Provide a dispatcher to dispatch events.
//
//@CLASSES:
// blpapi::EventDispatcher: Dispatches events from one or more Sessions through
// callbacks
//
//@DESCRIPTION: This component implements a dispatcher to dispatch events from
// one or more 'Session' through callbacks.

#ifndef INCLUDED_BLPAPI_DEFS
#include <blpapi_defs.h>
#endif

#ifndef INCLUDED_BLPAPI_TYPES
#include <blpapi_types.h>
#endif

#include <stddef.h> // for size_t

#ifdef __cplusplus
extern "C" {
#endif

BLPAPI_EXPORT
blpapi_EventDispatcher_t *blpapi_EventDispatcher_create(
        size_t numDispatcherThreads);

BLPAPI_EXPORT
void blpapi_EventDispatcher_destroy(blpapi_EventDispatcher_t *handle);

BLPAPI_EXPORT
int blpapi_EventDispatcher_start(blpapi_EventDispatcher_t *handle);

BLPAPI_EXPORT
int blpapi_EventDispatcher_stop(blpapi_EventDispatcher_t *handle, int async);

BLPAPI_EXPORT
int blpapi_EventDispatcher_dispatchEvents(blpapi_EventDispatcher_t *handle);

#ifdef __cplusplus
} // extern "C"

namespace BloombergLP {
namespace blpapi {
// =====================
// class EventDispatcher
// =====================

class EventDispatcher {
    // Dispatches events from one or more Sessions through callbacks
    //
    // EventDispatcher objects are optionally specified when Session
    // objects are constructed. A single EventDispatcher can be shared
    // by multiple Session objects.
    //
    // The EventDispatcher provides an event-driven interface,
    // generating callbacks from one or more internal threads for one
    // or more sessions.

    blpapi_EventDispatcher_t *d_impl_p;

  private:
    // NOT IMPLEMENTED
    EventDispatcher(const EventDispatcher&);
    EventDispatcher& operator=(const EventDispatcher&);

  public:
    EventDispatcher(size_t numDispatcherThreads = 1);
    // Construct an EventDispatcher with the specified
    // 'numDispatcherThreads'. If 'numDispatcherThreads' is 1 (the
    // default) then a single internal thread is created to
    // dispatch events. If 'numDispatcherThreads' is greater than
    // 1 then an internal pool of 'numDispatcherThreads' threads
    // is created to dispatch events. The behavior is undefined
    // if 'numDispatcherThreads' is 0.

    ~EventDispatcher();
    // Destructor.

    int start();
    // Start generating callbacks for events from sessions
    // associated with this EventDispatcher. Return 0 on success
    // and a non zero value otherwise.

    int stop(bool async = false);
    // Shutdown this event dispatcher object and stop generating
    // callbacks for events from sessions associated with it.
    // If the specified 'async' is false (the default) then this
    // method blocks until all current callbacks which were dispatched
    // through this EventDispatcher have completed.
    // Return 0 on success and a non zero value otherwise.
    //
    // Note: Calling stop with 'async' of false from within a callback
    // dispatched by this EventDispatcher is undefined and may result
    // in a deadlock.

    blpapi_EventDispatcher_t *impl() const;
    // Returns the internal implementation.
};

// ============================================================================
//                      INLINE AND TEMPLATE FUNCTION IMPLEMENTATIONS
// ============================================================================

// ---------------------
// class EventDispatcher
// ---------------------

inline EventDispatcher::EventDispatcher(size_t numDispatcherThreads)
    : d_impl_p(blpapi_EventDispatcher_create(numDispatcherThreads))
{
}

inline EventDispatcher::~EventDispatcher()
{
    blpapi_EventDispatcher_destroy(d_impl_p);
}

inline int EventDispatcher::start()
{
    return blpapi_EventDispatcher_start(d_impl_p);
}

inline int EventDispatcher::stop(bool async)
{
    return blpapi_EventDispatcher_stop(d_impl_p, async);
}

inline blpapi_EventDispatcher_t *EventDispatcher::impl() const
{
    return d_impl_p;
}

} // close namespace blpapi
} // close namespace BloombergLP

#endif // ifdef __cplusplus

#endif // #ifndef INCLUDED_BLPAPI_EVENTDISPATCHER