src/core/callback.h
author Mathieu Lacage <mathieu.lacage@sophia.inria.fr>
Thu Nov 12 13:01:01 2009 +0100 (2009-11-12)
changeset 5505 c0ac392289c3
parent 4205 751691462d36
child 6273 8d70de29d514
permissions -rw-r--r--
replace RefCountBase with SimpleRefCount<> to avoid duplicate refcounting implementations.
mathieu@150
     1
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
mathieu@9
     2
/*
mathieu@9
     3
 * Copyright (c) 2005,2006 INRIA
mathieu@9
     4
 *
mathieu@9
     5
 * This program is free software; you can redistribute it and/or modify
mathieu@9
     6
 * it under the terms of the GNU General Public License version 2 as
mathieu@9
     7
 * published by the Free Software Foundation;
mathieu@9
     8
 *
mathieu@9
     9
 * This program is distributed in the hope that it will be useful,
mathieu@9
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
mathieu@9
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
mathieu@9
    12
 * GNU General Public License for more details.
mathieu@9
    13
 *
mathieu@9
    14
 * You should have received a copy of the GNU General Public License
mathieu@9
    15
 * along with this program; if not, write to the Free Software
mathieu@9
    16
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
mathieu@9
    17
 *
mathieu@9
    18
 * Author: Mathieu Lacage <mathieu.lacage@sophia.inria.fr>
mathieu@9
    19
 */
mathieu@9
    20
mathieu@9
    21
#ifndef CALLBACK_H
mathieu@9
    22
#define CALLBACK_H
mathieu@9
    23
mathieu@687
    24
#include "ptr.h"
mathieu@528
    25
#include "fatal-error.h"
mathieu@714
    26
#include "empty.h"
mathieu@1418
    27
#include "type-traits.h"
mathieu@3763
    28
#include "attribute.h"
mathieu@3763
    29
#include "attribute-helper.h"
mathieu@5505
    30
#include "simple-ref-count.h"
mathieu@3365
    31
#include <typeinfo>
mathieu@9
    32
mathieu@16
    33
namespace ns3 {
mathieu@9
    34
mathieu@9
    35
/***
mathieu@9
    36
 * \internal
mathieu@9
    37
 * This code was originally written based on the techniques 
mathieu@9
    38
 * described in http://www.codeproject.com/cpp/TTLFunction.asp
mathieu@9
    39
 * It was subsequently rewritten to follow the architecture
mathieu@9
    40
 * outlined in "Modern C++ Design" by Andrei Alexandrescu in 
mathieu@9
    41
 * chapter 5, "Generalized Functors".
mathieu@9
    42
 *
mathieu@9
    43
 * This code uses:
mathieu@9
    44
 *   - default template parameters to saves users from having to
mathieu@9
    45
 *     specify empty parameters when the number of parameters
mathieu@9
    46
 *     is smaller than the maximum supported number
mathieu@9
    47
 *   - the pimpl idiom: the Callback class is passed around by 
mathieu@9
    48
 *     value and delegates the crux of the work to its pimpl
mathieu@9
    49
 *     pointer.
mathieu@9
    50
 *   - two pimpl implementations which derive from CallbackImpl
mathieu@9
    51
 *     FunctorCallbackImpl can be used with any functor-type
mathieu@9
    52
 *     while MemPtrCallbackImpl can be used with pointers to
mathieu@9
    53
 *     member functions.
mathieu@9
    54
 *   - a reference list implementation to implement the Callback's
mathieu@9
    55
 *     value semantics.
mathieu@9
    56
 *
mathieu@9
    57
 * This code most notably departs from the alexandrescu 
mathieu@9
    58
 * implementation in that it does not use type lists to specify
mathieu@9
    59
 * and pass around the types of the callback arguments.
mathieu@9
    60
 * Of course, it also does not use copy-destruction semantics
mathieu@53
    61
 * and relies on a reference list rather than autoPtr to hold
mathieu@9
    62
 * the pointer.
mathieu@9
    63
 */
mathieu@685
    64
template <typename T>
mathieu@685
    65
struct CallbackTraits;
mathieu@685
    66
mathieu@685
    67
template <typename T>
mathieu@685
    68
struct CallbackTraits<T *>
mathieu@685
    69
{
mathieu@685
    70
  static T & GetReference (T * const p)
mathieu@685
    71
  {
mathieu@685
    72
    return *p;
mathieu@685
    73
  }
mathieu@685
    74
};
mathieu@685
    75
mathieu@5505
    76
class CallbackImplBase : public SimpleRefCount<CallbackImplBase>
mathieu@2370
    77
{
mathieu@219
    78
public:
mathieu@219
    79
  virtual ~CallbackImplBase () {}
mathieu@2206
    80
  virtual bool IsEqual (Ptr<const CallbackImplBase> other) const = 0;
mathieu@219
    81
};
mathieu@219
    82
mathieu@9
    83
// declare the CallbackImpl class
Providence@4072
    84
template <typename R, typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7, typename T8, typename T9>
mathieu@9
    85
class CallbackImpl;
mathieu@9
    86
// define CallbackImpl for 0 params
mathieu@9
    87
template <typename R>
Providence@4072
    88
class CallbackImpl<R,empty,empty,empty,empty,empty,empty,empty,empty,empty> : public CallbackImplBase {
mathieu@9
    89
public:
mathieu@150
    90
  virtual ~CallbackImpl () {}
mathieu@150
    91
  virtual R operator() (void) = 0;
mathieu@9
    92
};
mathieu@9
    93
// define CallbackImpl for 1 params
mathieu@9
    94
template <typename R, typename T1>
Providence@4072
    95
class CallbackImpl<R,T1,empty,empty,empty,empty,empty,empty,empty,empty> : public CallbackImplBase {
mathieu@9
    96
public:
mathieu@150
    97
  virtual ~CallbackImpl () {}
mathieu@150
    98
  virtual R operator() (T1) = 0;
mathieu@9
    99
};
mathieu@9
   100
// define CallbackImpl for 2 params
mathieu@9
   101
template <typename R, typename T1, typename T2>
Providence@4072
   102
class CallbackImpl<R,T1,T2,empty,empty,empty,empty,empty,empty,empty> : public CallbackImplBase {
mathieu@9
   103
public:
mathieu@150
   104
  virtual ~CallbackImpl () {}
mathieu@150
   105
  virtual R operator() (T1, T2) = 0;
mathieu@9
   106
};
mathieu@9
   107
// define CallbackImpl for 3 params
mathieu@9
   108
template <typename R, typename T1, typename T2, typename T3>
Providence@4072
   109
class CallbackImpl<R,T1,T2,T3,empty,empty,empty,empty,empty,empty> : public CallbackImplBase {
mathieu@9
   110
public:
mathieu@150
   111
  virtual ~CallbackImpl () {}
mathieu@150
   112
  virtual R operator() (T1, T2, T3) = 0;
mathieu@9
   113
};
mathieu@9
   114
// define CallbackImpl for 4 params
mathieu@9
   115
template <typename R, typename T1, typename T2, typename T3, typename T4>
Providence@4072
   116
class CallbackImpl<R,T1,T2,T3,T4,empty,empty,empty,empty,empty> : public CallbackImplBase {
mathieu@9
   117
public:
mathieu@150
   118
  virtual ~CallbackImpl () {}
mathieu@150
   119
  virtual R operator() (T1, T2, T3, T4) = 0;
mathieu@9
   120
};
mathieu@9
   121
// define CallbackImpl for 5 params
mathieu@9
   122
template <typename R, typename T1, typename T2, typename T3, typename T4, typename T5>
Providence@4072
   123
class CallbackImpl<R,T1,T2,T3,T4,T5,empty,empty,empty,empty> : public CallbackImplBase {
mathieu@9
   124
public:
mathieu@150
   125
  virtual ~CallbackImpl () {}
mathieu@150
   126
  virtual R operator() (T1, T2, T3, T4, T5) = 0;
mathieu@9
   127
};
mathieu@1417
   128
// define CallbackImpl for 6 params
Providence@4072
   129
template <typename R, typename T1, typename T2, typename T3, typename T4, typename T5, typename T6>
Providence@4072
   130
class CallbackImpl<R,T1,T2,T3,T4,T5,T6,empty,empty,empty> : public CallbackImplBase {
mathieu@1417
   131
public:
mathieu@1417
   132
  virtual ~CallbackImpl () {}
mathieu@1417
   133
  virtual R operator() (T1, T2, T3, T4, T5, T6) = 0;
mathieu@1417
   134
};
Providence@4072
   135
// define CallbackImpl for 7 params
Providence@4072
   136
template <typename R, typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7>
Providence@4072
   137
class CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,empty,empty> : public CallbackImplBase {
Providence@4072
   138
public:
Providence@4072
   139
  virtual ~CallbackImpl () {}
Providence@4072
   140
  virtual R operator() (T1, T2, T3, T4, T5, T6, T7) = 0;
Providence@4072
   141
};
Providence@4072
   142
// define CallbackImpl for 8 params
Providence@4072
   143
template <typename R, typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7, typename T8>
Providence@4072
   144
class CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,T8,empty> : public CallbackImplBase {
Providence@4072
   145
public:
Providence@4072
   146
  virtual ~CallbackImpl () {}
Providence@4072
   147
  virtual R operator() (T1, T2, T3, T4, T5, T6, T7, T8) = 0;
Providence@4072
   148
};
Providence@4072
   149
// define CallbackImpl for 9 params
Providence@4072
   150
template <typename R, typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7, typename T8, typename T9>
Providence@4072
   151
class CallbackImpl : public CallbackImplBase {
Providence@4072
   152
public:
Providence@4072
   153
  virtual ~CallbackImpl () {}
Providence@4072
   154
  virtual R operator() (T1, T2, T3, T4, T5, T6, T7, T8, T9) = 0;
Providence@4072
   155
};
mathieu@9
   156
mathieu@9
   157
mathieu@9
   158
// an impl for Functors:
Providence@4072
   159
template <typename T, typename R, typename T1, typename T2, typename T3, typename T4,typename T5, typename T6, typename T7, typename T8, typename T9>
Providence@4072
   160
class FunctorCallbackImpl : public CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> {
mathieu@9
   161
public:
mathieu@150
   162
  FunctorCallbackImpl (T const &functor)
mathieu@687
   163
    : m_functor (functor) {}
mathieu@150
   164
  virtual ~FunctorCallbackImpl () {}
mathieu@150
   165
  R operator() (void) {
Providence@4072
   166
    return m_functor ();
mathieu@150
   167
  }
mathieu@150
   168
  R operator() (T1 a1) {
Providence@4072
   169
    return m_functor (a1);
mathieu@150
   170
  }
mathieu@150
   171
  R operator() (T1 a1,T2 a2) {
Providence@4072
   172
    return m_functor (a1,a2);
mathieu@150
   173
  }
mathieu@150
   174
  R operator() (T1 a1,T2 a2,T3 a3) {
Providence@4072
   175
    return m_functor (a1,a2,a3);
mathieu@150
   176
  }
mathieu@150
   177
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4) {
Providence@4072
   178
    return m_functor (a1,a2,a3,a4);
mathieu@150
   179
  }
mathieu@150
   180
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5) {
Providence@4072
   181
    return m_functor (a1,a2,a3,a4,a5);
mathieu@150
   182
  }
mathieu@1417
   183
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6) {
mathieu@1417
   184
    return m_functor (a1,a2,a3,a4,a5,a6);
mathieu@1417
   185
  }
Providence@4072
   186
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6,T7 a7) {
Providence@4072
   187
    return m_functor (a1,a2,a3,a4,a5,a6,a7);
Providence@4072
   188
  }
Providence@4072
   189
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6,T7 a7,T8 a8) {
Providence@4072
   190
    return m_functor (a1,a2,a3,a4,a5,a6,a7,a8);
Providence@4072
   191
  }
Providence@4072
   192
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6,T7 a7,T8 a8,T9 a9) {
Providence@4072
   193
    return m_functor (a1,a2,a3,a4,a5,a6,a7,a8,a9);
Providence@4072
   194
  }
mathieu@2206
   195
  virtual bool IsEqual (Ptr<const CallbackImplBase> other) const {
Providence@4072
   196
    FunctorCallbackImpl<T,R,T1,T2,T3,T4,T5,T6,T7,T8,T9> const *otherDerived = 
Providence@4072
   197
      dynamic_cast<FunctorCallbackImpl<T,R,T1,T2,T3,T4,T5,T6,T7,T8,T9> const *> (PeekPointer(other));
tomh@345
   198
    if (otherDerived == 0)
tomh@345
   199
      {
tomh@345
   200
        return false;
tomh@345
   201
      }
tomh@345
   202
    else if (otherDerived->m_functor != m_functor)
tomh@345
   203
      {
tomh@345
   204
        return false;
tomh@345
   205
      }
tomh@345
   206
    return true;
tomh@345
   207
  }
mathieu@9
   208
private:
mathieu@150
   209
  T m_functor;
mathieu@9
   210
};
mathieu@9
   211
mathieu@9
   212
// an impl for pointer to member functions
Providence@4072
   213
template <typename OBJ_PTR, typename MEM_PTR, typename R, typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7, typename T8, typename T9>
Providence@4072
   214
class MemPtrCallbackImpl : public CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> {
mathieu@9
   215
public:
mathieu@150
   216
  MemPtrCallbackImpl (OBJ_PTR const&objPtr, MEM_PTR mem_ptr)
mathieu@687
   217
    : m_objPtr (objPtr), m_memPtr (mem_ptr) {}
mathieu@150
   218
  virtual ~MemPtrCallbackImpl () {}
mathieu@150
   219
  R operator() (void) {
mathieu@685
   220
    return ((CallbackTraits<OBJ_PTR>::GetReference (m_objPtr)).*m_memPtr) ();
mathieu@150
   221
  }
mathieu@150
   222
  R operator() (T1 a1) {
mathieu@685
   223
    return ((CallbackTraits<OBJ_PTR>::GetReference (m_objPtr)).*m_memPtr) (a1);
mathieu@150
   224
  }
mathieu@150
   225
  R operator() (T1 a1,T2 a2) {
mathieu@685
   226
    return ((CallbackTraits<OBJ_PTR>::GetReference (m_objPtr)).*m_memPtr) (a1, a2);
mathieu@150
   227
  }
mathieu@150
   228
  R operator() (T1 a1,T2 a2,T3 a3) {
mathieu@685
   229
    return ((CallbackTraits<OBJ_PTR>::GetReference (m_objPtr)).*m_memPtr) (a1, a2, a3);
mathieu@150
   230
  }
mathieu@150
   231
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4) {
mathieu@685
   232
    return ((CallbackTraits<OBJ_PTR>::GetReference (m_objPtr)).*m_memPtr) (a1, a2, a3, a4);
mathieu@150
   233
  }
mathieu@150
   234
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5) {
mathieu@685
   235
    return ((CallbackTraits<OBJ_PTR>::GetReference (m_objPtr)).*m_memPtr) (a1, a2, a3, a4, a5);
mathieu@150
   236
  }
mathieu@1417
   237
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6) {
mathieu@1417
   238
    return ((CallbackTraits<OBJ_PTR>::GetReference (m_objPtr)).*m_memPtr) (a1, a2, a3, a4, a5, a6);
mathieu@1417
   239
  }
Providence@4072
   240
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6,T7 a7) {
Providence@4072
   241
    return ((CallbackTraits<OBJ_PTR>::GetReference (m_objPtr)).*m_memPtr) (a1, a2, a3, a4, a5, a6, a7);
Providence@4072
   242
  }
Providence@4072
   243
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6,T7 a7,T8 a8) {
Providence@4072
   244
    return ((CallbackTraits<OBJ_PTR>::GetReference (m_objPtr)).*m_memPtr) (a1, a2, a3, a4, a5, a6, a7, a8);
Providence@4072
   245
  }
Providence@4072
   246
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6,T7 a7,T8 a8, T9 a9) {
Providence@4072
   247
    return ((CallbackTraits<OBJ_PTR>::GetReference (m_objPtr)).*m_memPtr) (a1, a2, a3, a4, a5, a6, a7, a8, a9);
Providence@4072
   248
  }
mathieu@2206
   249
  virtual bool IsEqual (Ptr<const CallbackImplBase> other) const {
Providence@4072
   250
    MemPtrCallbackImpl<OBJ_PTR,MEM_PTR,R,T1,T2,T3,T4,T5,T6,T7,T8,T9> const *otherDerived = 
Providence@4072
   251
      dynamic_cast<MemPtrCallbackImpl<OBJ_PTR,MEM_PTR,R,T1,T2,T3,T4,T5,T6,T7,T8,T9> const *> (PeekPointer (other));
tomh@345
   252
    if (otherDerived == 0)
tomh@345
   253
      {
tomh@345
   254
        return false;
tomh@345
   255
      }
tomh@345
   256
    else if (otherDerived->m_objPtr != m_objPtr ||
tomh@345
   257
             otherDerived->m_memPtr != m_memPtr)
tomh@345
   258
      {
tomh@345
   259
        return false;
tomh@345
   260
      }
tomh@345
   261
    return true;
tomh@345
   262
  }
mathieu@9
   263
private:
mathieu@150
   264
  OBJ_PTR const m_objPtr;
mathieu@150
   265
  MEM_PTR m_memPtr;
mathieu@9
   266
};
mathieu@9
   267
mathieu@2531
   268
// an impl for Bound Functors:
Providence@4072
   269
template <typename T, typename R, typename TX, typename T1, typename T2, typename T3, typename T4,typename T5, typename T6, typename T7, typename T8>
Providence@4072
   270
class BoundFunctorCallbackImpl : public CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,T8,empty> {
mathieu@2531
   271
public:
mathieu@2531
   272
  template <typename FUNCTOR, typename ARG>
mathieu@2531
   273
  BoundFunctorCallbackImpl (FUNCTOR functor, ARG a)
Providence@4072
   274
    : m_functor (functor), m_a (a) {}
mathieu@2531
   275
  virtual ~BoundFunctorCallbackImpl () {}
mathieu@2531
   276
  R operator() (void) {
Providence@4072
   277
    return m_functor (m_a);
mathieu@2531
   278
  }
mathieu@2531
   279
  R operator() (T1 a1) {
Providence@4072
   280
    return m_functor (m_a,a1);
mathieu@2531
   281
  }
mathieu@2531
   282
  R operator() (T1 a1,T2 a2) {
Providence@4072
   283
    return m_functor (m_a,a1,a2);
mathieu@2531
   284
  }
mathieu@2531
   285
  R operator() (T1 a1,T2 a2,T3 a3) {
Providence@4072
   286
    return m_functor (m_a,a1,a2,a3);
mathieu@2531
   287
  }
mathieu@2531
   288
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4) {
Providence@4072
   289
    return m_functor (m_a,a1,a2,a3,a4);
mathieu@2531
   290
  }
mathieu@2531
   291
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5) {
Providence@4072
   292
    return m_functor (m_a,a1,a2,a3,a4,a5);
Providence@4072
   293
  }
Providence@4072
   294
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6) {
Providence@4072
   295
    return m_functor (m_a,a1,a2,a3,a4,a5,a6);
Providence@4072
   296
  }
Providence@4072
   297
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6,T7 a7) {
Providence@4072
   298
    return m_functor (m_a,a1,a2,a3,a4,a5,a6,a7);
Providence@4072
   299
  }
Providence@4072
   300
  R operator() (T1 a1,T2 a2,T3 a3,T4 a4,T5 a5,T6 a6,T7 a7,T8 a8) {
Providence@4072
   301
    return m_functor (m_a,a1,a2,a3,a4,a5,a6,a7,a8);
mathieu@2531
   302
  }
mathieu@2531
   303
  virtual bool IsEqual (Ptr<const CallbackImplBase> other) const {
Providence@4072
   304
    BoundFunctorCallbackImpl<T,R,TX,T1,T2,T3,T4,T5,T6,T7,T8> const *otherDerived = 
Providence@4072
   305
      dynamic_cast<BoundFunctorCallbackImpl<T,R,TX,T1,T2,T3,T4,T5,T6,T7,T8> const *> (PeekPointer (other));
mathieu@2531
   306
    if (otherDerived == 0)
mathieu@2531
   307
      {
mathieu@2531
   308
        return false;
mathieu@2531
   309
      }
mathieu@2531
   310
    else if (otherDerived->m_functor != m_functor ||
mathieu@2531
   311
             otherDerived->m_a != m_a)
mathieu@2531
   312
      {
mathieu@2531
   313
        return false;
mathieu@2531
   314
      }
mathieu@2531
   315
    return true;
mathieu@2531
   316
  }
mathieu@2531
   317
private:
mathieu@2531
   318
  T m_functor;
mathieu@2531
   319
  typename TypeTraits<TX>::ReferencedType m_a;
mathieu@2531
   320
};
mathieu@2531
   321
mathieu@2531
   322
mathieu@219
   323
class CallbackBase {
mathieu@219
   324
public:
mathieu@2206
   325
  CallbackBase () : m_impl () {}
mathieu@2206
   326
  Ptr<CallbackImplBase> GetImpl (void) const {return m_impl;}
mathieu@2206
   327
protected:
mathieu@2206
   328
  CallbackBase (Ptr<CallbackImplBase> impl) : m_impl (impl) {}
mathieu@2206
   329
  Ptr<CallbackImplBase> m_impl;
timo@4205
   330
timo@4205
   331
  static std::string Demangle(const std::string& mangled);
mathieu@219
   332
};
mathieu@219
   333
mathieu@9
   334
/**
mathieu@9
   335
 * \brief Callback template class
mathieu@9
   336
 *
mathieu@9
   337
 * This class template implements the Functor Design Pattern.
mathieu@9
   338
 * It is used to declare the type of a Callback:
mathieu@9
   339
 *  - the first non-optional template argument represents
mathieu@9
   340
 *    the return type of the callback.
mathieu@9
   341
 *  - the second optional template argument represents
mathieu@9
   342
 *    the type of the first argument to the callback.
mathieu@9
   343
 *  - the third optional template argument represents
mathieu@9
   344
 *    the type of the second argument to the callback.
mathieu@9
   345
 *  - the fourth optional template argument represents
mathieu@9
   346
 *    the type of the third argument to the callback.
mathieu@9
   347
 *  - the fifth optional template argument represents
mathieu@9
   348
 *    the type of the fourth argument to the callback.
mathieu@9
   349
 *  - the sixth optional template argument represents
mathieu@9
   350
 *    the type of the fifth argument to the callback.
mathieu@9
   351
 *
mathieu@122
   352
 * Callback instances are built with the \ref MakeCallback
mathieu@9
   353
 * template functions. Callback instances have POD semantics:
mathieu@9
   354
 * the memory they allocate is managed automatically, without
mathieu@9
   355
 * user intervention which allows you to pass around Callback
mathieu@9
   356
 * instances by value.
mathieu@9
   357
 *
mathieu@9
   358
 * Sample code which shows how to use this class template 
mathieu@122
   359
 * as well as the function templates \ref MakeCallback :
mathieu@9
   360
 * \include samples/main-callback.cc
mathieu@9
   361
 */
tomh@345
   362
mathieu@9
   363
template<typename R, 
Providence@4072
   364
         typename T1 = empty, typename T2 = empty, 
Providence@4072
   365
         typename T3 = empty, typename T4 = empty,
Providence@4072
   366
         typename T5 = empty, typename T6 = empty,
Providence@4072
   367
         typename T7 = empty, typename T8 = empty,
Providence@4072
   368
         typename T9 = empty>
mathieu@219
   369
class Callback : public CallbackBase {
mathieu@9
   370
public:
mathieu@2206
   371
  Callback () {}
mathieu@2206
   372
tomh@345
   373
  // There are two dummy args below to ensure that this constructor is
tomh@345
   374
  // always properly disambiguited by the c++ compiler
mathieu@150
   375
  template <typename FUNCTOR>
tomh@345
   376
  Callback (FUNCTOR const &functor, bool, bool) 
Providence@4072
   377
    : CallbackBase (Create<FunctorCallbackImpl<FUNCTOR,R,T1,T2,T3,T4,T5,T6,T7,T8,T9> > (functor))
mathieu@150
   378
  {}
mathieu@9
   379
mathieu@150
   380
  template <typename OBJ_PTR, typename MEM_PTR>
mathieu@150
   381
  Callback (OBJ_PTR const &objPtr, MEM_PTR mem_ptr)
Providence@4072
   382
    : CallbackBase (Create<MemPtrCallbackImpl<OBJ_PTR,MEM_PTR,R,T1,T2,T3,T4,T5,T6,T7,T8,T9> > (objPtr, mem_ptr))
mathieu@150
   383
  {}
mathieu@9
   384
Providence@4072
   385
  Callback (Ptr<CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> > const &impl)
mathieu@2206
   386
    : CallbackBase (impl)
mathieu@150
   387
  {}
mathieu@9
   388
mathieu@2531
   389
  template <typename T>
Providence@4072
   390
  Callback<R,T2,T3,T4,T5,T6,T7,T8,T9> Bind (T a) {
Providence@4072
   391
    Ptr<CallbackImpl<R,T2,T3,T4,T5,T6,T7,T8,T9,empty> > impl =
Providence@4072
   392
      Ptr<CallbackImpl<R,T2,T3,T4,T5,T6,T7,T8,T9,empty> > (
Providence@4072
   393
                                                  new BoundFunctorCallbackImpl<
Providence@4072
   394
                                                  Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9>,
Providence@4072
   395
                                                  R,T1,T2,T3,T4,T5,T6,T7,T8,T9> (*this, a), false);
Providence@4072
   396
    return Callback<R,T2,T3,T4,T5,T6,T7,T8,T9> (impl);
mathieu@2531
   397
  }
mathieu@2531
   398
mathieu@879
   399
  bool IsNull (void) const {
mathieu@2206
   400
    return (DoPeekImpl () == 0)?true:false;
mathieu@150
   401
  }
mathieu@686
   402
  void Nullify (void) {
mathieu@686
   403
    m_impl = 0;
mathieu@686
   404
  }
mathieu@9
   405
tomh@345
   406
  R operator() (void) const {
mathieu@2206
   407
    return (*(DoPeekImpl ())) ();
mathieu@150
   408
  }
tomh@345
   409
  R operator() (T1 a1) const {
mathieu@2206
   410
    return (*(DoPeekImpl ())) (a1);
mathieu@150
   411
  }
tomh@345
   412
  R operator() (T1 a1, T2 a2) const {
mathieu@2206
   413
    return (*(DoPeekImpl ())) (a1,a2);
mathieu@150
   414
  }
tomh@345
   415
  R operator() (T1 a1, T2 a2, T3 a3) const {
mathieu@2206
   416
    return (*(DoPeekImpl ())) (a1,a2,a3);
mathieu@150
   417
  }
tomh@345
   418
  R operator() (T1 a1, T2 a2, T3 a3, T4 a4) const {
mathieu@2206
   419
    return (*(DoPeekImpl ())) (a1,a2,a3,a4);
mathieu@150
   420
  }
tomh@345
   421
  R operator() (T1 a1, T2 a2, T3 a3, T4 a4,T5 a5) const {
mathieu@2206
   422
    return (*(DoPeekImpl ())) (a1,a2,a3,a4,a5);
mathieu@150
   423
  }
mathieu@1417
   424
  R operator() (T1 a1, T2 a2, T3 a3, T4 a4,T5 a5,T6 a6) const {
mathieu@2206
   425
    return (*(DoPeekImpl ())) (a1,a2,a3,a4,a5,a6);
mathieu@1417
   426
  }
Providence@4072
   427
  R operator() (T1 a1, T2 a2, T3 a3, T4 a4,T5 a5,T6 a6,T7 a7) const {
Providence@4072
   428
    return (*(DoPeekImpl ())) (a1,a2,a3,a4,a5,a6,a7);
Providence@4072
   429
  }
Providence@4072
   430
  R operator() (T1 a1, T2 a2, T3 a3, T4 a4,T5 a5,T6 a6,T7 a7,T8 a8) const {
Providence@4072
   431
    return (*(DoPeekImpl ())) (a1,a2,a3,a4,a5,a6,a7,a8);
Providence@4072
   432
  }
Providence@4072
   433
  R operator() (T1 a1, T2 a2, T3 a3, T4 a4,T5 a5,T6 a6,T7 a7,T8 a8, T9 a9) const {
Providence@4072
   434
    return (*(DoPeekImpl ())) (a1,a2,a3,a4,a5,a6,a7,a8,a9);
Providence@4072
   435
  }
mathieu@219
   436
mathieu@2206
   437
  bool IsEqual (const CallbackBase &other) const {
mathieu@2206
   438
    return m_impl->IsEqual (other.GetImpl ());
tomh@345
   439
  }
tomh@345
   440
mathieu@2206
   441
  bool CheckType (const CallbackBase & other) const {
mathieu@2206
   442
    return DoCheckType (other.GetImpl ());
mathieu@2206
   443
  }
mathieu@2206
   444
  void Assign (const CallbackBase &other) {
mathieu@2206
   445
    DoAssign (other.GetImpl ());
mathieu@2206
   446
  }
mathieu@2206
   447
private:
Providence@4072
   448
  CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> *DoPeekImpl (void) const {
Providence@4072
   449
    return static_cast<CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> *> (PeekPointer (m_impl));
mathieu@2206
   450
  }
mathieu@2206
   451
  bool DoCheckType (Ptr<const CallbackImplBase> other) const {
Providence@4072
   452
    if (other != 0 && dynamic_cast<const CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> *> (PeekPointer (other)) != 0)
mathieu@3763
   453
      {
mathieu@3763
   454
        return true;
mathieu@3763
   455
      }
mathieu@3763
   456
    else if (other == 0)
mathieu@219
   457
      {
mathieu@219
   458
        return true;
mathieu@219
   459
      }
mathieu@219
   460
    else
mathieu@219
   461
      {
mathieu@219
   462
        return false;
mathieu@219
   463
      }
mathieu@219
   464
  }
mathieu@2206
   465
  void DoAssign (Ptr<const CallbackImplBase> other) {
mathieu@2206
   466
    if (!DoCheckType (other))
mathieu@528
   467
      {
timo@4205
   468
        NS_FATAL_ERROR ("Incompatible types. (feed to \"c++filt -t\" if needed)" << std::endl <<
timo@4205
   469
                        "got=" << Demangle ( typeid (*other).name () ) << std::endl <<
timo@4205
   470
                        "expected=" << Demangle ( typeid (CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> *).name () ));
mathieu@528
   471
      }
mathieu@2206
   472
    m_impl = const_cast<CallbackImplBase *> (PeekPointer (other));
mathieu@528
   473
  }
mathieu@9
   474
};
mathieu@9
   475
mathieu@2531
   476
mathieu@2531
   477
template <typename R, typename T1, typename T2,
mathieu@2531
   478
          typename T3, typename T4,
Providence@4072
   479
          typename T5, typename T6,
Providence@4072
   480
          typename T7, typename T8,
Providence@4072
   481
          typename T9>
Providence@4072
   482
bool operator != (Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> a, Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> b)
mathieu@2531
   483
{
mathieu@2531
   484
  return !a.IsEqual (b);
mathieu@2531
   485
}
mathieu@2531
   486
mathieu@9
   487
/**
tomh@2217
   488
 * \ingroup core
mathieu@122
   489
 * \defgroup MakeCallback MakeCallback
mathieu@9
   490
 *
mathieu@9
   491
 */
mathieu@9
   492
mathieu@9
   493
/**
mathieu@122
   494
 * \ingroup MakeCallback
mathieu@1250
   495
 * \param memPtr class method member pointer
mathieu@53
   496
 * \param objPtr class instance
mathieu@9
   497
 * \return a wrapper Callback
mathieu@9
   498
 * Build Callbacks for class method members which takes no arguments
mathieu@9
   499
 * and potentially return a value.
mathieu@9
   500
 */
mathieu@685
   501
template <typename T, typename OBJ, typename R>
mathieu@685
   502
Callback<R> MakeCallback (R (T::*memPtr) (void), OBJ objPtr) {
mathieu@685
   503
  return Callback<R> (objPtr, memPtr);
mathieu@9
   504
}
mathieu@685
   505
template <typename T, typename OBJ, typename R>
mathieu@1417
   506
Callback<R> MakeCallback (R (T::*mem_ptr) () const, OBJ objPtr) {
mathieu@443
   507
  return Callback<R> (objPtr, mem_ptr);
mathieu@443
   508
}
mathieu@9
   509
/**
mathieu@122
   510
 * \ingroup MakeCallback
mathieu@9
   511
 * \param mem_ptr class method member pointer
mathieu@53
   512
 * \param objPtr class instance
mathieu@9
   513
 * \return a wrapper Callback
mathieu@9
   514
 * Build Callbacks for class method members which takes one argument
mathieu@9
   515
 * and potentially return a value.
mathieu@9
   516
 */
mathieu@685
   517
template <typename T, typename OBJ, typename R, typename T1>
mathieu@1417
   518
Callback<R,T1> MakeCallback (R (T::*mem_ptr) (T1), OBJ objPtr) {
mathieu@150
   519
  return Callback<R,T1> (objPtr, mem_ptr);
mathieu@9
   520
}
mathieu@685
   521
template <typename T, typename OBJ, typename R, typename T1>
mathieu@1417
   522
Callback<R,T1> MakeCallback (R (T::*mem_ptr) (T1) const, OBJ objPtr) {
mathieu@443
   523
  return Callback<R,T1> (objPtr, mem_ptr);
mathieu@443
   524
}
mathieu@9
   525
/**
mathieu@122
   526
 * \ingroup MakeCallback
mathieu@9
   527
 * \param mem_ptr class method member pointer
mathieu@53
   528
 * \param objPtr class instance
mathieu@9
   529
 * \return a wrapper Callback
mathieu@9
   530
 * Build Callbacks for class method members which takes two arguments
mathieu@9
   531
 * and potentially return a value.
mathieu@9
   532
 */
mathieu@685
   533
template <typename T, typename OBJ, typename R, typename T1, typename T2>
mathieu@1417
   534
Callback<R,T1,T2> MakeCallback (R (T::*mem_ptr) (T1,T2), OBJ objPtr) {
mathieu@150
   535
  return Callback<R,T1,T2> (objPtr, mem_ptr);
mathieu@9
   536
}
mathieu@685
   537
template <typename T, typename OBJ, typename R, typename T1, typename T2>
mathieu@1417
   538
Callback<R,T1,T2> MakeCallback (R (T::*mem_ptr) (T1,T2) const, OBJ objPtr) {
mathieu@443
   539
  return Callback<R,T1,T2> (objPtr, mem_ptr);
mathieu@443
   540
}
mathieu@9
   541
/**
mathieu@122
   542
 * \ingroup MakeCallback
mathieu@9
   543
 * \param mem_ptr class method member pointer
mathieu@53
   544
 * \param objPtr class instance
mathieu@9
   545
 * \return a wrapper Callback
mathieu@9
   546
 * Build Callbacks for class method members which takes three arguments
mathieu@9
   547
 * and potentially return a value.
mathieu@9
   548
 */
mathieu@685
   549
template <typename T, typename OBJ, typename R, typename T1,typename T2, typename T3>
mathieu@1417
   550
Callback<R,T1,T2,T3> MakeCallback (R (T::*mem_ptr) (T1,T2,T3), OBJ objPtr) {
mathieu@150
   551
  return Callback<R,T1,T2,T3> (objPtr, mem_ptr);
mathieu@9
   552
}
mathieu@685
   553
template <typename T, typename OBJ, typename R, typename T1,typename T2, typename T3>
mathieu@1417
   554
Callback<R,T1,T2,T3> MakeCallback (R (T::*mem_ptr) (T1,T2,T3) const, OBJ objPtr) {
mathieu@443
   555
  return Callback<R,T1,T2,T3> (objPtr, mem_ptr);
mathieu@443
   556
}
mathieu@9
   557
/**
mathieu@122
   558
 * \ingroup MakeCallback
mathieu@9
   559
 * \param mem_ptr class method member pointer
mathieu@53
   560
 * \param objPtr class instance
mathieu@9
   561
 * \return a wrapper Callback
mathieu@9
   562
 * Build Callbacks for class method members which takes four arguments
mathieu@9
   563
 * and potentially return a value.
mathieu@9
   564
 */
mathieu@685
   565
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4>
mathieu@1417
   566
Callback<R,T1,T2,T3,T4> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4), OBJ objPtr) {
mathieu@150
   567
  return Callback<R,T1,T2,T3,T4> (objPtr, mem_ptr);
mathieu@9
   568
}
mathieu@685
   569
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4>
mathieu@1417
   570
Callback<R,T1,T2,T3,T4> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4) const, OBJ objPtr) {
mathieu@443
   571
  return Callback<R,T1,T2,T3,T4> (objPtr, mem_ptr);
mathieu@443
   572
}
mathieu@9
   573
/**
mathieu@122
   574
 * \ingroup MakeCallback
mathieu@9
   575
 * \param mem_ptr class method member pointer
mathieu@53
   576
 * \param objPtr class instance
mathieu@9
   577
 * \return a wrapper Callback
mathieu@9
   578
 * Build Callbacks for class method members which takes five arguments
mathieu@9
   579
 * and potentially return a value.
mathieu@9
   580
 */
mathieu@685
   581
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4,typename T5>
mathieu@1417
   582
Callback<R,T1,T2,T3,T4,T5> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4,T5), OBJ objPtr) {
mathieu@150
   583
  return Callback<R,T1,T2,T3,T4,T5> (objPtr, mem_ptr);
mathieu@9
   584
}
mathieu@685
   585
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4,typename T5>
mathieu@1417
   586
Callback<R,T1,T2,T3,T4,T5> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4,T5) const, OBJ objPtr) {
mathieu@443
   587
  return Callback<R,T1,T2,T3,T4,T5> (objPtr, mem_ptr);
mathieu@443
   588
}
mathieu@1417
   589
/**
mathieu@1417
   590
 * \ingroup MakeCallback
mathieu@1417
   591
 * \param mem_ptr class method member pointer
mathieu@1417
   592
 * \param objPtr class instance
mathieu@1417
   593
 * \return a wrapper Callback
Providence@4072
   594
 * Build Callbacks for class method members which takes six arguments
mathieu@1417
   595
 * and potentially return a value.
mathieu@1417
   596
 */
mathieu@1417
   597
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4,typename T5,typename T6>
mathieu@1417
   598
Callback<R,T1,T2,T3,T4,T5,T6> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4,T5,T6), OBJ objPtr) {
mathieu@1417
   599
  return Callback<R,T1,T2,T3,T4,T5,T6> (objPtr, mem_ptr);
mathieu@1417
   600
}
mathieu@1417
   601
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4,typename T5, typename T6>
mathieu@1417
   602
Callback<R,T1,T2,T3,T4,T5,T6> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4,T5,T6) const, OBJ objPtr) {
mathieu@1417
   603
  return Callback<R,T1,T2,T3,T4,T5,T6> (objPtr, mem_ptr);
mathieu@1417
   604
}
mathieu@9
   605
mathieu@9
   606
/**
mathieu@122
   607
 * \ingroup MakeCallback
Providence@4072
   608
 * \param mem_ptr class method member pointer
Providence@4072
   609
 * \param objPtr class instance
Providence@4072
   610
 * \return a wrapper Callback
Providence@4072
   611
 * Build Callbacks for class method members which takes seven arguments
Providence@4072
   612
 * and potentially return a value.
Providence@4072
   613
 */
Providence@4072
   614
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4,typename T5,typename T6, typename T7>
Providence@4072
   615
Callback<R,T1,T2,T3,T4,T5,T6,T7> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4,T5,T6,T7), OBJ objPtr) {
Providence@4072
   616
  return Callback<R,T1,T2,T3,T4,T5,T6,T7> (objPtr, mem_ptr);
Providence@4072
   617
}
Providence@4072
   618
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4,typename T5, typename T6, typename T7>
Providence@4072
   619
Callback<R,T1,T2,T3,T4,T5,T6,T7> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4,T5,T6,T7) const, OBJ objPtr) {
Providence@4072
   620
  return Callback<R,T1,T2,T3,T4,T5,T6,T7> (objPtr, mem_ptr);
Providence@4072
   621
}
Providence@4072
   622
Providence@4072
   623
Providence@4072
   624
/**
Providence@4072
   625
 * \ingroup MakeCallback
Providence@4072
   626
 * \param mem_ptr class method member pointer
Providence@4072
   627
 * \param objPtr class instance
Providence@4072
   628
 * \return a wrapper Callback
Providence@4072
   629
 * Build Callbacks for class method members which takes eight arguments
Providence@4072
   630
 * and potentially return a value.
Providence@4072
   631
 */
Providence@4072
   632
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4,typename T5,typename T6, typename T7, typename T8>
Providence@4072
   633
Callback<R,T1,T2,T3,T4,T5,T6,T7,T8> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4,T5,T6,T7,T8), OBJ objPtr) {
Providence@4072
   634
  return Callback<R,T1,T2,T3,T4,T5,T6,T7,T8> (objPtr, mem_ptr);
Providence@4072
   635
}
Providence@4072
   636
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4,typename T5, typename T6, typename T7, typename T8>
Providence@4072
   637
Callback<R,T1,T2,T3,T4,T5,T6,T7,T8> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4,T5,T6,T7,T8) const, OBJ objPtr) {
Providence@4072
   638
  return Callback<R,T1,T2,T3,T4,T5,T6,T7,T8> (objPtr, mem_ptr);
Providence@4072
   639
}
Providence@4072
   640
Providence@4072
   641
/**
Providence@4072
   642
 * \ingroup MakeCallback
Providence@4072
   643
 * \param mem_ptr class method member pointer
Providence@4072
   644
 * \param objPtr class instance
Providence@4072
   645
 * \return a wrapper Callback
Providence@4072
   646
 * Build Callbacks for class method members which takes nine arguments
Providence@4072
   647
 * and potentially return a value.
Providence@4072
   648
 */
Providence@4072
   649
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4,typename T5,typename T6, typename T7, typename T8, typename T9>
Providence@4072
   650
Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4,T5,T6,T7,T8,T9), OBJ objPtr) {
Providence@4072
   651
  return Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> (objPtr, mem_ptr);
Providence@4072
   652
}
Providence@4072
   653
template <typename T, typename OBJ, typename R, typename T1, typename T2, typename T3, typename T4,typename T5, typename T6, typename T7, typename T8, typename T9>
Providence@4072
   654
Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> MakeCallback (R (T::*mem_ptr) (T1,T2,T3,T4,T5,T6,T7,T8,T9) const, OBJ objPtr) {
Providence@4072
   655
  return Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> (objPtr, mem_ptr);
Providence@4072
   656
}
Providence@4072
   657
Providence@4072
   658
/**
Providence@4072
   659
 * \ingroup MakeCallback
mathieu@53
   660
 * \param fnPtr function pointer
mathieu@9
   661
 * \return a wrapper Callback
mathieu@9
   662
 * Build Callbacks for functions which takes no arguments
mathieu@9
   663
 * and potentially return a value.
mathieu@9
   664
 */
mathieu@9
   665
template <typename R>
mathieu@122
   666
Callback<R> MakeCallback (R (*fnPtr) ()) {
tomh@345
   667
  return Callback<R> (fnPtr, true, true);
mathieu@9
   668
}
mathieu@9
   669
/**
mathieu@122
   670
 * \ingroup MakeCallback
mathieu@53
   671
 * \param fnPtr function pointer
mathieu@9
   672
 * \return a wrapper Callback
mathieu@9
   673
 * Build Callbacks for functions which takes one argument
mathieu@9
   674
 * and potentially return a value.
mathieu@9
   675
 */
mathieu@9
   676
template <typename R, typename T1>
mathieu@122
   677
Callback<R,T1> MakeCallback (R (*fnPtr) (T1)) {
tomh@345
   678
  return Callback<R,T1> (fnPtr, true, true);
mathieu@9
   679
}
mathieu@9
   680
/**
mathieu@122
   681
 * \ingroup MakeCallback
mathieu@53
   682
 * \param fnPtr function pointer
mathieu@9
   683
 * \return a wrapper Callback
mathieu@9
   684
 * Build Callbacks for functions which takes two arguments
mathieu@9
   685
 * and potentially return a value.
mathieu@9
   686
 */
mathieu@9
   687
template <typename R, typename T1, typename T2>
mathieu@122
   688
Callback<R,T1,T2> MakeCallback (R (*fnPtr) (T1,T2)) {
tomh@345
   689
  return Callback<R,T1,T2> (fnPtr, true, true);
mathieu@9
   690
}
mathieu@9
   691
/**
mathieu@122
   692
 * \ingroup MakeCallback
mathieu@53
   693
 * \param fnPtr function pointer
mathieu@9
   694
 * \return a wrapper Callback
mathieu@9
   695
 * Build Callbacks for functions which takes three arguments
mathieu@9
   696
 * and potentially return a value.
mathieu@9
   697
 */
mathieu@9
   698
template <typename R, typename T1, typename T2,typename T3>
mathieu@122
   699
Callback<R,T1,T2,T3> MakeCallback (R (*fnPtr) (T1,T2,T3)) {
tomh@345
   700
  return Callback<R,T1,T2,T3> (fnPtr, true, true);
mathieu@9
   701
}
mathieu@9
   702
/**
mathieu@122
   703
 * \ingroup MakeCallback
mathieu@53
   704
 * \param fnPtr function pointer
mathieu@9
   705
 * \return a wrapper Callback
mathieu@9
   706
 * Build Callbacks for functions which takes four arguments
mathieu@9
   707
 * and potentially return a value.
mathieu@9
   708
 */
mathieu@9
   709
template <typename R, typename T1, typename T2,typename T3,typename T4>
mathieu@122
   710
Callback<R,T1,T2,T3,T4> MakeCallback (R (*fnPtr) (T1,T2,T3,T4)) {
tomh@345
   711
  return Callback<R,T1,T2,T3,T4> (fnPtr, true, true);
mathieu@9
   712
}
mathieu@9
   713
/**
mathieu@122
   714
 * \ingroup MakeCallback
mathieu@53
   715
 * \param fnPtr function pointer
mathieu@9
   716
 * \return a wrapper Callback
mathieu@9
   717
 * Build Callbacks for functions which takes five arguments
mathieu@9
   718
 * and potentially return a value.
mathieu@9
   719
 */
mathieu@9
   720
template <typename R, typename T1, typename T2,typename T3,typename T4,typename T5>
mathieu@122
   721
Callback<R,T1,T2,T3,T4,T5> MakeCallback (R (*fnPtr) (T1,T2,T3,T4,T5)) {
tomh@345
   722
  return Callback<R,T1,T2,T3,T4,T5> (fnPtr, true, true);
mathieu@9
   723
}
mathieu@1417
   724
/**
mathieu@1417
   725
 * \ingroup MakeCallback
mathieu@1417
   726
 * \param fnPtr function pointer
mathieu@1417
   727
 * \return a wrapper Callback
Providence@4072
   728
 * Build Callbacks for functions which takes six arguments
mathieu@1417
   729
 * and potentially return a value.
mathieu@1417
   730
 */
mathieu@1417
   731
template <typename R, typename T1, typename T2,typename T3,typename T4,typename T5,typename T6>
mathieu@1417
   732
Callback<R,T1,T2,T3,T4,T5,T6> MakeCallback (R (*fnPtr) (T1,T2,T3,T4,T5,T6)) {
mathieu@1417
   733
  return Callback<R,T1,T2,T3,T4,T5,T6> (fnPtr, true, true);
mathieu@1417
   734
}
mathieu@9
   735
Providence@4072
   736
/**
Providence@4072
   737
 * \ingroup MakeCallback
Providence@4072
   738
 * \param fnPtr function pointer
Providence@4072
   739
 * \return a wrapper Callback
Providence@4072
   740
 * Build Callbacks for functions which takes seven arguments
Providence@4072
   741
 * and potentially return a value.
Providence@4072
   742
 */
Providence@4072
   743
template <typename R, typename T1, typename T2,typename T3,typename T4,typename T5,typename T6, typename T7>
Providence@4072
   744
Callback<R,T1,T2,T3,T4,T5,T6,T7> MakeCallback (R (*fnPtr) (T1,T2,T3,T4,T5,T6,T7)) {
Providence@4072
   745
  return Callback<R,T1,T2,T3,T4,T5,T6,T7> (fnPtr, true, true);
Providence@4072
   746
}
Providence@4072
   747
Providence@4072
   748
/**
Providence@4072
   749
 * \ingroup MakeCallback
Providence@4072
   750
 * \param fnPtr function pointer
Providence@4072
   751
 * \return a wrapper Callback
Providence@4072
   752
 * Build Callbacks for functions which takes eight arguments
Providence@4072
   753
 * and potentially return a value.
Providence@4072
   754
 */
Providence@4072
   755
template <typename R, typename T1, typename T2,typename T3,typename T4,typename T5,typename T6, typename T7, typename T8>
Providence@4072
   756
Callback<R,T1,T2,T3,T4,T5,T6,T7,T8> MakeCallback (R (*fnPtr) (T1,T2,T3,T4,T5,T6,T7,T8)) {
Providence@4072
   757
  return Callback<R,T1,T2,T3,T4,T5,T6,T7,T8> (fnPtr, true, true);
Providence@4072
   758
}
Providence@4072
   759
Providence@4072
   760
/**
Providence@4072
   761
 * \ingroup MakeCallback
Providence@4072
   762
 * \param fnPtr function pointer
Providence@4072
   763
 * \return a wrapper Callback
Providence@4072
   764
 * Build Callbacks for functions which takes nine arguments
Providence@4072
   765
 * and potentially return a value.
Providence@4072
   766
 */
Providence@4072
   767
template <typename R, typename T1, typename T2,typename T3,typename T4,typename T5,typename T6, typename T7, typename T8, typename T9>
Providence@4072
   768
Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> MakeCallback (R (*fnPtr) (T1,T2,T3,T4,T5,T6,T7,T8,T9)) {
Providence@4072
   769
  return Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> (fnPtr, true, true);
Providence@4072
   770
}
Providence@4072
   771
mathieu@9
   772
mathieu@9
   773
mathieu@9
   774
/**
mathieu@122
   775
 * \ingroup MakeCallback
mathieu@9
   776
 * \return a wrapper Callback
mathieu@9
   777
 * Build a null callback which takes no arguments
mathieu@9
   778
 * and potentially return a value.
mathieu@9
   779
 */
mathieu@9
   780
template <typename R>
mathieu@122
   781
Callback<R> MakeNullCallback (void) {
mathieu@150
   782
  return Callback<R> ();
mathieu@9
   783
}
mathieu@9
   784
/**
mathieu@122
   785
 * \ingroup MakeCallback
mathieu@416
   786
 * \overload Callback<R> MakeNullCallback (void)
mathieu@9
   787
 * \return a wrapper Callback
mathieu@9
   788
 * Build a null callback which takes one argument
mathieu@9
   789
 * and potentially return a value.
mathieu@9
   790
 */
mathieu@9
   791
template <typename R, typename T1>
mathieu@122
   792
Callback<R,T1> MakeNullCallback (void) {
mathieu@150
   793
  return Callback<R,T1> ();
mathieu@9
   794
}
mathieu@9
   795
/**
mathieu@122
   796
 * \ingroup MakeCallback
mathieu@416
   797
 * \overload Callback<R> MakeNullCallback (void)
mathieu@9
   798
 * \return a wrapper Callback
mathieu@9
   799
 * Build a null callback which takes two arguments
mathieu@9
   800
 * and potentially return a value.
mathieu@9
   801
 */
mathieu@9
   802
template <typename R, typename T1, typename T2>
mathieu@122
   803
Callback<R,T1,T2> MakeNullCallback (void) {
mathieu@150
   804
  return Callback<R,T1,T2> ();
mathieu@9
   805
}
mathieu@9
   806
/**
mathieu@122
   807
 * \ingroup MakeCallback
mathieu@416
   808
 * \overload Callback<R> MakeNullCallback (void)
mathieu@9
   809
 * \return a wrapper Callback
mathieu@9
   810
 * Build a null callback which takes three arguments
mathieu@9
   811
 * and potentially return a value.
mathieu@9
   812
 */
mathieu@9
   813
template <typename R, typename T1, typename T2,typename T3>
mathieu@122
   814
Callback<R,T1,T2,T3> MakeNullCallback (void) {
mathieu@150
   815
  return Callback<R,T1,T2,T3> ();
mathieu@9
   816
}
mathieu@9
   817
/**
mathieu@122
   818
 * \ingroup MakeCallback
mathieu@416
   819
 * \overload Callback<R> MakeNullCallback (void)
mathieu@9
   820
 * \return a wrapper Callback
mathieu@9
   821
 * Build a null callback which takes four arguments
mathieu@9
   822
 * and potentially return a value.
mathieu@9
   823
 */
mathieu@9
   824
template <typename R, typename T1, typename T2,typename T3,typename T4>
mathieu@122
   825
Callback<R,T1,T2,T3,T4> MakeNullCallback (void) {
mathieu@150
   826
  return Callback<R,T1,T2,T3,T4> ();
mathieu@9
   827
}
mathieu@9
   828
/**
mathieu@122
   829
 * \ingroup MakeCallback
mathieu@416
   830
 * \overload Callback<R> MakeNullCallback (void)
mathieu@9
   831
 * \return a wrapper Callback
mathieu@9
   832
 * Build a null callback which takes five arguments
mathieu@9
   833
 * and potentially return a value.
mathieu@9
   834
 */
mathieu@9
   835
template <typename R, typename T1, typename T2,typename T3,typename T4,typename T5>
mathieu@122
   836
Callback<R,T1,T2,T3,T4,T5> MakeNullCallback (void) {
mathieu@150
   837
  return Callback<R,T1,T2,T3,T4,T5> ();
mathieu@9
   838
}
mathieu@1417
   839
/**
mathieu@1417
   840
 * \ingroup MakeCallback
mathieu@1417
   841
 * \overload Callback<R> MakeNullCallback (void)
mathieu@1417
   842
 * \return a wrapper Callback
Providence@4072
   843
 * Build a null callback which takes six arguments
mathieu@1417
   844
 * and potentially return a value.
mathieu@1417
   845
 */
mathieu@1417
   846
template <typename R, typename T1, typename T2,typename T3,typename T4,typename T5,typename T6>
mathieu@1417
   847
Callback<R,T1,T2,T3,T4,T5,T6> MakeNullCallback (void) {
mathieu@1417
   848
  return Callback<R,T1,T2,T3,T4,T5,T6> ();
mathieu@1417
   849
}
mathieu@9
   850
Providence@4072
   851
/**
Providence@4072
   852
 * \ingroup MakeCallback
Providence@4072
   853
 * \overload Callback<R> MakeNullCallback (void)
Providence@4072
   854
 * \return a wrapper Callback
Providence@4072
   855
 * Build a null callback which takes seven arguments
Providence@4072
   856
 * and potentially return a value.
Providence@4072
   857
 */
Providence@4072
   858
template <typename R, typename T1, typename T2,typename T3,typename T4,typename T5,typename T6, typename T7>
Providence@4072
   859
Callback<R,T1,T2,T3,T4,T5,T6,T7> MakeNullCallback (void) {
Providence@4072
   860
  return Callback<R,T1,T2,T3,T4,T5,T6,T7> ();
Providence@4072
   861
}
Providence@4072
   862
Providence@4072
   863
/**
Providence@4072
   864
 * \ingroup MakeCallback
Providence@4072
   865
 * \overload Callback<R> MakeNullCallback (void)
Providence@4072
   866
 * \return a wrapper Callback
Providence@4072
   867
 * Build a null callback which takes eight arguments
Providence@4072
   868
 * and potentially return a value.
Providence@4072
   869
 */
Providence@4072
   870
template <typename R, typename T1, typename T2,typename T3,typename T4,typename T5,typename T6, typename T7, typename T8>
Providence@4072
   871
Callback<R,T1,T2,T3,T4,T5,T6,T7,T8> MakeNullCallback (void) {
Providence@4072
   872
  return Callback<R,T1,T2,T3,T4,T5,T6,T7,T8> ();
Providence@4072
   873
}
Providence@4072
   874
Providence@4072
   875
/**
Providence@4072
   876
 * \ingroup MakeCallback
Providence@4072
   877
 * \overload Callback<R> MakeNullCallback (void)
Providence@4072
   878
 * \return a wrapper Callback
Providence@4072
   879
 * Build a null callback which takes nine arguments
Providence@4072
   880
 * and potentially return a value.
Providence@4072
   881
 */
Providence@4072
   882
template <typename R, typename T1, typename T2,typename T3,typename T4,typename T5,typename T6, typename T7, typename T8, typename T9>
Providence@4072
   883
Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> MakeNullCallback (void) {
Providence@4072
   884
  return Callback<R,T1,T2,T3,T4,T5,T6,T7,T8,T9> ();
Providence@4072
   885
}
Providence@4072
   886
mathieu@73
   887
raj@632
   888
/*
mathieu@73
   889
 * The following is experimental code. It works but we have
mathieu@73
   890
 * not yet determined whether or not it is really useful and whether
mathieu@73
   891
 * or not we really want to use it.
mathieu@73
   892
 */
mathieu@73
   893
mathieu@1418
   894
template <typename R, typename TX, typename ARG>
mathieu@1418
   895
Callback<R> MakeBoundCallback (R (*fnPtr) (TX), ARG a) {
Providence@4072
   896
  Ptr<CallbackImpl<R,empty,empty,empty,empty,empty,empty,empty,empty,empty> > impl =
Providence@4072
   897
    Create<BoundFunctorCallbackImpl<R (*) (TX),R,TX,empty,empty,empty,empty,empty,empty,empty,empty> >(fnPtr, a);
tomh@345
   898
  return Callback<R> (impl);
tomh@345
   899
}
tomh@345
   900
mathieu@1418
   901
template <typename R, typename TX, typename ARG, 
mathieu@1418
   902
          typename T1>
mathieu@1418
   903
Callback<R,T1> MakeBoundCallback (R (*fnPtr) (TX,T1), ARG a) {
Providence@4072
   904
  Ptr<CallbackImpl<R,T1,empty,empty,empty,empty,empty,empty,empty,empty> > impl =
Providence@4072
   905
    Create<BoundFunctorCallbackImpl<R (*) (TX,T1),R,TX,T1,empty,empty,empty,empty,empty,empty,empty> > (fnPtr, a);
mathieu@150
   906
  return Callback<R,T1> (impl);
mathieu@9
   907
}
mathieu@1418
   908
template <typename R, typename TX, typename ARG, 
mathieu@1418
   909
          typename T1, typename T2>
mathieu@1418
   910
Callback<R,T1,T2> MakeBoundCallback (R (*fnPtr) (TX,T1,T2), ARG a) {
Providence@4072
   911
  Ptr<CallbackImpl<R,T1,T2,empty,empty,empty,empty,empty,empty,empty> > impl =
Providence@4072
   912
    Create<BoundFunctorCallbackImpl<R (*) (TX,T1,T2),R,TX,T1,T2,empty,empty,empty,empty,empty,empty> > (fnPtr, a);
mathieu@150
   913
  return Callback<R,T1,T2> (impl);
mathieu@9
   914
}
mathieu@1418
   915
template <typename R, typename TX, typename ARG,
mathieu@1418
   916
          typename T1, typename T2,typename T3>
mathieu@1418
   917
Callback<R,T1,T2,T3> MakeBoundCallback (R (*fnPtr) (TX,T1,T2,T3), ARG a) {
Providence@4072
   918
  Ptr<CallbackImpl<R,T1,T2,T3,empty,empty,empty,empty,empty,empty> > impl =
Providence@4072
   919
    Create<BoundFunctorCallbackImpl<R (*) (TX,T1,T2,T3),R,TX,T1,T2,T3,empty,empty,empty,empty,empty> > (fnPtr, a);
mathieu@1418
   920
  return Callback<R,T1,T2,T3> (impl);
mathieu@1418
   921
}
mathieu@1418
   922
template <typename R, typename TX, typename ARG,
mathieu@1418
   923
          typename T1, typename T2,typename T3,typename T4>
mathieu@1418
   924
Callback<R,T1,T2,T3,T4> MakeBoundCallback (R (*fnPtr) (TX,T1,T2,T3,T4), ARG a) {
Providence@4072
   925
  Ptr<CallbackImpl<R,T1,T2,T3,T4,empty,empty,empty,empty,empty> > impl =
Providence@4072
   926
    Create<BoundFunctorCallbackImpl<R (*) (TX,T1,T2,T3,T4),R,TX,T1,T2,T3,T4,empty,empty,empty,empty> > (fnPtr, a);
mathieu@150
   927
  return Callback<R,T1,T2,T3,T4> (impl);
mathieu@9
   928
}
mathieu@1418
   929
template <typename R, typename TX, typename ARG,
mathieu@1418
   930
          typename T1, typename T2,typename T3,typename T4,typename T5>
mathieu@1418
   931
Callback<R,T1,T2,T3,T4,T5> MakeBoundCallback (R (*fnPtr) (TX,T1,T2,T3,T4,T5), ARG a) {
Providence@4072
   932
  Ptr<CallbackImpl<R,T1,T2,T3,T4,T5,empty,empty,empty,empty> > impl =
Providence@4072
   933
    Create<BoundFunctorCallbackImpl<R (*) (TX,T1,T2,T3,T4,T5),R,TX,T1,T2,T3,T4,T5,empty,empty,empty> > (fnPtr, a);
mathieu@150
   934
  return Callback<R,T1,T2,T3,T4,T5> (impl);
mathieu@9
   935
}
Providence@4072
   936
template <typename R, typename TX, typename ARG,
Providence@4072
   937
          typename T1, typename T2,typename T3,typename T4,typename T5, typename T6>
Providence@4072
   938
Callback<R,T1,T2,T3,T4,T5,T6> MakeBoundCallback (R (*fnPtr) (TX,T1,T2,T3,T4,T5,T6), ARG a) {
Providence@4072
   939
  Ptr<CallbackImpl<R,T1,T2,T3,T4,T5,T6,empty,empty,empty> > impl =
Providence@4072
   940
    Create<BoundFunctorCallbackImpl<R (*) (TX,T1,T2,T3,T4,T5,T6),R,TX,T1,T2,T3,T4,T5,T6,empty,empty> > (fnPtr, a);
Providence@4072
   941
  return Callback<R,T1,T2,T3,T4,T5,T6> (impl);
Providence@4072
   942
}
Providence@4072
   943
template <typename R, typename TX, typename ARG,
Providence@4072
   944
          typename T1, typename T2,typename T3,typename T4,typename T5, typename T6, typename T7>
Providence@4072
   945
Callback<R,T1,T2,T3,T4,T5,T6,T7> MakeBoundCallback (R (*fnPtr) (TX,T1,T2,T3,T4,T5,T6,T7), ARG a) {
Providence@4072
   946
  Ptr<CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,empty,empty> > impl =
Providence@4072
   947
    Create<BoundFunctorCallbackImpl<R (*) (TX,T1,T2,T3,T4,T5,T6,T7),R,TX,T1,T2,T3,T4,T5,T6,T7,empty> > (fnPtr, a);
Providence@4072
   948
  return Callback<R,T1,T2,T3,T4,T5,T6,T7> (impl);
Providence@4072
   949
}
Providence@4072
   950
template <typename R, typename TX, typename ARG,
Providence@4072
   951
          typename T1, typename T2,typename T3,typename T4,typename T5, typename T6, typename T7, typename T8>
Providence@4072
   952
Callback<R,T1,T2,T3,T4,T5,T6,T7,T8> MakeBoundCallback (R (*fnPtr) (TX,T1,T2,T3,T4,T5,T6,T7,T8), ARG a) {
Providence@4072
   953
  Ptr<CallbackImpl<R,T1,T2,T3,T4,T5,T6,T7,T8,empty> > impl =
Providence@4072
   954
    Create<BoundFunctorCallbackImpl<R (*) (TX,T1,T2,T3,T4,T5,T6,T7,T8),R,TX,T1,T2,T3,T4,T5,T6,T7,T8> > (fnPtr, a);
Providence@4072
   955
  return Callback<R,T1,T2,T3,T4,T5,T6,T7,T8> (impl);
Providence@4072
   956
}
mathieu@3763
   957
} // namespace ns3
mathieu@9
   958
mathieu@3763
   959
namespace ns3 {
mathieu@9
   960
mathieu@3763
   961
class CallbackValue : public AttributeValue
mathieu@3763
   962
{
mathieu@3763
   963
public:
mathieu@3763
   964
  CallbackValue ();
mathieu@3763
   965
  CallbackValue (const CallbackBase &base);
mathieu@3763
   966
  virtual ~CallbackValue ();
mathieu@3763
   967
  void Set (CallbackBase base);
mathieu@3763
   968
  template <typename T>
mathieu@3763
   969
  bool GetAccessor (T &value) const;
mathieu@3763
   970
  virtual Ptr<AttributeValue> Copy (void) const;
mathieu@3763
   971
  virtual std::string SerializeToString (Ptr<const AttributeChecker> checker) const;
mathieu@3763
   972
  virtual bool DeserializeFromString (std::string value, Ptr<const AttributeChecker> checker);
mathieu@3763
   973
private:
mathieu@3763
   974
  CallbackBase m_value;
mathieu@3763
   975
};
mathieu@3763
   976
mathieu@3763
   977
ATTRIBUTE_ACCESSOR_DEFINE(Callback);
mathieu@3763
   978
ATTRIBUTE_CHECKER_DEFINE (Callback);
mathieu@3763
   979
mathieu@3763
   980
} // namespace ns3
mathieu@3763
   981
mathieu@3763
   982
namespace ns3 {
mathieu@3763
   983
mathieu@3763
   984
template <typename T>
mathieu@3763
   985
bool CallbackValue::GetAccessor (T &value) const
mathieu@3763
   986
{
mathieu@3763
   987
  if (value.CheckType (m_value))
mathieu@3763
   988
    {
mathieu@3763
   989
      value.Assign (m_value);
mathieu@3763
   990
      return true;
mathieu@3763
   991
    }
mathieu@3763
   992
  return false;
mathieu@3763
   993
}
mathieu@3763
   994
mathieu@3763
   995
} // namespace ns3
mathieu@9
   996
mathieu@9
   997
mathieu@9
   998
#endif /* CALLBACK_H */