1 
2 //===----------------------------------------------------------------------===//
3 //
4 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5 // See https://llvm.org/LICENSE.txt for license information.
6 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #ifndef KMP_SAFE_C_API_H
11 #define KMP_SAFE_C_API_H
12 
13 #include <type_traits>
14 #include "kmp_platform.h"
15 #include <string.h>
16 
17 // Replacement for banned C API
18 
19 // Not every unsafe call listed here is handled now, but keeping everything
20 // in one place should be handy for future maintenance.
21 #if KMP_OS_WINDOWS && KMP_MSVC_COMPAT
22 
23 #define RSIZE_MAX_STR (4UL << 10) // 4KB
24 
25 // _malloca was suggested, but it is not a drop-in replacement for _alloca
26 #define KMP_ALLOCA _alloca
27 
28 #define KMP_MEMCPY_S memcpy_s
29 #define KMP_SNPRINTF sprintf_s
30 #define KMP_SSCANF sscanf_s
31 #define KMP_STRCPY_S strcpy_s
32 #define KMP_STRNCPY_S strncpy_s
33 #define KMP_STRNCAT_S strncat_s
34 
35 // Use this only when buffer size is unknown
36 #define KMP_MEMCPY(dst, src, cnt) memcpy_s(dst, cnt, src, cnt)
37 
38 template <typename T, bool B = std::is_array<T>::value>
39 struct kmp_get_rmax_t {};
40 template <typename T> struct kmp_get_rmax_t<T, false> {
41   static const size_t value = RSIZE_MAX_STR;
42 };
43 template <typename T> struct kmp_get_rmax_t<T, true> {
44   static const size_t value = sizeof(T);
45 };
46 #define KMP_STRLEN(str) strnlen_s(str, kmp_get_rmax_t<decltype(str)>::value)
47 
48 // Use this only when buffer size is unknown
49 #define KMP_STRNCPY(dst, src, cnt) strncpy_s(dst, cnt, src, cnt)
50 
51 // _TRUNCATE insures buffer size > max string to print.
52 #define KMP_VSNPRINTF(dst, cnt, fmt, arg)                                      \
53   vsnprintf_s(dst, cnt, _TRUNCATE, fmt, arg)
54 
55 #else // KMP_OS_WINDOWS
56 
57 // For now, these macros use the existing API.
58 
59 #if KMP_OS_NETBSD
60 #define KMP_ALLOCA __builtin_alloca
61 #else
62 #define KMP_ALLOCA alloca
63 #endif
64 #define KMP_MEMCPY_S(dst, bsz, src, cnt) memcpy(dst, src, cnt)
65 #define KMP_SNPRINTF snprintf
66 #define KMP_SSCANF sscanf
67 #define KMP_STRCPY_S(dst, bsz, src) strcpy(dst, src)
68 #define KMP_STRNCPY_S(dst, bsz, src, cnt) strncpy(dst, src, cnt)
69 #define KMP_STRNCAT_S(dst, bsz, src, cnt) strncat(dst, src, cnt)
70 #define KMP_VSNPRINTF vsnprintf
71 #define KMP_STRNCPY strncpy
72 #define KMP_STRLEN strlen
73 #define KMP_MEMCPY memcpy
74 
75 #endif // KMP_OS_WINDOWS
76 
77 // Offer truncated version of strncpy
78 static inline void __kmp_strncpy_truncate(char *buffer, size_t buf_size,
79                                           char const *src, size_t src_size) {
80   if (src_size >= buf_size) {
81     src_size = buf_size - 1;
82   }
83   KMP_STRNCPY_S(buffer, buf_size, src, src_size);
84   buffer[src_size] = '\0';
85 }
86 
87 #endif // KMP_SAFE_C_API_H
88