blob: ce057e57bfe79f895354fd40a070771b7daf00cb [file] [log] [blame]
Damien Martin-Guillerezbf6281d2015-11-19 16:41:33 +00001// Copyright 2015 The Bazel Authors. All rights reserved.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01002//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14//
15// ijar.cpp -- .jar -> _interface.jar tool.
16//
17
18#include <stdio.h>
19#include <string.h>
20#include <stdlib.h>
21#include <limits.h>
Damien Martin-Guillerez08441122015-05-28 11:12:31 +000022#include <errno.h>
23#include <memory>
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010024
Damien Martin-Guillerez08441122015-05-28 11:12:31 +000025#include "third_party/ijar/zip.h"
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010026
Damien Martin-Guillerez08441122015-05-28 11:12:31 +000027namespace devtools_ijar {
28
29bool verbose = false;
30
31// Reads a JVM class from classdata_in (of the specified length), and
32// writes out a simplified class to classdata_out, advancing the
Liam Miller-Cushon31c88782016-04-11 22:59:37 +000033// pointer. Returns true if the class should be kept.
34bool StripClass(u1*& classdata_out, const u1* classdata_in, size_t in_length);
Damien Martin-Guillerez08441122015-05-28 11:12:31 +000035
36const char* CLASS_EXTENSION = ".class";
37const size_t CLASS_EXTENSION_LENGTH = strlen(CLASS_EXTENSION);
38
39// ZipExtractorProcessor that select only .class file and use
40// StripClass to generate an interface class, storing as a new file
41// in the specified ZipBuilder.
42class JarStripperProcessor : public ZipExtractorProcessor {
43 public:
44 JarStripperProcessor() {}
45 virtual ~JarStripperProcessor() {}
46
47 virtual void Process(const char* filename, const u4 attr,
48 const u1* data, const size_t size);
49 virtual bool Accept(const char* filename, const u4 attr);
50
51 private:
52 // Not owned by JarStripperProcessor, see SetZipBuilder().
53 ZipBuilder* builder;
54
55 public:
56 // Set the ZipBuilder to add the ijar class to the output zip file.
57 // This pointer should not be deleted while this class is still in use and
58 // it should be set before any call to the Process() method.
59 void SetZipBuilder(ZipBuilder* builder) {
60 this->builder = builder;
61 }
62};
63
64bool JarStripperProcessor::Accept(const char* filename, const u4 attr) {
John Cater0dff43a2017-03-01 16:36:32 +000065 const size_t filename_len = strlen(filename);
Laszlo Csomor479e18d2016-12-02 15:11:08 +000066 if (filename_len >= CLASS_EXTENSION_LENGTH) {
67 return strcmp(filename + filename_len - CLASS_EXTENSION_LENGTH,
68 CLASS_EXTENSION) == 0;
Damien Martin-Guillerez08441122015-05-28 11:12:31 +000069 }
70 return false;
71}
72
73void JarStripperProcessor::Process(const char* filename, const u4 attr,
74 const u1* data, const size_t size) {
75 if (verbose) {
76 fprintf(stderr, "INFO: StripClass: %s\n", filename);
77 }
Liam Miller-Cushon31c88782016-04-11 22:59:37 +000078 u1* buf = reinterpret_cast<u1*>(malloc(size));
79 u1* classdata_out = buf;
80 if (!StripClass(buf, data, size)) {
81 free(classdata_out);
82 return;
83 }
84 u1* q = builder->NewFile(filename, 0);
85 size_t out_length = buf - classdata_out;
86 memcpy(q, classdata_out, out_length);
lberki7ced4c02017-07-13 12:18:56 +020087 builder->FinishFile(out_length, false, true);
Liam Miller-Cushon31c88782016-04-11 22:59:37 +000088 free(classdata_out);
Damien Martin-Guillerez08441122015-05-28 11:12:31 +000089}
90
91// Opens "file_in" (a .jar file) for reading, and writes an interface
92// .jar to "file_out".
93void OpenFilesAndProcessJar(const char *file_out, const char *file_in) {
94 JarStripperProcessor processor;
95 std::unique_ptr<ZipExtractor> in(ZipExtractor::Create(file_in, &processor));
96 if (in.get() == NULL) {
97 fprintf(stderr, "Unable to open Zip file %s: %s\n", file_in,
98 strerror(errno));
99 abort();
100 }
101 u8 output_length = in->CalculateOutputLength();
102 std::unique_ptr<ZipBuilder> out(ZipBuilder::Create(file_out, output_length));
103 if (out.get() == NULL) {
104 fprintf(stderr, "Unable to open output file %s: %s\n", file_out,
105 strerror(errno));
106 abort();
107 }
108 processor.SetZipBuilder(out.get());
109
110 // Process all files in the zip
111 if (in->ProcessAll() < 0) {
112 fprintf(stderr, "%s\n", in->GetError());
113 abort();
114 }
115
116 // Add dummy file, since javac doesn't like truly empty jars.
117 if (out->GetNumberFiles() == 0) {
118 out->WriteEmptyFile("dummy");
119 }
120 // Finish writing the output file
121 if (out->Finish() < 0) {
122 fprintf(stderr, "%s\n", out->GetError());
123 abort();
124 }
125 // Get all file size
126 size_t in_length = in->GetSize();
127 size_t out_length = out->GetSize();
128 if (verbose) {
129 fprintf(stderr, "INFO: produced interface jar: %s -> %s (%d%%).\n",
130 file_in, file_out,
131 static_cast<int>(100.0 * out_length / in_length));
132 }
133}
134
135} // namespace devtools_ijar
136
137//
138// main method
139//
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100140static void usage() {
141 fprintf(stderr, "Usage: ijar [-v] x.jar [x_interface.jar>]\n");
142 fprintf(stderr, "Creates an interface jar from the specified jar file.\n");
143 exit(1);
144}
145
146int main(int argc, char **argv) {
147 const char *filename_in = NULL;
148 const char *filename_out = NULL;
149
150 for (int ii = 1; ii < argc; ++ii) {
151 if (strcmp(argv[ii], "-v") == 0) {
152 devtools_ijar::verbose = true;
153 } else if (filename_in == NULL) {
154 filename_in = argv[ii];
155 } else if (filename_out == NULL) {
156 filename_out = argv[ii];
157 } else {
158 usage();
159 }
160 }
161
162 if (filename_in == NULL) {
163 usage();
164 }
165
166 // Guess output filename from input:
167 char filename_out_buf[PATH_MAX];
168 if (filename_out == NULL) {
169 size_t len = strlen(filename_in);
170 if (len > 4 && strncmp(filename_in + len - 4, ".jar", 4) == 0) {
171 strcpy(filename_out_buf, filename_in);
172 strcpy(filename_out_buf + len - 4, "-interface.jar");
173 filename_out = filename_out_buf;
174 } else {
175 fprintf(stderr, "Can't determine output filename since input filename "
176 "doesn't end with '.jar'.\n");
177 return 1;
178 }
179 }
180
181 if (devtools_ijar::verbose) {
182 fprintf(stderr, "INFO: writing to '%s'.\n", filename_out);
183 }
184
Damien Martin-Guillerez08441122015-05-28 11:12:31 +0000185 devtools_ijar::OpenFilesAndProcessJar(filename_out, filename_in);
186 return 0;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100187}