SAM4E单片机之旅——22、GMAC和PHY的介绍与初始化
网络通信的作用不用多说,而这次进行的工作即是对以太网通信过程中,需要用到的硬件部分进行初始化,也介绍了发送和接收数据的方法。
由于较为复杂,所以使用了ASF框架。但是也会对用到的库函数的实现做一个介绍。
一、 MAC、PHY和MII
IEEE 802.3是现在常用的以太网标准,它定义了物理层(Physical Layer, PHY)和介质访问控制层(Media Access Control, MAC)的标准。另外,在OSI模型中,MAC则处于数据链路层的底层。
而在硬件实现上,M4使用的GMAC外设实现了802.3中MAC的功能。开发板携带型号为KSZ8051MNL的PHY芯片以及RJ45接口则实现了物理层的功能:
MAC和PHY之间交互的接口则是介质独立接口(Media Independent Interface,MII)。MII包含一个数据通信接口,以及一个管理接口(Management Data Input/Output ,MDIO)。由于PHY的接口是面向MAC的,所以我们需要通过MAC来对PHY进行管理及数据交互。
另外,更早制定的Ethernet II帧则是现在以太网传输中常使用的帧格式。
二、 GMAC的DMA缓冲区
GMAC使用了一个DMA接口。和M4的通用DMAC一样,它也可以自动进行多次传输,但是方式稍微有点区别。GMAC的DMA 对发送和接收使用不同的缓冲区列表,而缓冲区描述符列表是一个数组,而不是DMAC所使用的链表。数组的起始位置保存在寄存器(GMAC_RBQB、GMAC_TBQB)中,且缓冲区描述符中有一个字段(Wrap)指示其是否为数组中的最后一个描述符。如接收缓冲区:
在工作过程中,DMA会顺序访问每个缓冲区描述符,而在访问最后一个描述符时,就会重新开始遍历。
对于接收缓冲,列表中每一个缓冲区的长度是一样的,这个长度由DMA配置寄存器(GMAC_DCFGR)中的DRBS字段指定。在DMA将数据写入接收缓冲时,也会设置描述符相应的字段,以表明每帧的起始与结束;同时,也会标注相关的信息,如是否为广播帧等。
对于发送缓冲,其帧长度、是否需要添加CRC等控制信息也均在描述符中表示。在准备好数据后,向GMAC_NCR寄存器写入TSTART字段即可触发发送操作。
三、 使用ASF初始化GMAC
由于PHY是通过MAC访问的,所有在设置PHY前要完成GMAC的设置。
GMAC约有94个寄存器,其中约有40个为统计寄存器,约15个寄存器与1588和PTP相关,约15个寄存器与特殊地址和ID有关。另外,在有些状态寄存器中,需要向特定位写入1才会清除该位的状态。
使用的ASF模块为Ethernet GMAC,然后conf_eth.h中可以设置MAC地址,IP地址,子网掩码,网关以及缓冲区大小等参数。
然后调用gmac_dev_init( ) 函数即可对GMAC进行初始化:
pmc_enable_periph_clk(ID_GMAC);
// MAC地址
uint8_t mac_address[] =
{ ETHERNET_CONF_ETHADDR0, ETHERNET_CONF_ETHADDR1,
ETHERNET_CONF_ETHADDR2,ETHERNET_CONF_ETHADDR3,
ETHERNET_CONF_ETHADDR4, ETHERNET_CONF_ETHADDR5
};
// GMAC选项
gmac_options_t gmac_option;
gmac_option.uc_copy_all_frame = 0; // 不拷贝所有帧
gmac_option.uc_no_boardcast = 0; // 不忽略广播
memcpy(gmac_option.uc_mac_addr,
mac_address, sizeof(mac_address)); //拷贝MAC地址
// GMAC驱动设置
gmac_device_t gmac_dev;
gs_gmac_dev.p_hw = GMAC; // 指定GMAC寄存器基址
// 初始化GMAC
gmac_dev_init(GMAC, &gmac_dev, &gmac_option);
gmac_dev_init(Gmac* p_gmac, gmac_device_t* p_gmac_dev, gmac_options_t* p_opt) 函数完成了以下的工作:
- 禁用发送接收,禁用GMAC所有中断;清除统计寄存器,以及发送接受状态寄存器。
- 设置GMAC_NCFGR寄存器。根据p_opt,判断是否拷贝所有帧,以及是否忽略广播。同时,追加GMAC_NCFGR_PEN 和 GMAC_NCFGR_IRXFCS位。(我觉得这里应该是个BUG,从其注释判断需要追加的应该是GMAC_NCFGR_RFCS位)
- 设置好DMA缓冲,然后调用gmac_init_mem( ) 对缓冲区描述符等进行初始化。这个函数里也会使能发送和接收,同时也会启用一系列的中断。设置完成后,DMA缓冲的信息将储存在p_gmac_dev中。
- 将MAC地址写入特别地址寄存器1。
四、 PHY的地址
在MDIO通信过程中,每个PHY都会有一个4位的地址。而开发板携带的KSZ8051MNL芯片,可以在上电或复位时,根据引脚设置地址的低3位:
在开发板中,上电时PHYAD[2:0]的值为001,即其地址为0x1。
特别指出,地址0可作为该芯片的广播地址,而开发板也做了这样的配置。另外,在ASF中,错误地将PHY的地址定义成了0。这样能正确工作的原因仅是0为广播地址,而开发板只有一个PHY芯片。严谨起见,将这个地址修改为正确的值:
#ifdef BOARD_GMAC_PHY_ADDR
#undef BOARD_GMAC_PHY_ADDR
#endif
#define BOARD_GMAC_PHY_ADDR 1
五、 在ASF中使用PHY
使用的模块为Ethernet Physical Transceiver。需要在conf_board.h 中声明宏:
/* 使用 ETH PHY: KSZ8051MNL */
#define CONF_BOARD_KSZ8051MNL
初始化。
在PHY上电后,需要等待一段时间让其运行稳定。之后就可以对其进行初始化了:
if (ethernet_phy_init(GMAC, BOARD_GMAC_PHY_ADDR,sysclk_get_cpu_hz())
!= GMAC_OK) {
puts("PHY Initialize ERROR!\r");
return -1;
}在该ethernet_phy_init( ) 函数中,完成了以下工作:
- 设置MDIO的时钟MDC。
- 通过MDIO向PHY发送重置命令。
- 检查地址是否正确。检查的逻辑是先读取PHY的PHYID1的内容,再判断读出的内容是否正确。KSZ8051MNL芯片中,该寄存器的值是0x22。
- 如果地址无效的话,因为MDIO有效地址只有32个,就遍历这些地址。然后使用检查出的新地址重新发送一次重置命令。
- 如果初始化成功,则返回GMAC_OK。
自协商。
然后需要让PHY协商通信速率、双工模式:
ethernet_phy_auto_negotiate(GMAC, BOARD_GMAC_PHY_ADDR);
if (ethernet_phy_set_link(GMAC, BOARD_GMAC_PHY_ADDR, 0)
!= GMAC_OK) {
puts("Set link ERROR!\r");
return -1;
}ethernet_phy_auto_negotiate() 函数就会完成PHY的协商工作,然后根据协商的结果设置GMAC的速率、双工模式。ethernet_phy_set_link() 函数则会检查链路的状态,同时可以根据参数(第3个)应用PHY的自协商结果至GMAC中。
中断处理。
ASF的GMAC模块需要获取相关的中断,以进行相关的工作:如更新发送缓冲区描述符相关的信息,或是调用用户定义的回调函数等。
// 需要在NVIC中启用相关中断
void GMAC_Handler(void)
{
gmac_handler(&gs_gmac_dev);
}数据接收。
先准备好一个缓冲,然后就可以调用gmac_dev_read() 读取出接收到的帧的内容。
//#define GMAC_FRAME_LENTGH_MAX 1536
uint8_t eth_buffer[GMAC_FRAME_LENTGH_MAX];
uint32_t frm_size;
gmac_dev_read(&gmac_dev, (uint8_t *) eth_buffer,
sizeof(eth_buffer), &frm_size);数据发送。
gmac_dev_write(&gmac_dev, (uint8_t *)eth_buffer, frm_size, NULL);
通过该函数即可使用GMAC发送数据,第4个参数是发送完成后的回调函数。该回调函数时是在gmac_handler() 中被调用的。
SAM4E单片机之旅——22、GMAC和PHY的介绍与初始化的更多相关文章
- SAM4E单片机之旅——24、使用DSP库求向量数量积
DSP(Digital Signal Processing,数字信号处理)中会使用大量的数学运算.Cortex-M4中,配置了一些强大的部件,以提高DSP能力.同时CMSIS提供了一个DSP库,提供了 ...
- SAM4E单片机之旅——23、在AS6(GCC)中使用FPU
浮点单元(Floating Point Unit,FPU),是用于处理浮点数运算的单元. 为使用FPU,除了需要启用FPU外,还需要对编译器进行设置,以使其针对浮点运算生成特殊的指令.虽然在Atmel ...
- SAM4E单片机之旅——18、通过AFEC(ADC)获取输入的电压
很多时候,一个电压不仅仅需要定性(高电平或者低电平),而且要定量(了解具体电压的数值).这个时候就可以用到模数转换器(ADC)了.这次的内容是测量开发板搭载的滑动变阻器(VR1)的电压,然后把ADC转 ...
- SAM4E单片机之旅——13、LCD之ASF初步
在Atmel Studio 6中,集成了Atmel Software Framework(ASF框架).通过它提供的库,可以很快速地完成新的项目. 这次的最终目标使用ASF在LCD上显示出文字“Hel ...
- SAM4E单片机之旅——3、LED闪烁之定时器中断
让一个LED灯闪烁不过瘾,我们应该让这块开发板完成一点更高难度的任务:比如让两个LED灯闪烁. …… 当然了,以我们的现在使用的空循环技术,还是可以实现这点的.但是这样显得略为低端.所以我们使用一个高 ...
- SAM4E单片机之旅——2、LED闪烁之轮询定时器
之前我们使用空循环,达到了延迟的目的,但是这样子的延迟比较不精确.现在就使用实时定时器(RTT)来进行更为精确的计时.RTT虽然不是特别通用,在某些单片机上可能没有,但它较为简单. RTT内部有一个计 ...
- SAM4E单片机之旅——1、LED闪烁之空循环
最近因为导师要写一本关于SAME4单片机的书籍,而我也作为一个嵌入式的初学者看了这本书.现在也让我写写几个小的程序,做做示例.既然写了文档之类的,就发到博客上来吧. 目前关于这芯片能参考的书籍大概就只 ...
- SAM4E单片机之旅——21、DMAC之USART回显
DMAC也可以和外设进行数据交互.之前我们曾使用PDC进行USART的数据回显,这次就使用DMAC完成相同的工作.而且由于DMAC有内部的缓冲区,实现起来更为简单. 一. USART设置 因为之前已经 ...
- SAM4E单片机之旅——20、DMAC之使用Multi-buffer进行内存拷贝
这次使用这个DMAC的Multi-buffer传输功能,将两个缓冲区的内容拷贝至一个连续的缓冲区中. 一. DMAC 在M4中,DMA控制器(DMAC)比外设DMA控制器(PDC)要复杂,但是功能更加 ...
随机推荐
- Swift_3.0_开启注释功能
1. Swift_3.0 没法快捷键(command+/)注释的原因:这个是因为苹果解决xcode ghost,把插件屏蔽了. 2. 解决办法: (1) 终端输入: sudo /usr/libexec ...
- Android的init过程(二):初始化语言(init.rc)解析
Android的init过程(一) 本文使用的软件版本 Android:4.2.2 Linux内核:3.1.10 在上一篇文章中介绍了init的初始化第一阶段,也就是处理各种属性.在本文将会详细分析i ...
- 三分套三分 --- HDU 3400 Line belt
Line belt Problem's Link: http://acm.hdu.edu.cn/showproblem.php?pid=3400 Mean: 给出两条平行的线段AB, CD,然后一 ...
- web服务器之nginx与apache
最近准备架设php的web服务器,以下内容可供参考. 1.nginx相对于apache的优点: 轻量级,同样起web 服务,比apache占用更少的内存及资源 抗并发,nginx 处理请求是异步非阻塞 ...
- javascript中怎样验证密码是否含有特殊符号、数字、大小写字母,长度是否大于6小于12
今天在温习了一下以前学过的知识,一般常用验证密码是通过正则表达式或是通过ASCII 一.用AscII码来验证
- 多准则决策模型-TOPSIS评价方法-源码
? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 ...
- 中国快递包裹总量的预测-基于SARIMA模型
code{white-space: pre;} pre:not([class]) { background-color: white; }if (window.hljs && docu ...
- Ahjesus Nodejs02 使用集成开发环境
下载最新版webstorm, 选择此集成开发环境是因为支持性较好,在vs下也有插件支持,不过感觉有些牵强 附vs插件 NTVS 详细介绍 安装好以后就需要配置npm NPM 国内高速镜像 source ...
- [.NET] SQL数据总笔数查询
[.NET] SQL数据总笔数查询 程序下载 范例下载:点此下载 原始码下载:点此下载 NuGet封装:点此下载 数据查询 开发系统时,使用C#执行SQL查询指令,就可以从SQL数据库里查询所需数据. ...
- IOS6学习笔记(四)
1.GCD设置一个timer计时器 - (void)awakeFromNib { __weak id weakSelf = self; double delayInSeconds = 0.25; _t ...