blob: d99d3b41b785b4de1c188543b420cccbaf22435b [file] [log] [blame]
/*
* Copyright (c) 2017, The OpenThread Authors.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of the copyright holder nor the
* names of its contributors may be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#include <stdint.h>
#include <string.h>
#include <assert.h>
#include <openthread/types.h>
#include <openthread/platform/alarm.h>
#include <utils/code_utils.h>
#include <utils/flash.h>
#include "platform-nrf5.h"
#include "softdevice.h"
#define MIN(a, b) (((a) < (b)) ? (a) : (b))
#define FLASH_PAGE_SIZE 4096
#define FLASH_TIMEOUT 1000
typedef enum {
FLASH_STATUS_IDLE,
FLASH_STATUS_PENDING,
FLASH_STATUS_SUCCESS,
FLASH_STATUS_FAILED
} SdFlashStatus;
static volatile SdFlashStatus sFlashStatus;
void nrf5SdSocFlashProcess(uint32_t aEvtId)
{
switch (aEvtId)
{
case NRF_EVT_FLASH_OPERATION_SUCCESS:
if (sFlashStatus == FLASH_STATUS_PENDING)
{
sFlashStatus = FLASH_STATUS_SUCCESS;
}
break;
case NRF_EVT_FLASH_OPERATION_ERROR:
if (sFlashStatus == FLASH_STATUS_PENDING)
{
sFlashStatus = FLASH_STATUS_FAILED;
}
break;
default:
break;
}
}
static otError sdFlashSingleWrite(uint32_t aAddress, uint8_t *aData, uint32_t aSize)
{
uint32_t retval;
uint32_t startTime = otPlatAlarmGetNow();
// Expect SotfDevice Flash Complete event.
sFlashStatus = FLASH_STATUS_PENDING;
retval = sd_flash_write((uint32_t *)aAddress, (uint32_t *)aData, aSize);
if (retval == NRF_SUCCESS)
{
// Wait for SoftDevice Flash Complete event.
do
{
if (sFlashStatus != FLASH_STATUS_PENDING)
{
break;
}
}
while (otPlatAlarmGetNow() - startTime < FLASH_TIMEOUT);
if (sFlashStatus != FLASH_STATUS_SUCCESS)
{
retval = NRF_ERROR_INTERNAL;
}
}
sFlashStatus = FLASH_STATUS_IDLE;
return nrf5SdErrorToOtError(retval);
}
otError nrf5FlashPageErase(uint32_t aAddress)
{
uint32_t retval;
uint32_t startTime = otPlatAlarmGetNow();
// Expect SotfDevice Flash Complete event.
sFlashStatus = FLASH_STATUS_PENDING;
retval = sd_flash_page_erase(aAddress / FLASH_PAGE_SIZE);
if (retval == NRF_SUCCESS)
{
// Wait for SoftDevice Flash Complete event.
do
{
if (sFlashStatus != FLASH_STATUS_PENDING)
{
break;
}
}
while (otPlatAlarmGetNow() - startTime < FLASH_TIMEOUT);
if (sFlashStatus != FLASH_STATUS_SUCCESS)
{
retval = NRF_ERROR_INTERNAL;
}
}
sFlashStatus = FLASH_STATUS_IDLE;
return nrf5SdErrorToOtError(retval);
}
bool nrf5FlashIsBusy(void)
{
return sFlashStatus != FLASH_STATUS_IDLE;
}
uint32_t nrf5FlashWrite(uint32_t aAddress, uint8_t *aData, uint32_t aSize)
{
otError error = OT_ERROR_NONE;
uint32_t result = 0;
uint32_t remainder = (aAddress % sizeof(uint32_t));
uint32_t blockSize;
uint32_t blockValue;
otEXPECT(sFlashStatus == FLASH_STATUS_IDLE);
// Check if @p aAddress is aligned to full word size. If not, make additional
// flash write at the beginning.
if (remainder)
{
blockSize = MIN((sizeof(uint32_t) - remainder), aSize);
blockValue = 0xffffffff;
memcpy((uint8_t *)&blockValue + remainder, aData, blockSize);
error = sdFlashSingleWrite(aAddress - remainder,
(uint8_t *)&blockValue,
sizeof(blockValue) / sizeof(uint32_t));
otEXPECT(error == OT_ERROR_NONE);
aAddress += blockSize;
aData += blockSize;
aSize -= blockSize;
result += blockSize;
}
otEXPECT(aSize);
// Store the middle block of data.
remainder = aSize % sizeof(uint32_t);
blockSize = aSize - remainder;
error = sdFlashSingleWrite(aAddress,
aData,
blockSize / sizeof(uint32_t));
otEXPECT(error == OT_ERROR_NONE);
aAddress += blockSize;
aData += blockSize;
aSize -= blockSize;
result += blockSize;
// Store any additional bytes that didn't fit into middle block.
if (remainder)
{
blockValue = 0xffffffff;
memcpy((uint8_t *)&blockValue, aData, remainder);
error = sdFlashSingleWrite(aAddress,
(uint8_t *)&blockValue,
sizeof(blockValue) / sizeof(uint32_t));
otEXPECT(error == OT_ERROR_NONE);
result += remainder;
}
exit:
return result;
}