diff options
| author | 2025-03-07 14:55:20 -0800 | |
|---|---|---|
| committer | 2025-03-07 14:55:20 -0800 | |
| commit | fa522a19cc5e7c94e56c4be5c8ffcdbf8b92f9d2 (patch) | |
| tree | 1bb6d237b25822dfc0e48d8023d94bdfa664cc8f | |
| parent | 7297bd67f3188dafb3c32b6e6052e3db442ea69c (diff) | |
Introduce Parcel.{,un}marshall methods that work with ByteBuffers
Flag: EXEMPT test-only (not used in any production code yet)
Bug: 401362825
Change-Id: If4c8a0edfa33aac374a2aebbeaed1cd0b0ae6b74
| -rw-r--r-- | core/java/android/os/Parcel.java | 78 | ||||
| -rw-r--r-- | core/jni/android_os_Parcel.cpp | 87 | ||||
| -rw-r--r-- | core/tests/coretests/src/android/os/ParcelTest.java | 61 | 
3 files changed, 224 insertions, 2 deletions
| diff --git a/core/java/android/os/Parcel.java b/core/java/android/os/Parcel.java index 49d3f06eb80f..6cb49b3ea166 100644 --- a/core/java/android/os/Parcel.java +++ b/core/java/android/os/Parcel.java @@ -52,6 +52,8 @@ import com.android.internal.util.ArrayUtils;  import dalvik.annotation.optimization.CriticalNative;  import dalvik.annotation.optimization.FastNative; +import java.nio.BufferOverflowException; +import java.nio.ReadOnlyBufferException;  import libcore.util.SneakyThrow;  import java.io.ByteArrayInputStream; @@ -62,6 +64,7 @@ import java.io.ObjectInputStream;  import java.io.ObjectOutputStream;  import java.io.ObjectStreamClass;  import java.io.Serializable; +import java.nio.ByteBuffer;  import java.lang.annotation.Retention;  import java.lang.annotation.RetentionPolicy;  import java.lang.reflect.Array; @@ -457,8 +460,15 @@ public final class Parcel {      private static native void nativeDestroy(long nativePtr);      private static native byte[] nativeMarshall(long nativePtr); +    private static native int nativeMarshallArray( +            long nativePtr, byte[] data, int offset, int length); +    private static native int nativeMarshallBuffer( +            long nativePtr, ByteBuffer buffer, int offset, int length);      private static native void nativeUnmarshall(              long nativePtr, byte[] data, int offset, int length); +    private static native void nativeUnmarshallBuffer( +            long nativePtr, ByteBuffer buffer, int offset, int length); +      private static native int nativeCompareData(long thisNativePtr, long otherNativePtr);      private static native boolean nativeCompareDataInRange(              long ptrA, int offsetA, long ptrB, int offsetB, int length); @@ -814,12 +824,80 @@ public final class Parcel {      }      /** +     * Writes the raw bytes of the parcel to a buffer. +     * +     * <p class="note">The data you retrieve here <strong>must not</strong> +     * be placed in any kind of persistent storage (on local disk, across +     * a network, etc).  For that, you should use standard serialization +     * or another kind of general serialization mechanism.  The Parcel +     * marshalled representation is highly optimized for local IPC, and as +     * such does not attempt to maintain compatibility with data created +     * in different versions of the platform. +     * +     * @param buffer The ByteBuffer to write the data to. +     * @throws ReadOnlyBufferException if the buffer is read-only. +     * @throws BufferOverflowException if the buffer is too small. +     * +     * @hide +     */ +    public final void marshall(@NonNull ByteBuffer buffer) { +        if (buffer == null) { +            throw new NullPointerException(); +        } +        if (buffer.isReadOnly()) { +            throw new ReadOnlyBufferException(); +        } + +        final int position = buffer.position(); +        final int remaining = buffer.remaining(); + +        int marshalledSize = 0; +        if (buffer.isDirect()) { +            marshalledSize = nativeMarshallBuffer(mNativePtr, buffer, position, remaining); +        } else if (buffer.hasArray()) { +            marshalledSize = nativeMarshallArray( +                    mNativePtr, buffer.array(), buffer.arrayOffset() + position, remaining); +        } else { +            throw new IllegalArgumentException(); +        } + +        buffer.position(position + marshalledSize); +    } + +    /**       * Fills the raw bytes of this Parcel with the supplied data.       */      public final void unmarshall(@NonNull byte[] data, int offset, int length) {          nativeUnmarshall(mNativePtr, data, offset, length);      } +    /** +     * Fills the raw bytes of this Parcel with data from the supplied buffer. +     * +     * @param buffer will read buffer.remaining() bytes from the buffer. +     * +     * @hide +     */ +    public final void unmarshall(@NonNull ByteBuffer buffer) { +        if (buffer == null) { +            throw new NullPointerException(); +        } + +        final int position = buffer.position(); +        final int remaining = buffer.remaining(); + +        if (buffer.isDirect()) { +            nativeUnmarshallBuffer(mNativePtr, buffer, position, remaining); +        } else if (buffer.hasArray()) { +            nativeUnmarshall( +                    mNativePtr, buffer.array(), buffer.arrayOffset() + position, remaining); +        } else { +            throw new IllegalArgumentException(); +        } + +        buffer.position(position + remaining); +    } +      public final void appendFrom(Parcel parcel, int offset, int length) {          nativeAppendFrom(mNativePtr, parcel.mNativePtr, offset, length);      } diff --git a/core/jni/android_os_Parcel.cpp b/core/jni/android_os_Parcel.cpp index dec724b6a7ff..b4c58b9b246a 100644 --- a/core/jni/android_os_Parcel.cpp +++ b/core/jni/android_os_Parcel.cpp @@ -558,8 +558,7 @@ static void android_os_Parcel_destroy(JNIEnv* env, jclass clazz, jlong nativePtr      delete parcel;  } -static jbyteArray android_os_Parcel_marshall(JNIEnv* env, jclass clazz, jlong nativePtr) -{ +static Parcel* parcel_for_marshall(JNIEnv* env, jlong nativePtr) {      Parcel* parcel = reinterpret_cast<Parcel*>(nativePtr);      if (parcel == NULL) {         return NULL; @@ -577,6 +576,16 @@ static jbyteArray android_os_Parcel_marshall(JNIEnv* env, jclass clazz, jlong na          return NULL;      } +    return parcel; +} + +static jbyteArray android_os_Parcel_marshall(JNIEnv* env, jclass clazz, jlong nativePtr) +{ +    Parcel* parcel = parcel_for_marshall(env, nativePtr); +    if (parcel == NULL) { +       return NULL; +    } +      jbyteArray ret = env->NewByteArray(parcel->dataSize());      if (ret != NULL) @@ -592,6 +601,58 @@ static jbyteArray android_os_Parcel_marshall(JNIEnv* env, jclass clazz, jlong na      return ret;  } +static long ensure_capacity(JNIEnv* env, Parcel* parcel, jint remaining) { +    long dataSize = parcel->dataSize(); +    if (remaining < dataSize) { +        jniThrowExceptionFmt(env, "java/nio/BufferOverflowException", +                             "Destination buffer remaining capacity %d is less than the Parcel data size %d.", +                             remaining, dataSize); +        return -1; +    } +    return dataSize; +} + +static int android_os_Parcel_marshall_array(JNIEnv* env, jclass clazz, jlong nativePtr, +                                            jbyteArray data, jint offset, jint remaining) +{ +    Parcel* parcel = parcel_for_marshall(env, nativePtr); +    if (parcel == NULL) { +       return 0; +    } + +    long data_size = ensure_capacity(env, parcel, remaining); +    if (data_size < 0) { +        return 0; +    } + +    jbyte* array = (jbyte*)env->GetPrimitiveArrayCritical(data, 0); +    if (array != NULL) +    { +        memcpy(array + offset, parcel->data(), data_size); +        env->ReleasePrimitiveArrayCritical(data, array, 0); +    } +    return data_size; +} + +static int android_os_Parcel_marshall_buffer(JNIEnv* env, jclass clazz, jlong nativePtr, +                                             jobject javaBuffer, jint offset, jint remaining) { +    Parcel* parcel = parcel_for_marshall(env, nativePtr); +    if (parcel == NULL) { +       return 0; +    } + +    long data_size = ensure_capacity(env, parcel, remaining); +    if (data_size < 0) { +        return 0; +    } + +    jbyte* buffer = (jbyte*)env->GetDirectBufferAddress(javaBuffer); +    if (buffer != NULL) { +        memcpy(buffer + offset, parcel->data(), data_size); +    } +    return data_size; +} +  static void android_os_Parcel_unmarshall(JNIEnv* env, jclass clazz, jlong nativePtr,                                            jbyteArray data, jint offset, jint length)  { @@ -613,6 +674,25 @@ static void android_os_Parcel_unmarshall(JNIEnv* env, jclass clazz, jlong native      }  } +static void android_os_Parcel_unmarshall_buffer(JNIEnv* env, jclass clazz, jlong nativePtr, +                                                jobject javaBuffer, jint offset, jint length) +{ +    Parcel* parcel = reinterpret_cast<Parcel*>(nativePtr); +    if (parcel == NULL || length < 0) { +       return; +    } + +    jbyte* buffer = (jbyte*)env->GetDirectBufferAddress(javaBuffer); +    if (buffer) +    { +        parcel->setDataSize(length); +        parcel->setDataPosition(0); + +        void* raw = parcel->writeInplace(length); +        memcpy(raw, (buffer + offset), length); +    } +} +  static jint android_os_Parcel_compareData(JNIEnv* env, jclass clazz, jlong thisNativePtr,                                            jlong otherNativePtr)  { @@ -911,7 +991,10 @@ static const JNINativeMethod gParcelMethods[] = {      {"nativeDestroy",             "(J)V", (void*)android_os_Parcel_destroy},      {"nativeMarshall",            "(J)[B", (void*)android_os_Parcel_marshall}, +    {"nativeMarshallArray",       "(J[BII)I", (void*)android_os_Parcel_marshall_array}, +    {"nativeMarshallBuffer",      "(JLjava/nio/ByteBuffer;II)I", (void*)android_os_Parcel_marshall_buffer},      {"nativeUnmarshall",          "(J[BII)V", (void*)android_os_Parcel_unmarshall}, +    {"nativeUnmarshallBuffer",    "(JLjava/nio/ByteBuffer;II)V", (void*)android_os_Parcel_unmarshall_buffer},      {"nativeCompareData",         "(JJ)I", (void*)android_os_Parcel_compareData},      {"nativeCompareDataInRange",  "(JIJII)Z", (void*)android_os_Parcel_compareDataInRange},      {"nativeAppendFrom",          "(JJII)V", (void*)android_os_Parcel_appendFrom}, diff --git a/core/tests/coretests/src/android/os/ParcelTest.java b/core/tests/coretests/src/android/os/ParcelTest.java index 3e6520106ab0..bb059108d4b6 100644 --- a/core/tests/coretests/src/android/os/ParcelTest.java +++ b/core/tests/coretests/src/android/os/ParcelTest.java @@ -29,6 +29,8 @@ import android.util.Log;  import androidx.test.ext.junit.runners.AndroidJUnit4; +import java.nio.BufferOverflowException; +import java.nio.ByteBuffer;  import org.junit.Rule;  import org.junit.Test;  import org.junit.runner.RunWith; @@ -416,4 +418,63 @@ public class ParcelTest {          int binderEndPos = pA.dataPosition();          assertTrue(pA.hasBinders(binderStartPos, binderEndPos - binderStartPos));      } + +    private static final byte[] TEST_DATA = new byte[] {4, 8, 15, 16, 23, 42}; + +    // Allow for some Parcel overhead +    private static final int TEST_DATA_LENGTH = TEST_DATA.length + 100; + +    @Test +    public void testMarshall_ByteBuffer_wrapped() { +        ByteBuffer bb = ByteBuffer.allocate(TEST_DATA_LENGTH); +        testMarshall_ByteBuffer(bb); +    } + +    @Test +    public void testMarshall_DirectByteBuffer() { +        ByteBuffer bb = ByteBuffer.allocateDirect(TEST_DATA_LENGTH); +        testMarshall_ByteBuffer(bb); +    } + +    private void testMarshall_ByteBuffer(ByteBuffer bb) { +        // Ensure that Parcel respects the starting offset by not starting at 0 +        bb.position(1); +        bb.mark(); + +        // Parcel test data, then marshall into the ByteBuffer +        Parcel p1 = Parcel.obtain(); +        p1.writeByteArray(TEST_DATA); +        p1.marshall(bb); +        p1.recycle(); + +        assertTrue(bb.position() > 1); +        bb.reset(); + +        // Unmarshall test data into a new Parcel +        Parcel p2 = Parcel.obtain(); +        bb.reset(); +        p2.unmarshall(bb); +        assertTrue(bb.position() > 1); +        p2.setDataPosition(0); +        byte[] marshalled = p2.marshall(); + +        bb.reset(); +        for (int i = 0; i < TEST_DATA.length; i++) { +            assertEquals(bb.get(), marshalled[i]); +        } + +        byte[] testDataCopy = new byte[TEST_DATA.length]; +        p2.setDataPosition(0); +        p2.readByteArray(testDataCopy); +        for (int i = 0; i < TEST_DATA.length; i++) { +            assertEquals(TEST_DATA[i], testDataCopy[i]); +        } + +        // Test that overflowing the buffer throws an exception +        bb.reset(); +        // Leave certainly not enough room for the test data +        bb.limit(bb.position() + TEST_DATA.length - 1); +        assertThrows(BufferOverflowException.class, () -> p2.marshall(bb)); +        p2.recycle(); +    }  } |