-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathmain.c
230 lines (187 loc) · 8.06 KB
/
main.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
/** \file main.c
* \brief Sample STM32F4 project
* \details This file holds a very basic code for STM32F407VG. This code
* enables the FPU, all GPIO ports, configures Flash wait-states and enables
* the PLL to achieve the highest allowed frequency for STM32F407VG (168MHz).
* Main code block just blinks the LED. The LED port and pin are defined in
* config.h file. Target core frequency and quartz crystal resonator
* frequency are defined there as well.
*
* \author Freddie Chopin, http://www.freddiechopin.info/
* \date 2012-03-23
*/
/******************************************************************************
* project: stm32f4_blink_led
* chip: STM32F407VG
* compiler: arm-none-eabi-gcc (GNU Tools for ARM Embedded Processors) 4.6.2
* 20110921 (release) [ARM/embedded-4_6-branch revision 182083]
*
* prefix: (none)
*
* available global functions:
* int main(void)
*
* available local functions:
* static void flash_latency(uint32_t frequency)
* static void fpu_enable(void)
* static uint32_t pll_start(uint32_t crystal, uint32_t frequency)
* static void system_init(void)
*
* available interrupt handlers:
******************************************************************************/
/*
+=============================================================================+
| includes
+=============================================================================+
*/
#include <stdint.h>
#include "inc/stm32f4xx.h"
#include "config.h"
#include "hdr/hdr_rcc.h"
#include "hdr/hdr_gpio.h"
#include "gpio.h"
#include <usbd_audio.h>
#include <usbd_desc.h>
#include <usbd_usr.h>
/*
+=============================================================================+
| module variables
+=============================================================================+
*/
extern USB_OTG_CORE_HANDLE USB_Dev;
/*
+=============================================================================+
| local functions' declarations
+=============================================================================+
*/
static void flash_latency(uint32_t frequency);
static void fpu_enable(void);
static uint32_t pll_start(uint32_t crystal, uint32_t frequency);
static void system_init(void);
/*
+=============================================================================+
| global functions
+=============================================================================+
*/
/*------------------------------------------------------------------------*//**
* \brief main code block
* \details Call some static initialization functions and blink the led with
* frequency defined via count_max variable.
*//*-------------------------------------------------------------------------*/
int main(void)
{
fpu_enable();
system_init();
pll_start(CRYSTAL, FREQUENCY);
gpio_pin_cfg(GPIOD, 12, GPIO_OUT_PP_25MHz);
gpio_pin_cfg(GPIOD, 13, GPIO_OUT_PP_25MHz);
gpio_pin_cfg(GPIOD, 14, GPIO_OUT_PP_25MHz);
gpio_pin_cfg(GPIOD, 15, GPIO_OUT_PP_25MHz);
USBD_Init(&USB_Dev, USB_OTG_FS_CORE_ID, &USR_desc, &AUDIO_cb, &USR_cb);
while(1);
}
/*
+=============================================================================+
| local functions
+=============================================================================+
*/
/*------------------------------------------------------------------------*//**
* \brief Configures Flash latency.
* \details Configures Flash latency (wait-states) which allows the chip to run
* at higher speeds. The constants used for calculations are valid when chip's
* voltage is in 2.7V to 3.6V range.
*
* \param [in] frequency defines the target frequency of the core
*//*-------------------------------------------------------------------------*/
static void flash_latency(uint32_t frequency)
{
uint32_t wait_states;
wait_states = frequency / 30000000ul; // calculate wait_states (30M is valid for 2.7V to 3.6V voltage range, use 24M for 2.4V to 2.7V, 18M for 2.1V to 2.4V or 16M for 1.8V to 2.1V)
wait_states &= 7; // trim to max allowed value - 7
FLASH->ACR = wait_states; // set wait_states, disable all caches and prefetch
FLASH->ACR = FLASH_ACR_DCRST | FLASH_ACR_ICRST | wait_states; // reset caches
FLASH->ACR = FLASH_ACR_DCEN | FLASH_ACR_ICEN | FLASH_ACR_PRFTEN | wait_states; // enable caches and prefetch
}
/*------------------------------------------------------------------------*//**
* \brief Enables FPU
* \details Enables FPU in Cortex-M4 for both privileged and user mode. This is
* done by enabling CP10 and CP11 coprocessors in CPACR register (possible only
* when in privileged mode).
*//*-------------------------------------------------------------------------*/
static void fpu_enable(void)
{
#if (__FPU_PRESENT == 1) && (__FPU_USED == 1)
SCB->CPACR |= ((3UL << 10*2)|(3UL << 11*2)); // set CP10 and CP11 Full Access
#endif
}
/*------------------------------------------------------------------------*//**
* \brief Starts the PLL
* \details Configure and enable PLL to achieve some frequency with some crystal.
* Before the speed change Flash latency is configured via flash_latency(). PLL
* parameters are based on both function parameters. The PLL is set up,
* started and connected. AHB clock ratio is set to 1:1 with core clock, APB1 -
* 1:4 (max 42MHz), APB2 - 1:2 (max 84MHz). USB OTG, RNG and SDIO clock is set
* to (VCO output / 9) (max 48MHz).
*
* \param [in] crystal is the frequency of the crystal resonator connected to the
* STM32F407VG
* \param [in] frequency is the desired target frequency after enabling the PLL
*
* \return real frequency that was set
*//*-------------------------------------------------------------------------*/
static uint32_t pll_start(uint32_t crystal, uint32_t frequency)
{
uint32_t div, mul, div_core, vco_input_frequency, vco_output_frequency, frequency_core;
uint32_t best_div = 0, best_mul = 0, best_div_core = 0, best_frequency_core = 0;
RCC_CR_HSEON_bb = 1; // enable HSE clock
flash_latency(frequency); // configure Flash latency for desired frequency
for (div = 2; div <= 63; div++) // PLLM in [2; 63]
{
vco_input_frequency = crystal / div;
if ((vco_input_frequency < 1000000ul) || (vco_input_frequency > 2000000)) // skip invalid settings
continue;
for (mul = 64; mul <= 432; mul++) // PLLN in [64; 432]
{
vco_output_frequency = vco_input_frequency * mul;
if ((vco_output_frequency < 64000000ul) || (vco_output_frequency > 432000000ul)) // skip invalid settings
continue;
for (div_core = 2; div_core <= 8; div_core += 2) // PLLP in {2, 4, 6, 8}
{
frequency_core = vco_output_frequency / div_core;
if (frequency_core > frequency) // skip values over desired frequency
continue;
if (frequency_core > best_frequency_core) // is this configuration better than previous one?
{
best_frequency_core = frequency_core; // yes - save values
best_div = div;
best_mul = mul;
best_div_core = div_core;
}
}
}
}
RCC->PLLCFGR = (best_div << RCC_PLLCFGR_PLLM_bit) | (best_mul << RCC_PLLCFGR_PLLN_bit) | ((best_div_core / 2 - 1) << RCC_PLLCFGR_PLLP_bit) | RCC_PLLCFGR_PLLQ_DIV7 | RCC_PLLCFGR_PLLSRC_HSE; // configure PLL factors, always divide USB clock by 9
RCC->CFGR = RCC_CFGR_PPRE2_DIV2 | RCC_CFGR_PPRE1_DIV4 | RCC_CFGR_HPRE_DIV1; // AHB - no prescaler, APB1 - divide by 4, APB2 - divide by 2
while (!RCC_CR_HSERDY_bb); // wait for stable clock
RCC_CR_PLLON_bb = 1; // enable PLL
while (!RCC_CR_PLLRDY_bb); // wait for PLL lock
RCC->CFGR |= RCC_CFGR_SW_PLL; // change SYSCLK to PLL
while (((RCC->CFGR) & RCC_CFGR_SWS) != RCC_CFGR_SWS_PLL); // wait for switch
return best_frequency_core;
}
/*------------------------------------------------------------------------*//**
* \brief Initializes system
* \details Enables all GPIO ports
*//*-------------------------------------------------------------------------*/
static void system_init(void)
{
gpio_init();
}
/*
+=============================================================================+
| ISRs
+=============================================================================+
*/
/******************************************************************************
* END OF FILE
******************************************************************************/