| /****************************************************************************** |
| * |
| * Copyright (C) 1999-2012 Broadcom Corporation |
| * |
| * 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. |
| * |
| ******************************************************************************/ |
| #include "OverrideLog.h" |
| #define LOG_TAG "NfcNciHal" |
| #include "gki.h" |
| extern "C" |
| { |
| #include "nfc_hal_target.h" |
| #include "nfc_hal_nv_ci.h" |
| } |
| #include "config.h" |
| #include "CrcChecksum.h" |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| #include <fcntl.h> |
| #include <errno.h> |
| #include <string> |
| |
| |
| //directory of HAL's non-volatile storage |
| static const char* default_location = "/data/nfc"; |
| static const char* filename_prefix = "/halStorage.bin"; |
| static const std::string get_storage_location (); |
| void delete_hal_non_volatile_store (bool forceDelete); |
| void verify_hal_non_volatile_store (); |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function nfc_hal_nv_co_read |
| ** |
| ** Description This function is called by NFA to read in data from the |
| ** previously opened file. |
| ** |
| ** Parameters p_buf - buffer to read the data into. |
| ** nbytes - number of bytes to read into the buffer. |
| ** |
| ** Returns void |
| ** |
| ** Note: Upon completion of the request, nfc_hal_nv_ci_read () is |
| ** called with the buffer of data, along with the number |
| ** of bytes read into the buffer, and a status. The |
| ** call-in function should only be called when ALL requested |
| ** bytes have been read, the end of file has been detected, |
| ** or an error has occurred. |
| ** |
| *******************************************************************************/ |
| void nfc_hal_nv_co_read (UINT8 *p_buf, UINT16 nbytes, UINT8 block) |
| { |
| std::string fn = get_storage_location(); |
| char filename[256]; |
| |
| fn.append (filename_prefix); |
| if (fn.length() > 200) |
| { |
| ALOGE ("%s: filename too long", __FUNCTION__); |
| return; |
| } |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), block); |
| |
| ALOGD ("%s: buffer len=%u; file=%s", __FUNCTION__, nbytes, filename); |
| int fileStream = open (filename, O_RDONLY); |
| if (fileStream >= 0) |
| { |
| unsigned short checksum = 0; |
| size_t actualReadCrc = read (fileStream, &checksum, sizeof(checksum)); |
| size_t actualReadData = read (fileStream, p_buf, nbytes); |
| close (fileStream); |
| if (actualReadData > 0) |
| { |
| ALOGD ("%s: data size=%u", __FUNCTION__, actualReadData); |
| nfc_hal_nv_ci_read (actualReadData, NFC_HAL_NV_CO_OK, block); |
| } |
| else |
| { |
| ALOGE ("%s: fail to read", __FUNCTION__); |
| nfc_hal_nv_ci_read (0, NFC_HAL_NV_CO_FAIL, block); |
| } |
| } |
| else |
| { |
| ALOGD ("%s: fail to open", __FUNCTION__); |
| nfc_hal_nv_ci_read (0, NFC_HAL_NV_CO_FAIL, block); |
| } |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function nfc_hal_nv_co_write |
| ** |
| ** Description This function is called by io to send file data to the |
| ** phone. |
| ** |
| ** Parameters p_buf - buffer to read the data from. |
| ** nbytes - number of bytes to write out to the file. |
| ** |
| ** Returns void |
| ** |
| ** Note: Upon completion of the request, nfc_hal_nv_ci_write () is |
| ** called with the file descriptor and the status. The |
| ** call-in function should only be called when ALL requested |
| ** bytes have been written, or an error has been detected, |
| ** |
| *******************************************************************************/ |
| void nfc_hal_nv_co_write (const UINT8 *p_buf, UINT16 nbytes, UINT8 block) |
| { |
| std::string fn = get_storage_location(); |
| char filename[256]; |
| int fileStream = 0; |
| |
| fn.append (filename_prefix); |
| if (fn.length() > 200) |
| { |
| ALOGE ("%s: filename too long", __FUNCTION__); |
| return; |
| } |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), block); |
| ALOGD ("%s: bytes=%u; file=%s", __FUNCTION__, nbytes, filename); |
| |
| fileStream = open (filename, O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR); |
| if (fileStream >= 0) |
| { |
| unsigned short checksum = crcChecksumCompute (p_buf, nbytes); |
| size_t actualWrittenCrc = write (fileStream, &checksum, sizeof(checksum)); |
| size_t actualWrittenData = write (fileStream, p_buf, nbytes); |
| ALOGD ("%s: %d bytes written", __FUNCTION__, actualWrittenData); |
| if ((actualWrittenData == nbytes) && (actualWrittenCrc == sizeof(checksum))) |
| { |
| nfc_hal_nv_ci_write (NFC_HAL_NV_CO_OK); |
| } |
| else |
| { |
| ALOGE ("%s: fail to write", __FUNCTION__); |
| nfc_hal_nv_ci_write (NFC_HAL_NV_CO_FAIL); |
| } |
| close (fileStream); |
| } |
| else |
| { |
| ALOGE ("%s: fail to open, error = %d", __FUNCTION__, errno); |
| nfc_hal_nv_ci_write (NFC_HAL_NV_CO_FAIL); |
| } |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function get_storage_location |
| ** |
| ** Description Get the absolute directory path of the HAL's storage location. |
| ** |
| ** Parameters none |
| ** |
| ** Returns Absolute path. |
| ** |
| *******************************************************************************/ |
| const std::string get_storage_location () |
| { |
| char buffer [100]; |
| memset (buffer, 0, sizeof(buffer)); |
| if (!GetStrValue (NAME_NFA_STORAGE, buffer, sizeof(buffer))) |
| return default_location; |
| else |
| return std::string (buffer); |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function delete_hal_non_volatile_store |
| ** |
| ** Description Delete all the content of the HAL's storage location. |
| ** |
| ** Parameters forceDelete: unconditionally delete the storage. |
| ** |
| ** Returns none |
| ** |
| *******************************************************************************/ |
| void delete_hal_non_volatile_store (bool forceDelete) |
| { |
| static bool firstTime = true; |
| std::string fn = get_storage_location(); |
| char filename[256]; |
| int stat = 0; |
| |
| if ((firstTime == false) && (forceDelete == false)) |
| return; |
| firstTime = false; |
| |
| ALOGD ("%s", __FUNCTION__); |
| |
| fn.append (filename_prefix); |
| if (fn.length() > 200) |
| { |
| ALOGE ("%s: filename too long", __FUNCTION__); |
| return; |
| } |
| |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), DH_NV_BLOCK); |
| remove (filename); |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), HC_F3_NV_BLOCK); |
| remove (filename); |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), HC_F4_NV_BLOCK); |
| remove (filename); |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), HC_F2_NV_BLOCK); |
| remove (filename); |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), HC_F5_NV_BLOCK); |
| remove (filename); |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function verify_hal_non_volatile_store |
| ** |
| ** Description Verify the content of all non-volatile store. |
| ** |
| ** Parameters none |
| ** |
| ** Returns none |
| ** |
| *******************************************************************************/ |
| void verify_hal_non_volatile_store () |
| { |
| ALOGD ("%s", __FUNCTION__); |
| std::string fn = get_storage_location(); |
| char filename[256]; |
| bool isValid = false; |
| |
| fn.append (filename_prefix); |
| if (fn.length() > 200) |
| { |
| ALOGE ("%s: filename too long", __FUNCTION__); |
| return; |
| } |
| |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), DH_NV_BLOCK); |
| if (crcChecksumVerifyIntegrity (filename)) |
| { |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), HC_F3_NV_BLOCK); |
| if (crcChecksumVerifyIntegrity (filename)) |
| { |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), HC_F4_NV_BLOCK); |
| if (crcChecksumVerifyIntegrity (filename)) |
| { |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), HC_F2_NV_BLOCK); |
| if (crcChecksumVerifyIntegrity (filename)) |
| { |
| snprintf (filename, sizeof(filename), "%s%u", fn.c_str(), HC_F5_NV_BLOCK); |
| if (crcChecksumVerifyIntegrity (filename)) |
| isValid = true; |
| } |
| } |
| } |
| } |
| |
| if (isValid == false) |
| delete_hal_non_volatile_store (true); |
| } |