diff options
author | Samuel Huang <huangs@chromium.org> | 2018-04-20 17:40:00 +0000 |
---|---|---|
committer | Edward Lesmes <ehmaldonado@google.com> | 2021-07-23 22:08:14 +0000 |
commit | 451ff5de400706acdfcfdb9bf28ca6d4c0670b81 (patch) | |
tree | 59b8f9de7d363363412a8215166a66f4035c8daa /zucchini_gen.cc | |
parent | 07c31a327fc75fd62023d1774e900f85ce982ee7 (diff) | |
download | zucchini-451ff5de400706acdfcfdb9bf28ca6d4c0670b81.tar.gz |
[Zucchini] Introduce ReferenceBytesMixer.
Some architectures (e.g., ARM) have references that mix operation bits
with payload bits. ReferenceBytesMixer is a class to isloate operation
bit changes to Layer 1 patching.
This CL introduces only the stub for ReferenceBytesMixer (the remainder
would require ARM support). Having this now brings Zucchini-gen closer
to trunk code, and sets up upcoming work for a mock disassembler.
Change-Id: I2e1aa56bf265e4b61b33d323be2ed9a456e14f3b
Reviewed-on: https://chromium-review.googlesource.com/1021650
Commit-Queue: Samuel Huang <huangs@chromium.org>
Reviewed-by: Samuel Huang <huangs@chromium.org>
Reviewed-by: Greg Thompson <grt@chromium.org>
Cr-Commit-Position: refs/heads/master@{#552373}
NOKEYCHECK=True
GitOrigin-RevId: fdb4806d8d03c52ffe4ca7b14076f483384eedfb
Diffstat (limited to 'zucchini_gen.cc')
-rw-r--r-- | zucchini_gen.cc | 50 |
1 files changed, 39 insertions, 11 deletions
diff --git a/zucchini_gen.cc b/zucchini_gen.cc index af7c7dd..e88d37b 100644 --- a/zucchini_gen.cc +++ b/zucchini_gen.cc @@ -22,6 +22,7 @@ #include "components/zucchini/heuristic_ensemble_matcher.h" #include "components/zucchini/image_index.h" #include "components/zucchini/patch_writer.h" +#include "components/zucchini/reference_bytes_mixer.h" #include "components/zucchini/suffix_array.h" #include "components/zucchini/targets_affinity.h" @@ -120,6 +121,7 @@ bool GenerateRawDelta(ConstBufferView old_image, ConstBufferView new_image, const EquivalenceMap& equivalence_map, const ImageIndex& new_image_index, + ReferenceBytesMixer* reference_bytes_mixer, PatchElementWriter* patch_writer) { RawDeltaSink raw_delta_sink; @@ -130,14 +132,37 @@ bool GenerateRawDelta(ConstBufferView old_image, Equivalence equivalence = candidate.eq; // For each bytewise delta from |old_image| to |new_image|, compute "copy // offset" and pass it along with delta to the sink. - for (offset_t i = 0; i < equivalence.length; ++i) { - if (new_image_index.IsReference(equivalence.dst_offset + i)) - continue; // Skip references since they're handled elsewhere. - - int8_t diff = new_image[equivalence.dst_offset + i] - - old_image[equivalence.src_offset + i]; - if (diff) - raw_delta_sink.PutNext({base_copy_offset + i, diff}); + for (offset_t i = 0; i < equivalence.length;) { + if (new_image_index.IsReference(equivalence.dst_offset + i)) { + DCHECK(new_image_index.IsToken(equivalence.dst_offset + i)); + TypeTag type_tag = + new_image_index.LookupType(equivalence.dst_offset + i); + + // Reference delta has its own flow. On some architectures (e.g., x86) + // this does not involve raw delta, so we skip. On other architectures + // (e.g., ARM) references are mixed with other bits that may change, so + // we need to "mix" data and store some changed bits into raw delta. + int num_bytes = reference_bytes_mixer->NumBytes(type_tag.value()); + if (num_bytes) { + ConstBufferView mixed_ref_bytes = reference_bytes_mixer->Mix( + type_tag.value(), old_image.begin(), equivalence.src_offset + i, + new_image.begin(), equivalence.dst_offset + i); + for (int j = 0; j < num_bytes; ++j) { + int8_t diff = + mixed_ref_bytes[j] - old_image[equivalence.src_offset + i + j]; + if (diff) + raw_delta_sink.PutNext({base_copy_offset + i + j, diff}); + } + } + i += new_image_index.refs(type_tag).width(); + DCHECK_LE(i, equivalence.length); + } else { + int8_t diff = new_image[equivalence.dst_offset + i] - + old_image[equivalence.src_offset + i]; + if (diff) + raw_delta_sink.PutNext({base_copy_offset + i, diff}); + ++i; + } } base_copy_offset += equivalence.length; } @@ -225,10 +250,12 @@ bool GenerateRawElement(const std::vector<offset_t>& old_sa, kMinEquivalenceSimilarity); patch_writer->SetReferenceDeltaSink({}); + + ReferenceBytesMixer no_op_bytes_mixer; return GenerateEquivalencesAndExtraData(new_image, equivalences, patch_writer) && GenerateRawDelta(old_image, new_image, equivalences, new_image_index, - patch_writer); + &no_op_bytes_mixer, patch_writer); } bool GenerateExecutableElement(ExecutableType exe_type, @@ -282,11 +309,12 @@ bool GenerateExecutableElement(ExecutableType exe_type, } } patch_writer->SetReferenceDeltaSink(std::move(reference_delta_sink)); - + std::unique_ptr<ReferenceBytesMixer> reference_bytes_mixer = + ReferenceBytesMixer::Create(*old_disasm, *new_disasm); return GenerateEquivalencesAndExtraData(new_image, equivalences, patch_writer) && GenerateRawDelta(old_image, new_image, equivalences, new_image_index, - patch_writer); + reference_bytes_mixer.get(), patch_writer); } /******** Exported Functions ********/ |