|  | // Copyright 2022 The Bazel Authors. All rights reserved. | 
|  | // | 
|  | // 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 <cstdint> | 
|  | #include <cstdio> | 
|  | #include <cstdlib> | 
|  | #include <cstring> | 
|  | #include <fstream> | 
|  | #include <string> | 
|  |  | 
|  | #ifdef _WIN32 | 
|  | #include "src/main/cpp/util/path_platform.h" | 
|  | #endif  // _WIN32 | 
|  |  | 
|  | //  This is a replacement for | 
|  | //  third_party/bazel/src/main/java/com/google/devtools/build/lib/analysis/actions/LauncherFileWriteAction.java | 
|  | // | 
|  | //  It takes exactly 3 arguments: | 
|  | //    1) The path to the actual launcher executable | 
|  | //    2) The multi-line .params file containing the launcher info data | 
|  | //    3) The path of the output executable | 
|  | // | 
|  | //  The program copies the launcher executable as is to the output, and then | 
|  | //  appends each line of the launch info as a null-terminated string. At the | 
|  | //  end, the size of the launch data written is appended as a long value (8 | 
|  | //  bytes). | 
|  |  | 
|  | #ifdef _WIN32 | 
|  |  | 
|  | #define STRING_TYPE std::wstring | 
|  | #define STRING_FORMAT "%ls" | 
|  | std::wstring convert_path(char* path) { | 
|  | std::string error; | 
|  | std::wstring wpath; | 
|  | if (!blaze_util::AsAbsoluteWindowsPath(path, &wpath, &error)) { | 
|  | fprintf(stderr, "Failed to make absolute path for %s: %s\n", path, | 
|  | error.c_str()); | 
|  | exit(1); | 
|  | } | 
|  | return wpath; | 
|  | } | 
|  |  | 
|  | #else  // _WIN32 | 
|  |  | 
|  | #define STRING_TYPE std::string | 
|  | #define STRING_FORMAT "%s" | 
|  | std::string convert_path(char* path) { return path; } | 
|  |  | 
|  | #endif  // _WIN32 | 
|  |  | 
|  | int main(int argc, char** argv) { | 
|  | if (argc < 4) { | 
|  | fprintf(stderr, "Expected 3 arguments, got %d\n", argc); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | STRING_TYPE launcher_path = convert_path(argv[1]); | 
|  | STRING_TYPE info_params = convert_path(argv[2]); | 
|  | STRING_TYPE output_path = convert_path(argv[3]); | 
|  |  | 
|  | std::ifstream src(launcher_path.c_str(), std::ios::binary); | 
|  | if (!src.good()) { | 
|  | fprintf(stderr, "Failed to open " STRING_FORMAT ": %s\n", | 
|  | launcher_path.c_str(), strerror(errno)); | 
|  | return 1; | 
|  | } | 
|  | std::ofstream dst(output_path.c_str(), std::ios::binary); | 
|  | if (!dst.good()) { | 
|  | fprintf(stderr, "Failed to create " STRING_FORMAT ": %s\n", | 
|  | output_path.c_str(), strerror(errno)); | 
|  | return 1; | 
|  | } | 
|  | dst << src.rdbuf(); | 
|  |  | 
|  | std::ifstream info_file(info_params.c_str()); | 
|  | if (!info_file.good()) { | 
|  | fprintf(stderr, "Failed to open " STRING_FORMAT ": %s\n", | 
|  | info_params.c_str(), strerror(errno)); | 
|  | return 1; | 
|  | } | 
|  | int64_t bytes = 0; | 
|  | std::string line; | 
|  | while (std::getline(info_file, line)) { | 
|  | dst << line; | 
|  | bytes += line.length(); | 
|  | dst << '\0'; | 
|  | bytes++; | 
|  | } | 
|  |  | 
|  | dst.write(reinterpret_cast<const char*>(&bytes), sizeof(bytes)); | 
|  | return 0; | 
|  | } |