WinStd
Windows Win32 API using Standard C++
Loading...
Searching...
No Matches
Common.h
1/*
2 SPDX-License-Identifier: MIT
3 Copyright © 1991-2024 Amebis
4 Copyright © 2016 GÉANT
5*/
6
7#pragma once
8
9#include <Windows.h>
10#include <assert.h>
11#include <intsafe.h>
12#include <stdarg.h>
13#include <tchar.h>
14#include <iostream>
15#include <memory>
16#include <stdexcept>
17#include <string>
18#include <vector>
19
36
39
43#ifndef __L
44#define __L(x) L ## x
45#endif
46
50#ifndef _L
51#define _L(x) __L(x)
52#endif
53
57#define WINSTD_STRING_IMPL(x) #x
58
62#define WINSTD_STRING(x) WINSTD_STRING_IMPL(x)
63
67#define WINSTD_NONCOPYABLE(C) \
68private: \
69 C (_In_ const C &h) noexcept; \
70 C& operator=(_In_ const C &h) noexcept;
71
75#define WINSTD_NONMOVABLE(C) \
76private: \
77 C (_Inout_ C &&h) noexcept; \
78 C& operator=(_Inout_ C &&h) noexcept;
79
80#ifndef WINSTD_STACK_BUFFER_BYTES
94#define WINSTD_STACK_BUFFER_BYTES 1024
95#endif
96
98
101
105#ifdef UNICODE
106#define PRINTF_LPTSTR "ls"
107#else
108#define PRINTF_LPTSTR "s"
109#endif
110
114#ifdef OLE2ANSI
115#define PRINTF_LPOLESTR "hs"
116#else
117#define PRINTF_LPOLESTR "ls"
118#endif
119
123#ifdef _UNICODE
124#define _tcin (std::wcin )
125#else
126#define _tcin (std::cin )
127#endif
128
132#ifdef _UNICODE
133#define _tcout (std::wcout)
134#else
135#define _tcout (std::cout)
136#endif
137
141#ifdef _UNICODE
142#define _tcerr (std::wcerr)
143#else
144#define _tcerr (std::cerr)
145#endif
146
150#ifdef _UNICODE
151#define _tclog (std::wclog)
152#else
153#define _tclog (std::clog)
154#endif
155
157
160
164#define WINSTD_HANDLE_IMPL(C, T, INVAL) \
165public: \
166 C ( ) noexcept {} \
167 C (_In_opt_ T h) noexcept : handle<T, INVAL>( h ) {} \
168 C (_Inout_ C &&h) noexcept : handle<T, INVAL>(std::move(h)) {} \
169 C& operator=(_In_opt_ T h) noexcept { handle<T, INVAL>::operator=( h ); return *this; } \
170 C& operator=(_Inout_ C &&h) noexcept { handle<T, INVAL>::operator=(std::move(h)); return *this; } \
171WINSTD_NONCOPYABLE(C)
172
176#define WINSTD_DPLHANDLE_IMPL(C, T, INVAL) \
177public: \
178 C ( ) noexcept {} \
179 C (_In_opt_ T h) noexcept : dplhandle<T, INVAL>( h ) {} \
180 C (_In_ const C &h) noexcept : dplhandle<T, INVAL>(duplicate_internal(h.m_h)) {} \
181 C (_Inout_ C &&h) noexcept : dplhandle<T, INVAL>(std::move (h )) {} \
182 C& operator=(_In_opt_ T h) noexcept { dplhandle<T, INVAL>::operator=( h ); return *this; } \
183 C& operator=(_In_ const C &h) noexcept { dplhandle<T, INVAL>::operator=( h ); return *this; } \
184 C& operator=(_Inout_ C &&h) noexcept { dplhandle<T, INVAL>::operator=(std::move(h)); return *this; } \
185private:
186
188
189#ifndef _FormatMessage_format_string_
190#define _FormatMessage_format_string_ _In_z_
191#endif
192
194#ifndef _LPCBYTE_DEFINED
195#define _LPCBYTE_DEFINED
196typedef const BYTE *LPCBYTE;
197#endif
199
200#pragma warning(push)
201// Do not use _vsnprintf_s/_vsnwprintf_s(), since it terminates string by force even when we explicitly want to write unterminated string.
202// Threfore turn off compiler warning instead. ;)
203#pragma warning(disable: 4995)
204#pragma warning(disable: 4996)
205#pragma warning(disable: 4505) // Don't warn on unused code
206
207#ifdef _WIN64
208inline ULONGLONG ULongLongMult(ULONGLONG a, ULONGLONG b)
209{
210 ULONGLONG result;
211 if (SUCCEEDED(ULongLongMult(a, b, &result)))
212 return result;
213 throw std::invalid_argument("multiply overflow");
214}
215#else
216inline SIZE_T SIZETMult(SIZE_T a, SIZE_T b)
217{
218 SIZE_T result;
219 if (SUCCEEDED(SIZETMult(a, b, &result)))
220 return result;
221 throw std::invalid_argument("multiply overflow");
222}
223#endif
224
225#ifdef _WIN64
226inline ULONGLONG ULongLongAdd(ULONGLONG a, ULONGLONG b)
227{
228 ULONGLONG result;
229 if (SUCCEEDED(ULongLongAdd(a, b, &result)))
230 return result;
231 throw std::invalid_argument("add overflow");
232}
233#else
234inline SIZE_T SIZETAdd(SIZE_T a, SIZE_T b)
235{
236 SIZE_T result;
237 if (SUCCEEDED(SIZETAdd(a, b, &result)))
238 return result;
239 throw std::invalid_argument("add overflow");
240}
241#endif
242
245
255template<class _Traits, class _Ax>
256static int vsprintf(_Inout_ std::basic_string<char, _Traits, _Ax> &str, _In_z_ _Printf_format_string_ const char *format, _In_ va_list arg)
257{
258 char buf[WINSTD_STACK_BUFFER_BYTES/sizeof(char)];
259
260 // Try with stack buffer first.
261 int count = _vsnprintf(buf, _countof(buf), format, arg);
262 if (0 <= count && count < _countof(buf)) {
263 // Copy from stack.
264 str.append(buf, count);
265 return count;
266 }
267 if (count < 0) {
268 switch (errno) {
269 case 0:
270 count = _vsnprintf(NULL, 0, format, arg);
271 assert(count >= 0);
272 break;
273 case EINVAL: throw std::invalid_argument("invalid vsnprintf arguments");
274 case EILSEQ: throw std::runtime_error("encoding error");
275 default: throw std::runtime_error("failed to format string");
276 }
277 }
278 size_t offset = str.size();
279 str.resize(offset + count);
280 if (_vsnprintf(&str[offset], count + 1, format, arg) != count)
281 throw std::runtime_error("failed to format string");
282 return count;
283}
284
294template<class _Traits, class _Ax>
295static int vsprintf(_Inout_ std::basic_string<wchar_t, _Traits, _Ax> &str, _In_z_ _Printf_format_string_ const wchar_t *format, _In_ va_list arg)
296{
297 wchar_t buf[WINSTD_STACK_BUFFER_BYTES/sizeof(wchar_t)];
298
299 // Try with stack buffer first.
300 int count = _vsnwprintf(buf, _countof(buf), format, arg);
301 if (0 <= count && count < _countof(buf)) {
302 // Copy from stack.
303 str.append(buf, count);
304 return count;
305 }
306 if (count < 0) {
307 switch (errno) {
308 case 0:
309 count = _vsnwprintf(NULL, 0, format, arg);
310 assert(count >= 0);
311 break;
312 case EINVAL: throw std::invalid_argument("invalid vsnprintf arguments");
313 case EILSEQ: throw std::runtime_error("encoding error");
314 default: throw std::runtime_error("failed to format string");
315 }
316 }
317 size_t offset = str.size();
318 str.resize(offset + count);
319 if (_vsnwprintf(&str[offset], count + 1, format, arg) != count)
320 throw std::runtime_error("failed to format string");
321 return count;
322}
323
332template<class _Elem, class _Traits, class _Ax>
333static int sprintf(_Inout_ std::basic_string<_Elem, _Traits, _Ax> &str, _In_z_ _Printf_format_string_ const _Elem *format, ...)
334{
335 va_list arg;
336 va_start(arg, format);
337 const int res = vsprintf(str, format, arg);
338 va_end(arg);
339 return res;
340}
341
347template<class _Traits, class _Ax>
348static _Success_(return != 0) int WideCharToMultiByte(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_z_count_(cchWideChar) LPCWSTR lpWideCharStr, _In_ int cchWideChar, _Out_ std::basic_string<char, _Traits, _Ax> &sMultiByteStr, _In_opt_z_ LPCSTR lpDefaultChar, _Out_opt_ LPBOOL lpUsedDefaultChar) noexcept
349{
350 CHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(CHAR)];
351
352 // Try to convert to stack buffer first.
353 int cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, szStackBuffer, _countof(szStackBuffer), lpDefaultChar, lpUsedDefaultChar);
354 if (cch) {
355 // Copy from stack. Be careful not to include zero terminator.
356 sMultiByteStr.assign(szStackBuffer, cchWideChar != -1 ? strnlen(szStackBuffer, cch) : (size_t)cch - 1);
357 }
358 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
359 // Query the required output size. Allocate buffer. Then convert again.
360 cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, NULL, 0, lpDefaultChar, lpUsedDefaultChar);
361 sMultiByteStr.resize(cch);
362 cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, &sMultiByteStr[0], cch, lpDefaultChar, lpUsedDefaultChar);
363 sMultiByteStr.resize(cchWideChar != -1 ? strnlen(&sMultiByteStr[0], cch) : (size_t)cch - 1);
364 }
365
366 return cch;
367}
368
374template<class _Ax>
375static _Success_(return != 0) int WideCharToMultiByte(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_z_count_(cchWideChar) LPCWSTR lpWideCharStr, _In_ int cchWideChar, _Out_ std::vector<char, _Ax> &sMultiByteStr, _In_opt_z_ LPCSTR lpDefaultChar, _Out_opt_ LPBOOL lpUsedDefaultChar) noexcept
376{
377 CHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(CHAR)];
378
379 // Try to convert to stack buffer first.
380 int cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, szStackBuffer, _countof(szStackBuffer), lpDefaultChar, lpUsedDefaultChar);
381 if (cch) {
382 // Copy from stack.
383 sMultiByteStr.assign(szStackBuffer, szStackBuffer + cch);
384 }
385 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
386 // Query the required output size. Allocate buffer. Then convert again.
387 cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, NULL, 0, lpDefaultChar, lpUsedDefaultChar);
388 sMultiByteStr.resize(cch);
389 cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, sMultiByteStr.data(), cch, lpDefaultChar, lpUsedDefaultChar);
390 }
391
392 return cch;
393}
394
400template<class _Traits1, class _Ax1, class _Traits2, class _Ax2>
401static _Success_(return != 0) int WideCharToMultiByte(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_ std::basic_string<wchar_t, _Traits1, _Ax1> sWideCharStr, _Out_ std::basic_string<char, _Traits2, _Ax2> &sMultiByteStr, _In_opt_z_ LPCSTR lpDefaultChar, _Out_opt_ LPBOOL lpUsedDefaultChar) noexcept
402{
403 CHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(CHAR)];
404
405 // Try to convert to stack buffer first.
406 int cch = ::WideCharToMultiByte(CodePage, dwFlags, sWideCharStr.c_str(), (int)sWideCharStr.length(), szStackBuffer, _countof(szStackBuffer), lpDefaultChar, lpUsedDefaultChar);
407 if (cch) {
408 // Copy from stack.
409 sMultiByteStr.assign(szStackBuffer, cch);
410 }
411 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
412 // Query the required output size. Allocate buffer. Then convert again.
413 cch = ::WideCharToMultiByte(CodePage, dwFlags, sWideCharStr.c_str(), (int)sWideCharStr.length(), NULL, 0, lpDefaultChar, lpUsedDefaultChar);
414 sMultiByteStr.resize(cch);
415 cch = ::WideCharToMultiByte(CodePage, dwFlags, sWideCharStr.c_str(), (int)sWideCharStr.length(), &sMultiByteStr[0], cch, lpDefaultChar, lpUsedDefaultChar);
416 }
417
418 return cch;
419}
420
428template<class _Traits, class _Ax>
429static _Success_(return != 0) int SecureWideCharToMultiByte(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_z_count_(cchWideChar) LPCWSTR lpWideCharStr, _In_ int cchWideChar, _Out_ std::basic_string<char, _Traits, _Ax> &sMultiByteStr, _In_opt_z_ LPCSTR lpDefaultChar, _Out_opt_ LPBOOL lpUsedDefaultChar) noexcept
430{
431 CHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(CHAR)];
432
433 // Try to convert to stack buffer first.
434 int cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, szStackBuffer, _countof(szStackBuffer), lpDefaultChar, lpUsedDefaultChar);
435 if (cch) {
436 // Copy from stack. Be careful not to include zero terminator.
437 sMultiByteStr.assign(szStackBuffer, cchWideChar != -1 ? strnlen(szStackBuffer, cch) : (size_t)cch - 1);
438 }
439 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
440 // Query the required output size. Allocate buffer. Then convert again.
441 cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, NULL, 0, lpDefaultChar, lpUsedDefaultChar);
442 sMultiByteStr.resize(cch);
443 cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, &sMultiByteStr[0], cch, lpDefaultChar, lpUsedDefaultChar);
444 sMultiByteStr.resize(cchWideChar != -1 ? strnlen(&sMultiByteStr[0], cch) : (size_t)cch - 1);
445 }
446
447 SecureZeroMemory(szStackBuffer, sizeof(szStackBuffer));
448
449 return cch;
450}
451
459template<class _Ax>
460static _Success_(return != 0) int SecureWideCharToMultiByte(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_z_count_(cchWideChar) LPCWSTR lpWideCharStr, _In_ int cchWideChar, _Out_ std::vector<char, _Ax> &sMultiByteStr, _In_opt_z_ LPCSTR lpDefaultChar, _Out_opt_ LPBOOL lpUsedDefaultChar) noexcept
461{
462 CHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(CHAR)];
463
464 // Try to convert to stack buffer first.
465 int cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, szStackBuffer, _countof(szStackBuffer), lpDefaultChar, lpUsedDefaultChar);
466 if (cch) {
467 // Copy from stack.
468 sMultiByteStr.assign(szStackBuffer, szStackBuffer + cch);
469 }
470 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
471 // Query the required output size. Allocate buffer. Then convert again.
472 cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, NULL, 0, lpDefaultChar, lpUsedDefaultChar);
473 sMultiByteStr.resize(cch);
474 cch = ::WideCharToMultiByte(CodePage, dwFlags, lpWideCharStr, cchWideChar, sMultiByteStr.data(), cch, lpDefaultChar, lpUsedDefaultChar);
475 }
476
477 SecureZeroMemory(szStackBuffer, sizeof(szStackBuffer));
478
479 return cch;
480}
481
489template<class _Traits1, class _Ax1, class _Traits2, class _Ax2>
490static _Success_(return != 0) int SecureWideCharToMultiByte(_In_ UINT CodePage, _In_ DWORD dwFlags, _Out_ std::basic_string<wchar_t, _Traits1, _Ax1> sWideCharStr, _Out_ std::basic_string<char, _Traits2, _Ax2> &sMultiByteStr, _In_opt_z_ LPCSTR lpDefaultChar, _Out_opt_ LPBOOL lpUsedDefaultChar) noexcept
491{
492 CHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(CHAR)];
493
494 // Try to convert to stack buffer first.
495 int cch = ::WideCharToMultiByte(CodePage, dwFlags, sWideCharStr.c_str(), (int)sWideCharStr.length(), szStackBuffer, _countof(szStackBuffer), lpDefaultChar, lpUsedDefaultChar);
496 if (cch) {
497 // Copy from stack.
498 sMultiByteStr.assign(szStackBuffer, cch);
499 }
500 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
501 // Query the required output size. Allocate buffer. Then convert again.
502 cch = ::WideCharToMultiByte(CodePage, dwFlags, sWideCharStr.c_str(), (int)sWideCharStr.length(), NULL, 0, lpDefaultChar, lpUsedDefaultChar);
503 sMultiByteStr.resize(cch);
504 cch = ::WideCharToMultiByte(CodePage, dwFlags, sWideCharStr.c_str(), (int)sWideCharStr.length(), &sMultiByteStr[0], cch, lpDefaultChar, lpUsedDefaultChar);
505 }
506
507 SecureZeroMemory(szStackBuffer, sizeof(szStackBuffer));
508
509 return cch;
510}
511
517template<class _Traits, class _Ax>
518static _Success_(return != 0) int MultiByteToWideChar(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_z_count_(cbMultiByte) LPCSTR lpMultiByteStr, _In_ int cbMultiByte, _Out_ std::basic_string<wchar_t, _Traits, _Ax> &sWideCharStr) noexcept
519{
520 WCHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(WCHAR)];
521
522 // Try to convert to stack buffer first.
523 int cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, szStackBuffer, _countof(szStackBuffer));
524 if (cch) {
525 // Copy from stack.
526 sWideCharStr.assign(szStackBuffer, cbMultiByte != -1 ? wcsnlen(szStackBuffer, cch) : (size_t)cch - 1);
527 }
528 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
529 // Query the required output size. Allocate buffer. Then convert again.
530 cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, NULL, 0);
531 sWideCharStr.resize(cch);
532 cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, &sWideCharStr[0], cch);
533 sWideCharStr.resize(cbMultiByte != -1 ? wcsnlen(&sWideCharStr[0], cch) : (size_t)cch - 1);
534 }
535
536 return cch;
537}
538
544template<class _Ax>
545static _Success_(return != 0) int MultiByteToWideChar(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_z_count_(cbMultiByte) LPCSTR lpMultiByteStr, _In_ int cbMultiByte, _Out_ std::vector<wchar_t, _Ax> &sWideCharStr) noexcept
546{
547 WCHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(WCHAR)];
548
549 // Try to convert to stack buffer first.
550 int cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, szStackBuffer, _countof(szStackBuffer));
551 if (cch) {
552 // Copy from stack.
553 sWideCharStr.assign(szStackBuffer, szStackBuffer + cch);
554 }
555 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
556 // Query the required output size. Allocate buffer. Then convert again.
557 cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, NULL, 0);
558 sWideCharStr.resize(cch);
559 cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, sWideCharStr.data(), cch);
560 }
561
562 return cch;
563}
564
570template<class _Traits1, class _Ax1, class _Traits2, class _Ax2>
571static _Success_(return != 0) int MultiByteToWideChar(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_ const std::basic_string<char, _Traits1, _Ax1> &sMultiByteStr, _Out_ std::basic_string<wchar_t, _Traits2, _Ax2> &sWideCharStr) noexcept
572{
573 WCHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(WCHAR)];
574
575 // Try to convert to stack buffer first.
576 int cch = ::MultiByteToWideChar(CodePage, dwFlags, sMultiByteStr.c_str(), (int)sMultiByteStr.length(), szStackBuffer, _countof(szStackBuffer));
577 if (cch) {
578 // Copy from stack.
579 sWideCharStr.assign(szStackBuffer, cch);
580 }
581 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
582 // Query the required output size. Allocate buffer. Then convert again.
583 cch = ::MultiByteToWideChar(CodePage, dwFlags, sMultiByteStr.c_str(), (int)sMultiByteStr.length(), NULL, 0);
584 sWideCharStr.resize(cch);
585 cch = ::MultiByteToWideChar(CodePage, dwFlags, sMultiByteStr.c_str(), (int)sMultiByteStr.length(), &sWideCharStr[0], cch);
586 }
587
588 return cch;
589}
590
598template<class _Traits, class _Ax>
599static _Success_(return != 0) int SecureMultiByteToWideChar(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_z_count_(cbMultiByte) LPCSTR lpMultiByteStr, _In_ int cbMultiByte, _Out_ std::basic_string<wchar_t, _Traits, _Ax> &sWideCharStr) noexcept
600{
601 WCHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(WCHAR)];
602
603 // Try to convert to stack buffer first.
604 int cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, szStackBuffer, _countof(szStackBuffer));
605 if (cch) {
606 // Copy from stack.
607 sWideCharStr.assign(szStackBuffer, cbMultiByte != -1 ? wcsnlen(szStackBuffer, cch) : (size_t)cch - 1);
608 }
609 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
610 // Query the required output size. Allocate buffer. Then convert again.
611 cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, NULL, 0);
612 sWideCharStr.resize(cch);
613 cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, &sWideCharStr[0], cch);
614 sWideCharStr.resize(cbMultiByte != -1 ? wcsnlen(&sWideCharStr[0], cch) : (size_t)cch - 1);
615 }
616
617 SecureZeroMemory(szStackBuffer, sizeof(szStackBuffer));
618
619 return cch;
620}
621
629template<class _Ax>
630static _Success_(return != 0) int SecureMultiByteToWideChar(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_z_count_(cbMultiByte) LPCSTR lpMultiByteStr, _In_ int cbMultiByte, _Out_ std::vector<wchar_t, _Ax> &sWideCharStr) noexcept
631{
632 WCHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(WCHAR)];
633
634 // Try to convert to stack buffer first.
635 int cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, szStackBuffer, _countof(szStackBuffer));
636 if (cch) {
637 // Copy from stack.
638 sWideCharStr.assign(szStackBuffer, szStackBuffer + cch);
639 }
640 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
641 // Query the required output size. Allocate buffer. Then convert again.
642 cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, NULL, 0);
643 sWideCharStr.resize(cch);
644 cch = ::MultiByteToWideChar(CodePage, dwFlags, lpMultiByteStr, cbMultiByte, sWideCharStr.data(), cch);
645 }
646
647 SecureZeroMemory(szStackBuffer, sizeof(szStackBuffer));
648
649 return cch;
650}
651
659template<class _Traits1, class _Ax1, class _Traits2, class _Ax2>
660static _Success_(return != 0) int SecureMultiByteToWideChar(_In_ UINT CodePage, _In_ DWORD dwFlags, _In_ const std::basic_string<char, _Traits1, _Ax1> &sMultiByteStr, _Out_ std::basic_string<wchar_t, _Traits2, _Ax2> &sWideCharStr) noexcept
661{
662 WCHAR szStackBuffer[WINSTD_STACK_BUFFER_BYTES / sizeof(WCHAR)];
663
664 // Try to convert to stack buffer first.
665 int cch = ::MultiByteToWideChar(CodePage, dwFlags, sMultiByteStr.c_str(), (int)sMultiByteStr.length(), szStackBuffer, _countof(szStackBuffer));
666 if (cch) {
667 // Copy from stack.
668 sWideCharStr.assign(szStackBuffer, cch);
669 }
670 else if (::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
671 // Query the required output size. Allocate buffer. Then convert again.
672 cch = ::MultiByteToWideChar(CodePage, dwFlags, sMultiByteStr.c_str(), (int)sMultiByteStr.length(), NULL, 0);
673 sWideCharStr.resize(cch);
674 cch = ::MultiByteToWideChar(CodePage, dwFlags, sMultiByteStr.c_str(), (int)sMultiByteStr.length(), &sWideCharStr[0], cch);
675 }
676
677 SecureZeroMemory(szStackBuffer, sizeof(szStackBuffer));
678
679 return cch;
680}
681
687template<class _Traits, class _Ax>
688static DWORD FormatMessageA(_In_ DWORD dwFlags, _In_opt_ LPCVOID lpSource, _In_ DWORD dwMessageId, _In_ DWORD dwLanguageId, _Inout_ std::basic_string<char, _Traits, _Ax> &str, _In_opt_ va_list *Arguments)
689{
690 LPSTR lpBuffer;
691 DWORD dwResult = FormatMessageA(dwFlags | FORMAT_MESSAGE_ALLOCATE_BUFFER, lpSource, dwMessageId, dwLanguageId, reinterpret_cast<LPSTR>(&lpBuffer), 0, Arguments);
692 if (dwResult) {
693 str.assign(lpBuffer, dwResult);
694 LocalFree(lpBuffer);
695 }
696 return dwResult;
697}
698
704template<class _Traits, class _Ax>
705static DWORD FormatMessageW(_In_ DWORD dwFlags, _In_opt_ LPCVOID lpSource, _In_ DWORD dwMessageId, _In_ DWORD dwLanguageId, _Inout_ std::basic_string<wchar_t, _Traits, _Ax> &str, _In_opt_ va_list *Arguments)
706{
707 LPWSTR lpBuffer;
708 DWORD dwResult = FormatMessageW(dwFlags | FORMAT_MESSAGE_ALLOCATE_BUFFER, lpSource, dwMessageId, dwLanguageId, reinterpret_cast<LPWSTR>(&lpBuffer), 0, Arguments);
709 if (dwResult) {
710 str.assign(lpBuffer, dwResult);
711 LocalFree(lpBuffer);
712 }
713 return dwResult;
714}
715
717
718#pragma warning(pop)
719
720namespace winstd
721{
724
728#ifdef _UNICODE
729 typedef std::wstring tstring;
730#else
731 typedef std::string tstring;
732#endif
733
737 template <class _Ty>
739 {
741
746
750 template <class _Ty2> LocalFree_delete(const LocalFree_delete<_Ty2>&) {}
751
757 void operator()(_Frees_ptr_opt_ _Ty *_Ptr) const
758 {
759 LocalFree(_Ptr);
760 }
761 };
762
766 template <class _Ty>
767 struct LocalFree_delete<_Ty[]>
768 {
770
774 LocalFree_delete() noexcept {}
775
779 void operator()(_Frees_ptr_opt_ _Ty *_Ptr) const noexcept
780 {
781 LocalFree(_Ptr);
782 }
783
789 template<class _Other>
790 void operator()(_Other *) const
791 {
792 LocalFree(_Ptr);
793 }
794 };
795
800 {
805
811 void operator()(HGLOBAL _Ptr) const
812 {
813 GlobalFree(_Ptr);
814 }
815 };
816
820 template <class T>
822 {
825
826 public:
832 globalmem_accessor(_In_ HGLOBAL hMem) : m_h(hMem)
833 {
834 m_data = reinterpret_cast<T*>(GlobalLock(hMem));
835 if (!m_data)
836 throw win_runtime_error("GlobalLock failed");
837 }
838
845 {
846 GlobalUnlock(m_h);
847 }
848
852 T* data() const noexcept
853 {
854 return m_data;
855 }
856
857 protected:
858 HGLOBAL m_h;
860 };
861
863
866
872 template <class T, const T INVAL>
873 class handle
874 {
875 public:
879 typedef T handle_type;
880
884 static const T invalid;
885
889 handle() noexcept : m_h(invalid)
890 {}
891
897 handle(_In_opt_ handle_type h) noexcept : m_h(h)
898 {}
899
905 handle(_Inout_ handle<handle_type, INVAL> &&h) noexcept
906 {
907 // Transfer handle.
908 m_h = h.m_h;
909 h.m_h = invalid;
910 }
911
912 private:
913 // This class is noncopyable.
914 handle(_In_ const handle<handle_type, INVAL> &h) noexcept {};
915 handle<handle_type, INVAL>& operator=(_In_ const handle<handle_type, INVAL> &h) noexcept {};
916
917 public:
924 {
925 attach(h);
926 return *this;
927 }
928
934 #pragma warning(suppress: 26432) // Move constructor is also present, but not detected by code analysis somehow.
936 {
937 if (this != std::addressof(h)) {
938 // Transfer handle.
939 if (m_h != invalid)
941 m_h = h.m_h;
942 h.m_h = invalid;
943 }
944 return *this;
945 }
946
952 operator handle_type() const
953 {
954 return m_h;
955 }
956
963 {
964 assert(m_h != invalid);
965 return *m_h;
966 }
967
973 {
974 assert(m_h == invalid);
975 return &m_h;
976 }
977
984 {
985 assert(m_h != invalid);
986 return m_h;
987 }
988
999 bool operator!() const
1000 {
1001 return m_h == invalid;
1002 }
1003
1012 bool operator<(_In_opt_ handle_type h) const
1013 {
1014 return m_h < h;
1015 }
1016
1025 bool operator<=(_In_opt_ handle_type h) const
1026 {
1027 return !operator>(h);
1028 }
1029
1038 bool operator>=(_In_opt_ handle_type h) const
1039 {
1040 return !operator<(h);
1041 }
1042
1051 bool operator>(_In_opt_ handle_type h) const
1052 {
1053 return h < m_h;
1054 }
1055
1064 bool operator!=(_In_opt_ handle_type h) const
1065 {
1066 return !operator==(h);
1067 }
1068
1077 bool operator==(_In_opt_ handle_type h) const
1078 {
1079 return m_h == h;
1080 }
1081
1089 void attach(_In_opt_ handle_type h) noexcept
1090 {
1091 if (m_h != invalid)
1092 free_internal();
1093 m_h = h;
1094 }
1095
1102 {
1103 handle_type h = m_h;
1104 m_h = invalid;
1105 return h;
1106 }
1107
1111 void free()
1112 {
1113 if (m_h != invalid) {
1114 free_internal();
1115 m_h = invalid;
1116 }
1117 }
1118
1119 protected:
1123 virtual void free_internal() noexcept = 0;
1124
1125 protected:
1127 };
1128
1129 template <class T, const T INVAL>
1130 const T handle<T, INVAL>::invalid = INVAL;
1131
1135 template <class T, T INVAL>
1136 class dplhandle : public handle<T, INVAL>
1137 {
1138 public:
1142 dplhandle() noexcept
1143 {}
1144
1151 {}
1152
1158 dplhandle<handle_type, INVAL>(_In_ const dplhandle<handle_type, INVAL> &h) : handle<handle_type, INVAL>(duplicate_internal(h.m_h))
1159 {}
1160
1166 dplhandle<handle_type, INVAL>(_Inout_ dplhandle<handle_type, INVAL> &&h) noexcept : handle<handle_type, INVAL>(std::move(h))
1167 {}
1168
1175 {
1177 return *this;
1178 }
1179
1186 {
1187 if (this != std::addressof(h)) {
1188 if (h.m_h != invalid) {
1189 handle_type h_new = duplicate_internal(h.m_h);
1190
1191 if (m_h != invalid)
1192 free_internal();
1193
1194 m_h = h_new;
1195 } else {
1196 if (m_h != invalid)
1197 free_internal();
1198
1199 m_h = invalid;
1200 }
1201 }
1202 return *this;
1203 }
1204
1210 #pragma warning(disable: 26432) // Move constructor is also present, but not detected by code analysis somehow.
1212 {
1214 return *this;
1215 }
1216
1223 {
1224 return m_h != invalid ? duplicate_internal(m_h) : invalid;
1225 }
1226
1233 {
1234 if (m_h != invalid)
1235 free_internal();
1236
1237 m_h = h != invalid ? duplicate_internal(h) : invalid;
1238 }
1239
1240 protected:
1249 virtual handle_type duplicate_internal(_In_ handle_type h) const = 0;
1250 };
1251
1253
1256
1266 inline std::string load_msg_from_res(_In_opt_ HMODULE hModule, _In_ UINT nId, _In_ WORD wLanguage)
1267 {
1268 std::string sResult;
1269 HRSRC hFoundRes = FindResourceExW(hModule, MAKEINTRESOURCEW(6), MAKEINTRESOURCEW(nId), wLanguage);
1270 if (hFoundRes) {
1271 DWORD dwSize = SizeofResource(hModule, hFoundRes);
1272 if (dwSize) {
1273 HGLOBAL hLoadedRes = LoadResource(hModule, hFoundRes);
1274 if (hLoadedRes) {
1275 LPCWSTR szMessage = reinterpret_cast<LPCWSTR>(LockResource(hLoadedRes));
1276 if (szMessage) {
1277 WideCharToMultiByte(CP_UTF8, 0, szMessage, dwSize / sizeof(*szMessage), sResult, NULL, NULL);
1278 return sResult;
1279 } else
1280 SetLastError(ERROR_LOCK_FAILED);
1281 }
1282 }
1283 }
1284 sprintf(sResult, "msg %u", nId);
1285 return sResult;
1286 }
1287
1297 inline std::string fmt_msg_from_res(_In_opt_ HMODULE hModule, _In_ UINT nId, _In_ WORD wLanguage, ...)
1298 {
1299 std::string sResult;
1300 HRSRC hFoundRes = FindResourceExW(hModule, MAKEINTRESOURCEW(6), MAKEINTRESOURCEW(nId), wLanguage);
1301 if (hFoundRes) {
1302 DWORD dwSize = SizeofResource(hModule, hFoundRes);
1303 if (dwSize) {
1304 HGLOBAL hLoadedRes = LoadResource(hModule, hFoundRes);
1305 if (hLoadedRes) {
1306 LPCWSTR szFormat = reinterpret_cast<LPCWSTR>(LockResource(hLoadedRes));
1307 if (szFormat) {
1308 dwSize /= sizeof(*szFormat);
1309 assert(wcsnlen(szFormat, dwSize) < dwSize); // Resource strings must be zero-terminated to make strings directly usable with sprintf.
1310 va_list arg;
1311 va_start(arg, wLanguage);
1312 std::wstring sMessage;
1313 vsprintf(sMessage, szFormat, arg);
1314 va_end(arg);
1315 WideCharToMultiByte(CP_UTF8, 0, sMessage, sResult, NULL, NULL);
1316 return sResult;
1317 } else
1318 SetLastError(ERROR_LOCK_FAILED);
1319 }
1320 }
1321 }
1322 sprintf(sResult, "msg %u", nId);
1323 return sResult;
1324 }
1325
1329 template <typename _Tn>
1330 class num_runtime_error : public std::runtime_error
1331 {
1332 public:
1333 typedef _Tn error_type;
1334
1335 public:
1342 num_runtime_error(_In_ error_type num, _In_ const std::string& msg) :
1343 m_num(num),
1344 runtime_error(msg)
1345 {}
1346
1353 num_runtime_error(_In_ error_type num, _In_opt_z_ const char *msg = nullptr) :
1354 m_num(num),
1355 runtime_error(msg)
1356 {}
1357
1362 {
1363 return m_num;
1364 }
1365
1366 protected:
1368 };
1369
1374 {
1375 public:
1381 last_error_saver(_In_ DWORD error = GetLastError()) :
1382 m_error(error)
1383 {}
1384
1391 {
1392 SetLastError(m_error);
1393 }
1394
1395 protected:
1396 DWORD m_error;
1397 };
1398
1403 {
1404 public:
1410 win_runtime_error(_In_ error_type num) : num_runtime_error<DWORD>(num, message(num))
1411 {}
1412
1419 win_runtime_error(_In_ error_type num, _In_ const std::string& msg) : num_runtime_error<DWORD>(num, msg + ": " + message(num))
1420 {}
1421
1428 win_runtime_error(_In_ error_type num, _In_z_ const char *msg) : num_runtime_error<DWORD>(num, std::string(msg) + ": " + message(num))
1429 {}
1430
1434 win_runtime_error() : num_runtime_error<DWORD>(GetLastError(), message(GetLastError()))
1435 {}
1436
1442 win_runtime_error(_In_ const std::string& msg) : num_runtime_error<DWORD>(GetLastError(), msg + ": " + message(GetLastError()))
1443 {}
1444
1450 win_runtime_error(_In_z_ const char *msg) : num_runtime_error<DWORD>(GetLastError(), std::string(msg) + ": " + message(GetLastError()))
1451 {}
1452
1453 protected:
1460 static std::string message(_In_ error_type num, _In_opt_ DWORD dwLanguageId = 0)
1461 {
1462 last_error_saver last_error_save;
1463 std::wstring wstr;
1464 if (FormatMessageW(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, 0, num, dwLanguageId, wstr, NULL)) {
1465 // Stock Windows error messages contain CRLF. Well... Trim all the trailing white space.
1466 wstr.erase(wstr.find_last_not_of(L" \t\n\r\f\v") + 1);
1467 } else
1468 sprintf(wstr, num >= 0x10000 ? L"Error 0x%X" : L"Error %u", num);
1469 std::string str;
1470 WideCharToMultiByte(CP_UTF8, 0, wstr, str, NULL, NULL);
1471 return str;
1472 }
1473 };
1474
1476
1479
1483 template<class _Elem, class _Traits, class _Ax>
1484 class basic_string_printf : public std::basic_string<_Elem, _Traits, _Ax>
1485 {
1486 public:
1489
1495 basic_string_printf(_In_z_ _Printf_format_string_ const _Elem *format, ...)
1496 {
1497 va_list arg;
1498 va_start(arg, format);
1499 vsprintf(*this, format, arg);
1500 va_end(arg);
1501 }
1502
1504
1507
1514 basic_string_printf(_In_ HINSTANCE hInstance, _In_ UINT nFormatID, ...)
1515 {
1516 _Myt format;
1517 ATLENSURE(format.LoadString(hInstance, nFormatID));
1518
1519 va_list arg;
1520 va_start(arg, nFormatID);
1521 vsprintf(*this, format, arg);
1522 va_end(arg);
1523 }
1524
1532 basic_string_printf(_In_ HINSTANCE hInstance, _In_ WORD wLanguageID, _In_ UINT nFormatID, ...)
1533 {
1534 _Myt format;
1535 ATLENSURE(format.LoadString(hInstance, nFormatID, wLanguageID));
1536
1537 va_list arg;
1538 va_start(arg, nFormatID);
1539 vsprintf(*this, format, arg);
1540 va_end(arg);
1541 }
1542
1544 };
1545
1550
1555
1559#ifdef _UNICODE
1561#else
1563#endif
1564
1568 template<class _Elem, class _Traits, class _Ax>
1569 class basic_string_msg : public std::basic_string<_Elem, _Traits, _Ax>
1570 {
1571 public:
1574
1580 basic_string_msg(_In_z_ _FormatMessage_format_string_ const _Elem *format, ...)
1581 {
1582 va_list arg;
1583 va_start(arg, format);
1584 FormatMessage(FORMAT_MESSAGE_FROM_STRING, format, 0, 0, *this, &arg);
1585 va_end(arg);
1586 }
1587
1589
1592
1599 basic_string_msg(_In_ HINSTANCE hInstance, _In_ UINT nFormatID, ...)
1600 {
1601 _Myt format(GetManager());
1602 ATLENSURE(format.LoadString(hInstance, nFormatID));
1603
1604 va_list arg;
1605 va_start(arg, nFormatID);
1606 FormatMessage(FORMAT_MESSAGE_FROM_STRING, format, 0, 0, *this, &arg);
1607 va_end(arg);
1608 }
1609
1617 basic_string_msg(_In_ HINSTANCE hInstance, _In_ WORD wLanguageID, _In_ UINT nFormatID, ...)
1618 {
1619 _Myt format(GetManager());
1620 ATLENSURE(format.LoadString(hInstance, nFormatID, wLanguageID));
1621
1622 va_list arg;
1623 va_start(arg, nFormatID);
1624 FormatMessage(FORMAT_MESSAGE_FROM_STRING, format, 0, 0, *this, &arg);
1625 va_end(arg);
1626 }
1627
1629
1635 basic_string_msg(_In_ DWORD dwFlags, _In_opt_ LPCVOID lpSource, _In_ DWORD dwMessageId, _In_ DWORD dwLanguageId, _In_opt_ va_list *Arguments)
1636 {
1637 FormatMessage(dwFlags & ~FORMAT_MESSAGE_ARGUMENT_ARRAY, lpSource, dwMessageId, dwLanguageId, *this, Arguments);
1638 }
1639
1645 basic_string_msg(_In_ DWORD dwFlags, _In_opt_ LPCVOID lpSource, _In_ DWORD dwMessageId, _In_ DWORD dwLanguageId, _In_opt_ DWORD_PTR *Arguments)
1646 {
1647 FormatMessage(dwFlags | FORMAT_MESSAGE_ARGUMENT_ARRAY, lpSource, dwMessageId, dwLanguageId, *this, (va_list*)Arguments);
1648 }
1649
1655 basic_string_msg(_In_ DWORD dwFlags, _In_z_ LPCTSTR pszFormat, _In_opt_ va_list *Arguments)
1656 {
1657 FormatMessage(dwFlags & ~FORMAT_MESSAGE_ARGUMENT_ARRAY | FORMAT_MESSAGE_FROM_STRING, pszFormat, 0, 0, *this, Arguments);
1658 }
1659
1665 basic_string_msg(_In_ DWORD dwFlags, _In_z_ LPCTSTR pszFormat, _In_opt_ DWORD_PTR *Arguments)
1666 {
1667 FormatMessage(dwFlags | FORMAT_MESSAGE_ARGUMENT_ARRAY | FORMAT_MESSAGE_FROM_STRING, pszFormat, 0, 0, *this, (va_list*)Arguments);
1668 }
1669 };
1670
1675
1680
1684#ifdef _UNICODE
1685 typedef wstring_msg tstring_msg;
1686#else
1688#endif
1689
1693 template<class _Elem, class _Traits, class _Ax>
1694 class basic_string_guid : public std::basic_string<_Elem, _Traits, _Ax>
1695 {
1696 public:
1699
1706 basic_string_guid(_In_ const GUID &guid, _In_z_ _Printf_format_string_ const _Elem *format)
1707 {
1708 sprintf<_Elem, _Traits, _Ax>(*this, format,
1709 guid.Data1,
1710 guid.Data2,
1711 guid.Data3,
1712 guid.Data4[0], guid.Data4[1],
1713 guid.Data4[2], guid.Data4[3], guid.Data4[4], guid.Data4[5], guid.Data4[6], guid.Data4[7]);
1714 }
1715
1717 };
1718
1722 class string_guid : public basic_string_guid<char, std::char_traits<char>, std::allocator<char> >
1723 {
1724 public:
1727
1733 string_guid(_In_ const GUID &guid) :
1734 basic_string_guid<char, std::char_traits<char>, std::allocator<char> >(guid, "{%08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}")
1735 {}
1736
1738 };
1739
1743 class wstring_guid : public basic_string_guid<wchar_t, std::char_traits<wchar_t>, std::allocator<wchar_t> >
1744 {
1745 public:
1748
1754 wstring_guid(_In_ const GUID &guid) :
1755 basic_string_guid<wchar_t, std::char_traits<wchar_t>, std::allocator<wchar_t> >(guid, L"{%08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}")
1756 {}
1757
1759 };
1760
1764#ifdef _UNICODE
1765 typedef wstring_guid tstring_guid;
1766#else
1768#endif
1769
1771
1774
1775 // winstd::sanitizing_allocator::destroy() member generates _Ptr parameter not used warning for primitive datatypes _Ty.
1776 #pragma warning(push)
1777 #pragma warning(disable: 4100)
1778
1786 template<class _Ty>
1787 class sanitizing_allocator : public std::allocator<_Ty>
1788 {
1789 public:
1790 typedef std::allocator<_Ty> _Mybase;
1791
1795 template<class _Other>
1796 struct rebind
1797 {
1799 };
1800
1805 {}
1806
1811 {}
1812
1816 template<class _Other>
1817 sanitizing_allocator(_In_ const sanitizing_allocator<_Other> &_Othr) noexcept : _Mybase(_Othr)
1818 {}
1819
1823 void deallocate(_In_ _Ty* const _Ptr, _In_ const std::size_t _Count)
1824 {
1825 // Sanitize then free.
1826 SecureZeroMemory(_Ptr, sizeof(_Ty) * _Count);
1827 _Mybase::deallocate(_Ptr, _Count);
1828 }
1829 };
1830
1831 #pragma warning(pop)
1832
1840 typedef std::basic_string<char, std::char_traits<char>, sanitizing_allocator<char> > sanitizing_string;
1841
1849 typedef std::basic_string<wchar_t, std::char_traits<wchar_t>, sanitizing_allocator<wchar_t> > sanitizing_wstring;
1850
1854#ifdef _UNICODE
1856#else
1858#endif
1859
1863 template<size_t N>
1865 {
1866 public:
1871 {
1872 ZeroMemory(m_data, N);
1873 }
1874
1879 {
1880 SecureZeroMemory(m_data, N);
1881 }
1882
1883 public:
1884 unsigned char m_data[N];
1885 };
1886
1888}
Base template class to support converting GUID to string.
Definition Common.h:1695
basic_string_guid(const GUID &guid, const _Elem *format)
Initializes a new string and formats its contents to string representation of given GUID.
Definition Common.h:1706
Base template class to support string formatting using FormatMessage() style templates.
Definition Common.h:1570
basic_string_msg(DWORD dwFlags, LPCVOID lpSource, DWORD dwMessageId, DWORD dwLanguageId, DWORD_PTR *Arguments)
Initializes a new string and formats its contents using FormatMessage() style.
Definition Common.h:1645
basic_string_msg(DWORD dwFlags, LPCTSTR pszFormat, va_list *Arguments)
Initializes a new string and formats its contents using FormatMessage() style.
Definition Common.h:1655
basic_string_msg(HINSTANCE hInstance, WORD wLanguageID, UINT nFormatID,...)
Initializes a new string and formats its contents using FormatMessage() style template in resources.
Definition Common.h:1617
basic_string_msg(DWORD dwFlags, LPCVOID lpSource, DWORD dwMessageId, DWORD dwLanguageId, va_list *Arguments)
Initializes a new string and formats its contents using FormatMessage() style.
Definition Common.h:1635
basic_string_msg(const _Elem *format,...)
Initializes a new string and formats its contents using FormatMessage() style template.
Definition Common.h:1580
basic_string_msg(HINSTANCE hInstance, UINT nFormatID,...)
Initializes a new string and formats its contents using FormatMessage() style template in resources.
Definition Common.h:1599
basic_string_msg(DWORD dwFlags, LPCTSTR pszFormat, DWORD_PTR *Arguments)
Initializes a new string and formats its contents using FormatMessage() style.
Definition Common.h:1665
Base template class to support string formatting using printf() style templates.
Definition Common.h:1485
basic_string_printf(const _Elem *format,...)
Initializes a new string and formats its contents using printf() style template.
Definition Common.h:1495
basic_string_printf(HINSTANCE hInstance, WORD wLanguageID, UINT nFormatID,...)
Initializes a new string and formats its contents using printf() style template in resources.
Definition Common.h:1532
basic_string_printf(HINSTANCE hInstance, UINT nFormatID,...)
Initializes a new string and formats its contents using printf() style template in resources.
Definition Common.h:1514
Base abstract template class to support object handle keeping for objects that support trivial handle...
Definition Common.h:1137
virtual handle_type duplicate_internal(handle_type h) const =0
Abstract member function that must be implemented by child classes to do the actual object handle dup...
dplhandle< handle_type, INVAL > & operator=(handle_type h) noexcept
Attaches already available object handle.
Definition Common.h:1174
handle_type duplicate() const
Duplicates and returns a new object handle.
Definition Common.h:1222
dplhandle< handle_type, INVAL > & operator=(dplhandle< handle_type, INVAL > &&h) noexcept
Moves the object.
Definition Common.h:1211
void attach_duplicated(handle_type h)
Duplicates an object handle and sets a new object handle.
Definition Common.h:1232
dplhandle(handle_type h) noexcept
Initializes a new class instance with an already available object handle.
Definition Common.h:1150
dplhandle< handle_type, INVAL > & operator=(const dplhandle< handle_type, INVAL > &h) noexcept
Duplicates the object.
Definition Common.h:1185
dplhandle() noexcept
Initializes a new class instance with the object handle set to INVAL.
Definition Common.h:1142
Context scope automatic GlobalAlloc (un)access.
Definition Common.h:822
HGLOBAL m_h
memory handle
Definition Common.h:858
virtual ~globalmem_accessor()
Decrements the lock count associated with a memory object.
Definition Common.h:844
T * m_data
memory pointer
Definition Common.h:859
T * data() const noexcept
Return data pointer.
Definition Common.h:852
globalmem_accessor(HGLOBAL hMem)
Locks a global memory object and returns a pointer to the first byte of the object's memory block.
Definition Common.h:832
Base abstract template class to support generic object handle keeping.
Definition Common.h:874
handle_type *& operator*() const
Returns the object handle value when the object handle is a pointer to a value (class,...
Definition Common.h:962
virtual void free_internal() noexcept=0
Abstract member function that must be implemented by child classes to do the actual object destructio...
handle() noexcept
Initializes a new class instance with the object handle set to INVAL.
Definition Common.h:889
bool operator>=(handle_type h) const
Is handle greater than or equal to?
Definition Common.h:1038
handle_type operator->() const
Provides object handle member access when the object handle is a pointer to a class or struct.
Definition Common.h:983
handle_type * operator&()
Returns the object handle reference.
Definition Common.h:972
handle(handle_type h) noexcept
Initializes a new class instance with an already available object handle.
Definition Common.h:897
bool operator<(handle_type h) const
Is handle less than?
Definition Common.h:1012
handle< handle_type, INVAL > & operator=(handle_type h) noexcept
Attaches already available object handle.
Definition Common.h:923
bool operator!() const
Tests if the object handle is invalid.
Definition Common.h:999
handle< handle_type, INVAL > & operator=(handle< handle_type, INVAL > &&h) noexcept
Move assignment.
Definition Common.h:935
bool operator!=(handle_type h) const
Is handle not equal to?
Definition Common.h:1064
void free()
Destroys the object.
Definition Common.h:1111
handle_type m_h
Object handle.
Definition Common.h:1126
void attach(handle_type h) noexcept
Sets a new object handle for the class.
Definition Common.h:1089
bool operator==(handle_type h) const
Is handle equal to?
Definition Common.h:1077
handle(handle< handle_type, INVAL > &&h) noexcept
Move constructor.
Definition Common.h:905
handle_type detach()
Dismisses the object handle from this class.
Definition Common.h:1101
T handle_type
Datatype of the object handle this template class handles.
Definition Common.h:879
bool operator>(handle_type h) const
Is handle greater than?
Definition Common.h:1051
bool operator<=(handle_type h) const
Is handle less than or equal to?
Definition Common.h:1025
Saves GetLastError and restores SetLastError when going out of scope.
Definition Common.h:1374
~last_error_saver()
Sets the last-error code for the calling thread.
Definition Common.h:1390
last_error_saver(DWORD error=GetLastError())
Saves the calling thread's last-error code value.
Definition Common.h:1381
Numerical runtime error.
Definition Common.h:1331
num_runtime_error(error_type num, const char *msg=nullptr)
Constructs an exception.
Definition Common.h:1353
num_runtime_error(error_type num, const std::string &msg)
Constructs an exception.
Definition Common.h:1342
error_type number() const
Returns the error number.
Definition Common.h:1361
error_type m_num
Numeric error code.
Definition Common.h:1367
_Tn error_type
Error number type.
Definition Common.h:1333
An allocator template that sanitizes each memory block before it is destroyed or reallocated.
Definition Common.h:1788
sanitizing_allocator(const sanitizing_allocator< _Ty > &_Othr)
Construct by copying.
Definition Common.h:1810
std::allocator< _Ty > _Mybase
Base type.
Definition Common.h:1790
sanitizing_allocator(const sanitizing_allocator< _Other > &_Othr) noexcept
Construct from a related allocator.
Definition Common.h:1817
void deallocate(_Ty *const _Ptr, const std::size_t _Count)
Deallocate object at _Ptr sanitizing its content first.
Definition Common.h:1823
sanitizing_allocator() noexcept
Construct default allocator.
Definition Common.h:1804
Sanitizing BLOB.
Definition Common.h:1865
sanitizing_blob()
Constructs uninitialized BLOB.
Definition Common.h:1870
~sanitizing_blob()
Sanitizes BLOB.
Definition Common.h:1878
Single-byte character implementation of a class to support converting GUID to string.
Definition Common.h:1723
string_guid(const GUID &guid)
Initializes a new string and formats its contents to string representation of given GUID.
Definition Common.h:1733
Windows runtime error.
Definition Common.h:1403
win_runtime_error(const char *msg)
Constructs an exception using GetLastError()
Definition Common.h:1450
win_runtime_error(error_type num, const char *msg)
Constructs an exception.
Definition Common.h:1428
win_runtime_error(error_type num)
Constructs an exception.
Definition Common.h:1410
win_runtime_error()
Constructs an exception using GetLastError()
Definition Common.h:1434
static std::string message(error_type num, DWORD dwLanguageId=0)
Returns a user-readable Windows error message. As std::exception messages may only be char*,...
Definition Common.h:1460
win_runtime_error(const std::string &msg)
Constructs an exception using GetLastError()
Definition Common.h:1442
win_runtime_error(error_type num, const std::string &msg)
Constructs an exception.
Definition Common.h:1419
Wide character implementation of a class to support converting GUID to string.
Definition Common.h:1744
wstring_guid(const GUID &guid)
Initializes a new string and formats its contents to string representation of given GUID.
Definition Common.h:1754
std::string load_msg_from_res(HMODULE hModule, UINT nId, WORD wLanguage)
Loads exception message string from resources and converts it to UTF-8.
Definition Common.h:1266
std::string fmt_msg_from_res(HMODULE hModule, UINT nId, WORD wLanguage,...)
Loads exception message sprintf template from resources, formats it and converts it to UTF-8.
Definition Common.h:1297
#define WINSTD_NONCOPYABLE(C)
Declares a class as non-copyable.
Definition Common.h:67
#define WINSTD_STACK_BUFFER_BYTES
Size of the stack buffer in bytes used for initial system function call.
Definition Common.h:94
std::string tstring
Multi-byte / Wide-character string (according to _UNICODE)
Definition Common.h:731
#define WINSTD_NONMOVABLE(C)
Declares a class as non-movable.
Definition Common.h:75
std::basic_string< wchar_t, std::char_traits< wchar_t >, sanitizing_allocator< wchar_t > > sanitizing_wstring
A sanitizing variant of std::wstring.
Definition Common.h:1849
sanitizing_string sanitizing_tstring
Multi-byte / Wide-character sanitizing string (according to _UNICODE)
Definition Common.h:1857
std::basic_string< char, std::char_traits< char >, sanitizing_allocator< char > > sanitizing_string
A sanitizing variant of std::string.
Definition Common.h:1840
static int SecureWideCharToMultiByte(UINT CodePage, DWORD dwFlags, LPCWSTR lpWideCharStr, int cchWideChar, std::basic_string< char, _Traits, _Ax > &sMultiByteStr, LPCSTR lpDefaultChar, LPBOOL lpUsedDefaultChar) noexcept
Maps a UTF-16 (wide character) string to a std::string. The new character string is not necessarily f...
Definition Common.h:429
basic_string_printf< wchar_t, std::char_traits< wchar_t >, std::allocator< wchar_t > > wstring_printf
Wide character implementation of a class to support string formatting using printf() style templates.
Definition Common.h:1554
static int MultiByteToWideChar(UINT CodePage, DWORD dwFlags, LPCSTR lpMultiByteStr, int cbMultiByte, std::basic_string< wchar_t, _Traits, _Ax > &sWideCharStr) noexcept
Maps a character string to a UTF-16 (wide character) std::wstring. The character string is not necess...
Definition Common.h:518
static DWORD FormatMessageW(DWORD dwFlags, LPCVOID lpSource, DWORD dwMessageId, DWORD dwLanguageId, std::basic_string< wchar_t, _Traits, _Ax > &str, va_list *Arguments)
Formats a message string.
Definition Common.h:705
static DWORD FormatMessageA(DWORD dwFlags, LPCVOID lpSource, DWORD dwMessageId, DWORD dwLanguageId, std::basic_string< char, _Traits, _Ax > &str, va_list *Arguments)
Formats a message string.
Definition Common.h:688
string_guid tstring_guid
Multi-byte / Wide-character string GUID (according to _UNICODE)
Definition Common.h:1767
basic_string_msg< wchar_t, std::char_traits< wchar_t >, std::allocator< wchar_t > > wstring_msg
Wide character implementation of a class to support string formatting using FormatMessage() style tem...
Definition Common.h:1679
basic_string_printf< char, std::char_traits< char >, std::allocator< char > > string_printf
Single-byte character implementation of a class to support string formatting using printf() style tem...
Definition Common.h:1549
static int SecureMultiByteToWideChar(UINT CodePage, DWORD dwFlags, LPCSTR lpMultiByteStr, int cbMultiByte, std::basic_string< wchar_t, _Traits, _Ax > &sWideCharStr) noexcept
Maps a character string to a UTF-16 (wide character) std::wstring. The character string is not necess...
Definition Common.h:599
string_printf tstring_printf
Multi-byte / Wide-character formatted string (according to _UNICODE)
Definition Common.h:1562
static int WideCharToMultiByte(UINT CodePage, DWORD dwFlags, LPCWSTR lpWideCharStr, int cchWideChar, std::basic_string< char, _Traits, _Ax > &sMultiByteStr, LPCSTR lpDefaultChar, LPBOOL lpUsedDefaultChar) noexcept
Maps a UTF-16 (wide character) string to a std::string. The new character string is not necessarily f...
Definition Common.h:348
static int sprintf(std::basic_string< _Elem, _Traits, _Ax > &str, const _Elem *format,...)
Formats string using printf().
Definition Common.h:333
static int vsprintf(std::basic_string< char, _Traits, _Ax > &str, const char *format, va_list arg)
Formats string using printf().
Definition Common.h:256
basic_string_msg< char, std::char_traits< char >, std::allocator< char > > string_msg
Single-byte character implementation of a class to support string formatting using FormatMessage() st...
Definition Common.h:1674
string_msg tstring_msg
Multi-byte / Wide-character formatted string (according to _UNICODE)
Definition Common.h:1687
static const T invalid
Invalid handle value.
Definition Common.h:884
Deleter for unique_ptr using GlobalFree.
Definition Common.h:800
GlobalFree_delete()
Default construct.
Definition Common.h:804
void operator()(HGLOBAL _Ptr) const
Delete a pointer.
Definition Common.h:811
LocalFree_delete< _Ty > _Myt
This type.
Definition Common.h:769
LocalFree_delete() noexcept
Default construct.
Definition Common.h:774
void operator()(_Other *) const
Delete a pointer of another type.
Definition Common.h:790
void operator()(_Ty *_Ptr) const noexcept
Delete a pointer.
Definition Common.h:779
Deleter for unique_ptr using LocalFree.
Definition Common.h:739
LocalFree_delete< _Ty > _Myt
This type.
Definition Common.h:740
LocalFree_delete(const LocalFree_delete< _Ty2 > &)
Construct from another LocalFree_delete.
Definition Common.h:750
void operator()(_Ty *_Ptr) const
Delete a pointer.
Definition Common.h:757
LocalFree_delete()
Default construct.
Definition Common.h:745
Convert this type to sanitizing_allocator<_Other>
Definition Common.h:1797
sanitizing_allocator< _Other > other
Other type.
Definition Common.h:1798