реклама на сайте
подробности

 
 
2 страниц V   1 2 >  
Reply to this topicStart new topic
> Не могу решить проблему вызова функций IAR, используется 1 раз, но вызывается call
haker_fox
сообщение Oct 26 2005, 23:29
Сообщение #1


Познающий...
******

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



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

В общем мой вопрос так звучит: как сделать, чтоб однократно используемые функции не вызывались, а их тело помещалось вместо вызова подпрограммы.


--------------------
Выбор.
Go to the top of the page
 
+Quote Post
prottoss
сообщение Oct 26 2005, 23:39
Сообщение #2


Гуру
******

Группа: Свой
Сообщений: 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.


--------------------
Go to the top of the page
 
+Quote Post
haker_fox
сообщение Oct 26 2005, 23:58
Сообщение #3


Познающий...
******

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



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


А нельзя ли это автоматизировать, что сам компилятор это выполнял. А то трудно уследить какие функции будут использоваться однократно.


--------------------
Выбор.
Go to the top of the page
 
+Quote Post
prottoss
сообщение Oct 27 2005, 00:36
Сообщение #4


Гуру
******

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



Цитата(haker_fox @ Oct 27 2005, 07:58)
А нельзя ли это автоматизировать, что сам компилятор это выполнял. А то трудно уследить какие функции будут использоваться однократно.


Я использую директиву #pragma inline=forced в основном в прерываниях, для разделения задачи на более мелкие подзадачи - так удобнее потом в этом разбираться и мне, и тем, для кого я пишу. Да и при написания процедуры обработки прерывания, если внутри процедуры нет вызовов call, код компилируется меньше за счет сокращения числа сохраненных РОН. А вообще наверное есть смысл подставлять данную директиву везде, если есть вероятность вызова функции всего один раз. На счет автоматизации этого процесса ничего сказать не могу.


--------------------
Go to the top of the page
 
+Quote Post
haker_fox
сообщение Oct 27 2005, 00:56
Сообщение #5


Познающий...
******

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



2prottoss: спасибо! Я поставля эти директивы в обработчиках прерываний, где функции точно вызываются единожды.


--------------------
Выбор.
Go to the top of the page
 
+Quote Post
starter48
сообщение Oct 28 2005, 10:55
Сообщение #6


Частый гость
**

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



Цитата(haker_fox @ Oct 27 2005, 06:29)
В общем мой вопрос так звучит: как сделать, чтоб однократно используемые функции не вызывались, а их тело помещалось вместо вызова подпрограммы.
*

1) Нужно чтобы вызываемые ф-ии были в одном модуле с вызывающей ф-ей.
2) Нужно написать модификатор static перед вызываемой ф-ей.
3) К этим пунктами иногда тербуется включить оптимизацию по скорости (проверь экспериментально). Тогда компилятор сам достаточно умно решает инлайнить ф-ю или нет.
Go to the top of the page
 
+Quote Post
haker_fox
сообщение Oct 28 2005, 14:29
Сообщение #7


Познающий...
******

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



Но если разместить функции в одном модуле, то пропадет наглядность программы, видимо надо чихнуть на этот "беспредел" компилятора.


--------------------
Выбор.
Go to the top of the page
 
+Quote Post
starter48
сообщение Oct 28 2005, 14:41
Сообщение #8


Частый гость
**

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



Цитата(haker_fox @ Oct 28 2005, 21:29)
Но если разместить функции в одном модуле, то пропадет наглядность программы, видимо надо чихнуть на этот "беспредел" компилятора.
*

Можешь модуль разбить на несколько с помощью #include smile.gif
Вообще это свойство компилятора. Ф-ии для работающие вместе должны объединяться в модулях. Если есть ф-я, которую нужно инлайнить в нескольких модулях , то объяви её static и запихни в подключенный в эти моули .h файл.
Т.о. модули компилируются отдельно, при этом компилятор ничего не знает о ф-ях в других модулях, кроме прототипов. И только потом линкером сгенерированный из модулей код объединяется.
Go to the top of the page
 
+Quote Post
prottoss
сообщение Oct 28 2005, 15:15
Сообщение #9


Гуру
******

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



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


Компилятор то знает все обо всех функциях, на то он и компилятор. Другое дело, что он не подставляет тело функции из одного модуля, в тело функции из другого. Но если написать тело в .h файл, возможно это и прокатит.
Но тут такая белиберда получится, что ой-ой.

Допустим в модуле объявлена глобальная переменная, а нужная нам инлайн-функция должна работать с этой пременной. Как этой функции, которая описанна в хедере, рассказать что в одноименном модуле для него приготовленна переменная. Писать еще одну функцию в модуле, которая будет возвращать значение переменной инлайн-функции?


--------------------
Go to the top of the page
 
+Quote Post
starter48
сообщение Oct 28 2005, 15:52
Сообщение #10


Частый гость
**

Группа: Свой
Сообщений: 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
Go to the top of the page
 
+Quote Post
prottoss
сообщение Oct 28 2005, 16:12
Сообщение #11


Гуру
******

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



Цитата(starter48 @ Oct 28 2005, 23:52)
Нужно объявить переменную как extern в .h файле с функцией.


Тогда объясните зачем вообще нужны отдельные модули, раз уж на то пошло.
Давайте будем писать все в одном модуле, чтоб все все знали обо всех :-)


--------------------
Go to the top of the page
 
+Quote Post
starter48
сообщение Oct 28 2005, 16:39
Сообщение #12


Частый гость
**

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



Цитата(prottoss @ Oct 28 2005, 23:12)
Тогда объясните зачем вообще нужны отдельные модули, раз уж на то пошло.

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

Да, если проект не очень большой, то так обычно и делается.
Go to the top of the page
 
+Quote Post
prottoss
сообщение Oct 28 2005, 16:53
Сообщение #13


Гуру
******

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



Цитата(starter48 @ Oct 29 2005, 00:39)
1) Проще разбираться в исходниках, когда программа разбита на идеологически законченные блоки.
2) К программе можно прилинковать модуль написанный на другом языке.
3) Каждый модуль можно скомпилировать со своими настройками компилятора.
4) Программа быстрее перекомпилируется, т.к. компилятор компилирует только изменившиеся модули.


К этому Вы забыли еще добавить
5) разграничение пространства имен.


--------------------
Go to the top of the page
 
+Quote Post
starter48
сообщение Oct 28 2005, 17:11
Сообщение #14


Частый гость
**

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



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


К этому Вы забыли еще добавить
5) разграничение пространства имен.
*


Да, причём, это можно поставить на первое место.
Зачем спрашивал, если сам всё знаешь? smile.gif
Go to the top of the page
 
+Quote Post
prottoss
сообщение Oct 28 2005, 18:59
Сообщение #15


Гуру
******

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



Цитата(starter48 @ Oct 29 2005, 01:11)
Зачем спрашивал, если сам всё знаешь? smile.gif


Цитата(starter48 @ Oct 28 2005 @ 23:52)
Нужно объявить переменную как extern в .h файле с функцией.


Где ж тут разграничение пространства имен? Если глобальные переменные одного модуля известны всем модулям?


--------------------
Go to the top of the page
 
+Quote Post

2 страниц V   1 2 >
Reply to this topicStart new topic
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 


RSS Текстовая версия Сейчас: 18th July 2025 - 13:56
Рейтинг@Mail.ru


Страница сгенерированна за 0.01491 секунд с 7
ELECTRONIX ©2004-2016