//===- CXString.cpp - Routines for manipulating CXStrings -----------------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // This file defines routines for manipulating CXStrings. It should be the // only file that has internal knowledge of the encoding of the data in // CXStrings. // //===----------------------------------------------------------------------===// #include "CXString.h" #include "CXTranslationUnit.h" #include "clang/Frontend/ASTUnit.h" #include "clang-c/Index.h" #include "llvm/ADT/SmallString.h" #include "llvm/Support/ErrorHandling.h" using namespace clang; using namespace clang::cxstring; enum CXStringFlag { CXS_Unmanaged, CXS_Malloc, CXS_StringBuf }; //===----------------------------------------------------------------------===// // Basic generation of CXStrings. //===----------------------------------------------------------------------===// CXString cxstring::createCXString(const char *String, bool DupString){ CXString Str; if (DupString) { Str.data = strdup(String); Str.private_flags = (unsigned) CXS_Malloc; } else { Str.data = (void*)String; Str.private_flags = (unsigned) CXS_Unmanaged; } return Str; } CXString cxstring::createCXString(StringRef String, bool DupString) { CXString Result; if (DupString || (!String.empty() && String.data()[String.size()] != 0)) { char *Spelling = (char *)malloc(String.size() + 1); memmove(Spelling, String.data(), String.size()); Spelling[String.size()] = 0; Result.data = Spelling; Result.private_flags = (unsigned) CXS_Malloc; } else { Result.data = (void*) String.data(); Result.private_flags = (unsigned) CXS_Unmanaged; } return Result; } CXString cxstring::createCXString(CXStringBuf *buf) { CXString Str; Str.data = buf; Str.private_flags = (unsigned) CXS_StringBuf; return Str; } //===----------------------------------------------------------------------===// // String pools. //===----------------------------------------------------------------------===// typedef std::vector<CXStringBuf *> CXStringPool; void *cxstring::createCXStringPool() { return new CXStringPool(); } void cxstring::disposeCXStringPool(void *p) { CXStringPool *pool = static_cast<CXStringPool*>(p); if (pool) { for (CXStringPool::iterator I = pool->begin(), E = pool->end(); I != E; ++I) { delete *I; } delete pool; } } CXStringBuf *cxstring::getCXStringBuf(CXTranslationUnit TU) { CXStringPool *pool = static_cast<CXStringPool*>(TU->StringPool); if (pool->empty()) return new CXStringBuf(TU); CXStringBuf *buf = pool->back(); buf->Data.clear(); pool->pop_back(); return buf; } void cxstring::disposeCXStringBuf(CXStringBuf *buf) { if (buf) static_cast<CXStringPool*>(buf->TU->StringPool)->push_back(buf); } bool cxstring::isManagedByPool(CXString str) { return ((CXStringFlag) str.private_flags) == CXS_StringBuf; } //===----------------------------------------------------------------------===// // libClang public APIs. //===----------------------------------------------------------------------===// extern "C" { const char *clang_getCString(CXString string) { if (string.private_flags == (unsigned) CXS_StringBuf) { return ((CXStringBuf*)string.data)->Data.data(); } return (const char*) string.data; } void clang_disposeString(CXString string) { switch ((CXStringFlag) string.private_flags) { case CXS_Unmanaged: break; case CXS_Malloc: if (string.data) free((void*)string.data); break; case CXS_StringBuf: disposeCXStringBuf((CXStringBuf *) string.data); break; } } } // end: extern "C"