|
Не могу решить проблему вызова функций IAR, используется 1 раз, но вызывается call |
|
|
|
Oct 26 2005, 23:29
|

Познающий...
     
Группа: Свой
Сообщений: 2 963
Регистрация: 1-09-05
Из: г. Иркутск
Пользователь №: 8 125

|
Здравствуйте! Моя программа состоит из нескольких модулей. Есть главный модуль main.c, в нем для инициализации различных частей программы вызываются функции из других модулей. Причем эти функции используются однократно, при запуске программы. Уровень оптимизации стоит максимальный и по размеру кода. Просмотрев дизассемблированный текст, я увидел, что эти функции вызываются с помощью rcall. Нельзя ли что-нибудь настроить, чтоб тело этих функции помещалось на место команды rcall, т.е. чтобы они не вызывались (ведь больше они не где не используются), таким образом экономим память (оптимизация то по размеру).
В общем мой вопрос так звучит: как сделать, чтоб однократно используемые функции не вызывались, а их тело помещалось вместо вызова подпрограммы.
--------------------
Выбор.
|
|
|
|
|
Oct 26 2005, 23:39
|

Гуру
     
Группа: Свой
Сообщений: 2 720
Регистрация: 24-03-05
Пользователь №: 3 659

|
Цитата(haker_fox @ Oct 27 2005, 07:29) В общем мой вопрос так звучит: как сделать, чтоб однократно используемые функции не вызывались, а их тело помещалось вместо вызова подпрограммы. Вообще для этого в IAR (не знаю как в других средах) есть директива #pragma inline=forced. Но язаметил такую особенность - если функция , например void if(), объявленная как inline, и функция которая вызывает void if(), находятся в одном модуле, то void if(), компилируется как inline без call и ret. Если же функции в разных модулях, то функция void if(), вызывается через call.
--------------------
|
|
|
|
|
Oct 26 2005, 23:58
|

Познающий...
     
Группа: Свой
Сообщений: 2 963
Регистрация: 1-09-05
Из: г. Иркутск
Пользователь №: 8 125

|
Цитата Вообще для этого в IAR (не знаю как в других средах) есть директива #pragma inline=forced. Но язаметил такую особенность - если функция , например void if(), объявленная как inline, и функция которая вызывает void if(), находятся в одном модуле, то void if(), компилируется как inline без call и ret. Если же функции в разных модулях, то функция void if(), вызывается через call. А нельзя ли это автоматизировать, что сам компилятор это выполнял. А то трудно уследить какие функции будут использоваться однократно.
--------------------
Выбор.
|
|
|
|
|
Oct 28 2005, 10:55
|

Частый гость
 
Группа: Свой
Сообщений: 112
Регистрация: 15-10-05
Из: Томск
Пользователь №: 9 680

|
Цитата(haker_fox @ Oct 27 2005, 06:29) В общем мой вопрос так звучит: как сделать, чтоб однократно используемые функции не вызывались, а их тело помещалось вместо вызова подпрограммы. 1) Нужно чтобы вызываемые ф-ии были в одном модуле с вызывающей ф-ей. 2) Нужно написать модификатор static перед вызываемой ф-ей. 3) К этим пунктами иногда тербуется включить оптимизацию по скорости (проверь экспериментально). Тогда компилятор сам достаточно умно решает инлайнить ф-ю или нет.
|
|
|
|
|
Oct 28 2005, 14:41
|

Частый гость
 
Группа: Свой
Сообщений: 112
Регистрация: 15-10-05
Из: Томск
Пользователь №: 9 680

|
Цитата(haker_fox @ Oct 28 2005, 21:29) Но если разместить функции в одном модуле, то пропадет наглядность программы, видимо надо чихнуть на этот "беспредел" компилятора. Можешь модуль разбить на несколько с помощью #include  Вообще это свойство компилятора. Ф-ии для работающие вместе должны объединяться в модулях. Если есть ф-я, которую нужно инлайнить в нескольких модулях , то объяви её static и запихни в подключенный в эти моули .h файл. Т.о. модули компилируются отдельно, при этом компилятор ничего не знает о ф-ях в других модулях, кроме прототипов. И только потом линкером сгенерированный из модулей код объединяется.
|
|
|
|
|
Oct 28 2005, 15:15
|

Гуру
     
Группа: Свой
Сообщений: 2 720
Регистрация: 24-03-05
Пользователь №: 3 659

|
Цитата(starter48 @ Oct 28 2005, 22:41) Можешь модуль разбить на несколько с помощью #include  Вообще это свойство компилятора. Ф-ии для работающие вместе должны объединяться в модулях. Если есть ф-я, которую нужно инлайнить в нескольких модулях , то объяви её static и запихни в подключенный в эти моули .h файл. Т.о. модули компилируются отдельно, при этом компилятор ничего не знает о ф-ях в других модулях, кроме прототипов. И только потом линкером сгенерированный из модулей код объединяется. Компилятор то знает все обо всех функциях, на то он и компилятор. Другое дело, что он не подставляет тело функции из одного модуля, в тело функции из другого. Но если написать тело в .h файл, возможно это и прокатит. Но тут такая белиберда получится, что ой-ой. Допустим в модуле объявлена глобальная переменная, а нужная нам инлайн-функция должна работать с этой пременной. Как этой функции, которая описанна в хедере, рассказать что в одноименном модуле для него приготовленна переменная. Писать еще одну функцию в модуле, которая будет возвращать значение переменной инлайн-функции?
--------------------
|
|
|
|
|
Oct 28 2005, 15:52
|

Частый гость
 
Группа: Свой
Сообщений: 112
Регистрация: 15-10-05
Из: Томск
Пользователь №: 9 680

|
Цитата(prottoss @ Oct 28 2005, 22:15) Компилятор то знает все обо всех функциях, на то он и компилятор. Другое дело, что он не подставляет тело функции из одного модуля, в тело функции из другого. Нет, компилятор знает только прототип ф-ии из .h для внешнего модуля, но не тело ф-ии. Исходя из прототипа производится проверка числа и типов аргументов, а в объектный файл ставится rcall без адреса. Реальный адрес проставляется линкером на этапе линковки. Цитата(prottoss @ Oct 28 2005, 22:15) Но если написать тело в .h файл, возможно это и прокатит. Но тут такая белиберда получится, что ой-ой. Допустим в модуле объявлена глобальная переменная, а нужная нам инлайн-функция должна работать с этой пременной. Как этой функции, которая описанна в хедере, рассказать что в одноименном модуле для него приготовленна переменная. Писать еще одну функцию в модуле, которая будет возвращать значение переменной инлайн-функции? Нужно объявить переменную как extern в .h файле с функцией. Т.е. в основном модуле пишем: Код #include"submodule.h"
int myvar;
void main (void) { myfunc(); } а в submodule.h пишем: Код #ifndef __submodule_h #define __submodule_h
extern int myvar;
static void myfunc (void) { ..... здесь используем myvar }
#endif
|
|
|
|
|
Oct 28 2005, 16:39
|

Частый гость
 
Группа: Свой
Сообщений: 112
Регистрация: 15-10-05
Из: Томск
Пользователь №: 9 680

|
Цитата(prottoss @ Oct 28 2005, 23:12) Тогда объясните зачем вообще нужны отдельные модули, раз уж на то пошло. 1) Проще разбираться в исходниках, когда программа разбита на идеологически законченные блоки. 2) К программе можно прилинковать модуль написанный на другом языке. 3) Каждый модуль можно скомпилировать со своими настройками компилятора. 4) Программа быстрее перекомпилируется, т.к. компилятор компилирует только изменившиеся модули. Цитата(prottoss @ Oct 28 2005, 23:12) Давайте будем писать все в одном модуле, чтоб все все знали обо всех :-) Да, если проект не очень большой, то так обычно и делается.
|
|
|
|
|
Oct 28 2005, 17:11
|

Частый гость
 
Группа: Свой
Сообщений: 112
Регистрация: 15-10-05
Из: Томск
Пользователь №: 9 680

|
Цитата(prottoss @ Oct 28 2005, 23:53) Цитата(starter48 @ Oct 29 2005, 00:39) 1) Проще разбираться в исходниках, когда программа разбита на идеологически законченные блоки. 2) К программе можно прилинковать модуль написанный на другом языке. 3) Каждый модуль можно скомпилировать со своими настройками компилятора. 4) Программа быстрее перекомпилируется, т.к. компилятор компилирует только изменившиеся модули. К этому Вы забыли еще добавить 5) разграничение пространства имен. Да, причём, это можно поставить на первое место. Зачем спрашивал, если сам всё знаешь?
|
|
|
|
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0
|
|
|