2010-06-09 00:52:24 +08:00
|
|
|
//===-- StringList.cpp ------------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "lldb/Core/StringList.h"
|
2011-07-15 10:26:42 +08:00
|
|
|
|
|
|
|
#include "lldb/Core/StreamString.h"
|
2011-02-18 08:54:25 +08:00
|
|
|
#include "lldb/Host/FileSpec.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
using namespace lldb_private;
|
|
|
|
|
|
|
|
StringList::StringList () :
|
|
|
|
m_strings ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
StringList::StringList (const char *str) :
|
|
|
|
m_strings ()
|
|
|
|
{
|
|
|
|
if (str)
|
|
|
|
m_strings.push_back (str);
|
|
|
|
}
|
|
|
|
|
|
|
|
StringList::StringList (const char **strv, int strc) :
|
|
|
|
m_strings ()
|
|
|
|
{
|
|
|
|
for (int i = 0; i < strc; ++i)
|
|
|
|
{
|
|
|
|
if (strv[i])
|
|
|
|
m_strings.push_back (strv[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
StringList::~StringList ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
StringList::AppendString (const char *str)
|
|
|
|
{
|
|
|
|
if (str)
|
|
|
|
m_strings.push_back (str);
|
|
|
|
}
|
|
|
|
|
2012-08-23 01:17:09 +08:00
|
|
|
void
|
|
|
|
StringList::AppendString (const std::string &s)
|
|
|
|
{
|
|
|
|
m_strings.push_back (s);
|
|
|
|
}
|
|
|
|
|
2014-01-28 07:43:24 +08:00
|
|
|
void
|
|
|
|
StringList::AppendString (std::string &&s)
|
|
|
|
{
|
|
|
|
m_strings.push_back (s);
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
void
|
|
|
|
StringList::AppendString (const char *str, size_t str_len)
|
|
|
|
{
|
|
|
|
if (str)
|
|
|
|
m_strings.push_back (std::string (str, str_len));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
StringList::AppendList (const char **strv, int strc)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < strc; ++i)
|
|
|
|
{
|
|
|
|
if (strv[i])
|
|
|
|
m_strings.push_back (strv[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
StringList::AppendList (StringList strings)
|
|
|
|
{
|
2013-01-26 02:06:21 +08:00
|
|
|
size_t len = strings.GetSize();
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2013-01-26 02:06:21 +08:00
|
|
|
for (size_t i = 0; i < len; ++i)
|
2010-06-09 00:52:24 +08:00
|
|
|
m_strings.push_back (strings.GetStringAtIndex(i));
|
|
|
|
}
|
|
|
|
|
2011-02-18 08:54:25 +08:00
|
|
|
bool
|
|
|
|
StringList::ReadFileLines (FileSpec &input_file)
|
|
|
|
{
|
|
|
|
return input_file.ReadFileLines (m_strings);
|
|
|
|
}
|
|
|
|
|
2013-01-26 02:06:21 +08:00
|
|
|
size_t
|
2011-03-11 09:48:52 +08:00
|
|
|
StringList::GetSize () const
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
return m_strings.size();
|
|
|
|
}
|
|
|
|
|
2014-01-28 07:43:24 +08:00
|
|
|
size_t
|
|
|
|
StringList::GetMaxStringLength () const
|
|
|
|
{
|
|
|
|
size_t max_length = 0;
|
|
|
|
for (const auto &s : m_strings)
|
|
|
|
{
|
|
|
|
const size_t len = s.size();
|
|
|
|
if (max_length < len)
|
|
|
|
max_length = len;
|
|
|
|
}
|
|
|
|
return max_length;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
const char *
|
2011-03-11 09:48:52 +08:00
|
|
|
StringList::GetStringAtIndex (size_t idx) const
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
if (idx < m_strings.size())
|
|
|
|
return m_strings[idx].c_str();
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2012-05-27 01:21:14 +08:00
|
|
|
void
|
|
|
|
StringList::Join (const char *separator, Stream &strm)
|
2012-05-26 08:32:39 +08:00
|
|
|
{
|
2013-01-26 02:06:21 +08:00
|
|
|
size_t size = GetSize();
|
2012-05-27 01:21:14 +08:00
|
|
|
|
2012-05-26 08:32:39 +08:00
|
|
|
if (size == 0)
|
2012-05-27 01:21:14 +08:00
|
|
|
return;
|
|
|
|
|
2012-05-26 08:32:39 +08:00
|
|
|
for (uint32_t i = 0; i < size; ++i)
|
|
|
|
{
|
|
|
|
if (i > 0)
|
2012-05-27 01:21:14 +08:00
|
|
|
strm.PutCString(separator);
|
|
|
|
strm.PutCString(GetStringAtIndex(i));
|
2012-05-26 08:32:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
void
|
|
|
|
StringList::Clear ()
|
|
|
|
{
|
|
|
|
m_strings.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
StringList::LongestCommonPrefix (std::string &common_prefix)
|
|
|
|
{
|
2014-01-28 07:43:24 +08:00
|
|
|
const size_t num_strings = m_strings.size();
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2014-01-28 07:43:24 +08:00
|
|
|
if (num_strings == 0)
|
|
|
|
{
|
2010-06-09 00:52:24 +08:00
|
|
|
common_prefix.clear();
|
2014-01-28 07:43:24 +08:00
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
else
|
|
|
|
{
|
2014-01-28 07:43:24 +08:00
|
|
|
common_prefix = m_strings.front();
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2014-01-28 07:43:24 +08:00
|
|
|
for (size_t idx = 1; idx < num_strings; ++idx)
|
|
|
|
{
|
|
|
|
std::string &curr_string = m_strings[idx];
|
|
|
|
size_t new_size = curr_string.size();
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2014-01-28 07:43:24 +08:00
|
|
|
// First trim common_prefix if it is longer than the current element:
|
|
|
|
if (common_prefix.size() > new_size)
|
|
|
|
common_prefix.erase (new_size);
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2014-01-28 07:43:24 +08:00
|
|
|
// Then trim it at the first disparity:
|
|
|
|
for (size_t i = 0; i < common_prefix.size(); i++)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2014-01-28 07:43:24 +08:00
|
|
|
if (curr_string[i] != common_prefix[i])
|
|
|
|
{
|
|
|
|
common_prefix.erase(i);
|
|
|
|
break;
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2014-01-28 07:43:24 +08:00
|
|
|
// If we've emptied the common prefix, we're done.
|
|
|
|
if (common_prefix.empty())
|
|
|
|
break;
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
StringList::InsertStringAtIndex (size_t idx, const char *str)
|
|
|
|
{
|
|
|
|
if (str)
|
|
|
|
{
|
|
|
|
if (idx < m_strings.size())
|
|
|
|
m_strings.insert (m_strings.begin() + idx, str);
|
|
|
|
else
|
|
|
|
m_strings.push_back (str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-28 07:43:24 +08:00
|
|
|
void
|
|
|
|
StringList::InsertStringAtIndex (size_t idx, const std::string &str)
|
|
|
|
{
|
|
|
|
if (idx < m_strings.size())
|
|
|
|
m_strings.insert (m_strings.begin() + idx, str);
|
|
|
|
else
|
|
|
|
m_strings.push_back (str);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
StringList::InsertStringAtIndex (size_t idx, std::string &&str)
|
|
|
|
{
|
|
|
|
if (idx < m_strings.size())
|
|
|
|
m_strings.insert (m_strings.begin() + idx, str);
|
|
|
|
else
|
|
|
|
m_strings.push_back (str);
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
void
|
|
|
|
StringList::DeleteStringAtIndex (size_t idx)
|
|
|
|
{
|
|
|
|
if (idx < m_strings.size())
|
|
|
|
m_strings.erase (m_strings.begin() + idx);
|
|
|
|
}
|
|
|
|
|
2014-01-28 07:43:24 +08:00
|
|
|
size_t
|
|
|
|
StringList::SplitIntoLines (const std::string &lines)
|
|
|
|
{
|
|
|
|
return SplitIntoLines (lines.c_str(), lines.size());
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
size_t
|
|
|
|
StringList::SplitIntoLines (const char *lines, size_t len)
|
|
|
|
{
|
|
|
|
const size_t orig_size = m_strings.size();
|
|
|
|
|
|
|
|
if (len == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
const char *k_newline_chars = "\r\n";
|
|
|
|
const char *p = lines;
|
|
|
|
const char *end = lines + len;
|
|
|
|
while (p < end)
|
|
|
|
{
|
|
|
|
size_t count = strcspn (p, k_newline_chars);
|
|
|
|
if (count == 0)
|
|
|
|
{
|
|
|
|
if (p[count] == '\r' || p[count] == '\n')
|
|
|
|
m_strings.push_back(std::string());
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (p + count > end)
|
|
|
|
count = end - p;
|
|
|
|
m_strings.push_back(std::string(p, count));
|
|
|
|
}
|
|
|
|
if (p[count] == '\r' && p[count+1] == '\n')
|
|
|
|
count++; // Skip an extra newline char for the DOS newline
|
|
|
|
count++; // Skip the newline character
|
|
|
|
p += count;
|
|
|
|
}
|
|
|
|
return m_strings.size() - orig_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
StringList::RemoveBlankLines ()
|
|
|
|
{
|
|
|
|
if (GetSize() == 0)
|
|
|
|
return;
|
|
|
|
|
2010-07-10 04:39:50 +08:00
|
|
|
size_t idx = 0;
|
2010-06-09 00:52:24 +08:00
|
|
|
while (idx < m_strings.size())
|
|
|
|
{
|
|
|
|
if (m_strings[idx].empty())
|
|
|
|
DeleteStringAtIndex(idx);
|
|
|
|
else
|
|
|
|
idx++;
|
|
|
|
}
|
|
|
|
}
|
2011-07-15 10:26:42 +08:00
|
|
|
|
|
|
|
std::string
|
2014-01-28 07:43:24 +08:00
|
|
|
StringList::CopyList(const char* item_preamble, const char* items_sep) const
|
2011-07-15 10:26:42 +08:00
|
|
|
{
|
|
|
|
StreamString strm;
|
2013-06-20 03:04:53 +08:00
|
|
|
for (size_t i = 0; i < GetSize(); i++)
|
2011-07-15 10:26:42 +08:00
|
|
|
{
|
|
|
|
if (i && items_sep && items_sep[0])
|
|
|
|
strm << items_sep;
|
|
|
|
if (item_preamble)
|
|
|
|
strm << item_preamble;
|
|
|
|
strm << GetStringAtIndex(i);
|
|
|
|
}
|
|
|
|
return std::string(strm.GetData());
|
|
|
|
}
|
|
|
|
|
|
|
|
StringList&
|
|
|
|
StringList::operator << (const char* str)
|
|
|
|
{
|
|
|
|
AppendString(str);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringList&
|
|
|
|
StringList::operator << (StringList strings)
|
|
|
|
{
|
|
|
|
AppendList(strings);
|
|
|
|
return *this;
|
2011-08-10 10:10:13 +08:00
|
|
|
}
|
2012-08-23 01:17:09 +08:00
|
|
|
|
|
|
|
size_t
|
|
|
|
StringList::AutoComplete (const char *s, StringList &matches, size_t &exact_idx) const
|
|
|
|
{
|
|
|
|
matches.Clear();
|
|
|
|
exact_idx = SIZE_MAX;
|
|
|
|
if (s && s[0])
|
|
|
|
{
|
|
|
|
const size_t s_len = strlen (s);
|
|
|
|
const size_t num_strings = m_strings.size();
|
|
|
|
|
|
|
|
for (size_t i=0; i<num_strings; ++i)
|
|
|
|
{
|
|
|
|
if (m_strings[i].find(s) == 0)
|
|
|
|
{
|
|
|
|
if (exact_idx == SIZE_MAX && m_strings[i].size() == s_len)
|
|
|
|
exact_idx = matches.GetSize();
|
|
|
|
matches.AppendString (m_strings[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-08-23 08:22:02 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// No string, so it matches everything
|
|
|
|
matches = *this;
|
|
|
|
}
|
2012-08-23 01:17:09 +08:00
|
|
|
return matches.GetSize();
|
|
|
|
}
|
|
|
|
|