-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwdxs_file.c
317 lines (281 loc) · 10.8 KB
/
wdxs_file.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
/*************************************************************************************************/
/*!
* \file
*
* \brief Wireless Data Exchange profile implementation - File Example.
*
* Copyright (c) 2013-2018 Arm Ltd. All Rights Reserved.
* ARM Ltd. confidential and proprietary.
*
* IMPORTANT. Your use of this file is governed by a Software License Agreement
* ("Agreement") that must be accepted in order to download or otherwise receive a
* copy of this file. You may not use or copy this file for any purpose other than
* as described in the Agreement. If you do not agree to all of the terms of the
* Agreement do not use this file and delete all copies in your possession or control;
* if you do not have a copy of the Agreement, you must contact ARM Ltd. prior
* to any use, copying or further distribution of this software.
*/
/*************************************************************************************************/
#include <string.h>
#include "mxc_device.h"
#include "wsf_types.h"
#include "util/wstr.h"
#include "wsf_trace.h"
#include "wsf_assert.h"
#include "wsf_efs.h"
#include "wsf_cs.h"
#include "util/bstream.h"
#include "svc_wdxs.h"
#include "wdxs/wdxs_api.h"
#include "wdxs/wdxs_main.h"
#include "wdxs_file.h"
#include "dm_api.h"
#include "att_api.h"
#include "app_api.h"
#include "flc.h"
#ifndef FW_VERSION
#define FW_VERSION 1
#endif
static volatile uint32_t verifyLen;
static volatile uint8_t* lastWriteAddr;
static volatile uint32_t lastWriteLen;
/* Prototypes for file functions */
static uint8_t wdxsFileInitMedia(void);
static uint8_t wdxsFileErase(uint8_t* address, uint32_t size);
static uint8_t wdxsFileRead(uint8_t *pBuf, uint8_t *pAddress, uint32_t size);
static uint8_t wdxsFileWrite(const uint8_t *pBuf, uint8_t *pAddress, uint32_t size);
static uint8_t wsfFileHandle(uint8_t cmd, uint32_t param);
extern uint32_t _flash_update;
extern uint32_t _eflash_update;
/* Use the second half of the flash space for scratch space */
static const wsfEfsMedia_t WDXS_FileMedia = {
/* uint32_t startAddress; Start address. */ ((uint32_t)&_flash_update),
/* uint32_t endAddress; End address. */ ((uint32_t)&_eflash_update),
/* uint32_t pageSize; Page size. */ MXC_FLASH_PAGE_SIZE,
/* wsfMediaInitFunc_t *init; Media initialization callback. */ wdxsFileInitMedia,
/* wsfMediaEraseFunc_t *erase; Media erase callback. */ wdxsFileErase,
/* wsfMediaReadFunc_t *read; Media read callback. */ wdxsFileRead,
/* wsfMediaWriteFunc_t *write; Media write callback. */ wdxsFileWrite,
/* wsfMediaHandleCmdFunc_t *handleCmd; Media command handler callback. */ wsfFileHandle
};
/*************************************************************************************************/
/*!
* \brief Media Init function, called when media is registered.
*
* \return Status of the operation.
*/
/*************************************************************************************************/
static uint8_t wdxsFileInitMedia(void)
{
MXC_FLC_Init();
APP_TRACE_INFO1("FW_VERSION: %d", FW_VERSION);
return WSF_EFS_SUCCESS;
}
/*************************************************************************************************/
/*!
* \brief File erase function. Must be page aligned.
*
* \param pAddress Address in media to start erasing.
* \param size Number of bytes to erase.
*
* \return Status of the operation.
*/
/*************************************************************************************************/
static uint8_t wdxsFileErase(uint8_t* address, uint32_t size)
{
int err;
volatile uint32_t address32 = (uint32_t)address;
while(size) {
// WsfCsEnter();
err = MXC_FLC_PageErase((uint32_t)address32);
// WsfCsExit();
if(err != E_NO_ERROR) {
return WSF_EFS_FAILURE;
}
address32 += MXC_FLASH_PAGE_SIZE;
size -= MXC_FLASH_PAGE_SIZE;
}
return WSF_EFS_SUCCESS;
}
/*************************************************************************************************/
/*!
* \brief Media Read function.
*
* \param pBuf Buffer to hold data.
* \param pAddress Address in media to read from.
* \param size Size of pBuf in bytes.
*
* \return Status of the operation.
*/
/*************************************************************************************************/
static uint8_t wdxsFileRead(uint8_t *pBuf, uint8_t *pAddress, uint32_t size)
{
memcpy(pBuf, pAddress, size);
return WSF_EFS_SUCCESS;
}
/*************************************************************************************************/
/*!
* \brief File Write function.
*
* \param pBuf Buffer with data to be written.
* \param address Address in media to write to.
* \param size Size of pBuf in bytes.
*
* \return Status of the operation.
*/
/*************************************************************************************************/
static uint8_t wdxsFileWrite(const uint8_t *pBuf, uint8_t *pAddress, uint32_t size)
{
int err;
// WsfCsEnter();
err = MXC_FLC_Write((uint32_t)pAddress, size, (uint32_t *)pBuf);
// WsfCsExit();
if(err == E_NO_ERROR) {
lastWriteAddr = pAddress;
lastWriteLen = size;
return WSF_EFS_SUCCESS;
}
APP_TRACE_ERR1("Error writing to flash 0x%08X", (uint32_t)pAddress);
return WSF_EFS_FAILURE;
}
/* http://home.thep.lu.se/~bjorn/crc/ */
/*************************************************************************************************/
/*!
* \brief Create the CRC32 table.
*
* \param r Index into the table
*
* \return None.
*/
/*************************************************************************************************/
uint32_t crc32_for_byte(uint32_t r)
{
for(int j = 0; j < 8; ++j)
r = (r & 1? 0: (uint32_t)0xEDB88320L) ^ r >> 1;
return r ^ (uint32_t)0xFF000000L;
}
/*************************************************************************************************/
/*!
* \brief Calculate the CRC32 value for the given buffer.
*
* \param data Pointer to the data.
* \param n_bytes Number of bytes in the buffer.
* \param crc Pointer to store the result.
*
* \return None.
*/
/*************************************************************************************************/
static uint32_t table[0x100] = {0};
void crc32(const void *data, size_t n_bytes, uint32_t* crc)
{
if(!*table) {
for(size_t i = 0; i < 0x100; ++i)
table[i] = crc32_for_byte(i);
}
for(size_t i = 0; i < n_bytes; ++i) {
*crc = table[(uint8_t)*crc ^ ((uint8_t*)data)[i]] ^ *crc >> 8;
}
}
/*************************************************************************************************/
/*!
* \brief Media Specific Command handler.
*
* \param cmd Identifier of the media specific command.
* \param param Optional Parameter to the command.
*
* \return Status of the operation.
*/
/*************************************************************************************************/
static uint8_t wsfFileHandle(uint8_t cmd, uint32_t param)
{
switch(cmd) {
case WSF_EFS_WDXS_PUT_COMPLETE_CMD: {
/* Currently unimplemented */
return WDX_FTC_ST_SUCCESS;
}
break;
case WSF_EFS_VALIDATE_CMD:
default: {
/* Validate the image with CRC32 */
uint32_t crcResult = 0;
uint32_t crcFile;
verifyLen = (uint32_t)lastWriteAddr - WDXS_FileMedia.startAddress;
APP_TRACE_INFO2("CRC start addr: 0x%08X Len: 0x%08X", WDXS_FileMedia.startAddress, verifyLen);
crc32((const void*)WDXS_FileMedia.startAddress, verifyLen, &crcResult);
memcpy(&crcFile, (const char *)(WDXS_FileMedia.startAddress + verifyLen), 4);
/* Check the calculated CRC32 against what was received, 32 bits is 4 bytes */
if(crcFile != crcResult) {
APP_TRACE_INFO0("Update file verification failure");
APP_TRACE_INFO2("File CRC: 0x%08X Calculated CRC: 0x%08X", crcFile, crcResult);
return WDX_FTC_ST_VERIFICATION;
}
return WDX_FTC_ST_SUCCESS;
}
break;
}
return WDX_FTC_ST_SUCCESS;
}
/*************************************************************************************************/
/*!
* \brief Example of creating a WDXS stream.
*
* \param none
*
* \return None.
*/
/*************************************************************************************************/
void WdxsFileInit(void)
{
wsfEsfAttributes_t attr;
char versionString[WSF_EFS_VERSION_LEN];
/* Add major number */
versionString[0] = FW_VERSION & 0xFF;
/* Add "." */
versionString[1] = (FW_VERSION & 0xFF00) >> 8;
/* Minor number */
versionString[2] = (FW_VERSION & 0xFF0000) >> 16;
/* Add termination character */
versionString[3] = 0;
/* Register the media for the stream */
WsfEfsRegisterMedia(&WDXS_FileMedia, WDX_FLASH_MEDIA);
/* Set the attributes for the stream */
attr.permissions = (
WSF_EFS_REMOTE_GET_PERMITTED |
WSF_EFS_REMOTE_PUT_PERMITTED |
WSF_EFS_REMOTE_ERASE_PERMITTED |
WSF_EFS_REMOTE_VERIFY_PERMITTED |
WSF_EFS_LOCAL_GET_PERMITTED |
WSF_EFS_LOCAL_PUT_PERMITTED |
WSF_EFS_LOCAL_ERASE_PERMITTED |
WSF_EFS_REMOTE_VISIBLE);
attr.type = WSF_EFS_FILE_TYPE_BULK;
/* Potential buffer overrun is intentional to zero out fixed length field */
/* coverity[overrun-buffer-arg] */
WstrnCpy(attr.name, "File", WSF_EFS_NAME_LEN);
/* coverity[overrun-buffer-arg] */
WstrnCpy(attr.version, versionString, WSF_EFS_VERSION_LEN);
/* Add a file for the stream */
WsfEfsAddFile(WDXS_FileMedia.endAddress - WDXS_FileMedia.startAddress, WDX_FLASH_MEDIA, &attr, 0);
}
/*************************************************************************************************/
/*!
* \brief Get the base address of the WDXS file.
*
* \return Base address of WDXS file.
*/
/*************************************************************************************************/
uint32_t WdxsFileGetBaseAddr(void)
{
return WDXS_FileMedia.startAddress;
}
/*************************************************************************************************/
/*!
* \brief Get the length of the last verified WDXS file.
*
* \return Verified length of WDXS file.
*/
/*************************************************************************************************/
uint32_t WdxsFileGetVerifiedLength(void)
{
return verifyLen;
}