2008-08-22 20:38:05 +00:00
|
|
|
// Protocol Buffers - Google's data interchange format
|
|
|
|
// Copyright 2008 Google Inc.
|
|
|
|
// http://code.google.com/p/protobuf/
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
// Author: kenton@google.com (Kenton Varda)
|
|
|
|
// Based on original Protocol Buffers design by
|
|
|
|
// Sanjay Ghemawat, Jeff Dean, and others.
|
|
|
|
|
|
|
|
// Modified to implement C code by Dave Benson.
|
|
|
|
|
|
|
|
#include <google/protobuf/compiler/c/c_generator.h>
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
#include <google/protobuf/compiler/c/c_file.h>
|
|
|
|
#include <google/protobuf/compiler/c/c_helpers.h>
|
|
|
|
#include <google/protobuf/io/printer.h>
|
|
|
|
#include <google/protobuf/io/zero_copy_stream.h>
|
|
|
|
#include <google/protobuf/descriptor.pb.h>
|
|
|
|
|
|
|
|
namespace google {
|
|
|
|
namespace protobuf {
|
|
|
|
namespace compiler {
|
|
|
|
namespace c {
|
|
|
|
|
|
|
|
// Parses a set of comma-delimited name/value pairs, e.g.:
|
|
|
|
// "foo=bar,baz,qux=corge"
|
|
|
|
// parses to the pairs:
|
|
|
|
// ("foo", "bar"), ("baz", ""), ("qux", "corge")
|
|
|
|
void ParseOptions(const string& text, vector<pair<string, string> >* output) {
|
|
|
|
vector<string> parts;
|
|
|
|
SplitStringUsing(text, ",", &parts);
|
|
|
|
|
2011-03-10 19:12:35 +00:00
|
|
|
for (unsigned i = 0; i < parts.size(); i++) {
|
2008-08-22 20:38:05 +00:00
|
|
|
string::size_type equals_pos = parts[i].find_first_of('=');
|
|
|
|
pair<string, string> value;
|
|
|
|
if (equals_pos == string::npos) {
|
|
|
|
value.first = parts[i];
|
|
|
|
value.second = "";
|
|
|
|
} else {
|
|
|
|
value.first = parts[i].substr(0, equals_pos);
|
|
|
|
value.second = parts[i].substr(equals_pos + 1);
|
|
|
|
}
|
|
|
|
output->push_back(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CGenerator::CGenerator() {}
|
|
|
|
CGenerator::~CGenerator() {}
|
|
|
|
|
|
|
|
bool CGenerator::Generate(const FileDescriptor* file,
|
|
|
|
const string& parameter,
|
|
|
|
OutputDirectory* output_directory,
|
|
|
|
string* error) const {
|
|
|
|
vector<pair<string, string> > options;
|
|
|
|
ParseOptions(parameter, &options);
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------
|
|
|
|
// parse generator options
|
|
|
|
|
|
|
|
// TODO(kenton): If we ever have more options, we may want to create a
|
|
|
|
// class that encapsulates them which we can pass down to all the
|
|
|
|
// generator classes. Currently we pass dllexport_decl down to all of
|
|
|
|
// them via the constructors, but we don't want to have to add another
|
|
|
|
// constructor parameter for every option.
|
|
|
|
|
|
|
|
// If the dllexport_decl option is passed to the compiler, we need to write
|
|
|
|
// it in front of every symbol that should be exported if this .proto is
|
|
|
|
// compiled into a Windows DLL. E.g., if the user invokes the protocol
|
|
|
|
// compiler as:
|
|
|
|
// protoc --cpp_out=dllexport_decl=FOO_EXPORT:outdir foo.proto
|
|
|
|
// then we'll define classes like this:
|
|
|
|
// class FOO_EXPORT Foo {
|
|
|
|
// ...
|
|
|
|
// }
|
|
|
|
// FOO_EXPORT is a macro which should expand to __declspec(dllexport) or
|
|
|
|
// __declspec(dllimport) depending on what is being compiled.
|
|
|
|
string dllexport_decl;
|
|
|
|
|
2011-03-10 19:12:35 +00:00
|
|
|
for (unsigned i = 0; i < options.size(); i++) {
|
2008-08-22 20:38:05 +00:00
|
|
|
if (options[i].first == "dllexport_decl") {
|
|
|
|
dllexport_decl = options[i].second;
|
|
|
|
} else {
|
|
|
|
*error = "Unknown generator option: " + options[i].first;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
string basename = StripProto(file->name());
|
|
|
|
basename.append(".pb-c");
|
|
|
|
|
|
|
|
FileGenerator file_generator(file, dllexport_decl);
|
|
|
|
|
|
|
|
// Generate header.
|
|
|
|
{
|
|
|
|
scoped_ptr<io::ZeroCopyOutputStream> output(
|
|
|
|
output_directory->Open(basename + ".h"));
|
|
|
|
io::Printer printer(output.get(), '$');
|
|
|
|
file_generator.GenerateHeader(&printer);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate cc file.
|
|
|
|
{
|
|
|
|
scoped_ptr<io::ZeroCopyOutputStream> output(
|
|
|
|
output_directory->Open(basename + ".c"));
|
|
|
|
io::Printer printer(output.get(), '$');
|
|
|
|
file_generator.GenerateSource(&printer);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace c
|
|
|
|
} // namespace compiler
|
|
|
|
} // namespace protobuf
|
|
|
|
} // namespace google
|