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

 
 
> Важно ! uC-TCPIP bug @ SAM7X port, Некорректная обработка пакетов
_dem
сообщение Dec 27 2007, 18:47
Сообщение #1


Местный
***

Группа: Свой
Сообщений: 263
Регистрация: 2-02-07
Из: CN, Ukraine
Пользователь №: 24 970



uCOS/II v 2.85
ucTCPIP v 1.86
MCU - AT91SAM7X256 @ 43Mhz + RTL8201BL, порт ucTCPIP от micrium.com - свежий по состоянию на 27.12.2007

Долго отлавливал следующий баг в порте uTCP-IP на SAM7X -
в случае, если на плату приходит пара пакетов из Ethernet с малым расстоянием между ними (менее 1 мс), возможно "зависание" стека до получения следующего пакета из сети.

Картина выглядит так - приходит подряд два пакета, на каждый из которых плата должна ответить.
Плата отвечает на первый пакет, потом сетевая подсистема "зависает" (API работает, но обмена нет) до прихода еще одного пакета (неважно какого, например ARP).

Вместо
Код
---------- time ------------------------------->
[ request1 ][ request2 ] [ reply1][ reply2 ] --<time>------[ request3 ][ reply3 ]
имеем
Код
---------- time ------------------------------->
[ request1 ][ request2 ] [ reply1] ----<time>------[ any ethernet packet ][ reply2 ]


Если подряд принимаются три или более пакетов, проблема не возникает, т.к. прерывание по третьему пакету обрабатывает второй и третий.

Проблема связана с тем, что за время обработки пакета в прерывании успевает придти еще один пакет, повторно же обработка не запускается (прерывание срабатывает, но флаги EMAC_RSR корректно не установлены, т.к. были сброшены при выходе из прерывания на прошлом пакете) - код не в определяет причину прерывания.
Код
---------- time ------------------------------->
[ packet1 ][ packet 2]
[irq1]-----[irq2]-------
[-----irq1_processing:packet ready-----][----irq2_processing:reason unknown----]


Проблема решается следующим образом :
В файле net_nic.c вместо кода
Код
    n_new = NIC_GetNRdy() - NIC_RxNRdyCtr;                              /* Determine how many NEW packets have been received    */
    
    while (n_new > 0) {


нужен код примерно такой :
Код
    
while ((n_new = NIC_GetNRdy() - NIC_RxNRdyCtr) > 0) {


Это позволяет отловить вышеописанную ситуацию и отреагировать на второй пакет.
Go to the top of the page
 
+Quote Post
 
Start new topic
Ответов
_dem
сообщение Dec 28 2007, 06:43
Сообщение #2


Местный
***

Группа: Свой
Сообщений: 263
Регистрация: 2-02-07
Из: CN, Ukraine
Пользователь №: 24 970



Цитата
идеология неправильная imho. чем обрабатывать пакет в прерывании, надо бы просто потушить флаг в RSR и взвести семафор чтобы RX таск начал вынимать и обрабатывать пакеты.


идеология - в общем - да, неправильная. Но учитывая то, что у ARMа прерывания - приоритезируемые и прерывание EMAC имеет низкий приоритет, вполне допустимая.
Понятно, что неплохо бы переделать код, но если в готовом девайсе вдруг возникает такой вот баг - sad.gif увы, этим никто не будет заниматся.

ps Тем более, что проблема возникает при малом расстоянии между пакетами - менее 1 msec ( 50-100 usec по данным ethereal). Вполне возможно, что и "короткий" обработчик за это время не успеет выполнится.

Сообщение отредактировал _dem - Dec 28 2007, 07:02
Go to the top of the page
 
+Quote Post



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

 


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


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