ReactOS 0.4.16-dev-937-g7afcd2a
onexit.cpp File Reference
#include <corecrt_internal.h>
Include dependency graph for onexit.cpp:

Go to the source code of this file.

Enumerations

enum  : size_t { initial_table_count = 32 , minimum_table_increment = 4 , maximum_table_increment = 512 }
 

Functions

int __cdecl _crt_atexit (_PVFV const function)
 
int __cdecl _crt_at_quick_exit (_PVFV const function)
 
int __cdecl _initialize_onexit_table (_onexit_table_t *const table)
 
int __cdecl _register_onexit_function (_onexit_table_t *const table, _onexit_t const function)
 
int __cdecl _execute_onexit_table (_onexit_table_t *const table)
 

Variables

_onexit_table_t __acrt_atexit_table {}
 
_onexit_table_t __acrt_at_quick_exit_table {}
 

Enumeration Type Documentation

◆ anonymous enum

anonymous enum : size_t
Enumerator
initial_table_count 
minimum_table_increment 
maximum_table_increment 

Definition at line 30 of file onexit.cpp.

31{
35};
@ maximum_table_increment
Definition: onexit.cpp:34
@ minimum_table_increment
Definition: onexit.cpp:33
@ initial_table_count
Definition: onexit.cpp:32

Function Documentation

◆ _crt_at_quick_exit()

int __cdecl _crt_at_quick_exit ( _PVFV const  function)

Definition at line 46 of file onexit.cpp.

47{
48 return _register_onexit_function(&__acrt_at_quick_exit_table, reinterpret_cast<_onexit_t>(function));
49}
int(__cdecl * _onexit_t)(void)
Definition: stdlib.h:28
_onexit_table_t __acrt_at_quick_exit_table
Definition: onexit.cpp:26
int __cdecl _register_onexit_function(_onexit_table_t *const table, _onexit_t const function)
Definition: onexit.cpp:82

◆ _crt_atexit()

int __cdecl _crt_atexit ( _PVFV const  function)

Definition at line 41 of file onexit.cpp.

42{
43 return _register_onexit_function(&__acrt_atexit_table, reinterpret_cast<_onexit_t>(function));
44}
_onexit_table_t __acrt_atexit_table
Definition: onexit.cpp:25

Referenced by atexit(), test_call_exit(), and test_call_quick_exit().

◆ _execute_onexit_table()

int __cdecl _execute_onexit_table ( _onexit_table_t *const  table)

Definition at line 158 of file onexit.cpp.

159{
160 return __acrt_lock_and_call(__acrt_select_exit_lock(), [&]
161 {
162 if (!table)
163 {
164 return -1;
165 }
166
167 _PVFV* first = __crt_fast_decode_pointer(table->_first);
168 _PVFV* last = __crt_fast_decode_pointer(table->_last);
169 if (!first || first == reinterpret_cast<_PVFV*>(-1))
170 {
171 return 0;
172 }
173
174 // This loop calls through caller-provided function pointers. We must
175 // save and reset the global state mode before calling them, to maintain
176 // proper mode nesting. (These calls to caller-provided function pointers
177 // are the only non-trivial calls, so we can do this once for the entire
178 // loop.)
179 {
180 __crt_state_management::scoped_global_state_reset saved_state;
181
182 _PVFV const encoded_nullptr = __crt_fast_encode_pointer(nullptr);
183
184 _PVFV* saved_first = first;
185 _PVFV* saved_last = last;
186 for (;;)
187 {
188 // Find the last valid function pointer to call:
189 while (--last >= first && *last == encoded_nullptr)
190 {
191 // Keep going backwards
192 }
193
194 if (last < first)
195 {
196 // There are no more valid entries in the list; we are done:
197 break;
198 }
199
200 // Store the function pointer and mark it as visited in the list:
201 _PVFV const function = __crt_fast_decode_pointer(*last);
202 *last = encoded_nullptr;
203
204 function();
205
206 _PVFV* const new_first = __crt_fast_decode_pointer(table->_first);
207 _PVFV* const new_last = __crt_fast_decode_pointer(table->_last);
208
209 // Reset iteration if either the begin or end pointer has changed:
210 if (new_first != saved_first || new_last != saved_last)
211 {
212 first = saved_first = new_first;
213 last = saved_last = new_last;
214 }
215 }
216 }
217
218 if (first != reinterpret_cast<_PVFV*>(-1))
219 {
221 }
222
223 _PVFV* const encoded_nullptr = __crt_fast_encode_pointer(nullptr);
224
225 table->_first = encoded_nullptr;
226 table->_last = encoded_nullptr;
227 table->_end = encoded_nullptr;
228
229 return 0;
230 });
231}
#define __acrt_select_exit_lock()
void(__cdecl * _PVFV)(void)
const GLint * first
Definition: glext.h:5794
#define _free_crt
static UINT UINT last
Definition: font.c:45

◆ _initialize_onexit_table()

int __cdecl _initialize_onexit_table ( _onexit_table_t *const  table)

Definition at line 53 of file onexit.cpp.

54{
55 if (!table)
56 {
57 return -1;
58 }
59
60 // If the table has already been initialized, do not do anything. Note that
61 // this handles both the case where the table was value initialized and where
62 // the table was initialized with encoded null pointers.
63 if (table->_first != table->_end)
64 {
65 return 0;
66 }
67
68 _PVFV* const encoded_nullptr = __crt_fast_encode_pointer(nullptr);
69
70 table->_first = encoded_nullptr;
71 table->_last = encoded_nullptr;
72 table->_end = encoded_nullptr;
73
74 return 0;
75}

◆ _register_onexit_function()

int __cdecl _register_onexit_function ( _onexit_table_t *const  table,
_onexit_t const  function 
)

Definition at line 82 of file onexit.cpp.

83{
84 return __acrt_lock_and_call(__acrt_select_exit_lock(), [&]
85 {
86 if (!table)
87 {
88 return -1;
89 }
90
91 _PVFV* first = __crt_fast_decode_pointer(table->_first);
92 _PVFV* last = __crt_fast_decode_pointer(table->_last);
93 _PVFV* end = __crt_fast_decode_pointer(table->_end);
94
95 // If there is no room for the new entry, reallocate a larger table:
96 if (last == end)
97 {
98 size_t const old_count = end - first;
99
100 size_t const increment = old_count > maximum_table_increment ? maximum_table_increment : old_count;
101
102 // First, try to double the capacity of the table:
103 size_t new_count = old_count + increment;
104 if (new_count == 0)
105 {
106 new_count = initial_table_count;
107 }
108
109 _PVFV* new_first = nullptr;
110 if (new_count >= old_count)
111 {
112 new_first = _recalloc_crt_t(_PVFV, first, new_count).detach();
113 }
114
115 // If that didn't work, try to allocate a smaller increment:
116 if (new_first == nullptr)
117 {
118 new_count = old_count + minimum_table_increment;
119 new_first = _recalloc_crt_t(_PVFV, first, new_count).detach();
120 }
121
122 if (new_first == nullptr)
123 {
124 return -1;
125 }
126
127 first = new_first;
128 last = new_first + old_count;
129 end = new_first + new_count;
130
131 // The "additional" storage obtained from recalloc is sero-initialized.
132 // The array holds encoded function pointers, so we need to fill the
133 // storage with encoded nullptrs:
134 _PVFV const encoded_nullptr = __crt_fast_encode_pointer(nullptr);
135 for (auto it = last; it != end; ++it)
136 {
137 *it = encoded_nullptr;
138 }
139 }
140
141 *last++ = reinterpret_cast<_PVFV>(__crt_fast_encode_pointer(function));
142
143 table->_first = __crt_fast_encode_pointer(first);
144 table->_last = __crt_fast_encode_pointer(last);
145 table->_end = __crt_fast_encode_pointer(end);
146
147 return 0;
148 });
149}
GLuint GLuint end
Definition: gl.h:1545

Referenced by _crt_at_quick_exit(), and _crt_atexit().

Variable Documentation

◆ __acrt_at_quick_exit_table

_onexit_table_t __acrt_at_quick_exit_table {}

Definition at line 26 of file onexit.cpp.

Referenced by _crt_at_quick_exit(), and initialize_c().

◆ __acrt_atexit_table

_onexit_table_t __acrt_atexit_table {}

Definition at line 25 of file onexit.cpp.

Referenced by _crt_atexit(), and initialize_c().