blob: 1523034dbddbb25da747873f0177dc5e636ea5e1 [file] [log] [blame]
Ted Kremenekea834df2010-11-16 02:03:55 +00001//===- CXString.cpp - Routines for manipulating CXStrings -----------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file defines routines for manipulating CXStrings. It should be the
11// only file that has internal knowledge of the encoding of the data in
12// CXStrings.
13//
14//===----------------------------------------------------------------------===//
15
16#include "CXString.h"
Ted Kremenek0a90d322010-11-17 23:24:11 +000017#include "CXTranslationUnit.h"
Ted Kremenekea834df2010-11-16 02:03:55 +000018#include "clang-c/Index.h"
Chandler Carruthf59edb92012-12-04 09:25:21 +000019#include "clang/Frontend/ASTUnit.h"
Ted Kremeneka60ed472010-11-16 08:15:36 +000020#include "llvm/ADT/SmallString.h"
Ted Kremenekea834df2010-11-16 02:03:55 +000021#include "llvm/Support/ErrorHandling.h"
22
23using namespace clang;
Ted Kremenekea834df2010-11-16 02:03:55 +000024
Dmitri Gribenko17c1a242013-02-01 16:36:31 +000025/// Describes the kind of underlying data in CXString.
26enum CXStringFlag {
27 /// CXString contains a 'const char *' that it doesn't own.
28 CXS_Unmanaged,
29
30 /// CXString contains a 'const char *' that it allocated with malloc().
31 CXS_Malloc,
32
33 /// CXString contains a CXStringBuf that needs to be returned to the
34 /// CXStringPool.
35 CXS_StringBuf
36};
Ted Kremeneka60ed472010-11-16 08:15:36 +000037
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000038namespace clang {
39namespace cxstring {
40
Ted Kremeneka60ed472010-11-16 08:15:36 +000041//===----------------------------------------------------------------------===//
42// Basic generation of CXStrings.
43//===----------------------------------------------------------------------===//
Ted Kremenekea834df2010-11-16 02:03:55 +000044
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000045CXString createEmpty() {
Dmitri Gribenkodc66adb2013-02-01 14:21:22 +000046 CXString Str;
47 Str.data = "";
48 Str.private_flags = CXS_Unmanaged;
49 return Str;
50}
51
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000052CXString createNull() {
Dmitri Gribenkodad4c1a2013-02-01 14:13:32 +000053 CXString Str;
54 Str.data = 0;
55 Str.private_flags = CXS_Unmanaged;
56 return Str;
57}
58
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000059CXString createRef(const char *String) {
Dmitri Gribenko0c4394c2013-02-02 00:02:12 +000060 if (String && String[0] == '\0')
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000061 return createEmpty();
Dmitri Gribenko0c4394c2013-02-02 00:02:12 +000062
Ted Kremenekea834df2010-11-16 02:03:55 +000063 CXString Str;
Dmitri Gribenko0c4394c2013-02-02 00:02:12 +000064 Str.data = String;
65 Str.private_flags = CXS_Unmanaged;
66 return Str;
67}
68
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000069CXString createDup(const char *String) {
Dmitri Gribenko0c4394c2013-02-02 00:02:12 +000070 if (!String)
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000071 return createNull();
Dmitri Gribenko0c4394c2013-02-02 00:02:12 +000072
73 if (String[0] == '\0')
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000074 return createEmpty();
Dmitri Gribenko0c4394c2013-02-02 00:02:12 +000075
76 CXString Str;
77 Str.data = strdup(String);
78 Str.private_flags = CXS_Malloc;
Ted Kremenekea834df2010-11-16 02:03:55 +000079 return Str;
80}
81
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000082CXString createRef(StringRef String) {
Dmitri Gribenko5595ded2013-02-02 02:19:29 +000083 // If the string is not nul-terminated, we have to make a copy.
84 // This is doing a one past end read, and should be removed!
85 if (!String.empty() && String.data()[String.size()] != 0)
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000086 return createDup(String);
Dmitri Gribenko5595ded2013-02-02 02:19:29 +000087
Ted Kremenekea834df2010-11-16 02:03:55 +000088 CXString Result;
Dmitri Gribenko5595ded2013-02-02 02:19:29 +000089 Result.data = String.data();
90 Result.private_flags = (unsigned) CXS_Unmanaged;
91 return Result;
92}
93
Dmitri Gribenkof06faad2013-02-03 13:54:26 +000094CXString createDup(StringRef String) {
Dmitri Gribenko5595ded2013-02-02 02:19:29 +000095 CXString Result;
96 char *Spelling = static_cast<char *>(malloc(String.size() + 1));
97 memmove(Spelling, String.data(), String.size());
98 Spelling[String.size()] = 0;
99 Result.data = Spelling;
100 Result.private_flags = (unsigned) CXS_Malloc;
Ted Kremenekea834df2010-11-16 02:03:55 +0000101 return Result;
102}
103
Dmitri Gribenkof06faad2013-02-03 13:54:26 +0000104CXString createCXString(CXStringBuf *buf) {
Ted Kremeneka60ed472010-11-16 08:15:36 +0000105 CXString Str;
106 Str.data = buf;
107 Str.private_flags = (unsigned) CXS_StringBuf;
108 return Str;
109}
110
111
112//===----------------------------------------------------------------------===//
113// String pools.
114//===----------------------------------------------------------------------===//
115
Dmitri Gribenkof06faad2013-02-03 13:54:26 +0000116CXStringPool::~CXStringPool() {
Dmitri Gribenko9c48d162013-01-26 22:44:19 +0000117 for (std::vector<CXStringBuf *>::iterator I = Pool.begin(), E = Pool.end();
118 I != E; ++I) {
119 delete *I;
Ted Kremeneka60ed472010-11-16 08:15:36 +0000120 }
121}
122
Dmitri Gribenkof06faad2013-02-03 13:54:26 +0000123CXStringBuf *CXStringPool::getCXStringBuf(CXTranslationUnit TU) {
Dmitri Gribenko9c48d162013-01-26 22:44:19 +0000124 if (Pool.empty())
Ted Kremeneka60ed472010-11-16 08:15:36 +0000125 return new CXStringBuf(TU);
Dmitri Gribenko9c48d162013-01-26 22:44:19 +0000126
127 CXStringBuf *Buf = Pool.back();
128 Buf->Data.clear();
129 Pool.pop_back();
130 return Buf;
Ted Kremeneka60ed472010-11-16 08:15:36 +0000131}
132
Dmitri Gribenkof06faad2013-02-03 13:54:26 +0000133CXStringBuf *getCXStringBuf(CXTranslationUnit TU) {
Dmitri Gribenko9c48d162013-01-26 22:44:19 +0000134 return TU->StringPool->getCXStringBuf(TU);
135}
136
Dmitri Gribenkof06faad2013-02-03 13:54:26 +0000137void CXStringBuf::dispose() {
Dmitri Gribenko9c48d162013-01-26 22:44:19 +0000138 TU->StringPool->Pool.push_back(this);
Ted Kremeneka60ed472010-11-16 08:15:36 +0000139}
140
Dmitri Gribenkof06faad2013-02-03 13:54:26 +0000141bool isManagedByPool(CXString str) {
Ted Kremenekbaf82b02011-08-17 22:19:53 +0000142 return ((CXStringFlag) str.private_flags) == CXS_StringBuf;
143}
144
Dmitri Gribenkof06faad2013-02-03 13:54:26 +0000145} // end namespace cxstring
146} // end namespace clang
147
Ted Kremenekea834df2010-11-16 02:03:55 +0000148//===----------------------------------------------------------------------===//
149// libClang public APIs.
150//===----------------------------------------------------------------------===//
151
152extern "C" {
153const char *clang_getCString(CXString string) {
Ted Kremeneka60ed472010-11-16 08:15:36 +0000154 if (string.private_flags == (unsigned) CXS_StringBuf) {
Dmitri Gribenkof06faad2013-02-03 13:54:26 +0000155 return static_cast<const cxstring::CXStringBuf *>(string.data)->Data.data();
Ted Kremeneka60ed472010-11-16 08:15:36 +0000156 }
Dmitri Gribenkoe358fd52013-01-11 23:13:36 +0000157 return static_cast<const char *>(string.data);
Ted Kremenekea834df2010-11-16 02:03:55 +0000158}
159
160void clang_disposeString(CXString string) {
Ted Kremeneka60ed472010-11-16 08:15:36 +0000161 switch ((CXStringFlag) string.private_flags) {
162 case CXS_Unmanaged:
163 break;
164 case CXS_Malloc:
165 if (string.data)
Dmitri Gribenko689d7472013-01-11 23:08:18 +0000166 free(const_cast<void *>(string.data));
Ted Kremeneka60ed472010-11-16 08:15:36 +0000167 break;
168 case CXS_StringBuf:
Dmitri Gribenkof06faad2013-02-03 13:54:26 +0000169 static_cast<cxstring::CXStringBuf *>(
Dmitri Gribenko9c48d162013-01-26 22:44:19 +0000170 const_cast<void *>(string.data))->dispose();
Ted Kremeneka60ed472010-11-16 08:15:36 +0000171 break;
172 }
Ted Kremenekea834df2010-11-16 02:03:55 +0000173}
174} // end: extern "C"
175