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
25 /******************************************************************************************************************************************
26 * 函数名称: DMA_CH_Init()
27 * 功能说明: DMA通道初始化
28 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
29 * DMA_InitStructure * initStruct 包含DMA通道相关设定值的结构体
30 * 输 出: 无
31 * 注意事项: 无
32 ******************************************************************************************************************************************/
DMA_CH_Init(uint32_t chn,DMA_InitStructure * initStruct)33 void DMA_CH_Init(uint32_t chn, DMA_InitStructure * initStruct)
34 {
35 DMA->EN = 1; //每个通道都有自己独立的开关控制,所以总开关可以是一直开启的
36
37 DMA_CH_Close(chn); //关闭后配置
38
39 DMA->CH[chn].CR = (initStruct->Mode << DMA_CR_AUTORE_Pos) |
40 ((initStruct->Count ? initStruct->Count - 1 : 0) << DMA_CR_LEN_Pos);
41
42 DMA->CH[chn].SRC = initStruct->SrcAddr;
43 DMA->CH[chn].DST = initStruct->DstAddr;
44
45 DMA->CH[chn].AM = (initStruct->SrcAddrInc << DMA_AM_SRCAM_Pos) |
46 (initStruct->DstAddrInc << DMA_AM_DSTAM_Pos) |
47 (initStruct->Unit << DMA_AM_SRCBIT_Pos) |
48 (initStruct->Unit << DMA_AM_DSTBIT_Pos);
49
50 switch(initStruct->Handshake & DMA_HS_MSK)
51 {
52 case DMA_HS_NO:
53 DMA->CH[chn].MUX = 0;
54 break;
55
56 case DMA_HS_SRC:
57 DMA->CH[chn].MUX = ((initStruct->Handshake & 0xF) << DMA_MUX_SRCHSSIG_Pos) | (1 << DMA_MUX_SRCHSEN_Pos);
58 break;
59
60 case DMA_HS_DST:
61 DMA->CH[chn].MUX = ((initStruct->Handshake & 0xF) << DMA_MUX_DSTHSSIG_Pos) | (1 << DMA_MUX_DSTHSEN_Pos);
62 break;
63
64 case DMA_HS_EXT:
65 DMA->CH[chn].MUX = ((initStruct->Handshake & 0xF) << DMA_MUX_EXTHSSIG_Pos) | (1 << DMA_MUX_EXTHSEN_Pos);
66 DMA->CH[chn].CR |= (1 << DMA_CR_STEPOP_Pos);
67 break;
68
69 default:
70 break;
71 }
72
73 int totalBytes = initStruct->Count * (1 << initStruct->Unit);
74
75 if(initStruct->DstAddrInc == 2) // Destination Scatter-Gather Transfer
76 {
77 DMA->CH[chn].DSTSGADDR1 = initStruct->DstAddr + totalBytes / 4 * 1;
78 DMA->CH[chn].DSTSGADDR2 = initStruct->DstAddr + totalBytes / 4 * 2;
79 DMA->CH[chn].DSTSGADDR3 = initStruct->DstAddr + totalBytes / 4 * 3;
80 }
81 if(initStruct->SrcAddrInc == 2) // Source Scatter-Gather Transfer
82 {
83 DMA->CH[chn].SRCSGADDR1 = initStruct->SrcAddr + totalBytes / 4 * 1;
84 DMA->CH[chn].SRCSGADDR2 = initStruct->SrcAddr + totalBytes / 4 * 2;
85 DMA->CH[chn].SRCSGADDR3 = initStruct->SrcAddr + totalBytes / 4 * 3;
86 }
87
88 DMA->PRI &= ~(1 << chn);
89 DMA->PRI |= (initStruct->Priority << chn);
90
91 DMA->IM |= (1 << chn); // 默认全部关闭
92 DMA->DSTSGIM |= (3 << (chn * 2));
93 DMA->SRCSGIM |= (3 << (chn * 2));
94 DMA->IE |= (1 << chn); // 标志总是可查
95 DMA->DSTSGIE |= (3 << (chn * 2));
96 DMA->SRCSGIE |= (3 << (chn * 2));
97
98 DMA_CH_INTClr(chn, initStruct->INTEn);
99 DMA_CH_INTEn(chn, initStruct->INTEn);
100
101 if(initStruct->INTEn) NVIC_EnableIRQ(DMA_IRQn);
102 }
103
104 /******************************************************************************************************************************************
105 * 函数名称: DMA_CH_Open()
106 * 功能说明: DMA通道开启,对于软件启动通道,开启后立即传输;对于硬件触发通道,开启后还需等出现触发信号后才开始搬运
107 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
108 * 输 出: 无
109 * 注意事项: 无
110 ******************************************************************************************************************************************/
DMA_CH_Open(uint32_t chn)111 void DMA_CH_Open(uint32_t chn)
112 {
113 DMA->CH[chn].CR |= (1 << DMA_CR_RXEN_Pos);
114 }
115
116 /******************************************************************************************************************************************
117 * 函数名称: DMA_CH_Close()
118 * 功能说明: DMA通道关闭
119 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
120 * 输 出: 无
121 * 注意事项: 无
122 ******************************************************************************************************************************************/
DMA_CH_Close(uint32_t chn)123 void DMA_CH_Close(uint32_t chn)
124 {
125 DMA->CH[chn].CR &= ~(DMA_CR_TXEN_Msk | DMA_CR_RXEN_Msk);
126 }
127
128 /******************************************************************************************************************************************
129 * 函数名称: DMA_CH_SetCount()
130 * 功能说明: 设置传输 Unit 个数
131 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
132 * uint32_t count 传输 Unit 个数,最大取值0x100000
133 * 输 出: 无
134 * 注意事项: 无
135 ******************************************************************************************************************************************/
DMA_CH_SetCount(uint32_t chn,uint32_t count)136 void DMA_CH_SetCount(uint32_t chn, uint32_t count)
137 {
138 DMA->CH[chn].CR &= ~DMA_CR_LEN_Msk;
139 DMA->CH[chn].CR |= ((count - 1) << DMA_CR_LEN_Pos);
140 }
141
142 /******************************************************************************************************************************************
143 * 函数名称: DMA_CH_GetRemaining()
144 * 功能说明: 查询剩余的传输 Unit 个数
145 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
146 * 输 出: uint32_t 剩余的传输 Unit 个数
147 * 注意事项: 无
148 ******************************************************************************************************************************************/
DMA_CH_GetRemaining(uint32_t chn)149 uint32_t DMA_CH_GetRemaining(uint32_t chn)
150 {
151 return (DMA->CH[chn].DSTSR & DMA_DSTSR_LEN_Msk);
152 }
153
154 /******************************************************************************************************************************************
155 * 函数名称: DMA_CH_SetSrcAddress()
156 * 功能说明: 设置传输源地址
157 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
158 * uint32_t address 源地址
159 * 输 出: 无
160 * 注意事项: 无
161 ******************************************************************************************************************************************/
DMA_CH_SetSrcAddress(uint32_t chn,uint32_t address)162 void DMA_CH_SetSrcAddress(uint32_t chn, uint32_t address)
163 {
164 DMA->CH[chn].SRC = address;
165 }
166
167 /******************************************************************************************************************************************
168 * 函数名称: DMA_CH_SetDstAddress()
169 * 功能说明: 设置传输目的地址
170 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
171 * uint32_t address 目的地址
172 * 输 出: 无
173 * 注意事项: 无
174 ******************************************************************************************************************************************/
DMA_CH_SetDstAddress(uint32_t chn,uint32_t address)175 void DMA_CH_SetDstAddress(uint32_t chn, uint32_t address)
176 {
177 DMA->CH[chn].DST = address;
178 }
179
180 /******************************************************************************************************************************************
181 * 函数名称: DMA_CH_INTEn()
182 * 功能说明: DMA中断使能
183 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
184 * uint32_t it interrupt type,有效值有 DMA_IT_DONE、DMA_IT_DSTSG_HALF、DMA_IT_DSTSG_DONE、DMA_IT_SRCSG_HALF、
185 * DMA_IT_SRCSG_DONE 及其“或”
186 * 输 出: 无
187 * 注意事项: 无
188 ******************************************************************************************************************************************/
DMA_CH_INTEn(uint32_t chn,uint32_t it)189 void DMA_CH_INTEn(uint32_t chn, uint32_t it)
190 {
191 DMA->IM &= ~(it << chn);
192 DMA->DSTSGIM &= ~((it >> 8) << (chn * 2));
193 DMA->SRCSGIM &= ~((it >> 16) << (chn * 2));
194 }
195
196 /******************************************************************************************************************************************
197 * 函数名称: DMA_CH_INTDis()
198 * 功能说明: DMA中断禁止
199 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
200 * uint32_t it interrupt type,有效值有 DMA_IT_DONE、DMA_IT_DSTSG_HALF、DMA_IT_DSTSG_DONE、DMA_IT_SRCSG_HALF、
201 * DMA_IT_SRCSG_DONE 及其“或”
202 * 输 出: 无
203 * 注意事项: 无
204 ******************************************************************************************************************************************/
DMA_CH_INTDis(uint32_t chn,uint32_t it)205 void DMA_CH_INTDis(uint32_t chn, uint32_t it)
206 {
207 DMA->IM |= (it << chn);
208 DMA->DSTSGIM |= ((it >> 8) << (chn * 2));
209 DMA->SRCSGIM |= ((it >> 16) << (chn * 2));
210 }
211
212 /******************************************************************************************************************************************
213 * 函数名称: DMA_CH_INTClr()
214 * 功能说明: DMA中断标志清除
215 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
216 * uint32_t it interrupt type,有效值有 DMA_IT_DONE、DMA_IT_DSTSG_HALF、DMA_IT_DSTSG_DONE、DMA_IT_SRCSG_HALF、
217 * DMA_IT_SRCSG_DONE 及其“或”
218 * 输 出: 无
219 * 注意事项: 无
220 ******************************************************************************************************************************************/
DMA_CH_INTClr(uint32_t chn,uint32_t it)221 void DMA_CH_INTClr(uint32_t chn, uint32_t it)
222 {
223 DMA->IF = (it << chn);
224 DMA->DSTSGIF = ((it >> 8) << (chn * 2));
225 DMA->SRCSGIF = ((it >> 16) << (chn * 2));
226 }
227
228 /******************************************************************************************************************************************
229 * 函数名称: DMA_CH_INTStat()
230 * 功能说明: DMA中断状态查询
231 * 输 入: uint32_t chn 指定要配置的通道,有效值有DMA_CH0、DMA_CH1、DMA_CH2、DMA_CH3
232 * uint32_t it interrupt type,有效值有 DMA_IT_DONE、DMA_IT_DSTSG_HALF、DMA_IT_DSTSG_DONE、DMA_IT_SRCSG_HALF、
233 * DMA_IT_SRCSG_DONE 及其“或”
234 * 输 出: uint32_t 1 指定中断已发生 0 指定中断未发生
235 * 注意事项: 无
236 ******************************************************************************************************************************************/
DMA_CH_INTStat(uint32_t chn,uint32_t it)237 uint32_t DMA_CH_INTStat(uint32_t chn, uint32_t it)
238 {
239 return ((DMA->IF & (it << chn)) ||
240 (DMA->DSTSGIF & ((it >> 8) << (chn * 2))) ||
241 (DMA->SRCSGIF & ((it >> 16) << (chn * 2))));
242 }
243