/***************************************************************************** * Copyright ©2017-2019 Gemalto – a Thales Company. All rights Reserved. * * This copy is 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 or https://www.apache.org/licenses/LICENSE-2.0.html * * 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. ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "se-gto/libse-gto.h" #include "SecureElement.h" #if defined(ST_LIB_32) #define VENDOR_LIB_PATH "/vendor/lib/" #else #define VENDOR_LIB_PATH "/vendor/lib64/" #endif #define VENDOR_LIB_EXT ".so" namespace android { namespace hardware { namespace secure_element { namespace V1_2 { namespace implementation { #ifndef MAX_CHANNELS #define MAX_CHANNELS 0x04 #endif #ifndef BASIC_CHANNEL #define BASIC_CHANNEL 0x00 #endif #ifndef LOG_HAL_LEVEL #define LOG_HAL_LEVEL 4 #endif #ifndef MAX_AID_LEN #define MAX_AID_LEN 16 #endif uint8_t getResponse[5] = {0x00, 0xC0, 0x00, 0x00, 0x00}; static struct se_gto_ctx *ctx; bool debug_log_enabled = false; SecureElement::SecureElement(const char* ese_name){ nbrOpenChannel = 0; ctx = NULL; strncpy(ese_flag_name, ese_name, 4); if (strncmp(ese_flag_name, "eSE2", 4) == 0) { strncpy(config_filename, "/vendor/etc/libse-gto-hal2.conf", 31); } else { strncpy(config_filename, "/vendor/etc/libse-gto-hal.conf", 30); } } int SecureElement::resetSE(){ int n; isBasicChannelOpen = false; nbrOpenChannel = 0; ALOGD("SecureElement:%s se_gto_reset start", __func__); n = se_gto_reset(ctx, atr, sizeof(atr)); if (n >= 0) { atr_size = n; ALOGD("SecureElement:%s received ATR of %d bytes\n", __func__, n); dump_bytes("ATR: ", ':', atr, n, stdout); } else { ALOGE("SecureElement:%s Failed to reset and get ATR: %s\n", __func__, strerror(errno)); } return n; } sp SecureElement::internalClientCallback = nullptr; sp SecureElement::internalClientCallback_v1_1 = nullptr; int SecureElement::initializeSE() { int n; int ret = 0; ALOGD("SecureElement:%s start", __func__); if (checkSeUp) { ALOGD("SecureElement:%s Already initialized", __func__); ALOGD("SecureElement:%s end", __func__); return EXIT_SUCCESS; } if (se_gto_new(&ctx) < 0) { ALOGE("SecureElement:%s se_gto_new FATAL:%s", __func__,strerror(errno)); return EXIT_FAILURE; } se_gto_set_log_level(ctx, 3); openConfigFile(1); if (se_gto_open(ctx) < 0) { ALOGE("SecureElement:%s se_gto_open FATAL:%s", __func__,strerror(errno)); return EXIT_FAILURE; } ret = resetSE(); if (ret < 0 && (strncmp(ese_flag_name, "eSE2", 4) == 0)) { sleep(6); ALOGE("SecureElement:%s retry resetSE", __func__); ret = resetSE(); } if (ret < 0) { se_gto_close(ctx); ctx = NULL; return EXIT_FAILURE; } checkSeUp = true; ALOGD("SecureElement:%s end", __func__); return EXIT_SUCCESS; } Return SecureElement::init(const sp<::android::hardware::secure_element::V1_0::ISecureElementHalCallback>& clientCallback) { ALOGD("SecureElement:%s start", __func__); if (clientCallback == nullptr) { ALOGE("SecureElement:%s clientCallback == nullptr", __func__); return Void(); } else { internalClientCallback = clientCallback; internalClientCallback_v1_1 = nullptr; if (!internalClientCallback->linkToDeath(this, 0)) { ALOGE("SecureElement:%s: linkToDeath Failed", __func__); } } if (initializeSE() != EXIT_SUCCESS) { ALOGE("SecureElement:%s initializeSE Failed", __func__); auto ret = clientCallback->onStateChange(false); if (!ret.isOk()) { ALOGW("failed to send onStateChange event!"); } } else { ALOGD("SecureElement:%s initializeSE Success", __func__); auto ret = clientCallback->onStateChange(true); if (!ret.isOk()) { ALOGW("failed to send onStateChange event!"); } } ALOGD("SecureElement:%s end", __func__); return Void(); } Return SecureElement::init_1_1(const sp<::android::hardware::secure_element::V1_1::ISecureElementHalCallback>& clientCallback) { ALOGD("SecureElement:%s start", __func__); if (clientCallback == nullptr) { ALOGE("SecureElement:%s clientCallback == nullptr", __func__); return Void(); } else { internalClientCallback = nullptr; internalClientCallback_v1_1 = clientCallback; if (!internalClientCallback_v1_1->linkToDeath(this, 0)) { ALOGE("SecureElement:%s: linkToDeath Failed", __func__); } } if (initializeSE() != EXIT_SUCCESS) { ALOGE("SecureElement:%s initializeSE Failed", __func__); auto ret = clientCallback->onStateChange_1_1(false, "SE Initialized failed"); if (!ret.isOk()) { ALOGW("failed to send onStateChange event!"); } } else { ALOGD("SecureElement:%s initializeSE Success", __func__); auto ret = clientCallback->onStateChange_1_1(true, "SE Initialized"); if (!ret.isOk()) { ALOGW("failed to send onStateChange event!"); } } ALOGD("SecureElement:%s end", __func__); return Void(); } Return SecureElement::getAtr(getAtr_cb _hidl_cb) { hidl_vec response; response.resize(atr_size); memcpy(&response[0], atr, atr_size); _hidl_cb(response); return Void(); } Return SecureElement::isCardPresent() { return true; } Return SecureElement::transmit(const hidl_vec& data, transmit_cb _hidl_cb) { uint8_t *apdu; uint8_t *resp; int status = 0 ; int apdu_len = data.size(); int resp_len = 0; apdu = (uint8_t*)malloc(apdu_len * sizeof(uint8_t)); resp = (uint8_t*)malloc(65536 * sizeof(uint8_t)); hidl_vec result; if (checkSeUp && nbrOpenChannel != 0) { if (apdu != NULL) { memcpy(apdu, data.data(), data.size()); dump_bytes("CMD: ", ':', apdu, apdu_len, stdout); resp_len = se_gto_apdu_transmit(ctx, apdu, apdu_len, resp, 65536); } if (resp_len < 0) { ALOGE("SecureElement:%s: transmit failed", __func__); if (deinitializeSE() != SecureElementStatus::SUCCESS) { ALOGE("SecureElement:%s deinitializeSE Failed", __func__); } } else { dump_bytes("RESP: ", ':', resp, resp_len, stdout); result.resize(resp_len); memcpy(&result[0], resp, resp_len); } } else { ALOGE("SecureElement:%s: transmit failed! No channel is open", __func__); } _hidl_cb(result); if(apdu) free(apdu); if(resp) free(resp); return Void(); } Return SecureElement::openLogicalChannel(const hidl_vec& aid, uint8_t p2, openLogicalChannel_cb _hidl_cb) { ALOGD("SecureElement:%s start", __func__); LogicalChannelResponse resApduBuff; resApduBuff.channelNumber = 0xff; memset(&resApduBuff, 0x00, sizeof(resApduBuff)); if (!checkSeUp) { if (initializeSE() != EXIT_SUCCESS) { ALOGE("SecureElement:%s: Failed to re-initialise the eSE HAL", __func__); notify(false, "SE Initialized failed"); _hidl_cb(resApduBuff, SecureElementStatus::IOERROR); return Void(); } } SecureElementStatus mSecureElementStatus = SecureElementStatus::IOERROR; if (aid.size() > MAX_AID_LEN) { ALOGE("SecureElement:%s: Bad AID size", __func__); _hidl_cb(resApduBuff, SecureElementStatus::FAILED); return Void(); } uint8_t *apdu; //65536 int apdu_len = 0; uint8_t *resp; int resp_len = 0; uint8_t index = 0; int getResponseOffset = 0; apdu_len = 5; apdu = (uint8_t*)malloc(apdu_len * sizeof(uint8_t)); resp = (uint8_t*)malloc(65536 * sizeof(uint8_t)); if (apdu != NULL && resp!=NULL) { index = 0; apdu[index++] = 0x00; apdu[index++] = 0x70; apdu[index++] = 0x00; apdu[index++] = 0x00; apdu[index++] = 0x01; dump_bytes("CMD: ", ':', apdu, apdu_len, stdout); resp_len = se_gto_apdu_transmit(ctx, apdu, apdu_len, resp, 65536); ALOGD("SecureElement:%s Manage channel resp_len = %d", __func__,resp_len); } if (resp_len >= 0) dump_bytes("RESP: ", ':', resp, resp_len, stdout); if (resp_len < 0) { if (deinitializeSE() != SecureElementStatus::SUCCESS) { ALOGE("SecureElement:%s deinitializeSE Failed", __func__); } mSecureElementStatus = SecureElementStatus::IOERROR; ALOGD("SecureElement:%s Free memory after manage channel after ERROR", __func__); if(apdu) free(apdu); if(resp) free(resp); _hidl_cb(resApduBuff, mSecureElementStatus); return Void(); } else if (resp[resp_len - 2] == 0x90 && resp[resp_len - 1] == 0x00) { resApduBuff.channelNumber = resp[0]; nbrOpenChannel++; mSecureElementStatus = SecureElementStatus::SUCCESS; } else { if (resp[resp_len - 2] == 0x6A && resp[resp_len - 1] == 0x81) { mSecureElementStatus = SecureElementStatus::CHANNEL_NOT_AVAILABLE; }else if (resp[resp_len - 2] == 0x68 && resp[resp_len - 1] == 0x81) { mSecureElementStatus = SecureElementStatus::CHANNEL_NOT_AVAILABLE; } else { mSecureElementStatus = SecureElementStatus::IOERROR; } ALOGD("SecureElement:%s Free memory after manage channel after ERROR", __func__); if(apdu) free(apdu); if(resp) free(resp); _hidl_cb(resApduBuff, mSecureElementStatus); return Void(); } if(apdu) free(apdu); if(resp) free(resp); ALOGD("SecureElement:%s Free memory after manage channel", __func__); ALOGD("SecureElement:%s mSecureElementStatus = %d", __func__, (int)mSecureElementStatus); /*Start Sending select command after Manage Channel is successful.*/ ALOGD("SecureElement:%s Sending selectApdu", __func__); mSecureElementStatus = SecureElementStatus::IOERROR; apdu_len = (int32_t)(6 + aid.size()); resp_len = 0; apdu = (uint8_t*)malloc(apdu_len * sizeof(uint8_t)); resp = (uint8_t*)malloc(65536 * sizeof(uint8_t)); if (apdu != NULL && resp!=NULL) { index = 0; apdu[index++] = resApduBuff.channelNumber; apdu[index++] = 0xA4; apdu[index++] = 0x04; apdu[index++] = p2; apdu[index++] = aid.size(); memcpy(&apdu[index], aid.data(), aid.size()); index += aid.size(); apdu[index] = 0x00; send_logical: dump_bytes("CMD: ", ':', apdu, apdu_len, stdout); resp_len = se_gto_apdu_transmit(ctx, apdu, apdu_len, resp, 65536); ALOGD("SecureElement:%s selectApdu resp_len = %d", __func__,resp_len); } if (resp_len < 0) { ALOGE("SecureElement:%s selectApdu resp_len = %d", __func__,resp_len); if (deinitializeSE() != SecureElementStatus::SUCCESS) { ALOGE("SecureElement:%s deinitializeSE Failed", __func__); } mSecureElementStatus = SecureElementStatus::IOERROR; } else { dump_bytes("RESP: ", ':', resp, resp_len, stdout); if (resp[resp_len - 2] == 0x90 || resp[resp_len - 2] == 0x62 || resp[resp_len - 2] == 0x63) { resApduBuff.selectResponse.resize(getResponseOffset + resp_len); memcpy(&resApduBuff.selectResponse[getResponseOffset], resp, resp_len); mSecureElementStatus = SecureElementStatus::SUCCESS; } else if (resp[resp_len - 2] == 0x61) { resApduBuff.selectResponse.resize(getResponseOffset + resp_len - 2); memcpy(&resApduBuff.selectResponse[getResponseOffset], resp, resp_len - 2); getResponseOffset += (resp_len - 2); getResponse[4] = resp[resp_len - 1]; getResponse[0] = apdu[0]; dump_bytes("getResponse CMD: ", ':', getResponse, 5, stdout); free(apdu); apdu_len = 5; apdu = (uint8_t*)malloc(apdu_len * sizeof(uint8_t)); memset(resp, 0, resp_len); memcpy(apdu, getResponse, apdu_len); apdu[0] = resApduBuff.channelNumber; goto send_logical; } else if (resp[resp_len - 2] == 0x6C) { resApduBuff.selectResponse.resize(getResponseOffset + resp_len - 2); memcpy(&resApduBuff.selectResponse[getResponseOffset], resp, resp_len - 2); getResponseOffset += (resp_len - 2); apdu[4] = resp[resp_len - 1]; dump_bytes("case2 getResponse CMD: ", ':', apdu, 5, stdout); memset(resp, 0, resp_len); goto send_logical; } else if (resp[resp_len - 2] == 0x6A && resp[resp_len - 1] == 0x80) { mSecureElementStatus = SecureElementStatus::IOERROR; } else if (resp[resp_len - 2] == 0x6A && resp[resp_len - 1] == 0x81) { mSecureElementStatus = SecureElementStatus::IOERROR; } else if (resp[resp_len - 2] == 0x6A && resp[resp_len - 1] == 0x82) { mSecureElementStatus = SecureElementStatus::NO_SUCH_ELEMENT_ERROR; } else if (resp[resp_len - 2] == 0x6A && resp[resp_len - 1] == 0x86) { mSecureElementStatus = SecureElementStatus::UNSUPPORTED_OPERATION; } else if (resp[resp_len - 2] == 0x6A && resp[resp_len - 1] == 0x87) { mSecureElementStatus = SecureElementStatus::UNSUPPORTED_OPERATION; } } /*Check if SELECT command failed, close oppened channel*/ if (mSecureElementStatus != SecureElementStatus::SUCCESS) { closeChannel(resApduBuff.channelNumber); } ALOGD("SecureElement:%s mSecureElementStatus = %d", __func__, (int)mSecureElementStatus); _hidl_cb(resApduBuff, mSecureElementStatus); ALOGD("SecureElement:%s Free memory after selectApdu", __func__); if(apdu) free(apdu); if(resp) free(resp); return Void(); } Return SecureElement::openBasicChannel(const hidl_vec& aid, uint8_t p2, openBasicChannel_cb _hidl_cb) { hidl_vec result; SecureElementStatus mSecureElementStatus = SecureElementStatus::IOERROR; uint8_t *apdu; //65536 int apdu_len = 0; uint8_t *resp; int resp_len = 0; int getResponseOffset = 0; uint8_t index = 0; if (isBasicChannelOpen) { ALOGE("SecureElement:%s: Basic Channel already open", __func__); _hidl_cb(result, SecureElementStatus::CHANNEL_NOT_AVAILABLE); return Void(); } if (!checkSeUp) { if (initializeSE() != EXIT_SUCCESS) { ALOGE("SecureElement:%s: Failed to re-initialise the eSE HAL", __func__); notify(false, "SE Initialized failed"); _hidl_cb(result, SecureElementStatus::IOERROR); return Void(); } } if (aid.size() > MAX_AID_LEN) { ALOGE("SecureElement:%s: Bad AID size", __func__); _hidl_cb(result, SecureElementStatus::FAILED); return Void(); } apdu_len = (int32_t)(6 + aid.size()); resp_len = 0; apdu = (uint8_t*)malloc(apdu_len * sizeof(uint8_t)); resp = (uint8_t*)malloc(65536 * sizeof(uint8_t)); if (apdu != NULL) { index = 0; apdu[index++] = 0x00; apdu[index++] = 0xA4; apdu[index++] = 0x04; apdu[index++] = p2; apdu[index++] = aid.size(); memcpy(&apdu[index], aid.data(), aid.size()); index += aid.size(); apdu[index] = 0x00; send_basic: dump_bytes("CMD: ", ':', apdu, apdu_len, stdout); resp_len = se_gto_apdu_transmit(ctx, apdu, apdu_len, resp, 65536); ALOGD("SecureElement:%s selectApdu resp_len = %d", __func__,resp_len); } if (resp_len < 0) { if (deinitializeSE() != SecureElementStatus::SUCCESS) { ALOGE("SecureElement:%s deinitializeSE Failed", __func__); } mSecureElementStatus = SecureElementStatus::IOERROR; } else { dump_bytes("RESP: ", ':', resp, resp_len, stdout); if (resp[resp_len - 2] == 0x90 || resp[resp_len - 2] == 0x62 || resp[resp_len - 2] == 0x63) { result.resize(getResponseOffset + resp_len); memcpy(&result[getResponseOffset], resp, resp_len); isBasicChannelOpen = true; nbrOpenChannel++; mSecureElementStatus = SecureElementStatus::SUCCESS; } else if (resp[resp_len - 2] == 0x61) { result.resize(getResponseOffset + resp_len - 2); memcpy(&result[getResponseOffset], resp, resp_len - 2); getResponseOffset += (resp_len - 2); getResponse[4] = resp[resp_len - 1]; getResponse[0] = apdu[0]; dump_bytes("getResponse CMD: ", ':', getResponse, 5, stdout); free(apdu); apdu_len = 5; apdu = (uint8_t*)malloc(apdu_len * sizeof(uint8_t)); memset(resp, 0, resp_len); memcpy(apdu, getResponse, apdu_len); goto send_basic; } else if (resp[resp_len - 2] == 0x6C) { result.resize(getResponseOffset + resp_len - 2); memcpy(&result[getResponseOffset], resp, resp_len - 2); getResponseOffset += (resp_len - 2); apdu[4] = resp[resp_len - 1]; dump_bytes("case2 getResponse CMD: ", ':', apdu, 5, stdout); apdu_len = 5; memset(resp, 0, resp_len); goto send_basic; } else if (resp[resp_len - 2] == 0x68 && resp[resp_len - 1] == 0x81) { mSecureElementStatus = SecureElementStatus::CHANNEL_NOT_AVAILABLE; } else if (resp[resp_len - 2] == 0x6A && resp[resp_len - 1] == 0x81) { mSecureElementStatus = SecureElementStatus::CHANNEL_NOT_AVAILABLE; } } if ((mSecureElementStatus != SecureElementStatus::SUCCESS) && isBasicChannelOpen) { closeChannel(BASIC_CHANNEL); } ALOGD("SecureElement:%s mSecureElementStatus = %d", __func__, (int)mSecureElementStatus); _hidl_cb(result, mSecureElementStatus); ALOGD("SecureElement:%s Free memory after openBasicChannel", __func__); if(apdu) free(apdu); if(resp) free(resp); return Void(); } Return<::android::hardware::secure_element::V1_0::SecureElementStatus> SecureElement::closeChannel(uint8_t channelNumber) { ALOGD("SecureElement:%s start", __func__); SecureElementStatus mSecureElementStatus = SecureElementStatus::FAILED; uint8_t *apdu; //65536 int apdu_len = 10; uint8_t *resp; int resp_len = 0; if (!checkSeUp) { ALOGE("SecureElement:%s cannot closeChannel, HAL is deinitialized", __func__); mSecureElementStatus = SecureElementStatus::FAILED; ALOGD("SecureElement:%s end", __func__); return mSecureElementStatus; } if ((channelNumber < 0) || (channelNumber >= MAX_CHANNELS)) { ALOGE("SecureElement:%s Channel not supported", __func__); mSecureElementStatus = SecureElementStatus::FAILED; } else if (channelNumber == 0) { isBasicChannelOpen = false; mSecureElementStatus = SecureElementStatus::SUCCESS; nbrOpenChannel--; } else { apdu = (uint8_t*)malloc(apdu_len * sizeof(uint8_t)); resp = (uint8_t*)malloc(65536 * sizeof(uint8_t)); if (apdu != NULL) { uint8_t index = 0; apdu[index++] = channelNumber; apdu[index++] = 0x70; apdu[index++] = 0x80; apdu[index++] = channelNumber; apdu[index++] = 0x00; apdu_len = index; resp_len = se_gto_apdu_transmit(ctx, apdu, apdu_len, resp, 65536); } if (resp_len < 0) { mSecureElementStatus = SecureElementStatus::FAILED; if (deinitializeSE() != SecureElementStatus::SUCCESS) { ALOGE("SecureElement:%s deinitializeSE Failed", __func__); } } else if ((resp[resp_len - 2] == 0x90) && (resp[resp_len - 1] == 0x00)) { mSecureElementStatus = SecureElementStatus::SUCCESS; nbrOpenChannel--; } else { mSecureElementStatus = SecureElementStatus::FAILED; } if(apdu) free(apdu); if(resp) free(resp); } if (nbrOpenChannel == 0 && isBasicChannelOpen == false) { ALOGD("SecureElement:%s All Channels are closed", __func__); if (deinitializeSE() != SecureElementStatus::SUCCESS) { ALOGE("SecureElement:%s deinitializeSE Failed", __func__); } } ALOGD("SecureElement:%s end", __func__); return mSecureElementStatus; } void SecureElement::notify(bool state, const char *message) { if (internalClientCallback_v1_1 != nullptr) { auto ret = internalClientCallback_v1_1->onStateChange_1_1(state, message); if (!ret.isOk()) { ALOGW("failed to send onStateChange event!"); } } else { auto ret = internalClientCallback->onStateChange(state); if (!ret.isOk()) { ALOGW("failed to send onStateChange event!"); } } } void SecureElement::dump_bytes(const char *pf, char sep, const uint8_t *p, int n, FILE *out) { const uint8_t *s = p; char *msg; int len = 0; int input_len = n; if (!debug_log_enabled) return; msg = (char*) malloc ( (pf ? strlen(pf) : 0) + input_len * 3 + 1); if(!msg) { errno = ENOMEM; return; } if (pf) { len += sprintf(msg , "%s" , pf); } while (input_len--) { len += sprintf(msg + len, "%02X" , *s++); if (input_len && sep) { len += sprintf(msg + len, ":"); } } sprintf(msg + len, "\n"); ALOGD("SecureElement:%s ==> size = %d data = %s", __func__, n, msg); if(msg) free(msg); } int SecureElement::toint(char c) { if ((c >= '0') && (c <= '9')) return c - '0'; if ((c >= 'A') && (c <= 'F')) return c - 'A' + 10; if ((c >= 'a') && (c <= 'f')) return c - 'a' + 10; return 0; } int SecureElement::run_apdu(struct se_gto_ctx *ctx, const uint8_t *apdu, uint8_t *resp, int n, int verbose) { int sw; if (verbose) dump_bytes("APDU: ", ':', apdu, n, stdout); n = se_gto_apdu_transmit(ctx, apdu, n, resp, sizeof(resp)); if (n < 0) { ALOGE("SecureElement:%s FAILED: APDU transmit (%s).\n\n", __func__, strerror(errno)); return -2; } else if (n < 2) { dump_bytes("RESP: ", ':', resp, n, stdout); ALOGE("SecureElement:%s FAILED: not enough data to have a status word.\n", __func__); return -2; } if (verbose) { sw = (resp[n - 2] << 8) | resp[n - 1]; printf("%d bytes, SW=0x%04x\n", n - 2, sw); if (n > 2) dump_bytes("RESP: ", ':', resp, n - 2, stdout); } return 0; } int SecureElement::parseConfigFile(FILE *f, int verbose) { static char buf[65536 * 2 + 2]; int line; char * pch; line = 0; while (feof(f) == 0) { char *s; s = fgets(buf, sizeof buf, f); if (s == NULL) break; if (s[0] == '#') { continue; } pch = strtok(s," =;"); if (strcmp("GTO_DEV", pch) == 0) { pch = strtok (NULL, " =;"); ALOGD("SecureElement:%s Defined node : %s", __func__, pch); if (strlen(pch) > 0 && strcmp("\n", pch) != 0 && strcmp("\0", pch) != 0 ) { se_gto_set_gtodev(ctx, pch); } } else if (strcmp("GTO_DEBUG", pch) == 0) { pch = strtok(NULL, " =;"); ALOGD("SecureElement:%s Log state : %s", __func__, pch); if (strlen(pch) > 0 && strcmp("\n", pch) != 0 && strcmp("\0", pch) != 0 ) { if (strcmp(pch, "enable") == 0) { debug_log_enabled = true; se_gto_set_log_level(ctx, 4); } else { debug_log_enabled = false; se_gto_set_log_level(ctx, 3); } } } } return 0; } int SecureElement::openConfigFile(int verbose) { int r; FILE *f; /* filename is not NULL */ ALOGD("SecureElement:%s Open Config file : %s", __func__, config_filename); f = fopen(config_filename, "r"); if (f) { r = parseConfigFile(f, verbose); if (r == -1) { perror(config_filename); ALOGE("SecureElement:%s Error parse %s Failed", __func__, config_filename); } if (fclose(f) != 0) { r = -1; ALOGE("SecureElement:%s Error close %s Failed", __func__, config_filename); } } else { r = -1; ALOGE("SecureElement:%s Error open %s Failed", __func__, config_filename); } return r; } void SecureElement::serviceDied(uint64_t, const wp&) { ALOGD("SecureElement:%s SecureElement serviceDied", __func__); SecureElementStatus mSecureElementStatus = deinitializeSE(); if (mSecureElementStatus != SecureElementStatus::SUCCESS) { ALOGE("SecureElement:%s deinitializeSE Failed", __func__); } if (internalClientCallback != nullptr) { internalClientCallback->unlinkToDeath(this); internalClientCallback = nullptr; } if (internalClientCallback_v1_1 != nullptr) { internalClientCallback_v1_1->unlinkToDeath(this); internalClientCallback_v1_1 = nullptr; } } Return<::android::hardware::secure_element::V1_0::SecureElementStatus> SecureElement::deinitializeSE() { SecureElementStatus mSecureElementStatus = SecureElementStatus::FAILED; ALOGD("SecureElement:%s start", __func__); if(checkSeUp){ if (se_gto_close(ctx) < 0) { mSecureElementStatus = SecureElementStatus::FAILED; notify(false, "SE Initialized failed"); } else { ctx = NULL; mSecureElementStatus = SecureElementStatus::SUCCESS; isBasicChannelOpen = false; nbrOpenChannel = 0; } checkSeUp = false; }else{ ALOGD("SecureElement:%s No need to deinitialize SE", __func__); mSecureElementStatus = SecureElementStatus::SUCCESS; } ALOGD("SecureElement:%s end", __func__); return mSecureElementStatus; } Return<::android::hardware::secure_element::V1_0::SecureElementStatus> SecureElement::reset() { SecureElementStatus status = SecureElementStatus::FAILED; std::string eSE1ResetToolStr; int ret = 0; ALOGD("SecureElement:%s start", __func__); if (strncmp(ese_flag_name, "eSE1", 4) == 0) { eSE1ResetToolStr = android::base::GetProperty("persist.vendor.se.streset", ""); } if (deinitializeSE() != SecureElementStatus::SUCCESS) { ALOGE("SecureElement:%s deinitializeSE Failed", __func__); } notify(false, "reset the SE"); if (eSE1ResetToolStr.length() > 0) { typedef int (*STEseReset)(); eSE1ResetToolStr = VENDOR_LIB_PATH + eSE1ResetToolStr + VENDOR_LIB_EXT; void *stdll = dlopen(eSE1ResetToolStr.c_str(), RTLD_NOW); STEseReset fn = (STEseReset)dlsym(stdll, "direct_reset"); ret = fn(); ALOGD("SecureElement:%s STResetTool ret : %d", __func__, ret); if (ret == 0) { ALOGD("SecureElement:%s STResetTool Success", __func__); if (initializeSE() == EXIT_SUCCESS) { status = SecureElementStatus::SUCCESS; notify(true, "SE Initialized"); } } else { ALOGE("SecureElement:%s STResetTool Failed!", __func__); } dlclose(stdll); } else { if (initializeSE() == EXIT_SUCCESS) { notify(true, "SE Initialized"); status = SecureElementStatus::SUCCESS; } } ALOGD("SecureElement:%s end", __func__); return status; } } // namespace implementation } // namespace V1_2 } // namespace secure_element } // namespace hardware } // namespace android