Привет, мир, металлическая доска Beagleboard - PullRequest
10 голосов
/ 04 февраля 2012

Я пытаюсь запустить программу типа «Привет, мир» на моем Beagleboard-xm rev.C, вызывая функцию C puts из сборки.

До сих пор я использовал это в качестве ссылки: http://wiki.osdev.org/ARM_Beagleboard

Вот то, что я пока имею, но нетвывод.

hello.c

volatile unsigned int * const UART3DR = (unsigned int *)0x49020000;

void puts(const char *s) {
  while(*s != '\0') { 
    *UART3DR = (unsigned int)(*s); 
    s++; 
  }
}

void hello() {
  puts("Hello, Beagleboard!\n");
}

boot.asm

.global start
start:
   ldr sp, =stack_bottom
   bl hello
   b .

linker.ld

ENTRY(start)

MEMORY
{
    ram : ORIGIN = 0x80200000, LENGTH = 0x10000
}

SECTIONS
{
    .hello : { hello.o(.text) } > ram
    .text : { *(.text) } > ram
    .data : { *(.data) } > ram
    .bss : { *(.bss) } > ram
     . = . + 0x5000; /* 4kB of stack memory */
    stack_bottom = .;

}

Makefile

ARMGNU = arm-linux-gnueabi

AOPS = --warn --fatal-warnings
COPS = -Wall -Werror -O2 -nostdlib -nostartfiles -ffreestanding

boot.bin: boot.asm
   $(ARMGNU)-as boot.asm -o boot.o
   $(ARMGNU)-gcc-4.6 -c $(COPS) hello.c -o hello.o
   $(ARMGNU)-ld -T linker.ld hello.o boot.o -o boot.elf
   $(ARMGNU)-objdump -D boot.elf > boot.list
   $(ARMGNU)-objcopy boot.elf -O srec boot.srec
   $(ARMGNU)-objcopy boot.elf -O binary boot.bin

Использование только asm-файла, как это работает.

.equ UART3.BASE,        0x49020000
start:
   ldr r0,=UART3.BASE
   mov r1,#'c'

Вот некоторые Beagleboard/ Информация, связанная с Minicom: http://paste.ubuntu.com/829072/

Есть указатели?:)


Я также пытался

void hello() {
  *UART3DR = 'c';
}

Я использую Minicom и отправляю файл через ymodem, затем я пытаюсь запустить его с:

go 0x80200000

Поток управления аппаратным и программным обеспечением в minicom выключен.

Ответы [ 3 ]

3 голосов
/ 05 февраля 2012

это должно сработать для вас. Вот некоторый код, который я откопал в далеком прошлом, когда не пробовал его сегодня на beagleboard, просто убедился, что он скомпилирован, он работал одно время ...

startup.s:

    .code 32

.globl _start
_start:

    bl main
hang: b hang

.globl PUT32
PUT32:
    str r1,[r0]
    bx lr

.globl GET32
GET32:
    ldr r0,[r0]
    bx lr

hello.c:

extern void PUT32 ( unsigned int, unsigned int );
extern unsigned int GET32 ( unsigned int );
void uart_send ( unsigned char x )
{
    while((GET32(0x49020014)&0x20)==0x00) continue;
    PUT32(0x49020000,x);
}
void hexstring ( unsigned int d )
{
    //unsigned int ra;
    unsigned int rb;
    unsigned int rc;

    rb=32;
    while(1)
    {
        rb-=4;
        rc=(d>>rb)&0xF;
        if(rc>9) rc+=0x37; else rc+=0x30;
        uart_send(rc);
        if(rb==0) break;
    }
    uart_send(0x0D);
    uart_send(0x0A);
}
int main ( void )
{
    hexstring(0x12345678);
    return(0);
}

memmap (скрипт компоновщика):

MEMORY
{
    ram : ORIGIN = 0x82000000, LENGTH = 256K
}

SECTIONS
{
    ROM : { startup.o } > ram
}

Makefile:

CROSS_COMPILE = arm-none-eabi

AOPS = --warn --fatal-warnings 
COPS = -Wall -Werror -O2 -nostdlib -nostartfiles -ffreestanding 

all : hello.bin

hello.bin : startup.o hello.o memmap
    $(CROSS_COMPILE)-ld startup.o hello.o -T memmap -o hello.elf  
    $(CROSS_COMPILE)-objdump -D hello.elf > hello.list
    $(CROSS_COMPILE)-objcopy hello.elf -O binary hello.bin

startup.o : startup.s
    $(CROSS_COMPILE)-as $(AOPS) startup.s -o startup.o

hello.o : hello.c 
    $(CROSS_COMPILE)-gcc -c $(COPS) hello.c -o hello.o

clean :
    rm -f *.o
    rm -f *.elf
    rm -f *.bin
    rm -f *.list

Похоже, я просто оставил указатель стека там, где его имел загрузчик. Точно так же, как вы и предполагали, загрузчик инициализировал последовательный порт.

Я полагаю, у вас работает доступ к последовательному порту, вы видите uboot и можете набирать команды для загрузки этой программы (xmodem или чего-то еще) в ram плат? Если вы не можете сделать это, то, возможно, вы не подключены к последовательному порту прямо. последовательный порт beagleboards не очень удобен, возможно, вам понадобится собственный кабель.

2 голосов
/ 06 февраля 2012

Мне не хватает повторения, чтобы комментировать .. Но мой ответ на

Работает в любом случае. Теперь странно то, что я могу печатать отдельные символы с символом uart_send ('c'), например, но не могут печатать строки print_string (char * str) {while (* str! = '\ 0') uart_send (* Ул ++); } print_string ("Test"); , Есть мысли по этому поводу?

есть:

Вы пишете быстрее в буфере вывода, так как UART может отправлять. Поэтому перед отправкой нового символа вы должны проверить, пуст ли буфер вывода.

Я сделал это в коде моего блога (http://hardwarefreak.wordpress.com/2011/08/30/some-experience-with-the-beagleboard-xm-part-2/)

2 голосов
/ 04 февраля 2012

Вы не можете просто слепо записать строку символов в UART - вам нужно проверить состояние каждого символа - это работает в примере с одним символом, потому что UART всегда будет готов для первого символа, но длявторой и последующие символы, которые нужно опрашивать (или еще лучше использовать ISR, но давайте пройдемся до того, как запустим).

Здесь есть хороший пример кода: http://hardwarefreak.wordpress.com/2011/08/30/some-experience-with-the-beagleboard-xm-part-2/

...