summaryrefslogtreecommitdiff
path: root/src/mir_core/src/mstring.cpp
blob: f7fa57ccfb7223f5d4aa42fa4ee7f33d0539ee38 (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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
/*

Miranda NG: the free IM client for Microsoft* Windows*

Copyright (c) 2012-17 Miranda NG project (https://miranda-ng.org),
Copyright (c) 2000-12 Miranda IM project,
all portions of this codebase are copyrighted to the people
listed in contributors.txt.

This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
*/

#include "stdafx.h"

/////////////////////////////////////////////////////////////////////////////////////////
// CMBaseString

class CNilMStringData : public CMStringData
{
public:
	CNilMStringData();

public:
	wchar_t achNil[2];
};

CNilMStringData::CNilMStringData()
{
	nRefs = 2;  // Never gets freed
	nDataLength = 0;
	nAllocLength = 0;
	achNil[0] = 0;
	achNil[1] = 0;
}

static CNilMStringData *m_nil = nullptr;

/////////////////////////////////////////////////////////////////////////////////////////
// CMBaseString

MIR_CORE_DLL(CMStringData*) mirstr_allocate(int nChars, int nCharSize)
{
	nChars++; // nil char
	size_t nDataBytes = nCharSize * nChars;
	size_t nTotalSize = nDataBytes + sizeof(CMStringData);

	CMStringData *pData = static_cast<CMStringData*>(malloc(nTotalSize));
	if (pData == nullptr)
		return nullptr;

	pData->nRefs = 1;
	pData->nAllocLength = nChars - 1;
	pData->nDataLength = 0;
	return pData;
}

MIR_CORE_DLL(void) mirstr_free(CMStringData *pData)
{
	free(pData);
}

MIR_CORE_DLL(CMStringData*) mirstr_realloc(CMStringData* pData, int nChars, int nCharSize)
{
	nChars++; // nil char
	ULONG nDataBytes = nCharSize * nChars;
	ULONG nTotalSize = nDataBytes + sizeof(CMStringData);

	CMStringData *pNewData = static_cast<CMStringData*>(realloc(pData, nTotalSize));
	if (pNewData == nullptr)
		return nullptr;

	pNewData->nAllocLength = nChars - 1;
	return pNewData;
}

MIR_CORE_DLL(CMStringData*) mirstr_getNil()
{
	if (m_nil == nullptr)
		m_nil = new CNilMStringData();
	m_nil->AddRef();
	return m_nil;
}

/////////////////////////////////////////////////////////////////////////////////////////
// CMStringData

MIR_CORE_DLL(void) mirstr_lock(CMStringData* pThis)
{
	pThis->nRefs--;  // Locked buffers can't be shared, so no interlocked operation necessary
	if (pThis->nRefs == 0)
		pThis->nRefs = -1;
}

MIR_CORE_DLL(void) mirstr_release(CMStringData* pThis)
{
	if (InterlockedDecrement(&pThis->nRefs) <= 0)
		mirstr_free(pThis);
}

MIR_CORE_DLL(void) mirstr_unlock(CMStringData* pThis)
{
	if (pThis->IsLocked())
	{
		pThis->nRefs++;  // Locked buffers can't be shared, so no interlocked operation necessary
		if (pThis->nRefs == 0)
			pThis->nRefs = 1;
	}
}

/////////////////////////////////////////////////////////////////////////////////////////
// don't remove it
// this code just instantiates templates for CMStringW[A/W]

template CMStringW;
template MIR_CORE_EXPORT CMStringW CALLBACK operator + (const CMStringW& str1, const CMStringW& str2);
template MIR_CORE_EXPORT CMStringW CALLBACK operator+(const CMStringW& str1, const wchar_t *psz2);
template MIR_CORE_EXPORT CMStringW CALLBACK operator+(const wchar_t *psz1, const CMStringW& str2);
template MIR_CORE_EXPORT CMStringW CALLBACK operator+(const CMStringW& str1, wchar_t ch2);
template MIR_CORE_EXPORT CMStringW CALLBACK operator+(const CMStringW& str1, char ch2);
template MIR_CORE_EXPORT CMStringW CALLBACK operator+(wchar_t ch1, const CMStringW& str2);
template MIR_CORE_EXPORT CMStringW CALLBACK operator+(char ch1, const CMStringW& str2);

template CMStringA;
template MIR_CORE_EXPORT CMStringA CALLBACK operator+(const CMStringA& str1, const CMStringA& str2);
template MIR_CORE_EXPORT CMStringA CALLBACK operator+(const CMStringA& str1, const char *psz2);
template MIR_CORE_EXPORT CMStringA CALLBACK operator+(const char *psz1, const CMStringA& str2);
template MIR_CORE_EXPORT CMStringA CALLBACK operator+(const CMStringA& str1, wchar_t ch2);
template MIR_CORE_EXPORT CMStringA CALLBACK operator+(const CMStringA& str1, char ch2);
template MIR_CORE_EXPORT CMStringA CALLBACK operator+(wchar_t ch1, const CMStringA& str2);
template MIR_CORE_EXPORT CMStringA CALLBACK operator+(char ch1, const CMStringA& str2);