SWM341_dma.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. /******************************************************************************************************************************************
  2. * 文件名称: SWM341_dma.c
  3. * 功能说明: SWM341单片机的DMA功能驱动库
  4. * 技术支持: http://www.synwit.com.cn/e/tool/gbook/?bid=1
  5. * 注意事项:
  6. * 版本日期: V1.0.0 2016年1月30日
  7. * 升级记录:
  8. *
  9. *
  10. *******************************************************************************************************************************************
  11. * @attention
  12. *
  13. * THE PRESENT FIRMWARE WHICH IS FOR GUIDANCE ONLY AIMS AT PROVIDING CUSTOMERS WITH CODING INFORMATION
  14. * REGARDING THEIR PRODUCTS IN ORDER FOR THEM TO SAVE TIME. AS A RESULT, SYNWIT SHALL NOT BE HELD LIABLE
  15. * FOR ANY DIRECT, INDIRECT OR CONSEQUENTIAL DAMAGES WITH RESPECT TO ANY CLAIMS ARISING FROM THE CONTENT
  16. * OF SUCH FIRMWARE AND/OR THE USE MADE BY CUSTOMERS OF THE CODING INFORMATION CONTAINED HEREIN IN CONN-
  17. * -ECTION WITH THEIR PRODUCTS.
  18. *
  19. * COPYRIGHT 2012 Synwit Technology
  20. *******************************************************************************************************************************************/
  21. #include "SWM341.h"
  22. #include "SWM341_dma.h"
  23. /******************************************************************************************************************************************
  24. * 函数名称: DMA_CH_Init()
  25. * 功能说明: DMA通道初始化
  26. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  27. * DMA_InitStructure * initStruct 包含DMA通道相关设定值的结构体
  28. * 输 出: 无
  29. * 注意事项: 无
  30. ******************************************************************************************************************************************/
  31. void DMA_CH_Init(uint32_t chn, DMA_InitStructure * initStruct)
  32. {
  33. DMA->EN = 1; //每个通道都有自己独立的开关控制,所以总开关可以是一直开启的
  34. DMA_CH_Close(chn); //关闭后配置
  35. DMA->CH[chn].CR = (initStruct->Mode << DMA_CR_AUTORE_Pos) |
  36. ((initStruct->Count ? initStruct->Count - 1 : 0) << DMA_CR_LEN_Pos);
  37. DMA->CH[chn].SRC = initStruct->SrcAddr;
  38. DMA->CH[chn].DST = initStruct->DstAddr;
  39. DMA->CH[chn].AM = (initStruct->SrcAddrInc << DMA_AM_SRCAM_Pos) |
  40. (initStruct->DstAddrInc << DMA_AM_DSTAM_Pos) |
  41. (initStruct->Unit << DMA_AM_SRCBIT_Pos) |
  42. (initStruct->Unit << DMA_AM_DSTBIT_Pos);
  43. switch(initStruct->Handshake & DMA_HS_MSK)
  44. {
  45. case DMA_HS_NO:
  46. DMA->CH[chn].MUX = 0;
  47. break;
  48. case DMA_HS_SRC:
  49. DMA->CH[chn].MUX = ((initStruct->Handshake & 0xF) << DMA_MUX_SRCHSSIG_Pos) | (1 << DMA_MUX_SRCHSEN_Pos);
  50. break;
  51. case DMA_HS_DST:
  52. DMA->CH[chn].MUX = ((initStruct->Handshake & 0xF) << DMA_MUX_DSTHSSIG_Pos) | (1 << DMA_MUX_DSTHSEN_Pos);
  53. break;
  54. case DMA_HS_EXT:
  55. DMA->CH[chn].MUX = ((initStruct->Handshake & 0xF) << DMA_MUX_EXTHSSIG_Pos) | (1 << DMA_MUX_EXTHSEN_Pos);
  56. DMA->CH[chn].CR |= (1 << DMA_CR_STEPOP_Pos);
  57. break;
  58. default:
  59. break;
  60. }
  61. int totalBytes = initStruct->Count * (1 << initStruct->Unit);
  62. if(initStruct->DstAddrInc == 2) // Destination Scatter-Gather Transfer
  63. {
  64. DMA->CH[chn].DSTSGADDR1 = initStruct->DstAddr + totalBytes / 4 * 1;
  65. DMA->CH[chn].DSTSGADDR2 = initStruct->DstAddr + totalBytes / 4 * 2;
  66. DMA->CH[chn].DSTSGADDR3 = initStruct->DstAddr + totalBytes / 4 * 3;
  67. }
  68. if(initStruct->SrcAddrInc == 2) // Source Scatter-Gather Transfer
  69. {
  70. DMA->CH[chn].SRCSGADDR1 = initStruct->SrcAddr + totalBytes / 4 * 1;
  71. DMA->CH[chn].SRCSGADDR2 = initStruct->SrcAddr + totalBytes / 4 * 2;
  72. DMA->CH[chn].SRCSGADDR3 = initStruct->SrcAddr + totalBytes / 4 * 3;
  73. }
  74. DMA->PRI &= ~(1 << chn);
  75. DMA->PRI |= (initStruct->Priority << chn);
  76. DMA->IM |= (1 << chn); // 默认全部关闭
  77. DMA->DSTSGIM |= (3 << (chn * 2));
  78. DMA->SRCSGIM |= (3 << (chn * 2));
  79. DMA->IE |= (1 << chn); // 标志总是可查
  80. DMA->DSTSGIE |= (3 << (chn * 2));
  81. DMA->SRCSGIE |= (3 << (chn * 2));
  82. DMA_CH_INTClr(chn, initStruct->INTEn);
  83. DMA_CH_INTEn(chn, initStruct->INTEn);
  84. if(initStruct->INTEn) NVIC_EnableIRQ(DMA_IRQn);
  85. }
  86. /******************************************************************************************************************************************
  87. * 函数名称: DMA_CH_Open()
  88. * 功能说明: DMA通道开启,对于软件启动通道,开启后立即传输;对于硬件触发通道,开启后还需等出现触发信号后才开始搬运
  89. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  90. * 输 出: 无
  91. * 注意事项: 无
  92. ******************************************************************************************************************************************/
  93. void DMA_CH_Open(uint32_t chn)
  94. {
  95. DMA->CH[chn].CR |= (1 << DMA_CR_RXEN_Pos);
  96. }
  97. /******************************************************************************************************************************************
  98. * 函数名称: DMA_CH_Close()
  99. * 功能说明: DMA通道关闭
  100. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  101. * 输 出: 无
  102. * 注意事项: 无
  103. ******************************************************************************************************************************************/
  104. void DMA_CH_Close(uint32_t chn)
  105. {
  106. DMA->CH[chn].CR &= ~(DMA_CR_TXEN_Msk | DMA_CR_RXEN_Msk);
  107. }
  108. /******************************************************************************************************************************************
  109. * 函数名称: DMA_CH_SetCount()
  110. * 功能说明: 设置传输 Unit 个数
  111. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  112. * uint32_t count 传输 Unit 个数,最大取值0x100000
  113. * 输 出: 无
  114. * 注意事项: 无
  115. ******************************************************************************************************************************************/
  116. void DMA_CH_SetCount(uint32_t chn, uint32_t count)
  117. {
  118. DMA->CH[chn].CR &= ~DMA_CR_LEN_Msk;
  119. DMA->CH[chn].CR |= ((count - 1) << DMA_CR_LEN_Pos);
  120. }
  121. /******************************************************************************************************************************************
  122. * 函数名称: DMA_CH_GetRemaining()
  123. * 功能说明: 查询剩余的传输 Unit 个数
  124. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  125. * 输 出: uint32_t 剩余的传输 Unit 个数
  126. * 注意事项: 无
  127. ******************************************************************************************************************************************/
  128. uint32_t DMA_CH_GetRemaining(uint32_t chn)
  129. {
  130. return (DMA->CH[chn].DSTSR & DMA_DSTSR_LEN_Msk);
  131. }
  132. /******************************************************************************************************************************************
  133. * 函数名称: DMA_CH_SetSrcAddress()
  134. * 功能说明: 设置传输源地址
  135. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  136. * uint32_t address 源地址
  137. * 输 出: 无
  138. * 注意事项: 无
  139. ******************************************************************************************************************************************/
  140. void DMA_CH_SetSrcAddress(uint32_t chn, uint32_t address)
  141. {
  142. DMA->CH[chn].SRC = address;
  143. }
  144. /******************************************************************************************************************************************
  145. * 函数名称: DMA_CH_SetDstAddress()
  146. * 功能说明: 设置传输目的地址
  147. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  148. * uint32_t address 目的地址
  149. * 输 出: 无
  150. * 注意事项: 无
  151. ******************************************************************************************************************************************/
  152. void DMA_CH_SetDstAddress(uint32_t chn, uint32_t address)
  153. {
  154. DMA->CH[chn].DST = address;
  155. }
  156. /******************************************************************************************************************************************
  157. * 函数名称: DMA_CH_INTEn()
  158. * 功能说明: DMA中断使能
  159. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  160. * uint32_t it interrupt type,有效值有 DMA_IT_DONE、DMA_IT_DSTSG_HALF、DMA_IT_DSTSG_DONE、DMA_IT_SRCSG_HALF、
  161. * DMA_IT_SRCSG_DONE 及其“或”
  162. * 输 出: 无
  163. * 注意事项: 无
  164. ******************************************************************************************************************************************/
  165. void DMA_CH_INTEn(uint32_t chn, uint32_t it)
  166. {
  167. DMA->IM &= ~(it << chn);
  168. DMA->DSTSGIM &= ~((it >> 8) << (chn * 2));
  169. DMA->SRCSGIM &= ~((it >> 16) << (chn * 2));
  170. }
  171. /******************************************************************************************************************************************
  172. * 函数名称: DMA_CH_INTDis()
  173. * 功能说明: DMA中断禁止
  174. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  175. * uint32_t it interrupt type,有效值有 DMA_IT_DONE、DMA_IT_DSTSG_HALF、DMA_IT_DSTSG_DONE、DMA_IT_SRCSG_HALF、
  176. * DMA_IT_SRCSG_DONE 及其“或”
  177. * 输 出: 无
  178. * 注意事项: 无
  179. ******************************************************************************************************************************************/
  180. void DMA_CH_INTDis(uint32_t chn, uint32_t it)
  181. {
  182. DMA->IM |= (it << chn);
  183. DMA->DSTSGIM |= ((it >> 8) << (chn * 2));
  184. DMA->SRCSGIM |= ((it >> 16) << (chn * 2));
  185. }
  186. /******************************************************************************************************************************************
  187. * 函数名称: DMA_CH_INTClr()
  188. * 功能说明: DMA中断标志清除
  189. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  190. * uint32_t it interrupt type,有效值有 DMA_IT_DONE、DMA_IT_DSTSG_HALF、DMA_IT_DSTSG_DONE、DMA_IT_SRCSG_HALF、
  191. * DMA_IT_SRCSG_DONE 及其“或”
  192. * 输 出: 无
  193. * 注意事项: 无
  194. ******************************************************************************************************************************************/
  195. void DMA_CH_INTClr(uint32_t chn, uint32_t it)
  196. {
  197. DMA->IF = (it << chn);
  198. DMA->DSTSGIF = ((it >> 8) << (chn * 2));
  199. DMA->SRCSGIF = ((it >> 16) << (chn * 2));
  200. }
  201. /******************************************************************************************************************************************
  202. * 函数名称: DMA_CH_INTStat()
  203. * 功能说明: DMA中断状态查询
  204. * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
  205. * uint32_t it interrupt type,有效值有 DMA_IT_DONE、DMA_IT_DSTSG_HALF、DMA_IT_DSTSG_DONE、DMA_IT_SRCSG_HALF、
  206. * DMA_IT_SRCSG_DONE 及其“或”
  207. * 输 出: uint32_t 1 指定中断已发生 0 指定中断未发生
  208. * 注意事项: 无
  209. ******************************************************************************************************************************************/
  210. uint32_t DMA_CH_INTStat(uint32_t chn, uint32_t it)
  211. {
  212. return ((DMA->IF & (it << chn)) ||
  213. (DMA->DSTSGIF & ((it >> 8) << (chn * 2))) ||
  214. (DMA->SRCSGIF & ((it >> 16) << (chn * 2))));
  215. }