1 /*
2  * Copyright (c) 1999, 2013, Oracle and/or its affiliates. All rights reserved.
3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4  *
5  * This code is free software; you can redistribute it and/or modify it
6  * under the terms of the GNU General Public License version 2 only, as
7  * published by the Free Software Foundation.  Oracle designates this
8  * particular file as subject to the "Classpath" exception as provided
9  * by Oracle in the LICENSE file that accompanied this code.
10  *
11  * This code is distributed in the hope that it will be useful, but WITHOUT
12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
14  * version 2 for more details (a copy is included in the LICENSE file that
15  * accompanied this code).
16  *
17  * You should have received a copy of the GNU General Public License version
18  * 2 along with this work; if not, write to the Free Software Foundation,
19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20  *
21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22  * or visit www.oracle.com if you need additional information or have any
23  * questions.
24  */
25 
26 package javax.crypto;
27 
28 import java.security.AlgorithmParameters;
29 import java.security.Provider;
30 import java.security.Key;
31 import java.security.Security;
32 import java.security.NoSuchAlgorithmException;
33 import java.security.NoSuchProviderException;
34 import java.security.InvalidKeyException;
35 import java.security.InvalidAlgorithmParameterException;
36 import java.security.spec.AlgorithmParameterSpec;
37 
38 import sun.security.jca.GetInstance.Instance;
39 
40 /**
41  * This class provides the functionality of an exemption mechanism, examples
42  * of which are <i>key recovery</i>, <i>key weakening</i>, and
43  * <i>key escrow</i>.
44  *
45  * <p>Applications or applets that use an exemption mechanism may be granted
46  * stronger encryption capabilities than those which don't.
47  *
48  * @since 1.4
49  */
50 
51 public class ExemptionMechanism {
52 
53     // The provider
54     private Provider provider;
55 
56     // The provider implementation (delegate)
57     private ExemptionMechanismSpi exmechSpi;
58 
59     // The name of the exemption mechanism.
60     private String mechanism;
61 
62     // Flag which indicates whether this ExemptionMechanism
63     // result is generated successfully.
64     private boolean done = false;
65 
66     // State information
67     private boolean initialized = false;
68 
69     // Store away the key at init() time for later comparison.
70     private Key keyStored = null;
71 
72     /**
73      * Creates a ExemptionMechanism object.
74      *
75      * @param exmechSpi the delegate
76      * @param provider the provider
77      * @param mechanism the exemption mechanism
78      */
ExemptionMechanism(ExemptionMechanismSpi exmechSpi, Provider provider, String mechanism)79     protected ExemptionMechanism(ExemptionMechanismSpi exmechSpi,
80                                  Provider provider,
81                                  String mechanism) {
82         this.exmechSpi = exmechSpi;
83         this.provider = provider;
84         this.mechanism = mechanism;
85     }
86 
87     /**
88      * Returns the exemption mechanism name of this
89      * <code>ExemptionMechanism</code> object.
90      *
91      * <p>This is the same name that was specified in one of the
92      * <code>getInstance</code> calls that created this
93      * <code>ExemptionMechanism</code> object.
94      *
95      * @return the exemption mechanism name of this
96      * <code>ExemptionMechanism</code> object.
97      */
getName()98     public final String getName() {
99         return this.mechanism;
100     }
101 
102     /**
103      * Returns an <code>ExemptionMechanism</code> object that implements the
104      * specified exemption mechanism algorithm.
105      *
106      * <p> This method traverses the list of registered security Providers,
107      * starting with the most preferred Provider.
108      * A new ExemptionMechanism object encapsulating the
109      * ExemptionMechanismSpi implementation from the first
110      * Provider that supports the specified algorithm is returned.
111      *
112      * <p> Note that the list of registered providers may be retrieved via
113      * the {@link Security#getProviders() Security.getProviders()} method.
114      *
115      * @param algorithm the standard name of the requested exemption
116      * mechanism.
117      * See the ExemptionMechanism section in the
118      * <a href=
119      *   "{@docRoot}/../technotes/guides/security/StandardNames.html#Exemption">
120      * Java Cryptography Architecture Standard Algorithm Name Documentation</a>
121      * for information about standard exemption mechanism names.
122      *
123      * @return the new <code>ExemptionMechanism</code> object.
124      *
125      * @exception NullPointerException if <code>algorithm</code>
126      *          is null.
127      *
128      * @exception NoSuchAlgorithmException if no Provider supports an
129      *          ExemptionMechanismSpi implementation for the
130      *          specified algorithm.
131      *
132      * @see java.security.Provider
133      */
getInstance(String algorithm)134     public static final ExemptionMechanism getInstance(String algorithm)
135             throws NoSuchAlgorithmException {
136         Instance instance = JceSecurity.getInstance("ExemptionMechanism",
137                 ExemptionMechanismSpi.class, algorithm);
138         return new ExemptionMechanism((ExemptionMechanismSpi)instance.impl,
139                 instance.provider, algorithm);
140     }
141 
142 
143     /**
144      * Returns an <code>ExemptionMechanism</code> object that implements the
145      * specified exemption mechanism algorithm.
146      *
147      * <p> A new ExemptionMechanism object encapsulating the
148      * ExemptionMechanismSpi implementation from the specified provider
149      * is returned.  The specified provider must be registered
150      * in the security provider list.
151      *
152      * <p> Note that the list of registered providers may be retrieved via
153      * the {@link Security#getProviders() Security.getProviders()} method.
154 
155      * @param algorithm the standard name of the requested exemption mechanism.
156      * See the ExemptionMechanism section in the
157      * <a href=
158      *   "{@docRoot}/../technotes/guides/security/StandardNames.html#Exemption">
159      * Java Cryptography Architecture Standard Algorithm Name Documentation</a>
160      * for information about standard exemption mechanism names.
161      *
162      * @param provider the name of the provider.
163      *
164      * @return the new <code>ExemptionMechanism</code> object.
165      *
166      * @exception NullPointerException if <code>algorithm</code>
167      *          is null.
168      *
169      * @exception NoSuchAlgorithmException if an ExemptionMechanismSpi
170      *          implementation for the specified algorithm is not
171      *          available from the specified provider.
172      *
173      * @exception NoSuchProviderException if the specified provider is not
174      *          registered in the security provider list.
175      *
176      * @exception IllegalArgumentException if the <code>provider</code>
177      *          is null or empty.
178      *
179      * @see java.security.Provider
180      */
getInstance(String algorithm, String provider)181     public static final ExemptionMechanism getInstance(String algorithm,
182             String provider) throws NoSuchAlgorithmException,
183             NoSuchProviderException {
184         Instance instance = JceSecurity.getInstance("ExemptionMechanism",
185                 ExemptionMechanismSpi.class, algorithm, provider);
186         return new ExemptionMechanism((ExemptionMechanismSpi)instance.impl,
187                 instance.provider, algorithm);
188     }
189 
190     /**
191      * Returns an <code>ExemptionMechanism</code> object that implements the
192      * specified exemption mechanism algorithm.
193      *
194      * <p> A new ExemptionMechanism object encapsulating the
195      * ExemptionMechanismSpi implementation from the specified Provider
196      * object is returned.  Note that the specified Provider object
197      * does not have to be registered in the provider list.
198      *
199      * @param algorithm the standard name of the requested exemption mechanism.
200      * See the ExemptionMechanism section in the
201      * <a href=
202      *   "{@docRoot}/../technotes/guides/security/StandardNames.html#Exemption">
203      * Java Cryptography Architecture Standard Algorithm Name Documentation</a>
204      * for information about standard exemption mechanism names.
205      *
206      * @param provider the provider.
207      *
208      * @return the new <code>ExemptionMechanism</code> object.
209      *
210      * @exception NullPointerException if <code>algorithm</code>
211      *          is null.
212      *
213      * @exception NoSuchAlgorithmException if an ExemptionMechanismSpi
214      *          implementation for the specified algorithm is not available
215      *          from the specified Provider object.
216      *
217      * @exception IllegalArgumentException if the <code>provider</code>
218      *          is null.
219      *
220      * @see java.security.Provider
221      */
getInstance(String algorithm, Provider provider)222     public static final ExemptionMechanism getInstance(String algorithm,
223             Provider provider) throws NoSuchAlgorithmException {
224         Instance instance = JceSecurity.getInstance("ExemptionMechanism",
225                 ExemptionMechanismSpi.class, algorithm, provider);
226         return new ExemptionMechanism((ExemptionMechanismSpi)instance.impl,
227                 instance.provider, algorithm);
228     }
229 
230     /**
231      * Returns the provider of this <code>ExemptionMechanism</code> object.
232      *
233      * @return the provider of this <code>ExemptionMechanism</code> object.
234      */
getProvider()235     public final Provider getProvider() {
236         return this.provider;
237     }
238 
239     /**
240      * Returns whether the result blob has been generated successfully by this
241      * exemption mechanism.
242      *
243      * <p>The method also makes sure that the key passed in is the same as
244      * the one this exemption mechanism used in initializing and generating
245      * phases.
246      *
247      * @param key the key the crypto is going to use.
248      *
249      * @return whether the result blob of the same key has been generated
250      * successfully by this exemption mechanism; false if <code>key</code>
251      * is null.
252      *
253      * @exception ExemptionMechanismException if problem(s) encountered
254      * while determining whether the result blob has been generated successfully
255      * by this exemption mechanism object.
256      */
isCryptoAllowed(Key key)257     public final boolean isCryptoAllowed(Key key)
258             throws ExemptionMechanismException {
259         boolean ret = false;
260         if (done && (key != null)) {
261             // Check if the key passed in is the same as the one
262             // this exemption mechanism used.
263             ret = keyStored.equals(key);
264         }
265         return ret;
266      }
267 
268     /**
269      * Returns the length in bytes that an output buffer would need to be in
270      * order to hold the result of the next
271      * {@link #genExemptionBlob(byte[]) genExemptionBlob}
272      * operation, given the input length <code>inputLen</code> (in bytes).
273      *
274      * <p>The actual output length of the next
275      * {@link #genExemptionBlob(byte[]) genExemptionBlob}
276      * call may be smaller than the length returned by this method.
277      *
278      * @param inputLen the input length (in bytes)
279      *
280      * @return the required output buffer size (in bytes)
281      *
282      * @exception IllegalStateException if this exemption mechanism is in a
283      * wrong state (e.g., has not yet been initialized)
284      */
getOutputSize(int inputLen)285     public final int getOutputSize(int inputLen) throws IllegalStateException {
286         if (!initialized) {
287             throw new IllegalStateException(
288                 "ExemptionMechanism not initialized");
289         }
290         if (inputLen < 0) {
291             throw new IllegalArgumentException(
292                 "Input size must be equal to " + "or greater than zero");
293         }
294         return exmechSpi.engineGetOutputSize(inputLen);
295     }
296 
297     /**
298      * Initializes this exemption mechanism with a key.
299      *
300      * <p>If this exemption mechanism requires any algorithm parameters
301      * that cannot be derived from the given <code>key</code>, the
302      * underlying exemption mechanism implementation is supposed to
303      * generate the required parameters itself (using provider-specific
304      * default values); in the case that algorithm parameters must be
305      * specified by the caller, an <code>InvalidKeyException</code> is raised.
306      *
307      * @param key the key for this exemption mechanism
308      *
309      * @exception InvalidKeyException if the given key is inappropriate for
310      * this exemption mechanism.
311      * @exception ExemptionMechanismException if problem(s) encountered in the
312      * process of initializing.
313      */
init(Key key)314     public final void init(Key key)
315             throws InvalidKeyException, ExemptionMechanismException {
316         done = false;
317         initialized = false;
318 
319         keyStored = key;
320         exmechSpi.engineInit(key);
321         initialized = true;
322     }
323 
324     /**
325      * Initializes this exemption mechanism with a key and a set of algorithm
326      * parameters.
327      *
328      * <p>If this exemption mechanism requires any algorithm parameters
329      * and <code>params</code> is null, the underlying exemption
330      * mechanism implementation is supposed to generate the required
331      * parameters itself (using provider-specific default values); in the case
332      * that algorithm parameters must be specified by the caller, an
333      * <code>InvalidAlgorithmParameterException</code> is raised.
334      *
335      * @param key the key for this exemption mechanism
336      * @param params the algorithm parameters
337      *
338      * @exception InvalidKeyException if the given key is inappropriate for
339      * this exemption mechanism.
340      * @exception InvalidAlgorithmParameterException if the given algorithm
341      * parameters are inappropriate for this exemption mechanism.
342      * @exception ExemptionMechanismException if problem(s) encountered in the
343      * process of initializing.
344      */
init(Key key, AlgorithmParameterSpec params)345     public final void init(Key key, AlgorithmParameterSpec params)
346             throws InvalidKeyException, InvalidAlgorithmParameterException,
347             ExemptionMechanismException {
348         done = false;
349         initialized = false;
350 
351         keyStored = key;
352         exmechSpi.engineInit(key, params);
353         initialized = true;
354     }
355 
356     /**
357      * Initializes this exemption mechanism with a key and a set of algorithm
358      * parameters.
359      *
360      * <p>If this exemption mechanism requires any algorithm parameters
361      * and <code>params</code> is null, the underlying exemption mechanism
362      * implementation is supposed to generate the required parameters itself
363      * (using provider-specific default values); in the case that algorithm
364      * parameters must be specified by the caller, an
365      * <code>InvalidAlgorithmParameterException</code> is raised.
366      *
367      * @param key the key for this exemption mechanism
368      * @param params the algorithm parameters
369      *
370      * @exception InvalidKeyException if the given key is inappropriate for
371      * this exemption mechanism.
372      * @exception InvalidAlgorithmParameterException if the given algorithm
373      * parameters are inappropriate for this exemption mechanism.
374      * @exception ExemptionMechanismException if problem(s) encountered in the
375      * process of initializing.
376      */
init(Key key, AlgorithmParameters params)377     public final void init(Key key, AlgorithmParameters params)
378             throws InvalidKeyException, InvalidAlgorithmParameterException,
379             ExemptionMechanismException {
380         done = false;
381         initialized = false;
382 
383         keyStored = key;
384         exmechSpi.engineInit(key, params);
385         initialized = true;
386     }
387 
388     /**
389      * Generates the exemption mechanism key blob.
390      *
391      * @return the new buffer with the result key blob.
392      *
393      * @exception IllegalStateException if this exemption mechanism is in
394      * a wrong state (e.g., has not been initialized).
395      * @exception ExemptionMechanismException if problem(s) encountered in the
396      * process of generating.
397      */
genExemptionBlob()398     public final byte[] genExemptionBlob() throws IllegalStateException,
399             ExemptionMechanismException {
400         if (!initialized) {
401             throw new IllegalStateException(
402                 "ExemptionMechanism not initialized");
403         }
404         byte[] blob = exmechSpi.engineGenExemptionBlob();
405         done = true;
406         return blob;
407     }
408 
409     /**
410      * Generates the exemption mechanism key blob, and stores the result in
411      * the <code>output</code> buffer.
412      *
413      * <p>If the <code>output</code> buffer is too small to hold the result,
414      * a <code>ShortBufferException</code> is thrown. In this case, repeat this
415      * call with a larger output buffer. Use
416      * {@link #getOutputSize(int) getOutputSize} to determine how big
417      * the output buffer should be.
418      *
419      * @param output the buffer for the result
420      *
421      * @return the number of bytes stored in <code>output</code>
422      *
423      * @exception IllegalStateException if this exemption mechanism is in
424      * a wrong state (e.g., has not been initialized).
425      * @exception ShortBufferException if the given output buffer is too small
426      * to hold the result.
427      * @exception ExemptionMechanismException if problem(s) encountered in the
428      * process of generating.
429      */
genExemptionBlob(byte[] output)430     public final int genExemptionBlob(byte[] output)
431             throws IllegalStateException, ShortBufferException,
432             ExemptionMechanismException {
433         if (!initialized) {
434             throw new IllegalStateException
435             ("ExemptionMechanism not initialized");
436         }
437         int n = exmechSpi.engineGenExemptionBlob(output, 0);
438         done = true;
439         return n;
440     }
441 
442     /**
443      * Generates the exemption mechanism key blob, and stores the result in
444      * the <code>output</code> buffer, starting at <code>outputOffset</code>
445      * inclusive.
446      *
447      * <p>If the <code>output</code> buffer is too small to hold the result,
448      * a <code>ShortBufferException</code> is thrown. In this case, repeat this
449      * call with a larger output buffer. Use
450      * {@link #getOutputSize(int) getOutputSize} to determine how big
451      * the output buffer should be.
452      *
453      * @param output the buffer for the result
454      * @param outputOffset the offset in <code>output</code> where the result
455      * is stored
456      *
457      * @return the number of bytes stored in <code>output</code>
458      *
459      * @exception IllegalStateException if this exemption mechanism is in
460      * a wrong state (e.g., has not been initialized).
461      * @exception ShortBufferException if the given output buffer is too small
462      * to hold the result.
463      * @exception ExemptionMechanismException if problem(s) encountered in the
464      * process of generating.
465      */
genExemptionBlob(byte[] output, int outputOffset)466     public final int genExemptionBlob(byte[] output, int outputOffset)
467             throws IllegalStateException, ShortBufferException,
468             ExemptionMechanismException {
469         if (!initialized) {
470             throw new IllegalStateException
471             ("ExemptionMechanism not initialized");
472         }
473         int n = exmechSpi.engineGenExemptionBlob(output, outputOffset);
474         done = true;
475         return n;
476     }
477 
478     // Android-removed: Unnecessary finalize() method.
479     // OpenJDK 9 also removed the method.
480     /*
481     protected void finalize() {
482         keyStored = null;
483         // Are there anything else we could do?
484     }
485     */
486 }
487