Created
April 28, 2015 13:50
-
-
Save kkmonster/109decfde98e4f3f96c2 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
****************************************************************************** | |
* File Name : main.c | |
* Date : 21/03/2015 17:50:15 | |
* Description : Main program body | |
****************************************************************************** | |
* | |
* COPYRIGHT(c) 2015 STMicroelectronics | |
* | |
* Redistribution and use in source and binary forms, with or without modification, | |
* are permitted provided that the following conditions are met: | |
* 1. Redistributions of source code must retain the above copyright notice, | |
* this list of conditions and the following disclaimer. | |
* 2. Redistributions in binary form must reproduce the above copyright notice, | |
* this list of conditions and the following disclaimer in the documentation | |
* and/or other materials provided with the distribution. | |
* 3. Neither the name of STMicroelectronics nor the names of its contributors | |
* may be used to endorse or promote products derived from this software | |
* without specific prior written permission. | |
* | |
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | |
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | |
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | |
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | |
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | |
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
* | |
****************************************************************************** | |
*/ | |
/* Includes ------------------------------------------------------------------*/ | |
#include "stm32f0xx_hal.h" | |
/* USER CODE BEGIN Includes */ | |
#define center_current 3077.0f // | |
#define offset_zero 0.0f // | |
/* USER CODE END Includes */ | |
/* Private variables ---------------------------------------------------------*/ | |
ADC_HandleTypeDef hadc; | |
DMA_HandleTypeDef hdma_adc; | |
TIM_HandleTypeDef htim3; | |
TIM_HandleTypeDef htim14; | |
TIM_HandleTypeDef htim17; | |
/* USER CODE BEGIN PV */ | |
uint16_t ADC_raw_data[4] = {0}; | |
float Iref = 0; | |
float Icurrent = 0; | |
float Pwm = 0; | |
float error = 0; | |
float a = 0; | |
float b = 0; | |
float c = 0; | |
float d = 0; | |
float e = 0; | |
/* USER CODE END PV */ | |
/* Private function prototypes -----------------------------------------------*/ | |
void SystemClock_Config(void); | |
static void MX_GPIO_Init(void); | |
static void MX_DMA_Init(void); | |
static void MX_ADC_Init(void); | |
static void MX_TIM3_Init(void); | |
static void MX_TIM14_Init(void); | |
static void MX_TIM17_Init(void); | |
/* USER CODE BEGIN PFP */ | |
volatile void PI_control(void); | |
volatile void F_drive(float tmp); | |
volatile void R_drive(float tmp); | |
volatile void disable_drive(void); | |
float Smooth_filter(float alfa, float new_data, float prev_data); | |
/* USER CODE END PFP */ | |
/* USER CODE BEGIN 0 */ | |
/* USER CODE END 0 */ | |
int main(void) | |
{ | |
/* USER CODE BEGIN 1 */ | |
/* USER CODE END 1 */ | |
/* MCU Configuration----------------------------------------------------------*/ | |
/* Reset of all peripherals, Initializes the Flash interface and the Systick. */ | |
HAL_Init(); | |
/* Configure the system clock */ | |
SystemClock_Config(); | |
/* Initialize all configured peripherals */ | |
MX_GPIO_Init(); | |
MX_DMA_Init(); | |
MX_ADC_Init(); | |
MX_TIM3_Init(); | |
MX_TIM14_Init(); | |
MX_TIM17_Init(); | |
/* USER CODE BEGIN 2 */ | |
disable_drive(); | |
// enable ADC | |
HAL_ADC_Start_DMA(&hadc ,(uint32_t*)ADC_raw_data, 4); | |
// enable Pwm | |
HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_1); | |
HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_2); | |
HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_4); | |
HAL_TIM_PWM_Start(&htim14, TIM_CHANNEL_1); | |
disable_drive(); | |
// enable sampling | |
HAL_TIM_Base_Start_IT(&htim17); | |
/* USER CODE END 2 */ | |
/* USER CODE BEGIN 3 */ | |
/* Infinite loop */ | |
while (1) | |
{ | |
HAL_GPIO_TogglePin(GPIOF, GPIO_PIN_0); | |
HAL_Delay(50); | |
a = ADC_raw_data[0]; | |
b = ADC_raw_data[1]; | |
c = ADC_raw_data[2]; | |
d = ADC_raw_data[3]; | |
e = Smooth_filter(0.7f, ((float)ADC_raw_data[0] * 0.5f + (float)ADC_raw_data[2] * 0.5f), e); | |
} | |
/* USER CODE END 3 */ | |
} | |
/** System Clock Configuration | |
*/ | |
void SystemClock_Config(void) | |
{ | |
RCC_OscInitTypeDef RCC_OscInitStruct; | |
RCC_ClkInitTypeDef RCC_ClkInitStruct; | |
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI|RCC_OSCILLATORTYPE_HSI14; | |
RCC_OscInitStruct.HSIState = RCC_HSI_ON; | |
RCC_OscInitStruct.HSI14State = RCC_HSI14_ON; | |
RCC_OscInitStruct.HSICalibrationValue = 16; | |
RCC_OscInitStruct.HSI14CalibrationValue = 16; | |
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; | |
RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSI; | |
RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL12; | |
RCC_OscInitStruct.PLL.PREDIV = RCC_PREDIV_DIV1; | |
HAL_RCC_OscConfig(&RCC_OscInitStruct); | |
RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_SYSCLK; | |
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK; | |
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; | |
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1; | |
HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_1); | |
__SYSCFG_CLK_ENABLE(); | |
} | |
/* ADC init function */ | |
void MX_ADC_Init(void) | |
{ | |
ADC_ChannelConfTypeDef sConfig; | |
/**Configure the global features of the ADC (Clock, Resolution, Data Alignment and number of conversion) | |
*/ | |
hadc.Instance = ADC1; | |
hadc.Init.ClockPrescaler = ADC_CLOCK_ASYNC; | |
hadc.Init.Resolution = ADC_RESOLUTION12b; | |
hadc.Init.DataAlign = ADC_DATAALIGN_RIGHT; | |
hadc.Init.ScanConvMode = ADC_SCAN_DIRECTION_FORWARD; | |
hadc.Init.EOCSelection = EOC_SINGLE_CONV; | |
hadc.Init.LowPowerAutoWait = DISABLE; | |
hadc.Init.LowPowerAutoPowerOff = DISABLE; | |
hadc.Init.ContinuousConvMode = ENABLE; | |
hadc.Init.DiscontinuousConvMode = DISABLE; | |
hadc.Init.ExternalTrigConvEdge = ADC_EXTERNALTRIGCONVEDGE_NONE; | |
hadc.Init.DMAContinuousRequests = ENABLE; | |
hadc.Init.Overrun = OVR_DATA_PRESERVED; | |
HAL_ADC_Init(&hadc); | |
/**Configure for the selected ADC regular channel to be converted. | |
*/ | |
sConfig.Channel = ADC_CHANNEL_0; | |
sConfig.Rank = 1; | |
sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5; | |
HAL_ADC_ConfigChannel(&hadc, &sConfig); | |
/**Configure for the selected ADC regular channel to be converted. | |
*/ | |
sConfig.Channel = ADC_CHANNEL_1; | |
sConfig.Rank = 2; | |
HAL_ADC_ConfigChannel(&hadc, &sConfig); | |
/**Configure for the selected ADC regular channel to be converted. | |
*/ | |
sConfig.Channel = ADC_CHANNEL_0; | |
sConfig.Rank = 3; | |
HAL_ADC_ConfigChannel(&hadc, &sConfig); | |
/**Configure for the selected ADC regular channel to be converted. | |
*/ | |
sConfig.Channel = ADC_CHANNEL_1; | |
sConfig.Rank = 4; | |
HAL_ADC_ConfigChannel(&hadc, &sConfig); | |
} | |
/* TIM3 init function */ | |
void MX_TIM3_Init(void) | |
{ | |
TIM_MasterConfigTypeDef sMasterConfig; | |
TIM_OC_InitTypeDef sConfigOC; | |
htim3.Instance = TIM3; | |
htim3.Init.Prescaler = 0; | |
htim3.Init.CounterMode = TIM_COUNTERMODE_UP; | |
htim3.Init.Period = 2399; | |
htim3.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; | |
HAL_TIM_PWM_Init(&htim3); | |
sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET; | |
sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE; | |
HAL_TIMEx_MasterConfigSynchronization(&htim3, &sMasterConfig); | |
sConfigOC.OCMode = TIM_OCMODE_PWM1; | |
sConfigOC.Pulse = 0; | |
sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH; | |
sConfigOC.OCFastMode = TIM_OCFAST_DISABLE; | |
HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_1); | |
HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_2); | |
sConfigOC.OCMode = TIM_OCMODE_PWM2; | |
HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_4); | |
} | |
/* TIM14 init function */ | |
void MX_TIM14_Init(void) | |
{ | |
TIM_OC_InitTypeDef sConfigOC; | |
htim14.Instance = TIM14; | |
htim14.Init.Prescaler = 0; | |
htim14.Init.CounterMode = TIM_COUNTERMODE_UP; | |
htim14.Init.Period = 2399; | |
htim14.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; | |
HAL_TIM_Base_Init(&htim14); | |
HAL_TIM_PWM_Init(&htim14); | |
sConfigOC.OCMode = TIM_OCMODE_PWM2; | |
sConfigOC.Pulse = 0; | |
sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH; | |
sConfigOC.OCFastMode = TIM_OCFAST_DISABLE; | |
HAL_TIM_PWM_ConfigChannel(&htim14, &sConfigOC, TIM_CHANNEL_1); | |
} | |
/* TIM17 init function */ | |
void MX_TIM17_Init(void) | |
{ | |
htim17.Instance = TIM17; | |
htim17.Init.Prescaler = 0; | |
htim17.Init.CounterMode = TIM_COUNTERMODE_UP; | |
htim17.Init.Period = 2399; | |
htim17.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; | |
htim17.Init.RepetitionCounter = 0; | |
HAL_TIM_Base_Init(&htim17); | |
} | |
/** | |
* Enable DMA controller clock | |
*/ | |
void MX_DMA_Init(void) | |
{ | |
/* DMA controller clock enable */ | |
__DMA1_CLK_ENABLE(); | |
/* DMA interrupt init */ | |
} | |
/** Configure pins as | |
* Analog | |
* Input | |
* Output | |
* EVENT_OUT | |
* EXTI | |
*/ | |
void MX_GPIO_Init(void) | |
{ | |
GPIO_InitTypeDef GPIO_InitStruct; | |
/* GPIO Ports Clock Enable */ | |
__GPIOF_CLK_ENABLE(); | |
__GPIOA_CLK_ENABLE(); | |
__GPIOB_CLK_ENABLE(); | |
/*Configure GPIO pins : PF0 PF1 */ | |
GPIO_InitStruct.Pin = GPIO_PIN_0|GPIO_PIN_1; | |
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; | |
GPIO_InitStruct.Pull = GPIO_NOPULL; | |
GPIO_InitStruct.Speed = GPIO_SPEED_LOW; | |
HAL_GPIO_Init(GPIOF, &GPIO_InitStruct); | |
/*Configure GPIO pins : PA2 PA3 */ | |
GPIO_InitStruct.Pin = GPIO_PIN_2|GPIO_PIN_3; | |
GPIO_InitStruct.Mode = GPIO_MODE_INPUT; | |
GPIO_InitStruct.Pull = GPIO_PULLDOWN; | |
HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); | |
/*Configure GPIO pins : PA5 PA9 PA10 */ | |
GPIO_InitStruct.Pin = GPIO_PIN_5|GPIO_PIN_9|GPIO_PIN_10; | |
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; | |
GPIO_InitStruct.Pull = GPIO_NOPULL; | |
GPIO_InitStruct.Speed = GPIO_SPEED_LOW; | |
HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); | |
} | |
/* USER CODE BEGIN 4 */ | |
volatile void PI_control(void) | |
{ | |
HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_SET); | |
// Icurrent = center_current - ((float)ADC_raw_data[0] * 0.5f + (float)ADC_raw_data[2] * 0.5f); | |
Icurrent = Smooth_filter(0.5f, center_current - ((float)ADC_raw_data[0] * 0.5f + (float)ADC_raw_data[2] * 0.5f), Icurrent); | |
if (Icurrent < 0) Icurrent = 0; | |
// Iref = ((float)ADC_raw_data[1] * 0.5f + (float)ADC_raw_data[3] * 0.5f) * 0.6f; | |
Iref = Smooth_filter(0.5f, (((float)ADC_raw_data[1] * 0.5f + (float)ADC_raw_data[3] * 0.5f)* 0.6f) - offset_zero, Iref); | |
error = Iref - Icurrent; | |
if (error < 50 && error > -50) | |
{ | |
HAL_GPIO_WritePin(GPIOA, GPIO_PIN_10, GPIO_PIN_SET); | |
}else{ | |
HAL_GPIO_WritePin(GPIOA, GPIO_PIN_10, GPIO_PIN_RESET); | |
} | |
Pwm += error*0.025f; | |
if (Pwm >2300) Pwm = 2300; | |
if (Pwm < 0) Pwm = 0; | |
if (HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_3) == GPIO_PIN_SET ) | |
{ | |
if (HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_2) == GPIO_PIN_SET ) | |
{ | |
F_drive(Pwm); | |
}else{ | |
R_drive(Pwm); | |
} | |
}else{ | |
disable_drive(); | |
} | |
HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_RESET); | |
} | |
volatile void F_drive(float tmp) | |
{ | |
TIM3->CNT = 0; | |
TIM14->CNT = 0; | |
TIM3 -> CCR1 = 0; | |
TIM3 -> CCR2 = 0; | |
TIM3 -> CCR4 = 0; | |
TIM14 -> CCR1 = 0; | |
// HAL_GPIO_WritePin(GPIOB, GPIO_PIN_1, GPIO_PIN_SET); | |
// HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_RESET); | |
if (tmp > 2300) tmp = 2300 ; | |
if (tmp < 0) tmp = 0 ; | |
TIM3 -> CCR2 = tmp; | |
TIM14 -> CCR1 = tmp; | |
} | |
volatile void R_drive(float tmp) | |
{ | |
TIM3->CNT = 0; | |
TIM14->CNT = 0; | |
TIM3 -> CCR1 = 0; | |
TIM3 -> CCR2 = 0; | |
TIM3 -> CCR4 = 0; | |
TIM14 -> CCR1 = 0; | |
// HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_SET); | |
// HAL_GPIO_WritePin(GPIOB, GPIO_PIN_1, GPIO_PIN_RESET); | |
if (tmp > 2300) tmp = 2300; | |
if (tmp < 0) tmp = 0; | |
TIM3 -> CCR1 = tmp; | |
TIM3 -> CCR4 = tmp; | |
} | |
volatile void disable_drive(void) | |
{ | |
TIM3->CNT = 0; | |
TIM14->CNT = 0; | |
// HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_SET); | |
// HAL_GPIO_WritePin(GPIOB, GPIO_PIN_1, GPIO_PIN_SET); | |
TIM3 -> CCR1 = 0; | |
TIM3 -> CCR2 = 0; | |
TIM3 -> CCR4 = 0; | |
TIM14 -> CCR1 = 0; | |
} | |
float Smooth_filter(float alfa, float new_data, float prev_data) | |
{ | |
float output = prev_data + (alfa * (new_data - prev_data)); | |
return output; | |
} | |
/* USER CODE END 4 */ | |
#ifdef USE_FULL_ASSERT | |
/** | |
* @brief Reports the name of the source file and the source line number | |
* where the assert_param error has occurred. | |
* @param file: pointer to the source file name | |
* @param line: assert_param error line source number | |
* @retval None | |
*/ | |
void assert_failed(uint8_t* file, uint32_t line) | |
{ | |
/* USER CODE BEGIN 6 */ | |
/* User can add his own implementation to report the file name and line number, | |
ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */ | |
/* USER CODE END 6 */ | |
} | |
#endif | |
/** | |
* @} | |
*/ | |
/** | |
* @} | |
*/ | |
/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment