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 17 package android.location.cts.asn1.base; 18 19 import com.google.common.base.Preconditions; 20 import com.google.common.collect.ImmutableList; 21 22 import java.math.BigInteger; 23 import java.util.Arrays; 24 25 /** 26 * Basic algorithms for aligned PER encoding and decoding, ASN.1 X.691-0207. 27 * 28 */ 29 public class PerAlignedUtils { 30 31 public static final int ONE_K = 1024; 32 public static final int SIXTEEN_K = 16 * ONE_K; 33 public static final int SIXTYFOUR_K = 64 * ONE_K; 34 35 36 /** 37 * Encodes whole numbers up to 64K range according to X.691-0207, 10.5. 38 */ encodeSmallConstrainedWholeNumber(int value, int minimumValue, int maximumValue)39 public static BitStream encodeSmallConstrainedWholeNumber(int value, 40 int minimumValue, 41 int maximumValue) { 42 int normalizedValue = value - minimumValue; 43 // Note: range here means one less than in ASN.1 X.691-0207, 10.5. 44 int range = maximumValue - minimumValue; 45 return encodeNormalizedSmallConstrainedWholeNumber(normalizedValue, range); 46 } 47 48 /** 49 * Encodes the difference between the actual value and the minimum allowed 50 * value, the {@code normalizedValue}, for whole numbers up to 64K range 51 * according to X.691-0207, 10.5. 52 * 53 * <p>Note: range here means one less than in ASN.1 X.691-0207, 10.5., i.e. 54 * here it is the difference between the maximum allowed value and the minimum 55 * allowed value. 56 */ encodeNormalizedSmallConstrainedWholeNumber( int normalizedValue, int range)57 public static BitStream encodeNormalizedSmallConstrainedWholeNumber( 58 int normalizedValue, int range) { 59 Preconditions.checkArgument(range < SIXTYFOUR_K, "range >= 64K"); 60 Preconditions.checkArgument(normalizedValue >= 0, 61 "negative normalized value"); 62 BitStream result = new BitStream(); 63 if (range == 0) { 64 return result; 65 } 66 if (range < 128) { 67 result.appendLowBits(leastBitsToEncode((byte) range), 68 (byte) normalizedValue); 69 return result; 70 } 71 if (range >= 255) { 72 result.setBeginByteAligned(); 73 } 74 if (range < 256) { 75 result.appendByte((byte) (normalizedValue)); 76 return result; 77 } 78 result.appendByte((byte) ((normalizedValue & 0xFF00) >>> 8)); 79 result.appendByte((byte) (normalizedValue & 0x00FF)); 80 return result; 81 } 82 83 /** 84 * Decodes whole numbers up to 64K range according to X.691-0207, 10.5. 85 */ decodeSmallConstrainedWholeNumber(BitStreamReader reader, int minimumValue, int maximumValue)86 public static int decodeSmallConstrainedWholeNumber(BitStreamReader reader, 87 int minimumValue, 88 int maximumValue) { 89 // Note: range here means one less than in ASN.1 X.691-0207, 10.5. 90 int range = maximumValue - minimumValue; 91 int normalizedResult = 92 decodeNormalizedSmallConstrainedWholeNumber(reader, range); 93 return normalizedResult + minimumValue; 94 } 95 96 /** 97 * Decodes the difference between the actual value and the minimum allowed 98 * value for whole numbers up to 64K range according to X.691-0207, 10.5. 99 * 100 * <p>Note: range here means one less than in ASN.1 X.691-0207, 10.5., i.e. 101 * here it is the difference between the maximum allowed value and the minimum 102 * allowed value. 103 */ decodeNormalizedSmallConstrainedWholeNumber( BitStreamReader reader, int range)104 public static int decodeNormalizedSmallConstrainedWholeNumber( 105 BitStreamReader reader, int range) { 106 if (range < 0) { 107 throw new IllegalArgumentException("range < 0"); 108 } 109 if (range >= SIXTYFOUR_K) { 110 throw new IllegalArgumentException("range >= 64K"); 111 } 112 if (range == 0) { 113 return 0; 114 } 115 if (range < 128) { 116 return reader.readLowBits(leastBitsToEncode((byte) range)); 117 } 118 if (range >= 255) { 119 reader.spoolToByteBoundary(); 120 } 121 if (range < 256) { 122 return (reader.readByte() & 0xFF); 123 } 124 return ((reader.readByte() & 0xFF) << 8) 125 + (reader.readByte() & 0xFF); 126 } 127 leastBitsToEncode(byte value)128 private static int leastBitsToEncode(byte value) { 129 int unsignedByte = value & 0xFF; 130 for (int bits = 1; bits < 8; bits++) { 131 if (unsignedByte < (1 << bits)) { 132 return bits; 133 } 134 } 135 return 8; 136 } 137 encodeNormallySmallWholeNumber(int value)138 public static Iterable<BitStream> encodeNormallySmallWholeNumber(int value) { 139 if (value < 64) { 140 BitStream result = new BitStream(); 141 result.appendBit(false); 142 result.appendLowBits(6, (byte) value); 143 return ImmutableList.of(result); 144 } 145 throw new UnsupportedOperationException("normally small numbers >= 64 " 146 + "unimplemented"); 147 } 148 decodeNormallySmallWholeNumber(BitStreamReader reader)149 public static int decodeNormallySmallWholeNumber(BitStreamReader reader) { 150 if (reader.readBit()) { 151 throw new UnsupportedOperationException("normally small numbers >= 64 " 152 + "unimplemented"); 153 } 154 return reader.readLowBits(6) & 0xFF; 155 } 156 157 /** 158 * Encodes length determinant for a constrained length byte[] according to 159 * X.691-0207, 10.9.3.3 and up. 160 */ encodeConstrainedLengthOfBytes( byte[] bytes, int minimumLength, int maximumLength)161 public static Iterable<BitStream> encodeConstrainedLengthOfBytes( 162 byte[] bytes, int minimumLength, int maximumLength) { 163 if (maximumLength >= SIXTYFOUR_K) { 164 return encodeSemiConstrainedLengthOfBytes(bytes); 165 } 166 167 BitStream lengthDeterminant = encodeSmallConstrainedWholeNumber( 168 bytes.length, minimumLength, maximumLength); 169 if (bytes.length == 0) { 170 return ImmutableList.of(lengthDeterminant); 171 } 172 BitStream value = new BitStream(); 173 value.setBeginByteAligned(); 174 for (byte aByte : bytes) { 175 value.appendByte(aByte); 176 } 177 return ImmutableList.of(lengthDeterminant, value); 178 179 } 180 181 /** 182 * Decodes a constrained length byte[] with length determinant according to 183 * X.691-0207, 10.9.3.3 and up. 184 */ decodeConstrainedLengthOfBytes( BitStreamReader reader, int minimumLength, int maximumLength)185 public static byte[] decodeConstrainedLengthOfBytes( 186 BitStreamReader reader, int minimumLength, int maximumLength) { 187 if (maximumLength >= SIXTYFOUR_K) { 188 return decodeSemiConstrainedLengthOfBytes(reader); 189 } 190 int length = decodeSmallConstrainedWholeNumber( 191 reader, minimumLength, maximumLength); 192 if (length == 0) { 193 return new byte[0]; 194 } 195 byte[] result = new byte[length]; 196 reader.spoolToByteBoundary(); 197 for (int i = 0; i < length; i++) { 198 result[i] = reader.readByte(); 199 } 200 return result; 201 } 202 203 /** 204 * Encodes length determinant for a semi-constrained length byte[] according 205 * to X.691-0207, 10.9.3.5. 206 */ encodeSemiConstrainedLengthOfBytes( byte[] bytes)207 public static Iterable<BitStream> encodeSemiConstrainedLengthOfBytes( 208 byte[] bytes) { 209 int n = bytes.length; 210 if (n < SIXTEEN_K) { 211 BitStream result = encodeSemiConstrainedLength(n); 212 result.setBeginByteAligned(); 213 for (byte b : bytes) { 214 result.appendByte(b); 215 } 216 return ImmutableList.of(result); 217 } 218 throw new UnsupportedOperationException("Arrays > 16K unimplemented."); 219 } 220 /** 221 * Encodes length determinant for a semi-constrained length byte[] according 222 * to X.691-0207, 10.9.3.5. 223 */ encodeUnconstrainedLengthOfBytes( byte[] bytes)224 public static Iterable<BitStream> encodeUnconstrainedLengthOfBytes( 225 byte[] bytes) { 226 // Length for unconsrtained and semiconstrained integers is encoded by the 227 // same rules. 228 return encodeSemiConstrainedLengthOfBytes(bytes); 229 } 230 /** 231 * Decodes length determinant for a semi-constrained length byte[] according 232 * to X.691-0207, 10.9.3.5. 233 */ decodeSemiConstrainedLengthOfBytes( BitStreamReader reader)234 public static byte[] decodeSemiConstrainedLengthOfBytes( 235 BitStreamReader reader) { 236 reader.spoolToByteBoundary(); 237 int length = decodeSemiConstrainedLength(reader); 238 byte[] result = new byte[length]; 239 for (int i = 0; i < length; i++) { 240 result[i] = reader.readByte(); 241 } 242 return result; 243 } 244 /** 245 * Decodes length determinant for a semi-constrained length byte[] according 246 * to X.691-0207, 10.9.3.5. 247 */ decodeUnconstrainedLengthOfBytes( BitStreamReader reader)248 public static byte[] decodeUnconstrainedLengthOfBytes( 249 BitStreamReader reader) { 250 // Length for unconsrtained and semiconstrained integers is encoded by the 251 // same rules. 252 return decodeSemiConstrainedLengthOfBytes(reader); 253 } 254 /** 255 * Encodes non-negative numbers according to X.691-0207, 10.3. 256 */ encodeBigNonNegativeWholeNumber(BigInteger bigInteger)257 public static byte[] encodeBigNonNegativeWholeNumber(BigInteger bigInteger) { 258 byte[] twosComplement = bigInteger.toByteArray(); 259 return twosComplement[0] == 0 260 ? Arrays.copyOfRange(twosComplement, 1, twosComplement.length) 261 : twosComplement; 262 } 263 264 /** 265 * Decodes non-negative numbers according to X.691-0207, 10.3. 266 */ decodeBigNonNegativeWholeNumber(byte[] encoded)267 public static BigInteger decodeBigNonNegativeWholeNumber(byte[] encoded) { 268 return new BigInteger(1, encoded); 269 } 270 271 /** 272 * Encodes length determinant according to X.691-0207, 10.9.3.6. 273 */ encodeSemiConstrainedLength(int value)274 public static BitStream encodeSemiConstrainedLength(int value) { 275 if (value <= 127) { 276 BitStream result = new BitStream(); 277 result.setBeginByteAligned(); 278 result.appendBit(false); 279 result.appendLowBits(7, (byte) value); 280 return result; 281 } else if (value < SIXTEEN_K) { 282 BitStream result = new BitStream(); 283 result.setBeginByteAligned(); 284 result.appendBit(true); 285 result.appendBit(false); 286 result.appendLowBits(6, (byte) (value >>> 8)); 287 result.appendByte((byte) (value & 0xFF)); 288 return result; 289 } 290 throw new UnsupportedOperationException("Length values > " + 291 SIXTEEN_K + "unimplemented"); 292 } 293 294 /** 295 * Decodes length determinant according to X.691-0207, 10.9.3.6. 296 */ decodeSemiConstrainedLength(BitStreamReader reader)297 public static int decodeSemiConstrainedLength(BitStreamReader reader) { 298 reader.spoolToByteBoundary(); 299 if (!reader.readBit()) { 300 return reader.readLowBits(7); 301 } else if (!reader.readBit()) { 302 return (reader.readLowBits(6) << 8) + (reader.readByte() & 0xFF); 303 } else { 304 throw new UnsupportedOperationException("Length values > " + 305 SIXTEEN_K + "unimplemented"); 306 } 307 } 308 309 /* 310 * Encodes an Asn1Object into a Open type field (X.691-0207, 10.2), used 311 * mostly for encoding Sequence and SetOf extension additions. A decode method 312 * hasn't been added as the extension additions should decoded 313 * by their relevant Asn1Object decoders. 314 */ encodeOpenTypeField( Asn1Object object)315 public static Iterable<BitStream> encodeOpenTypeField( 316 Asn1Object object){ 317 PacketBuilder packetBuilder = new PacketBuilder(); 318 packetBuilder.appendAll(object.encodePerAligned()); 319 return encodeSemiConstrainedLengthOfBytes(packetBuilder.getPaddedBytes()); 320 } 321 decodeOpenTypeField( BitStreamReader reader, Asn1Object asn1Object)322 public static Asn1Object decodeOpenTypeField( 323 BitStreamReader reader, Asn1Object asn1Object) { 324 byte [] encodedBytes = decodeSemiConstrainedLengthOfBytes(reader); 325 asn1Object.decodePerAligned(new BitStreamReader(encodedBytes)); 326 return asn1Object; 327 } 328 } 329