// Protocol Buffers - Google's data interchange format // Copyright 2008 Google Inc. All rights reserved. // http://code.google.com/p/protobuf/ // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above // copyright notice, this list of conditions and the following disclaimer // in the documentation and/or other materials provided with the // distribution. // * Neither the name of Google Inc. nor the names of its // contributors may be used to endorse or promote products derived from // this software without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // Author: kenton@google.com (Kenton Varda) // Based on original Protocol Buffers design by // Sanjay Ghemawat, Jeff Dean, and others. // Copyright (c) 2008-2013, Dave Benson. All rights reserved. // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // // * Redistributions in binary form must reproduce the above // copyright notice, this list of conditions and the following disclaimer // in the documentation and/or other materials provided with the // distribution. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // Modified to implement C code by Dave Benson. #include #include #include #include // for snprintf #include #include #include namespace google { namespace protobuf { namespace compiler { namespace c { #if defined(_MSC_VER) // FIXME: In the case where the generated string is longer than the buffer, // _snprint() returns a negative value, where snprintf() returns the number // of characters that *would* have been stored, had there been room. // That is fundamental, as it allows snprintf() to be used to find the size // necessary for the buffer, simply by calling it with the size of the buffer // passed in as zero. // Note: at the present moment return value isn't used in the code. #define snprintf _snprintf #pragma warning(disable:4800) #pragma warning(disable:4996) #endif std::string DotsToUnderscores(const std::string& name) { return StringReplace(name, ".", "_", true); } std::string DotsToColons(const std::string& name) { return StringReplace(name, ".", "::", true); } std::string SimpleFtoa(float f) { char buf[100]; snprintf(buf,sizeof(buf),"%.*g", FLT_DIG, f); buf[sizeof(buf)-1] = 0; /* should NOT be necessary */ return buf; } std::string SimpleDtoa(double d) { char buf[100]; snprintf(buf,sizeof(buf),"%.*g", DBL_DIG, d); buf[sizeof(buf)-1] = 0; /* should NOT be necessary */ return buf; } std::string CamelToUpper(const std::string &name) { bool was_upper = true; // suppress initial _ std::string rv = ""; int len = name.length(); for (int i = 0; i < len; i++) { bool is_upper = isupper(name[i]); if (is_upper) { if (!was_upper) rv += '_'; rv += name[i]; } else { rv += toupper(name[i]); } was_upper = is_upper; } return rv; } std::string CamelToLower(const std::string &name) { bool was_upper = true; // suppress initial _ std::string rv = ""; int len = name.length(); for (int i = 0; i < len; i++) { bool is_upper = isupper(name[i]); if (is_upper) { if (!was_upper) rv += '_'; rv += tolower(name[i]); } else { rv += name[i]; } was_upper = is_upper; } return rv; } std::string ToUpper(const std::string &name) { std::string rv = ""; int len = name.length(); for (int i = 0; i < len; i++) { rv += toupper(name[i]); } return rv; } std::string ToLower(const std::string &name) { std::string rv = ""; int len = name.length(); for (int i = 0; i < len; i++) { rv += tolower(name[i]); } return rv; } std::string ToCamel(const std::string &name) { std::string rv = ""; int len = name.length(); bool next_is_upper = true; for (int i = 0; i < len; i++) { if (name[i] == '_') { next_is_upper = true; } else if (next_is_upper) { rv += toupper (name[i]); next_is_upper = false; } else { rv += name[i]; } } return rv; } std::string OverrideFullName(const std::string &full_name, const FileDescriptor *file) { const ProtobufCFileOptions opt = file->options().GetExtension(pb_c_file); if (!opt.has_c_package()) return full_name; std::string new_name = opt.c_package(); if (file->package().empty()) new_name += "."; return new_name + full_name.substr(file->package().length()); } std::string FullNameToLower(const std::string &full_name, const FileDescriptor *file) { std::vector pieces; SplitStringUsing(OverrideFullName(full_name, file), ".", &pieces); std::string rv = ""; for (unsigned i = 0; i < pieces.size(); i++) { if (pieces[i] == "") continue; if (rv != "") rv += "__"; rv += CamelToLower(pieces[i]); } return rv; } std::string FullNameToUpper(const std::string &full_name, const FileDescriptor *file) { std::vector pieces; SplitStringUsing(OverrideFullName(full_name, file), ".", &pieces); std::string rv = ""; for (unsigned i = 0; i < pieces.size(); i++) { if (pieces[i] == "") continue; if (rv != "") rv += "__"; rv += CamelToUpper(pieces[i]); } return rv; } std::string FullNameToC(const std::string &full_name, const FileDescriptor *file) { std::vector pieces; SplitStringUsing(OverrideFullName(full_name, file), ".", &pieces); std::string rv = ""; for (unsigned i = 0; i < pieces.size(); i++) { if (pieces[i] == "") continue; if (rv != "") rv += "__"; rv += ToCamel(pieces[i]); } return rv; } void PrintComment (io::Printer* printer, std::string comment) { if (!comment.empty()) { std::vector comment_lines; SplitStringUsing (comment, "\r\n", &comment_lines); printer->Print ("/*\n"); for (size_t i = 0; i < comment_lines.size(); i++) { if (!comment_lines[i].empty()) { /* Make sure we don't inadvertently close the comment block */ if (comment_lines[i][0] == '/') comment_lines[i] = ' ' + comment_lines[i]; /* Or cause other compiler issues. */ size_t delim_i; while ((delim_i = comment_lines[i].find("/*")) != std::string::npos) comment_lines[i][delim_i] = ' '; while ((delim_i = comment_lines[i].find("*/")) != std::string::npos) comment_lines[i][delim_i + 1] = ' '; printer->Print (" *$line$\n", "line", comment_lines[i]); } } printer->Print (" */\n"); } } std::string ConvertToSpaces(const std::string &input) { return std::string(input.size(), ' '); } int compare_name_indices_by_name(const void *a, const void *b) { const NameIndex *ni_a = (const NameIndex *) a; const NameIndex *ni_b = (const NameIndex *) b; return strcmp (ni_a->name, ni_b->name); } std::string CEscape(const std::string& src); const char* const kKeywordList[] = { "and", "and_eq", "asm", "auto", "bitand", "bitor", "bool", "break", "case", "catch", "char", "class", "compl", "const", "const_cast", "continue", "default", "delete", "do", "double", "dynamic_cast", "else", "enum", "explicit", "extern", "false", "float", "for", "friend", "goto", "if", "inline", "int", "long", "mutable", "namespace", "new", "not", "not_eq", "operator", "or", "or_eq", "private", "protected", "public", "register", "reinterpret_cast", "return", "short", "signed", "sizeof", "static", "static_cast", "struct", "switch", "template", "this", "throw", "true", "try", "typedef", "typeid", "typename", "union", "unsigned", "using", "virtual", "void", "volatile", "wchar_t", "while", "xor", "xor_eq" }; std::set MakeKeywordsMap() { std::set result; for (size_t i = 0; i < GOOGLE_ARRAYSIZE(kKeywordList); i++) { result.insert(kKeywordList[i]); } return result; } std::set kKeywords = MakeKeywordsMap(); std::string FieldName(const FieldDescriptor* field) { std::string result = ToLower(field->name()); if (kKeywords.count(result) > 0) { result.append("_"); } return result; } std::string FieldDeprecated(const FieldDescriptor* field) { if (field->options().deprecated()) { return " PROTOBUF_C__DEPRECATED"; } return ""; } std::string StripProto(const std::string& filename) { if (HasSuffixString(filename, ".protodevel")) { return StripSuffixString(filename, ".protodevel"); } else { return StripSuffixString(filename, ".proto"); } } // Convert a file name into a valid identifier. std::string FilenameIdentifier(const std::string& filename) { std::string result; for (unsigned i = 0; i < filename.size(); i++) { if (isalnum(filename[i])) { result.push_back(filename[i]); } else { // Not alphanumeric. To avoid any possibility of name conflicts we // use the hex code for the character. result.push_back('_'); char buffer[32]; result.append(FastHexToBuffer(static_cast(filename[i]), buffer)); } } return result; } // Return the name of the BuildDescriptors() function for a given file. std::string GlobalBuildDescriptorsName(const std::string& filename) { return "proto_BuildDescriptors_" + FilenameIdentifier(filename); } std::string GetLabelName(FieldDescriptor::Label label) { switch (label) { case FieldDescriptor::LABEL_OPTIONAL: return "optional"; case FieldDescriptor::LABEL_REQUIRED: return "required"; case FieldDescriptor::LABEL_REPEATED: return "repeated"; } return "bad-label"; } unsigned WriteIntRanges(io::Printer* printer, int n_values, const int *values, const std::string &name) { std::map vars; vars["name"] = name; if (n_values > 0) { int n_ranges = 1; for (int i = 1; i < n_values; i++) { if (values[i-1] + 1 != values[i]) n_ranges++; } vars["n_ranges"] = SimpleItoa(n_ranges); printer->Print(vars, "static const ProtobufCIntRange $name$[$n_ranges$ + 1] =\n" "{\n"); int last_range_start = 0; for (int i = 1; i < n_values; i++) { if (values[i-1] + 1 != values[i]) { int count = i - last_range_start; int expected = values[i-1] + 1; vars["start_value"] = SimpleItoa(expected - count); vars["orig_offset"] = SimpleItoa(last_range_start); printer->Print (vars, " { $start_value$, $orig_offset$ },\n"); last_range_start = i; } } // write last real entry { int i = n_values; int count = i - last_range_start; int expected = values[i-1] + 1; vars["start_value"] = SimpleItoa(expected - count); vars["orig_offset"] = SimpleItoa(last_range_start); printer->Print (vars, " { $start_value$, $orig_offset$ },\n"); } // write sentinel entry vars["n_entries"] = SimpleItoa(n_values); printer->Print (vars, " { 0, $n_entries$ }\n"); printer->Print (vars, "};\n"); return n_ranges; } else { printer->Print (vars, "#define $name$ NULL\n"); return 0; } } // XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXx // XXXXXXXXX this stuff is copied from strutils.cc !!!! XXXXXXXXXXXXXXXXXXXXXXXXXXXXx // XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXx // ---------------------------------------------------------------------- // StringReplace() // Replace the "old" pattern with the "new" pattern in a string, // and append the result to "res". If replace_all is false, // it only replaces the first instance of "old." // ---------------------------------------------------------------------- void StringReplace(const std::string& s, const std::string& oldsub, const std::string& newsub, bool replace_all, std::string* res) { if (oldsub.empty()) { res->append(s); // if empty, append the given string. return; } std::string::size_type start_pos = 0; std::string::size_type pos; do { pos = s.find(oldsub, start_pos); if (pos == std::string::npos) { break; } res->append(s, start_pos, pos - start_pos); res->append(newsub); start_pos = pos + oldsub.size(); // start searching again after the "old" } while (replace_all); res->append(s, start_pos, s.length() - start_pos); } // ---------------------------------------------------------------------- // StringReplace() // Give me a string and two patterns "old" and "new", and I replace // the first instance of "old" in the string with "new", if it // exists. If "global" is true; call this repeatedly until it // fails. RETURN a new string, regardless of whether the replacement // happened or not. // ---------------------------------------------------------------------- std::string StringReplace(const std::string& s, const std::string& oldsub, const std::string& newsub, bool replace_all) { std::string ret; StringReplace(s, oldsub, newsub, replace_all, &ret); return ret; } // ---------------------------------------------------------------------- // SplitStringUsing() // Split a string using a character delimiter. Append the components // to 'result'. // // Note: For multi-character delimiters, this routine will split on *ANY* of // the characters in the string, not the entire string as a single delimiter. // ---------------------------------------------------------------------- template static inline void SplitStringToIteratorUsing(const std::string& full, const char* delim, ITR& result) { // Optimize the common case where delim is a single character. if (delim[0] != '\0' && delim[1] == '\0') { char c = delim[0]; const char* p = full.data(); const char* end = p + full.size(); while (p != end) { if (*p == c) { ++p; } else { const char* start = p; while (++p != end && *p != c); *result++ = std::string(start, p - start); } } return; } std::string::size_type begin_index, end_index; begin_index = full.find_first_not_of(delim); while (begin_index != std::string::npos) { end_index = full.find_first_of(delim, begin_index); if (end_index == std::string::npos) { *result++ = full.substr(begin_index); return; } *result++ = full.substr(begin_index, (end_index - begin_index)); begin_index = full.find_first_not_of(delim, end_index); } } void SplitStringUsing(const std::string& full, const char* delim, std::vector* result) { std::back_insert_iterator< std::vector > it(*result); SplitStringToIteratorUsing(full, delim, it); } char* FastHexToBuffer(int i, char* buffer) { snprintf(buffer, 16, "%x", i); return buffer; } static int CEscapeInternal(const char* src, int src_len, char* dest, int dest_len, bool use_hex) { const char* src_end = src + src_len; int used = 0; bool last_hex_escape = false; // true if last output char was \xNN for (; src < src_end; src++) { if (dest_len - used < 2) // Need space for two letter escape return -1; bool is_hex_escape = false; switch (*src) { case '\n': dest[used++] = '\\'; dest[used++] = 'n'; break; case '\r': dest[used++] = '\\'; dest[used++] = 'r'; break; case '\t': dest[used++] = '\\'; dest[used++] = 't'; break; case '\"': dest[used++] = '\\'; dest[used++] = '\"'; break; case '\'': dest[used++] = '\\'; dest[used++] = '\''; break; case '\\': dest[used++] = '\\'; dest[used++] = '\\'; break; default: // Note that if we emit \xNN and the src character after that is a hex // digit then that digit must be escaped too to prevent it being // interpreted as part of the character code by C. if (!isprint(*src) || (last_hex_escape && isxdigit(*src))) { if (dest_len - used < 4) // need space for 4 letter escape return -1; snprintf(dest + used, dest_len - used, (use_hex ? "\\x%02x" : "\\%03o"), static_cast(*src)); is_hex_escape = use_hex; used += 4; } else { dest[used++] = *src; break; } } last_hex_escape = is_hex_escape; } if (dest_len - used < 1) // make sure that there is room for \0 return -1; dest[used] = '\0'; // doesn't count towards return value though return used; } std::string CEscape(const std::string& src) { const int dest_length = src.size() * 4 + 1; // Maximum possible expansion std::unique_ptr dest(new char[dest_length]); const int len = CEscapeInternal(src.data(), src.size(), dest.get(), dest_length, false); GOOGLE_DCHECK_GE(len, 0); return std::string(dest.get(), len); } } // namespace c } // namespace compiler } // namespace protobuf } // namespace google