Посторонним В...
Jan 26 2013, 06:59
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 - все скомпилилось отлично
кто то решал такую проблему???
где копать код???
Посторонним В...
Jan 26 2013, 08:01
Цитата(AHTOXA @ Jan 26 2013, 07:17)

а printf() заработает или примитивы допиливать надо???
AHTOXA
Jan 26 2013, 08:15
Надо определить функцию putChar(). (См. ф-ю _write())
Посторонним В...
Jan 26 2013, 08:18
Цитата(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
Jan 26 2013, 08:25
_heap и _eheap должны быть определены в скрипте линкера.
Можете посмотреть пример в
скриптах для scmRTOS.
Или вот я выкладывал тестовые проекты:
Посторонним В...
Jan 26 2013, 08:34
нашел другую немного реализацию 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);
}
//------------------------------------------------------------------------------
Посторонним В...
Jan 26 2013, 11:30
меня смущает существующая секция...
/* 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, 12:14
Цитата(Посторонним В... @ 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]
Посторонним В...
Jan 26 2013, 14:06
блин...
теперь размер проджекта с 16кб вырос до 50кб
как то можно уменьшить его...???
_Артём_
Jan 26 2013, 14:28
Цитата(Посторонним В... @ Jan 26 2013, 16:06)

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

Можно - выкинуть sprintf и всё что с ним.
Или взять sprintf попроще (без heap и проч. )
где???
Посторонним В...
Jan 26 2013, 15:50
Цитата(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
ошибку выдает
как подключить то либу эту???
Сергей Борщ
Jan 26 2013, 19:41
QUOTE (Посторонним В... @ Jan 26 2013, 17:50)

как подключить то либу эту???
Которую из? Которая к сообщению прикреплена? Просто добавить в проект. А вы что, через #include ее подключали? Так нельзя. Надо делать #include <stdio.h>