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

 
 
 
Reply to this topicStart new topic
> Как определить размер структуры в макросе?, в макросе sizeof() не работает
AndreyS
сообщение Sep 27 2010, 13:16
Сообщение #1


Местный
***

Группа: Участник
Сообщений: 235
Регистрация: 28-01-05
Из: Санкт-Петербург
Пользователь №: 2 276



Добрый день.

Дано:
8 битный микроконтроллер
Keil Си
Структура для хранения переменных во внешней I2C EPROM
Размер EPROM константа 256 байт
Структура постепенно растет и изменяется

Хотел включить макрос проверки размера структура за предел размер памяти EPROM на стадии компиляции и в случае превышеняи размера выводить ошибку.

Что-то типа того:
Код
typdef struct
{
...
}структура;

#define размер EPROM 256

#if (sizeof(структура)>размер EPROM)
#error "ОШИБКА"
#endif


Но sizeof в данном случае использовать нельзя.
Подскажите пожалуйста, как мне решить мою задачу? (считать вручную уже надоело)

Сообщение отредактировал AndreyS - Sep 27 2010, 13:20


--------------------
Удачи.
Go to the top of the page
 
+Quote Post
neiver
сообщение Sep 27 2010, 13:56
Сообщение #2


Местный
***

Группа: Участник
Сообщений: 214
Регистрация: 22-03-10
Из: Саратов
Пользователь №: 56 123



Можно вот такой хак применить, воспользовавшись, тем фактом, что массив не может иметь отрицательный размер:

Код
struct A
{
    int a, b, c, d;
    char e[300];
};

enum{EPROM_SIZE = 256};

static int error_struct_a_is_too_big[sizeof(A) > EPROM_SIZE ? -1: 0];


Если sizeof(A) будет больше EPROM_SIZE, то будет ошибка. Если нет - массив нулевого размера, который оптимизатор выкинет.
Собственно, сообщение нужно закодировать в имени этого массива. Пример сообщения об ошибке, которое выдаёт GCC:
../main.cpp:19: error: size of array 'error_struct_a_is_too_big' is negative


А в макросах это никак не сделать, макросы работают на уровне текстовых строк, им побоку любые языковые конструкции и sizeof в частности.
Go to the top of the page
 
+Quote Post
sergeeff
сообщение Sep 27 2010, 14:01
Сообщение #3


Профессионал
*****

Группа: Свой
Сообщений: 1 481
Регистрация: 10-04-05
Пользователь №: 4 007



Макросы обрабатываются препроцессором, то есть до компиляции. sizeof определяется на этапе компиляции. Посему ответ: в лоб на С - никак.

http://msdn.microsoft.com/en-us/library/ew...28VS.80%29.aspx
Go to the top of the page
 
+Quote Post
vmp
сообщение Sep 27 2010, 14:13
Сообщение #4


Местный
***

Группа: Свой
Сообщений: 426
Регистрация: 20-01-05
Из: Зеленоград
Пользователь №: 2 070



sizeof в препроцессоре понимали какие-то ранние версии компиляторов IAR.
А для проверки можно тупо написать if в теле программы. Все равно компилятор при оптимизации его выкинет, если условие не выполняется.
Можно даже вызывать какую-нибуь левую подпрограииу, чтобы обругался линкер:

Код
void linker_error(void);


void main(void)
{
  if (sizeof(структура)>размер EPROM)
   {
     linker_error();
     for (;;);
   }
}

В этом случае сначала компилятор даст warning на недостижимый код из-за for (;;);, а потом линкер не найдет подпрограмму linker_error.


Go to the top of the page
 
+Quote Post
neiver
сообщение Sep 27 2010, 15:02
Сообщение #5


Местный
***

Группа: Участник
Сообщений: 214
Регистрация: 22-03-10
Из: Саратов
Пользователь №: 56 123



Можно довольно много языковых конструкций использовать для этих целей. Особенно тех, правильность которых зависит от значений целочисленных выражений. switch, например, можно использовать с дублирующимися метками. Да много чего ещё.
А вот в С++ есть такая штука как STATIC_ASSERT, реализацию которого можно найти, например, в boost (А как утвердят новый стандарт -это будет часть языка).
И можно писать в любом месте программы так:

Код
BOOST_STATIC_ASSERT(sizeof(A) < 256)


Если условие итненно - всё нормально, ложно - ошибка компиляции.
Я лично именно BOOST_STATIC_ASSERT использую в таких случаях (и пишу на С++ соответственно).
Go to the top of the page
 
+Quote Post
sergeeff
сообщение Sep 27 2010, 16:36
Сообщение #6


Профессионал
*****

Группа: Свой
Сообщений: 1 481
Регистрация: 10-04-05
Пользователь №: 4 007



Цитата(neiver @ Sep 27 2010, 19:02) *
Я лично именно BOOST_STATIC_ASSERT использую в таких случаях (и пишу на С++ соответственно).


STATIC_ASSERT уже является частью Visual Studio 2010
Go to the top of the page
 
+Quote Post
neiver
сообщение Sep 27 2010, 17:02
Сообщение #7


Местный
***

Группа: Участник
Сообщений: 214
Регистрация: 22-03-10
Из: Саратов
Пользователь №: 56 123



Цитата(sergeeff @ Sep 27 2010, 20:36) *
STATIC_ASSERT уже является частью Visual Studio 2010

Он является частью нового стандарта С++.
И в GCC он уже тоже есть. Но стандарт ещё не принят.
Go to the top of the page
 
+Quote Post
AndreyS
сообщение Sep 30 2010, 06:29
Сообщение #8


Местный
***

Группа: Участник
Сообщений: 235
Регистрация: 28-01-05
Из: Санкт-Петербург
Пользователь №: 2 276



Всем большое спасибо.

Сделал костыль предложенный neiver (хоть и пришлось пожертвовать одним байтом ОЗУ. 0 длины массив не создал. Вернее не выкинул его, а ругнулся при требуемом типе оптимизации.)

Жалко конструкция vmp на Keil не сработала. Он упорно пытался найти все что под if (хотя после компиляции его выкинул)


--------------------
Удачи.
Go to the top of the page
 
+Quote Post
Diz
сообщение Sep 30 2010, 18:42
Сообщение #9


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

Группа: Участник
Сообщений: 84
Регистрация: 1-08-06
Пользователь №: 19 250



Если волнует, что компилятор выделит память переменной (при ненулевом уровне оптимизации должен выбросить),
можно добавить extern:
CODE
#define CAT(a, b) a##b
#define CAT_WRAP(prefix, line) CAT(prefix, line)
#define PANIC_IF(arg) extern char CAT_WRAP(AssertAtLine, __LINE__) [ (arg) ? -1 : 1 ]


Go to the top of the page
 
+Quote Post
Dog Pawlowa
сообщение Oct 1 2010, 06:25
Сообщение #10


Гуру
******

Группа: Свой
Сообщений: 2 702
Регистрация: 14-07-06
Пользователь №: 18 823



Есть такой макрос определения смещения поля в структуре, в ИАРе он встроенный, его использовать.

#define OFFSET(type,field) ((uchar *)&(((type *)0)->field) - (uchar *)0)


--------------------
Уходя, оставьте свет...
Go to the top of the page
 
+Quote Post
_Pasha
сообщение Oct 1 2010, 09:34
Сообщение #11


;
******

Группа: Участник
Сообщений: 5 646
Регистрация: 1-08-07
Пользователь №: 29 509



Цитата(Dog Pawlowa @ Oct 1 2010, 09:25) *
#define OFFSET...

Вообще-то он должен offsetof называться....
Go to the top of the page
 
+Quote Post
Dog Pawlowa
сообщение Oct 2 2010, 18:30
Сообщение #12


Гуру
******

Группа: Свой
Сообщений: 2 702
Регистрация: 14-07-06
Пользователь №: 18 823



Цитата(_Pasha @ Oct 1 2010, 12:34) *
offsetof

Может быть, я то свой определил.


--------------------
Уходя, оставьте свет...
Go to the top of the page
 
+Quote Post
esaulenka
сообщение Oct 4 2010, 09:45
Сообщение #13


Профессионал
*****

Группа: Свой
Сообщений: 1 032
Регистрация: 13-03-08
Из: Маськва
Пользователь №: 35 877



Цитата(Dog Pawlowa @ Oct 1 2010, 10:25) *
Есть такой макрос определения смещения поля в структуре, в ИАРе он встроенный, его использовать.
#define OFFSET(type,field) ((uchar *)&(((type *)0)->field) - (uchar *)0)

Плюс адин к мнению _Pasha, лучше делать #include <stddef.h> и использовать стандартный offsetof().

Вот только топикстартера интересовал другой вопрос smile.gif


--------------------
Тут обсуждается творческий порыв, а не соответствие каким-либо стандартам ©
Go to the top of the page
 
+Quote Post

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

 


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


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