XLKit  0.1.0
 All Classes Files Functions Typedefs Macros Groups
xlkit.hpp
Go to the documentation of this file.
1 
6 // Copyright (c) 2014 Edward Lam
7 //
8 // All rights reserved. This software is distributed under the
9 // Mozilla Public License, v. 2.0 ( http://www.mozilla.org/MPL/2.0/ ).
10 //
11 // Redistributions of source code must retain the above copyright
12 // and license notice and the following restrictions and disclaimer.
13 //
14 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
15 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
16 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
17 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
18 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY INDIRECT, INCIDENTAL,
19 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
20 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
21 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
22 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
24 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25 
26 #ifndef XLKIT_HPP
27 #define XLKIT_HPP
28 
29 #include <xlkit/xldebug.hpp>
30 #include <xlkit/xlException.hpp>
31 #include <xlkit/xlOperand.hpp>
32 #include <xlkit/xlversion.hpp>
33 
34 #define BOOST_FT_AUTODETECT_CALLING_CONVENTIONS
35 #include <boost/function_types/components.hpp>
36 #include <boost/function_types/is_nonmember_callable_builtin.hpp>
37 
38 #include <boost/mpl/begin_end.hpp>
39 #include <boost/mpl/deref.hpp>
40 #include <boost/unordered_map.hpp>
41 #include <boost/utility/enable_if.hpp>
42 
43 #include <string>
44 #include <vector>
45 #include <stdio.h>
46 
47 namespace xlkit {
49 namespace XLKIT_VERSION_NAME {
50 
51 namespace mpl = boost::mpl;
52 namespace ft = boost::function_types;
53 
56  public:
57 
60 
63  ResultOperandPtr(const xlOperand& copy);
64 
65  operator xlOperand*() {
66  return myOperand;
67  }
68 
69  xlOperand* operator->() {
70  return myOperand;
71  }
72  xlOperand& operator*() {
73  return *myOperand;
74  }
75 
76  private:
77  xlOperand* myOperand;
78 };
79 
80 namespace detail {
81 
82 // Empty help
83 struct empty_ { };
84 
85 // Info for type T
86 template <typename T>
87 struct TypeInfo;
88 
89 #define XLKIT_TYPEINFO(TYPE, CODE, HELP) \
90  template <> \
91  struct TypeInfo<TYPE> { \
92  static size_t size() { return sizeof(TYPE); } \
93  static char code() { return CODE; } \
94  static const char* name() { return #TYPE; } \
95  static const char* help() { return HELP; } \
96  }; \
97 
98 
99 XLKIT_TYPEINFO(double, 'B', "Number")
100 XLKIT_TYPEINFO(const char*, 'C', "String")
101 XLKIT_TYPEINFO(uint16_t, 'H', "Unsigned Integer")
102 XLKIT_TYPEINFO(int16_t, 'I', "Signed Integer")
103 XLKIT_TYPEINFO(int32_t, 'J', "Signed Integer")
104 XLKIT_TYPEINFO(xlOperand*, 'P', "Cell or Cell Range")
105 XLKIT_TYPEINFO(const xlOperand*, 'P', "Cell or Cell Range")
106 XLKIT_TYPEINFO(ResultOperandPtr, 'P', "Cell or Cell Range")
107 
108 #undef XLKIT_TYPEINFO
109 
110 // Error result for type T
111 template <typename T>
112 struct ErrorResult {
113  static T value() {
114  return T(0);
115  }
116  template <typename S>
117  static T value(S s) {
118  return T(0);
119  }
120 };
121 template <>
122 struct ErrorResult<ResultOperandPtr> {
123  static ResultOperandPtr value() {
124  return ResultOperandPtr();
125  }
126  template <typename S>
127  static ResultOperandPtr value(S s) {
128  return ResultOperandPtr(xlOperand(s));
129  }
130 };
131 
132 // Label for type T, defaults to nothing for unknown types.
133 template <typename T>
134 struct ParmHelp {
135  static const char* name() {
136  return NULL;
137  }
138  static const char* help() {
139  return NULL;
140  }
141 };
142 
143 } // namespace detail
144 
146 template <typename T, typename PARM_HELP = detail::empty_>
147 class Parm {
148  public:
149  typedef T type;
150  typedef PARM_HELP parm_help_tag;
151 
153  static const char *name() {
154  return detail::ParmHelp<PARM_HELP>::name();
155  }
157  static const char *help() {
158  return detail::ParmHelp<PARM_HELP>::help();
159  }
160 
163  const T& value() const {
164  return myValue;
165  }
166  operator const T&() const {
167  return myValue;
168  }
169  operator T&() {
170  return myValue;
171  }
173 
174  private:
175  T myValue;
176 };
177 
178 namespace detail {
179 
180 template <typename T, typename PARM_HELP>
181 struct TypeInfo< Parm<T, PARM_HELP> > {
182  static size_t size() {
183  return TypeInfo<T>::size();
184  }
185  static char code() {
186  return TypeInfo<T>::code();
187  }
188  static const char* name() {
189  const char* name = Parm<T, PARM_HELP>::name();
190  if (!name)
191  return TypeInfo<T>::name();
192  return name;
193  }
194  static const char* help() {
195  const char* help = Parm<T, PARM_HELP>::help();
196  if (!help)
197  return TypeInfo<T>::help();
198  return help;
199  }
200 };
201 
202 } // namespace detail
203 
205 class Registry {
206 
207  public:
208 
210  static Registry& instance() {
211  if (!theInstance)
212  theInstance = new Registry;
213  return *theInstance;
214  }
215 
216  void setAddinLabel(const char* label);
217 
221  template <typename F>
222  void addFunction(const std::string& name, F f, const char* help) {
223  myFunctions[name] = makeWrapper<F>(name, f, help);
224  }
226  template <typename F>
227  void addFunction(const std::string& excel_name,
228  const std::string& name, F f, const char* help) {
229  myFunctions[name] = makeWrapper<F>(excel_name, f, help);
230  }
232 
234  void dump() {
235  for (const auto& i : myFunctions) {
236  printf("'%s' -> '%s' [", i.first.c_str(), i.second.myTypes.c_str());
237  for (int j = 0, n = i.second.myParmHelp.size(); j < n; ++j) {
238  if (j > 0)
239  printf(",");
240  printf("%s", i.second.myParmHelp[j].c_str());
241  }
242  printf("]\n");
243  }
244  }
245 
247  struct Wrapper {
248  Wrapper() {
249  }
250  Wrapper(const std::string& func_name,
251  const std::string& sig,
252  const std::string& func_help,
253  const std::string& arg_names,
254  const std::vector<std::string>& parm_help)
255  : myFuncName(func_name)
256  , myTypes(sig)
257  , myFuncHelp(func_help)
258  , myArgNames(arg_names)
259  , myParmHelp(parm_help) {
260  }
261 
262  std::string myFuncName;
263  std::string myTypes;
264  std::string myFuncHelp;
265  std::string myArgNames;
266  std::vector<std::string> myParmHelp;
267  };
268 
269  typedef boost::unordered_map<std::string, Wrapper> NameMap;
270 
272  const NameMap& functions() const {
273  return myFunctions;
274  }
275 
276  private:
277 
278  Registry() {}
279 
280  // Make a wrapper object for function F
281  template <typename F>
282  Wrapper
283  makeWrapper(const std::string&name, F f, const char* func_help,
284  typename boost::enable_if< ft::is_nonmember_callable_builtin<F, ft::stdcall_cc>
285  >::type *dummy = 0) {
286 
287  typedef Func<F> FuncT;
288 
289  std::string types;
290  Func<F>::getTypes(types);
291 
292  // Skip first type, which is the return type
293  typedef typename mpl::next<mpl::begin< ft::components<F> >::type>::type Second;
294  typedef typename mpl::end< ft::components<F> >::type End;
295 
296  std::string arg_names;
297  Func<F, Second, End>::getArgNames(arg_names);
298 
299  std::vector<std::string> parm_help;
300  Func<F, Second, End>::getParmHelp(parm_help);
301 
302  return Wrapper(name, types, func_help, arg_names, parm_help);
303  }
304 
305  // Get the information for function F (first arg is return type)
306  template
307  < typename F
308  , class Beg = typename mpl::begin< ft::components<F> >::type
309  , class End = typename mpl::end< ft::components<F> >::type
310  >
311  struct Func;
312 
313  private:
314 
315  NameMap myFunctions;
316 
317  static Registry* theInstance;
318 };
319 
320 inline void
321 dumpRegistry() {
323 }
324 
325 //
326 // Specializations for function introspection
327 //
328 template <typename F, typename Curr, typename End>
329 struct Registry::Func {
330  // Type string signature
331  static void getTypes(std::string& types) {
332  typedef typename mpl::next<Curr>::type Next;
333  typedef typename mpl::deref<Curr>::type Type;
334 
335  types += detail::TypeInfo<Type>::code();
336  Func<F, Next, End>::getTypes(types);
337  }
338  // Names of the arguments
339  static void getArgNames(std::string& arg_names) {
340  typedef typename mpl::next<Curr>::type Next;
341  typedef typename mpl::deref<Curr>::type Type;
342 
343  if (arg_names.size() > 0)
344  arg_names += ", ";
345  arg_names += detail::TypeInfo<Type>::name();
346  Func<F, Next, End>::getArgNames(arg_names);
347  }
348  // Help description for individual arguments
349  static void getParmHelp(std::vector<std::string>& parm_help) {
350  typedef typename mpl::next<Curr>::type Next;
351  typedef typename mpl::deref<Curr>::type Type;
352 
353  parm_help.push_back(detail::TypeInfo<Type>::help());
354  Func<F, Next, End>::getParmHelp(parm_help);
355  }
356 };
357 
358 template <typename F, typename End>
359 struct Registry::Func<F, End, End> {
360  static void getTypes(std::string&) { }
361  static void getArgNames(std::string&) { }
362  static void getParmHelp(std::vector<std::string>&) { }
363 };
364 
365 } // namespace XLKIT_VERSION_NAME
366 } // namespace xlkit
367 
371 
374 template <typename T, typename PARM_HELP = xlkit::detail::empty_>
375 using xlParm = xlkit::Parm<T, PARM_HELP>;
376 
379 typedef xlkit::ResultOperandPtr xlResultOperandPtr;
380 
382 
386 
389 #define XLKIT_INIT_ADDIN_LABEL(LABEL) \
390  struct XLInitAddinLabel { \
391  XLInitAddinLabel() { \
392  xlkit::Registry::instance().setAddinLabel(LABEL); \
393  } \
394  }; \
395  static XLInitAddinLabel theInitAddinLabel; \
396 
397 
398 
401 #define XLKIT_PARM(VALUE_TYPE, NAME, HELP) \
402  struct HELP_FOR_##NAME { }; \
403  namespace xlkit { \
404  XLKIT_USE_VERSION_NAMESPACE \
405  namespace XLKIT_VERSION_NAME { \
406  namespace detail { \
407  template <> struct ParmHelp<HELP_FOR_##NAME> { \
408  static const char* name() { return #NAME; } \
409  static const char* help() { return HELP; } \
410  }; \
411  } } } \
412  typedef xlParm<VALUE_TYPE, HELP_FOR_##NAME> xlParm##NAME;
413 
414 
416 #define XLKIT_REGISTER(FUNC, HELP) \
417  struct FUNC##Registrar { \
418  FUNC##Registrar() { \
419  xlkit::Registry::instance().addFunction(#FUNC,FUNC,HELP); \
420  } \
421  }; \
422  static FUNC##Registrar the##FUNC##Registrar; \
423 
424 
427 #define XLKIT_REGISTER_AS(XLNAME, FUNC, HELP) \
428  struct FUNC##Registrar { \
429  FUNC##Registrar() { \
430  xlkit::Registry::instance().addFunction(XLNAME,#FUNC,FUNC,HELP); \
431  } \
432  }; \
433  static FUNC##Registrar the##FUNC##Registrar; \
434 
435 
437 #define XLKIT_API __stdcall
438 
440 #define XLKIT_PRAGMA_DLL_EXPORT \
441  __pragma(comment(linker, "/EXPORT:" __FUNCTION__ "=" __FUNCDNAME__))
442 
444 #define XLKIT_BEGIN_FUNCTION \
445  XLKIT_PRAGMA_DLL_EXPORT \
446  try { \
447 
448 
451 #define XLKIT_END_FUNCTION(RESULT_T) \
452  } catch (xlkit::xlException& err) { \
453  XLDBG("Exception caught: %s", err.what()); \
454  return xlkit::detail::ErrorResult<RESULT_T>::value(); \
455  } catch (std::exception& err){ \
456  XLDBG("Exception caught: %s", err.what()); \
457  return xlkit::detail::ErrorResult<RESULT_T>::value(err.what()); \
458  } catch (xlkit::xlError& err){ \
459  XLDBG("Exception caught: %s", err.str().c_str()); \
460  return xlkit::detail::ErrorResult<RESULT_T>::value(err); \
461  } catch (...) { \
462  XLDBG("Unknown exception caught"); \
463  return xlkit::detail::ErrorResult<RESULT_T>::value(); \
464  } \
465 
466 
468 
469 #endif // XLKIT_HPP
Stores all registry of functions for the XLL.
Definition: xlkit.hpp:205
ResultOperandPtr()
Get pointer to TLS copy and default initialize it.
Definition: xlkit.cpp:332
xlkit::ResultOperandPtr xlResultOperandPtr
Return value for XLL functions. See xlResultOperandPtr.
Definition: xlkit.hpp:379
xlkit::xlOperand xlOperand
C++ wrapper for XLOPER struct. See xlOper4.
Definition: xlOperand.hpp:547
xlkit::xlOperand class
void addFunction(const std::string &name, F f, const char *help)
Register a new function.Name of the Excel function is the same as the C++ function name...
Definition: xlkit.hpp:222
Return value for XLL functions.
Definition: xlkit.hpp:55
void dump()
Print a list of all registered functions for debugging.
Definition: xlkit.hpp:234
#define XLKIT_VERSION_NAME
Version namespace for this library.
Definition: xlversion.hpp:33
Information for a registered function.
Definition: xlkit.hpp:247
C++ methods that operate on top of an XLOPER struct.
Definition: xlOperand.hpp:130
static Registry & instance()
Get the singleton instance.
Definition: xlkit.hpp:210
XLKit debugging facility for the XLDBG() macro.
const NameMap & functions() const
List of registered functions.
Definition: xlkit.hpp:272
#define XLKIT_USE_VERSION_NAMESPACE
Macro used to pull the versioned namespace into the main xlkit namepsace.
Definition: xlversion.hpp:52
xlkit::Parm< T, PARM_HELP > xlParm
An xlkit function parameter of type T, with optional help for convenience. See Parm.
Definition: xlkit.hpp:375
static const char * help()
Help of the parameter for Excel.
Definition: xlkit.hpp:157
XLKit version defines.
Function parameter with associated help.
Definition: xlkit.hpp:147
void addFunction(const std::string &excel_name, const std::string &name, F f, const char *help)
Name of the Excel function is different from the the C++ function name.
Definition: xlkit.hpp:227
const T & value() const
Value of the parameter.
Definition: xlkit.hpp:163
static const char * name()
Name of the parameter for Excel.
Definition: xlkit.hpp:153
Provides XLKIT_THROW() macro and xlException class.