STM32F4 的FPU 的配置 编译环境为EWARM-6601 首先看下面一段代码:来自core_cm4.h文件 /*!< __FPU_USED to be checked prior to making use of FPU specific registers and functions */ #if defined ( __CC_ARM ) #if defined __TARGET_FPU_VFP #if (__FPU_PRESENT == 1) #define __FPU_USED 1 #else #warning "Compiler generates FPU instructions for a device without an FPU (check __FPU_PRESENT)" #define __FPU_USED 0 #endif #else #define __FPU_USED 0 #endif #elif defined ( __ICCARM__ ) //IAR编译工具的开关 #if defined __ARMVFP__ //要求编译出的文件使用FPU功能 #if (__FPU_PRESENT == 1) //检查是否有FPU功能 #define __FPU_USED 1 #else #warning "Compiler generates FPU instructions for a device without an FPU (check __FPU_PRESENT)" #define __FPU_USED 0 #endif #else #define __FPU_USED 0 #endif #elif defined ( __GNUC__ ) #if defined (__VFP_FP__) && !defined(__SOFTFP__) #if (__FPU_PRESENT == 1) #define __FPU_USED 1 #else #warning "Compiler generates FPU instructions for a device without an FPU (check __FPU_PRESENT)" #define __FPU_USED 0 #endif #else #define __FPU_USED 0 #endif #elif defined ( __TASKING__ ) /* add preprocessor checks to define __FPU_USED */ #define __FPU_USED 0 #endif 这里针对各种工具链设置了相应的条件编译选项,这里只说IAR的,其他工具链的用户请参考本文自行修改. 最上层的开关是 __CC_ARM 这个宏,很明显这个开关是针对MDK工具的,同理IAR工具的开关就叫做 __ICCARM__ 了; 然后一个开关就是__ARMVFP__ 了,这个开关要求编译出的文件使用FPU功能,但是m4的FPU并不是必装设备,某些厂家可能会不装这个东西,所以需要检查是否有FPU,即 __FPU_PRESENT 这个开关。在core_cm4.h中有这样的代码: #ifndef __FPU_PRESENT #define __FPU_PRESENT 0 #warning "__FPU_PRESENT not defined in device header file; using default!" #endif #ifndef __MPU_PRESENT #define __MPU_PRESENT 0 #warning "__MPU_PRESENT not defined in device header file; using default!" #endif 由上述代码可以知道CMSIS默认的定义是没有FPU的,所以需要我们手动添加FPU存在的定义 #define __FPU_PRESENT 1 这样FPU存在与否的检查就通过了,所以后面就自动定义了 __FPU_USED 这一开关。因此就有了下面的定义 #define __FPU_USED 1 因为m4增加了FPU(单精度浮点数),所以很多数学函数都应该用m4专用的,优化过的,而不是之前的通用函数。 打开arm_math.h文件 * Each library project have differant pre-processor macros. * * <b>ARM_MATH_CMx:</b> * Define macro ARM_MATH_CM4 for building the library on Cortex-M4 target, * ARM_MATH_CM3 for building library on Cortex-M3 target * and ARM_MATH_CM0 for building library on cortex-M0 target. * * <b>ARM_MATH_BIG_ENDIAN:</b> * Define macro ARM_MATH_BIG_ENDIAN to build the library for big endian targets. By default library builds for little endian targets. * * <b>ARM_MATH_MATRIX_CHECK:</b> * Define macro for checking on the input and output sizes of matrices * * <b>ARM_MATH_ROUNDING:</b> * Define macro for rounding on support functions * * <b>__FPU_PRESENT:</b> * Initialize macro __FPU_PRESENT = 1 when building on FPU supported Targets. Enable this macro for M4bf and M4lf libraries 以上代码说明:要用ARM的数学函数得定义ARM_MATH_CMx即得有下面的定义 #define ARM_MATH_CM4 如果你想使用STM32F4的FPU功能而又不想管Project中的设置那么使用下面的代码: #define __FPU_PRESENT 1 // FPU开关 #ifndef __ARMVFP__ //要求在生成的代码中使用FPU #define __ARMVFP__ #endif #ifndef ARM_MATH_CM4 //要求使用m4的数学库函数 #define ARM_MATH_CM4 #endif 也可以在软件中开FPU功能IAR的设置如下:
MDK中的设置如下:
对于Cortex-M0或者M3处理器类型,由于没有FPU因此无法直接进行浮点运算,只能将浮点数进行Q规格化 (q7、 q15或Q31)处理,如开平方运算: M0/M3只能通过迭代法 ( 标准数学函数库 ) 计算,而M4直接调用VSQRT指令完成。 DSP_Lib的文件结构
事实上arm_math.h文件中的定义就为源文件组使用的。 因此在将源文件组编译为库时,在应用工程中添加这个库和arm_math.h即可访问所有DSP库功能。也可以根据需要只添加arm_math.h和需要的源文件,以缩短程序长度。 (1) BasicMathFunctions源文件组 提供浮点数的各种基本运算函数,如加减乘除等运算。对于M0/M3只能用Q运算,即文件夹下以 _q7、_q15和 _q31结尾的文件;而M4能直接硬件浮点计算,属于文件夹下以 _f32结尾的文件。 (2) CommonTables源文件组 arm_common_tables.c文件提供位翻转或相关参数表。 (3) ComplexMathFunctions源文件组 复数数学功能,如向量处理,求模运算的。 (4) ControllerFunctions源文件组 控制功能,主要为PID控制函数。arm_sin_cos_f32/-q31.c函数提供360点正余弦函数表和任意角度的正余弦函数值计算功能。 (5) FastMathFunctions源文件组 快速数学功能函数, 提供256点正余弦函数表和任意角度的正余弦函数值计算功能,和Q值开平方运算: Arm_cos_f32/_q15/_q31.c:提供256点余弦函数表和任意角度余弦值计算功能。 Arm_sin_f32/_q15/_q31.c:提供256点正弦函数表和任意角度正弦值计算功能。 Arm_sqrt_q15/q31.c :提供迭代法计算平方根的函数。 对于M4的平方根运算,通过执行VSQRT指令完成。 (6) FilteringFunctions源文件组 滤波函数功能,主要为FIR和LMS (最小均方根)滤波函数。 (7) MatrixFunctions源文件组 矩阵处理函数。 (8) StatisticsFunctions源文件组 统计功能函数,如求平均值、计算RMS 、计算方差 / 标准差等。 (9) SupportFunctions源文件组 支持功能函数,如数据拷贝, Q格式和浮点格式相互转换, Q任意格式相互转换。 (10) TransformFunctions源文件组 变换功能,包括复数FFT ( CFFT) / 复数FFT逆运算 ( CIFFT) 、 实数FFT (RFFT) / 实数FFT逆运算(RIFFT ) 、和DCT (离散余弦变换)和配套的初始化函数。 在IAR的环境下运行一个简单的浮点运算: 主函数如下:
通过IAR软件Terminal I/O 窗口观察的输出结果 |