All Downloads are FREE. Search and download functionalities are using the official Maven repository.

android.se.omapi.Channel Maven / Gradle / Ivy

Go to download

A library jar that provides APIs for Applications written for the Google Android Platform.

There is a newer version: 14-robolectric-10818077
Show newest version
/*
 * 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:
*
    *
  • MANAGE_CHANNEL commands are not allowed.
  • *
  • SELECT by DF Name (p1=04) are not allowed.
  • *
  • CLA bytes with channel numbers are de-masked.
  • *
* * @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 he 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()); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy