aboutsummaryrefslogtreecommitdiff
path: root/odb/details/win32/tls.txx
blob: 5bbe28f29bd8002c1b623ea96d2ad1d9e4d06bb5 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
// file      : odb/details/win32/tls.txx
// copyright : Copyright (c) 2009-2013 Code Synthesis Tools CC
// license   : GNU GPL v2; see accompanying LICENSE file

#include <odb/details/unique-ptr.hxx>
#include <odb/details/win32/exceptions.hxx>

namespace odb
{
  namespace details
  {
    // tls<T>
    //
    template <typename T>
    win32_once_t tls<T>::once_= WIN32_ONCE_INIT;

    template <typename T>
    size_t tls<T>::key_;

    template <typename T>
    T& tls<T>::
    get () const
    {
      win32_once (once_, key_init);

      if (void* v = _get (key_))
        return *static_cast<T*> (v);

      unique_ptr<T> p (new T);
      _set (key_, p.get ());

      T& r (*p);
      p.release ();
      return r;
    }

    template <typename T>
    void tls<T>::
    free ()
    {
      win32_once (once_, key_init);

      if (void* v = _get (key_))
      {
        _set (key_, 0);
        delete static_cast<T*> (v);
      }
    }

    template <typename T>
    void tls<T>::
    key_init ()
    {
      key_ = _allocate (destructor);
    }

    template <typename T>
    void tls<T>::
    destructor (void* v)
    {
      delete static_cast<T*> (v);
    }

    // tls<T*>
    //
    template <typename T>
    win32_once_t tls<T*>::once_ = WIN32_ONCE_INIT;

    template <typename T>
    size_t tls<T*>::key_;

    template <typename T>
    T* tls<T*>::
    get () const
    {
      win32_once (once_, key_init);
      return static_cast<T*> (_get (key_));
    }

    template <typename T>
    void tls<T*>::
    set (T* p)
    {
      win32_once (once_, key_init);
      _set (key_, p);
    }

    template <typename T>
    void tls<T*>::
    key_init ()
    {
      key_ = _allocate (0);
    }
  }
}