Project

General

Profile

Работа с Ethernet (Си) » History » Version 18

Version 17 (krufter_multiclet, 04/22/2013 12:34 PM) → Version 18/23 (krufter_multiclet, 04/22/2013 12:35 PM)

h1. Работа с Ethernet (Си)

Рассмотрим простой пример по отправке одного пакета с заданным IP-адресом в виде широковещательной рассылки.
Но сначала заметим, что настройка микросхемы, реализующей физический уровень Ethernet происходит программно.

На Си реализованы функции записи и чтения в регистры микросхемы, реализующей физический уровень Ethernet.

1) Функция записи в регистры PHY

<pre>
void mdio_wr(int phy_addr, int reg, int data)
</pre>

phy_addr - адрес микросхемы PHY (по умолчанию 0x00)
reg - номер регистра
data - значение для установки в регистр

2) Функция чтения регистра PHY

<pre>
int mdio_rd(int phy_addr, int reg)
</pre>

phy_addr - адрес микросхемы PHY (по умолчанию 0x00)
reg - номер регистра
Функция возвращает 32 разрядное значение регистра

Перейдём непосредственно к рассмотрению примеров отправки и приёма пакетов.

h2. а) Передача пакета по Ethernet

Перед передачей пакета необходимо учесть, что области размещения дескрипторов передачи и приёма должны быть
выравнены на 1 килобайт. Области памяти, содержащие данные для передачи или принятые данные должны быть
выравнены на 4 байта.

h3. 1. Объявим переменные и массивы данных с выравниванием

<pre>
static unsigned char buf_tx_desc[2048];
static unsigned char buf_rx_desc[2048];
unsigned int buf_tx[400];
unsigned int buf_rx[400];
unsigned char *eth_tx_desc;
unsigned char *eth_rx_desc;

eth_tx_desc=(unsigned char *)(((int)buf_tx_desc+1024)& ~(1024-1)); //указатель на область памяти дескрипторов передачи
eth_rx_desc=(unsigned char *)(((int)buf_rx_desc+1024)& ~(1024-1)); //указатель на область памяти дескрипторов приема
</pre>


h3. 2. Сформируем простую функцию передачи данных

<pre>
void send_pack(unsigned int *buf_tx, int size, unsigned char *eth_tx_desc)
{
int pre;
GPIOA->BPS = 0x0000FFFF; //разрешаем альтернативные функции порта для работы Ethernet
ETH->TDP = 0xE0200000 + (unsigned int)(eth_tx_desc); //задаём область размещения дескрипторов передачи
pre = 0xE0200000 + (unsigned int)buf_tx; //область размещения данных в ПД

//переворачиваем pre из big endian в little endian и формируем вторую часть дескриптора передачи
eth_tx_desc[3] = (pre & 0xFF);
eth_tx_desc[2] = ((pre & 0xFF00)>>8);
eth_tx_desc[1] = ((pre & 0xFF0000)>>16);
eth_tx_desc[0] = ((pre & 0xFF000000)>>24);

pre = (size & 0x3FF) + 0x1800; //первая часть дескриптора передачи EN, WP - "1"

eth_tx_desc[7] = (pre & 0xFF);
eth_tx_desc[6] = ((pre & 0xFF00)>>8);
eth_tx_desc[5] = 0x00;
eth_tx_desc[4] = 0x00;

ETH->CR = 0x00000001; //разрешаем передачу
while((ETH->CR & 1) != 0);
}
</pre>

3. Заполним данные на передачу

<pre>
for(i=0;i<400;i++)
{
buf_tx[i] = 0xFFFFFFFF;
}
</pre>

Первые 12 байт отводятся в пакете под МАК адреса отправителя и получателя, если заполнить
МАК адреса единицами, то получим широковещательную посылку.
В примере, который будет доступен в хранилище задаётся тип пакета IPv4 и IP адрес с данными.