Работа с Ethernet (Си) » History » Version 12
Version 11 (krufter_multiclet, 04/22/2013 12:08 PM) → Version 12/23 (krufter_multiclet, 04/22/2013 12:24 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>
2. Сформируем простую функцию передачи данных
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);
}
Рассмотрим простой пример по отправке одного пакета с заданным 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>
2. Сформируем простую функцию передачи данных
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);
}