1 /*
2 * Copyright (c) 2016, Freescale Semiconductor, Inc.
3 * Copyright 2016-2019 NXP
4 * All rights reserved.
5 *
6 * SPDX-License-Identifier: BSD-3-Clause
7 */
8 #ifndef _FSL_SPIFI_H_
9 #define _FSL_SPIFI_H_
10
11 #include "fsl_common.h"
12
13 /*!
14 * @addtogroup spifi
15 * @{
16 */
17
18 /*******************************************************************************
19 * Definitions
20 ******************************************************************************/
21
22 /*! @name Driver version */
23 /*@{*/
24 /*! @brief SPIFI driver version 2.0.3. */
25 #define FSL_SPIFI_DRIVER_VERSION (MAKE_VERSION(2, 0, 3))
26 /*@}*/
27
28 /*! @brief Status structure of SPIFI.*/
29 enum
30 {
31 kStatus_SPIFI_Idle = MAKE_STATUS(kStatusGroup_SPIFI, 0), /*!< SPIFI is in idle state */
32 kStatus_SPIFI_Busy = MAKE_STATUS(kStatusGroup_SPIFI, 1), /*!< SPIFI is busy */
33 kStatus_SPIFI_Error = MAKE_STATUS(kStatusGroup_SPIFI, 2), /*!< Error occurred during SPIFI transfer */
34 };
35
36 /*! @brief SPIFI interrupt source */
37 typedef enum _spifi_interrupt_enable
38 {
39 kSPIFI_CommandFinishInterruptEnable = SPIFI_CTRL_INTEN_MASK, /*!< Interrupt while command finished */
40 } spifi_interrupt_enable_t;
41
42 /*! @brief SPIFI SPI mode select */
43 typedef enum _spifi_spi_mode
44 {
45 kSPIFI_SPISckLow = 0x0U, /*!< SCK low after last bit of command, keeps low while CS high */
46 kSPIFI_SPISckHigh = 0x1U /*!< SCK high after last bit of command and while CS high */
47 } spifi_spi_mode_t;
48
49 /*! @brief SPIFI dual mode select */
50 typedef enum _spifi_dual_mode
51 {
52 kSPIFI_QuadMode = 0x0U, /*!< SPIFI uses IO3:0 */
53 kSPIFI_DualMode = 0x1U /*!< SPIFI uses IO1:0 */
54 } spifi_dual_mode_t;
55
56 /*! @brief SPIFI data direction */
57 typedef enum _spifi_data_direction
58 {
59 kSPIFI_DataInput = 0x0U, /*!< Data input from serial flash. */
60 kSPIFI_DataOutput = 0x1U /*!< Data output to serial flash. */
61 } spifi_data_direction_t;
62
63 /*! @brief SPIFI command opcode format */
64 typedef enum _spifi_command_format
65 {
66 kSPIFI_CommandAllSerial = 0x0, /*!< All fields of command are serial. */
67 kSPIFI_CommandDataQuad = 0x1U, /*!< Only data field is dual/quad, others are serial. */
68 kSPIFI_CommandOpcodeSerial = 0x2U, /*!< Only opcode field is serial, others are quad/dual. */
69 kSPIFI_CommandAllQuad = 0x3U /*!< All fields of command are dual/quad mode. */
70 } spifi_command_format_t;
71
72 /*! @brief SPIFI command type */
73 typedef enum _spifi_command_type
74 {
75 kSPIFI_CommandOpcodeOnly = 0x1U, /*!< Command only have opcode, no address field */
76 kSPIFI_CommandOpcodeAddrOneByte = 0x2U, /*!< Command have opcode and also one byte address field */
77 kSPIFI_CommandOpcodeAddrTwoBytes = 0x3U, /*!< Command have opcode and also two bytes address field */
78 kSPIFI_CommandOpcodeAddrThreeBytes = 0x4U, /*!< Command have opcode and also three bytes address field. */
79 kSPIFI_CommandOpcodeAddrFourBytes = 0x5U, /*!< Command have opcode and also four bytes address field */
80 kSPIFI_CommandNoOpcodeAddrThreeBytes = 0x6U, /*!< Command have no opcode and three bytes address field */
81 kSPIFI_CommandNoOpcodeAddrFourBytes = 0x7U /*!< Command have no opcode and four bytes address field */
82 } spifi_command_type_t;
83
84 /*! @brief SPIFI status flags */
85 enum
86 {
87 kSPIFI_MemoryCommandWriteFinished = SPIFI_STAT_MCINIT_MASK, /*!< Memory command write finished */
88 kSPIFI_CommandWriteFinished = SPIFI_STAT_CMD_MASK, /*!< Command write finished */
89 kSPIFI_InterruptRequest = SPIFI_STAT_INTRQ_MASK /*!< CMD flag from 1 to 0, means command execute finished */
90 };
91
92 /*! @brief SPIFI command structure */
93 typedef struct _spifi_command
94 {
95 uint16_t dataLen; /*!< How many data bytes are needed in this command. */
96 bool isPollMode; /*!< For command need to read data from serial flash */
97 spifi_data_direction_t direction; /*!< Data direction of this command. */
98 uint8_t intermediateBytes; /*!< How many intermediate bytes needed */
99 spifi_command_format_t format; /*!< Command format */
100 spifi_command_type_t type; /*!< Command type */
101 uint8_t opcode; /*!< Command opcode value */
102 } spifi_command_t;
103
104 /*!
105 * @brief SPIFI region configuration structure.
106 */
107 typedef struct _spifi_config
108 {
109 uint16_t timeout; /*!< SPI transfer timeout, the unit is SCK cycles */
110 uint8_t csHighTime; /*!< CS high time cycles */
111 bool disablePrefetch; /*!< True means SPIFI will not attempt a speculative prefetch. */
112 bool disableCachePrefech; /*!< Disable prefetch of cache line */
113 bool isFeedbackClock; /*!< Is data sample uses feedback clock. */
114 spifi_spi_mode_t spiMode; /*!< SPIFI spi mode select */
115 bool isReadFullClockCycle; /*!< If enable read full clock cycle. */
116 spifi_dual_mode_t dualMode; /*!< SPIFI dual mode, dual or quad. */
117 } spifi_config_t;
118
119 /*! @brief Transfer structure for SPIFI */
120 typedef struct _spifi_transfer
121 {
122 uint8_t *data; /*!< Pointer to data to transmit */
123 size_t dataSize; /*!< Bytes to be transmit */
124 } spifi_transfer_t;
125
126 /*******************************************************************************
127 * API
128 ******************************************************************************/
129
130 #if defined(__cplusplus)
131 extern "C" {
132 #endif /* _cplusplus */
133
134 /*!
135 * @name Initialization and deinitialization
136 * @{
137 */
138
139 /*!
140 * @brief Get the SPIFI instance from peripheral base address.
141 *
142 * @param base SPIFI peripheral base address.
143 * @return SPIFI instance.
144 */
145 uint32_t SPIFI_GetInstance(SPIFI_Type *base);
146
147 /*!
148 * @brief Initializes the SPIFI with the user configuration structure.
149 *
150 * This function configures the SPIFI module with the user-defined configuration.
151 *
152 * @param base SPIFI peripheral base address.
153 * @param config The pointer to the configuration structure.
154 */
155 void SPIFI_Init(SPIFI_Type *base, const spifi_config_t *config);
156
157 /*!
158 * @brief Get SPIFI default configure settings.
159 *
160 * @param config SPIFI config structure pointer.
161 */
162 void SPIFI_GetDefaultConfig(spifi_config_t *config);
163
164 /*!
165 * @brief Deinitializes the SPIFI regions.
166 *
167 * @param base SPIFI peripheral base address.
168 */
169 void SPIFI_Deinit(SPIFI_Type *base);
170
171 /* @}*/
172
173 /*!
174 * @name Basic Control Operations
175 * @{
176 */
177
178 /*!
179 * @brief Set SPIFI flash command.
180 *
181 * @param base SPIFI peripheral base address.
182 * @param cmd SPIFI command structure pointer.
183 */
184 void SPIFI_SetCommand(SPIFI_Type *base, spifi_command_t *cmd);
185
186 /*!
187 * @brief Set SPIFI command address.
188 *
189 * @param base SPIFI peripheral base address.
190 * @param addr Address value for the command.
191 */
SPIFI_SetCommandAddress(SPIFI_Type * base,uint32_t addr)192 static inline void SPIFI_SetCommandAddress(SPIFI_Type *base, uint32_t addr)
193 {
194 base->ADDR = addr;
195 }
196
197 /*!
198 * @brief Set SPIFI intermediate data.
199 *
200 * Before writing a command wihch needs specific intermediate value, users shall call this function to write it.
201 * The main use of this function for current serial flash is to select no-opcode mode and cancelling this mode. As
202 * dummy cycle do not care about the value, no need to call this function.
203 *
204 * @param base SPIFI peripheral base address.
205 * @param val Intermediate data.
206 */
SPIFI_SetIntermediateData(SPIFI_Type * base,uint32_t val)207 static inline void SPIFI_SetIntermediateData(SPIFI_Type *base, uint32_t val)
208 {
209 base->IDATA = val;
210 }
211
212 /*!
213 * @brief Set SPIFI Cache limit value.
214 *
215 * SPIFI includes caching of prevously-accessed data to improve performance. Software can write an address to this
216 * function, to prevent such caching at and above the address.
217 *
218 * @param base SPIFI peripheral base address.
219 * @param val Zero-based upper limit of cacheable memory.
220 */
SPIFI_SetCacheLimit(SPIFI_Type * base,uint32_t val)221 static inline void SPIFI_SetCacheLimit(SPIFI_Type *base, uint32_t val)
222 {
223 base->CLIMIT = val;
224 }
225
226 /*!
227 * @brief Reset the command field of SPIFI.
228 *
229 * This function is used to abort the current command or memory mode.
230 *
231 * @param base SPIFI peripheral base address.
232 */
SPIFI_ResetCommand(SPIFI_Type * base)233 static inline void SPIFI_ResetCommand(SPIFI_Type *base)
234 {
235 base->STAT = SPIFI_STAT_RESET_MASK;
236 /* Wait for the RESET flag cleared by HW */
237 while ((base->STAT & SPIFI_STAT_RESET_MASK) != 0x00U)
238 {
239 }
240 }
241
242 /*!
243 * @brief Set SPIFI flash AHB read command.
244 *
245 * Call this function means SPIFI enters to memory mode, while users need to use command, a SPIFI_ResetCommand shall
246 * be called.
247 *
248 * @param base SPIFI peripheral base address.
249 * @param cmd SPIFI command structure pointer.
250 */
251 void SPIFI_SetMemoryCommand(SPIFI_Type *base, spifi_command_t *cmd);
252
253 /*!
254 * @brief Enable SPIFI interrupt.
255 *
256 * The interrupt is triggered only in command mode, and it means the command now is finished.
257 *
258 * @param base SPIFI peripheral base address.
259 * @param mask SPIFI interrupt enable mask. It is a logic OR of members the
260 * enumeration :: spifi_interrupt_enable_t
261 */
SPIFI_EnableInterrupt(SPIFI_Type * base,uint32_t mask)262 static inline void SPIFI_EnableInterrupt(SPIFI_Type *base, uint32_t mask)
263 {
264 base->CTRL |= mask;
265 }
266
267 /*!
268 * @brief Disable SPIFI interrupt.
269 *
270 * The interrupt is triggered only in command mode, and it means the command now is finished.
271 *
272 * @param base SPIFI peripheral base address.
273 * @param mask SPIFI interrupt enable mask. It is a logic OR of members the
274 * enumeration :: spifi_interrupt_enable_t
275 */
SPIFI_DisableInterrupt(SPIFI_Type * base,uint32_t mask)276 static inline void SPIFI_DisableInterrupt(SPIFI_Type *base, uint32_t mask)
277 {
278 base->CTRL &= ~mask;
279 }
280
281 /*!
282 * @name Status
283 * @{
284 */
285
286 /*!
287 * @brief Get the status of all interrupt flags for SPIFI.
288 *
289 * @param base SPIFI peripheral base address.
290 * @return SPIFI flag status
291 */
SPIFI_GetStatusFlag(SPIFI_Type * base)292 static inline uint32_t SPIFI_GetStatusFlag(SPIFI_Type *base)
293 {
294 return base->STAT;
295 }
296
297 /* @}*/
298
299 /*!
300 * @brief Enable or disable DMA request for SPIFI.
301 *
302 * @param base SPIFI peripheral base address.
303 * @param enable True means enable DMA and false means disable DMA.
304 */
SPIFI_EnableDMA(SPIFI_Type * base,bool enable)305 static inline void SPIFI_EnableDMA(SPIFI_Type *base, bool enable)
306 {
307 if (enable)
308 {
309 base->CTRL |= SPIFI_CTRL_DMAEN_MASK;
310 }
311 else
312 {
313 base->CTRL &= ~SPIFI_CTRL_DMAEN_MASK;
314 }
315 }
316
317 /*!
318 * @brief Gets the SPIFI data register address.
319 *
320 * This API is used to provide a transfer address for the SPIFI DMA transfer configuration.
321 *
322 * @param base SPIFI base pointer
323 * @return data register address
324 */
SPIFI_GetDataRegisterAddress(SPIFI_Type * base)325 static inline uint32_t SPIFI_GetDataRegisterAddress(SPIFI_Type *base)
326 {
327 return (uint32_t)(&(base->DATA));
328 }
329
330 /*!
331 * @brief Write a word data in address of SPIFI.
332 *
333 * Users can write a page or at least a word data into SPIFI address.
334 *
335 * @param base SPIFI peripheral base address.
336 * @param data Data need be write.
337 */
SPIFI_WriteData(SPIFI_Type * base,uint32_t data)338 static inline void SPIFI_WriteData(SPIFI_Type *base, uint32_t data)
339 {
340 base->DATA = data;
341 }
342
343 /*!
344 * @brief Write a byte data in address of SPIFI.
345 *
346 * Users can write a byte data into SPIFI address.
347 *
348 * @param base SPIFI peripheral base address.
349 * @param data Data need be write.
350 */
SPIFI_WriteDataByte(SPIFI_Type * base,uint8_t data)351 static inline void SPIFI_WriteDataByte(SPIFI_Type *base, uint8_t data)
352 {
353 *((volatile uint8_t *)(&(base->DATA))) = data;
354 }
355
356 /*!
357 * @brief Write a halfword data in address of SPIFI.
358 *
359 * Users can write a halfword data into SPIFI address.
360 *
361 * @param base SPIFI peripheral base address.
362 * @param data Data need be write.
363 */
364 void SPIFI_WriteDataHalfword(SPIFI_Type *base, uint16_t data);
365
366 /*!
367 * @brief Read data from serial flash.
368 *
369 * Users should notice before call this function, the data length field in command register shall larger
370 * than 4, otherwise a hardfault will happen.
371 *
372 * @param base SPIFI peripheral base address.
373 * @return Data input from flash.
374 */
SPIFI_ReadData(SPIFI_Type * base)375 static inline uint32_t SPIFI_ReadData(SPIFI_Type *base)
376 {
377 return base->DATA;
378 }
379
380 /*!
381 * @brief Read a byte data from serial flash.
382 *
383 * @param base SPIFI peripheral base address.
384 * @return Data input from flash.
385 */
SPIFI_ReadDataByte(SPIFI_Type * base)386 static inline uint8_t SPIFI_ReadDataByte(SPIFI_Type *base)
387 {
388 return *((volatile uint8_t *)(&(base->DATA)));
389 }
390
391 /*!
392 * @brief Read a halfword data from serial flash.
393 *
394 * @param base SPIFI peripheral base address.
395 * @return Data input from flash.
396 */
397 uint16_t SPIFI_ReadDataHalfword(SPIFI_Type *base);
398
399 /* @} */
400
401 #if defined(__cplusplus)
402 }
403 #endif
404
405 /*! @}*/
406
407 #endif /* _FSL_SPIFI_H_ */
408