/*
* 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.
*/
#ifndef APDU_H_
#define APDU_H_
#include <cstddef>
#include <cstdint>
#include <iterator>
#include <vector>
namespace android {
/**
* Helper to build an APDU command. If a data section is needed, it is left empty with dataBegin
* and dataEnd able to return iterators to where the data should be filled in.
*
* The command bytes are stored sequentially in the same manner as std::vector.
*/
class CommandApdu {
public:
CommandApdu(uint8_t cla, uint8_t ins, uint8_t p1, uint8_t p2)
: CommandApdu(cla, ins, p1, p2, 0, 0) {}
CommandApdu(uint8_t cla, uint8_t ins, uint8_t p1, uint8_t p2, size_t lc, size_t le);
using iterator = std::vector<uint8_t>::iterator;
using const_iterator = std::vector<uint8_t>::const_iterator;
iterator begin() { return mCommand.begin(); }
iterator end() { return mCommand.end(); }
const_iterator begin() const { return mCommand.begin(); }
const_iterator end() const { return mCommand.end(); }
iterator dataBegin() { return mDataBegin; }
iterator dataEnd() { return mDataEnd; }
const_iterator dataBegin() const { return mDataBegin; }
const_iterator dataEnd() const { return mDataEnd; }
size_t size() const { return mCommand.size(); }
size_t dataSize() const { return std::distance(mDataBegin, mDataEnd); }
const std::vector<uint8_t>& vector() const { return mCommand; }
private:
std::vector<uint8_t> mCommand;
std::vector<uint8_t>::iterator mDataBegin;
std::vector<uint8_t>::iterator mDataEnd;
};
/**
* Helper to deconstruct a response APDU. This wraps a reference to an iterable byte container.
*/
template<typename T>
class ResponseApdu {
static constexpr size_t STATUS_SIZE = 2;
static constexpr uint8_t BYTES_AVAILABLE = 0x61;
static constexpr uint8_t SW1_WARNING_NON_VOLATILE_MEMORY_UNCHANGED = 0x62;
static constexpr uint8_t SW1_WARNING_NON_VOLATILE_MEMORY_CHANGED = 0x63;
static constexpr uint8_t SW1_FIRST_EXECUTION_ERROR = 0x64;
static constexpr uint8_t SW1_LAST_EXECUTION_ERROR = 0x66;
static constexpr uint8_t SW1_FIRST_CHECKING_ERROR = 0x67;
static constexpr uint8_t SW1_LAST_CHECKING_ERROR = 0x6f;
public:
ResponseApdu(const T& data) : mData(data) {}
bool ok() const {
return static_cast<size_t>(
std::distance(std::begin(mData), std::end(mData))) >= STATUS_SIZE;
}
uint8_t sw1() const { return *(std::end(mData) - 2); }
uint8_t sw2() const { return *(std::end(mData) - 1); }
uint16_t status() const { return (static_cast<uint16_t>(sw1()) << 8) | sw2(); }
int8_t remainingBytes() const { return sw1() == BYTES_AVAILABLE ? sw2() : 0; }
bool isWarning() const {
const uint8_t sw1 = this->sw1();
return sw1 == SW1_WARNING_NON_VOLATILE_MEMORY_UNCHANGED
|| sw1 == SW1_WARNING_NON_VOLATILE_MEMORY_CHANGED;
}
bool isExecutionError() const {
const uint8_t sw1 = this->sw1();
return sw1 >= SW1_FIRST_EXECUTION_ERROR && sw1 <= SW1_LAST_EXECUTION_ERROR;
}
bool isCheckingError() const {
const uint8_t sw1 = this->sw1();
return sw1 >= SW1_FIRST_CHECKING_ERROR && sw1 <= SW1_LAST_CHECKING_ERROR;
}
bool isError() const { return isExecutionError() || isCheckingError(); }
auto dataBegin() const { return std::begin(mData); }
auto dataEnd() const { return std::end(mData) - STATUS_SIZE; }
size_t dataSize() const { return std::distance(dataBegin(), dataEnd()); }
private:
const T& mData;
};
} // namespace android
#endif // APDU_H_