blob: 73ae362ede279314052d6797a0de5171286a2701 [file] [log] [blame]
// Copyright (c) 2017, the R8 project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
package com.android.tools.r8.naming;
import com.android.tools.r8.DiagnosticsHandler;
import com.android.tools.r8.StringConsumer;
import com.android.tools.r8.Version;
import com.android.tools.r8.errors.Unreachable;
import com.android.tools.r8.naming.mappinginformation.MetaInfMappingInformation;
import com.android.tools.r8.utils.Box;
import com.android.tools.r8.utils.ChainableStringConsumer;
import com.android.tools.r8.utils.ExceptionUtils;
import com.android.tools.r8.utils.InternalOptions;
import com.android.tools.r8.utils.Reporter;
import com.android.tools.r8.utils.VersionProperties;
import com.google.common.hash.Hasher;
import com.google.common.hash.Hashing;
import java.io.IOException;
public class ProguardMapSupplier {
public static final String MARKER_KEY_COMPILER = "compiler";
public static final String MARKER_VALUE_COMPILER = "R8";
public static final String MARKER_KEY_COMPILER_VERSION = "compiler_version";
public static final String MARKER_KEY_COMPILER_HASH = "compiler_hash";
public static final String MARKER_KEY_MIN_API = "min_api";
public static final String MARKER_KEY_PG_MAP_ID = "pg_map_id";
public static int PG_MAP_ID_LENGTH = 7;
// Truncated murmur hash of the non-whitespace codepoints of the Proguard map (excluding the
// marker).
public static class ProguardMapId extends Box<String> {
private ProguardMapId(String id) {
super(id);
assert id != null;
assert id.length() == PG_MAP_ID_LENGTH;
}
}
private final ClassNameMapper classNameMapper;
private final StringConsumer consumer;
private final InternalOptions options;
private final Reporter reporter;
private ProguardMapSupplier(ClassNameMapper classNameMapper, InternalOptions options) {
assert classNameMapper != null;
assert !classNameMapper.isEmpty();
this.classNameMapper = classNameMapper.sorted();
this.consumer =
InternalOptions.assertionsEnabled()
? new ProguardMapChecker(options.proguardMapConsumer)
: options.proguardMapConsumer;
this.options = options;
this.reporter = options.reporter;
}
public static ProguardMapSupplier create(
ClassNameMapper classNameMapper, InternalOptions options) {
return classNameMapper.isEmpty() ? null : new ProguardMapSupplier(classNameMapper, options);
}
public ProguardMapId writeProguardMap() {
ProguardMapId id = computeProguardMapId();
writeMarker(id);
writeBody();
ExceptionUtils.withFinishedResourceHandler(reporter, consumer);
return id;
}
private ProguardMapId computeProguardMapId() {
ProguardMapIdBuilder builder = new ProguardMapIdBuilder();
classNameMapper.write(builder);
return builder.build();
}
private void writeBody() {
classNameMapper.write(new ProguardMapWriter());
}
private void writeMarker(ProguardMapId id) {
StringBuilder builder = new StringBuilder();
builder.append(
"# "
+ MARKER_KEY_COMPILER
+ ": "
+ MARKER_VALUE_COMPILER
+ "\n"
+ "# "
+ MARKER_KEY_COMPILER_VERSION
+ ": "
+ Version.LABEL
+ "\n");
if (options.isGeneratingDex()) {
builder.append("# " + MARKER_KEY_MIN_API + ": " + options.minApiLevel + "\n");
}
if (Version.isDevelopmentVersion()) {
builder.append(
"# " + MARKER_KEY_COMPILER_HASH + ": " + VersionProperties.INSTANCE.getSha() + "\n");
}
builder.append("# " + MARKER_KEY_PG_MAP_ID + ": " + id.get() + "\n");
// Turn off linting of the mapping file in some build systems.
builder.append("# common_typos_disable" + "\n");
// Emit the R8 specific map-file version.
MapVersion mapVersion = options.getMapFileVersion();
if (mapVersion.isGreaterThan(MapVersion.MapVersionNone)) {
builder
.append("# ")
.append(new MetaInfMappingInformation(mapVersion).serialize())
.append("\n");
}
consumer.accept(builder.toString(), reporter);
}
static class ProguardMapIdBuilder implements ChainableStringConsumer {
private final Hasher hasher = Hashing.murmur3_32().newHasher();
@Override
public ProguardMapIdBuilder accept(String string) {
for (int i = 0; i < string.length(); i++) {
char c = string.charAt(i);
if (!Character.isWhitespace(c)) {
hasher.putInt(c);
}
}
return this;
}
public ProguardMapId build() {
return new ProguardMapId(hasher.hash().toString().substring(0, PG_MAP_ID_LENGTH));
}
}
class ProguardMapWriter implements ChainableStringConsumer {
@Override
public ProguardMapWriter accept(String string) {
consumer.accept(string, reporter);
return this;
}
}
static class ProguardMapChecker implements StringConsumer {
private final StringConsumer inner;
private final StringBuilder contents = new StringBuilder();
ProguardMapChecker(StringConsumer inner) {
if (!InternalOptions.assertionsEnabled()) {
// Make sure we never get here without assertions enabled.
throw new Unreachable();
}
this.inner = inner;
}
@Override
public void accept(String string, DiagnosticsHandler handler) {
inner.accept(string, handler);
contents.append(string);
}
@Override
public void finished(DiagnosticsHandler handler) {
inner.finished(handler);
assert validateProguardMapParses(contents.toString());
}
private static boolean validateProguardMapParses(String content) {
try {
ClassNameMapper.mapperFromString(content);
} catch (IOException e) {
e.printStackTrace();
return false;
}
return true;
}
}
}