-
Notifications
You must be signed in to change notification settings - Fork 1
/
47x16.h
369 lines (306 loc) · 17.9 KB
/
47x16.h
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
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
/*!*****************************************************************************
* @file 47x16.h
* @author Fabien 'Emandhal' MAILLY
* @version 1.2.0
* @date 04/06/2023
* @brief EERAM47x16 driver
* @details I2C-Compatible (2-wire) 16-Kbit (2kB x 8) Serial EERAM
* Follow datasheet 47L04/47C04/47L16/47C16 Rev.C (Jun 2016)
******************************************************************************/
/* @page License
*
* Copyright (c) 2020-2023 Fabien MAILLY
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS,
* IMPLIED OR STATUTORY, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO
* EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES
* OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*****************************************************************************/
/* Revision history:
* 1.2.0 Add EEPROM genericness
* 1.1.0 I2C interface rework for I2C DMA use
* 1.0.0 Release version
*****************************************************************************/
#ifndef EERAM47x16_H_INC
#define EERAM47x16_H_INC
//=============================================================================
//-----------------------------------------------------------------------------
#include "ErrorsDef.h"
#include "I2C_Interface.h"
//-----------------------------------------------------------------------------
#ifdef USE_EEPROM_GENERICNESS
# include "EEPROM.h"
#endif
//-----------------------------------------------------------------------------
#ifdef __cplusplus
extern "C" {
# define __EERAM47x16_PACKED__
# define EERAM47x16_PACKITEM __pragma(pack(push, 1))
# define EERAM47x16_UNPACKITEM __pragma(pack(pop))
#else
# define __EERAM47x16_PACKED__ __attribute__((packed))
# define EERAM47x16_PACKITEM
# define EERAM47x16_UNPACKITEM
#endif
//-----------------------------------------------------------------------------
//! This macro is used to check the size of an object. If not, it will raise a "divide by 0" error at compile time
#define EERAM47x16_CONTROL_ITEM_SIZE(item, size) enum { item##_size_must_be_##size##_bytes = 1 / (int)(!!(sizeof(item) == size)) }
//-----------------------------------------------------------------------------
#ifdef USE_EEPROM_GENERICNESS
// 47(L/C)16 EERAM configurations
extern const EEPROM_Conf EERAM47L16_Conf, EERAM47C16_Conf;
#endif
//-----------------------------------------------------------------------------
// Limits definitions
#define EERAM47x16_I2CCLOCK_MAX ( 1000000u ) //!< Max I2C clock frequency
// Device definitions
#define EERAM47x16_SRAM_CHIPADDRESS_BASE ( 0xA0 ) //!< SRAM chip base address
#define EERAM47x16_REG_CHIPADDRESS_BASE ( 0x30 ) //!< Control Register chip base address
#define EERAM47x16_CHIPADDRESS_BASE_MASK ( 0xF0 ) //!< Base chip address mask
#define EERAM47x16_CHIPADDRESS_MASK ( 0xFC ) //!< Chip address mask
#define EERAM47x16_STATUS_REGISTER_ADDR ( 0x00 ) //!< Address to the status register
#define EERAM47x16_COMMAND_REGISTER_ADDR ( 0x55 ) //!< Address to the command register
#define EERAM47x16_STORE_COMMAND ( 0b00110011 ) //!< Command to store SRAM data to EEPROM
#define EERAM47x16_RECALL_COMMAND ( 0b11011101 ) //!< Command to recall data from EEPROM to SRAM
#define EERAM47x16_EERAM_SIZE ( 2048 ) //!< 47x16 total memory size
#define EERAM47x16_STORE_TIMEOUT ( 25 ) //!< Store Operation Duration: 25ms
#define EERAM47x16_RECALL_TIMEOUT ( 5 ) //!< Recall Operation Duration: 5ms
/*! @brief Generate the EERAM47x16 chip configurable address following the state of A1, and A2
* You shall set '1' (when corresponding pin is connected to +V) or '0' (when corresponding pin is connected to Ground) on each parameter
*/
#define EERAM47x16_ADDR(A2, A1) ( (uint8_t)((((A2) & 0x01) << 3) | (((A1) & 0x01) << 2)) )
//-----------------------------------------------------------------------------
//********************************************************************************************************************
// EERAM47x16 Specific Controller Registers
//********************************************************************************************************************
//! Status Register
EERAM47x16_PACKITEM
typedef union __EERAM47x16_PACKED__ EERAM47x16_Status_Register
{
uint8_t Status;
struct
{
uint8_t EVENT: 1; //!< 0 - Event Detect bit: '1' = An event was detected on the HS pin ; '0' = No event was detected on the HS pin
uint8_t ASE : 1; //!< 1 - Auto-Store Enable bit: '1' = Auto-Store feature is enabled ; '0' = Auto-Store feature is disabled
uint8_t BP : 3; //!< 2-4 - Block Protect bits
uint8_t : 2; //!< 5-6
uint8_t AM : 1; //!< 7 - Array Modified bit: '1' = SRAM array has been modified ; '0' = SRAM array has not been modified
} Bits;
} EERAM47x16_Status_Register;
EERAM47x16_UNPACKITEM;
EERAM47x16_CONTROL_ITEM_SIZE(EERAM47x16_Status_Register, 1);
#define EERAM47x16_EVENT_DETECTED (0x1u << 0) //!< An event was detected on the HS pin
#define EERAM47x16_ASE_ENABLE (0x1u << 1) //!< Enable Auto-Store feature
#define EERAM47x16_ASE_DISABLE (0x0u << 1) //!< Disable Auto-Store feature
//! Block Protect
typedef enum
{
EERAM47x16_NO_WRITE_PROTECT = 0b000, //!< No write protect
EERAM47x16_PROTECT_7E0h_7FFh = 0b001, //!< Protect range: Upper 1/64 (7E0h-7FFh)
EERAM47x16_PROTECT_7C0h_7FFh = 0b010, //!< Protect range: Upper 1/32 (7C0h-7FFh)
EERAM47x16_PROTECT_780h_7FFh = 0b011, //!< Protect range: Upper 1/16 (780h-7FFh)
EERAM47x16_PROTECT_700h_7FFh = 0b100, //!< Protect range: Upper 1/8 (700h-7FFh)
EERAM47x16_PROTECT_600h_7FFh = 0b101, //!< Protect range: Upper 1/4 (600h-7FFh)
EERAM47x16_PROTECT_400h_7FFh = 0b110, //!< Protect range: Upper 1/2 (400h-7FFh)
EERAM47x16_PROTECT_000h_7FFh = 0b111, //!< Protect range: All blocks
} eEERAM47x16_BlockProtect;
#define EERAM47x16_BP_Pos 2
#define EERAM47x16_BP_Mask (0x7u << EERAM47x16_BP_Pos)
#define EERAM47x16_BP_SET(value) (((uint8_t)(value) << EERAM47x16_BP_Pos) & EERAM47x16_BP_Mask) //!< Set Block Protect bits
#define EERAM47x16_BP_GET(value) (((uint8_t)(value) & EERAM47x16_BP_Mask) >> EERAM47x16_BP_Pos) //!< Get Block Protect bits
#define EERAM47x16_ARRAY_MODIFIED (0x1u << 7) //!< SRAM array has been modified
//-----------------------------------------------------------------------------
//********************************************************************************************************************
// EERAM47x16 Driver API
//********************************************************************************************************************
#define EERAM47x16_DMA_TRANSFER_IN_PROGRESS_Pos ( 0 )
#define EERAM47x16_DMA_TRANSFER_IN_PROGRESS ( 1u << EERAM47x16_DMA_TRANSFER_IN_PROGRESS_Pos ) // DMA transfer in progress
#define EERAM47x16_IS_DMA_TRANSFER_IN_PROGRESS(value) (((uint16_t)(value) & (1u << EERAM47x16_DMA_TRANSFER_IN_PROGRESS_Pos)) > 0) // Is DMA transfer in progress?
#define EERAM47x16_NO_DMA_TRANSFER_IN_PROGRESS_SET (~EERAM47x16_DMA_TRANSFER_IN_PROGRESS) // Mask to set no DMA transfer to the device
#define EERAM47x16_TRANSACTION_NUMBER_Pos ( 1 )
#define EERAM47x16_TRANSACTION_NUMBER_Mask ( 0x3F << EERAM47x16_TRANSACTION_NUMBER_Pos )
#define EERAM47x16_TRANSACTION_NUMBER_SET(value) (((uint16_t)(value) << EERAM47x16_TRANSACTION_NUMBER_Pos) & EERAM47x16_TRANSACTION_NUMBER_Mask) // Set the transaction number to internal config
#define EERAM47x16_TRANSACTION_NUMBER_GET(value) (((uint16_t)(value) & EERAM47x16_TRANSACTION_NUMBER_Mask) >> EERAM47x16_TRANSACTION_NUMBER_Pos) // Get the transaction number to internal config
#define EERAM47x16_TRANSACTION_NUMBER_CLEAR(value) value &= ~EERAM47x16_TRANSACTION_NUMBER_Mask // Clears the transaction number of internal config
//-----------------------------------------------------------------------------
typedef struct EERAM47x16 EERAM47x16; //! Typedef of EERAM47x16 device object structure
typedef uint8_t TEERAM47x16DriverInternal; //! Alias for Driver Internal data flags
//-----------------------------------------------------------------------------
#if !defined(USE_EEPROM_GENERICNESS)
/*! @brief Function that gives the current millisecond of the system to the driver
*
* This function will be called when the driver needs to get current millisecond
* @return Returns the current millisecond of the system
*/
typedef uint32_t (*GetCurrentms_Func)(void);
#endif
//-----------------------------------------------------------------------------
//! EERAM47x16 device object structure
struct EERAM47x16
{
#ifdef USE_EEPROM_GENERICNESS
struct EEPROM Eeprom;
#else
struct EEPROM_EERAM47x16
{
void *UserDriverData; //!< Optional, can be used to store driver data or NULL
TEERAM47x16DriverInternal InternalConfig; //!< DO NOT USE OR CHANGE THIS VALUE, IT'S THE INTERNAL DRIVER CONFIGURATION
//--- Interface driver call functions ---
# ifdef USE_DYNAMIC_INTERFACE
I2C_Interface* I2C; //!< This is the I2C_Interface descriptor pointer that will be used to communicate with the device
# else
I2C_Interface I2C; //!< This is the I2C_Interface descriptor that will be used to communicate with the device
# endif
uint32_t I2CclockSpeed; //!< Clock frequency of the I2C interface in Hertz
//--- Time call function ---
GetCurrentms_Func fnGetCurrentms; //!< This function will be called when the driver need to get current millisecond
//--- Device address ---
uint8_t AddrA2A1A0; //!< Device configurable address A2, and A1. A0 is not used. You can use the macro EERAM47x16_ADDR() to help filling this parameter. Only these 3 lower bits are used: ....21.. where 2 is A2, 1 is A1, and '.' are fixed by device
} Eeprom;
#endif // USE_EEPROM_GENERICNESS
};
//-----------------------------------------------------------------------------
/*! @brief EERAM47x16 initialization
*
* This function initializes the EERAM47x16 driver and call the initialization of the interface driver (I2C).
* Next it checks parameters and configures the EERAM47x16
* @param[in] *pComp Is the pointed structure of the device to be initialized
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT Init_EERAM47x16(EERAM47x16 *pComp);
/*! @brief Is the EERAM47x16 device ready
*
* Poll the acknowledge from the EERAM47x16
* @param[in] *pComp Is the pointed structure of the device to be used
* @return Returns 'true' if ready else 'false'
*/
bool EERAM47x16_IsReady(EERAM47x16 *pComp);
//-----------------------------------------------------------------------------
/*! @brief Read SRAM data from the EERAM47x16 device
*
* This function reads data from the SRAM area of a EERAM47x16 device
* @param[in] *pComp Is the pointed structure of the device to be used
* @param[in] address Is the address to read
* @param[out] *data Is where the data will be stored
* @param[in] size Is the size of the data array to read
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_ReadSRAMData(EERAM47x16 *pComp, uint16_t address, uint8_t* data, size_t size);
/*! @brief Read Control register from the EERAM47x16 device
*
* This function reads the content of the Status register of a EERAM47x16 device
* @param[in] *pComp Is the pointed structure of the device to be used
* @param[out] *data Is where the data will be stored
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_ReadRegister(EERAM47x16 *pComp, uint8_t* data);
/*! @brief Read SRAM data with DMA from the EERAM47x16 device
*
* To know the state of the DMA transfer, call this function. When the function returns ERR_NONE, it means that the transfer is complete else it returns the current state/error
* In case of no DMA, the function act like a EERAM47x16_ReadSRAMData() function
* @warning Never touch the data processed by the DMA before its completion
* @param[in] *pComp Is the pointed structure of the device to be used
* @param[in] address Is the address to read
* @param[out] *data Is where the data will be stored
* @param[in] size Is the size of the data array to read
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_ReadSRAMDataWithDMA(EERAM47x16 *pComp, uint16_t address, uint8_t* data, size_t size);
//-----------------------------------------------------------------------------
/*! @brief Write SRAM data to the EERAM47x16 device
*
* This function writes data to the SRAM area of a EERAM47x16 device
* @param[in] *pComp Is the pointed structure of the device to be used
* @param[in] address Is the address where data will be stored
* @param[in] *data Is the data array to store
* @param[in] size Is the size of the data array to write
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_WriteSRAMData(EERAM47x16 *pComp, uint16_t address, const uint8_t* data, size_t size);
/*! @brief Write Control register to the EERAM47x16 device
*
* This function writes data to the status or the command register of a EERAM47x16 device
* Use the address EERAM47x16_STATUS_REGISTER_ADDR (0x00) to write to the status register
* Use the address EERAM47x16_COMMAND_REGISTER_ADDR (0x55) to write to the command register
* @param[in] *pComp Is the pointed structure of the device to be used
* @param[in] address Is the address where data will be stored
* @param[in] data Is the data array to store
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_WriteRegister(EERAM47x16 *pComp, uint8_t address, const uint8_t data);
/*! @brief Write SRAM data with DMA to the EERAM47x16 device
*
* To know the state of the DMA transfer, call this function. When the function returns ERR_NONE, it means that the transfer is complete else it returns the current state/error
* In case of no DMA, the function act like a EERAM47x16_WriteSRAMData() function
* @warning Never touch the data processed by the DMA before its completion
* @param[in] *pComp Is the pointed structure of the device to be used
* @param[in] address Is the address where data will be stored
* @param[in] *data Is the data array to store
* @param[in] size Is the size of the data array to write
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_WriteSRAMDataWithDMA(EERAM47x16 *pComp, uint16_t address, const uint8_t* data, size_t size);
//-----------------------------------------------------------------------------
/*! @brief Store all the SRAM to the EEPROM of the EERAM47x16 device
*
* @param[in] *pComp Is the pointed structure of the device to be used
* @param[in] force This force the store of SRAM regardless of its status modified or not
* @param[in] waitEndOfStore This indicate that the function do not return before the end of store operation duration. If 'false', the function return directly after the send of store operation
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_StoreSRAMtoEEPROM(EERAM47x16 *pComp, bool force, bool waitEndOfStore);
/*! @brief Recall all data from EEPROM to SRAM of the EERAM47x16 device
*
* @param[in] *pComp Is the pointed structure of the device to be used
* @param[in] waitEndOfRecall This indicate that the function do not return before the end of recall operation duration. If 'false', the function return directly after the send of recall operation
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_RecallEEPROMtoSRAM(EERAM47x16 *pComp, bool waitEndOfRecall);
/*! @brief Activate the Auto-Store of the EERAM47x16 device
*
* @param[in] *pComp Is the pointed structure of the device to be used
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_ActivateAutoStore(EERAM47x16 *pComp);
/*! @brief Deactivate the Auto-Store of the EERAM47x16 device
*
* @param[in] *pComp Is the pointed structure of the device to be used
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_DeactivateAutoStore(EERAM47x16 *pComp);
/*! @brief Set block write protect of the EERAM47x16 device
*
* @param[in] *pComp Is the pointed structure of the device to be used
* @param[in] blockProtect Indicate what location to write protect
* @return Returns an #eERRORRESULT value enum
*/
eERRORRESULT EERAM47x16_SetBlockWriteProtect(EERAM47x16 *pComp, eEERAM47x16_BlockProtect blockProtect);
/*! @brief Get the status of the EERAM47x16 device
*
* @param[in] *pComp Is the pointed structure of the device to be used
* @param[in] *status Is where the result will be saved
* @return Returns an #eERRORRESULT value enum
*/
inline eERRORRESULT EERAM47x16_GetStatus(EERAM47x16 *pComp, EERAM47x16_Status_Register* status)
{
return EERAM47x16_ReadRegister(pComp, &status->Status); // Get the status register
}
//-----------------------------------------------------------------------------
#ifdef __cplusplus
}
#endif
//-----------------------------------------------------------------------------
#endif /* EERAM47x16_H_INC */