W tym poście chciałbym opisać drugi sposób przygotowania projektu STM32H7 z użyciem LWIP.
Bufory:
Na samym początku przejdziemy przez rozmieszczenie buforów w pamięci.
Bufory DMA RX oraz TX umieszczamy w pamięci RAMD2. Pamięć ta składa się z dwóch części SRAM1 o rozmiarze 128KB (zakres adresów to 0x30000000 do 0x3001FFFF). Druga część tej pamięci to SRAM2 także o rozmiarze 128KB (zakres adresów to 0x30020000 do 0x3003FFFF).
Do rozmieszczenia w pamięci namy następujące bufory:
- DMARxDscrTab - tablica odbiorcza deskryptorów DMA.
- DMATxDscrTab - tablica nadawcza deskryptorów DMA.
- memp_memory_RX_POOL_base - wskaźnik dla bufora pamięci wykorzystywanego przez stos LWIP do przechowywania odebranych pakietów.
- LWIP heap - rozmiar pamięci dla LWIP. Zarządzanie tą cześcią pamięci odbywa się przez LWIP. Pamięc jest tam alokowana dla funkcji sieciowych, buforowania danych oraz innych operacji dotyczących zarządzania komunikacją sieciową.
Przy wykorzystywaniu 256 bajtów, mamy miejsce na szesnaście lub osiem deskryptorów. Zależy to od wielkości descryptora (16 lub 32 bajty). STM32H7 wykorzystuje wersje 32 bajtową.
Należy pamiętać, że z tej pamięci korzystają takie interfejsy jak QUAD-SPI, USB OTH HS czy SDMMC2. Jeśli w projekcie będą one wykorzystywane to należy zmniejszyć rozmiar buforów wykorzystywanych przez ETH i LWIP, lub część z nich przenieść do innej pamięci. W przykładzie znalezionym na stronie ST start bufforów rozpoczyna się od adresu 0x30020000, co daje wystarczający zapas na inne operacje.
Standardowa konfiguracja jaką znalazłem, rozmieszcza bufory w następujący sposób:
- DMARxDscrTab - 256 bajtów - adres 0x30040000,
- DMATxDscrTab - 256 bajtów - adres 0x30040100,
- memp_memory_RX_POOL_base - adres 0x30040200,
- LWIP heap - adres 0x30020000.
memp_memory_RX_POOL_base przy takiej adresacji zyskuje możliwość przechowywania 12 buforów 1536 + 24. Wartość 1526 jest to standardowo przyjęta wartość jaką się stosuje dla ramek ethernetowych. W tym przypadku zastosowana wartości 1536 wynika z wyrównania pamięci. Jest to wartość z zapasem, która pozwala na umieszczenie wszystkich danych wraz z nagłówkami. Dodatkowe 24 bajty będą wykorzystywane na dane dodatkowe jak struktury czy metadane.
Ogólnie podczas doborów obszarów pamięci należy pamiętać by DMARxDscrTab było możliwie największe (minimum to 96). Jeśli chodzi o deskryptor DMATxDscrTab to tutaj jest to głównie zależne od tego czym się aplikacja będzie zajmować i ile pakietów będzie musiała przygotowywać przed wysłaniem. W tym przypadku zalecane minimum to także 96. Natomiast w jednym przykładzie od ST skonfigurowana była wartość 32. Należy tutaj pamiętać, że gdy te wartości będą za małe mogą wystąpić problemy z poprawną komunikacją z urządzeniem.
Deskryptory przechowywane są w następujący sposób:
- typedef struct
- {
- __IO uint32_t DESC0;
- __IO uint32_t DESC1;
- __IO uint32_t DESC2;
- __IO uint32_t DESC3;
- uint32_t BackupAddr0; /* used to store rx buffer 1 address */
- uint32_t BackupAddr1; /* used to store rx buffer 2 address */
- } ETH_DMADescTypeDef;
Cały deskryptor zajmuje 24 bajty danych. Tablicę tych deskryptorów zapisujemy w pamięci w sekcji RxDecripSection oraz TxDecriptSection. Wobec tego deklaracje tych tablic (plik ethernetif.c) muszą być dostosowane do ilości miejsca jakie na nie przeznaczamy:
- #if defined ( __ICCARM__ ) /*!< IAR Compiler */
- #pragma location=0x30040000
- ETH_DMADescTypeDef DMARxDscrTab[ETH_RX_DESC_CNT]; /* Ethernet Rx DMA Descriptors */
- #pragma location=0x30040100
- ETH_DMADescTypeDef DMATxDscrTab[ETH_TX_DESC_CNT]; /* Ethernet Tx DMA Descriptors */
- #elif defined ( __CC_ARM ) /* MDK ARM Compiler */
- __attribute__((at(0x30040000))) ETH_DMADescTypeDef DMARxDscrTab[ETH_RX_DESC_CNT]; /* Ethernet Rx DMA Descriptors */
- __attribute__((at(0x30040100))) ETH_DMADescTypeDef DMATxDscrTab[ETH_TX_DESC_CNT]; /* Ethernet Tx DMA Descriptors */
- #elif defined ( __GNUC__ ) /* GNU Compiler */
- ETH_DMADescTypeDef DMARxDscrTab[ETH_RX_DESC_CNT] __attribute__((section(".RxDecripSection"))); /* Ethernet Rx DMA Descriptors */
- ETH_DMADescTypeDef DMATxDscrTab[ETH_TX_DESC_CNT] __attribute__((section(".TxDecripSection"))); /* Ethernet Tx DMA Descriptors */
- #endif
ETH_RX_DESC_CNT oraz ETH_TX_DESC_CNT są domyślnie ustawione na 4. Co odpowiada opiasnej wcześniej minimalnej wartości 96 bajtów. Jeśli przeznaczymy na deskryptory za mało pamięci to nie uda się skompilować projektu.
Konfiguracja biblioteki:
Kolejnym elementem jest ustawienie wartości w bibliotece LWIP:
- TCP_MSS - maksymalny rozmiar segmentu. Domyślnie przyjęta jest wartości 536. Należy tutaj wprowadzić wartość 1460. Wynika ona z maksymalnego rozmiaru ramki wynoszący 1500 czy 1526 gdy wliczymy całość pakietu. Wartość ta odpowiada maksymalnemu rozmiarowi danych w warstwie aplikacji. Jeśli zostanie wpisana mniejsza wartości, to będzie konieczna fragmentacja ramki.
- TCP_SND_BUF - maksymalny rozmiar wysyłanego bufora. W tym przypadku zalecana wartość to 5840, co odpowiada czteroktorności MSS. Pozwala to na buforowanie wystarczającej ilości danych.
- TCP_WND - rozmiar okna odbiorczego. Dobrze jest ustawić go na pasujący do TCP_SND_BUF. Czyli przy odbiorze możemy obsłużyć cztery segmenty danych. Zapewni to maksymlanie zrównoważone nadawanie z odbieraniem pakietów.
- TCP_SND_QUEUELEN - rozmiar kolejki dla wysyłanych danych. Ta wartości powinna być wielokrotnością ilości pakietów jakie są przechowywane w buforze. Tutaj zalecana wartość początkowa wynosi 16.
Jeśli wprowadzimy takie ustawienia, bądź inne, to trzeba się upewnić, że starczy miejsca w pamięci na wszystkie wprowadzone ustawienia:
TCP_SND_BUF - 5840 (4 * 1460)
TCP_WND - 5840 (4 * 1460)
TCP_SND_QUEUELEN - 23360 (16 * 1460)
W tym przypadku na bufory potrzeba zarezerwować 35040 bajtów danych. Nie są to oczywiście wszystkie dane potrzebne dla biblioteki LWIP.
Za ilość pamieci jaka została zarezerwowana dla LWIP odpowiada define:
- #define MEM_SIZE 131048
Konfiguracja MPU w kodzie wygląda następująco:
- void MPU_Config(void)
- {
- MPU_Region_InitTypeDef MPU_InitStruct = {0};
- /* Disables the MPU */
- HAL_MPU_Disable();
- /** Initializes and configures the Region and the memory to be protected
- */
- MPU_InitStruct.Enable = MPU_REGION_ENABLE;
- MPU_InitStruct.Number = MPU_REGION_NUMBER0;
- MPU_InitStruct.BaseAddress = 0x0;
- MPU_InitStruct.Size = MPU_REGION_SIZE_4GB;
- MPU_InitStruct.SubRegionDisable = 0x87;
- MPU_InitStruct.TypeExtField = MPU_TEX_LEVEL0;
- MPU_InitStruct.AccessPermission = MPU_REGION_FULL_ACCESS;
- MPU_InitStruct.DisableExec = MPU_INSTRUCTION_ACCESS_DISABLE;
- MPU_InitStruct.IsShareable = MPU_ACCESS_SHAREABLE;
- MPU_InitStruct.IsCacheable = MPU_ACCESS_NOT_CACHEABLE;
- MPU_InitStruct.IsBufferable = MPU_ACCESS_NOT_BUFFERABLE;
- HAL_MPU_ConfigRegion(&MPU_InitStruct);
- /** Initializes and configures the Region and the memory to be protected
- */
- MPU_InitStruct.Number = MPU_REGION_NUMBER1;
- MPU_InitStruct.BaseAddress = 0x30020000;
- MPU_InitStruct.Size = MPU_REGION_SIZE_128KB;
- MPU_InitStruct.SubRegionDisable = 0x0;
- MPU_InitStruct.TypeExtField = MPU_TEX_LEVEL1;
- MPU_InitStruct.IsShareable = MPU_ACCESS_NOT_SHAREABLE;
- HAL_MPU_ConfigRegion(&MPU_InitStruct);
- /** Initializes and configures the Region and the memory to be protected
- */
- MPU_InitStruct.Number = MPU_REGION_NUMBER2;
- MPU_InitStruct.BaseAddress = 0x30040000;
- MPU_InitStruct.Size = MPU_REGION_SIZE_512B;
- MPU_InitStruct.TypeExtField = MPU_TEX_LEVEL0;
- MPU_InitStruct.IsShareable = MPU_ACCESS_SHAREABLE;
- MPU_InitStruct.IsBufferable = MPU_ACCESS_BUFFERABLE;
- HAL_MPU_ConfigRegion(&MPU_InitStruct);
- /* Enables the MPU */
- HAL_MPU_Enable(MPU_PRIVILEGED_DEFAULT);
- }
Dla pamieci flash wprowadzamy następującą konfigurację:
- MEMORY
- {
- DTCMRAM (xrw) : ORIGIN = 0x20000000, LENGTH = 128K
- ITCMRAM (xrw) : ORIGIN = 0x00000000, LENGTH = 64K
- RAM_D1 (xrw) : ORIGIN = 0x24000000, LENGTH = 512K
- RAM_D2 (xrw) : ORIGIN = 0x30000000, LENGTH = 288K
- RAM_D3 (xrw) : ORIGIN = 0x38000000, LENGTH = 64K
- FLASH (rx) : ORIGIN = 0x8000000, LENGTH = 2048K
- }
- .lwip_sec (NOLOAD) :
- {
- . = ABSOLUTE(0x30040000);
- *(.RxDecripSection)
- . = ABSOLUTE(0x30040100);
- *(.TxDecripSection)
- . = ABSOLUTE(0x30040200);
- *(.Rx_PoolSection)
- } >RAM_D2
Testowałem też inne wartości dla Rx oraz Tx. Przy wykorzystywaniu prostych aplikacji TCP Echo Serwer wraz z UDP Serwer nie miało to większego znaczenia. Może przy bardziej rozbudowanych aplikacjach miało by to większe znaczenie.
Zgodnie z tym co udało mi się znaleść w internecie najlepiej ustawić bufor RX ustawić jako:
- ETH_RX_BUFFER_CNT == ETH_RX_DESC_CNT
Powinno to zapewnić najlepsze wykorzystanie zasobów. Ponieważ bufor RX może być odrazu przekazywany do deskryptorów. Co oznacza brak konieczności zarządzania przyjętymi buforami, zanim trafią one do deskryptorów. Oczywiście należało by wtedy wprowadzić większą wartość RX_BUFFER_CNT niż 4 (najlepiej 12). To z kolei oznacza przeznaczenie większej ilości pamięci. Dla tego układu raczej to nie byłoby problemem. Jednak w przypadku korzystania z H723 czy H725, gdzie ilość pamięci RAMD2 jest bardziej ograniczona mogło by to już stanowić pewien problem.
Dla takiej konfiguracji pamięć powinna wyglądać np. tak:
- .lwip_sec (NOLOAD) :
- {
- //ETH_RX_BUFFER_CNT == ETH_RX_DESC_CNT (12) == (12)
- . = ABSOLUTE(0x30040000);
- *(.RxDecripSection)
- . = ABSOLUTE(0x30040120);
- *(.TxDecripSection)
- . = ABSOLUTE(0x30040200);
- *(.Rx_PoolSection)
- } >RAM_D2
Przy takiej konfiguracji można zwiększyć ETH_TX_DESC_CNT na 5 ( 5*24 => 120 (DEC) => 78 (HEX)).
Testowałem również poniższe ustawienia w pamięci z odpowiednią modyfikacją MPU.
- //----------------------------------------
- .lwip_sec (NOLOAD) :
- {
- . = ABSOLUTE(0x30040000);
- *(.RxDecripSection)
- . = ABSOLUTE(0x30040100);
- *(.TxDecripSection)
- . = ABSOLUTE(0x30040200);
- *(.Rx_PoolSection)
- } >RAM_D2
- //----------------------------------------
- .lwip_sec (NOLOAD) :
- {
- . = ABSOLUTE(0x30040000);
- *(.RxDecripSection)
- . = ABSOLUTE(0x30040080);
- *(.TxDecripSection)
- . = ABSOLUTE(0x30040100);
- *(.Rx_PoolSection)
- } >RAM_D2
- //----------------------------------------
- .lwip_sec (NOLOAD) :
- {
- . = ABSOLUTE(0x30040000);
- *(.RxDecripSection)
- . = ABSOLUTE(0x30040120);
- *(.TxDecripSection)
- . = ABSOLUTE(0x30040200);
- *(.Rx_PoolSection)
- } >RAM_D2
- //----------------------------------------
- .lwip_sec (NOLOAD) :
- {
- . = ABSOLUTE(0x30000000);
- *(.RxDecripSection)
- . = ABSOLUTE(0x30000100);
- *(.TxDecripSection)
- . = ABSOLUTE(0x30000200);
- *(.Rx_PoolSection)
- } >RAM_D2
- //----------------------------------------
Ogólnie nie ma tutaj większego znaczenia gdzie to w pamięci umieścimy. Należy pamiętać o poprawnym zarezerwowaniu miejsca, aby nie wykraczać poza buffor. Dodatkowo należy pamiętać aby bufory były dobrane odpowiednio do aplikacji. W moim przypadku wykonywałem dosyć proste testy UDP, TCP czy HTTP. Natomiast przy bardziej wymagających aplikacjach należy na te ustawienia zwrócić większą uwagę.
Pliki projektu do pobrania z dysku Google.