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

 
 
> WINAVR: pgmspace, eeprom, отражаем актуальные проблемы
_Pasha
сообщение Apr 8 2008, 18:50
Сообщение #1


;
******

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



Предлагаю в этой ветке отражать передовые методы работы с pgmspace.h и eeprom.h
Может, потом закрепить тему. Тем более, что видны динамичные изменения в этой области.

Для начала. WINAVR-20071221 + Avr Studio 4.13 sp2 build 571

Делаю большой проект. Выложить не могу. Вкратце суть проблемы.
Использую хедер, в котором сведены вместе все параметры, хранящиеся в EEPROM
paramset.h
Код
#ifndef PARAMSET_H
#define PARAMSET_H 1
EEMEM uint16_t nominal_rpm = 500;
EEMEM uint8_t amplitude = 208;
................... etc
#endif


Хедер включается в разные модули. Их много. Везде константы из еепрома пользуются популярностью smile.gif
И, BTW, надо иметь на выходе файл *.eep для начальной заливки.

Для того, чтобы все работало, приходится объявлять параметры дважды - в основном модуле с инициализацией, а в остальных - как extern, т.е.
Код
#ifndef PARAMSET_H
#define PARAMSET_H 1
#ifdef _IN_MAIN_
EEMEM uint16_t nominal_rpm = 500;
EEMEM uint8_t amplitude = 208;
................... etc
#else
extern EEMEM uint16_t nominal_rpm;
extern EEMEM uint8_t amplitude;
................... etc
#endif

#endif


Потом, в основной программе
Код
#define _IN_MAIN_ 1
#include "paramset.h"


Иначе - чушь в распределении адресов.
Как избежать двойного объявления переменных в приведенном примере, ессно, без использования макросов?
Возможно ли это вообще?
Go to the top of the page
 
+Quote Post
 
Start new topic
Ответов
_Pasha
сообщение Apr 8 2008, 23:42
Сообщение #2


;
******

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



Как автор темы позволю себе подытожить по этому примеру smile.gif

- Нечитабельность при инициализации достаточно большой структуры.
- Постоянное добавление имени владельца структуры при обращении

+ Структура позволяет решить проблему описаний и легко заливать defaults из флеша.
Если же последнее не предусмотрено, то
Макрос, который напрашивается здесь, имеет вид
Код
#define PARAM(_TYPE_ , _NAME_ , _INIT_) #ifdef _IN_MAIN_\
EEMEM _TYPE_ _NAME_ = _INIT_\
#else\
extern EEMEM _TYPE_ _NAME_\
#endif


НаписАл я эту довольно банальную вещь как раз для того, чтобы ее покритиковали.
Go to the top of the page
 
+Quote Post
ReAl
сообщение Apr 9 2008, 06:47
Сообщение #3


Нечётный пользователь.
******

Группа: Свой
Сообщений: 2 033
Регистрация: 26-05-05
Из: Бровари, Україна
Пользователь №: 5 417



Цитата(_Pasha @ Apr 9 2008, 02:42) *
- Нечитабельность при инициализации достаточно большой структуры.
C99 позволяет
Код
typedef struct {
    uint8_t a;
    int b;
    char c;
} cfg_t;

// даже порядок инициализации не важен, всё будет правильно
cfg_t cfg EEMEM = {
  .b = -1 ,
  .c = '2' ,
  .a = 3
};

char arr[5] EEMEM = {
   [2] = 'c' ,
   [4] = 'e' ,
   [0] = 'a'  // элементы массива 1 и 3 инициализированы нулями по умолчанию
};
К сожалению, в С++ такого нет.

Цитата(_Pasha @ Apr 9 2008, 02:42) *
Макрос, который напрашивается здесь, имеет вид
Код
#define PARAM(_TYPE_ , _NAME_ , _INIT_) #ifdef _IN_MAIN_\
EEMEM _TYPE_ _NAME_ = _INIT_\
#else\
extern EEMEM _TYPE_ _NAME_\
#endif

Нельзя делать проверку условия внутри макроса. И нельзя всё одной строкой. Надо так:

Код
#ifdef _IN_MAIN_
#define PARAM(_TYPE_ , _NAME_ , _INIT_)  \
   EEMEM _TYPE_ _NAME_ = _INIT_
#else
#define PARAM(_TYPE_ , _NAME_ , _INIT_) extern EEMEM _TYPE_ _NAME_
#endif


Кстати, С99 позволяет это даже не при инициализации делать, можно делать эдакие "литералы"
Код
void foo(cfg_t *p)
{
    *p = (cfg_t){ .c = 'c', .b = -1, .a = 0 };
    // ну или     *p = (cfg_t){ 0, -1, 'c'};  для тех, кто помнит порядок полей;-)
}
вместо
Код
void foo(cfg_t *p)
{
    p->c = 'c';
    p->b = -1;
    p->a = 0;
}
И avr-gcc чудесно такое компилирует, я иногда пользуюсь.
А потом как вспомню, что далеко не все так стандарт поддерживают, как gcc, так и начинаю стараться не привыкать.

Ну и С++ такого не позволяет...
Даже с -std=gnu++98, не говоря уже о -std=c++98 --pedantic


--------------------
Ну, я пошёл… Если что – звоните…
Go to the top of the page
 
+Quote Post
_Pasha
сообщение Apr 9 2008, 14:38
Сообщение #4


;
******

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



Цитата(ReAl @ Apr 9 2008, 10:47) *
C99 позволяет
даже порядок инициализации не важен, всё будет правильно

А потом как вспомню, что далеко не все так стандарт поддерживают, как gcc, так и начинаю стараться не привыкать.


+1000
Потому что поддерживаемые в gcc операции с EEPROM тоже, мягко говоря, уникальны smile.gif
Так что с подводной лодки далеко не уплывешь... и можно и привыкнуть, ничего страшного.
Go to the top of the page
 
+Quote Post
ReAl
сообщение Apr 9 2008, 20:48
Сообщение #5


Нечётный пользователь.
******

Группа: Свой
Сообщений: 2 033
Регистрация: 26-05-05
Из: Бровари, Україна
Пользователь №: 5 417



Цитата(_Pasha @ Apr 9 2008, 17:38) *
+1000
Потому что поддерживаемые в gcc операции с EEPROM тоже, мягко говоря, уникальны smile.gif
Так что с подводной лодки далеко не уплывешь... и можно и привыкнуть, ничего страшного.
Увы, это разные вещи. Одно дело чисто gcc-шные расширения, типа диапазонных case в switch и вложенных функций. Приятные, но абсолютно нестандартные. Не так и жалко.
Другое дело - то, что с 1999 года в стандарте, уже скоро 10 лет, но от MS до IAR почти никто не поддерживает. Вот за это обидно. Вещи-то из СТАНДАРТА, а "виноват" потом окажется тот, кто этот стандарт поддержал.


--------------------
Ну, я пошёл… Если что – звоните…
Go to the top of the page
 
+Quote Post



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

 


RSS Текстовая версия Сейчас: 21st July 2025 - 11:16
Рейтинг@Mail.ru


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