1 /* 2 * Copyright (C) 2017 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 package com.android.server.usb.descriptors; 17 18 // Framework builds and Android Studio builds use different imports for NonNull. 19 // This one for Framework builds 20 import android.annotation.NonNull; 21 // this one in the AndroidStudio project 22 // import android.support.annotation.NonNull; 23 24 /** 25 * @hide 26 * A stream interface wrapping a byte array. Very much like a java.io.ByteArrayInputStream 27 * but with the capability to "back up" in situations where the parser discovers that a 28 * UsbDescriptor has overrun its length. 29 */ 30 public final class ByteStream { 31 private static final String TAG = "ByteStream"; 32 33 /** The byte array being wrapped */ 34 @NonNull 35 private final byte[] mBytes; // this is never null. 36 37 /** 38 * The index into the byte array to be read next. 39 * This value is altered by reading data out of the stream 40 * (using either the getByte() or unpack*() methods), or alternatively 41 * by explicitly offseting the stream position with either 42 * advance() or reverse(). 43 */ 44 private int mIndex; 45 46 /* 47 * This member used with resetReadCount() & getReadCount() can be used to determine how many 48 * bytes a UsbDescriptor subclass ACTUALLY reads (as opposed to what its length field says). 49 * using this info, the parser can mark a descriptor as valid or invalid and correct the stream 50 * position with advance() & reverse() to keep from "getting lost" in the descriptor stream. 51 */ 52 private int mReadCount; 53 54 /** 55 * Create a ByteStream object wrapping the specified byte array. 56 * 57 * @param bytes The byte array containing the raw descriptor information retrieved from 58 * the USB device. 59 * @throws IllegalArgumentException 60 */ ByteStream(@onNull byte[] bytes)61 public ByteStream(@NonNull byte[] bytes) { 62 if (bytes == null) { 63 throw new IllegalArgumentException(); 64 } 65 mBytes = bytes; 66 } 67 68 /** 69 * Resets the running count of bytes read so that later we can see how much more has been read. 70 */ resetReadCount()71 public void resetReadCount() { 72 mReadCount = 0; 73 } 74 75 /** 76 * Retrieves the running count of bytes read from the stream. 77 */ getReadCount()78 public int getReadCount() { 79 return mReadCount; 80 } 81 82 /** 83 * @return The value of the next byte in the stream without advancing the stream. 84 * Does not affect the running count as the byte hasn't been "consumed". 85 * @throws IndexOutOfBoundsException 86 */ peekByte()87 public byte peekByte() { 88 if (available() > 0) { 89 return mBytes[mIndex + 1]; 90 } else { 91 throw new IndexOutOfBoundsException(); 92 } 93 } 94 95 /** 96 * @return the next byte from the stream and advances the stream and the read count. Note 97 * that this is a signed byte (as is the case of byte in Java). The user may need to understand 98 * from context if it should be interpreted as an unsigned value. 99 * @throws IndexOutOfBoundsException 100 */ getByte()101 public byte getByte() { 102 if (available() > 0) { 103 mReadCount++; 104 return mBytes[mIndex++]; 105 } else { 106 throw new IndexOutOfBoundsException(); 107 } 108 } 109 110 /** 111 * @return the next byte from the stream and advances the stream and the read count. Note 112 * that this is an unsigned byte encoded in a Java int. 113 * @throws IndexOutOfBoundsException 114 */ getUnsignedByte()115 public int getUnsignedByte() { 116 if (available() > 0) { 117 mReadCount++; 118 return mBytes[mIndex++] & 0x000000FF; 119 } else { 120 throw new IndexOutOfBoundsException(); 121 } 122 } 123 124 /** 125 * Reads 2 bytes in *little endian format* from the stream and composes a 16-bit integer. 126 * As we are storing the 2-byte value in a 4-byte integer, the upper 2 bytes are always 127 * 0, essentially making the returned value *unsigned*. 128 * @return The 16-bit integer (packed into the lower 2 bytes of an int) encoded by the 129 * next 2 bytes in the stream. 130 * @throws IndexOutOfBoundsException 131 */ unpackUsbShort()132 public int unpackUsbShort() { 133 if (available() >= 2) { 134 int b0 = getUnsignedByte(); 135 int b1 = getUnsignedByte(); 136 return (b1 << 8) | b0; 137 } else { 138 throw new IndexOutOfBoundsException(); 139 } 140 } 141 142 /** 143 * Reads 3 bytes in *little endian format* from the stream and composes a 24-bit integer. 144 * As we are storing the 3-byte value in a 4-byte integer, the upper byte is always 145 * 0, essentially making the returned value *unsigned*. 146 * @return The 24-bit integer (packed into the lower 3 bytes of an int) encoded by the 147 * next 3 bytes in the stream. 148 * @throws IndexOutOfBoundsException 149 */ unpackUsbTriple()150 public int unpackUsbTriple() { 151 if (available() >= 3) { 152 int b0 = getUnsignedByte(); 153 int b1 = getUnsignedByte(); 154 int b2 = getUnsignedByte(); 155 return (b2 << 16) | (b1 << 8) | b0; 156 } else { 157 throw new IndexOutOfBoundsException(); 158 } 159 } 160 161 /** 162 * Reads 4 bytes in *little endian format* from the stream and composes a 32-bit integer. 163 * @return The 32-bit integer encoded by the next 4 bytes in the stream. 164 * @throws IndexOutOfBoundsException 165 */ unpackUsbInt()166 public int unpackUsbInt() { 167 if (available() >= 4) { 168 int b0 = getUnsignedByte(); 169 int b1 = getUnsignedByte(); 170 int b2 = getUnsignedByte(); 171 int b3 = getUnsignedByte(); 172 return (b3 << 24) | (b2 << 16) | (b1 << 8) | b0; 173 } else { 174 throw new IndexOutOfBoundsException(); 175 } 176 } 177 /** 178 * Advances the logical position in the stream. Affects the running count also. 179 * @param numBytes The number of bytes to advance. 180 * @throws IndexOutOfBoundsException 181 * @throws IllegalArgumentException 182 */ advance(int numBytes)183 public void advance(int numBytes) { 184 if (numBytes < 0) { 185 // Positive offsets only 186 throw new IllegalArgumentException(); 187 } 188 // do arithmetic and comparison in long to avoid potential integer overflow 189 long longNewIndex = (long) mIndex + (long) numBytes; 190 if (longNewIndex <= (long) mBytes.length) { 191 mReadCount += numBytes; 192 mIndex += numBytes; 193 } else { 194 // Position the stream to the end so available() will return 0 195 mIndex = mBytes.length; 196 throw new IndexOutOfBoundsException(); 197 } 198 } 199 200 /** 201 * Reverse the logical position in the stream. Affects the running count also. 202 * @param numBytes The (positive) number of bytes to reverse. 203 * @throws IndexOutOfBoundsException 204 * @throws IllegalArgumentException 205 */ reverse(int numBytes)206 public void reverse(int numBytes) { 207 if (numBytes < 0) { 208 // Positive (reverse) offsets only 209 throw new IllegalArgumentException(); 210 } 211 if (mIndex >= numBytes) { 212 mReadCount -= numBytes; 213 mIndex -= numBytes; 214 } else { 215 mIndex = 0; 216 throw new IndexOutOfBoundsException(); 217 } 218 } 219 220 /** 221 * @return The number of bytes available to be read in the stream. 222 */ available()223 public int available() { 224 return mBytes.length - mIndex; 225 } 226 } 227