mirror of https://gitlab.com/cppit/libclangmm
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
111 lines
3.4 KiB
111 lines
3.4 KiB
#include "utility.hpp" |
|
#include <regex> |
|
|
|
std::string clangmm::to_string(CXString cx_string) { |
|
std::string string; |
|
if(cx_string.data != NULL) { |
|
string = clang_getCString(cx_string); |
|
clang_disposeString(cx_string); |
|
} |
|
return string; |
|
} |
|
|
|
clangmm::String::String(const CXString &cx_string) : cx_string(cx_string) { |
|
if(cx_string.data != NULL) |
|
c_str = clang_getCString(cx_string); |
|
else |
|
c_str = ""; |
|
} |
|
|
|
clangmm::String::~String() { |
|
if(cx_string.data != NULL) |
|
clang_disposeString(cx_string); |
|
} |
|
|
|
void clangmm::remove_include_guard(std::string &buffer) { |
|
static std::regex ifndef_regex1("^[ \t]*#[ \t]*ifndef[ \t]+([A-Za-z0-9_]+).*$"); |
|
static std::regex ifndef_regex2("^[ \t]*#[ \t]*if[ \t]+![ \t]*defined[ \t]*\\([ \t]*([A-Za-z0-9_]+).*$"); |
|
static std::regex define_regex("^[ \t]*#[ \t]*define[ \t]+([A-Za-z0-9_]+).*$"); |
|
static std::regex endif_regex("^[ \t]*#[ \t]*endif.*$"); |
|
std::vector<std::pair<size_t, size_t>> ranges; |
|
bool found_ifndef = false, found_define = false; |
|
bool line_comment = false, multiline_comment = false; |
|
size_t start_of_line = 0; |
|
std::string line; |
|
std::string preprocessor_identifier; |
|
for(size_t c = 0; c < buffer.size(); ++c) { |
|
if(!line_comment && !multiline_comment && buffer[c] == '/' && c + 1 < buffer.size() && (buffer[c + 1] == '/' || buffer[c + 1] == '*')) { |
|
if(buffer[c + 1] == '/') |
|
line_comment = true; |
|
else |
|
multiline_comment = true; |
|
++c; |
|
} |
|
else if(multiline_comment && buffer[c] == '*' && c + 1 < buffer.size() && buffer[c + 1] == '/') { |
|
multiline_comment = false; |
|
++c; |
|
} |
|
else if(buffer[c] == '\n') { |
|
bool empty_line = true; |
|
for(auto &chr : line) { |
|
if(chr != ' ' && chr != '\t') { |
|
empty_line = false; |
|
break; |
|
} |
|
} |
|
|
|
std::smatch sm; |
|
if(empty_line) { |
|
} |
|
else if(!found_ifndef && (std::regex_match(line, sm, ifndef_regex1) || std::regex_match(line, sm, ifndef_regex2))) { |
|
found_ifndef = true; |
|
ranges.emplace_back(start_of_line, c); |
|
preprocessor_identifier = sm[1].str(); |
|
} |
|
else if(found_ifndef && std::regex_match(line, sm, define_regex)) { |
|
found_define = true; |
|
ranges.emplace_back(start_of_line, c); |
|
if(preprocessor_identifier != sm[1].str()) |
|
return; |
|
break; |
|
} |
|
else |
|
return; |
|
|
|
line_comment = false; |
|
line.clear(); |
|
if(c + 1 < buffer.size()) |
|
start_of_line = c + 1; |
|
else |
|
return; |
|
} |
|
else if(!line_comment && !multiline_comment && buffer[c] != '\r') |
|
line += buffer[c]; |
|
} |
|
if(found_ifndef && found_define) { |
|
size_t last_char_pos = std::string::npos; |
|
for(size_t c = buffer.size() - 1; c != std::string::npos; --c) { |
|
if(last_char_pos == std::string::npos) { |
|
if(buffer[c] != ' ' && buffer[c] != '\t' && buffer[c] != '\r' && buffer[c] != '\n') |
|
last_char_pos = c; |
|
} |
|
else { |
|
if(buffer[c] == '\n' && c + 1 < buffer.size()) { |
|
auto line = buffer.substr(c + 1, last_char_pos - c); |
|
std::smatch sm; |
|
if(std::regex_match(line, sm, endif_regex)) { |
|
ranges.emplace_back(c + 1, last_char_pos + 1); |
|
for(auto &range : ranges) { |
|
for(size_t c = range.first; c < range.second; ++c) { |
|
if(buffer[c] != '\r') |
|
buffer[c] = ' '; |
|
} |
|
} |
|
return; |
|
} |
|
return; |
|
} |
|
} |
|
} |
|
} |
|
}
|
|
|