Sergey_Aleksandrovi4 2 25 октября, 2009 Опубликовано 25 октября, 2009 · Жалоба Всегда в проектах (пишу на асме в AVRstudio) создаю подобную конструкцию .org (FLASHEND - 64) .db " Encoder&Keyboard v1.4 25.10.2009" Т.е. в конец флеша записываю строку с названием проекта, версией прошивки и датой билда, что порой очень выручает. На днях узнал про "предопределённые макросы" __DATE__ __TIME__, был приятно удивлён. Но не очень устраивает формат вывода даты макросом __DATE__ . Он выдаёт "Oct 25 2009", а хочется именно "25.10.2009". В хэлпе по AVRasm2 приводят: -FDformat -FTformat Specify the format of the __DATE__ and __TIME__ predefined macros, respectively. The format string is passed directly to the strftime(3) C library function. The __DATE__ and __TIME__ preprocessor macros will always be string tokens, i.e., their values will appear in double quotes. The default formats are "%b %d %Y" and "%H:%M:%S", respectively. Example: To specify ISO format for __DATE__, specify -FD"%Y-%m-%d" See note below These formats may only be specified at the command line, there are no corresponding #pragma directives. Important note: The Windows command interpreter (cmd.exe or command.com) may interpret a character sequence starting and ending with a '%' character as an environment variable to be expanded even when it is quoted. This may cause the date/time format strings to be changed by the command interpreter and not work as expected. A workaround that will work in many cases is to use double '%' characters to specify the format directives, e.g., -FD"%%Y-%%m-%%d" for the example above. The exact behaviour of the command interpreter seems to be inconsistent and vary depending on a number of circumstances, for one, it is different in batch and interactive mode. The effect of the format directives should be tested. It is recommended to put the following line in the source file for testing this: #message "__DATE__ =" __DATE__ "__TIME__ =" __TIME__ This will print the value of the date and time macros when the program is assembled, making verification easy (see #message directive documentation). An alternative syntax for the format specification may be considered in future AVRASM2 versions to avoid this problem. Some relevant strftime() format specifiers (see strftime(3) manual page for full details): %Y - Year, 4 digits %y - Year, 2 digits %m - Month number (01-12) %b - Abbreviated month name %B - Full month name %d - Day number in month (01-31) %a - Abbreviated weekday name %A - Full weekday name %H - Hour, 24-hour clock (00-23) %I - Hour, 12-hour clock (01-12) %p - "AM" or "PM" for 12-hour clock %M - Minute (00-59) %S - Second (00-59) Куда писать эти спецификаторы - ума не приложу :-[ Имею подозрения, что поле "Additional Parameters" окна "Project->Assembler Options" для этих целей используется. Также вопрос вдогонку. При присваивании константам текстовых имён хочу делать проверку на допустимость значения .equ defineParam1 = 1 ...... ...... #if (defineParam1 < 3) #error "defineParam1 must be more then 3" #endif Но вот как в тексте ошибки вернуть значение этой константы? Чтобы при неверно заданном параметре был виден не только текст констатирующий ошибку, но и первопричину этой ошибки, что-то вроде #error "defineParam1 must be more then 3. Param1=", (????). Спасибо всем откликнувшимся. Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться
Sergey_Aleksandrovi4 2 30 октября, 2009 Опубликовано 30 октября, 2009 · Жалоба Сам вопрос задал, сам же на него и отвечу (вдруг кому и пригодится). В поле "Additional Parameters" окна настроек ассемблера "Project->Assembler Options" необходимо ввести спецификатор -FD"%%Y.%%m.%%d" для вывода даты вида "YYYY.MM.DD" (например 2009.10.30). Аналогично предопределяется и макрос __TIME__ . 4 года пользуюсь студией, а о подобных настройках узню только сейчас(( PS А вот как к строке возвращаемой директивой #error приаттачивать число, так и не придумал/нашёл. Может у кого есть соображения по этому вопросу? Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться
CSS 0 29 ноября, 2009 Опубликовано 29 ноября, 2009 · Жалоба Насчёт #error - нету. Я тут ещё с номером билда баловался - наваял за полчаса программку, которая ТУПО инкрементит одно число в файле типа .db "build xxx", и этот файл инклудю в код. PDFLASHData.Inc ==== _CSVersionString: .DB "CS PowerDimmer v2.6 " .INCLUDE "PDVersion.Inc" .DB " @AVR ATMega 8. Copyright © 2009 CS.", NUL, FF PDVersion.Inc ====== ;AUTO-GENERATED SCRIPT!! DO NOT EDIT!!!!!!!!!! .DB "[0900]" Прога юзается в шаге Project -> Assebler Options -> Avdanced Options -> Pre-assebling: IncVersion.Exe PDVersion.Inc Формат даты-времени ставил, как вы уже нашли. IncVersion.rar Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться
ae_ 2 5 декабря, 2009 Опубликовано 5 декабря, 2009 · Жалоба При присваивании константам текстовых имён хочу делать проверку на допустимость значения .equ defineParam1 = 1 ...... ...... #if (defineParam1 < 3) #error "defineParam1 must be more then 3" #endif Но вот как в тексте ошибки вернуть значение этой константы? Использовать #define вместо .equ #define defineParam1 2 #if (defineParam1 < 3) #error "defineParam1 must be equal or more then 3. defineParam1 = " defineParam1 #endif Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться
adc 0 6 декабря, 2009 Опубликовано 6 декабря, 2009 · Жалоба На днях узнал про "предопределённые макросы" __DATE__ __TIME__, был приятно удивлён. Тоже был приятно удивлен на днях))) В этой ветке http://electronix.ru/forum/index.php?s=&am...st&p=685111 ув. V_G написал следующую форму: Version: .DB __DAY__,__MONTH__,__YEAR__,__HOUR__,__MINUTE__,__SECOND__;версия ПО Эта схема возвращает нужный Вам формат т.е. 11.12.2009.... (все в шестнадцатеричном виде )Если нужно, можно добавить точки. .DB __DAY__,'.',__MONTH__'.',__YEAR_ Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться
Sergey_Aleksandrovi4 2 23 мая, 2011 Опубликовано 23 мая, 2011 · Жалоба Спустя время пересел на IAR EWAVR 5.1. Здесь форматы макросов __DATE__ и __TIME__ определены жёстко и строки инициализации наподобие -FD"%%Y.%%m.%%d" используемых в AVRStudio не действуют. С форматом времени разобрался #define HOURS (__TIME__[0]), (__TIME__[1]) #define MINUTES (__TIME__[3]), (__TIME__[4]) #define SECONDS (__TIME__[6]), (__TIME__[7]) __flash __root unsigned char PROJECT_INFO4[] = {' ', HOURS, ':', MINUTES, ':', SECONDS}; С датой не получается. Наткнувшись на пример http://www.embeddedrelated.com/groups/msp430/show/44591.php попытался реализовать конструкцию вида #if __DATE__[2] == 'b' //Feb #define MONTH '0','2' #elif __DATE__[2] == 'y' //May #define MONTH '0','5' #elif __DATE__[2] == 'l' //Jul #define MONTH '0','7' #elif __DATE__[2] == 'g' //Aug #define MONTH '0','8' #elif __DATE__[2] == 'p' //Sep #define MONTH '0','9' #elif __DATE__[2] == 't' //Oct #define MONTH '1','0' #elif __DATE__[2] == 'v' //Nov #define MONTH '1','1' #elif __DATE__[2] == 'c' //Dec #define MONTH '1','2' компилятор ругается на сравнение #if __DATE__[2] == 'b', т.е. не может взять член массива и сравнить с константой. Выдаёт ошибку "Error[Pe029]: expected an expression". У кого есть мысли, как обойти эту проблему. Или может есть другой, более элегантный способ отредактировать __DATE__? Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться
demiurg_spb 0 26 мая, 2011 Опубликовано 26 мая, 2011 · Жалоба компилятор ругается на сравнение #if __DATE__[2] == 'b', т.е. не может взять член массива и сравнить с константой. Выдаёт ошибку "Error[Pe029]: expected an expression".А если условие взять в скобочки? #if (__DATE__[2] == 'b') Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться
ReAl 0 26 мая, 2011 Опубликовано 26 мая, 2011 · Жалоба Препроцессор не должен уметь залазить внутрь строк. Нодо пробовать что-то в духе #define MONTH \ (__DATE__[2] == 'b' ? "02" : \ __DATE__[2] == 'y' ? "05" : \ __DATE__[2] == 'l' ? "07" : И так далее. Потом только MONTH можно будет использовать только для того, чтобы инициалищировать отедльную строку либо подставить в печать. А вообще -- чего только люди не сделают, чтобы не разбираться с make и всем прочим // d.c char date[] = DATE; foo: avr-gcc -DDATE=\"`date +%Y.%m.%d`\" -Os -S d.c make -f mk.mk cat d.s .file "d.c" __SREG__ = 0x3f __SP_H__ = 0x3e __SP_L__ = 0x3d __CCP__ = 0x34 __tmp_reg__ = 0 __zero_reg__ = 1 .global __do_copy_data .global __do_clear_bss .global date .data .type date, @object .size date, 11 date: .string "2011.05.26" Под win при наличии make.exe sh.exe должно получаться аналогично. Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться
kbvsoft 0 25 декабря, 2017 Опубликовано 25 декабря, 2017 · Жалоба У кого есть мысли, как обойти эту проблему. Или может есть другой, более элегантный способ отредактировать __DATE__? Конечно скобочки в конструкции #if (__DATE__[2] == 'b') обязательны, но это не поможет. Похоже на этапе выбора условий компиляций (директивы препроцессора #if, #ifdef) компилятор IAR ещё не сформировал макросы __DATE__ и __TIME__ и при любом обращении к ним в этих директивах пустые значения __DATE__ и __TIME__ никогда не включат нужную ветку условной компиляции. Поэтому условие компиляции типа #if (__DATE__[2] == 'b') //Feb #define MONTH '0','2' ... никогда не выполнится (т.к. на самом деле в директиве #if элемент макроса __DATE__[2] ещё равен 0) Задание же макросов (директивы препроцессора #define) или символических констант (ключевое слово const происходит на следующем этапе компиляции, когда IAR уже сформировал макросы __DATE__ и __TIME__, тогда уже следующие конструкции прекрасно работают: #define HOURS (__TIME__[0]), (__TIME__[1]) #define YEAR __DATE__[7], __DATE__[8], __DATE__[9], __DATE__[10]; __flash char BUILD_TIME[] = {__TIME__}; Исходя их вышесказанного, преобразовать __DATE__ в нужный формат на этапе компиляции никак не получится, по крайней мере, у меня не получилось. :laughing: Зато в runtime всё прекрасно работает. :yeah: Следующий код возвращает дату компиляции в нужном нам формате 'DD.MM.YYYY' в строке TempStr: char TempStr[11]; const char m1 = __DATE__[1]; const char m2 = __DATE__[2]; const char d1 = __DATE__[4]; const char d2 = __DATE__[5]; const char y1 = __DATE__[7]; const char y2 = __DATE__[8]; const char y3 = __DATE__[9]; const char y4 = __DATE__[10]; void BuildDateToTempStr (void ) { /* Формат макроса IAR '__DATE__' на примере '30.01.2000': 'Jan 30 2000' 01234567890 Jan - 01 Feb - 02 Mar - 03 Apr - 04 May - 05 Jun - 06 Jul - 07 Aug - 08 Sep - 09 Oct - 10 Nov - 11 Dec - 12 */ char M1, M2; //•••••••••••••••••••••••• TempStr[0] = d1; TempStr[1] = d2; TempStr[2] = '.'; E1 = '0'; if (m1 == 'a' && m2 == 'n') M2 = '1'; if (m1 == 'e' && m1 == 'b') M2 = '2'; if (m1 == 'a' && m2 == 'r') M2 = '3'; if (m1 == 'p' && m2 == 'r') M2 = '4'; if (m1 == 'a' && m2 == 'y') M2 = '5'; if (m1 == 'u' && m2 == 'n') M2 = '6'; if (m1 == 'u' && m2 == 'l') M2 = '7'; if (m1 == 'u' && m2 == 'g') M2 = '8'; if (m1 == 'e' && m2 == 'p') M2 = '9'; if (m1 == 'c' && m2 == 't') {M1 = '1'; M2 = '0';} if (m1 == 'o' && m2 == 'v') {M1 = '1'; M2 = '1';} if (m1 == 'e' && m2 == 'c') {M1 = '1'; M2 = '2';} TempStr[3] = M1; TempStr[4] = M2; TempStr[5] = '.'; TempStr[6] = y1; TempStr[7] = y2; TempStr[8] = y3; TempStr[9] = y4; //•••••••••••••••••••••••• TempStr[10] = 0; } m1 … y4 – обязательно!!! должны объявляться как константы через const. Тогда условные операторы типа if (m1 == 'a' && m2 == 'n') в скомпилированный код вообще не включаются, т.к. сравнивать константу с константой нет смысла, поэтому в полученном коде элементам строки просто присваиваются реальные значения даты компиляции в нужном формате и всё! Если же m1 … y4 объявить через #define, на этапе компиляции дата размещается в памяти данных и все обращения к ней как к памяти, в итоге скомпилированный код раз в 10 больше!. Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться
Baser 5 25 декабря, 2017 Опубликовано 25 декабря, 2017 · Жалоба Когда-то давно, на просторах этого форума, пробегал чей-то макрос, которым я все это время без проблем пользуюсь: #ifndef _COMPILE_DATE_TIME_ #define _COMPILE_DATE_TIME_ #define COMPILE_HOUR (((__TIME__[0]-'0')*10) + (__TIME__[1]-'0')) #define COMPILE_MINUTE (((__TIME__[3]-'0')*10) + (__TIME__[4]-'0')) #define COMPILE_SECOND (((__TIME__[6]-'0')*10) + (__TIME__[7]-'0')) #define COMPILE_YEAR ((((__DATE__[7]-'0')*10+(__DATE__[8]-'0'))*10+(__DATE__[9]-'0'))*10+(__DATE__[10]-'0')) #define COMPILE_MONTH ((__DATE__[2] == 'n' ? (__DATE__ [1] == 'a'? 0 : 5) \ : __DATE__[2] == 'b' ? 1 \ : __DATE__[2] == 'r' ? (__DATE__ [0] == 'M'? 2 : 3) \ : __DATE__[2] == 'y' ? 4 \ : __DATE__[2] == 'l' ? 6 \ : __DATE__[2] == 'g' ? 7 \ : __DATE__[2] == 'p' ? 8 \ : __DATE__[2] == 't' ? 9 \ : __DATE__[2] == 'v' ? 10 : 11)+1) #define COMPILE_DAY ((__DATE__[4]==' ' ? 0 : __DATE__[4]-'0')*10+(__DATE__[5]-'0')) #endif Автору большое спасибо :a14: Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться
AVI-crak 0 26 декабря, 2017 Опубликовано 26 декабря, 2017 · Жалоба Вариант практического использования для регистров часов и даты stm32f4-7. Устанавливает дату и время на момент сборки прошивки, макрос сокращается до одиночной записи в регистр. RTC->TR = ((__TIME__[0]-'0') << 20) // hour 10 | ((__TIME__[1]-'0') << 16) // hour 1 | ((__TIME__[3]-'0') << 12) // minutes 10 | ((__TIME__[4]-'0') << 8) // minutes 1 | ((__TIME__[6]-'0') << 4) // seconds 10 | (__TIME__[7]-'0'); // seconds 1 RTC->DR = ((__DATE__[9]-'0') << 20) // year 10 | ((__DATE__[10]-'0') << 16) // year 1 | ((__TIMESTAMP__[2]=='e'?2:__TIMESTAMP__[2]=='d'?3 \ :__TIMESTAMP__[2]=='u'?4:__TIMESTAMP__[2]=='i'?5 \ :__TIMESTAMP__[2]=='t'?6:__TIMESTAMP__[0]=='M'?1:7) << 13) // Week day | ((((__DATE__[2]=='n'?(__DATE__[1]=='a'?0:5):__DATE__[2]=='b'?1 \ :__DATE__[2]=='r'?(__DATE__[0]=='M'?2:3):__DATE__[2]=='y'?4 \ :__DATE__[2]=='l'?6:__DATE__[2]=='g'?7:__DATE__[2]=='p'?8 \ :__DATE__[2] =='t'?9:__DATE__[2]=='v'?10:11)+1)/10) << 12) // Month 10 | ((((__DATE__[2]=='n'?(__DATE__[1]=='a'?0:5):__DATE__[2]=='b'?1 \ :__DATE__[2]=='r'?(__DATE__[0]=='M'?2:3):__DATE__[2]=='y'?4 \ :__DATE__[2]=='l'?6:__DATE__[2]=='g'?7:__DATE__[2]=='p'?8 \ :__DATE__[2] =='t'?9:__DATE__[2]=='v'?10:11)+1)%10) << 8) // Month 1 | ((__DATE__[4]==' ' ? 0 : __DATE__[4]-'0') << 4) // day 10 | (__DATE__[5]-'0'); // day 1 Цитата Поделиться сообщением Ссылка на сообщение Поделиться на другие сайты Поделиться