blob: 2db411d53d55d81dc909c967275f90abc571382c [file] [log] [blame]
// Copyright 2018 The Bazel Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.unsafe;
import static com.google.devtools.build.lib.unsafe.UnsafeProvider.unsafe;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.util.Arrays;
/**
* Provides direct access to the string implementation used by JDK9.
*
* <p>As of JDK9, a string is two fields: <code>byte coder</code>, and <code>byte[] value</code>.
* The <code>coder</code> field has value 0 if the encoding is LATIN-1, and 2 if the encoding is
* UTF-16 (the classic JDK8 encoding).
*
* <p>The <code>value</code> field contains the actual bytes.
*/
public final class StringUnsafe {
// Fields corresponding to the coder
public static final byte LATIN1 = 0;
public static final byte UTF16 = 1;
private static final StringUnsafe INSTANCE = new StringUnsafe();
private final Constructor<String> constructor;
private final long valueOffset;
private final long coderOffset;
public static StringUnsafe getInstance() {
return INSTANCE;
}
private StringUnsafe() {
Field valueField;
Field coderField;
try {
this.constructor = String.class.getDeclaredConstructor(byte[].class, byte.class);
valueField = String.class.getDeclaredField("value");
coderField = String.class.getDeclaredField("coder");
} catch (ReflectiveOperationException e) {
throw new IllegalStateException(
"Bad fields/constructor: "
+ Arrays.toString(String.class.getDeclaredConstructors())
+ ", "
+ Arrays.toString(String.class.getDeclaredFields()),
e);
}
this.constructor.setAccessible(true);
valueOffset = unsafe().objectFieldOffset(valueField);
coderOffset = unsafe().objectFieldOffset(coderField);
}
/** Returns the coder used for this string. See {@link #LATIN1} and {@link #UTF16}. */
public byte getCoder(String obj) {
return unsafe().getByte(obj, coderOffset);
}
/**
* Returns the internal byte array, encoded according to {@link #getCoder}.
*
* <p>Use of this is unsafe. The representation may change from one JDK version to the next.
* Ensure you do not mutate this byte array in any way.
*/
public byte[] getByteArray(String obj) {
return (byte[]) unsafe().getObject(obj, valueOffset);
}
/** Constructs a new string from a byte array and coder. */
public String newInstance(byte[] bytes, byte coder) throws ReflectiveOperationException {
return constructor.newInstance(bytes, coder);
}
}