Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: Размещение функции в RAM
Форум разработчиков электроники ELECTRONIX.ru > Сайт и форум > В помощь начинающему > ARM, 32bit
toweroff
Попробовал разместить критичные, с точки зрения времени выполнения, функции в RAM. До этого не пользовался, но сравнения для )
Keil 4.11, использовал следующие комбинации для кода и скаттера:

1.
Код
Код
#pragma arm section code = "ramfunc"
тело
#pragma arm section code


scatter
Код
LR_IROM1 0x00000000  0x00080000 {  ; load region size_region
  ER_IROM1 0x00000000
  {; load address = execution address
   STARTUP.o (RESET, +First)
   *(InRoot$$Sections)
   .ANY (+RO)
}
  RW_IRAM1 0x40000000 0x00010000  {; RW data
   *(ramfunc)
   .ANY (+RW +ZI)
  }
}


2.
По совету, найденному в примерах Keil, добавил для всего С-файла в свойствах "Memory Assignment->Code/Const->IRAM1(0x40000000-0x4000FFFF)" и собрал все со сгенеренным scatter'ом

Код - убрал прагмы

scatter
Код
LR_IROM1 0x00000000 0x00080000  {   ; load region size_region
  ER_IROM1 0x00000000 0x00080000  { ; load address = execution address
   *.o (RESET, +First)
   *(InRoot$$Sections)
   .ANY (+RO)
  }
  RW_IRAM1 0x40000000 0x00010000  { ; RW data
    mscuser.o (+RO)
   .ANY (+RW +ZI)
  }
}


Итого

Размер используемой RAM при компиляции не изменяется никак.
Так должно быть (с объемом), или я что-то сделал не так и компилятор с линкером мне должны сказать (хотя бы по количеству), что RAM пользуется больше?
aaarrr
Цитата(toweroff @ Jun 9 2010, 23:23) *
использовал следующие комбинации для кода и скаттера

Все правильно сделали.

Цитата(toweroff @ Jun 9 2010, 23:23) *
По совету, найденному в примерах Keil, добавил для всего С-файла в свойствах "Memory Assignment->Code/Const->IRAM1(0x40000000-0x4000FFFF)"

А это плохой совет.

Цитата(toweroff @ Jun 9 2010, 23:23) *
Размер используемой RAM при компиляции не изменяется никак.
Так должно быть (с объемом), или я что-то сделал не так и компилятор с линкером мне должны сказать (хотя бы по количеству), что RAM пользуется больше?

Смотреть нужно не размер использованной памяти, а map файл.

Не знаю как с Keil'ом, а на RVDS 2.2 есть баг: компилятор может заинлайнить функцию, проигнорировав #pragma arm section. Баг можно обойти установкой более низкого уровня оптимизации (#pragma o1) для конкретной функции, или вынесением ее в отдельный модуль.
sonycman
Цитата(aaarrr @ Jun 9 2010, 23:30) *
А это плохой совет.

Но почему?
Зачем лазить в каракули скаттера, когда можно вынести необходимые функции в один файл и в его свойствах указать расположение кода в ОЗУ?
Что в этом плохого?
aaarrr
Цитата(sonycman @ Jun 9 2010, 23:35) *
Что в этом плохого?

Плохо подстраивать (уродовать, точнее) структуру проекта в угоду "галочке" в среде.
Плохо лениться разобраться с нехитрым скриптом скаттера.
toweroff
Цитата(aaarrr @ Jun 9 2010, 23:30) *
Баг можно обойти установкой более низкого уровня оптимизации (#pragma o1) для конкретной функции, или вынесением ее в отдельный модуль.

в отдельном модуле, оптимизация _всего_ кода с установленной для данного модуля O1 дало уменьшение используемой RAM -4 байта

Цитата(sonycman @ Jun 9 2010, 23:35) *
Но почему?
Зачем лазить в каракули скаттера, когда можно вынести необходимые функции в один файл и в его свойствах указать расположение кода в ОЗУ?
Что в этом плохого?

как сказал ув.zltigo, лучше руками, чем галочками smile.gif за цитату не ручаюсь, но суть верна smile.gif
aaarrr
Цитата(toweroff @ Jun 9 2010, 23:47) *
в отдельном модуле, оптимизация _всего_ кода с установленной для данного модуля O1 дало уменьшение используемой RAM -4 байта

Да при чем тут размер, в map-файле что написано?
toweroff
Посмотрел *.map
модуль расположился в адресах IRAM, вроде ок
отложим до завтра линковку, проверю работоспособность и скорость

пока по коду - собственно, кто конкретно расположит мой модуль в RAM? механизьмаф не знаю, потому спрашиваю smile.gif просто надобности не было
aaarrr
Цитата(toweroff @ Jun 9 2010, 23:53) *
пока по коду - собственно, кто конкретно расположит мой модуль в RAM? механизьмаф не знаю, потому спрашиваю smile.gif просто надобности не было

__main разложит после startup'а.
toweroff
Цитата(aaarrr @ Jun 10 2010, 00:01) *
__main разложит после startup'а.

спасибо, завтра протестирую

Цитата(aaarrr @ Jun 10 2010, 00:01) *
__main разложит после startup'а.

и, все-таки, пока не завтра...
в стартапе

Код
                IMPORT  __main
                LDR     R0, =__main
                BX      R0


далее все стартует на мой код в main() бред

В стартапе:
до того - настройка слоков и EMC, все лишнее отключено/поудалено... да и не было там ранее вроде копирования никакого областей никуда

так кто же копирует код из flash в RAM для выполнения? неверно

как отследить __main?
aaarrr
Цитата(toweroff @ Jun 10 2010, 00:34) *
и, все-таки, пока не завтра...
в стартапе

Код
                IMPORT  __main
                LDR     R0, =__main
                BX      R0


далее все стартует на мой код в main()
до того - настройка слоков и EMC, все лишнее отключено/поудалено... да и не было там ранее вроде копирования никакого областей никуда

__main != main. И кто же тогда ранее инициализировал переменные, __user_initial_stackheap вызывал?
Стартует оно потому что галочка "Run to main()" стоит.
toweroff
Цитата
__main != main.

знаю.. теперь smile.gif

Цитата
И кто же тогда ранее инициализировал переменные, __user_initial_stackheap вызывал?

копаюсь...

Цитата
Стартует оно потому что галочка "Run to main()" стоит.

навсегда off...

Цитата( @ Jun 10 2010, 00:42) *
__user_initial_stackheap вызывал?

нет

там начался _scatterload_*
разбираюсь опять... но, похоже, должно быть оно... по крайней мере, по названию
sonycman
Цитата(toweroff @ Jun 10 2010, 01:03) *
навсегда off...

А это зря, делать Вам нечего в дебрях стандартного сишного стартапа.
Он всё разложит в соответствии с "галочками", ну или с "ручками", если они достаточно прямые smile.gif
aaarrr
Цитата(toweroff @ Jun 10 2010, 01:03) *
копаюсь...

Про вызов __user_initial_stackheap и инициализацию переменных - это был не вопрос. Все эти действия, как и вызов __scatter*, выполняет библиотечный __main.

Цитата(sonycman @ Jun 10 2010, 01:08) *
А это зря, делать Вам нечего в дебрях стандартного сишного стартапа.
Он всё разложит в соответствии с "галочками", ну или с "ручками", если они достаточно прямые smile.gif

В дебрях библиотек - действительно практически нечего, разве что полюбопытствовать или посмотреть вызов вышеупомянутого stackheap. А вот в собственном стартапе бывает что и нужно.
toweroff
Цитата(sonycman @ Jun 10 2010, 01:08) *
А это зря, делать Вам нечего в дебрях стандартного сишного стартапа.
Он всё разложит в соответствии с "галочками", ну или с "ручками", если они достаточно прямые smile.gif

не сомневаюсь, но хотя бы один раз посмотреть необходимо. Успокоения и осознания для smile.gif

Цитата(sonycman @ Jun 10 2010, 01:08) *
А это зря, делать Вам нечего в дебрях стандартного сишного стартапа.
Он всё разложит в соответствии с "галочками", ну или с "ручками", если они достаточно прямые smile.gif

я просто точки останова ставил там, где необходимо smile.gif

кстати, всякие __rt_lib_* откуда? я не пользую RL-ARM, и листбокс в Operation System стоит как None......
aaarrr
__rt* - компоненты стандартных библиотек, к RL-ARM никакого отношения не имеют.
toweroff
Цитата(aaarrr @ Jun 10 2010, 01:31) *
__rt* - компоненты стандартных библиотек, к RL-ARM никакого отношения не имеют.

понял, спасибо
Сергей Борщ
Цитата(sonycman @ Jun 10 2010, 00:08) *
А это зря, делать Вам нечего в дебрях стандартного сишного стартапа.
Как будто в самом начале main() есть много чего делать smile.gif Сразу точку останова в проблемное место и побежали, без пит-стопа в main(). Да и железная точка останова освобождается.
toweroff
Протестил софт (занимается передачей по USB данных, полученных через static EMC) с USB функциями, работающими в RAM
Результат - ноль прироста и ноль падения производительности. Как была передача ~950 Кбайт/сек, так и осталась. Конечно, размещение в RAM - дело полезное, но мне не помогло
sonycman
950 кб/сек. - практически потолок для Full Speed USB.
Какую скорость Вы бы хотели получить?
toweroff
Цитата(sonycman @ Jun 10 2010, 16:52) *
950 кб/сек. - практически потолок для Full Speed USB.
Какую скорость Вы бы хотели получить?

в том-то и дело, что "практически". Хотелось максимум того, что возможно smile.gif
Поэтому придется все-таки ставить High-speed интерфейсы
alt3857
Подскажите каким образом создать секции допустим fast и slow, все, что в секции fast будет выполняться из RAM, а все, что в секции slow - из флеш. А в программе где нужно обрамлять pragma. Как это сделать? Примеры, которые удалось найти требуют помещения в нужный участок памяти всего модуля *.o - в scatter-файле. А как сделать через pragma, т.е. чтобы в любом месте кода прагмой можно было определить откуда он должен выполняться?
toweroff
Цитата(alt3857 @ Dec 9 2011, 15:18) *
Подскажите каким образом создать секции допустим fast и slow, все, что в секции fast будет выполняться из RAM, а все, что в секции slow - из флеш. А в программе где нужно обрамлять pragma. Как это сделать? Примеры, которые удалось найти требуют помещения в нужный участок памяти всего модуля *.o - в scatter-файле. А как сделать через pragma, т.е. чтобы в любом месте кода прагмой можно было определить откуда он должен выполняться?


Вот здесь

И про структуру скаттер-файла почитайте, внизу есть ссылка
alt3857
toweroff, как секцию обрамить прагмой я понимаю, как ее (секцию) правильно в scatter-файле описать??? Я пробовал так:

LR_IROM1 0x00000000 0x00080000 { ; load region size_region
ER_IROM1 0x00000000 0x00080000 { ; load address = execution address
*.o (RESET, +First)
*(InRoot$$Sections)
.ANY (+RO)
}
RW_IRAM1 0x10000000 0x00004000 { ; RW data
.ANY (+RW +ZI)
}
ER_FAST 0x10004000 0x8000 {
*(fastcode)
}
}

Не работает.
toweroff
примерно так
Код
; *************************************************************
; *** Scatter-Loading Description File generated by uVision ***
; *************************************************************

LR_IROM1 0x20000000 0x00010000  { ; load region size_region
  RO_IRAM1 0x20000000 0x00010000  {; RO data
   *.o (RESET, +First)
   *(InRoot$$Sections)
   .ANY (+RO)
  }
  RW_IRAM2 0x80000000 0x0000E000  {; RW data
   *.o (RAMFUNC)
   .ANY (+RW, +ZI)
  }
}


потом проверьте в map-файле, куда что легло
alt3857
Сделал:

LR_IROM1 0x00000000 0x00080000 { ; load region size_region
ER_IROM1 0x00000000 0x00080000 { ; load address = execution address
*.o (RESET, +First)
*(InRoot$$Sections)
.ANY (+RO)
}
RW_IRAM1 0x10000000 0x00004000 { ; RW data
.ANY (+RW +ZI)
}
ER_FAST 0x10004000 0x8000 {
*.o (fast)
.ANY (+RW, +ZI)
}
}
Код функции такой (делаю в симуляторе):
#pragma arm section code = "fast"
int func2(int a)
{
return (a*a*a);
}
#pragma arm section code

В map-файле все норм. разместилось в секции fast, но программа не работает.
toweroff
а в стартапе что?
alt3857
В стартапе ничего не менял - на сайте arm-а ничего про стартап не пишут
toweroff
ну как... startup.s в проекте есть? или что-то, где настраиваются вектора, стеки и т.д.
alt3857
Цитата
ну как... startup.s в проекте есть? или что-то, где настраиваются вектора, стеки и т.д.

Есть конечно, там я ничего не менял, подскажите если там тоже нужно что-то добавить?
toweroff
Цитата(alt3857 @ Dec 9 2011, 19:11) *
Есть конечно, там я ничего не менял, подскажите если там тоже нужно что-то добавить?

ну так выкладывайте sm.gif

Цитата(alt3857 @ Dec 9 2011, 15:56) *
Код функции такой (делаю в симуляторе):

В map-файле все норм. разместилось в секции fast, но программа не работает.

так в симуляторе не работает или в железке?
alt3857
Код
;/*****************************************************************************
; * @file:    startup_LPC17xx.s
; * @purpose: CMSIS Cortex-M3 Core Device Startup File
; *           for the NXP LPC17xx Device Series
; * @version: V1.0
; * @date:    25. Nov. 2008
; *------- <<< Use Configuration Wizard in Context Menu >>> ------------------
; *
; * Copyright (C) 2008 ARM Limited. All rights reserved.
; * ARM Limited (ARM) is supplying this software for use with Cortex-M3
; * processor based microcontrollers.  This file can be freely distributed
; * within development tools that are supporting such ARM based processors.
; *
; * THIS SOFTWARE IS PROVIDED "AS IS".  NO WARRANTIES, WHETHER EXPRESS, IMPLIED
; * OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF
; * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE.
; * ARM SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL, OR
; * CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.
; *
; *****************************************************************************/


; <h> Stack Configuration
;   <o> Stack Size (in Bytes) <0x0-0xFFFFFFFF:8>
; </h>

Stack_Size      EQU     0x00000200

                AREA    STACK, NOINIT, READWRITE, ALIGN=3
Stack_Mem       SPACE   Stack_Size
__initial_sp


; <h> Heap Configuration
;   <o>  Heap Size (in Bytes) <0x0-0xFFFFFFFF:8>
; </h>

Heap_Size       EQU     0x00000000

                AREA    HEAP, NOINIT, READWRITE, ALIGN=3
__heap_base
Heap_Mem        SPACE   Heap_Size
__heap_limit


                PRESERVE8
                THUMB


; Vector Table Mapped to Address 0 at Reset

                AREA    RESET, DATA, READONLY
                EXPORT  __Vectors

__Vectors       DCD     __initial_sp             ; Top of Stack
                DCD     Reset_Handler            ; Reset Handler
                DCD     NMI_Handler              ; NMI Handler
                DCD     HardFault_Handler        ; Hard Fault Handler
                DCD     MemManage_Handler        ; MPU Fault Handler
                DCD     BusFault_Handler         ; Bus Fault Handler
                DCD     UsageFault_Handler       ; Usage Fault Handler
                DCD     0                        ; Reserved
                DCD     0                        ; Reserved
                DCD     0                        ; Reserved
                DCD     0                        ; Reserved
                DCD     SVC_Handler              ; SVCall Handler
                DCD     DebugMon_Handler         ; Debug Monitor Handler
                DCD     0                        ; Reserved
                DCD     PendSV_Handler           ; PendSV Handler
                DCD     SysTick_Handler          ; SysTick Handler

               ; External Interrupts
                DCD     WDT_IRQHandler           ; 16: Watchdog Timer
                DCD     TIMER0_IRQHandler        ; 17: Timer0
                DCD     TIMER1_IRQHandler        ; 18: Timer1
                DCD     TIMER2_IRQHandler        ; 19: Timer2
                DCD     TIMER3_IRQHandler        ; 20: Timer3
                DCD     UART0_IRQHandler         ; 21: UART0
                DCD     UART1_IRQHandler         ; 22: UART1
                DCD     UART2_IRQHandler         ; 23: UART2
                DCD     UART3_IRQHandler         ; 24: UART3
                DCD     PWM1_IRQHandler          ; 25: PWM1
                DCD     I2C0_IRQHandler          ; 26: I2C0
                DCD     I2C1_IRQHandler          ; 27: I2C1
                DCD     I2C2_IRQHandler          ; 28: I2C2
                DCD     SPI_IRQHandler           ; 29: SPI
                DCD     SSP0_IRQHandler          ; 30: SSP0
                DCD     SSP1_IRQHandler          ; 31: SSP1
                DCD     PLL0_IRQHandler          ; 32: PLL0 Lock (Main PLL)
                DCD     RTC_IRQHandler           ; 33: Real Time Clock
                DCD     EINT0_IRQHandler         ; 34: External Interrupt 0
                DCD     EINT1_IRQHandler         ; 35: External Interrupt 1
                DCD     EINT2_IRQHandler         ; 36: External Interrupt 2
                DCD     EINT3_IRQHandler         ; 37: External Interrupt 3
                DCD     ADC_IRQHandler           ; 38: A/D Converter
                DCD     BOD_IRQHandler           ; 39: Brown-Out Detect
                DCD     USB_IRQHandler           ; 40: USB
                DCD     CAN_IRQHandler           ; 41: CAN
                DCD     DMA_IRQHandler           ; 42: General Purpose DMA
                DCD     I2S_IRQHandler           ; 43: I2S
                DCD     ENET_IRQHandler          ; 44: Ethernet
                DCD     RIT_IRQHandler           ; 45: Repetitive Interrupt Timer
                DCD     MCPWM_IRQHandler         ; 46: Motor Control PWM
                DCD     QEI_IRQHandler           ; 47: Quadrature Encoder Interface
                DCD     PLL1_IRQHandler          ; 48: PLL1 Lock (USB PLL)


                IF      :LNOT::DEF:NO_CRP
                AREA    |.ARM.__at_0x02FC|, CODE, READONLY
CRP_Key         DCD     0xFFFFFFFF
                ENDIF


                AREA    |.text|, CODE, READONLY


; Reset Handler

Reset_Handler   PROC
                EXPORT  Reset_Handler             [WEAK]
                IMPORT  __main
                LDR     R0, =__main
                BX      R0
                ENDP


; Dummy Exception Handlers (infinite loops which can be modified)                

NMI_Handler     PROC
                EXPORT  NMI_Handler               [WEAK]
                B       .
                ENDP
HardFault_Handler\
                PROC
                EXPORT  HardFault_Handler         [WEAK]
                B       .
                ENDP
MemManage_Handler\
                PROC
                EXPORT  MemManage_Handler         [WEAK]
                B       .
                ENDP
BusFault_Handler\
                PROC
                EXPORT  BusFault_Handler          [WEAK]
                B       .
                ENDP
UsageFault_Handler\
                PROC
                EXPORT  UsageFault_Handler        [WEAK]
                B       .
                ENDP
SVC_Handler     PROC
                EXPORT  SVC_Handler               [WEAK]
                B       .
                ENDP
DebugMon_Handler\
                PROC
                EXPORT  DebugMon_Handler          [WEAK]
                B       .
                ENDP
PendSV_Handler  PROC
                EXPORT  PendSV_Handler            [WEAK]
                B       .
                ENDP
SysTick_Handler PROC
                EXPORT  SysTick_Handler           [WEAK]
                B       .
                ENDP

Default_Handler PROC

                EXPORT  WDT_IRQHandler            [WEAK]
                EXPORT  TIMER0_IRQHandler         [WEAK]
                EXPORT  TIMER1_IRQHandler         [WEAK]
                EXPORT  TIMER2_IRQHandler         [WEAK]
                EXPORT  TIMER3_IRQHandler         [WEAK]
                EXPORT  UART0_IRQHandler          [WEAK]
                EXPORT  UART1_IRQHandler          [WEAK]
                EXPORT  UART2_IRQHandler          [WEAK]
                EXPORT  UART3_IRQHandler          [WEAK]
                EXPORT  PWM1_IRQHandler           [WEAK]
                EXPORT  I2C0_IRQHandler           [WEAK]
                EXPORT  I2C1_IRQHandler           [WEAK]
                EXPORT  I2C2_IRQHandler           [WEAK]
                EXPORT  SPI_IRQHandler            [WEAK]
                EXPORT  SSP0_IRQHandler           [WEAK]
                EXPORT  SSP1_IRQHandler           [WEAK]
                EXPORT  PLL0_IRQHandler           [WEAK]
                EXPORT  RTC_IRQHandler            [WEAK]
                EXPORT  EINT0_IRQHandler          [WEAK]
                EXPORT  EINT1_IRQHandler          [WEAK]
                EXPORT  EINT2_IRQHandler          [WEAK]
                EXPORT  EINT3_IRQHandler          [WEAK]
                EXPORT  ADC_IRQHandler            [WEAK]
                EXPORT  BOD_IRQHandler            [WEAK]
                EXPORT  USB_IRQHandler            [WEAK]
                EXPORT  CAN_IRQHandler            [WEAK]
                EXPORT  DMA_IRQHandler            [WEAK]
                EXPORT  I2S_IRQHandler            [WEAK]
                EXPORT  ENET_IRQHandler           [WEAK]
                EXPORT  RIT_IRQHandler            [WEAK]
                EXPORT  MCPWM_IRQHandler          [WEAK]
                EXPORT  QEI_IRQHandler            [WEAK]
                EXPORT  PLL1_IRQHandler           [WEAK]

WDT_IRQHandler          
TIMER0_IRQHandler        
TIMER1_IRQHandler        
TIMER2_IRQHandler        
TIMER3_IRQHandler        
UART0_IRQHandler          
UART1_IRQHandler          
UART2_IRQHandler          
UART3_IRQHandler          
PWM1_IRQHandler          
I2C0_IRQHandler          
I2C1_IRQHandler          
I2C2_IRQHandler          
SPI_IRQHandler            
SSP0_IRQHandler          
SSP1_IRQHandler          
PLL0_IRQHandler          
RTC_IRQHandler            
EINT0_IRQHandler          
EINT1_IRQHandler          
EINT2_IRQHandler          
EINT3_IRQHandler          
ADC_IRQHandler            
BOD_IRQHandler            
USB_IRQHandler            
CAN_IRQHandler            
DMA_IRQHandler          
I2S_IRQHandler            
ENET_IRQHandler      
RIT_IRQHandler          
MCPWM_IRQHandler            
QEI_IRQHandler            
PLL1_IRQHandler          

                B       .

                ENDP


                ALIGN


; User Initial Stack & Heap

                IF      :DEF:__MICROLIB
                
                EXPORT  __initial_sp
                EXPORT  __heap_base
                EXPORT  __heap_limit
                
                ELSE
                
                IMPORT  __use_two_region_memory
                EXPORT  __user_initial_stackheap
__user_initial_stackheap

                LDR     R0, =  Heap_Mem
                LDR     R1, =(Stack_Mem + Stack_Size)
                LDR     R2, = (Heap_Mem +  Heap_Size)
                LDR     R3, = Stack_Mem
                BX      LR

                ALIGN

                ENDIF


                END


Стандартный startup.
Для просмотра полной версии этой страницы, пожалуйста, пройдите по ссылке.
Invision Power Board © 2001-2025 Invision Power Services, Inc.