|
функция sprintf() и компилятор arm-none-eabi-gcc, не компилирует... вылетает ошибка... |
|
|
|
Jan 26 2013, 06:59
|
Участник

Группа: Участник
Сообщений: 41
Регистрация: 2-08-12
Пользователь №: 72 984

|
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
|
Участник

Группа: Участник
Сообщений: 41
Регистрация: 2-08-12
Пользователь №: 72 984

|
Цитата(AHTOXA @ Jan 26 2013, 07:17)  а printf() заработает или примитивы допиливать надо???
|
|
|
|
|
Jan 26 2013, 08:18
|
Участник

Группа: Участник
Сообщений: 41
Регистрация: 2-08-12
Пользователь №: 72 984

|
Цитата(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
|
|
|
|
|
Jan 26 2013, 08:34
|
Участник

Группа: Участник
Сообщений: 41
Регистрация: 2-08-12
Пользователь №: 72 984

|
нашел другую немного реализацию 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
|
Участник

Группа: Участник
Сообщений: 41
Регистрация: 2-08-12
Пользователь №: 72 984

|
меня смущает существующая секция...
/* 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)); } }
Сообщение отредактировал Посторонним В... - Jan 26 2013, 11:31
|
|
|
|
|
Jan 26 2013, 12:14
|

фанат дивана
     
Группа: Свой
Сообщений: 3 387
Регистрация: 9-08-07
Из: Уфа
Пользователь №: 29 684

|
Цитата(Посторонним В... @ 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, 12:28
|
Участник

Группа: Участник
Сообщений: 41
Регистрация: 2-08-12
Пользователь №: 72 984

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

Группа: Участник
Сообщений: 41
Регистрация: 2-08-12
Пользователь №: 72 984

|
блин...
теперь размер проджекта с 16кб вырос до 50кб
как то можно уменьшить его...???
|
|
|
|
|
Jan 26 2013, 14:30
|
Участник

Группа: Участник
Сообщений: 41
Регистрация: 2-08-12
Пользователь №: 72 984

|
Цитата(_Артём_ @ Jan 26 2013, 15:28)  Можно - выкинуть sprintf и всё что с ним.
Или взять sprintf попроще (без heap и проч. ) где???
|
|
|
|
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0
|
|
|