Home | History | Annotate | Download | only in pm
      1 /*
      2  * Copyright (C) 2008 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.content.pm;
     18 
     19 import android.os.Parcel;
     20 import android.os.Parcelable;
     21 
     22 import com.android.internal.util.ArrayUtils;
     23 
     24 import java.io.ByteArrayInputStream;
     25 import java.io.InputStream;
     26 import java.lang.ref.SoftReference;
     27 import java.security.PublicKey;
     28 import java.security.cert.Certificate;
     29 import java.security.cert.CertificateEncodingException;
     30 import java.security.cert.CertificateException;
     31 import java.security.cert.CertificateFactory;
     32 import java.security.cert.X509Certificate;
     33 import java.util.Arrays;
     34 
     35 /**
     36  * Opaque, immutable representation of a signing certificate associated with an
     37  * application package.
     38  * <p>
     39  * This class name is slightly misleading, since it's not actually a signature.
     40  */
     41 public class Signature implements Parcelable {
     42     private final byte[] mSignature;
     43     private int mHashCode;
     44     private boolean mHaveHashCode;
     45     private SoftReference<String> mStringRef;
     46     private Certificate[] mCertificateChain;
     47 
     48     /**
     49      * Create Signature from an existing raw byte array.
     50      */
     51     public Signature(byte[] signature) {
     52         mSignature = signature.clone();
     53         mCertificateChain = null;
     54     }
     55 
     56     /**
     57      * Create signature from a certificate chain. Used for backward
     58      * compatibility.
     59      *
     60      * @throws CertificateEncodingException
     61      * @hide
     62      */
     63     public Signature(Certificate[] certificateChain) throws CertificateEncodingException {
     64         mSignature = certificateChain[0].getEncoded();
     65         if (certificateChain.length > 1) {
     66             mCertificateChain = Arrays.copyOfRange(certificateChain, 1, certificateChain.length);
     67         }
     68     }
     69 
     70     private static final int parseHexDigit(int nibble) {
     71         if ('0' <= nibble && nibble <= '9') {
     72             return nibble - '0';
     73         } else if ('a' <= nibble && nibble <= 'f') {
     74             return nibble - 'a' + 10;
     75         } else if ('A' <= nibble && nibble <= 'F') {
     76             return nibble - 'A' + 10;
     77         } else {
     78             throw new IllegalArgumentException("Invalid character " + nibble + " in hex string");
     79         }
     80     }
     81 
     82     /**
     83      * Create Signature from a text representation previously returned by
     84      * {@link #toChars} or {@link #toCharsString()}. Signatures are expected to
     85      * be a hex-encoded ASCII string.
     86      *
     87      * @param text hex-encoded string representing the signature
     88      * @throws IllegalArgumentException when signature is odd-length
     89      */
     90     public Signature(String text) {
     91         final byte[] input = text.getBytes();
     92         final int N = input.length;
     93 
     94         if (N % 2 != 0) {
     95             throw new IllegalArgumentException("text size " + N + " is not even");
     96         }
     97 
     98         final byte[] sig = new byte[N / 2];
     99         int sigIndex = 0;
    100 
    101         for (int i = 0; i < N;) {
    102             final int hi = parseHexDigit(input[i++]);
    103             final int lo = parseHexDigit(input[i++]);
    104             sig[sigIndex++] = (byte) ((hi << 4) | lo);
    105         }
    106 
    107         mSignature = sig;
    108     }
    109 
    110     /**
    111      * Encode the Signature as ASCII text.
    112      */
    113     public char[] toChars() {
    114         return toChars(null, null);
    115     }
    116 
    117     /**
    118      * Encode the Signature as ASCII text in to an existing array.
    119      *
    120      * @param existingArray Existing char array or null.
    121      * @param outLen Output parameter for the number of characters written in
    122      * to the array.
    123      * @return Returns either <var>existingArray</var> if it was large enough
    124      * to hold the ASCII representation, or a newly created char[] array if
    125      * needed.
    126      */
    127     public char[] toChars(char[] existingArray, int[] outLen) {
    128         byte[] sig = mSignature;
    129         final int N = sig.length;
    130         final int N2 = N*2;
    131         char[] text = existingArray == null || N2 > existingArray.length
    132                 ? new char[N2] : existingArray;
    133         for (int j=0; j<N; j++) {
    134             byte v = sig[j];
    135             int d = (v>>4)&0xf;
    136             text[j*2] = (char)(d >= 10 ? ('a' + d - 10) : ('0' + d));
    137             d = v&0xf;
    138             text[j*2+1] = (char)(d >= 10 ? ('a' + d - 10) : ('0' + d));
    139         }
    140         if (outLen != null) outLen[0] = N;
    141         return text;
    142     }
    143 
    144     /**
    145      * Return the result of {@link #toChars()} as a String.
    146      */
    147     public String toCharsString() {
    148         String str = mStringRef == null ? null : mStringRef.get();
    149         if (str != null) {
    150             return str;
    151         }
    152         str = new String(toChars());
    153         mStringRef = new SoftReference<String>(str);
    154         return str;
    155     }
    156 
    157     /**
    158      * @return the contents of this signature as a byte array.
    159      */
    160     public byte[] toByteArray() {
    161         byte[] bytes = new byte[mSignature.length];
    162         System.arraycopy(mSignature, 0, bytes, 0, mSignature.length);
    163         return bytes;
    164     }
    165 
    166     /**
    167      * Returns the public key for this signature.
    168      *
    169      * @throws CertificateException when Signature isn't a valid X.509
    170      *             certificate; shouldn't happen.
    171      * @hide
    172      */
    173     public PublicKey getPublicKey() throws CertificateException {
    174         final CertificateFactory certFactory = CertificateFactory.getInstance("X.509");
    175         final ByteArrayInputStream bais = new ByteArrayInputStream(mSignature);
    176         final Certificate cert = certFactory.generateCertificate(bais);
    177         return cert.getPublicKey();
    178     }
    179 
    180     /**
    181      * Used for compatibility code that needs to check the certificate chain
    182      * during upgrades.
    183      *
    184      * @throws CertificateEncodingException
    185      * @hide
    186      */
    187     public Signature[] getChainSignatures() throws CertificateEncodingException {
    188         if (mCertificateChain == null) {
    189             return new Signature[] { this };
    190         }
    191 
    192         Signature[] chain = new Signature[1 + mCertificateChain.length];
    193         chain[0] = this;
    194 
    195         int i = 1;
    196         for (Certificate c : mCertificateChain) {
    197             chain[i++] = new Signature(c.getEncoded());
    198         }
    199 
    200         return chain;
    201     }
    202 
    203     @Override
    204     public boolean equals(Object obj) {
    205         try {
    206             if (obj != null) {
    207                 Signature other = (Signature)obj;
    208                 return this == other || Arrays.equals(mSignature, other.mSignature);
    209             }
    210         } catch (ClassCastException e) {
    211         }
    212         return false;
    213     }
    214 
    215     @Override
    216     public int hashCode() {
    217         if (mHaveHashCode) {
    218             return mHashCode;
    219         }
    220         mHashCode = Arrays.hashCode(mSignature);
    221         mHaveHashCode = true;
    222         return mHashCode;
    223     }
    224 
    225     public int describeContents() {
    226         return 0;
    227     }
    228 
    229     public void writeToParcel(Parcel dest, int parcelableFlags) {
    230         dest.writeByteArray(mSignature);
    231     }
    232 
    233     public static final Parcelable.Creator<Signature> CREATOR
    234             = new Parcelable.Creator<Signature>() {
    235         public Signature createFromParcel(Parcel source) {
    236             return new Signature(source);
    237         }
    238 
    239         public Signature[] newArray(int size) {
    240             return new Signature[size];
    241         }
    242     };
    243 
    244     private Signature(Parcel source) {
    245         mSignature = source.createByteArray();
    246     }
    247 
    248     /**
    249      * Test if given {@link Signature} sets are exactly equal.
    250      *
    251      * @hide
    252      */
    253     public static boolean areExactMatch(Signature[] a, Signature[] b) {
    254         return (a.length == b.length) && ArrayUtils.containsAll(a, b)
    255                 && ArrayUtils.containsAll(b, a);
    256     }
    257 
    258     /**
    259      * Test if given {@link Signature} sets are effectively equal. In rare
    260      * cases, certificates can have slightly malformed encoding which causes
    261      * exact-byte checks to fail.
    262      * <p>
    263      * To identify effective equality, we bounce the certificates through an
    264      * decode/encode pass before doing the exact-byte check. To reduce attack
    265      * surface area, we only allow a byte size delta of a few bytes.
    266      *
    267      * @throws CertificateException if the before/after length differs
    268      *             substantially, usually a signal of something fishy going on.
    269      * @hide
    270      */
    271     public static boolean areEffectiveMatch(Signature[] a, Signature[] b)
    272             throws CertificateException {
    273         final CertificateFactory cf = CertificateFactory.getInstance("X.509");
    274 
    275         final Signature[] aPrime = new Signature[a.length];
    276         for (int i = 0; i < a.length; i++) {
    277             aPrime[i] = bounce(cf, a[i]);
    278         }
    279         final Signature[] bPrime = new Signature[b.length];
    280         for (int i = 0; i < b.length; i++) {
    281             bPrime[i] = bounce(cf, b[i]);
    282         }
    283 
    284         return areExactMatch(aPrime, bPrime);
    285     }
    286 
    287     /**
    288      * Test if given {@link Signature} objects are effectively equal. In rare
    289      * cases, certificates can have slightly malformed encoding which causes
    290      * exact-byte checks to fail.
    291      * <p>
    292      * To identify effective equality, we bounce the certificates through an
    293      * decode/encode pass before doing the exact-byte check. To reduce attack
    294      * surface area, we only allow a byte size delta of a few bytes.
    295      *
    296      * @throws CertificateException if the before/after length differs
    297      *             substantially, usually a signal of something fishy going on.
    298      * @hide
    299      */
    300     public static boolean areEffectiveMatch(Signature a, Signature b)
    301             throws CertificateException {
    302         final CertificateFactory cf = CertificateFactory.getInstance("X.509");
    303 
    304         final Signature aPrime = bounce(cf, a);
    305         final Signature bPrime = bounce(cf, b);
    306 
    307         return aPrime.equals(bPrime);
    308     }
    309 
    310     /**
    311      * Bounce the given {@link Signature} through a decode/encode cycle.
    312      *
    313      * @throws CertificateException if the before/after length differs
    314      *             substantially, usually a signal of something fishy going on.
    315      * @hide
    316      */
    317     public static Signature bounce(CertificateFactory cf, Signature s) throws CertificateException {
    318         final InputStream is = new ByteArrayInputStream(s.mSignature);
    319         final X509Certificate cert = (X509Certificate) cf.generateCertificate(is);
    320         final Signature sPrime = new Signature(cert.getEncoded());
    321 
    322         if (Math.abs(sPrime.mSignature.length - s.mSignature.length) > 2) {
    323             throw new CertificateException("Bounced cert length looks fishy; before "
    324                     + s.mSignature.length + ", after " + sPrime.mSignature.length);
    325         }
    326 
    327         return sPrime;
    328     }
    329 }
    330