|
Выполнение кода из ОЗУ, как задать области, которые туда копировать не нужно |
|
|
|
Apr 18 2018, 12:53
|
Группа: Участник
Сообщений: 11
Регистрация: 1-11-15
Пользователь №: 89 126
|
Реализовал выполнение кода из ОЗУ на STM32, отредактировав .icf файл. Вопрос в следующем: у меня в коде присутствует большой константный массив данных (картинка). Как мне сказать компилятору, чтобы он не копировал её в оперативку и не забивал память почем зря?
|
|
|
|
|
Apr 18 2018, 14:05
|
Группа: Участник
Сообщений: 11
Регистрация: 1-11-15
Пользователь №: 89 126
|
Цитата(jcxz @ Apr 18 2018, 14:09) Перенести этот массив во FLASH-регион описанный в .icf. Вот я и пытаюсь понять, как это сделать, чтобы не копировалось. Пробовал, например, #pragma location = ".rodata", не помогает .icf файл.: Код /*###ICF### Section handled by ICF editor, don't touch! ****/ /*-Editor annotation file-*/ /* IcfEditorFile="$TOOLKIT_DIR$\config\ide\IcfEditor\cortex_v1_0.xml" */ /*-Specials-*/ define symbol __ICFEDIT_intvec_start__ = 0x08000000; /*-Memory Regions-*/ define symbol __ICFEDIT_region_ROM_start__ = 0x08000000; define symbol __ICFEDIT_region_ROM_end__ = 0x080FFFFF; define symbol __ICFEDIT_region_RAM_start__ = 0x00000000; define symbol __ICFEDIT_region_RAM_end__ = 0x0002FFFF; define symbol __ICFEDIT_region_CCMRAM_start__ = 0x10000000; define symbol __ICFEDIT_region_CCMRAM_end__ = 0x1000FFFF; /*-Sizes-*/ define symbol __ICFEDIT_size_cstack__ = 0x2000; define symbol __ICFEDIT_size_heap__ = 0x2000; /**** End of ICF editor section. ###ICF###*/
/* intvec location in RAM after remapping in SystemInit */ define symbol RAM_intvec_start = 0x00000000;
define memory mem with size = 4G; define region ROM_region = mem:[from __ICFEDIT_region_ROM_start__ to __ICFEDIT_region_ROM_end__]; define region RAM_region = mem:[from __ICFEDIT_region_RAM_start__ to __ICFEDIT_region_RAM_end__]; define region CCMRAM_region = mem:[from __ICFEDIT_region_CCMRAM_start__ to __ICFEDIT_region_CCMRAM_end__];
define block CSTACK with alignment = 8, size = __ICFEDIT_size_cstack__ { }; define block HEAP with alignment = 8, size = __ICFEDIT_size_heap__ { };
initialize by copy { readonly, readwrite };
do not initialize { section .noinit };
place at address mem:__ICFEDIT_intvec_start__ { readonly section .intvec }; place at address mem:RAM_intvec_start { section .intvec_RAM }; /*place at address mem:RAM_intvec_start { readonly section .intvec };*/
place in ROM_region { readonly }; place in RAM_region { readwrite }; place in CCMRAM_region { section .ccmram, block CSTACK, block HEAP };
|
|
|
|
|
Apr 18 2018, 14:35
|
Группа: Участник
Сообщений: 11
Регистрация: 1-11-15
Пользователь №: 89 126
|
Попробовал, результат следующий: в отладчике массив лежит во Flash, как и хотелось, но теперь другой косяк - вместо значений из .cpp файла в нем одни нули UPD: а иногда одни 0xFF
Сообщение отредактировал AlphaLaiman - Apr 18 2018, 15:36
|
|
|
|
|
Apr 18 2018, 16:00
|
Группа: Участник
Сообщений: 11
Регистрация: 1-11-15
Пользователь №: 89 126
|
Цитата(jcxz @ Apr 18 2018, 16:50) Так ищите где у Вас косяк При замене Вашего кода Код .cpp: static char const t[] @ ".httpContent" = {...}; например на Код .cpp: static char const t[] @ ".ccmram" = {...}; все работает. Но ccmram тоже не хочется забивать этим массивом Цитата(VladislavS @ Apr 18 2018, 16:50) Предлагаю заглянуть в .map файл. Если воспользоваться советом jcxz, то в map файле следующее: Код "P1": 0x207c P1 s0 0x08000190 0x207c <Init block> .httpContent inited 0x08000190 0x207c graphics.o [1] - 0x0800220c 0x207c вроде тут все норм
Сообщение отредактировал AlphaLaiman - Apr 18 2018, 16:05
|
|
|
|
|
Apr 18 2018, 16:14
|
Гуру
Группа: Свой
Сообщений: 5 228
Регистрация: 3-07-08
Из: Омск
Пользователь №: 38 713
|
Цитата(AlphaLaiman @ Apr 18 2018, 19:00) .httpContent inited 0x08000190 0x207c graphics.o [1] вроде тут все норм Где же норм? Вы бы хоть поинтересовались в этом же .map как у Вас const-элементы выглядят! Вот это норм: .httpContent const 0x08020000 0x1444 content.cpio [1]А inited - это runtime-инициализируемая переменная: .data inited 0x20022468 0x4 dflash.o [2]
|
|
|
|
|
Apr 18 2018, 16:32
|
Группа: Участник
Сообщений: 11
Регистрация: 1-11-15
Пользователь №: 89 126
|
Но я ведь вроде все также сделал, как у Вас. Может, Вы не все показали? Инициализация массива: Код static BYTE const font[] @ ".httpContent" = { #include "table_char_5x7.h" }; icf файл: Код /*###ICF### Section handled by ICF editor, don't touch! ****/ /*-Editor annotation file-*/ /* IcfEditorFile="$TOOLKIT_DIR$\config\ide\IcfEditor\cortex_v1_0.xml" */ /*-Specials-*/ define symbol __ICFEDIT_intvec_start__ = 0x08000000; /*-Memory Regions-*/ define symbol __ICFEDIT_region_ROM_start__ = 0x08000000; define symbol __ICFEDIT_region_ROM_end__ = 0x080FFFFF; define symbol __ICFEDIT_region_RAM_start__ = 0x00000000; define symbol __ICFEDIT_region_RAM_end__ = 0x0002FFFF; define symbol __ICFEDIT_region_CCMRAM_start__ = 0x10000000; define symbol __ICFEDIT_region_CCMRAM_end__ = 0x1000FFFF; /*-Sizes-*/ define symbol __ICFEDIT_size_cstack__ = 0x2000; define symbol __ICFEDIT_size_heap__ = 0x2000; /**** End of ICF editor section. ###ICF###*/
/* intvec location in RAM after remapping in SystemInit */ define symbol RAM_intvec_start = 0x00000000;
define memory mem with size = 4G; define region FLASHC_regionC = mem:[from 0x08080000 to __ICFEDIT_region_ROM_end__]; define region ROM_region = mem:[from __ICFEDIT_region_ROM_start__ to 0x0807FFFF]; define region RAM_region = mem:[from __ICFEDIT_region_RAM_start__ to __ICFEDIT_region_RAM_end__]; define region CCMRAM_region = mem:[from __ICFEDIT_region_CCMRAM_start__ to __ICFEDIT_region_CCMRAM_end__];
define block CSTACK with alignment = 8, size = __ICFEDIT_size_cstack__ { }; define block HEAP with alignment = 8, size = __ICFEDIT_size_heap__ { };
initialize by copy { readonly, readwrite };
do not initialize { section .noinit };
place at address mem:__ICFEDIT_intvec_start__ { readonly section .intvec }; place at address mem:RAM_intvec_start { section .intvec_RAM }; /*place at address mem:RAM_intvec_start { readonly section .intvec };*/
place in FLASHC_regionC { section .httpContent}; place in ROM_region { readonly }; place in RAM_region { readwrite }; place in CCMRAM_region { section .ccmram, block CSTACK, block HEAP };
Сообщение отредактировал AlphaLaiman - Apr 18 2018, 16:33
|
|
|
|
|
Apr 18 2018, 16:35
|
Местный
Группа: Свой
Сообщений: 475
Регистрация: 14-04-05
Из: Москва
Пользователь №: 4 140
|
Код initialize by copy { readonly, readwrite }; Ничем не смущает?
|
|
|
|
|
Apr 18 2018, 16:43
|
Группа: Участник
Сообщений: 11
Регистрация: 1-11-15
Пользователь №: 89 126
|
Цитата(VladislavS @ Apr 18 2018, 17:35) Код initialize by copy { readonly, readwrite }; Ничем не смущает? Ну так я же в начале темы написал, что хочу, чтобы код выполнялся из RAM. Для этого копирую туда код
|
|
|
|
|
Apr 18 2018, 17:09
|
Местный
Группа: Свой
Сообщений: 475
Регистрация: 14-04-05
Из: Москва
Пользователь №: 4 140
|
Код initialize by copy { readonly, readwrite } except{ section .httpContent}; Cекцию .httpContent в исключения попробуй. Цитата(jcxz @ Apr 18 2018, 19:52) В readonly секции?? Вообще, мануал именно это и советует. Цитата If you want to copy the entire application from ROM to RAM at program startup, use the initilize by copy directive, for example: initialize by copy { readonly, readwrite }; The readwrite pattern will match all statically initialized variables and arrange for them to be initialized at startup. The readonly pattern will do the same for all read-only code and data, except for code and data needed for the initialization. Нелогично... Наверное, except{ section .httpContent} поможет. PS: А зачем для секции .httpContent свой регион заводить? Можно же просто в ROM_region её разместить. Даже больше, что-то мне подсказывает, что с атрибутом static const массив и так в readonly должен попасть. PPS: Я рассуждаю так. Регионы описывают физические области памяти, если они отличаются друг от друга свойствами или разнесены в адресном пространстве. Секции же описывают области программы. Если какие-то части программы (секции) требуют особенного размещения, то их размещают в нужном регионе, а если требуется то и по фиксированному адресу. Создавать пересекающиеся регионы - только осложнять жизнь линкеру.
Сообщение отредактировал VladislavS - Apr 18 2018, 17:50
|
|
|
|
|
Apr 18 2018, 22:00
|
Гуру
Группа: Свой
Сообщений: 5 228
Регистрация: 3-07-08
Из: Омск
Пользователь №: 38 713
|
Цитата(VladislavS @ Apr 18 2018, 20:09) Нелогично... Наверное, except{ section .httpContent} поможет. initialize by copy { readonly, readwrite } Тогда уж лучше указать конкретные секции в initialize by copy {}. А readonly, readwrite в указанной конструкции указывают на все секции имеющие соответствующие атрибуты. Цитата(VladislavS @ Apr 18 2018, 20:09) PS: А зачем для секции .httpContent свой регион заводить? Можно же просто в ROM_region её разместить. Даже больше, что-то мне подсказывает, что с атрибутом static const массив и так в readonly должен попасть. Его секция будет иметь атрибут readoinly. Ну и попадёт в тот регион, который для readonly. Я скопировал этот пример из .icf-файла для отладки в ОЗУ. Там все остальные секции идут в ОЗУ. Только .httpContent - во FLASH так как большой и не меняется - нет смысла грузить при каждой отладке. Цитата(VladislavS @ Apr 18 2018, 20:09) Создавать пересекающиеся регионы - только осложнять жизнь линкеру. У меня и нет пересекающихся регионов. Код define region FLASHC_regionA = mem:[from 0x08000000 to 0x08001FFF]; //PMU/FLASH (cached) define region FLASHC_regionB = mem:[from 0x08002000 to 0x08003FFF]; //PMU/FLASH (cached) define region FLASHC_regionC = mem:[from 0x08020000 to 0x081FFFFF]; //PMU/FLASH (cached) define region FLASHU_regionA = mem:[from 0x0C000000 to 0x0C1FFFFF]; //PMU/FLASH (uncached) define region RAM_regionA = mem:[from 0x1FFE8000 size 0x18000]; //PSRAM (code) define region RAM_regionB = mem:[from 0x20000100 size 0x1FF00]; //DSRAM1 (RW-data) define region RAM_regionC = mem:[from 0x20020000 size 0x20000]; //DSRAM2 (RO-data) PS: А "ROM" я называю именно ROM - Read Only Memory. FLASH != ROM
|
|
|
|
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0
|
|
|