在线时间282 小时
UID3301905
ST金币4288
蝴蝶豆7
注册时间2016-11-26
该用户从未签到
论坛元老
- 最后登录
- 2022-9-9
|
这几天一直都在捣鼓官方SDK中的例程,其中最令我兴奋的是找到了之前移植的个人工程失败的原因。原因是stm32f30x_it.c的SysTick_Handler()函数和stm32f30x_MC_it.c整个文件不能移除,必须被保留。SysTick_Handler()函数的作用是运用系统滴答定时器的计时溢出中断,周期性执行某个程序,如HAL_GPIO_TogglePin()翻转某个LED灯的电平状态,当然,SysTick_Handler()执行的间隔太短,肉眼是看不出LED翻转效果的。官方例程里面的SysTick_Handler()函数里面是TB_Scheduler()函数,涉及到一些电机状态控制和安全控制相关的函数,所以不能去掉。虽然不能去掉,但是可以把SysTick_Handler()整个搬到main.c的最下面,既方便修改,又减少了工程所包含文件的数量。至于stm32f30x_MC_it.c,里面的重要函数是一些定时器中断请求的函数如TIMx_UP_M2_IRQHandler(),也是非常重要不能去掉的。那么我是怎么发现这点的呢,是靠排除法发现的,我在进行STM32开发的时候最看不爽的就是XXXX_it.c结尾的中断服务函数源文件,一般情况下能删则删,不能删则一个个函数移植。在我操作官方SDK的时候不小心误删了stm32f30x_MC_it.c文件,然后电机也不转了,于是我就知道了stm32f30x_MC_it.c这个文件很重要,不能删。
既然移植的工程没有任何问题,那自然就可以开始进行自己的开发了。俗话说调通串口好办事,首要做的肯定是先调通串口以便以后监控各个变量的情况。由于是标准库,我首先想到的是套用正点原子战舰F103ZE的例程,但是初期套用没法调通,花了我将近一个小时,后来我将官方库的串口初始化代码移植过来,总算是成功了。以下是串口初始化代码,供大家参考:
- <font size="3">void USART2_Init(int baud)
- {
- RCC_APB1PeriphClockCmd(RCC_AHBPeriph_GPIOA|
- RCC_APB1Periph_USART2,ENABLE);
-
- USART_InitStructure.USART_BaudRate=115200;
- USART_InitStructure.USART_WordLength=USART_WordLength_8b;
- USART_InitStructure.USART_StopBits=USART_StopBits_1;
- USART_InitStructure.USART_Parity=USART_Parity_No;
-
- USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None;
- USART_InitStructure.USART_Mode = USART_Mode_Rx|USART_Mode_Tx;
- USART_Init(USART2,&USART_InitStructure);
- USART_Cmd(USART2, ENABLE);
-
- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
- GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
- GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
- GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
- GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
-
- GPIO_Init(GPIOA,&GPIO_InitStructure);
-
- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
- GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN;
- GPIO_InitStructure.GPIO_OType=GPIO_OType_OD;
- GPIO_Init(GPIOA,&GPIO_InitStructure);
-
- GPIO_PinAFConfig(GPIOA,GPIO_PinSource2,GPIO_AF_7);
- GPIO_PinAFConfig(GPIOA,GPIO_PinSource3,GPIO_AF_7);
-
- NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;
- NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;
- NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;
- NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
- NVIC_Init(&NVIC_InitStructure);
- }</font>
复制代码
这里顺便提提,官方的串口初始化函数是用结构体函数指针进行初始化的,串口的各种参数如波特率、校验位、停止位等等,串口的PORT脚和引脚等等都经过三层封装,放在三个不同的文件里,这点跟ST官方的Disco系列开发板的外设初始化程序很像:
另外,初始化函数多次提到了object(对象)这个单词,很容易让人联想到8月22日电机套件直播的时候主持人提到的【面向对象初始化】,并且官方FOC例程里面有诸多类似于XXXX_class.c(对象的实现)、XXXX_private.h(对象的声明)、XXXX_class.h(接口)三大类型文件的分别存放,很容易可以让开发者知道官方想要进行纯C环境下的面向对象开发的意图。这点是好还是不好呢,我也不想作过多评价,免得被社区里某些所谓的【自认为是老工程师的人】指指点点。
现在我就贴出部分官方初始化串口的函数:
- <font size="3">const USART_InitTypeDef UFCInitHW_str =
- {
- USART_SPEED, //USART_BaudRate
- USART_WordLength_8b, //USART_WordLength
- USART_StopBits_1, //USART_StopBits
- USART_Parity_No, //USART_Parity
- USART_Mode_Tx, //USART_Mode
- USART_HardwareFlowControl_None //USART_HardwareFlowControl
- };
- USARTParams_t USARTParams_str =
- {
- USART, // USART
- USART_GPIO_REMAP,
- // USART_REMAP GPIO_NoRemap_USART1 or GPIO_Remap_USART1 ...
- USART_CLK, // USART_CLK
- USART_RX_GPIO_PORT, // USART_RxPort
- USART_RX_GPIO_PIN, // USART_RxPin
- USART_TX_GPIO_PORT, // USART_TxPort
- USART_TX_GPIO_PIN, // USART_TxPin
- UI_IRQ_USART, // IRQ Number
- (USART_InitTypeDef*)(&USARTInitHW_str), // USART_InitStructure
- (NVIC_InitTypeDef*)(&NVICInitHW_str) // NVIC_InitStructure
- };</font>
复制代码- <font size="3">CUSART_COM USART_NewObject(pUSARTParams_t pUSARTParams)
- {
- _CCOM _oCOM;
- _CUSART _oUSART;
-
- _oCOM = (_CCOM)COM_NewObject();
- #ifdef MC_CLASS_DYNAMIC
- _oUSART = (_CUSART)calloc(1,sizeof(_CUSART_t));
- #else
- if (USART_COM_Allocated < MAX_USART_COM_NUM)
- {
- _oUSART = &USART_COMpool[USART_COM_Allocated++];
- }
- else
- {
- _oUSART = MC_NULL;
- }
- #endif
-
- _oUSART->pDParams_str = pUSARTParams;
- _oCOM->DerivedClass = (void*)_oUSART;
- _oCOM->Methods_str.pStartReceive = &USART_StartReceive;
- _oCOM->Methods_str.pStartTransmit = &USART_StartTransmit;
-
- _oCOM->Methods_str.pIRQ_Handler = &USART_IRQ_Handler;
- Set_UI_IRQ_Handler(pUSARTParams->bUIIRQn, (_CUIIRQ)_oCOM);
-
- //Init Struct communication
- COM_ResetTX((CCOM)_oCOM);
- COM_ResetRX((CCOM)_oCOM);
-
- USART_HWInit(pUSARTParams);
-
- return ((CUSART_COM)_oCOM);
- }
- void USART_HWInit(pUSARTParams_t pUSARTParams)
- {
- GPIO_InitTypeDef GPIO_InitStructure;
-
- /* Enable USART clock: UASRT1 -> APB2, USART2-5 -> APB1 */
- if (pUSARTParams->wUSARTClockSource == RCC_APB2Periph_USART1)
- {
- RCC_APB2PeriphClockCmd(pUSARTParams->wUSARTClockSource, ENABLE);
- }
- else
- {
- RCC_APB1PeriphClockCmd(pUSARTParams->wUSARTClockSource, ENABLE);
- }
-
- /* USART Init structure */
- /* Configure USART */
- USART_Init(pUSARTParams->USARTx, pUSARTParams->USART_InitStructure);
-
- GPIO_PinAFConfig(pUSARTParams->hRxPort, USART_GPIOPin2Source(pUSARTParams->hRxPin), USART_AF(pUSARTParams->USARTx));
- GPIO_PinAFConfig(pUSARTParams->hTxPort, USART_GPIOPin2Source(pUSARTParams->hTxPin), USART_AF(pUSARTParams->USARTx));
-
- /* Configure Rx, Tx pins */
- GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
- GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
- GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
- GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
-
- GPIO_InitStructure.GPIO_Pin = pUSARTParams->hRxPin;
- GPIO_Init(pUSARTParams->hRxPort, &GPIO_InitStructure);
-
- GPIO_InitStructure.GPIO_Pin = pUSARTParams->hTxPin;
- GPIO_Init(pUSARTParams->hTxPort, &GPIO_InitStructure);
-
- if (pUSARTParams->NVIC_InitStructure->NVIC_IRQChannelCmd == ENABLE)
- {
- /* Enable USART Receive and Transmit interrupts */
- if ((pUSARTParams->USART_InitStructure->USART_Mode & USART_Mode_Rx) == USART_Mode_Rx)
- {
- USART_ITConfig(pUSARTParams->USARTx, USART_IT_RXNE, ENABLE);
- }
- if ((pUSARTParams->USART_InitStructure->USART_Mode & USART_Mode_Tx) == USART_Mode_Tx)
- {
- USART_ITConfig(pUSARTParams->USARTx, USART_IT_TXE, DISABLE);
- }
- /* Enable the USARTy Interrupt */
- NVIC_Init(pUSARTParams->NVIC_InitStructure);
- }
-
- /* Enable the USART */
- USART_Cmd(pUSARTParams->USARTx, ENABLE);
- }</font>
复制代码
好不好用大家心中有数,大家喜欢正点原子风格还是官方Disco风格,自己决定吧。
除了串口以外,外部中断也是必须调通的一个外设。官方例程的按键处理程序是while轮询,但是我认为,电机控制这种事情,还是用外部中断最好。顺便说说F3XX的外部中断,用SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOC,EXTI_PinSource13)函数替代了F103的GPIO_EXTILineConfig(GPIO_PortSourceGPIOC,GPIO_PinSource13),为了找相关资料,又忙活了一下午。以下是外部中断初始化和中断服务函数:
- <font size="3">void EXTI13_Init()
- {
- EXTI_InitTypeDef EXTI_InitStructure;
- NVIC_InitTypeDef NVIC_InitStructure;
- GPIO_InitTypeDef GPIO_InitStructure;
-
- RCC_APB2PeriphClockCmd(RCC_AHBPeriph_GPIOC|
- RCC_APB2Periph_SYSCFG, ENABLE);
- GPIO_InitStructure.GPIO_PuPd=GPIO_PuPd_UP;
- GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN;
- GPIO_InitStructure.GPIO_Pin=GPIO_Pin_13;
- GPIO_Init(GPIOC,&GPIO_InitStructure);
-
- SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOC,EXTI_PinSource13);
- EXTI_InitStructure.EXTI_Line=EXTI_Line13;
- EXTI_InitStructure.EXTI_Mode=EXTI_Mode_Interrupt;
- EXTI_InitStructure.EXTI_Trigger=EXTI_Trigger_Falling;
- EXTI_InitStructure.EXTI_LineCmd=ENABLE;
- EXTI_Init(&EXTI_InitStructure);
- NVIC_InitStructure.NVIC_IRQChannel=EXTI15_10_IRQn;
- NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1;
- NVIC_InitStructure.NVIC_IRQChannelSubPriority=1;
- NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;
- NVIC_Init(&NVIC_InitStructure);
- }
- </font>
复制代码- <font size="3">void EXTI15_10_IRQHandler()
- {
- if(EXTI_GetITStatus(EXTI_Line13) != RESET)
- {
- flag=0;
- cont++;
- if(cont>5)cont=0;
- printf("进入外部中断,cont=%d\n",cont);
- EXTI_ClearITPendingBit(EXTI_Line13);
- }
- }</font>
复制代码
调通了外部中断之后,就可以结合之前的电机调速Task,进行实验了。每按一次按键,电机速度提升3000转,到了13000转之后速度重新归为1000转:- <font size="3">int main()
- {
- STM32F3_64MHz_Internal();
- MCboot(oMCI,oMCT);
- SysTick_Configuration();
- EXTI13_Init();
- USART2_Init(115200);
- printf("STM32社区 电机套件试用活动\n");
- //MCI_StartMotor(oMCI[M1]);
- while(1)
- {
-
- switch(cont)
- {
- case 1:
- if(flag==0)
- {
- flag=1;
- MCI_StartMotor(oMCI[M1]);
- MCI_ExecSpeedRamp(oMCI[0],1000/6,1000);
- printf("设置电机转速为1000\n");
- }
- break;
- case 2:
- if(flag==0)
- {
- flag=1;
- MCI_StartMotor(oMCI[M1]);
- MCI_ExecSpeedRamp(oMCI[0],4000/6,1000);
- printf("设置电机转速为4000\n");
- }
- break;
- case 3:
- if(flag==0)
- {
- flag=1;
- MCI_StartMotor(oMCI[M1]);
- MCI_ExecSpeedRamp(oMCI[0],7000/6,1000);
- printf("设置电机转速为7000\n");
- }
- break;
- case 4:
- if(flag==0)
- {
- flag=1;
- MCI_StartMotor(oMCI[M1]);
- MCI_ExecSpeedRamp(oMCI[0],10000/6,1000);
- printf("设置电机转速为10000\n");
- }
- break;
- case 5:
- if(flag==0)
- {
- flag=1;
- MCI_StartMotor(oMCI[M1]);
- MCI_ExecSpeedRamp(oMCI[0],13000/6,1000);
- printf("设置电机转速为13000\n");
- }
- break;
- }
- }
- }</font>
复制代码
上传工程文件。
电机套件工程.zip
(3.79 MB, 下载次数: 94)
|
|