/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* * Copyright (c) 2015-2017, The Linux Foundation. */ /* * Contributed by: Giesecke & Devrient GmbH. */ package android.se.omapi; import android.annotation.NonNull; import android.annotation.Nullable; import android.os.RemoteException; import android.os.ServiceSpecificException; import android.util.Log; import java.io.IOException; /** * Instances of this class represent an ISO/IEC 7816-4 channel opened to a * Secure Element. It can be either a logical channel or the basic channel. They * can be used to send APDUs to the secure element. Channels are opened by * calling the Session.openBasicChannel(byte[]) or * Session.openLogicalChannel(byte[]) methods. * * @see GlobalPlatform Open Mobile API */ public final class Channel implements java.nio.channels.Channel { private static final String TAG = "OMAPI.Channel"; private Session mSession; private final ISecureElementChannel mChannel; private final SEService mService; private final Object mLock = new Object(); Channel(@NonNull SEService service, @NonNull Session session, @NonNull ISecureElementChannel channel) { if (service == null || session == null || channel == null) { throw new IllegalArgumentException("Parameters cannot be null"); } mService = service; mSession = session; mChannel = channel; } /** * Closes this channel to the Secure Element. If the method is called when * the channel is already closed, this method will be ignored. The close() * method shall wait for completion of any pending transmit(byte[] command) * before closing the channel. */ public void close() { if (isOpen()) { synchronized (mLock) { try { mChannel.close(); } catch (Exception e) { Log.e(TAG, "Error closing channel", e); } } } } /** * Tells if this channel is open. * * @return false if the channel is closed or in case of an error. * true otherwise. */ public boolean isOpen() { if (!mService.isConnected()) { Log.e(TAG, "service not connected to system"); return false; } try { return !mChannel.isClosed(); } catch (RemoteException e) { Log.e(TAG, "Exception in isClosed()"); return false; } } /** * Returns a boolean telling if this channel is the basic channel. * * @return true if this channel is a basic channel. false if * this channel is a logical channel. */ public boolean isBasicChannel() { if (!mService.isConnected()) { throw new IllegalStateException("service not connected to system"); } try { return mChannel.isBasicChannel(); } catch (RemoteException e) { throw new IllegalStateException(e.getMessage()); } } /** * Transmit an APDU command (as per ISO/IEC 7816-4) to the Secure Element. The * underlying layers generate as many TPDUs as necessary to transport this APDU. The * API shall ensure that all available data returned from Secure Element, including * concatenated responses, are retrieved and made available to the calling application. If a * warning status code is received the API wont check for further response data but will * return all data received so far and the warning status code.
* The transport part is invisible from the application. The generated response is the * response of the APDU which means that all protocols related responses are handled * inside the API or the underlying implementation.
* The transmit method shall support extended length APDU commands independently of * the coding within the ATR.
* For status word '61 XX' the API or underlying implementation shall issue a GET * RESPONSE command as specified by ISO 7816-4 standard with LE=XX; for the status * word '6C XX', the API or underlying implementation shall reissue the input command * with LE=XX. For other status words, the API (or underlying implementation) shall return * the complete response including data and status word to the device application. The API * (or underlying implementation) shall not handle internally the received status words. The * channel shall not be closed even if the Secure Element answered with an error code. * The system ensures the synchronization between all the concurrent calls to this method, * and that only one APDU will be sent at a time, irrespective of the number of TPDUs that * might be required to transport it to the SE. The entire APDU communication to this SE is * locked to the APDU.
* The channel information in the class byte in the APDU will be ignored. The system will * add any required information to ensure the APDU is transported on this channel. * The only restrictions on the set of commands that can be sent is defined below, the API * implementation shall be able to send all other commands:
* * * @param command the APDU command to be transmitted, as a byte array. * * @return the response received, as a byte array. The returned byte array contains the data * bytes in the following order: * [<first data byte>, ..., <last data byte>, <sw1>, <sw2>] * * @throws IOException if there is a communication problem to the reader or the Secure Element. * @throws IllegalStateException if the channel is used after being closed. * @throws IllegalArgumentException if the command byte array is less than 4 bytes long. * @throws IllegalArgumentException if Lc byte is inconsistent with length of the byte array. * @throws IllegalArgumentException if CLA byte is invalid according to [2] (0xff). * @throws IllegalArgumentException if INS byte is invalid according to [2] (0x6x or 0x9x). * @throws SecurityException if the command is filtered by the security policy. * @throws NullPointerException if command is NULL. */ public @NonNull byte[] transmit(@NonNull byte[] command) throws IOException { if (!mService.isConnected()) { throw new IllegalStateException("service not connected to system"); } synchronized (mLock) { try { byte[] response = mChannel.transmit(command); if (response == null) { throw new IOException("Error in communicating with Secure Element"); } return response; } catch (ServiceSpecificException e) { throw new IOException(e.getMessage()); } catch (RemoteException e) { throw new IllegalStateException(e.getMessage()); } } } /** * Get the session that has opened this channel. * * @return the session object this channel is bound to. */ public @NonNull Session getSession() { return mSession; } /** * Returns the data as received from the application select command inclusively the status word * received at applet selection. * The returned byte array contains the data bytes in the following order: * [<first data byte>, ..., <last data byte>, <sw1>, <sw2>] * @return The data as returned by the application select command inclusively the status word. * Only the status word if the application select command has no returned data. * Returns null if an application select command has not been performed or the selection * response can not be retrieved by the reader implementation. */ public @Nullable byte[] getSelectResponse() { if (!mService.isConnected()) { throw new IllegalStateException("service not connected to system"); } byte[] response; try { response = mChannel.getSelectResponse(); } catch (RemoteException e) { throw new IllegalStateException(e.getMessage()); } if (response != null && response.length == 0) { response = null; } return response; } /** * Performs a selection of the next Applet on this channel that matches to the partial AID * specified in the openBasicChannel(byte[] aid) or openLogicalChannel(byte[] aid) method. * This mechanism can be used by a device application to iterate through all Applets * matching to the same partial AID. * If selectNext() returns true a new Applet was successfully selected on this channel. * If no further Applet exists with matches to the partial AID this method returns false * and the already selected Applet stays selected.
* * Since the API cannot distinguish between a partial and full AID the API shall rely on the * response of the Secure Element for the return value of this method.
* The implementation of the underlying SELECT command within this method shall use * the same values as the corresponding openBasicChannel(byte[] aid) or * openLogicalChannel(byte[] aid) command with the option:
* P2='02' (Next occurrence)
* The select response stored in the Channel object shall be updated with the APDU * response of the SELECT command. * @return true if new Applet was selected on this channel. false the already selected Applet stays selected on this channel. * * @throws IOException if there is a communication problem to the reader or the Secure Element. * @throws IllegalStateException if the channel is used after being closed. * @throws UnsupportedOperationException if this operation is not supported by the card. */ public boolean selectNext() throws IOException { if (!mService.isConnected()) { throw new IllegalStateException("service not connected to system"); } try { synchronized (mLock) { return mChannel.selectNext(); } } catch (ServiceSpecificException e) { throw new IOException(e.getMessage()); } catch (RemoteException e) { throw new IllegalStateException(e.getMessage()); } } }