Home | History | Annotate | Download | only in fortress
      1 /*
      2  *  Licensed to the Apache Software Foundation (ASF) under one or more
      3  *  contributor license agreements.  See the NOTICE file distributed with
      4  *  this work for additional information regarding copyright ownership.
      5  *  The ASF licenses this file to You under the Apache License, Version 2.0
      6  *  (the "License"); you may not use this file except in compliance with
      7  *  the License.  You may obtain a copy of the License at
      8  *
      9  *     http://www.apache.org/licenses/LICENSE-2.0
     10  *
     11  *  Unless required by applicable law or agreed to in writing, software
     12  *  distributed under the License is distributed on an "AS IS" BASIS,
     13  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     14  *  See the License for the specific language governing permissions and
     15  *  limitations under the License.
     16  */
     17 
     18 /**
     19 * @author Boris V. Kuznetsov
     20 * @version $Revision$
     21 */
     22 
     23 package org.apache.harmony.security.fortress;
     24 
     25 import java.security.NoSuchAlgorithmException;
     26 import java.security.Provider;
     27 import java.util.Locale;
     28 
     29 
     30 /**
     31  * This class implements common functionality for Provider supplied
     32  * classes. The usage pattern is to allocate static Engine instance
     33  * per service type and synchronize on that instance during calls to
     34  * {@code getInstance} and retreival of the selected {@code Provider}
     35  * and Service Provider Interface (SPI) results. Retreiving the
     36  * results with {@code getProvider} and {@code getSpi} sets the
     37  * internal {@code Engine} values to null to prevent memory leaks.
     38  *
     39  * <p>
     40  *
     41  * For example: <pre>   {@code
     42  *   public class Foo {
     43  *
     44  *       private static final Engine ENGINE = new Engine("Foo");
     45  *
     46  *       private final FooSpi spi;
     47  *       private final Provider provider;
     48  *       private final String algorithm;
     49  *
     50  *       protected Foo(FooSpi spi,
     51  *                     Provider provider,
     52  *                     String algorithm) {
     53  *           this.spi = spi;
     54  *           this.provider = provider;
     55  *           this.algorithm = algorithm;
     56  *       }
     57  *
     58  *       public static Foo getInstance(String algorithm) {
     59  *           Engine.SpiAndProvider sap = ENGINE.getInstance(algorithm, null);
     60  *           return new Foo((FooSpi) sap.spi, sap.provider, algorithm);
     61  *       }
     62  *
     63  *       public static Foo getInstance(String algorithm, Provider provider) {
     64  *           Object spi = ENGINE.getInstance(algorithm, provider, null);
     65  *           return new Foo((FooSpi) spi, provider, algorithm);
     66  *       }
     67  *
     68  *       ...
     69  *
     70  * }</pre>
     71  */
     72 public class Engine {
     73 
     74     /**
     75      * Access to package visible api in java.security
     76      */
     77     public static SecurityAccess door;
     78 
     79     /**
     80      * Service name such as Cipher or SSLContext
     81      */
     82     private final String serviceName;
     83 
     84     /**
     85      * Previous result for getInstance(String, Object) optimization.
     86      * Only this non-Provider version of getInstance is optimized
     87      * since the the Provider version does not require an expensive
     88      * Services.getService call.
     89      */
     90     private volatile ServiceCacheEntry serviceCache;
     91 
     92     private static final class ServiceCacheEntry {
     93         /** used to test for cache hit */
     94         private final String algorithm;
     95         /** used to test for cache validity */
     96         private final int cacheVersion;
     97         /** cached result */
     98         private final Provider.Service service;
     99 
    100         private ServiceCacheEntry(String algorithm,
    101                                   int cacheVersion,
    102                                   Provider.Service service) {
    103             this.algorithm = algorithm;
    104             this.cacheVersion = cacheVersion;
    105             this.service = service;
    106         }
    107     }
    108 
    109     public static final class SpiAndProvider {
    110         public final Object spi;
    111         public final Provider provider;
    112         private SpiAndProvider(Object spi, Provider provider) {
    113             this.spi = spi;
    114             this.provider = provider;
    115         }
    116     }
    117 
    118     /**
    119      * Creates a Engine object
    120      *
    121      * @param service
    122      */
    123     public Engine(String service) {
    124         this.serviceName = service;
    125     }
    126 
    127     /**
    128      * Finds the appropriate service implementation and returns an
    129      * {@code SpiAndProvider} instance containing a reference to SPI
    130      * and its {@code Provider}
    131      */
    132     public SpiAndProvider getInstance(String algorithm, Object param)
    133             throws NoSuchAlgorithmException {
    134         if (algorithm == null) {
    135             throw new NoSuchAlgorithmException("Null algorithm name");
    136         }
    137         int newCacheVersion = Services.getCacheVersion();
    138         Provider.Service service;
    139         ServiceCacheEntry cacheEntry = this.serviceCache;
    140         if (cacheEntry != null
    141                 && cacheEntry.algorithm.equalsIgnoreCase(algorithm)
    142                 && newCacheVersion == cacheEntry.cacheVersion) {
    143             service = cacheEntry.service;
    144         } else {
    145             if (Services.isEmpty()) {
    146                 throw notFound(serviceName, algorithm);
    147             }
    148             String name = this.serviceName + "." + algorithm.toUpperCase(Locale.US);
    149             service = Services.getService(name);
    150             if (service == null) {
    151                 throw notFound(serviceName, algorithm);
    152             }
    153             this.serviceCache = new ServiceCacheEntry(algorithm, newCacheVersion, service);
    154         }
    155         return new SpiAndProvider(service.newInstance(param), service.getProvider());
    156     }
    157 
    158     /**
    159      * Finds the appropriate service implementation and returns and
    160      * instance of the class that implements corresponding Service
    161      * Provider Interface.
    162      */
    163     public Object getInstance(String algorithm, Provider provider, Object param)
    164             throws NoSuchAlgorithmException {
    165         if (algorithm == null) {
    166             throw new NoSuchAlgorithmException("algorithm == null");
    167         }
    168         Provider.Service service = provider.getService(serviceName, algorithm);
    169         if (service == null) {
    170             throw notFound(serviceName, algorithm);
    171         }
    172         return service.newInstance(param);
    173     }
    174 
    175     private NoSuchAlgorithmException notFound(String serviceName, String algorithm)
    176             throws NoSuchAlgorithmException {
    177         throw new NoSuchAlgorithmException(serviceName + " " + algorithm
    178                                            + " implementation not found");
    179     }
    180 }
    181