/* * Copyright (C) 2011 The Android Open Source Project * * 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. */ #include <stdio.h> #include <stdlib.h> #include "errors.h" #include "EntryPoint.h" #include "strUtils.h" #include "ApiGen.h" #include "TypeFactory.h" #include "getopt.h" const std::string SPEC_EXTENSION = std::string(".in"); const std::string ATTRIB_EXTENSION = std::string(".attrib"); const std::string TYPES_EXTENTION = std::string(".types"); void usage(const char *filename) { fprintf(stderr, "Usage: %s [options] <base name>\n", filename); fprintf(stderr, "\t-h: This message\n"); fprintf(stderr, "\t-E <dir>: generate encoder into dir\n"); fprintf(stderr, "\t-D <dir>: generate decoder into dir\n"); fprintf(stderr, "\t-i: input dir, local directory by default\n"); fprintf(stderr, "\t-T : generate attribute template into the input directory\n\t\tno other files are generated\n"); fprintf(stderr, "\t-W : generate wrapper into dir\n"); } int main(int argc, char *argv[]) { std::string encoderDir = ""; std::string decoderDir = ""; std::string wrapperDir = ""; std::string inDir = "."; bool generateAttributesTemplate = false; int c; while((c = getopt(argc, argv, "TE:D:i:hW:")) != -1) { switch(c) { case 'W': wrapperDir = std::string(optarg); break; case 'T': generateAttributesTemplate = true; break; case 'h': usage(argv[0]); exit(0); break; case 'E': encoderDir = std::string(optarg); break; case 'D': decoderDir = std::string(optarg); break; case 'i': inDir = std::string(optarg); break; case ':': fprintf(stderr, "Missing argument !!\n"); [[fallthrough]]; default: usage(argv[0]); exit(0); } } if (optind >= argc) { fprintf(stderr, "Usage: %s [options] <base name> \n", argv[0]); return BAD_USAGE; } if (encoderDir.size() == 0 && decoderDir.size() == 0 && generateAttributesTemplate == false && wrapperDir.size() == 0) { fprintf(stderr, "No output specified - aborting\n"); return BAD_USAGE; } std::string baseName = std::string(argv[optind]); ApiGen apiEntries(baseName); // init types; std::string typesFilename = inDir + "/" + baseName + TYPES_EXTENTION; if (TypeFactory::instance()->initFromFile(typesFilename) < 0) { fprintf(stderr, "missing or error reading types file: %s...ignored\n", typesFilename.c_str()); } std::string filename = inDir + "/" + baseName + SPEC_EXTENSION; if (apiEntries.readSpec(filename) < 0) { perror(filename.c_str()); return BAD_SPEC_FILE; } if (generateAttributesTemplate) { apiEntries.genAttributesTemplate(inDir + "/" + baseName + ATTRIB_EXTENSION); exit(0); } std::string attribFileName = inDir + "/" + baseName + ATTRIB_EXTENSION; if (apiEntries.readAttributes(attribFileName) < 0) { perror(attribFileName.c_str()); fprintf(stderr, "failed to parse attributes\n"); exit(1); } if (encoderDir.size() != 0) { apiEntries.genOpcodes(encoderDir + "/" + baseName + "_opcodes.h"); apiEntries.genContext(encoderDir + "/" + baseName + "_client_context.h", ApiGen::CLIENT_SIDE); apiEntries.genContextImpl(encoderDir + "/" + baseName + "_client_context.cpp", ApiGen::CLIENT_SIDE); apiEntries.genProcTypes(encoderDir + "/" + baseName + "_client_proc.h", ApiGen::CLIENT_SIDE); apiEntries.genFuncTable(encoderDir + "/" + baseName + "_ftable.h", ApiGen::CLIENT_SIDE); apiEntries.genEntryPoints(encoderDir + "/" + baseName + "_entry.cpp", ApiGen::CLIENT_SIDE); apiEntries.genEncoderHeader(encoderDir + "/" + baseName + "_enc.h"); apiEntries.genEncoderImpl(encoderDir + "/" + baseName + "_enc.cpp"); } if (decoderDir.size() != 0) { apiEntries.genOpcodes(decoderDir + "/" + baseName + "_opcodes.h"); apiEntries.genProcTypes(decoderDir + "/" + baseName + "_server_proc.h", ApiGen::SERVER_SIDE); apiEntries.genContext(decoderDir + "/" + baseName + "_server_context.h", ApiGen::SERVER_SIDE); apiEntries.genContextImpl(decoderDir + "/" + baseName + "_server_context.cpp", ApiGen::SERVER_SIDE); apiEntries.genDecoderHeader(decoderDir + "/" + baseName + "_dec.h"); apiEntries.genDecoderImpl(decoderDir + "/" + baseName + "_dec.cpp"); } if (wrapperDir.size() != 0) { apiEntries.genProcTypes(wrapperDir + "/" + baseName + "_wrapper_proc.h", ApiGen::WRAPPER_SIDE); apiEntries.genContext(wrapperDir + "/" + baseName + "_wrapper_context.h", ApiGen::WRAPPER_SIDE); apiEntries.genContextImpl(wrapperDir + "/" + baseName + "_wrapper_context.cpp", ApiGen::WRAPPER_SIDE); apiEntries.genEntryPoints(wrapperDir + "/" + baseName + "_wrapper_entry.cpp", ApiGen::WRAPPER_SIDE); } #ifdef DEBUG_DUMP int withPointers = 0; printf("%d functions found\n", int(apiEntries.size())); for (int i = 0; i < apiEntries.size(); i++) { if (apiEntries[i].hasPointers()) { withPointers++; apiEntries[i].print(); } } fprintf(stdout, "%d entries has poitners\n", withPointers); #endif }