Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: функция sprintf() и компилятор arm-none-eabi-gcc
Форум разработчиков электроники ELECTRONIX.ru > Cистемный уровень проектирования > Операционные системы > scmRTOS
Посторонним В...
In function `_sbrk_r': sbrkr.c:(.text._sbrk_r+0xc): undefined reference to `_sbrk' collect2.exe: error: ld returned 1 exit status

поиск по гуглю показал что при компиляции надо указать какой то ключ то ли написать свои функции `_sbrk_r' и еще там парочка... и править ld файл
скомпилить функцию под gcc для msp430 - все скомпилилось отлично

кто то решал такую проблему???

где копать код???
AHTOXA
Вот тут решение.
Посторонним В...
Цитата(AHTOXA @ Jan 26 2013, 07:17) *
Вот тут решение.


а printf() заработает или примитивы допиливать надо???
AHTOXA
Надо определить функцию putChar(). (См. ф-ю _write())
Посторонним В...
Цитата(AHTOXA @ Jan 26 2013, 08:15) *
Надо определить функцию putChar(). (См. ф-ю _write())



другая ошибка выскочила....


In function `_sbrk':
./src/stf_syscalls_minimal.c:114: undefined reference to `_heap'
./src/stf_syscalls_minimal.c:114: undefined reference to `_heap'
makefile:383: recipe for target `bin/charger_test.elf' failed
./src/stf_syscalls_minimal.c:124: undefined reference to `_eheap'
./src/stf_syscalls_minimal.c:124: undefined reference to `_eheap'
collect2.exe: error: ld returned 1 exit status
make: *** [bin/charger_test.elf] Error 1
AHTOXA
_heap и _eheap должны быть определены в скрипте линкера.
Можете посмотреть пример в скриптах для scmRTOS.
Или вот я выкладывал тестовые проекты:

Посторонним В...
нашел другую немного реализацию syscalls.c :

с ним проект без правки ld файла компилица но нет возможности проверить работает или нет...

/****************************************************************************
* Copyright © 2009 by Michael Fischer. All rights reserved.
*
* 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 the author 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 OWNER 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.
*
****************************************************************************
* History:
*
* 28.03.09 mifi First Version, based on the original syscall.c from
* newlib version 1.17.0
****************************************************************************/

#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/types.h>

//------------------------------------------------------------------------------

int _read_r (struct _reent *r, int file, char * ptr, int len)
{
r = r;
file = file;
ptr = ptr;
len = len;

errno = EINVAL;

return (-1);
}

//------------------------------------------------------------------------------

int _lseek_r (struct _reent *r, int file, int ptr, int dir)
{
r = r;
file = file;
ptr = ptr;
dir = dir;

return (0);
}

//------------------------------------------------------------------------------

int _write_r (struct _reent *r, int file, char * ptr, int len)
{
r = r;
file = file;
ptr = ptr;

#if 0

int index;

/* For example, output string by UART */
for (index=0; index<len; index++)
{
if (ptr[index] == '\n')
{
uart_putc('\r');
}

uart_putc(ptr[index]);
}

#endif

return (len);
}

//------------------------------------------------------------------------------

int _close_r (struct _reent *r, int file)
{
return (0);
}

//------------------------------------------------------------------------------

/* Register name faking - works in collusion with the linker. */
register char *stack_ptr asm ("sp");

caddr_t _sbrk_r (struct _reent *r, int incr)
{
extern char end asm ("end"); /* Defined by the linker. */
static char *heap_end;
char *prev_heap_end;

if (heap_end == NULL) heap_end = &end;

prev_heap_end = heap_end;

if (heap_end + incr > stack_ptr)
{
/* Some of the libstdc++-v3 tests rely upon detecting out of memory errors, so do not abort here. */
#if 0

extern void abort (void);

_write (1, "_sbrk: Heap and stack collision\n", 32);

abort ();

#else

errno = ENOMEM;

return ((caddr_t)(-1));

#endif

}

heap_end += incr;

return ((caddr_t)(prev_heap_end));
}

//------------------------------------------------------------------------------

int _fstat_r (struct _reent *r, int file, struct stat *st)
{
r = r;
file = file;

memset (st, 0, sizeof (*st));

st->st_mode = S_IFCHR;

return (0);
}

//------------------------------------------------------------------------------

int _isatty_r(struct _reent *r, int fd)
{
r = r;
fd = fd;

return (1);
}

//------------------------------------------------------------------------------
// added
//------------------------------------------------------------------------------

int _getpid ()
{
return (1);
}

//------------------------------------------------------------------------------

void _exit (int i)
{
// printf("Program exit with code %d", i);

i = i;

while (1);
}

//------------------------------------------------------------------------------

int _kill (int pid, int sig)
{
(void)pid;
(void)sig;

errno = EINVAL;

return (-1);
}

//------------------------------------------------------------------------------
Посторонним В...

http://we.easyelectronics.ru/STM32/primene...-v-stm32f4.html
Посторонним В...
меня смущает существующая секция...

/* User_heap_stack section, used to check that there is enough RAM left */
._user_heap_stack :
{
. = ALIGN(4);
PROVIDE ( end = . );
PROVIDE ( _end = . );
. = . + _Min_Heap_Size;
. = . + _Min_Stack_Size;
. = ALIGN(4);
} >RAM

и вот это определение

PROVIDE ( _heap = _ebss );
PROVIDE ( _eheap = ALIGN(ORIGIN(RAM) + LENGTH(RAM) - 8 - _Min_Stack_Size , 8) );

это одно и то же или я ошибаюсь???



можно ли тогда функцию переписать вот так???

caddr_t _sbrk(int incr)
{
extern char end asm ("end"); /* Defined by the linker. */
extern char _end asm ("_end"); /* Defined by the linker. */

caddr_t prevHeap;
caddr_t nextHeap;

if (heap == NULL)
{
// first allocation
// heap = ((caddr_t)(&_heap));
heap = ((caddr_t)(&end));
}

prevHeap = heap;

// Always return data aligned on a 8 byte boundary
nextHeap = (caddr_t) (((unsigned int) (heap + incr) + 7) & ~7);

// Check enough space and there is no collision with stack coming the other way
// if stack is above start of heap
// if ( nextHeap >= ( (caddr_t)(&_eheap) ) )
if ( nextHeap >= ( (caddr_t)(&_end) ) )
{
errno = ENOMEM;

return (NULL); // error - no more memory
}
else
{
heap = nextHeap;

return ((caddr_t)(prevHeap));
}
}

AHTOXA
Цитата(Посторонним В... @ Jan 26 2013, 17:30) *
это одно и то же или я ошибаюсь???

Это не совсем одно и то же. Первый кусок просто проверяет, что после распределения всех переменных в ОЗУ осталось место для стека и кучи (Min_Heap_Size + _Min_Stack_Size).
Второй кусок - даёт имена началу и концу куска ОЗУ, предназначенного для кучи.
Цитата(Посторонним В... @ Jan 26 2013, 17:30) *
можно ли тогда функцию переписать вот так???
Код
caddr_t _sbrk(int incr)
{
...
//    if ( nextHeap >= ( (caddr_t)(&_eheap) ) )
    if ( nextHeap >= ( (caddr_t)(&_end) ) )
    {
...
}

_eheap и _end - это разные адреса. _eheap - это конец кучи, а _end - её начало (то есть конец секции bss).
ЗЫ. Используйте для оформления кода кнопку [cоde]
Посторонним В...
Цитата(AHTOXA @ Jan 26 2013, 13:14) *
Это не совсем одно и то же. Первый кусок просто проверяет, что после распределения всех переменных в ОЗУ осталось место для стека и кучи (Min_Heap_Size + _Min_Stack_Size).
Второй кусок - даёт имена началу и концу куска ОЗУ, предназначенного для кучи.

_eheap и _end - это разные адреса. _eheap - это конец кучи, а _end - её начало (то есть конец секции bss).
ЗЫ. Используйте для оформления кода кнопку [cоde]



все понятно...
Посторонним В...
блин...

теперь размер проджекта с 16кб вырос до 50кб

как то можно уменьшить его...???
_Артём_
Цитата(Посторонним В... @ Jan 26 2013, 16:06) *
блин...

теперь размер проджекта с 16кб вырос до 50кб

как то можно уменьшить его...???

Можно - выкинуть sprintf и всё что с ним.

Или взять sprintf попроще (без heap и проч. )
Посторонним В...
Цитата(_Артём_ @ Jan 26 2013, 15:28) *
Можно - выкинуть sprintf и всё что с ним.

Или взять sprintf попроще (без heap и проч. )



где???
AHTOXA
Например, вот здесь.
Посторонним В...
Цитата(AHTOXA @ Jan 26 2013, 15:46) *
Например, вот здесь.


сенкью...

проверю сейчас...

./src/printfs.c:42: multiple definition of `vsprintf'
./obj/printfs.o:./src/printfs.c:42: first defined here
./obj/debug.o: In function `sprintf':
./src/printfs.c:26: multiple definition of `sprintf'
makefile:323: recipe for target `bin/charger.elf' failed
./obj/printfs.o:./src/printfs.c:26: first defined here

ошибку выдает

как подключить то либу эту???
Сергей Борщ
QUOTE (Посторонним В... @ Jan 26 2013, 17:50) *
как подключить то либу эту???
Которую из? Которая к сообщению прикреплена? Просто добавить в проект. А вы что, через #include ее подключали? Так нельзя. Надо делать #include <stdio.h>
juvf
а при чём тут scmRTOS?
Для просмотра полной версии этой страницы, пожалуйста, пройдите по ссылке.
Invision Power Board © 2001-2025 Invision Power Services, Inc.