1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
8 #include "base/at_exit.h"
9 #include "base/basictypes.h"
10 #include "base/command_line.h"
11 #include "base/file_util.h"
12 #include "base/files/file_path.h"
13 #include "base/logging.h"
14 #include "base/strings/string_number_conversions.h"
15 #include "base/strings/string_util.h"
16 #include "base/strings/utf_string_conversions.h"
17 #include "courgette/courgette.h"
18 #include "courgette/streams.h"
19 #include "courgette/third_party/bsdiff.h"
25 " courgette -supported <executable_file>\n"
26 " courgette -dis <executable_file> <binary_assembly_file>\n"
27 " courgette -asm <binary_assembly_file> <executable_file>\n"
28 " courgette -disadj <executable_file> <reference> <binary_assembly_file>\n"
29 " courgette -gen <v1> <v2> <patch>\n"
30 " courgette -apply <v1> <patch> <v2>\n"
34 void UsageProblem(const char* message
) {
35 fprintf(stderr
, "%s", message
);
36 fprintf(stderr
, "\n");
41 void Problem(const char* format
, ...) {
43 va_start(args
, format
);
44 vfprintf(stderr
, format
, args
);
45 fprintf(stderr
, "\n");
50 std::string
ReadOrFail(const base::FilePath
& file_name
, const char* kind
) {
52 if (!file_util::GetFileSize(file_name
, &file_size
))
53 Problem("Can't read %s file.", kind
);
55 buffer
.reserve(static_cast<size_t>(file_size
));
56 if (!file_util::ReadFileToString(file_name
, &buffer
))
57 Problem("Can't read %s file.", kind
);
61 void WriteSinkToFile(const courgette::SinkStream
*sink
,
62 const base::FilePath
& output_file
) {
64 file_util::WriteFile(output_file
,
65 reinterpret_cast<const char*>(sink
->Buffer()),
66 static_cast<int>(sink
->Length()));
68 Problem("Can't write output.");
69 if (static_cast<size_t>(count
) != sink
->Length())
70 Problem("Incomplete write.");
73 void Disassemble(const base::FilePath
& input_file
,
74 const base::FilePath
& output_file
) {
75 std::string buffer
= ReadOrFail(input_file
, "input");
77 courgette::AssemblyProgram
* program
= NULL
;
78 const courgette::Status parse_status
=
79 courgette::ParseDetectedExecutable(buffer
.c_str(), buffer
.length(),
82 if (parse_status
!= courgette::C_OK
)
83 Problem("Can't parse input.");
85 courgette::EncodedProgram
* encoded
= NULL
;
86 const courgette::Status encode_status
= Encode(program
, &encoded
);
88 courgette::DeleteAssemblyProgram(program
);
90 if (encode_status
!= courgette::C_OK
)
91 Problem("Can't encode program.");
93 courgette::SinkStreamSet sinks
;
95 const courgette::Status write_status
=
96 courgette::WriteEncodedProgram(encoded
, &sinks
);
97 if (write_status
!= courgette::C_OK
)
98 Problem("Can't serialize encoded program.");
100 courgette::DeleteEncodedProgram(encoded
);
102 courgette::SinkStream sink
;
103 if (!sinks
.CopyTo(&sink
))
104 Problem("Can't combine serialized encoded program streams.");
106 WriteSinkToFile(&sink
, output_file
);
109 bool Supported(const base::FilePath
& input_file
) {
112 std::string buffer
= ReadOrFail(input_file
, "input");
114 courgette::ExecutableType type
;
115 size_t detected_length
;
117 DetectExecutableType(buffer
.c_str(), buffer
.length(),
121 // If the detection fails, we just fall back on UNKNOWN
122 std::string format
= "Unsupported";
126 case courgette::EXE_UNKNOWN
:
129 case courgette::EXE_WIN_32_X86
:
130 format
= "Windows 32 PE";
134 case courgette::EXE_ELF_32_X86
:
135 format
= "ELF 32 X86";
139 case courgette::EXE_ELF_32_ARM
:
140 format
= "ELF 32 ARM";
145 printf("%s Executable\n", format
.c_str());
149 void DisassembleAndAdjust(const base::FilePath
& program_file
,
150 const base::FilePath
& model_file
,
151 const base::FilePath
& output_file
) {
152 std::string program_buffer
= ReadOrFail(program_file
, "program");
153 std::string model_buffer
= ReadOrFail(model_file
, "reference");
155 courgette::AssemblyProgram
* program
= NULL
;
156 const courgette::Status parse_program_status
=
157 courgette::ParseDetectedExecutable(program_buffer
.c_str(),
158 program_buffer
.length(),
160 if (parse_program_status
!= courgette::C_OK
)
161 Problem("Can't parse program input.");
163 courgette::AssemblyProgram
* model
= NULL
;
164 const courgette::Status parse_model_status
=
165 courgette::ParseDetectedExecutable(model_buffer
.c_str(),
166 model_buffer
.length(),
168 if (parse_model_status
!= courgette::C_OK
)
169 Problem("Can't parse model input.");
171 const courgette::Status adjust_status
= Adjust(*model
, program
);
172 if (adjust_status
!= courgette::C_OK
)
173 Problem("Can't adjust program.");
175 courgette::EncodedProgram
* encoded
= NULL
;
176 const courgette::Status encode_status
= Encode(program
, &encoded
);
178 courgette::DeleteAssemblyProgram(program
);
180 if (encode_status
!= courgette::C_OK
)
181 Problem("Can't encode program.");
183 courgette::SinkStreamSet sinks
;
185 const courgette::Status write_status
=
186 courgette::WriteEncodedProgram(encoded
, &sinks
);
187 if (write_status
!= courgette::C_OK
)
188 Problem("Can't serialize encoded program.");
190 courgette::DeleteEncodedProgram(encoded
);
192 courgette::SinkStream sink
;
193 if (!sinks
.CopyTo(&sink
))
194 Problem("Can't combine serialized encoded program streams.");
196 WriteSinkToFile(&sink
, output_file
);
199 // Diffs two executable files, write a set of files for the diff, one file per
200 // stream of the EncodedProgram format. Each file is the bsdiff between the
201 // original file's stream and the new file's stream. This is completely
202 // uninteresting to users, but it is handy for seeing how much each which
203 // streams are contributing to the final file size. Adjustment is optional.
204 void DisassembleAdjustDiff(const base::FilePath
& model_file
,
205 const base::FilePath
& program_file
,
206 const base::FilePath
& output_file_root
,
208 std::string model_buffer
= ReadOrFail(model_file
, "'old'");
209 std::string program_buffer
= ReadOrFail(program_file
, "'new'");
211 courgette::AssemblyProgram
* model
= NULL
;
212 const courgette::Status parse_model_status
=
213 courgette::ParseDetectedExecutable(model_buffer
.c_str(),
214 model_buffer
.length(),
216 if (parse_model_status
!= courgette::C_OK
)
217 Problem("Can't parse model input.");
219 courgette::AssemblyProgram
* program
= NULL
;
220 const courgette::Status parse_program_status
=
221 courgette::ParseDetectedExecutable(program_buffer
.c_str(),
222 program_buffer
.length(),
224 if (parse_program_status
!= courgette::C_OK
)
225 Problem("Can't parse program input.");
228 const courgette::Status adjust_status
= Adjust(*model
, program
);
229 if (adjust_status
!= courgette::C_OK
)
230 Problem("Can't adjust program.");
233 courgette::EncodedProgram
* encoded_program
= NULL
;
234 const courgette::Status encode_program_status
=
235 Encode(program
, &encoded_program
);
236 courgette::DeleteAssemblyProgram(program
);
237 if (encode_program_status
!= courgette::C_OK
)
238 Problem("Can't encode program.");
240 courgette::EncodedProgram
* encoded_model
= NULL
;
241 const courgette::Status encode_model_status
= Encode(model
, &encoded_model
);
242 courgette::DeleteAssemblyProgram(model
);
243 if (encode_model_status
!= courgette::C_OK
)
244 Problem("Can't encode model.");
246 courgette::SinkStreamSet program_sinks
;
247 const courgette::Status write_program_status
=
248 courgette::WriteEncodedProgram(encoded_program
, &program_sinks
);
249 if (write_program_status
!= courgette::C_OK
)
250 Problem("Can't serialize encoded program.");
251 courgette::DeleteEncodedProgram(encoded_program
);
253 courgette::SinkStreamSet model_sinks
;
254 const courgette::Status write_model_status
=
255 courgette::WriteEncodedProgram(encoded_model
, &model_sinks
);
256 if (write_model_status
!= courgette::C_OK
)
257 Problem("Can't serialize encoded model.");
258 courgette::DeleteEncodedProgram(encoded_model
);
260 courgette::SinkStream empty_sink
;
261 for (int i
= 0; ; ++i
) {
262 courgette::SinkStream
* old_stream
= model_sinks
.stream(i
);
263 courgette::SinkStream
* new_stream
= program_sinks
.stream(i
);
264 if (old_stream
== NULL
&& new_stream
== NULL
)
267 courgette::SourceStream old_source
;
268 courgette::SourceStream new_source
;
269 old_source
.Init(old_stream
? *old_stream
: empty_sink
);
270 new_source
.Init(new_stream
? *new_stream
: empty_sink
);
271 courgette::SinkStream patch_stream
;
272 courgette::BSDiffStatus status
=
273 courgette::CreateBinaryPatch(&old_source
, &new_source
, &patch_stream
);
274 if (status
!= courgette::OK
) Problem("-xxx failed.");
276 std::string append
= std::string("-") + base::IntToString(i
);
278 WriteSinkToFile(&patch_stream
,
279 output_file_root
.InsertBeforeExtensionASCII(append
));
283 void Assemble(const base::FilePath
& input_file
,
284 const base::FilePath
& output_file
) {
285 std::string buffer
= ReadOrFail(input_file
, "input");
287 courgette::SourceStreamSet sources
;
288 if (!sources
.Init(buffer
.c_str(), buffer
.length()))
289 Problem("Bad input file.");
291 courgette::EncodedProgram
* encoded
= NULL
;
292 const courgette::Status read_status
= ReadEncodedProgram(&sources
, &encoded
);
293 if (read_status
!= courgette::C_OK
)
294 Problem("Bad encoded program.");
296 courgette::SinkStream sink
;
298 const courgette::Status assemble_status
= courgette::Assemble(encoded
, &sink
);
299 if (assemble_status
!= courgette::C_OK
)
300 Problem("Can't assemble.");
302 WriteSinkToFile(&sink
, output_file
);
305 void GenerateEnsemblePatch(const base::FilePath
& old_file
,
306 const base::FilePath
& new_file
,
307 const base::FilePath
& patch_file
) {
308 std::string old_buffer
= ReadOrFail(old_file
, "'old' input");
309 std::string new_buffer
= ReadOrFail(new_file
, "'new' input");
311 courgette::SourceStream old_stream
;
312 courgette::SourceStream new_stream
;
313 old_stream
.Init(old_buffer
);
314 new_stream
.Init(new_buffer
);
316 courgette::SinkStream patch_stream
;
317 courgette::Status status
=
318 courgette::GenerateEnsemblePatch(&old_stream
, &new_stream
, &patch_stream
);
320 if (status
!= courgette::C_OK
) Problem("-gen failed.");
322 WriteSinkToFile(&patch_stream
, patch_file
);
325 void ApplyEnsemblePatch(const base::FilePath
& old_file
,
326 const base::FilePath
& patch_file
,
327 const base::FilePath
& new_file
) {
328 // We do things a little differently here in order to call the same Courgette
329 // entry point as the installer. That entry point point takes file names and
330 // returns an status code but does not output any diagnostics.
332 courgette::Status status
=
333 courgette::ApplyEnsemblePatch(old_file
.value().c_str(),
334 patch_file
.value().c_str(),
335 new_file
.value().c_str());
337 if (status
== courgette::C_OK
)
340 // Diagnose the error.
342 case courgette::C_BAD_ENSEMBLE_MAGIC
:
343 Problem("Not a courgette patch");
346 case courgette::C_BAD_ENSEMBLE_VERSION
:
347 Problem("Wrong version patch");
350 case courgette::C_BAD_ENSEMBLE_HEADER
:
351 Problem("Corrupt patch");
354 case courgette::C_DISASSEMBLY_FAILED
:
355 Problem("Disassembly failed (could be because of memory issues)");
358 case courgette::C_STREAM_ERROR
:
359 Problem("Stream error (likely out of memory or disk space)");
366 // If we failed due to a missing input file, this will
367 // print the message.
368 std::string old_buffer
= ReadOrFail(old_file
, "'old' input");
370 std::string patch_buffer
= ReadOrFail(patch_file
, "'patch' input");
371 patch_buffer
.clear();
373 // Non-input related errors:
374 if (status
== courgette::C_WRITE_OPEN_ERROR
)
375 Problem("Can't open output");
376 if (status
== courgette::C_WRITE_ERROR
)
377 Problem("Can't write output");
379 Problem("-apply failed.");
382 void GenerateBSDiffPatch(const base::FilePath
& old_file
,
383 const base::FilePath
& new_file
,
384 const base::FilePath
& patch_file
) {
385 std::string old_buffer
= ReadOrFail(old_file
, "'old' input");
386 std::string new_buffer
= ReadOrFail(new_file
, "'new' input");
388 courgette::SourceStream old_stream
;
389 courgette::SourceStream new_stream
;
390 old_stream
.Init(old_buffer
);
391 new_stream
.Init(new_buffer
);
393 courgette::SinkStream patch_stream
;
394 courgette::BSDiffStatus status
=
395 courgette::CreateBinaryPatch(&old_stream
, &new_stream
, &patch_stream
);
397 if (status
!= courgette::OK
) Problem("-genbsdiff failed.");
399 WriteSinkToFile(&patch_stream
, patch_file
);
402 void ApplyBSDiffPatch(const base::FilePath
& old_file
,
403 const base::FilePath
& patch_file
,
404 const base::FilePath
& new_file
) {
405 std::string old_buffer
= ReadOrFail(old_file
, "'old' input");
406 std::string patch_buffer
= ReadOrFail(patch_file
, "'patch' input");
408 courgette::SourceStream old_stream
;
409 courgette::SourceStream patch_stream
;
410 old_stream
.Init(old_buffer
);
411 patch_stream
.Init(patch_buffer
);
413 courgette::SinkStream new_stream
;
414 courgette::BSDiffStatus status
=
415 courgette::ApplyBinaryPatch(&old_stream
, &patch_stream
, &new_stream
);
417 if (status
!= courgette::OK
) Problem("-applybsdiff failed.");
419 WriteSinkToFile(&new_stream
, new_file
);
422 int main(int argc
, const char* argv
[]) {
423 base::AtExitManager at_exit_manager
;
424 CommandLine::Init(argc
, argv
);
425 const CommandLine
& command_line
= *CommandLine::ForCurrentProcess();
427 logging::LoggingSettings settings
;
428 settings
.logging_dest
= logging::LOG_TO_ALL
;
429 settings
.log_file
= FILE_PATH_LITERAL("courgette.log");
430 (void)logging::InitLogging(settings
);
431 logging::SetMinLogLevel(logging::LOG_VERBOSE
);
433 bool cmd_sup
= command_line
.HasSwitch("supported");
434 bool cmd_dis
= command_line
.HasSwitch("dis");
435 bool cmd_asm
= command_line
.HasSwitch("asm");
436 bool cmd_disadj
= command_line
.HasSwitch("disadj");
437 bool cmd_make_patch
= command_line
.HasSwitch("gen");
438 bool cmd_apply_patch
= command_line
.HasSwitch("apply");
439 bool cmd_make_bsdiff_patch
= command_line
.HasSwitch("genbsdiff");
440 bool cmd_apply_bsdiff_patch
= command_line
.HasSwitch("applybsdiff");
441 bool cmd_spread_1_adjusted
= command_line
.HasSwitch("gen1a");
442 bool cmd_spread_1_unadjusted
= command_line
.HasSwitch("gen1u");
444 std::vector
<base::FilePath
> values
;
445 const CommandLine::StringVector
& args
= command_line
.GetArgs();
446 for (size_t i
= 0; i
< args
.size(); ++i
) {
447 values
.push_back(base::FilePath(args
[i
]));
450 // '-repeat=N' is for debugging. Running many iterations can reveal leaks and
452 int repeat_count
= 1;
453 std::string repeat_switch
= command_line
.GetSwitchValueASCII("repeat");
454 if (!repeat_switch
.empty())
455 if (!base::StringToInt(repeat_switch
, &repeat_count
))
458 if (cmd_sup
+ cmd_dis
+ cmd_asm
+ cmd_disadj
+ cmd_make_patch
+
459 cmd_apply_patch
+ cmd_make_bsdiff_patch
+ cmd_apply_bsdiff_patch
+
460 cmd_spread_1_adjusted
+ cmd_spread_1_unadjusted
463 "Must have exactly one of:\n"
464 " -supported -asm, -dis, -disadj, -gen or -apply, -genbsdiff"
465 " or -applybsdiff.");
467 while (repeat_count
-- > 0) {
469 if (values
.size() != 1)
470 UsageProblem("-supported <executable_file>");
471 return !Supported(values
[0]);
472 } else if (cmd_dis
) {
473 if (values
.size() != 2)
474 UsageProblem("-dis <executable_file> <courgette_file>");
475 Disassemble(values
[0], values
[1]);
476 } else if (cmd_asm
) {
477 if (values
.size() != 2)
478 UsageProblem("-asm <courgette_file_input> <executable_file_output>");
479 Assemble(values
[0], values
[1]);
480 } else if (cmd_disadj
) {
481 if (values
.size() != 3)
482 UsageProblem("-disadj <executable_file> <model> <courgette_file>");
483 DisassembleAndAdjust(values
[0], values
[1], values
[2]);
484 } else if (cmd_make_patch
) {
485 if (values
.size() != 3)
486 UsageProblem("-gen <old_file> <new_file> <patch_file>");
487 GenerateEnsemblePatch(values
[0], values
[1], values
[2]);
488 } else if (cmd_apply_patch
) {
489 if (values
.size() != 3)
490 UsageProblem("-apply <old_file> <patch_file> <new_file>");
491 ApplyEnsemblePatch(values
[0], values
[1], values
[2]);
492 } else if (cmd_make_bsdiff_patch
) {
493 if (values
.size() != 3)
494 UsageProblem("-genbsdiff <old_file> <new_file> <patch_file>");
495 GenerateBSDiffPatch(values
[0], values
[1], values
[2]);
496 } else if (cmd_apply_bsdiff_patch
) {
497 if (values
.size() != 3)
498 UsageProblem("-applybsdiff <old_file> <patch_file> <new_file>");
499 ApplyBSDiffPatch(values
[0], values
[1], values
[2]);
500 } else if (cmd_spread_1_adjusted
|| cmd_spread_1_unadjusted
) {
501 if (values
.size() != 3)
502 UsageProblem("-gen1[au] <old_file> <new_file> <patch_files_root>");
503 DisassembleAdjustDiff(values
[0], values
[1], values
[2],
504 cmd_spread_1_adjusted
);
506 UsageProblem("No operation specified");