1 /* 2 * Copyright (C) 2016 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.os; 18 19 import android.annotation.IntDef; 20 import android.annotation.NonNull; 21 import android.annotation.Nullable; 22 import android.annotation.SystemApi; 23 import android.compat.annotation.UnsupportedAppUsage; 24 25 import dalvik.annotation.optimization.FastNative; 26 27 import libcore.util.NativeAllocationRegistry; 28 29 import java.lang.annotation.Retention; 30 import java.lang.annotation.RetentionPolicy; 31 import java.util.ArrayList; 32 import java.util.Arrays; 33 34 /** @hide */ 35 @SystemApi 36 public class HwParcel { 37 private static final String TAG = "HwParcel"; 38 39 /** @hide */ 40 @IntDef(prefix = { "STATUS_" }, value = { 41 STATUS_SUCCESS, 42 }) 43 @Retention(RetentionPolicy.SOURCE) 44 public @interface Status {} 45 46 /** 47 * Success return error for a transaction. Written to parcels 48 * using writeStatus. 49 */ 50 public static final int STATUS_SUCCESS = 0; 51 52 private static final NativeAllocationRegistry sNativeRegistry; 53 54 @UnsupportedAppUsage HwParcel(boolean allocate)55 private HwParcel(boolean allocate) { 56 native_setup(allocate); 57 58 sNativeRegistry.registerNativeAllocation( 59 this, 60 mNativeContext); 61 } 62 63 /** 64 * Creates an initialized and empty parcel. 65 */ HwParcel()66 public HwParcel() { 67 native_setup(true /* allocate */); 68 69 sNativeRegistry.registerNativeAllocation( 70 this, 71 mNativeContext); 72 } 73 74 /** 75 * Writes an interface token into the parcel used to verify that 76 * a transaction has made it to the right type of interface. 77 * 78 * @param interfaceName fully qualified name of interface message 79 * is being sent to. 80 */ 81 @FastNative writeInterfaceToken(String interfaceName)82 public native final void writeInterfaceToken(String interfaceName); 83 /** 84 * Writes a boolean value to the end of the parcel. 85 * @param val to write 86 */ 87 @FastNative writeBool(boolean val)88 public native final void writeBool(boolean val); 89 /** 90 * Writes a byte value to the end of the parcel. 91 * @param val to write 92 */ 93 @FastNative writeInt8(byte val)94 public native final void writeInt8(byte val); 95 /** 96 * Writes a short value to the end of the parcel. 97 * @param val to write 98 */ 99 @FastNative writeInt16(short val)100 public native final void writeInt16(short val); 101 /** 102 * Writes a int value to the end of the parcel. 103 * @param val to write 104 */ 105 @FastNative writeInt32(int val)106 public native final void writeInt32(int val); 107 /** 108 * Writes a long value to the end of the parcel. 109 * @param val to write 110 */ 111 @FastNative writeInt64(long val)112 public native final void writeInt64(long val); 113 /** 114 * Writes a float value to the end of the parcel. 115 * @param val to write 116 */ 117 @FastNative writeFloat(float val)118 public native final void writeFloat(float val); 119 /** 120 * Writes a double value to the end of the parcel. 121 * @param val to write 122 */ 123 @FastNative writeDouble(double val)124 public native final void writeDouble(double val); 125 /** 126 * Writes a String value to the end of the parcel. 127 * 128 * Note, this will be converted to UTF-8 when it is written. 129 * 130 * @param val to write 131 */ 132 @FastNative writeString(String val)133 public native final void writeString(String val); 134 /** 135 * Writes a native handle (without duplicating the underlying 136 * file descriptors) to the end of the parcel. 137 * 138 * @param val to write 139 */ 140 @FastNative writeNativeHandle(@ullable NativeHandle val)141 public native final void writeNativeHandle(@Nullable NativeHandle val); 142 143 /** 144 * Writes an array of boolean values to the end of the parcel. 145 * @param val to write 146 */ 147 @FastNative writeBoolVector(boolean[] val)148 private native final void writeBoolVector(boolean[] val); 149 /** 150 * Writes an array of byte values to the end of the parcel. 151 * @param val to write 152 */ 153 @FastNative writeInt8Vector(byte[] val)154 private native final void writeInt8Vector(byte[] val); 155 /** 156 * Writes an array of short values to the end of the parcel. 157 * @param val to write 158 */ 159 @FastNative writeInt16Vector(short[] val)160 private native final void writeInt16Vector(short[] val); 161 /** 162 * Writes an array of int values to the end of the parcel. 163 * @param val to write 164 */ 165 @FastNative writeInt32Vector(int[] val)166 private native final void writeInt32Vector(int[] val); 167 /** 168 * Writes an array of long values to the end of the parcel. 169 * @param val to write 170 */ 171 @FastNative writeInt64Vector(long[] val)172 private native final void writeInt64Vector(long[] val); 173 /** 174 * Writes an array of float values to the end of the parcel. 175 * @param val to write 176 */ 177 @FastNative writeFloatVector(float[] val)178 private native final void writeFloatVector(float[] val); 179 /** 180 * Writes an array of double values to the end of the parcel. 181 * @param val to write 182 */ 183 @FastNative writeDoubleVector(double[] val)184 private native final void writeDoubleVector(double[] val); 185 /** 186 * Writes an array of String values to the end of the parcel. 187 * 188 * Note, these will be converted to UTF-8 as they are written. 189 * 190 * @param val to write 191 */ 192 @FastNative writeStringVector(String[] val)193 private native final void writeStringVector(String[] val); 194 /** 195 * Writes an array of native handles to the end of the parcel. 196 * 197 * Individual elements may be null but not the whole array. 198 * 199 * @param val array of {@link NativeHandle} objects to write 200 */ 201 @FastNative writeNativeHandleVector(NativeHandle[] val)202 private native final void writeNativeHandleVector(NativeHandle[] val); 203 204 /** 205 * Helper method to write a list of Booleans to val. 206 * @param val list to write 207 */ writeBoolVector(ArrayList<Boolean> val)208 public final void writeBoolVector(ArrayList<Boolean> val) { 209 final int n = val.size(); 210 boolean[] array = new boolean[n]; 211 for (int i = 0; i < n; ++i) { 212 array[i] = val.get(i); 213 } 214 215 writeBoolVector(array); 216 } 217 218 /** 219 * Helper method to write a list of Booleans to the end of the parcel. 220 * @param val list to write 221 */ writeInt8Vector(ArrayList<Byte> val)222 public final void writeInt8Vector(ArrayList<Byte> val) { 223 final int n = val.size(); 224 byte[] array = new byte[n]; 225 for (int i = 0; i < n; ++i) { 226 array[i] = val.get(i); 227 } 228 229 writeInt8Vector(array); 230 } 231 232 /** 233 * Helper method to write a list of Shorts to the end of the parcel. 234 * @param val list to write 235 */ writeInt16Vector(ArrayList<Short> val)236 public final void writeInt16Vector(ArrayList<Short> val) { 237 final int n = val.size(); 238 short[] array = new short[n]; 239 for (int i = 0; i < n; ++i) { 240 array[i] = val.get(i); 241 } 242 243 writeInt16Vector(array); 244 } 245 246 /** 247 * Helper method to write a list of Integers to the end of the parcel. 248 * @param val list to write 249 */ writeInt32Vector(ArrayList<Integer> val)250 public final void writeInt32Vector(ArrayList<Integer> val) { 251 final int n = val.size(); 252 int[] array = new int[n]; 253 for (int i = 0; i < n; ++i) { 254 array[i] = val.get(i); 255 } 256 257 writeInt32Vector(array); 258 } 259 260 /** 261 * Helper method to write a list of Longs to the end of the parcel. 262 * @param val list to write 263 */ writeInt64Vector(ArrayList<Long> val)264 public final void writeInt64Vector(ArrayList<Long> val) { 265 final int n = val.size(); 266 long[] array = new long[n]; 267 for (int i = 0; i < n; ++i) { 268 array[i] = val.get(i); 269 } 270 271 writeInt64Vector(array); 272 } 273 274 /** 275 * Helper method to write a list of Floats to the end of the parcel. 276 * @param val list to write 277 */ writeFloatVector(ArrayList<Float> val)278 public final void writeFloatVector(ArrayList<Float> val) { 279 final int n = val.size(); 280 float[] array = new float[n]; 281 for (int i = 0; i < n; ++i) { 282 array[i] = val.get(i); 283 } 284 285 writeFloatVector(array); 286 } 287 288 /** 289 * Helper method to write a list of Doubles to the end of the parcel. 290 * @param val list to write 291 */ writeDoubleVector(ArrayList<Double> val)292 public final void writeDoubleVector(ArrayList<Double> val) { 293 final int n = val.size(); 294 double[] array = new double[n]; 295 for (int i = 0; i < n; ++i) { 296 array[i] = val.get(i); 297 } 298 299 writeDoubleVector(array); 300 } 301 302 /** 303 * Helper method to write a list of Strings to the end of the parcel. 304 * @param val list to write 305 */ writeStringVector(ArrayList<String> val)306 public final void writeStringVector(ArrayList<String> val) { 307 writeStringVector(val.toArray(new String[val.size()])); 308 } 309 310 /** 311 * Helper method to write a list of native handles to the end of the parcel. 312 * @param val list of {@link NativeHandle} objects to write 313 */ writeNativeHandleVector(@onNull ArrayList<NativeHandle> val)314 public final void writeNativeHandleVector(@NonNull ArrayList<NativeHandle> val) { 315 writeNativeHandleVector(val.toArray(new NativeHandle[val.size()])); 316 } 317 318 /** 319 * Write a hwbinder object to the end of the parcel. 320 * @param binder value to write 321 */ 322 @FastNative writeStrongBinder(IHwBinder binder)323 public native final void writeStrongBinder(IHwBinder binder); 324 325 /** 326 * Write a HidlMemory object (without duplicating the underlying file descriptors) to the end 327 * of the parcel. 328 * 329 * @param memory value to write 330 */ 331 @FastNative writeHidlMemory(@onNull HidlMemory memory)332 public native final void writeHidlMemory(@NonNull HidlMemory memory); 333 334 /** 335 * Checks to make sure that the interface name matches the name written by the parcel 336 * sender by writeInterfaceToken 337 * 338 * @throws SecurityException interface doesn't match 339 */ enforceInterface(String interfaceName)340 public native final void enforceInterface(String interfaceName); 341 342 /** 343 * Reads a boolean value from the current location in the parcel. 344 * @return value parsed from the parcel 345 * @throws IllegalArgumentException if the parcel has no more data 346 */ 347 @FastNative readBool()348 public native final boolean readBool(); 349 /** 350 * Reads a byte value from the current location in the parcel. 351 * @return value parsed from the parcel 352 * @throws IllegalArgumentException if the parcel has no more data 353 */ 354 @FastNative readInt8()355 public native final byte readInt8(); 356 /** 357 * Reads a short value from the current location in the parcel. 358 * @return value parsed from the parcel 359 * @throws IllegalArgumentException if the parcel has no more data 360 */ 361 @FastNative readInt16()362 public native final short readInt16(); 363 /** 364 * Reads a int value from the current location in the parcel. 365 * @return value parsed from the parcel 366 * @throws IllegalArgumentException if the parcel has no more data 367 */ 368 @FastNative readInt32()369 public native final int readInt32(); 370 /** 371 * Reads a long value from the current location in the parcel. 372 * @return value parsed from the parcel 373 * @throws IllegalArgumentException if the parcel has no more data 374 */ 375 @FastNative readInt64()376 public native final long readInt64(); 377 /** 378 * Reads a float value from the current location in the parcel. 379 * @return value parsed from the parcel 380 * @throws IllegalArgumentException if the parcel has no more data 381 */ 382 @FastNative readFloat()383 public native final float readFloat(); 384 /** 385 * Reads a double value from the current location in the parcel. 386 * @return value parsed from the parcel 387 * @throws IllegalArgumentException if the parcel has no more data 388 */ 389 @FastNative readDouble()390 public native final double readDouble(); 391 /** 392 * Reads a String value from the current location in the parcel. 393 * @return value parsed from the parcel 394 * @throws IllegalArgumentException if the parcel has no more data 395 */ 396 @FastNative readString()397 public native final String readString(); 398 /** 399 * Reads a native handle (without duplicating the underlying file 400 * descriptors) from the parcel. These file descriptors will only 401 * be open for the duration that the binder window is open. If they 402 * are needed further, you must call {@link NativeHandle#dup()}. 403 * 404 * @return a {@link NativeHandle} instance parsed from the parcel 405 * @throws IllegalArgumentException if the parcel has no more data 406 */ 407 @FastNative readNativeHandle()408 public native final @Nullable NativeHandle readNativeHandle(); 409 /** 410 * Reads an embedded native handle (without duplicating the underlying 411 * file descriptors) from the parcel. These file descriptors will only 412 * be open for the duration that the binder window is open. If they 413 * are needed further, you must call {@link NativeHandle#dup()}. You 414 * do not need to call close on the NativeHandle returned from this. 415 * 416 * @param parentHandle handle from which to read the embedded object 417 * @param offset offset into parent 418 * @return a {@link NativeHandle} instance parsed from the parcel 419 * @throws IllegalArgumentException if the parcel has no more data 420 */ 421 @FastNative readEmbeddedNativeHandle( long parentHandle, long offset)422 public native final @Nullable NativeHandle readEmbeddedNativeHandle( 423 long parentHandle, long offset); 424 425 /** 426 * Reads an array of boolean values from the parcel. 427 * @return array of parsed values 428 * @throws IllegalArgumentException if the parcel has no more data 429 */ 430 @FastNative readBoolVectorAsArray()431 private native final boolean[] readBoolVectorAsArray(); 432 /** 433 * Reads an array of byte values from the parcel. 434 * @return array of parsed values 435 * @throws IllegalArgumentException if the parcel has no more data 436 */ 437 @FastNative readInt8VectorAsArray()438 private native final byte[] readInt8VectorAsArray(); 439 /** 440 * Reads an array of short values from the parcel. 441 * @return array of parsed values 442 * @throws IllegalArgumentException if the parcel has no more data 443 */ 444 @FastNative readInt16VectorAsArray()445 private native final short[] readInt16VectorAsArray(); 446 /** 447 * Reads an array of int values from the parcel. 448 * @return array of parsed values 449 * @throws IllegalArgumentException if the parcel has no more data 450 */ 451 @FastNative readInt32VectorAsArray()452 private native final int[] readInt32VectorAsArray(); 453 /** 454 * Reads an array of long values from the parcel. 455 * @return array of parsed values 456 * @throws IllegalArgumentException if the parcel has no more data 457 */ 458 @FastNative readInt64VectorAsArray()459 private native final long[] readInt64VectorAsArray(); 460 /** 461 * Reads an array of float values from the parcel. 462 * @return array of parsed values 463 * @throws IllegalArgumentException if the parcel has no more data 464 */ 465 @FastNative readFloatVectorAsArray()466 private native final float[] readFloatVectorAsArray(); 467 /** 468 * Reads an array of double values from the parcel. 469 * @return array of parsed values 470 * @throws IllegalArgumentException if the parcel has no more data 471 */ 472 @FastNative readDoubleVectorAsArray()473 private native final double[] readDoubleVectorAsArray(); 474 /** 475 * Reads an array of String values from the parcel. 476 * @return array of parsed values 477 * @throws IllegalArgumentException if the parcel has no more data 478 */ 479 @FastNative readStringVectorAsArray()480 private native final String[] readStringVectorAsArray(); 481 /** 482 * Reads an array of native handles from the parcel. 483 * @return array of {@link NativeHandle} objects 484 * @throws IllegalArgumentException if the parcel has no more data 485 */ 486 @FastNative readNativeHandleAsArray()487 private native final NativeHandle[] readNativeHandleAsArray(); 488 489 /** 490 * Convenience method to read a Boolean vector as an ArrayList. 491 * @return array of parsed values. 492 * @throws IllegalArgumentException if the parcel has no more data 493 */ readBoolVector()494 public final ArrayList<Boolean> readBoolVector() { 495 Boolean[] array = HwBlob.wrapArray(readBoolVectorAsArray()); 496 497 return new ArrayList<Boolean>(Arrays.asList(array)); 498 } 499 500 /** 501 * Convenience method to read a Byte vector as an ArrayList. 502 * @return array of parsed values. 503 * @throws IllegalArgumentException if the parcel has no more data 504 */ readInt8Vector()505 public final ArrayList<Byte> readInt8Vector() { 506 Byte[] array = HwBlob.wrapArray(readInt8VectorAsArray()); 507 508 return new ArrayList<Byte>(Arrays.asList(array)); 509 } 510 511 /** 512 * Convenience method to read a Short vector as an ArrayList. 513 * @return array of parsed values. 514 * @throws IllegalArgumentException if the parcel has no more data 515 */ readInt16Vector()516 public final ArrayList<Short> readInt16Vector() { 517 Short[] array = HwBlob.wrapArray(readInt16VectorAsArray()); 518 519 return new ArrayList<Short>(Arrays.asList(array)); 520 } 521 522 /** 523 * Convenience method to read a Integer vector as an ArrayList. 524 * @return array of parsed values. 525 * @throws IllegalArgumentException if the parcel has no more data 526 */ readInt32Vector()527 public final ArrayList<Integer> readInt32Vector() { 528 Integer[] array = HwBlob.wrapArray(readInt32VectorAsArray()); 529 530 return new ArrayList<Integer>(Arrays.asList(array)); 531 } 532 533 /** 534 * Convenience method to read a Long vector as an ArrayList. 535 * @return array of parsed values. 536 * @throws IllegalArgumentException if the parcel has no more data 537 */ readInt64Vector()538 public final ArrayList<Long> readInt64Vector() { 539 Long[] array = HwBlob.wrapArray(readInt64VectorAsArray()); 540 541 return new ArrayList<Long>(Arrays.asList(array)); 542 } 543 544 /** 545 * Convenience method to read a Float vector as an ArrayList. 546 * @return array of parsed values. 547 * @throws IllegalArgumentException if the parcel has no more data 548 */ readFloatVector()549 public final ArrayList<Float> readFloatVector() { 550 Float[] array = HwBlob.wrapArray(readFloatVectorAsArray()); 551 552 return new ArrayList<Float>(Arrays.asList(array)); 553 } 554 555 /** 556 * Convenience method to read a Double vector as an ArrayList. 557 * @return array of parsed values. 558 * @throws IllegalArgumentException if the parcel has no more data 559 */ readDoubleVector()560 public final ArrayList<Double> readDoubleVector() { 561 Double[] array = HwBlob.wrapArray(readDoubleVectorAsArray()); 562 563 return new ArrayList<Double>(Arrays.asList(array)); 564 } 565 566 /** 567 * Convenience method to read a String vector as an ArrayList. 568 * @return array of parsed values. 569 * @throws IllegalArgumentException if the parcel has no more data 570 */ readStringVector()571 public final ArrayList<String> readStringVector() { 572 return new ArrayList<String>(Arrays.asList(readStringVectorAsArray())); 573 } 574 575 /** 576 * Convenience method to read a vector of native handles as an ArrayList. 577 * @return array of {@link NativeHandle} objects. 578 * @throws IllegalArgumentException if the parcel has no more data 579 */ readNativeHandleVector()580 public final @NonNull ArrayList<NativeHandle> readNativeHandleVector() { 581 return new ArrayList<NativeHandle>(Arrays.asList(readNativeHandleAsArray())); 582 } 583 584 /** 585 * Reads a strong binder value from the parcel. 586 * @return binder object read from parcel or null if no binder can be read 587 * @throws IllegalArgumentException if the parcel has no more data 588 */ 589 @FastNative readStrongBinder()590 public native final IHwBinder readStrongBinder(); 591 592 /** 593 * Reads a HidlMemory value (without duplicating the underlying file 594 * descriptors) from the parcel. These file descriptors will only 595 * be open for the duration that the binder window is open. If they 596 * are needed further, you must call {@link HidlMemory#dup()}, which makes you also 597 * responsible for calling {@link HidlMemory#close()}. 598 * 599 * @return HidlMemory object read from parcel. 600 * @throws IllegalArgumentException if the parcel has no more data or is otherwise corrupt. 601 */ 602 @FastNative 603 @NonNull readHidlMemory()604 public native final HidlMemory readHidlMemory(); 605 606 /** 607 * Reads an embedded HidlMemory (without duplicating the underlying 608 * file descriptors) from the parcel. These file descriptors will only 609 * be open for the duration that the binder window is open. If they 610 * are needed further, you must call {@link HidlMemory#dup()}. You 611 * do not need to call close on the HidlMemory returned from this. 612 * 613 * @param fieldHandle handle of the field, obtained from the {@link HwBlob}. 614 * @param parentHandle parentHandle from which to read the embedded object 615 * @param offset offset into parent 616 * @return a {@link HidlMemory} instance parsed from the parcel 617 * @throws IllegalArgumentException if the parcel has no more data 618 */ 619 @FastNative 620 @NonNull 621 public native final readEmbeddedHidlMemory(long fieldHandle, long parentHandle, long offset)622 HidlMemory readEmbeddedHidlMemory(long fieldHandle, long parentHandle, long offset); 623 624 /** 625 * Read opaque segment of data as a blob. 626 * @return blob of size expectedSize 627 * @throws IllegalArgumentException if the parcel has no more data 628 */ 629 @FastNative readBuffer(long expectedSize)630 public native final HwBlob readBuffer(long expectedSize); 631 632 /** 633 * Read a buffer written using scatter gather. 634 * 635 * @param expectedSize size that buffer should be 636 * @param parentHandle handle from which to read the embedded buffer 637 * @param offset offset into parent 638 * @param nullable whether or not to allow for a null return 639 * @return blob of data with size expectedSize 640 * @throws NoSuchElementException if an embedded buffer is not available to read 641 * @throws IllegalArgumentException if expectedSize < 0 642 * @throws NullPointerException if the transaction specified the blob to be null 643 * but nullable is false 644 */ 645 @FastNative readEmbeddedBuffer( long expectedSize, long parentHandle, long offset, boolean nullable)646 public native final HwBlob readEmbeddedBuffer( 647 long expectedSize, long parentHandle, long offset, 648 boolean nullable); 649 650 /** 651 * Write a buffer into the transaction. 652 * @param blob blob to write into the parcel. 653 */ 654 @FastNative writeBuffer(HwBlob blob)655 public native final void writeBuffer(HwBlob blob); 656 /** 657 * Write a status value into the blob. 658 * @param status value to write 659 */ 660 @FastNative writeStatus(int status)661 public native final void writeStatus(int status); 662 /** 663 * @throws IllegalArgumentException if a success vaue cannot be read 664 * @throws RemoteException if success value indicates a transaction error 665 */ 666 @FastNative verifySuccess()667 public native final void verifySuccess(); 668 /** 669 * Should be called to reduce memory pressure when this object no longer needs 670 * to be written to. 671 */ 672 @FastNative releaseTemporaryStorage()673 public native final void releaseTemporaryStorage(); 674 /** 675 * Should be called when object is no longer needed to reduce possible memory 676 * pressure if the Java GC does not get to this object in time. 677 */ 678 @FastNative release()679 public native final void release(); 680 681 /** 682 * Sends the parcel to the specified destination. 683 */ send()684 public native final void send(); 685 686 // Returns address of the "freeFunction". native_init()687 private static native final long native_init(); 688 689 @FastNative native_setup(boolean allocate)690 private native final void native_setup(boolean allocate); 691 692 static { 693 long freeFunction = native_init(); 694 695 sNativeRegistry = new NativeAllocationRegistry( 696 HwParcel.class.getClassLoader(), 697 freeFunction, 698 128 /* size */); 699 } 700 701 private long mNativeContext; 702 } 703 704