dma2d_hal.c 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940
  1. /*
  2. * Copyright (c) 2020, Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. ******************************************************************************
  8. * @file dma2d_hal.c
  9. * @brief DMA2D HAL module driver.
  10. * This file provides firmware functions to manage the following
  11. * functionalities of the DMA2D peripheral:
  12. * + Initialization and de-initialization functions
  13. * + IO operation functions
  14. * + Peripheral Control functions
  15. * + Peripheral State and Errors functions
  16. *
  17. @verbatim
  18. ==============================================================================
  19. ##### How to use this driver #####
  20. ==============================================================================
  21. [..]
  22. (#) Initialize the DMA2D module using hal_dma2d_init() function.
  23. (#) Program the required configuration through the following parameters:
  24. the transfer mode, the output color mode and the output offset using
  25. hal_dma2d_config_output() function.
  26. (#) Program the required configuration through the following parameters:
  27. -@- the input color mode, the input color, the input alpha value, the alpha mode,
  28. the red/blue swap mode, the inverted alpha mode and the input offset using
  29. hal_dma2d_config_layer() function for foreground or/and background layer.
  30. -@- the rotation configuration using hal_dma2d_config_rotation.
  31. *** Polling mode IO operation ***
  32. =================================
  33. [..]
  34. (#) Configure pdata parameter (explained hereafter), destination and data length
  35. and enable the transfer using hal_dma2d_start().
  36. (#) Wait for end of transfer using hal_dma2d_poll_transfer(), at this stage
  37. user can specify the value of timeout according to his end application.
  38. *** Interrupt mode IO operation ***
  39. ===================================
  40. [..]
  41. (#) Use function @ref hal_dma2d_register_callback() to register user callbacks,
  42. xfer_cplt_callback and xfer_error_callback.
  43. (#) Configure pdata parameter, destination and data length and enable
  44. the transfer using hal_dma2d_start().
  45. (#) At the end of data transfer dma2d_device_handler() function is executed and user can
  46. add his own function by customization of function pointer xfer_cplt_callback (member
  47. of DMA2D handle structure).
  48. (#) In case of error, the dma2d_device_handler() function calls the callback
  49. xfer_error_callback.
  50. -@- In Register-to-Memory transfer mode, pdata parameter is the register
  51. color, in Memory-to-memory or Memory-to-Memory with pixel format
  52. conversion pdata is the source address.
  53. -@- Configure the foreground source address, the background source address,
  54. the destination and data length then Enable the transfer using
  55. hal_dma2d_blending_start() either in polling mode or interrupt mode.
  56. -@- hal_dma2d_blending_start() function is used if the memory to memory
  57. with blending transfer mode is selected.
  58. -@- hal_dma2d_rotation_start() function is used if the memory to memory
  59. with rotation transfer mode is selected.
  60. (#) To control the DMA2D state, use the following function: hal_dma2d_get_state().
  61. (#) To read the DMA2D error code, use the following function: hal_dma2d_get_error().
  62. *** Callback registration ***
  63. ===================================
  64. [..]
  65. (#) Use function @ref hal_dma2d_register_callback() to register a user callback.
  66. (#) Function @ref hal_dma2d_register_callback() allows to register following callbacks:
  67. (+) xfer_cplt_callback : callback for transfer complete.
  68. (+) xfer_error_callback : callback for transfer error.
  69. This function takes as parameters the HAL peripheral handle, the Callback ID
  70. and a pointer to the user callback function.
  71. (#) Use function @ref hal_dma2d_unregister_callback() to reset a callback to the default
  72. weak (surcharged) function.
  73. @ref hal_dma2d_unregister_callback() takes as parameters the HAL peripheral handle,
  74. and the Callback ID.
  75. This function allows to reset following callbacks:
  76. (+) xfer_cplt_callback : callback for transfer complete.
  77. (+) xfer_error_callback : callback for transfer error.
  78. [..]
  79. (@) You can refer to the DMA2D HAL driver header file for more useful macros
  80. @endverbatim
  81. */
  82. /* Includes ------------------------------------------------------------------*/
  83. #include <string.h>
  84. #include <assert.h>
  85. #include <spicache.h>
  86. #include <display/sw_math.h>
  87. #include <dma2d_hal.h>
  88. /** @defgroup DMA2D DMA2D
  89. * @brief DMA2D HAL module driver
  90. * @{
  91. */
  92. /* Private types -------------------------------------------------------------*/
  93. /* Private define ------------------------------------------------------------*/
  94. /** @defgroup DMA2D_Private_Constants DMA2D Private Constants
  95. * @{
  96. */
  97. /**
  98. * @}
  99. */
  100. /* Private variables ---------------------------------------------------------*/
  101. /* DMA2D global enable bit */
  102. static bool global_en = true;
  103. /* Private constants ---------------------------------------------------------*/
  104. /* Private macro -------------------------------------------------------------*/
  105. #define IS_DMA2D_LAYER(layer) (((layer) == HAL_DMA2D_BACKGROUND_LAYER) || ((layer) == HAL_DMA2D_FOREGROUND_LAYER))
  106. #define IS_DMA2D_MODE(mode) (((mode) == HAL_DMA2D_R2M) || ((mode) == HAL_DMA2D_M2M) || \
  107. ((mode) == HAL_DMA2D_M2M_BLEND) || ((mode) == HAL_DMA2D_M2M_BLEND_FG) || \
  108. ((mode) == HAL_DMA2D_M2M_BLEND_BG) || ((mode) == HAL_DMA2D_M2M_ROTATE))
  109. #define IS_DMA2D_LINE(line) ((line) <= HAL_DMA2D_LINE)
  110. #define IS_DMA2D_PIXEL(pixel) ((pixel) <= HAL_DMA2D_PIXEL)
  111. #define IS_DMA2D_OFFSET(ooffset) ((ooffset) <= HAL_DMA2D_OFFSET)
  112. #define IS_DMA2D_ALPHA_MODE(alpha_mode) (((alpha_mode) == HAL_DMA2D_NO_MODIF_ALPHA) || \
  113. ((alpha_mode) == HAL_DMA2D_REPLACE_ALPHA) || \
  114. ((alpha_mode) == HAL_DMA2D_COMBINE_ALPHA))
  115. #define IS_DMA2D_RB_SWAP(rb_swap) (((rb_swap) == HAL_DMA2D_RB_REGULAR) || \
  116. ((rb_swap) == HAL_DMA2D_RB_SWAP))
  117. #define IS_DMA2D_COLOR_MODE(color_mode) (((color_mode) == HAL_DMA2D_ARGB8888) || \
  118. ((color_mode) == HAL_DMA2D_ARGB6666) || \
  119. ((color_mode) == HAL_DMA2D_RGB888) || \
  120. ((color_mode) == HAL_DMA2D_RGB565) || \
  121. ((color_mode) == HAL_DMA2D_A8) || \
  122. ((color_mode) == HAL_DMA2D_A4) || \
  123. ((color_mode) == HAL_DMA2D_A1) || \
  124. ((color_mode) == HAL_DMA2D_A4_LE) || \
  125. ((color_mode) == HAL_DMA2D_A1_LE) || \
  126. ((color_mode) == HAL_DMA2D_RGB565_LE))
  127. /* Private function prototypes -----------------------------------------------*/
  128. /** @addtogroup DMA2D_Private_Functions DMA2D Private Functions
  129. * @{
  130. */
  131. static int dma2d_set_config(hal_dma2d_handle_t *hdma2d, uint32_t src_address1, uint32_t src_address2, uint32_t dst_address, uint16_t width, uint16_t height);
  132. /**
  133. * @}
  134. */
  135. /* Private functions ---------------------------------------------------------*/
  136. /**
  137. * @brief Get the DMA2D device
  138. * @retval Pointer to the DMA2D device structure
  139. */
  140. static const struct device *dma2d_get_device(void)
  141. {
  142. static const struct device *dma2d_dev;
  143. if (!dma2d_dev) {
  144. dma2d_dev = device_get_binding(CONFIG_DISPLAY_ENGINE_DEV_NAME);
  145. }
  146. assert(dma2d_dev != NULL);
  147. return dma2d_dev;
  148. }
  149. /**
  150. * @brief Covert the DMA2D ColorMode and rb_swap to display pixel format
  151. * @retval Pointer to the DMA2D device structure
  152. */
  153. static int32_t dma2d_get_display_format(uint16_t color_mode, uint16_t rb_swap)
  154. {
  155. switch (color_mode) {
  156. case HAL_DMA2D_RGB565:
  157. return (rb_swap == HAL_DMA2D_RB_REGULAR) ?
  158. PIXEL_FORMAT_RGB_565 : PIXEL_FORMAT_BGR_565;
  159. case HAL_DMA2D_ARGB8888:
  160. return (rb_swap == HAL_DMA2D_RB_REGULAR) ?
  161. PIXEL_FORMAT_ARGB_8888 : -1;
  162. case HAL_DMA2D_ARGB6666:
  163. return (rb_swap == HAL_DMA2D_RB_REGULAR) ?
  164. PIXEL_FORMAT_ARGB_6666 : PIXEL_FORMAT_ABGR_6666;
  165. case HAL_DMA2D_RGB565_LE:
  166. return (rb_swap == HAL_DMA2D_RB_REGULAR) ?
  167. PIXEL_FORMAT_RGB_565_LE : -1;
  168. case HAL_DMA2D_RGB888:
  169. return (rb_swap == HAL_DMA2D_RB_REGULAR) ?
  170. PIXEL_FORMAT_RGB_888 : -1;
  171. case HAL_DMA2D_A8:
  172. return PIXEL_FORMAT_A8;
  173. case HAL_DMA2D_A4:
  174. return PIXEL_FORMAT_A4;
  175. case HAL_DMA2D_A1:
  176. return PIXEL_FORMAT_A1;
  177. case HAL_DMA2D_A4_LE:
  178. return PIXEL_FORMAT_A4_LE;
  179. case HAL_DMA2D_A1_LE:
  180. return PIXEL_FORMAT_A1_LE;
  181. default:
  182. return -1;
  183. }
  184. }
  185. /* Exported functions --------------------------------------------------------*/
  186. /** @defgroup DMA2D_Exported_Functions DMA2D Exported Functions
  187. * @{
  188. */
  189. /** @defgroup DMA2D_Exported_Functions_Group1 Initialization and de-initialization functions
  190. * @brief Initialization and Configuration functions
  191. *
  192. @verbatim
  193. ===============================================================================
  194. ##### Initialization and Configuration functions #####
  195. ===============================================================================
  196. [..] This section provides functions allowing to:
  197. (+) Initialize and configure the DMA2D
  198. (+) De-initialize the DMA2D
  199. @endverbatim
  200. * @{
  201. */
  202. /**
  203. * @brief DMA2D callback from device driver, may called in interrupt routine
  204. * @param status DMA2D device status
  205. * @param cmd_seq Current DMA2D device command sequence
  206. * @param user_data Address of structure hal_dma2d_handle_t
  207. */
  208. static void dma2d_device_handler(int status, uint16_t cmd_seq, void *user_data)
  209. {
  210. hal_dma2d_handle_t *hdma2d = user_data;
  211. if (status != 0) {
  212. /* Update error code */
  213. hdma2d->error_code |= HAL_DMA2D_ERROR_TIMEOUT;
  214. if (hdma2d->xfer_error_callback != NULL) {
  215. /* Transfer error Callback */
  216. hdma2d->xfer_error_callback(hdma2d, cmd_seq);
  217. }
  218. } else {
  219. /* Update error code */
  220. hdma2d->error_code |= HAL_DMA2D_ERROR_NONE;
  221. if (hdma2d->xfer_cplt_callback != NULL) {
  222. /* Transfer error Callback */
  223. hdma2d->xfer_cplt_callback(hdma2d, cmd_seq);
  224. }
  225. }
  226. atomic_dec(&hdma2d->xfer_count);
  227. }
  228. /**
  229. * @brief Initialize the DMA2D peripheral and create the associated handle.
  230. * @param hdma2d pointer to a hal_dma2d_handle_t structure that contains
  231. * the configuration information for the DMA2D.
  232. * @retval 0 on success else negative errno code.
  233. */
  234. int hal_dma2d_init(hal_dma2d_handle_t *hdma2d)
  235. {
  236. const struct device *dma2d_dev = dma2d_get_device();
  237. /* Check the DMA2D peripheral existence */
  238. if (dma2d_dev == NULL) {
  239. return -ENODEV;
  240. }
  241. /* Check the DMA2D peripheral State */
  242. if (hdma2d == NULL) {
  243. return -EINVAL;
  244. }
  245. /* Open the DMA2D device instance */
  246. hdma2d->instance = display_engine_open(dma2d_dev, 0);
  247. if (hdma2d->instance < 0) {
  248. return -EBUSY;
  249. }
  250. /* Register the DMA2D device instance callback */
  251. display_engine_register_callback(dma2d_dev, hdma2d->instance, dma2d_device_handler, hdma2d);
  252. atomic_set(&hdma2d->xfer_count, 0);
  253. /* Update error code */
  254. hdma2d->error_code = HAL_DMA2D_ERROR_NONE;
  255. return 0;
  256. }
  257. /**
  258. * @brief Deinitializes the DMA2D peripheral registers to their default reset
  259. * values.
  260. * @param hdma2d pointer to a hal_dma2d_handle_t structure that contains
  261. * the configuration information for the DMA2D.
  262. * @retval 0 on success else negative errno code.
  263. */
  264. int hal_dma2d_deinit(hal_dma2d_handle_t *hdma2d)
  265. {
  266. const struct device *dma2d_dev = dma2d_get_device();
  267. /* Check the DMA2D peripheral existence */
  268. if (dma2d_dev == NULL) {
  269. return -ENODEV;
  270. }
  271. /* Check the DMA2D peripheral State */
  272. if (hdma2d == NULL || hdma2d->instance < 0) {
  273. return -EINVAL;
  274. }
  275. /* Close the DMA2D device instance */
  276. display_engine_close(dma2d_dev, hdma2d->instance);
  277. /* Assign invald value */
  278. hdma2d->instance = -1;
  279. /* Update error code */
  280. hdma2d->error_code = HAL_DMA2D_ERROR_NONE;
  281. return 0;
  282. }
  283. /**
  284. * @brief Register a User DMA2D Callback
  285. * To be used instead of the weak (surcharged) predefined callback
  286. * @param hdma2d DMA2D handle
  287. * @param callback_id ID of the callback to be registered
  288. * This parameter can be one of the following values:
  289. * @arg @ref HAL_DMA2D_TRANSFERCOMPLETE_CB_ID DMA2D transfer complete Callback ID
  290. * @arg @ref HAL_DMA2D_TRANSFERERROR_CB_ID DMA2D transfer error Callback ID
  291. * @param callback_fn pointer to the callback function
  292. * @retval 0 on success else negative errno code.
  293. */
  294. int hal_dma2d_register_callback(hal_dma2d_handle_t *hdma2d, hal_dma2d_callback_e callback_id, hal_dma2d_callback_t callback_fn)
  295. {
  296. int status = 0;
  297. if (callback_fn == NULL) {
  298. /* Update the error code */
  299. hdma2d->error_code |= HAL_DMA2D_ERROR_INVALID_CALLBACK;
  300. return -EINVAL;
  301. }
  302. if (HAL_DMA2D_STATE_READY == hal_dma2d_get_state(hdma2d)) {
  303. switch (callback_id) {
  304. case HAL_DMA2D_TRANSFERCOMPLETE_CB_ID :
  305. hdma2d->xfer_cplt_callback = callback_fn;
  306. break;
  307. case HAL_DMA2D_TRANSFERERROR_CB_ID :
  308. hdma2d->xfer_error_callback = callback_fn;
  309. break;
  310. default :
  311. /* Update the error code */
  312. hdma2d->error_code |= HAL_DMA2D_ERROR_INVALID_CALLBACK;
  313. /* update return status */
  314. status = -EINVAL;
  315. break;
  316. }
  317. } else {
  318. /* Update the error code */
  319. hdma2d->error_code |= HAL_DMA2D_ERROR_INVALID_CALLBACK;
  320. /* update return status */
  321. status = -EBUSY;
  322. }
  323. return status;
  324. }
  325. /**
  326. * @brief Unregister a DMA2D Callback
  327. * DMA2D Callback is redirected to the weak (surcharged) predefined callback
  328. * @param hdma2d DMA2D handle
  329. * @param callback_id ID of the callback to be unregistered
  330. * This parameter can be one of the following values:
  331. * @arg @ref HAL_DMA2D_TRANSFERCOMPLETE_CB_ID DMA2D transfer complete Callback ID
  332. * @arg @ref HAL_DMA2D_TRANSFERERROR_CB_ID DMA2D transfer error Callback ID
  333. * @retval 0 on success else negative errno code.
  334. */
  335. int hal_dma2d_unregister_callback(hal_dma2d_handle_t *hdma2d, hal_dma2d_callback_e callback_id)
  336. {
  337. int status = 0;
  338. if (HAL_DMA2D_STATE_READY == hal_dma2d_get_state(hdma2d)) {
  339. switch (callback_id) {
  340. case HAL_DMA2D_TRANSFERCOMPLETE_CB_ID :
  341. hdma2d->xfer_cplt_callback = NULL;
  342. break;
  343. case HAL_DMA2D_TRANSFERERROR_CB_ID :
  344. hdma2d->xfer_error_callback = NULL;
  345. break;
  346. default :
  347. /* Update the error code */
  348. hdma2d->error_code |= HAL_DMA2D_ERROR_INVALID_CALLBACK;
  349. /* update return status */
  350. status = -EINVAL;
  351. break;
  352. }
  353. } else {
  354. /* Update the error code */
  355. hdma2d->error_code |= HAL_DMA2D_ERROR_INVALID_CALLBACK;
  356. /* update return status */
  357. status = -EBUSY;
  358. }
  359. return status;
  360. }
  361. /**
  362. * @}
  363. */
  364. /** @defgroup DMA2D_Exported_Functions_Group2 IO operation functions
  365. * @brief IO operation functions
  366. *
  367. @verbatim
  368. ===============================================================================
  369. ##### IO operation functions #####
  370. ===============================================================================
  371. [..] This section provides functions allowing to:
  372. (+) Configure the pdata, destination address and data size then
  373. start the DMA2D transfer.
  374. (+) Configure the source for foreground and background, destination address
  375. and data size then start a MultiBuffer DMA2D transfer.
  376. (+) Configure the pdata, destination address and data size then
  377. start the DMA2D transfer with interrupt.
  378. (+) Configure the source for foreground and background, destination address
  379. and data size then start a MultiBuffer DMA2D transfer with interrupt.
  380. (+) Poll for transfer complete.
  381. (+) handle DMA2D interrupt request.
  382. @endverbatim
  383. * @{
  384. */
  385. /**
  386. * @brief Start the DMA2D Transfer.
  387. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  388. * the configuration information for the DMA2D.
  389. * @param pdata Configure the source memory Buffer address if
  390. * Memory-to-Memory or Memory-to-Memory with pixel format
  391. * conversion mode is selected, or configure
  392. * the color value if Register-to-Memory mode is selected.
  393. * @param dst_address The destination memory Buffer address.
  394. * @param width The width of data to be transferred from source to destination (expressed in number of pixels per line).
  395. * @param height The height of data to be transferred from source to destination (expressed in number of lines).
  396. * @retval command sequence (uint16_t) on success else negative errno code.
  397. */
  398. int hal_dma2d_start(hal_dma2d_handle_t *hdma2d, uint32_t pdata, uint32_t dst_address, uint16_t width, uint16_t height)
  399. {
  400. return dma2d_set_config(hdma2d, pdata, 0, dst_address, width, height);
  401. }
  402. /**
  403. * @brief Start the multi-source DMA2D Transfer.
  404. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  405. * the configuration information for the DMA2D.
  406. * @param src_address1 The source memory Buffer address for the foreground layer.
  407. * @param src_address2 The source memory Buffer address for the background layer.
  408. * @param dst_address The destination memory Buffer address.
  409. * @param width The width of data to be transferred from source to destination (expressed in number of pixels per line).
  410. * @param height The height of data to be transferred from source to destination (expressed in number of lines).
  411. * @retval command sequence (uint16_t) on success else negative errno code.
  412. */
  413. int hal_dma2d_blending_start(hal_dma2d_handle_t *hdma2d, uint32_t src_address1, uint32_t src_address2, uint32_t dst_address, uint16_t width, uint16_t height)
  414. {
  415. return dma2d_set_config(hdma2d, src_address1, src_address2, dst_address, width, height);
  416. }
  417. /**
  418. * @brief Start the DMA2D Rotation Transfer with interrupt enabled.
  419. *
  420. * The source size must be square, and only the ring area defined in hal_dma2d_rotation_cfg_t
  421. * is rotated, and the pixels inside the inner ring can be filled with constant
  422. * color defined in hal_dma2d_rotation_cfg_t.
  423. *
  424. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  425. * the configuration information for the DMA2D.
  426. * @param src_address The source memory Buffer start address.
  427. * @param dst_address The destination memory Buffer address.
  428. * @param start_line The start line to rotate.
  429. * @param num_lines Number of lines to rotate.
  430. * @retval command sequence (uint16_t) on success else negative errno code.
  431. */
  432. int hal_dma2d_rotation_start(hal_dma2d_handle_t *hdma2d, uint32_t src_address, uint32_t dst_address, uint16_t start_line, uint16_t num_lines)
  433. {
  434. const struct device *dma2d_dev = dma2d_get_device();
  435. hal_dma2d_output_cfg_t *ouput_cfg = &hdma2d->output_cfg;
  436. hal_dma2d_rotation_cfg_t *rot_cfg = &hdma2d->rotation_cfg;
  437. display_engine_rotation_t *hw_cfg = &rot_cfg->hw_cfg;
  438. uint8_t bytes_per_pixel;
  439. int res = 0;
  440. /* Check DMAD enabled */
  441. if (global_en == false) {
  442. return -EBUSY;
  443. }
  444. /* Check the peripheral existence */
  445. if (dma2d_dev == NULL) {
  446. return -ENODEV;
  447. }
  448. if (rot_cfg->color_mode != ouput_cfg->color_mode ||
  449. rot_cfg->rb_swap != ouput_cfg->rb_swap) {
  450. return -EINVAL;
  451. }
  452. if (start_line + num_lines > hw_cfg->outer_diameter) {
  453. return -EINVAL;
  454. }
  455. if (buf_is_psram(src_address)) {
  456. src_address = (uint32_t)cache_to_uncache((void *)src_address);
  457. }
  458. if (buf_is_psram(dst_address)) {
  459. dst_address = (uint32_t)cache_to_uncache((void *)dst_address);
  460. }
  461. bytes_per_pixel = display_format_get_bits_per_pixel(hw_cfg->pixel_format) / 8;
  462. hw_cfg->src_pitch = (hw_cfg->outer_diameter + rot_cfg->input_offset) * bytes_per_pixel;
  463. hw_cfg->dst_pitch = (hw_cfg->outer_diameter + ouput_cfg->output_offset) * bytes_per_pixel;
  464. hw_cfg->dst_address = dst_address;
  465. /* Always compute the variables for hardware parallelism */
  466. /* if (start_line == 0 || start_line != hw_cfg->line_end) */ {
  467. hw_cfg->dst_dist_sq = (hw_cfg->outer_diameter - 1) * (hw_cfg->outer_diameter - 1) +
  468. (hw_cfg->outer_diameter - 1 - 2 * start_line) * (hw_cfg->outer_diameter - 1 - 2 * start_line);
  469. hw_cfg->src_coord_x = rot_cfg->src_coord_x0 + start_line * hw_cfg->src_coord_dx_ay;
  470. hw_cfg->src_coord_y = rot_cfg->src_coord_y0 + start_line * hw_cfg->src_coord_dy_ay;
  471. hw_cfg->src_address = src_address +
  472. FLOOR_FIXEDPOINT12(hw_cfg->src_coord_y) * hw_cfg->src_pitch +
  473. FLOOR_FIXEDPOINT12(hw_cfg->src_coord_x) * bytes_per_pixel;
  474. }
  475. hw_cfg->line_start = start_line;
  476. hw_cfg->line_end = start_line + num_lines;
  477. res = display_engine_rotate(dma2d_dev, hdma2d->instance, hw_cfg);
  478. if (res < 0) {
  479. /* Update error code */
  480. hdma2d->error_code |= HAL_DMA2D_ERROR_CE;
  481. } else {
  482. atomic_inc(&hdma2d->xfer_count);
  483. }
  484. return res;
  485. }
  486. /**
  487. * @brief Polling for transfer complete.
  488. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  489. * the configuration information for the DMA2D.
  490. * @param timeout timeout duration in milliseconds, if negative, means wait forever
  491. * @retval 0 on success else negative errno code.
  492. */
  493. int hal_dma2d_poll_transfer(hal_dma2d_handle_t *hdma2d, int32_t timeout)
  494. {
  495. const struct device *dma2d_dev = dma2d_get_device();
  496. int status = 0;
  497. /* Check the peripheral existence */
  498. if (dma2d_dev == NULL) {
  499. return -ENODEV;
  500. }
  501. if (HAL_DMA2D_STATE_BUSY != hal_dma2d_get_state(hdma2d)) {
  502. return 0;
  503. }
  504. if (display_engine_poll(dma2d_dev, hdma2d->instance, timeout)) {
  505. /* Update error code */
  506. hdma2d->error_code |= HAL_DMA2D_ERROR_TIMEOUT;
  507. /* update return status */
  508. status = -ETIME;
  509. }
  510. return status;
  511. }
  512. /**
  513. * @}
  514. */
  515. /** @defgroup DMA2D_Exported_Functions_Group3 Peripheral Control functions
  516. * @brief Peripheral Control functions
  517. *
  518. @verbatim
  519. ===============================================================================
  520. ##### Peripheral Control functions #####
  521. ===============================================================================
  522. [..] This section provides functions allowing to:
  523. (+) Configure the DMA2D transfer mode and output parameters.
  524. (+) Configure the DMA2D foreground or background layer parameters.
  525. (+) Configure the DMA2D rotation parameters.
  526. @endverbatim
  527. * @{
  528. */
  529. /**
  530. * @brief Configure the DMA2D transfer mode and output according to the
  531. * specified parameters in the hal_dma2d_handle_t.
  532. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  533. * the configuration information for the DMA2D.
  534. * @retval 0 on success else negative errno code.
  535. */
  536. int hal_dma2d_config_output(hal_dma2d_handle_t *hdma2d)
  537. {
  538. #if 0 /* These will be checked in dma2d_set_config(), so skip it here */
  539. const struct device *dma2d_dev = dma2d_get_device();
  540. /* Check the peripheral existence */
  541. if (dma2d_dev == NULL) {
  542. return -ENODEV;
  543. }
  544. /* Check the parameters */
  545. assert(IS_DMA2D_MODE(hdma2d->output_cfg.mode));
  546. assert(IS_DMA2D_OFFSET(hdma2d->output_cfg.output_offset));
  547. assert(IS_DMA2D_RB_SWAP(hdma2d->output_cfg.rb_swap));
  548. assert(IS_DMA2D_COLOR_MODE(hdma2d->output_cfg.color_mode));
  549. int32_t display_format = dma2d_get_display_format(hdma2d->output_cfg.color_mode, hdma2d->output_cfg.rb_swap);
  550. if (display_format < 0) {
  551. return -EINVAL;
  552. }
  553. struct display_engine_capabilities capabilities;
  554. display_engine_get_capabilities(dma2d_dev, &capabilities);
  555. if (hdma2d->output_cfg.mode == HAL_DMA2D_M2M_ROTATE) {
  556. if ((capabilities.supported_rotate_pixel_formats & display_format) == 0) {
  557. return -ENOTSUP;
  558. }
  559. } else {
  560. if ((capabilities.supported_output_pixel_formats & display_format) == 0) {
  561. return -ENOTSUP;
  562. }
  563. if (hdma2d->output_cfg.mode == HAL_DMA2D_M2M_BLEND_FG && capabilities.support_blend_fg == 0) {
  564. return -ENOTSUP;
  565. }
  566. if (hdma2d->output_cfg.mode == HAL_DMA2D_M2M_BLEND_BG && capabilities.support_blend_bg == 0) {
  567. return -ENOTSUP;
  568. }
  569. }
  570. #endif
  571. return 0;
  572. }
  573. /**
  574. * @brief Configure the DMA2D Layer according to the specified
  575. * parameters in the hal_dma2d_handle_t.
  576. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  577. * the configuration information for the DMA2D.
  578. * @param layer_idx DMA2D Layer index.
  579. * This parameter can be one of the following values:
  580. * HAL_DMA2D_BACKGROUND_LAYER(0) / HAL_DMA2D_FOREGROUND_LAYER(1)
  581. * @retval 0 on success else negative errno code.
  582. */
  583. int hal_dma2d_config_layer(hal_dma2d_handle_t *hdma2d, uint16_t layer_idx)
  584. {
  585. hal_dma2d_layer_cfg_t *cfg = &hdma2d->layer_cfg[layer_idx];
  586. #if 0 /* These will be checked in dma2d_set_config(), so skip it here */
  587. const struct device *dma2d_dev = dma2d_get_device();
  588. /* Check the peripheral existence */
  589. if (dma2d_dev == NULL) {
  590. return -ENODEV;
  591. }
  592. /* Check the parameters */
  593. assert(IS_DMA2D_LAYER(layer_idx));
  594. assert(IS_DMA2D_OFFSET(cfg->input_offset));
  595. assert(IS_DMA2D_COLOR_MODE(cfg->color_mode));
  596. int32_t display_format = dma2d_get_display_format(cfg->color_mode, cfg->rb_swap);
  597. if (display_format < 0) {
  598. return -EINVAL;
  599. }
  600. struct display_engine_capabilities capabilities;
  601. display_engine_get_capabilities(dma2d_dev, &capabilities);
  602. if ((capabilities.supported_input_pixel_formats & display_format) == 0) {
  603. return -ENOTSUP;
  604. }
  605. #endif
  606. if (layer_idx == HAL_DMA2D_FOREGROUND_LAYER) {
  607. if (cfg->alpha_mode == HAL_DMA2D_NO_MODIF_ALPHA) {
  608. cfg->input_alpha |= 0xff000000;
  609. } else if (cfg->alpha_mode == HAL_DMA2D_REPLACE_ALPHA) {
  610. return -ENOTSUP;
  611. }
  612. }
  613. return 0;
  614. }
  615. /**
  616. * @brief Configure the DMA2D Rotation according to the specified
  617. * parameters in the hal_dma2d_handle_t.
  618. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  619. * the configuration information for the DMA2D.
  620. * @retval 0 on success else negative errno code.
  621. */
  622. int hal_dma2d_config_rotation(hal_dma2d_handle_t *hdma2d)
  623. {
  624. hal_dma2d_rotation_cfg_t *cfg = &hdma2d->rotation_cfg;
  625. display_engine_rotation_t *hw_cfg = &cfg->hw_cfg;
  626. uint16_t revert_angle;
  627. int32_t center;
  628. /* Check the parameters */
  629. assert(IS_DMA2D_OFFSET(cfg->input_offset));
  630. if (cfg->angle >= 3600 || cfg->outer_diameter <= 0 ||
  631. cfg->inner_diameter >= cfg->outer_diameter) {
  632. return -EINVAL;
  633. }
  634. hw_cfg->pixel_format = dma2d_get_display_format(cfg->color_mode, cfg->rb_swap);
  635. hw_cfg->outer_diameter = cfg->outer_diameter;
  636. hw_cfg->outer_radius_sq = (cfg->outer_diameter - 1) * (cfg->outer_diameter - 1);
  637. hw_cfg->inner_radius_sq = cfg->inner_diameter > 0 ?
  638. (cfg->inner_diameter - 1) * (cfg->inner_diameter - 1) : 0;
  639. hw_cfg->fill_color.full = cfg->fill_color;
  640. hw_cfg->fill_enable = cfg->fill_enable;
  641. hw_cfg->line_start = 0;
  642. hw_cfg->line_end = 0;
  643. center = FIXEDPOINT12(hw_cfg->outer_diameter) / 2;
  644. revert_angle = 3600 - cfg->angle;
  645. /* coordinates in the destination coordinate system */
  646. cfg->src_coord_x0 = PX_FIXEDPOINT12(0);
  647. cfg->src_coord_y0 = PX_FIXEDPOINT12(0);
  648. hw_cfg->src_coord_dx_ax = FIXEDPOINT12(1);
  649. hw_cfg->src_coord_dy_ax = FIXEDPOINT12(0);
  650. hw_cfg->src_coord_dx_ay = FIXEDPOINT12(0);
  651. hw_cfg->src_coord_dy_ay = FIXEDPOINT12(1);
  652. /* rotate back to the source coordinate system */
  653. sw_rotate_point32(&cfg->src_coord_x0, &cfg->src_coord_y0,
  654. cfg->src_coord_x0, cfg->src_coord_y0, center,
  655. center, revert_angle);
  656. sw_rotate_point32(&hw_cfg->src_coord_dx_ax, &hw_cfg->src_coord_dy_ax,
  657. hw_cfg->src_coord_dx_ax, hw_cfg->src_coord_dy_ax,
  658. FIXEDPOINT12(0), FIXEDPOINT12(0), revert_angle);
  659. sw_rotate_point32(&hw_cfg->src_coord_dx_ay, &hw_cfg->src_coord_dy_ay,
  660. hw_cfg->src_coord_dx_ay, hw_cfg->src_coord_dy_ay,
  661. FIXEDPOINT12(0), FIXEDPOINT12(0), revert_angle);
  662. /* move to the source pixel coordinate system */
  663. cfg->src_coord_x0 -= PX_FIXEDPOINT12(0);
  664. cfg->src_coord_y0 -= PX_FIXEDPOINT12(0);
  665. return 0;
  666. }
  667. /**
  668. * @}
  669. */
  670. /** @defgroup DMA2D_Exported_Functions_Group4 Peripheral State and Error functions
  671. * @brief Peripheral State functions
  672. *
  673. @verbatim
  674. ===============================================================================
  675. ##### Peripheral State and Errors functions #####
  676. ===============================================================================
  677. [..]
  678. This subsection provides functions allowing to:
  679. (+) Get the DMA2D state
  680. (+) Get the DMA2D error code
  681. @endverbatim
  682. * @{
  683. */
  684. /**
  685. * @brief Return the DMA2D state
  686. * @param hdma2d pointer to a hal_dma2d_handle_t structure that contains
  687. * the configuration information for the DMA2D.
  688. * @retval HAL state
  689. */
  690. hal_dma2d_state_e hal_dma2d_get_state(hal_dma2d_handle_t *hdma2d)
  691. {
  692. return (atomic_get(&hdma2d->xfer_count) > 0) ?
  693. HAL_DMA2D_STATE_BUSY : HAL_DMA2D_STATE_READY;
  694. }
  695. /**
  696. * @brief Return the DMA2D error code
  697. * @param hdma2d pointer to a hal_dma2d_handle_t structure that contains
  698. * the configuration information for DMA2D.
  699. * @retval DMA2D Error Code
  700. */
  701. uint32_t hal_dma2d_get_error(hal_dma2d_handle_t *hdma2d)
  702. {
  703. uint32_t error_code = hdma2d->error_code;
  704. /* clear the error code */
  705. hdma2d->error_code = 0;
  706. return error_code;
  707. }
  708. /**
  709. * @}
  710. */
  711. /**
  712. * @}
  713. */
  714. /** @defgroup DMA2D_Private_Functions DMA2D Private Functions
  715. * @{
  716. */
  717. static void dma2d_set_dst_buffer(hal_dma2d_handle_t *hdma2d, display_buffer_t *dst, uint32_t dst_address, uint16_t width, uint16_t height)
  718. {
  719. dst->desc.width = width;
  720. dst->desc.height = height;
  721. dst->desc.pixel_format = dma2d_get_display_format(hdma2d->output_cfg.color_mode, hdma2d->output_cfg.rb_swap);
  722. dst->desc.pitch = width + hdma2d->output_cfg.output_offset;
  723. dst->addr = dst_address;
  724. }
  725. static void dma2d_set_src_buffer(hal_dma2d_handle_t *hdma2d, display_buffer_t *src, uint16_t layer_idx, uint32_t src_address, uint16_t width, uint16_t height)
  726. {
  727. src->desc.width = width;
  728. src->desc.height = height;
  729. src->desc.pixel_format = dma2d_get_display_format(hdma2d->layer_cfg[layer_idx].color_mode, hdma2d->layer_cfg[layer_idx].rb_swap),
  730. src->desc.pitch = width + hdma2d->layer_cfg[layer_idx].input_offset;
  731. src->addr = src_address;
  732. }
  733. /**
  734. * @brief Set the DMA2D transfer parameters.
  735. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  736. * the configuration information for the specified DMA2D.
  737. * @param src_address1 The source memory Buffer address for the foreground layer.
  738. * @param src_address2 The source memory Buffer address for the background layer.
  739. * @param dst_address The destination memory Buffer address.
  740. * @param width The width of data to be transferred from source to destination.
  741. * @param height The height of data to be transferred from source to destination.
  742. * @retval command sequence (uint16_t) on success else negative errno code.
  743. */
  744. static int dma2d_set_config(hal_dma2d_handle_t *hdma2d, uint32_t src_address1, uint32_t src_address2, uint32_t dst_address, uint16_t width, uint16_t height)
  745. {
  746. const struct device *dma2d_dev = dma2d_get_device();
  747. display_buffer_t dst;
  748. display_buffer_t fg;
  749. display_color_t fg_color;
  750. display_buffer_t bg;
  751. display_color_t bg_color;
  752. int res = 0;
  753. /* Check DMAD enabled */
  754. if (global_en == false) {
  755. return -EBUSY;
  756. }
  757. /* Check the peripheral existence */
  758. if (dma2d_dev == NULL) {
  759. return -ENODEV;
  760. }
  761. /* Check the parameters */
  762. assert(IS_DMA2D_LINE(height));
  763. assert(IS_DMA2D_PIXEL(width));
  764. dma2d_set_dst_buffer(hdma2d, &dst, dst_address, width, height);
  765. fg_color.full = hdma2d->layer_cfg[HAL_DMA2D_FOREGROUND_LAYER].input_alpha; /* ARGB8888 */
  766. bg_color.full = hdma2d->layer_cfg[HAL_DMA2D_BACKGROUND_LAYER].input_alpha; /* ARGB8888 */
  767. switch (hdma2d->output_cfg.mode) {
  768. case HAL_DMA2D_R2M:
  769. fg_color.full = src_address1;
  770. res = display_engine_fill(dma2d_dev, hdma2d->instance, &dst, fg_color);
  771. break;
  772. case HAL_DMA2D_M2M:
  773. dma2d_set_src_buffer(hdma2d, &fg, HAL_DMA2D_FOREGROUND_LAYER, src_address1, width, height);
  774. res = display_engine_blit(dma2d_dev, hdma2d->instance, &dst, &fg, fg_color);
  775. break;
  776. case HAL_DMA2D_M2M_BLEND:
  777. dma2d_set_src_buffer(hdma2d, &fg, HAL_DMA2D_FOREGROUND_LAYER, src_address1, width, height);
  778. dma2d_set_src_buffer(hdma2d, &bg, HAL_DMA2D_BACKGROUND_LAYER, src_address2, width, height);
  779. res = display_engine_blend(dma2d_dev, hdma2d->instance, &dst, &fg, fg_color, &bg, bg_color);
  780. break;
  781. case HAL_DMA2D_M2M_BLEND_FG:
  782. fg_color.full = src_address1;
  783. dma2d_set_src_buffer(hdma2d, &bg, HAL_DMA2D_BACKGROUND_LAYER, src_address2, width, height);
  784. res = display_engine_blend(dma2d_dev, hdma2d->instance, &dst, NULL, fg_color, &bg, bg_color);
  785. break;
  786. case HAL_DMA2D_M2M_BLEND_BG:
  787. bg_color.full = src_address2;
  788. dma2d_set_src_buffer(hdma2d, &fg, HAL_DMA2D_FOREGROUND_LAYER, src_address1, width, height);
  789. res = display_engine_blend(dma2d_dev, hdma2d->instance, &dst, &fg, fg_color, NULL, bg_color);
  790. break;
  791. default:
  792. res = -EINVAL;
  793. break;
  794. }
  795. if (res < 0) {
  796. /* Update error code */
  797. hdma2d->error_code |= HAL_DMA2D_ERROR_CE;
  798. } else {
  799. atomic_inc(&hdma2d->xfer_count);
  800. }
  801. return res;
  802. }
  803. /**
  804. * @brief Global enable/disable DMAD functions
  805. * @param enabled enable or not
  806. * @retval N/A
  807. */
  808. void hal_dma2d_set_global_enabled(bool enabled)
  809. {
  810. global_en = enabled;
  811. }
  812. /**
  813. * @}
  814. */
  815. /**
  816. * @}
  817. */