dma2d_hal.c 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927
  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_transform.
  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. (#) Configure pdata parameter, destination and data length and enable
  43. the transfer using hal_dma2d_start().
  44. (#) At the end of data transfer dma2d_device_handler() function is executed and user can
  45. add his own function by customization of function pointer xfer_callback (member
  46. of DMA2D handle structure).
  47. (#) In case of error, the dma2d_device_handler() function calls the callback
  48. xfer_error_callback.
  49. -@- In Register-to-Memory transfer mode, pdata parameter is the register
  50. color, in Memory-to-memory or Memory-to-Memory with pixel format
  51. conversion pdata is the source address.
  52. -@- Configure the foreground source address, the background source address,
  53. the destination and data length then Enable the transfer using
  54. hal_dma2d_blending_start() either in polling mode or interrupt mode.
  55. -@- hal_dma2d_blending_start() function is used if the memory to memory
  56. with blending transfer mode is selected.
  57. -@- hal_dma2d_transform_start() function is used if the memory to memory
  58. with rotation transfer mode is selected.
  59. -@- hal_dma2d_clut_load_start() function is used if the layer's color lookup table
  60. is required to update.
  61. (#) To control the DMA2D state, use the following function: hal_dma2d_get_state().
  62. (#) To read the DMA2D error code, use the following function: hal_dma2d_get_error().
  63. *** Callback registration ***
  64. ===================================
  65. [..]
  66. (#) Use function @ref hal_dma2d_register_callback() to register a user callback.
  67. (#) Function @ref hal_dma2d_register_callback() allows to register the callbacks:
  68. This function takes as parameters the HAL peripheral handle
  69. and a pointer to the user callback function.
  70. (#) Use function @ref hal_dma2d_unregister_callback() to reset a callback to the default
  71. weak (surcharged) function.
  72. [..]
  73. (@) You can refer to the DMA2D HAL driver header file for more useful macros
  74. @endverbatim
  75. */
  76. /* Includes ------------------------------------------------------------------*/
  77. #include <string.h>
  78. #include <assert.h>
  79. #include <display/sw_math.h>
  80. #include <dma2d_hal.h>
  81. #include <board_cfg.h>
  82. #include <linker/linker-defs.h>
  83. #include <tracing/tracing.h>
  84. /** @defgroup DMA2D DMA2D
  85. * @brief DMA2D HAL module driver
  86. * @{
  87. */
  88. /* Private types -------------------------------------------------------------*/
  89. /* Private define ------------------------------------------------------------*/
  90. /** @defgroup DMA2D_Private_Constants DMA2D Private Constants
  91. * @{
  92. */
  93. #define FULL_DEV_IDX 0
  94. #define LITE_DEV_IDX 1
  95. /**
  96. * @}
  97. */
  98. /* Private variables ---------------------------------------------------------*/
  99. /* DMA2D global enable bit */
  100. static bool global_en = true;
  101. /* DMA2D device initialize or not ? */
  102. static bool dma2d_dev_initialized = false;
  103. /* DMA2D device handle */
  104. static const struct device *dma2d_dev[2];
  105. static struct display_engine_capabilities dma2d_cap[2];
  106. /* Private constants ---------------------------------------------------------*/
  107. /* Private macro -------------------------------------------------------------*/
  108. #define IS_DMA2D_LAYER(layer) (((layer) == HAL_DMA2D_BACKGROUND_LAYER) || ((layer) == HAL_DMA2D_FOREGROUND_LAYER))
  109. #define IS_DMA2D_MODE(mode) (((mode) == HAL_DMA2D_R2M) || ((mode) == HAL_DMA2D_M2M) || \
  110. ((mode) == HAL_DMA2D_M2M_BLEND) || ((mode) == HAL_DMA2D_M2M_BLEND_FG) || \
  111. ((mode) == HAL_DMA2D_M2M_BLEND_BG) || ((mode) == HAL_DMA2D_M2M_TRANSFORM))
  112. #define IS_DMA2D_LINE(line) ((line) <= HAL_DMA2D_LINE)
  113. #define IS_DMA2D_PIXEL(pixel) ((pixel) <= HAL_DMA2D_PIXEL)
  114. #define IS_DMA2D_PITCH(pitch) ((pitch) <= HAL_DMA2D_PITCH)
  115. #define IS_DMA2D_ALPHA_MODE(alpha_mode) (((alpha_mode) == HAL_DMA2D_NO_MODIF_ALPHA) || \
  116. ((alpha_mode) == HAL_DMA2D_REPLACE_ALPHA) || \
  117. ((alpha_mode) == HAL_DMA2D_COMBINE_ALPHA))
  118. /* Private function prototypes -----------------------------------------------*/
  119. /** @addtogroup DMA2D_Private_Functions DMA2D Private Functions
  120. * @{
  121. */
  122. __de_func static int dma2d_set_config(hal_dma2d_handle_t *hdma2d, uint32_t fg_address,
  123. uint32_t bg_address, uint32_t dst_address, uint32_t width,
  124. uint32_t height);
  125. __de_func static void dma2d_set_dst_buffer(hal_dma2d_handle_t *hdma2d, display_buffer_t *dst,
  126. uint32_t dst_address, uint32_t width, uint32_t height);
  127. __de_func static void dma2d_set_src_buffer(hal_dma2d_handle_t *hdma2d, display_buffer_t *src,
  128. uint16_t layer_idx, uint32_t src_address);
  129. /**
  130. * @}
  131. */
  132. /* Private functions ---------------------------------------------------------*/
  133. /**
  134. * @brief Get the DMA2D device
  135. * @retval Pointer to the DMA2D device structure
  136. */
  137. static const struct device *dma2d_get_device(uint32_t preferred_modes)
  138. {
  139. const uint32_t FULL_FEATURE_MODES = HAL_DMA2D_M2M_BLEND |
  140. HAL_DMA2D_M2M_BLEND_FG | HAL_DMA2D_M2M_BLEND_BG |
  141. HAL_DMA2D_M2M_TRANSFORM | HAL_DMA2D_M2M_TRANSFORM_BLEND |
  142. HAL_DMA2D_M2M_TRANSFORM_CIRCLE;
  143. const struct device *device = NULL;
  144. if (dma2d_dev_initialized == false) {
  145. dma2d_dev_initialized = true;
  146. dma2d_dev[FULL_DEV_IDX] = device_get_binding(CONFIG_DISPLAY_ENGINE_DEV_NAME);
  147. if (dma2d_dev[FULL_DEV_IDX])
  148. display_engine_get_capabilities(dma2d_dev[FULL_DEV_IDX], &dma2d_cap[FULL_DEV_IDX]);
  149. dma2d_dev[LITE_DEV_IDX] = device_get_binding(CONFIG_DMA2D_LITE_DEV_NAME);
  150. if (dma2d_dev[LITE_DEV_IDX])
  151. display_engine_get_capabilities(dma2d_dev[LITE_DEV_IDX], &dma2d_cap[LITE_DEV_IDX]);
  152. }
  153. if (!(preferred_modes & FULL_FEATURE_MODES) && dma2d_dev[LITE_DEV_IDX]) {
  154. if (!(preferred_modes & HAL_DMA2D_R2M) || dma2d_cap[LITE_DEV_IDX].support_fill) {
  155. device = dma2d_dev[LITE_DEV_IDX];
  156. }
  157. }
  158. if (device == NULL) {
  159. device = dma2d_dev[FULL_DEV_IDX];
  160. }
  161. assert(device != NULL);
  162. return device;
  163. }
  164. static bool is_dma2d_device_ready(const struct device *device)
  165. {
  166. if (device == NULL) {
  167. return false;
  168. }
  169. if (!global_en && device == dma2d_dev[FULL_DEV_IDX]) {
  170. return false;
  171. }
  172. return true;
  173. }
  174. /* Exported functions --------------------------------------------------------*/
  175. /** @defgroup DMA2D_Exported_Functions DMA2D Exported Functions
  176. * @{
  177. */
  178. /** @defgroup DMA2D_Exported_Functions_Group1 Initialization and de-initialization functions
  179. * @brief Initialization and Configuration functions
  180. *
  181. @verbatim
  182. ===============================================================================
  183. ##### Initialization and Configuration functions #####
  184. ===============================================================================
  185. [..] This section provides functions allowing to:
  186. (+) Initialize and configure the DMA2D
  187. (+) De-initialize the DMA2D
  188. @endverbatim
  189. * @{
  190. */
  191. /**
  192. * @brief DMA2D callback from device driver, may called in interrupt routine
  193. * @param err_code DMA2D device error code
  194. * @param cmd_seq Current DMA2D device command sequence
  195. * @param user_data Address of structure hal_dma2d_handle_t
  196. */
  197. __de_func void dma2d_device_handler(int err_code, uint16_t cmd_seq, void *user_data)
  198. {
  199. hal_dma2d_handle_t *hdma2d = user_data;
  200. uint32_t error_code;
  201. /* Update error code */
  202. error_code = (err_code != 0) ? HAL_DMA2D_ERROR_TIMEOUT : HAL_DMA2D_ERROR_NONE;
  203. hdma2d->error_code |= error_code;
  204. if (hdma2d->xfer_callback != NULL) {
  205. /* Transfer error Callback */
  206. hdma2d->xfer_callback(hdma2d, cmd_seq, error_code);
  207. }
  208. atomic_dec(&hdma2d->xfer_count);
  209. }
  210. /**
  211. * @brief Initialize the DMA2D peripheral and create the associated handle.
  212. * @param hdma2d pointer to a hal_dma2d_handle_t structure that contains
  213. * the configuration information for the DMA2D.
  214. * @param preferred_modes "bitwise or" of output modes that maybe used.
  215. * @retval 0 on success else negative errno code.
  216. */
  217. int hal_dma2d_init(hal_dma2d_handle_t *hdma2d, uint32_t preferred_modes)
  218. {
  219. /* Check the Parameter */
  220. if (hdma2d == NULL) {
  221. return -EINVAL;
  222. }
  223. /* Initialize the handle */
  224. memset(hdma2d, 0, sizeof(*hdma2d));
  225. /* Check the DMA2D peripheral existence */
  226. hdma2d->device = dma2d_get_device(preferred_modes);
  227. if (hdma2d->device == NULL) {
  228. return -ENODEV;
  229. }
  230. /* Open the DMA2D device instance */
  231. hdma2d->instance = display_engine_open(hdma2d->device, 0);
  232. if (hdma2d->instance < 0) {
  233. return -EBUSY;
  234. }
  235. /* Register the DMA2D device instance callback */
  236. display_engine_register_callback(hdma2d->device, hdma2d->instance, dma2d_device_handler, hdma2d);
  237. atomic_set(&hdma2d->xfer_count, 0);
  238. /* Update error code */
  239. hdma2d->error_code = HAL_DMA2D_ERROR_NONE;
  240. return 0;
  241. }
  242. /**
  243. * @brief Deinitializes the DMA2D peripheral registers to their default reset
  244. * values.
  245. * @param hdma2d pointer to a hal_dma2d_handle_t structure that contains
  246. * the configuration information for the DMA2D.
  247. * @retval 0 on success else negative errno code.
  248. */
  249. int hal_dma2d_deinit(hal_dma2d_handle_t *hdma2d)
  250. {
  251. /* Check the DMA2D peripheral State */
  252. if (hdma2d == NULL || hdma2d->device == NULL || hdma2d->instance < 0) {
  253. return -EINVAL;
  254. }
  255. /* Close the DMA2D device instance */
  256. display_engine_close(hdma2d->device, hdma2d->instance);
  257. /* Assign invald value */
  258. hdma2d->device = NULL;
  259. hdma2d->instance = -1;
  260. return 0;
  261. }
  262. /**
  263. * @brief Register a User DMA2D Callback
  264. * To be used instead of the weak (surcharged) predefined callback
  265. * @param hdma2d DMA2D handle
  266. * @param callback_fn pointer to the callback function
  267. * @retval 0 on success else negative errno code.
  268. */
  269. int hal_dma2d_register_callback(hal_dma2d_handle_t *hdma2d, hal_dma2d_callback_t callback_fn)
  270. {
  271. int status = 0;
  272. if (callback_fn == NULL) {
  273. /* Update the error code */
  274. hdma2d->error_code |= HAL_DMA2D_ERROR_INVALID_CALLBACK;
  275. return -EINVAL;
  276. }
  277. if (HAL_DMA2D_STATE_READY == hal_dma2d_get_state(hdma2d)) {
  278. hdma2d->xfer_callback = callback_fn;
  279. } else {
  280. /* Update the error code */
  281. hdma2d->error_code |= HAL_DMA2D_ERROR_INVALID_CALLBACK;
  282. /* update return status */
  283. status = -EBUSY;
  284. }
  285. return status;
  286. }
  287. /**
  288. * @brief Unregister a DMA2D Callback
  289. * DMA2D Callback is redirected to the weak (surcharged) predefined callback
  290. * @param hdma2d DMA2D handle
  291. * @retval 0 on success else negative errno code.
  292. */
  293. int hal_dma2d_unregister_callback(hal_dma2d_handle_t *hdma2d)
  294. {
  295. int status = 0;
  296. if (HAL_DMA2D_STATE_READY == hal_dma2d_get_state(hdma2d)) {
  297. hdma2d->xfer_callback = NULL;
  298. } else {
  299. /* Update the error code */
  300. hdma2d->error_code |= HAL_DMA2D_ERROR_INVALID_CALLBACK;
  301. /* update return status */
  302. status = -EBUSY;
  303. }
  304. return status;
  305. }
  306. /**
  307. * @}
  308. */
  309. /** @defgroup DMA2D_Exported_Functions_Group2 IO operation functions
  310. * @brief IO operation functions
  311. *
  312. @verbatim
  313. ===============================================================================
  314. ##### IO operation functions #####
  315. ===============================================================================
  316. [..] This section provides functions allowing to:
  317. (+) Configure the pdata, destination address and data size then
  318. start the DMA2D transfer.
  319. (+) Configure the source for foreground and background, destination address
  320. and data size then start a MultiBuffer DMA2D transfer.
  321. (+) Configure the pdata, destination address and data size then
  322. start the DMA2D transfer with interrupt.
  323. (+) Configure the source for foreground and background, destination address
  324. and data size then start a MultiBuffer DMA2D transfer with interrupt.
  325. (+) Poll for transfer complete.
  326. (+) handle DMA2D interrupt request.
  327. @endverbatim
  328. * @{
  329. */
  330. /**
  331. * @brief Start the DMA2D Transfer.
  332. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  333. * the configuration information for the DMA2D.
  334. * @param pdata Configure the source memory Buffer address if
  335. * Memory-to-Memory or Memory-to-Memory with pixel format
  336. * conversion mode is selected, or configure
  337. * the color value if Register-to-Memory mode is selected.
  338. * @param dst_address The destination memory Buffer address.
  339. * @param width The width of data to be transferred from source to destination (expressed in number of pixels per line).
  340. * @param height The height of data to be transferred from source to destination (expressed in number of lines).
  341. * @retval command sequence (uint16_t) on success else negative errno code.
  342. */
  343. __de_func int hal_dma2d_start(hal_dma2d_handle_t *hdma2d, uint32_t pdata, uint32_t dst_address,
  344. uint32_t width, uint32_t height)
  345. {
  346. return dma2d_set_config(hdma2d, pdata, 0, dst_address, width, height);
  347. }
  348. /**
  349. * @brief Start the multi-source DMA2D Transfer.
  350. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  351. * the configuration information for the DMA2D.
  352. * @param fg_address The source memory Buffer address for the foreground layer.
  353. * @param bg_address The source memory Buffer address for the background layer.
  354. * @param dst_address The destination memory Buffer address.
  355. * @param width The width of data to be transferred from source to destination (expressed in number of pixels per line).
  356. * @param height The height of data to be transferred from source to destination (expressed in number of lines).
  357. * @retval command sequence (uint16_t) on success else negative errno code.
  358. */
  359. __de_func int hal_dma2d_blending_start(hal_dma2d_handle_t *hdma2d, uint32_t fg_address,
  360. uint32_t bg_address, uint32_t dst_address, uint32_t width,
  361. uint32_t height)
  362. {
  363. return dma2d_set_config(hdma2d, fg_address, bg_address, dst_address, width, height);
  364. }
  365. /**
  366. * @brief Start the DMA2D Rotation Transfer with interrupt enabled.
  367. *
  368. * The source size must be square, and only the ring area defined in hal_dma2d_rotation_cfg_t
  369. * is rotated, and the pixels inside the inner ring can be filled with constant
  370. * color defined in hal_dma2d_rotation_cfg_t.
  371. *
  372. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  373. * the configuration information for the DMA2D.
  374. * @param src_address The source memory Buffer start address.
  375. * @param dst_address The destination memory Buffer address.
  376. * @param x X coord of top-left corner of dest area.
  377. * @param y Y coord of top-left corner of dest area.
  378. * @param width Width of dest area.
  379. * @param height Height of dest area.
  380. * @retval command sequence (uint16_t) on success else negative errno code.
  381. */
  382. int hal_dma2d_transform_start(hal_dma2d_handle_t *hdma2d, uint32_t src_address, uint32_t dst_address,
  383. int16_t x, int16_t y, uint16_t width, uint16_t height)
  384. {
  385. hal_dma2d_transform_cfg_t *cfg = &hdma2d->trans_cfg;
  386. display_engine_transform_param_t *hw_param = &cfg->hw_param;
  387. display_buffer_t dst, src;
  388. int32_t sw_x0, sw_y0;
  389. int res = 0;
  390. /* Check the peripheral existence */
  391. if (!is_dma2d_device_ready(hdma2d->device) || hdma2d->instance < 0) {
  392. return -ENODEV;
  393. }
  394. dma2d_set_dst_buffer(hdma2d, &dst, dst_address, width, height);
  395. dma2d_set_src_buffer(hdma2d, &src, HAL_DMA2D_FOREGROUND_LAYER, src_address);
  396. if (hdma2d->output_cfg.mode == HAL_DMA2D_M2M_TRANSFORM_CIRCLE) {
  397. if (x != cfg->image_x0 || width != cfg->circle.outer_diameter + 1)
  398. return -EINVAL;
  399. if (y < cfg->image_y0 || y + height > cfg->image_x0 + cfg->circle.outer_diameter + 1)
  400. return -EINVAL;
  401. if (src.desc.pixel_format != dst.desc.pixel_format ||
  402. src.desc.width != dst.desc.width ||
  403. src.desc.width != src.desc.height)
  404. return -EINVAL;
  405. hw_param->circle.line_start = y;
  406. }
  407. x -= cfg->image_x0;
  408. y -= cfg->image_y0;
  409. sw_x0 = hw_param->matrix.tx;
  410. sw_y0 = hw_param->matrix.ty;
  411. hw_param->matrix.tx += y * hw_param->matrix.shx + x * hw_param->matrix.sx;
  412. hw_param->matrix.ty += y * hw_param->matrix.sy + x * hw_param->matrix.shy;
  413. hw_param->color.full = hdma2d->layer_cfg[HAL_DMA2D_FOREGROUND_LAYER].input_alpha;
  414. res = display_engine_transform(hdma2d->device, hdma2d->instance, &dst, &src, hw_param);
  415. if (res < 0) {
  416. /* Update error code */
  417. hdma2d->error_code |= HAL_DMA2D_ERROR_CE;
  418. } else {
  419. atomic_inc(&hdma2d->xfer_count);
  420. }
  421. hw_param->matrix.tx = sw_x0;
  422. hw_param->matrix.ty = sw_y0;
  423. return res;
  424. }
  425. /**
  426. * @brief Start DMA2D CLUT Loading.
  427. * @param hdma2d Pointer to a DMA2D_HandleTypeDef structure that contains
  428. * the configuration information for the DMA2D.
  429. * @param layer_idx DMA2D Layer index.
  430. * This parameter can be one of the following values:
  431. * DMA2D_BACKGROUND_LAYER(0) / DMA2D_FOREGROUND_LAYER(1)
  432. * @param size Number of colors in the color look up table
  433. * @param clut Pointer to the color look up table.
  434. * @retval command sequence (uint16_t) on success else negative errno code.
  435. */
  436. int hal_dma2d_clut_load_start(hal_dma2d_handle_t *hdma2d, uint16_t layer_idx, uint16_t size, const uint32_t *clut)
  437. {
  438. int res = 0;
  439. /* Check the peripheral existence */
  440. if (!is_dma2d_device_ready(hdma2d->device) || hdma2d->instance < 0) {
  441. return -ENODEV;
  442. }
  443. if (layer_idx > HAL_DMA2D_FOREGROUND_LAYER) {
  444. return -EINVAL;
  445. }
  446. res = display_engine_set_clut(hdma2d->device, hdma2d->instance, layer_idx + 1, size, clut);
  447. if (res < 0) {
  448. /* Update error code */
  449. hdma2d->error_code |= HAL_DMA2D_ERROR_CE;
  450. } else {
  451. atomic_inc(&hdma2d->xfer_count);
  452. }
  453. return res;
  454. }
  455. /**
  456. * @brief Polling for transfer complete.
  457. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  458. * the configuration information for the DMA2D.
  459. * @param timeout timeout duration in milliseconds, if negative, means wait forever
  460. * @retval 0 on success else negative errno code.
  461. */
  462. int hal_dma2d_poll_transfer(hal_dma2d_handle_t *hdma2d, int32_t timeout)
  463. {
  464. int status = 0;
  465. /* Check the peripheral existence */
  466. if (hdma2d->device == NULL || hdma2d->instance < 0) {
  467. return -ENODEV;
  468. }
  469. if (HAL_DMA2D_STATE_BUSY != hal_dma2d_get_state(hdma2d)) {
  470. return 0;
  471. }
  472. if (display_engine_poll(hdma2d->device, hdma2d->instance, timeout)) {
  473. /* Update error code */
  474. hdma2d->error_code |= HAL_DMA2D_ERROR_TIMEOUT;
  475. /* update return status */
  476. status = -ETIME;
  477. }
  478. return status;
  479. }
  480. /**
  481. * @}
  482. */
  483. /** @defgroup DMA2D_Exported_Functions_Group3 Peripheral Control functions
  484. * @brief Peripheral Control functions
  485. *
  486. @verbatim
  487. ===============================================================================
  488. ##### Peripheral Control functions #####
  489. ===============================================================================
  490. [..] This section provides functions allowing to:
  491. (+) Configure the DMA2D transfer mode and output parameters.
  492. (+) Configure the DMA2D foreground or background layer parameters.
  493. (+) Configure the DMA2D rotation parameters.
  494. @endverbatim
  495. * @{
  496. */
  497. /**
  498. * @brief Configure the DMA2D transfer mode and output according to the
  499. * specified parameters in the hal_dma2d_handle_t.
  500. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  501. * the configuration information for the DMA2D.
  502. * @retval 0 on success else negative errno code.
  503. */
  504. __de_func int hal_dma2d_config_output(hal_dma2d_handle_t *hdma2d)
  505. {
  506. return 0;
  507. }
  508. /**
  509. * @brief Configure the DMA2D Layer according to the specified
  510. * parameters in the hal_dma2d_handle_t.
  511. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  512. * the configuration information for the DMA2D.
  513. * @param layer_idx DMA2D Layer index.
  514. * This parameter can be one of the following values:
  515. * HAL_DMA2D_BACKGROUND_LAYER(0) / HAL_DMA2D_FOREGROUND_LAYER(1)
  516. * @retval 0 on success else negative errno code.
  517. */
  518. __de_func int hal_dma2d_config_layer(hal_dma2d_handle_t *hdma2d, uint16_t layer_idx)
  519. {
  520. hal_dma2d_layer_cfg_t *cfg = &hdma2d->layer_cfg[layer_idx];
  521. if (layer_idx == HAL_DMA2D_FOREGROUND_LAYER) {
  522. if (cfg->alpha_mode == HAL_DMA2D_NO_MODIF_ALPHA) {
  523. cfg->input_alpha |= 0xff000000;
  524. } else if (cfg->alpha_mode == HAL_DMA2D_REPLACE_ALPHA) {
  525. return -ENOTSUP;
  526. }
  527. }
  528. return 0;
  529. }
  530. /**
  531. * @brief Configure the DMA2D Rotation according to the specified
  532. * parameters in the hal_dma2d_handle_t.
  533. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  534. * the configuration information for the DMA2D.
  535. * @retval 0 on success else negative errno code.
  536. */
  537. int hal_dma2d_config_transform(hal_dma2d_handle_t *hdma2d)
  538. {
  539. hal_dma2d_transform_cfg_t *cfg = &hdma2d->trans_cfg;
  540. display_engine_transform_param_t *hw_param = &cfg->hw_param;
  541. uint16_t revert_angle, revert_scale_x, revert_scale_y;
  542. int32_t pivot_x, pivot_y;
  543. int32_t width = hdma2d->layer_cfg[HAL_DMA2D_FOREGROUND_LAYER].input_width;
  544. int32_t height = hdma2d->layer_cfg[HAL_DMA2D_FOREGROUND_LAYER].input_height;
  545. /* Just assume the top-left coordinate of the source before transformation
  546. * is (0, 0) in the display coordinate system.
  547. */
  548. switch (cfg->mode) {
  549. case HAL_DMA2D_ROT_90:
  550. hw_param->matrix = (display_matrix_t) {
  551. .sx = 0, .shx = FIXEDPOINT12(1), .tx = 0,
  552. .shy = FIXEDPOINT12(-1), .sy = 0, .ty = FIXEDPOINT12(height - 1),
  553. };
  554. return 0;
  555. case HAL_DMA2D_ROT_180:
  556. hw_param->matrix = (display_matrix_t) {
  557. .sx = FIXEDPOINT12(-1), .shx = 0, .tx = FIXEDPOINT12(width - 1),
  558. .shy = 0, .sy = FIXEDPOINT12(-1), .ty = FIXEDPOINT12(height - 1),
  559. };
  560. return 0;
  561. case HAL_DMA2D_ROT_270:
  562. hw_param->matrix = (display_matrix_t) {
  563. .sx = 0, .shx = FIXEDPOINT12(-1), .tx = FIXEDPOINT12(width - 1),
  564. .shy = FIXEDPOINT12(1), .sy = 0, .ty = 0,
  565. };
  566. return 0;
  567. case HAL_DMA2D_FLIP_H:
  568. hw_param->matrix = (display_matrix_t) {
  569. .sx = FIXEDPOINT12(-1), .shx = 0, .tx = FIXEDPOINT12(width - 1),
  570. .shy = 0, .sy = FIXEDPOINT12(1), .ty = 0,
  571. };
  572. return 0;
  573. case HAL_DMA2D_FLIP_V:
  574. hw_param->matrix = (display_matrix_t) {
  575. .sx = FIXEDPOINT12(1), .shx = 0, .tx = 0,
  576. .shy = 0, .sy = FIXEDPOINT12(-1), .ty = FIXEDPOINT12(height - 1),
  577. };
  578. return 0;
  579. case HAL_DMA2D_ROT2D:
  580. default:
  581. break;
  582. }
  583. /* Check the parameters */
  584. if (cfg->angle >= 3600) {
  585. return -EINVAL;
  586. }
  587. hw_param->blend_en = (hdma2d->output_cfg.mode == HAL_DMA2D_M2M_TRANSFORM_BLEND);
  588. if (hdma2d->output_cfg.mode== HAL_DMA2D_M2M_TRANSFORM_CIRCLE) {
  589. if (cfg->circle.outer_diameter <= 0 ||
  590. cfg->circle.inner_diameter >= cfg->circle.outer_diameter) {
  591. return -EINVAL;
  592. }
  593. hw_param->is_circle = 1;
  594. hw_param->circle.outer_radius_sq = cfg->circle.outer_diameter * cfg->circle.outer_diameter;
  595. hw_param->circle.inner_radius_sq = cfg->circle.inner_diameter * cfg->circle.inner_diameter;
  596. hw_param->circle.line_start = 0;
  597. pivot_x = FIXEDPOINT12(cfg->circle.outer_diameter + 1) / 2 + PX_FIXEDPOINT12(0);
  598. pivot_y = FIXEDPOINT12(cfg->circle.outer_diameter + 1) / 2 + PX_FIXEDPOINT12(0);
  599. revert_scale_x = revert_scale_y = HAL_DMA2D_SCALE_NONE; /* no scale */
  600. } else {
  601. /* only support scaling up */
  602. if (cfg->rect.scale_x < HAL_DMA2D_SCALE_NONE ||
  603. cfg->rect.scale_y < HAL_DMA2D_SCALE_NONE) {
  604. return -EINVAL;
  605. }
  606. hw_param->is_circle = 0;
  607. pivot_x = PX_FIXEDPOINT12(cfg->rect.pivot_x);
  608. pivot_y = PX_FIXEDPOINT12(cfg->rect.pivot_y);
  609. revert_scale_x = HAL_DMA2D_SCALE_NONE * HAL_DMA2D_SCALE_NONE / cfg->rect.scale_x;
  610. revert_scale_y = HAL_DMA2D_SCALE_NONE * HAL_DMA2D_SCALE_NONE / cfg->rect.scale_y;
  611. }
  612. revert_angle = 3600 - cfg->angle;
  613. /* coordinates in the destination coordinate system */
  614. hw_param->matrix.tx = PX_FIXEDPOINT12(0);
  615. hw_param->matrix.ty = PX_FIXEDPOINT12(0);
  616. hw_param->matrix.sx = FIXEDPOINT12(1);
  617. hw_param->matrix.shy = FIXEDPOINT12(0);
  618. hw_param->matrix.shx = FIXEDPOINT12(0);
  619. hw_param->matrix.sy = FIXEDPOINT12(1);
  620. /* rotate back to the source coordinate system */
  621. sw_transform_point32_rot_first(&hw_param->matrix.tx, &hw_param->matrix.ty,
  622. hw_param->matrix.tx, hw_param->matrix.ty,
  623. pivot_x, pivot_y, revert_angle, revert_scale_x, revert_scale_y, 8);
  624. sw_transform_point32_rot_first(&hw_param->matrix.sx, &hw_param->matrix.shy,
  625. hw_param->matrix.sx, hw_param->matrix.shy,
  626. FIXEDPOINT12(0), FIXEDPOINT12(0), revert_angle, revert_scale_x, revert_scale_y, 8);
  627. sw_transform_point32_rot_first(&hw_param->matrix.shx, &hw_param->matrix.sy,
  628. hw_param->matrix.shx, hw_param->matrix.sy,
  629. FIXEDPOINT12(0), FIXEDPOINT12(0), revert_angle, revert_scale_x, revert_scale_y, 8);
  630. /* move to the source pixel coordinate system */
  631. hw_param->matrix.tx -= PX_FIXEDPOINT12(0);
  632. hw_param->matrix.ty -= PX_FIXEDPOINT12(0);
  633. return 0;
  634. }
  635. /**
  636. * @}
  637. */
  638. /** @defgroup DMA2D_Exported_Functions_Group4 Peripheral State and Error functions
  639. * @brief Peripheral State functions
  640. *
  641. @verbatim
  642. ===============================================================================
  643. ##### Peripheral State and Errors functions #####
  644. ===============================================================================
  645. [..]
  646. This subsection provides functions allowing to:
  647. (+) Get the DMA2D state
  648. (+) Get the DMA2D error code
  649. @endverbatim
  650. * @{
  651. */
  652. /**
  653. * @brief Return the DMA2D state
  654. * @param hdma2d pointer to a hal_dma2d_handle_t structure that contains
  655. * the configuration information for the DMA2D.
  656. * @retval HAL state
  657. */
  658. hal_dma2d_state_e hal_dma2d_get_state(hal_dma2d_handle_t *hdma2d)
  659. {
  660. return (atomic_get(&hdma2d->xfer_count) > 0) ?
  661. HAL_DMA2D_STATE_BUSY : HAL_DMA2D_STATE_READY;
  662. }
  663. /**
  664. * @brief Return the DMA2D error code
  665. * @param hdma2d pointer to a hal_dma2d_handle_t structure that contains
  666. * the configuration information for DMA2D.
  667. * @retval DMA2D Error Code
  668. */
  669. uint32_t hal_dma2d_get_error(hal_dma2d_handle_t *hdma2d)
  670. {
  671. uint32_t error_code = hdma2d->error_code;
  672. /* clear the error code */
  673. hdma2d->error_code = 0;
  674. return error_code;
  675. }
  676. /**
  677. * @}
  678. */
  679. /**
  680. * @}
  681. */
  682. /** @defgroup DMA2D_Private_Functions DMA2D Private Functions
  683. * @{
  684. */
  685. __de_func static void dma2d_set_dst_buffer(hal_dma2d_handle_t *hdma2d, display_buffer_t *dst,
  686. uint32_t dst_address, uint32_t width, uint32_t height)
  687. {
  688. dst->desc.width = width;
  689. dst->desc.height = height;
  690. dst->desc.pixel_format = hdma2d->output_cfg.color_format;
  691. dst->desc.pitch = hdma2d->output_cfg.output_pitch;
  692. dst->addr = dst_address;
  693. }
  694. __de_func static void dma2d_set_src_buffer(hal_dma2d_handle_t *hdma2d, display_buffer_t *src,
  695. uint16_t layer_idx, uint32_t src_address)
  696. {
  697. src->desc.pixel_format = hdma2d->layer_cfg[layer_idx].color_format,
  698. src->desc.height = hdma2d->layer_cfg[layer_idx].input_height;
  699. src->desc.width = hdma2d->layer_cfg[layer_idx].input_width;
  700. src->desc.pitch = hdma2d->layer_cfg[layer_idx].input_pitch;
  701. src->px_ofs = hdma2d->layer_cfg[layer_idx].input_xofs;
  702. src->addr = src_address;
  703. }
  704. /**
  705. * @brief Set the DMA2D transfer parameters.
  706. * @param hdma2d Pointer to a hal_dma2d_handle_t structure that contains
  707. * the configuration information for the specified DMA2D.
  708. * @param fg_address The source memory Buffer address for the foreground layer.
  709. * @param bg_address The source memory Buffer address for the background layer.
  710. * @param dst_address The destination memory Buffer address.
  711. * @param width The width of data to be transferred from source to destination.
  712. * @param height The height of data to be transferred from source to destination.
  713. * @retval command sequence (uint16_t) on success else negative errno code.
  714. */
  715. __de_func static int dma2d_set_config(hal_dma2d_handle_t *hdma2d, uint32_t fg_address,
  716. uint32_t bg_address, uint32_t dst_address, uint32_t width,
  717. uint32_t height)
  718. {
  719. display_buffer_t dst, fg, bg;
  720. display_color_t fg_color, bg_color;
  721. int res = 0;
  722. /* Check the peripheral existence */
  723. if (!is_dma2d_device_ready(hdma2d->device) || hdma2d->instance < 0) {
  724. return -ENODEV;
  725. }
  726. /* Check the parameters */
  727. if (!IS_DMA2D_PIXEL(width) || !IS_DMA2D_LINE(height)) {
  728. return -EINVAL;
  729. }
  730. dma2d_set_dst_buffer(hdma2d, &dst, dst_address, width, height);
  731. fg_color.full = hdma2d->layer_cfg[HAL_DMA2D_FOREGROUND_LAYER].input_alpha; /* ARGB8888 */
  732. bg_color.full = hdma2d->layer_cfg[HAL_DMA2D_BACKGROUND_LAYER].input_alpha; /* ARGB8888 */
  733. switch (hdma2d->output_cfg.mode) {
  734. case HAL_DMA2D_R2M:
  735. fg_color.full = fg_address;
  736. res = display_engine_fill(hdma2d->device, hdma2d->instance, &dst, fg_color);
  737. break;
  738. case HAL_DMA2D_M2M:
  739. dma2d_set_src_buffer(hdma2d, &fg, HAL_DMA2D_FOREGROUND_LAYER, fg_address);
  740. res = display_engine_blit(hdma2d->device, hdma2d->instance, &dst, &fg);
  741. break;
  742. case HAL_DMA2D_M2M_BLEND:
  743. dma2d_set_src_buffer(hdma2d, &fg, HAL_DMA2D_FOREGROUND_LAYER, fg_address);
  744. dma2d_set_src_buffer(hdma2d, &bg, HAL_DMA2D_BACKGROUND_LAYER, bg_address);
  745. res = display_engine_blend(hdma2d->device, hdma2d->instance, &dst, &fg, fg_color, &bg, bg_color);
  746. break;
  747. case HAL_DMA2D_M2M_BLEND_FG:
  748. fg_color.full = fg_address;
  749. dma2d_set_src_buffer(hdma2d, &bg, HAL_DMA2D_BACKGROUND_LAYER, bg_address);
  750. res = display_engine_blend(hdma2d->device, hdma2d->instance, &dst, NULL, fg_color, &bg, bg_color);
  751. break;
  752. case HAL_DMA2D_M2M_BLEND_BG:
  753. bg_color.full = bg_address;
  754. dma2d_set_src_buffer(hdma2d, &fg, HAL_DMA2D_FOREGROUND_LAYER, fg_address);
  755. res = display_engine_blend(hdma2d->device, hdma2d->instance, &dst, &fg, fg_color, NULL, bg_color);
  756. break;
  757. default:
  758. res = -EINVAL;
  759. break;
  760. }
  761. if (res < 0) {
  762. /* Update error code */
  763. hdma2d->error_code |= HAL_DMA2D_ERROR_CE;
  764. } else {
  765. atomic_inc(&hdma2d->xfer_count);
  766. }
  767. return res;
  768. }
  769. /**
  770. * @brief Global enable/disable DMAD functions
  771. * @param enabled enable or not
  772. * @retval N/A
  773. */
  774. void hal_dma2d_set_global_enabled(bool enabled)
  775. {
  776. const struct device *dma2d_dev = dma2d_get_device(HAL_DMA2D_M2M_BLEND);
  777. if (dma2d_dev == NULL || enabled == global_en) {
  778. return;
  779. }
  780. sys_trace_u32(SYS_TRACE_ID_DMA2D_EN, enabled);
  781. if (!enabled) {
  782. global_en = false;
  783. display_engine_control(dma2d_dev, DISPLAY_ENGINE_CTRL_WORK_MODE,
  784. (void *)DISPLAY_ENGINE_MODE_DISPLAY_ONLY, NULL);
  785. } else {
  786. global_en = true;
  787. display_engine_control(dma2d_dev, DISPLAY_ENGINE_CTRL_WORK_MODE,
  788. (void *)DISPLAY_ENGINE_MODE_DEFAULT, NULL);
  789. }
  790. sys_trace_end_call(SYS_TRACE_ID_DMA2D_EN);
  791. }
  792. /**
  793. * @}
  794. */
  795. /**
  796. * @}
  797. */