qxLib
string_data.inl
Go to the documentation of this file.
1 /**
2 
3  @file string_data.inl
4  @author Khrapov
5  @date 8.11.2020
6  @copyright © Nick Khrapov, 2021. All right reserved.
7 
8 **/
9 
10 namespace qx
11 {
12 
13 template<class traits_t>
14 typename string_data<traits_t>::pointer string_data<traits_t>::data() noexcept
15 {
16  static_assert(
17  sizeof(string_data) == 32 || sizeof(string_data) == 64 || sizeof(string_data) == 128
18  || sizeof(string_data) == 256 || sizeof(string_data) > 256,
19  "The buffer size should be such that the final size of the structure is aligned");
20 
21  if (is_small())
22  return m_Buffer.data();
23  else
24  return m_pData;
25 }
26 
27 template<class traits_t>
29 {
30  if (!is_small())
31  {
32  std::free(m_pData);
33  m_pData = nullptr;
34  }
35 
36  m_nSize = 0;
37  m_nAllocatedSize = 0;
38 }
39 
40 template<class traits_t>
41 bool string_data<traits_t>::resize(size_type nSymbols, size_type nAlign, string_resize_type eType) noexcept
42 {
43  bool bRet = true;
44 
45  typename traits_t::size_type nSymbolsToAllocate =
46  nAlign > 0 ? nAlign * ((nSymbols + 1) / nAlign + 1) : nSymbols + 1;
47 
48  if (eType == string_resize_type::shrink_to_fit // need to decrease size
49  || size() == 0 // string is empty
50  || nSymbolsToAllocate > capacity()) // need to increase size
51  {
52  const bool bSmallAtStart = is_small();
53  const size_type nNewSize = nSymbolsToAllocate * sizeof(value_type);
54 
55  buffer buff;
56 
57  if (nSymbolsToAllocate <= m_Buffer.size())
58  {
59  if (!bSmallAtStart && (traits_t::shrink_to_fit_when_small() || eType == string_resize_type::shrink_to_fit))
60  {
61  // free allocated memory and move string to buffer
62  std::memmove(buff.data(), m_pData, nNewSize);
63  free();
64  m_Buffer = buff;
65  }
66 
67  m_nSize = nSymbolsToAllocate - 1;
68  }
69  else
70  {
71  size_type nStartSize = 0;
72  if (bSmallAtStart)
73  {
74  buff = m_Buffer;
75  nStartSize = size() * sizeof(value_type);
76  }
77 
78  if (void* pNewBlock = std::realloc(bSmallAtStart ? nullptr : m_pData, nNewSize))
79  {
80  m_nAllocatedSize = nSymbolsToAllocate;
81  m_pData = static_cast<typename traits_t::value_type*>(pNewBlock);
82 
83  if (bSmallAtStart)
84  std::memmove(m_pData, buff.data(), nStartSize);
85  }
86  else
87  {
88  bRet = false;
89  }
90  }
91  }
92 
93  if (bRet && eType == string_resize_type::common)
94  m_nSize = nSymbols;
95 
96  return bRet;
97 }
98 
99 template<class traits_t>
100 typename string_data<traits_t>::size_type string_data<traits_t>::size() const noexcept
101 {
102  return m_nSize;
103 }
104 
105 template<class traits_t>
106 typename string_data<traits_t>::size_type string_data<traits_t>::capacity() const noexcept
107 {
108  if (is_small())
109  return m_Buffer.size();
110  else
111  return m_nAllocatedSize;
112 }
113 
114 template<class traits_t>
115 bool string_data<traits_t>::is_small() const noexcept
116 {
117  return m_nAllocatedSize == 0;
118 }
119 
120 } // namespace qx
Represents string data.
Definition: string_data.h:36
bool resize(size_type nSymbols, size_type nAlign, string_resize_type eType) noexcept
Resize string data.
Definition: string_data.inl:41
size_type size() const noexcept
Get string length.
bool is_small() const noexcept
Is string small and fits in the local buffer.
void free() noexcept
Free allocated memory.
Definition: string_data.inl:28
pointer data() noexcept
Get string data: from buffer or from pointer.
Definition: string_data.inl:14
size_type capacity() const noexcept
Get capacity of string.