source: rtems/bsps/arm/atsam/contrib/libraries/libchip/source/efc.c @ 54aabb7

5
Last change on this file since 54aabb7 was 54aabb7, checked in by Sebastian Huber <sebastian.huber@…>, on 04/22/18 at 13:11:43

bsp/atsam: Move libraries to bsps

This patch is a part of the BSP source reorganization.

Update #3285.

  • Property mode set to 100644
File size: 8.5 KB
Line 
1/* ---------------------------------------------------------------------------- */
2/*                  Atmel Microcontroller Software Support                      */
3/*                       SAM Software Package License                           */
4/* ---------------------------------------------------------------------------- */
5/* Copyright (c) 2015, Atmel Corporation                                        */
6/*                                                                              */
7/* All rights reserved.                                                         */
8/*                                                                              */
9/* Redistribution and use in source and binary forms, with or without           */
10/* modification, are permitted provided that the following condition is met:    */
11/*                                                                              */
12/* - Redistributions of source code must retain the above copyright notice,     */
13/* this list of conditions and the disclaimer below.                            */
14/*                                                                              */
15/* Atmel's name may not be used to endorse or promote products derived from     */
16/* this software without specific prior written permission.                     */
17/*                                                                              */
18/* DISCLAIMER:  THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR   */
19/* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF */
20/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE   */
21/* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,      */
22/* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT */
23/* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,  */
24/* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF    */
25/* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING         */
26/* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, */
27/* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.                           */
28/* ---------------------------------------------------------------------------- */
29
30/** \addtogroup efc_module Working with EEFC
31 *  \ingroup peripherals_module
32 *
33 * The EEFC driver provides the interface to configure and use the EEFC
34 * peripheral.
35 *
36 * The user needs to set the number of wait states depending on the frequency
37 * used.\n
38 * Configure number of cycles for flash read/write operations in the FWS field
39 * of EEFC_FMR.
40 *
41 * It offers a function to send flash command to EEFC and waits for the
42 * flash to be ready.
43 *
44 * To send flash command, the user could do in either of following way:
45 * <ul>
46 * <li>Write a correct key, command and argument in EEFC_FCR. </li>
47 * <li>Or, Use IAP (In Application Programming) function which is executed from
48 * ROM directly, this allows flash programming to be done by code running in
49 * flash.</li>
50 * <li>Once the command is achieved, it can be detected even by polling
51 * EEFC_FSR or interrupt.
52 * </ul>
53 *
54 * The command argument could be a page number,GPNVM number or nothing, it
55 * depends on the command itself. Some useful functions in this driver could
56 * help user translate physical flash address into a page number and vice verse.
57 *
58 * For more accurate information, please look at the EEFC section of the
59 * Datasheet.
60 *
61 * Related files :\n
62 * \ref efc.c\n
63 * \ref efc.h.\n
64 */
65/*@{*/
66/*@}*/
67
68
69/**
70 * \file
71 *
72 * Implementation of Enhanced Embedded Flash Controller (EEFC).
73 *
74 */
75
76
77/*----------------------------------------------------------------------------
78 *        Headers
79 *----------------------------------------------------------------------------*/
80#include "chip.h"
81
82#include <assert.h>
83
84
85/*----------------------------------------------------------------------------
86 *        Macro
87 *----------------------------------------------------------------------------*/
88#define EEFC_FCR_FCMD(value) ((EEFC_FCR_FCMD_Msk & ((value) << EEFC_FCR_FCMD_Pos)))
89
90/*----------------------------------------------------------------------------
91 *        Exported functions
92 *----------------------------------------------------------------------------*/
93
94extern void EFC_WriteFMR(Efc *efc, uint32_t dwFmr);
95
96#ifdef __ICCARM__
97        extern __ramfunc void EFC_WriteFMR(Efc *efc, uint32_t dwFmr)
98#else
99        __attribute__ ((section (".ramfunc")))
100        extern void EFC_WriteFMR(Efc *efc, uint32_t dwFmr)
101#endif
102{
103        efc->EEFC_FMR = dwFmr;
104}
105
106/**
107 * \brief Enables the flash ready interrupt source on the EEFC peripheral.
108 *
109 * \param efc  Pointer to a Efc instance
110 */
111extern void EFC_EnableFrdyIt(Efc *efc)
112{
113        uint32_t dwFmr;
114
115        dwFmr = efc->EEFC_FMR |= EEFC_FMR_FRDY;
116        EFC_WriteFMR(efc, dwFmr);
117}
118
119/**
120 * \brief Disables the flash ready interrupt source on the EEFC peripheral.
121 *
122 * \param efc  Pointer to a Efc instance
123 */
124extern void EFC_DisableFrdyIt(Efc *efc)
125{
126        uint32_t dwFmr;
127
128        dwFmr = efc->EEFC_FMR & (~EEFC_FMR_FRDY);
129        EFC_WriteFMR(efc, dwFmr);
130}
131
132
133/**
134 * \brief Set read/write wait state on the EEFC peripheral.
135 *
136 * \param efc  Pointer to a Efc instance
137 * \param cycles  the number of wait states in cycle.
138 */
139extern void EFC_SetWaitState(Efc *efc, uint8_t ucCycles)
140{
141        uint32_t dwFmr;
142
143        dwFmr = efc->EEFC_FMR;
144        dwFmr &= ~((uint32_t)EEFC_FMR_FWS_Msk);
145        dwFmr |= EEFC_FMR_FWS(ucCycles);
146        EFC_WriteFMR(efc, dwFmr);
147}
148
149/**
150 * \brief Returns the current status of the EEFC.
151 *
152 * \note Keep in mind that this function clears the value of some status bits
153 * (LOCKE, PROGE).
154 *
155 * \param efc  Pointer to a Efc instance
156 */
157extern uint32_t EFC_GetStatus(Efc *efc)
158{
159        return efc->EEFC_FSR;
160}
161
162/**
163 * \brief Returns the result of the last executed command.
164 *
165 * \param efc  Pointer to a Efc instance
166 */
167extern uint32_t EFC_GetResult(Efc *efc)
168{
169        return efc->EEFC_FRR;
170}
171
172/**
173 * \brief Translates the given address page and offset values.
174 * \note The resulting values are stored in the provided variables if they are
175 * not null.
176 *
177 * \param efc  Pointer to a Efc instance
178 * \param address  Address to translate.
179 * \param pPage  First page accessed.
180 * \param pOffset  Byte offset in first page.
181 */
182extern void EFC_TranslateAddress(Efc **ppEfc, uint32_t dwAddress,
183                                                                  uint16_t *pwPage,
184                                                                  uint16_t *pwOffset)
185{
186        assert(dwAddress >= IFLASH_ADDR);
187        assert(dwAddress <= (IFLASH_ADDR + IFLASH_SIZE));
188
189        /* Store values */
190        if (ppEfc)
191                *ppEfc = EFC;
192
193        if (pwPage)
194                *pwPage = (dwAddress - IFLASH_ADDR) / IFLASH_PAGE_SIZE;
195
196        if (pwOffset) {
197                *pwOffset = (dwAddress - IFLASH_ADDR) % IFLASH_PAGE_SIZE;;
198        }
199}
200
201
202/**
203 * \brief Computes the address of a flash access given the page and offset.
204 *
205 * \param efc  Pointer to a Efc instance
206 * \param page  Page number.
207 * \param offset  Byte offset inside page.
208 * \param pAddress  Computed address (optional).
209 */
210extern void EFC_ComputeAddress(Efc *efc, uint16_t wPage, uint16_t wOffset,
211                                                                uint32_t *pdwAddress)
212{
213        uint32_t dwAddress;
214
215        /* Stop warning */
216        efc = efc;
217
218        assert(efc);
219        assert(wPage <= IFLASH_NB_OF_PAGES);
220        assert(wOffset < IFLASH_PAGE_SIZE);
221        dwAddress = IFLASH_ADDR + wPage * IFLASH_PAGE_SIZE + wOffset;
222
223        /* Store result */
224        if (pdwAddress != NULL)
225                *pdwAddress = dwAddress;
226}
227
228/**
229 * \brief Performs the given command and wait until its completion (or an error).
230 *
231 * \param efc  Pointer to a Efc instance
232 * \param command  Command to perform.
233 * \param argument  Optional command argument.
234 *
235 * \return 0 if successful, otherwise returns an error code.
236 */
237
238extern uint32_t EFC_PerformCommand(Efc *efc, uint32_t dwCommand,
239                                                                        uint32_t dwArgument, uint32_t dwUseIAP)
240{
241        if (dwUseIAP != 0) {
242                /* Pointer on IAP function in ROM */
243                static uint32_t (*IAP_PerformCommand)(uint32_t, uint32_t);
244
245                IAP_PerformCommand = (uint32_t (*)(uint32_t, uint32_t))
246                                                         * ((uint32_t *)CHIP_FLASH_IAP_ADDRESS);
247
248                if (efc == EFC) {
249                        IAP_PerformCommand(0, EEFC_FCR_FKEY_PASSWD | EEFC_FCR_FARG(dwArgument)
250                                                                | EEFC_FCR_FCMD(dwCommand));
251                }
252
253                return (efc->EEFC_FSR & (EEFC_FSR_FLOCKE | EEFC_FSR_FCMDE | EEFC_FSR_FLERR));
254        } else {
255                uint32_t dwStatus;
256
257                efc->EEFC_FCR = EEFC_FCR_FKEY_PASSWD | EEFC_FCR_FARG(dwArgument)
258                                                | EEFC_FCR_FCMD(dwCommand);
259
260                do {
261                        dwStatus = efc->EEFC_FSR;
262                } while ((dwStatus & EEFC_FSR_FRDY) != EEFC_FSR_FRDY);
263
264                return (dwStatus & (EEFC_FSR_FLOCKE | EEFC_FSR_FCMDE | EEFC_FSR_FLERR));
265        }
266}
267
268/**
269 * \brief Set flash access mode.
270 *
271 * \param dwMode - 0:128-bit, (1<<24):64-bit
272 */
273extern void EFC_SetFlashAccessMode(Efc *efc, uint32_t dwMode)
274{
275        uint32_t dwFmr;
276
277        dwFmr = dwMode;
278        EFC_WriteFMR(efc, dwFmr);
279}
280
Note: See TracBrowser for help on using the repository browser.