召唤ARM工具链和eclipse
Posted
技术标签:
【中文标题】召唤ARM工具链和eclipse【英文标题】:summon-ARM toolchain and eclipse 【发布时间】:2013-12-17 08:32:23 【问题描述】:我很想在 Linuxmint Nadia MATE 中让召唤臂工具链与 eclipse 一起工作。我正在尝试编译一个简单的 LED 闪光灯示例。我已将 STM32F4 的所有目录添加到包含路径中,并且 #include 语句已解析,但我得到了各种未解析的引用。 ~/bin/sat 在我的 $PATH 中,并且很明显可以找到工具链。我还得到了一个有趣的未定义引用 `_exit',它指向我的下载目录中的某个地方,我不明白为什么 Eclipse 会看那里。
我是 Eclipse(可能很明显)和 ARM 平台的真正新手,但在 Windows 环境中编程 PIC 方面经验丰富。
我觉得我错过了一些相当简单的东西(除了那个 '__exit' 东西),但我还没有找到 SO 的启发性时刻(即使同样类型的问题不断出现,我也可以' t 似乎找到了解决方案)。想在尝试 CooCox 之前再问一次。
#include <stm32f4xx_conf.h>
#include "stm32f4xx_gpio.h"
#include "stm32f4_discovery.h"
#include "stm32f4xx_rcc.h"
void Delay(__IO uint32_t nCount)
while(nCount--)
void init_GPIO(void)
GPIO_InitTypeDef GPIO_InitStruct;
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE);
GPIO_InitStruct.GPIO_Pin = GPIO_Pin_15 | GPIO_Pin_14 | GPIO_Pin_13 | GPIO_Pin_12;
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_OUT;
GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStruct.GPIO_OType = GPIO_OType_PP;
GPIO_InitStruct.GPIO_PuPd = GPIO_PuPd_NOPULL;
GPIO_Init(GPIOD, &GPIO_InitStruct);
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0; // we want to configure PA0
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN; // we want it to be an input
GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;//this sets the GPIO modules clock speed
GPIO_InitStruct.GPIO_OType = GPIO_OType_PP; // this sets the pin type to push / pull (as opposed to open drain)
GPIO_InitStruct.GPIO_PuPd = GPIO_PuPd_DOWN; // this enables the pulldown resistor --> we want to detect a high level
GPIO_Init(GPIOA, &GPIO_InitStruct); // this passes the configuration to the Init function which takes care of the low level stuff
void main()
init_GPIO();
GPIOD->BSRRL = 0xF000; // set PD12 thru PD15
Delay(1000000L); // wait a short period of time
GPIOD->BSRRH = 0xF000; // reset PD12 thru PD15
uint8_t i = 0;
while(1)
/* Every GPIO port has an input and
* output data register, ODR and IDR
* respectively, which hold the status of the pin
*
* Here the IDR of GPIOA is checked whether bit 0 is
* set or not. If it's set the button is pressed
*/
if(GPIOA->IDR & 0x0001)
// if the number of button presses is greater than 4, reset the counter (we start counting from 0!)
if(i > 3)
i = 0;
else // if it's smaller than 4, switch the LEDs
switch(i)
case 0:
GPIOD->BSRRL = 0x1000; // this sets LED1 (green)
GPIOD->BSRRH = 0x8000; // this resets LED4 (blue)
break;
case 1:
GPIOD->BSRRL = 0x2000; // this sets LED2 (orange)
GPIOD->BSRRH = 0x1000; // this resets LED1
break;
case 2:
GPIOD->BSRRL = 0x4000; // this sets LED3 (red)
GPIOD->BSRRH = 0x2000; // this resets LED2
break;
case 3:
GPIOD->BSRRL = 0x8000; // this sets LED4
GPIOD->BSRRH = 0x4000; // this resets LED3
break;
i++; // increase the counter every time the switch is pressed
Delay(3000000L); // add a small delay to debounce the switch
当我构建时,我得到
make all
Building file: ../src/main.c
Invoking: Cross ARM C Compiler
arm-none-eabi-gcc -mcpu=cortex-m4 -mthumb -O0 -fmessage-length=0 -fsigned-char -ffunction-sections -fdata-sections -Wall -g3 -I"/home/scott/workspace/STM32 libraries/CMSIS/Include" -I"/home/scott/workspace/STM32 libraries/CMSIS/ST/STM32F4xx/Include" -I"/home/scott/workspace/STM32 libraries/STM32F4xx_StdPeriph_Driver/inc" -I"/home/scott/workspace/STM32 libraries/STM32F4-Discovery" -I/home/scott/workspace/blinky2/src -MMD -MP -MF"src/main.d" -MT"src/main.d" -c -o "src/main.o" "../src/main.c"
../src/main.c:41:6: warning: return type of 'main' is not 'int' [-Wmain]
void main()
^
Finished building: ../src/main.c
Building target: blinky2.elf
Invoking: Cross ARM C Linker
arm-none-eabi-gcc -mcpu=cortex-m4 -mthumb -O0 -fmessage-length=0 -fsigned-char -ffunction-sections -fdata-sections -Wall -g3 -Xlinker --gc-sections -Wl,-Map,"blinky2.map" -o "blinky2.elf" ./src/main.o
./src/main.o: In function `init_GPIO':
/home/scott/workspace/blinky2/Debug/../src/main.c:21: undefined reference to `RCC_AHB1PeriphClockCmd'
/home/scott/workspace/blinky2/Debug/../src/main.c:27: undefined reference to `GPIO_Init'
/home/scott/workspace/blinky2/Debug/../src/main.c:28: undefined reference to `RCC_AHB1PeriphClockCmd'
/home/scott/workspace/blinky2/Debug/../src/main.c:36: undefined reference to `GPIO_Init'
/home/scott/sat/lib/gcc/arm-none-eabi/4.8.2/../../../../arm-none-eabi/lib/thumb/cortex-m4/libg.a(lib_a-exit.o): In function `exit':
/home/scott/Downloads/summon-arm-toolchain-master/build/arm-none-eabi/thumb/cortex-m4/newlib/libc/stdlib/../../../../../../../gcc-linaro-4.8-2013.07-1/newlib/libc/stdlib/exit.c:65: undefined reference to `_exit'
collect2: error: ld returned 1 exit status
make: *** [blinky2.elf] Error 1
11:39:28 Build Finished (took 3s.124ms)
【问题讨论】:
您对我的回答满意吗? 不完全。主要部分是 stm32 库中未解决的调用,以及为什么退出调用正在查找它甚至应该知道的目录。不过谢谢。我开始认为我可能需要在我的链接路径中包含库中的 .c 文件,但还没有机会测试它 好的,我用解释编辑了我的答案。关于 stm32 库,您会收到未定义的引用错误,因为您没有添加库的 .c 文件,您必须将这些文件添加到正在编译的文件列表中。 谢谢。我将尝试验证是否需要在包含路径中包含库中的 c 文件。在过去的 PIC 时代,从未见过! 【参考方案1】:您的工具链正在使用 newlib C 库。 Newlib 是一个非常灵活的最小 C 库,这种灵活性是通过让您定义一些作为库的一部分的函数来实现的。这些函数称为“存根”,请参阅this site,了解一些 newlib 存根的示例实现。例如,通过定义这些函数,您可以使用 UART、USB 等拥有 printf
。
在您的情况下,文件exit.c
正在调用您的存根实现_exit
。您可以实现一个虚拟存根,因为在嵌入式系统上通常不需要此系统调用。
编辑:
您可能使用here 提供的构建脚本构建了召唤臂工具链。此脚本下载 newlib 并在 GCC 安装文件夹中添加一个指向 newlib 下载位置的符号链接:
# line 420 and 421 of the script:
log "Adding newlib symlink to gcc"
ln -f -s `pwd`/$NEWLIB/newlib $GCC
由于您可能在文件夹 /home/scott/Downloads/summon-arm-toolchain-master/
中构建了您的工具链,因此 newlib 库现在位于此路径的子文件夹中,构建脚本下载它的位置。
【讨论】:
谢谢!所以我只需要找到该链接并将其指向正确的路径,然后制作我自己的 exit.c以上是关于召唤ARM工具链和eclipse的主要内容,如果未能解决你的问题,请参考以下文章
带有 Eclipse 的 GNU blackfin 工具链所需的教程