Работа с Ethernet (Си) » History » Version 17
Version 16 (krufter_multiclet, 04/22/2013 12:32 PM) → Version 17/23 (krufter_multiclet, 04/22/2013 12:34 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>
Рассмотрим простой пример по отправке одного пакета с заданным 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>