Merge "Remove unused/unusable relocation packer files."
This commit is contained in:
commit
7f8d91ea8e
|
@ -1,135 +0,0 @@
|
|||
Introduction:
|
||||
-------------
|
||||
|
||||
Relative relocations are the bulk of dynamic relocations (the .rel.dyn
|
||||
or .rela.dyn sections) in libchrome.<version>.so. The ELF standard
|
||||
representation of them is wasteful.
|
||||
|
||||
Packing uses a combination of run length encoding, delta encoding, and LEB128
|
||||
encoding to store them more efficiently. Packed relocations are placed in
|
||||
a new .android.rel.dyn or .android.rela.dyn section. Packing reduces
|
||||
the footprint of libchrome.<version>.so in the filesystem, in APK downloads,
|
||||
and in memory when loaded on the device.
|
||||
|
||||
A packed libchrome.<version>.so is designed so that it can be loaded directly
|
||||
on Android, but requires the explicit support of a crazy linker that has been
|
||||
extended to understand packed relocations. Packed relocations are currently
|
||||
only supported on ARM.
|
||||
|
||||
A packed libchrome.<version>.so cannot currently be used with the standard
|
||||
Android runtime linker.
|
||||
|
||||
See src/*.h for design and implementation notes.
|
||||
|
||||
|
||||
Notes:
|
||||
------
|
||||
|
||||
Packing does not adjust debug data. An unstripped libchrome.<version>.so
|
||||
can be packed and will run, but may no longer be useful for debugging.
|
||||
|
||||
Unpacking on the device requires the explicit support of an extended crazy
|
||||
linker. Adds the following new .dynamic tags, used by the crazy linker to
|
||||
find the packed .android.rel.dyn or .android.rela.dyn section data:
|
||||
|
||||
DT_ANDROID_REL_OFFSET = DT_LOOS (Operating System specific: 0x6000000d)
|
||||
- The offset of packed relocation data in libchrome.<version>.so
|
||||
DT_ANDROID_REL_SIZE = DT_LOOS + 1 (Operating System Specific: 0x6000000e)
|
||||
- The size of packed relocation data in bytes
|
||||
|
||||
32 bit ARM libraries use relocations without addends. 64 bit ARM libraries
|
||||
use relocations with addends. The packing strategy necessarily differs for
|
||||
the two relocation types.
|
||||
|
||||
Where libchrome.<version>.so contains relocations without addends, the format
|
||||
of .android.rel.dyn data is:
|
||||
|
||||
"APR1" identifier
|
||||
N: the number of count-delta pairs in the encoding
|
||||
A: the initial offset
|
||||
N * C,D: N count-delta pairs
|
||||
|
||||
Where libchrome.<version>.so contains relocations with addends, the format
|
||||
of .android.rela.dyn data is:
|
||||
|
||||
"APA1" identifier
|
||||
N: the number of addr-addend delta pairs in the encoding
|
||||
N * A,V: N addr-addend delta pairs
|
||||
|
||||
All numbers in the encoding stream are stored as LEB128 values. For details
|
||||
see http://en.wikipedia.org/wiki/LEB128.
|
||||
|
||||
The streaming unpacking algorithm for 32 bit ARM is:
|
||||
|
||||
skip over "APR1"
|
||||
pairs, addr = next leb128 value, next leb128 value
|
||||
emit R_ARM_RELATIVE relocation with r_offset = addr
|
||||
while pairs:
|
||||
count, delta = next leb128 value, next leb128 value
|
||||
while count:
|
||||
addr += delta
|
||||
emit R_ARM_RELATIVE relocation with r_offset = addr
|
||||
count--
|
||||
pairs--
|
||||
|
||||
The streaming unpacking algorithm for 64 bit ARM is:
|
||||
|
||||
skip over "APA1"
|
||||
pairs = next signed leb128 value
|
||||
addr, addend = 0, 0
|
||||
while pairs:
|
||||
addr += next signed leb128 value
|
||||
addend += next signed leb128 value
|
||||
emit R_AARCH64_RELATIVE relocation with r_offset = addr, r_addend = addend
|
||||
pairs--
|
||||
|
||||
|
||||
Usage instructions:
|
||||
-------------------
|
||||
|
||||
To pack relocations, add an empty .android.rel.dyn or .android.rela.dyn and
|
||||
then run the tool:
|
||||
|
||||
echo -n 'NULL' >/tmp/small
|
||||
if file libchrome.<version>.so | grep -q 'ELF 32'; then
|
||||
arm-linux-androideabi-objcopy
|
||||
--add-section .android.rel.dyn=/tmp/small
|
||||
libchrome.<version>.so libchrome.<version>.so.packed
|
||||
else
|
||||
aarch64-linux-android-objcopy
|
||||
--add-section .android.rela.dyn=/tmp/small
|
||||
libchrome.<version>.so libchrome.<version>.so.packed
|
||||
fi
|
||||
rm /tmp/small
|
||||
relocation_packer libchrome.<version>.so.packed
|
||||
|
||||
To unpack and restore the shared library to its original state:
|
||||
|
||||
cp libchrome.<version>.so.packed unpackable
|
||||
relocation_packer -u unpackable
|
||||
if file libchrome.<version>.so | grep -q 'ELF 32'; then
|
||||
arm-linux-androideabi-objcopy \
|
||||
--remove-section=.android.rel.dyn unpackable libchrome.<version>.so
|
||||
else
|
||||
aarch64-linux-android-objcopy \
|
||||
--remove-section=.android.rela.dyn unpackable libchrome.<version>.so
|
||||
endif
|
||||
rm unpackable
|
||||
|
||||
|
||||
Bugs & TODOs:
|
||||
-------------
|
||||
|
||||
Requires two free slots in the .dynamic section. Uses these to add data that
|
||||
tells the crazy linker where to find the packed relocation data. Fails
|
||||
if insufficient free slots exist (use gold --spare-dynamic-slots to increase
|
||||
the allocation).
|
||||
|
||||
Requires libelf 0.158 or later. Earlier libelf releases may be buggy in
|
||||
ways that prevent the packer from working correctly.
|
||||
|
||||
|
||||
Testing:
|
||||
--------
|
||||
|
||||
Unittests run under gtest, on the host system.
|
|
@ -1,81 +0,0 @@
|
|||
// Copyright 2014 The Chromium Authors. All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style license that can be
|
||||
// found in the LICENSE file.
|
||||
|
||||
// Run-length encode and decode relative relocations.
|
||||
//
|
||||
// Relative relocations are the bulk of dynamic relocations (the
|
||||
// .rel.dyn or .rela.dyn sections) in libchrome.<version>.so, and the ELF
|
||||
// standard representation of them is wasteful. .rel.dyn contains
|
||||
// relocations without addends, .rela.dyn relocations with addends.
|
||||
//
|
||||
// A relocation with no addend is 8 bytes on 32 bit platforms and 16 bytes
|
||||
// on 64 bit plaforms, split into offset and info fields. Offsets strictly
|
||||
// increase, and each is commonly a few bytes different from its predecessor.
|
||||
// There are long runs where the difference does not change. The info field
|
||||
// is constant. Example, from 'readelf -x4 libchrome.<version>.so' 32 bit:
|
||||
//
|
||||
// offset info offset info
|
||||
// 808fef01 17000000 848fef01 17000000 ................
|
||||
// 888fef01 17000000 8c8fef01 17000000 ................
|
||||
// 908fef01 17000000 948fef01 17000000 ................
|
||||
//
|
||||
// Run length encoding packs this data more efficiently, by representing it
|
||||
// as a delta and a count of entries each differing from its predecessor
|
||||
// by this delta. The above can be represented as a start address followed
|
||||
// by an encoded count of 6 and offset difference of 4:
|
||||
//
|
||||
// start count diff
|
||||
// 01ef8f80 00000006 00000004
|
||||
//
|
||||
// Because relative relocation offsets strictly increase, the complete
|
||||
// set of relative relocations in libchrome.<version>.so can be
|
||||
// represented by a single start address followed by one or more difference
|
||||
// and count encoded word pairs:
|
||||
//
|
||||
// start run1 count run1 diff run2 count run2 diff
|
||||
// 01ef8f80 00000006 00000004 00000010 00000008 ...
|
||||
//
|
||||
// Decoding regenerates relative relocations beginning at address
|
||||
// 'start' and for each encoded run, incrementing the address by 'difference'
|
||||
// for 'count' iterations and emitting a new relative relocation.
|
||||
//
|
||||
// Once encoded, data is prefixed by a single word count of packed delta and
|
||||
// count pairs. A final run-length encoded relative relocations vector
|
||||
// might therefore look something like:
|
||||
//
|
||||
// pairs start run 1 run 2 ... run 15
|
||||
// 0000000f 01ef8f80 00000006 00000004 00000010 00000008 ...
|
||||
// Interpreted as:
|
||||
// pairs=15 start=.. count=6,delta=4 count=16,delta=8
|
||||
|
||||
#ifndef TOOLS_RELOCATION_PACKER_SRC_RUN_LENGTH_ENCODER_H_
|
||||
#define TOOLS_RELOCATION_PACKER_SRC_RUN_LENGTH_ENCODER_H_
|
||||
|
||||
#include <vector>
|
||||
|
||||
#include "elf.h"
|
||||
#include "elf_traits.h"
|
||||
|
||||
namespace relocation_packer {
|
||||
|
||||
// A RelocationRunLengthCodec packs vectors of relative relocations
|
||||
// into more compact forms, and unpacks them to reproduce the pre-packed data.
|
||||
class RelocationRunLengthCodec {
|
||||
public:
|
||||
// Encode relative relocations into a more compact form.
|
||||
// |relocations| is a vector of relative relocation structs.
|
||||
// |packed| is the vector of packed words into which relocations are packed.
|
||||
static void Encode(const std::vector<ELF::Rel>& relocations,
|
||||
std::vector<ELF::Xword>* packed);
|
||||
|
||||
// Decode relative relocations from their more compact form.
|
||||
// |packed| is the vector of packed relocations.
|
||||
// |relocations| is a vector of unpacked relative relocation structs.
|
||||
static void Decode(const std::vector<ELF::Xword>& packed,
|
||||
std::vector<ELF::Rel>* relocations);
|
||||
};
|
||||
|
||||
} // namespace relocation_packer
|
||||
|
||||
#endif // TOOLS_RELOCATION_PACKER_SRC_RUN_LENGTH_ENCODER_H_
|
|
@ -1,88 +0,0 @@
|
|||
#!/usr/bin/env python
|
||||
#
|
||||
# Copyright 2014 The Chromium Authors. All rights reserved.
|
||||
# Use of this source code is governed by a BSD-style license that can be
|
||||
# found in the LICENSE file.
|
||||
|
||||
"""Build relocation packer unit test data.
|
||||
|
||||
Uses a built relocation packer to generate 'golden' reference test data
|
||||
files for elf_file_unittests.cc.
|
||||
"""
|
||||
|
||||
import optparse
|
||||
import os
|
||||
import shutil
|
||||
import subprocess
|
||||
import sys
|
||||
import tempfile
|
||||
|
||||
def PackArmLibraryRelocations(android_pack_relocations,
|
||||
android_objcopy,
|
||||
added_section,
|
||||
input_path,
|
||||
output_path):
|
||||
# Copy and add a 'NULL' .android.rel.dyn section for the packing tool.
|
||||
with tempfile.NamedTemporaryFile() as stream:
|
||||
stream.write('NULL')
|
||||
stream.flush()
|
||||
objcopy_command = [android_objcopy,
|
||||
'--add-section', '%s=%s' % (added_section, stream.name),
|
||||
input_path, output_path]
|
||||
subprocess.check_call(objcopy_command)
|
||||
|
||||
# Pack relocations.
|
||||
pack_command = [android_pack_relocations, output_path]
|
||||
subprocess.check_call(pack_command)
|
||||
|
||||
|
||||
def UnpackArmLibraryRelocations(android_pack_relocations,
|
||||
input_path,
|
||||
output_path):
|
||||
shutil.copy(input_path, output_path)
|
||||
|
||||
# Unpack relocations. We leave the .android.rel.dyn or .android.rela.dyn
|
||||
# in place.
|
||||
unpack_command = [android_pack_relocations, '-u', output_path]
|
||||
subprocess.check_call(unpack_command)
|
||||
|
||||
|
||||
def main():
|
||||
parser = optparse.OptionParser()
|
||||
|
||||
parser.add_option('--android-pack-relocations',
|
||||
help='Path to the ARM relocations packer binary')
|
||||
parser.add_option('--android-objcopy',
|
||||
help='Path to the toolchain\'s objcopy binary')
|
||||
parser.add_option('--added-section',
|
||||
choices=['.android.rel.dyn', '.android.rela.dyn'],
|
||||
help='Section to add, one of ".android.rel.dyn" or ".android.rela.dyn"')
|
||||
parser.add_option('--test-file',
|
||||
help='Path to the input test file, an unpacked ARM .so')
|
||||
parser.add_option('--unpacked-output',
|
||||
help='Path to the output file for reference unpacked data')
|
||||
parser.add_option('--packed-output',
|
||||
help='Path to the output file for reference packed data')
|
||||
|
||||
options, _ = parser.parse_args()
|
||||
|
||||
for output in [options.unpacked_output, options.packed_output]:
|
||||
directory = os.path.dirname(output)
|
||||
if not os.path.exists(directory):
|
||||
os.makedirs(directory)
|
||||
|
||||
PackArmLibraryRelocations(options.android_pack_relocations,
|
||||
options.android_objcopy,
|
||||
options.added_section,
|
||||
options.test_file,
|
||||
options.packed_output)
|
||||
|
||||
UnpackArmLibraryRelocations(options.android_pack_relocations,
|
||||
options.packed_output,
|
||||
options.unpacked_output)
|
||||
|
||||
return 0
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
sys.exit(main())
|
|
@ -1,35 +0,0 @@
|
|||
#!/bin/bash
|
||||
#
|
||||
# Copyright 2014 The Chromium Authors. All rights reserved.
|
||||
# Use of this source code is governed by a BSD-style license that can be
|
||||
# found in the LICENSE file.
|
||||
|
||||
# Generates elf_file_unittest_relocs_arm{32,64}{,_packed}.so test data files
|
||||
# from elf_file_unittest_relocs.cc. Run once to create these test data
|
||||
# files; the files are checked into the source tree.
|
||||
#
|
||||
# To use:
|
||||
# ./generate_elf_file_unittest_relocs.sh
|
||||
# git add elf_file_unittest_relocs_arm{32,64}{,_packed}.so
|
||||
|
||||
function main() {
|
||||
local '-r' test_data_directory="$(pwd)"
|
||||
cd '../../..'
|
||||
|
||||
source tools/cr/cr-bash-helpers.sh
|
||||
local arch
|
||||
for arch in 'arm32' 'arm64'; do
|
||||
cr 'init' '--platform=android' '--type=Debug' '--architecture='"${arch}"
|
||||
cr 'build' 'relocation_packer_unittests_test_data'
|
||||
done
|
||||
|
||||
local '-r' packer='out_android/Debug/obj/tools/relocation_packer'
|
||||
local '-r' gen="${packer}/relocation_packer_unittests_test_data.gen"
|
||||
|
||||
cp "${gen}/elf_file_unittest_relocs_arm"{32,64}{,_packed}'.so' \
|
||||
"${test_data_directory}"
|
||||
|
||||
return 0
|
||||
}
|
||||
|
||||
main
|
Loading…
Reference in New Issue