热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

STM32(三)ENC28J60以太网(二)

3寄存器操作实现ENC28j60的寄存器操作分为222部分,分别为写寄存器和读寄存器部分,读缓冲区和写缓冲区部分,写PHY寄存器和读PH

3 寄存器操作实现

ENC28j60的寄存器操作分为2+2+2部分,分别为写寄存器和读寄存器部分,读缓冲区和写缓冲区部分,写PHY寄存器和读PHY寄存器部分。

3.1 读写寄存器

读或写寄存器的函数如下:

unsigned char enc28j60Read(unsigned char address)
{/* 设定寄存器地址区域 */enc28j60SetBank(address);/* 读取寄存器值 发送读寄存器命令和地址 */return enc28j60ReadOp(ENC28J60_READ_CTRL_REG, address);
}
void enc28j60Write(unsigned char address, unsigned char data)
{/* 设定寄存器地址区域 */enc28j60SetBank(address);/* 写寄存器值 发送写寄存器命令和地址 */enc28j60WriteOp(ENC28J60_WRITE_CTRL_REG, address, data);
}

读写寄存器的分为两步,第一步为选定寄存器的BANK编号,第二步使用写命令或读命令,操作指定地址的寄存器。在ENC28J60中,由ECON1中的低两位(BIT1-BIT1)保存BANK编号,ECON1是比较特殊的控制寄存器, 4个BANK均具有该寄存器且该寄存器的地址相同。Enc28j60Bank为全局变量,用于保存当前的BANK编号,如果两次操作控制寄存器在同一个BANK时,该变量保持不变,若两次操作的控制寄存器位于不同的BANK,那么BANK的值会变为新的BANK编号。

void enc28j60SetBank(unsigned char address)
{/* 计算本次寄存器地址在存取区域的位置 */if((address & BANK_MASK) != Enc28j60Bank){/* 清除ECON1的BSEL1 BSEL0 详见数据手册15页 */enc28j60WriteOp(ENC28J60_BIT_FIELD_CLR, ECON1, (ECON1_BSEL1|ECON1_BSEL0));/* 请注意寄存器地址的宏定义,bit6 bit5代码寄存器存储区域位置 */enc28j60WriteOp(ENC28J60_BIT_FIELD_SET, ECON1, (address & BANK_MASK)>>5);/* 重新确定当前寄存器存储区域 */Enc28j60Bank = (address & BANK_MASK);}
}unsigned char enc28j60ReadOp(unsigned char op, unsigned char address)
{unsigned char dat = 0;/* CS拉低 使能ENC28J60 */ENC28J60_CSL();/* 操作码和地址 */dat = op | (address & ADDR_MASK);/* 通过SPI写数据*/spi_sendbyte(dat);/* 通过SPI读出数据 */dat = spi_sendbyte(0xFF);/* 如果是MAC和MII寄存器,第一个读取的字节无效,该信息包含在地址的最高位 */if(address & 0x80){/* 再次通过SPI读取数据 */dat = spi_sendbyte(0xFF);}/* CS拉高 禁止ENC28J60 */ENC28J60_CSH();/* 返回数据 */return dat;
}

读控制寄存器实际上就是严格遵守数据手册的操作要求。由于读MAC和MII寄存器时,第一个接收到的字节为无效字节,第二个字节才为有效字节。程序通过寄存器地址的最高位来判断是否为MAC或MII寄存器。写寄存器函数较为简单,第一次字节包括操作码和寄存器地址,第二个字节为数据。在这两个函数中参数op为ENC28J60的指令,或称之为操作码,该指令占据SPI首字节的前3位,参数address为寄存器地址,参数data为寄存器的具体值。
ENC28J60_CSL()和ENC28J60_CSH()为操作CS端口的操作宏,而spi_sendbyte()可通过SPI发送一个字节。修改这些函数即可在其他平台上操作ENC28J60。不过请特别注意,在使用其他开发板时由于SPI总线上可能挂载多个设备,单独使用ENC28J60时需要把其他设备的CS端口拉高,或安装一个上拉电阻。

unsigned char enc28j60ReadOp(unsigned char op, unsigned char address)
{unsigned char dat = 0;/* CS拉低 使能ENC28J60 */ENC28J60_CSL();/* 操作码和地址 */dat = op | (address & ADDR_MASK);/* 通过SPI写数据*/spi_sendbyte(dat);/* 通过SPI读出数据 */dat = spi_sendbyte(0xFF);/* 如果是MAC和MII寄存器,第一个读取的字节无效,该信息包含在地址的最高位 */if(address & 0x80){/* 再次通过SPI读取数据 */dat = spi_sendbyte(0xFF);}/* CS拉高 禁止ENC28J60 */ENC28J60_CSH();/* 返回数据 */return dat;
}
void enc28j60WriteOp(unsigned char op, unsigned char address, unsigned char data)
{unsigned char dat = 0;/* 使能ENC28J60 */ ENC28J60_CSL(); /* 通过SPI发送 操作码和寄存器地址 */ dat = op | (address & ADDR_MASK);/* 通过SPI1发送数据 */spi_sendbyte(dat);/* 准备寄存器数值 */ dat = data;/* 通过SPI发送数据 */spi_sendbyte(dat);/* 禁止ENC28J60 */ ENC28J60_CSH();
}

 

3.2 读写缓冲区

读写缓冲区的操作也易于理解的。需要说明的是,两个函数具有相同的输入参数,参数len代表被操作数据的长度,pdata为被操作数据的指针。和寄存器读写函数相似,发送或接收数据之前需要发送特定的操作码。

void enc28j60ReadBuffer(unsigned int len, unsigned char* pdata)
{/* 使能ENC28J60 */ENC28J60_CSL();/* 通过SPI发送读取缓冲区命令*/spi_sendbyte(ENC28J60_READ_BUF_MEM);/* 循环读取 */while(len){len--;/* 读取数据 */*pdata = (unsigned char)spi_sendbyte(0);/* 地址指针累加 */pdata++;}/* 禁止ENC28J60 */ENC28J60_CSH();
}
void enc28j60WriteBuffer(unsigned int len, unsigned char* pdata)
{/* 使能ENC28J60 */ENC28J60_CSL();/* 通过SPI发送写取缓冲区命令*/spi_sendbyte(ENC28J60_WRITE_BUF_MEM);/* 循环发送 */while(len){len--;/* 发送数据 */spi_sendbyte(*pdata);/* 地址指针累加 */pdata++;}/* 禁止ENC28J60 */ENC28J60_CSH();
}

3.3 读写PHY寄存器

PHY寄存器和被ENC28J60控制的LED指示灯有关,控制该寄存器可以控制LED驱动方式和发生相应事件时LED显示方式。一般情况下,一个LED指示灯用于指示网络状态(常亮可理解为网络接通),另一个LED指示灯显示接收活动,有数据输入时产生一个点亮脉冲。PHY是比较特殊的寄存器,先要想一个控制寄存器写入PHY寄存器的地址,再向两个控制寄存器依次写入PHY寄存器的具体数据的高8位和低8位,最后等待PHY寄存器操作完成。

void enc28j60PhyWrite(unsigned char address, unsigned int data)
{/* 向MIREGADR写入地址 详见数据手册19页*/enc28j60Write(MIREGADR, address);/* 写入低8位数据 */enc28j60Write(MIWRL, data);/* 写入高8位数据 */enc28j60Write(MIWRH, data>>8);/* 等待PHY寄存器写入完成 */while(enc28j60Read(MISTAT) & MISTAT_BUSY);
}

转自 https://blog.csdn.net/xukai871105/article/details/14092167


推荐阅读
  • 深入理解Redis的数据结构与对象系统
    本文详细探讨了Redis中的数据结构和对象系统的实现,包括字符串、列表、集合、哈希表和有序集合等五种核心对象类型,以及它们所使用的底层数据结构。通过分析源码和相关文献,帮助读者更好地理解Redis的设计原理。 ... [详细]
  • 本教程涵盖OpenGL基础操作及直线光栅化技术,包括点的绘制、简单图形绘制、直线绘制以及DDA和中点画线算法。通过逐步实践,帮助读者掌握OpenGL的基本使用方法。 ... [详细]
  • 本文深入探讨了HTTP请求和响应对象的使用,详细介绍了如何通过响应对象向客户端发送数据、处理中文乱码问题以及常见的HTTP状态码。此外,还涵盖了文件下载、请求重定向、请求转发等高级功能。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • Splay Tree 区间操作优化
    本文详细介绍了使用Splay Tree进行区间操作的实现方法,包括插入、删除、修改、翻转和求和等操作。通过这些操作,可以高效地处理动态序列问题,并且代码实现具有一定的挑战性,有助于编程能力的提升。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 本文详细探讨了VxWorks操作系统中双向链表和环形缓冲区的实现原理及使用方法,通过具体示例代码加深理解。 ... [详细]
  • 基于KVM的SRIOV直通配置及性能测试
    SRIOV介绍、VF直通配置,以及包转发率性能测试小慢哥的原创文章,欢迎转载目录?1.SRIOV介绍?2.环境说明?3.开启SRIOV?4.生成VF?5.VF ... [详细]
  • 本文探讨了如何在模运算下高效计算组合数C(n, m),并详细介绍了乘法逆元的应用。通过扩展欧几里得算法求解乘法逆元,从而实现除法取余的计算。 ... [详细]
  • Codeforces Round #566 (Div. 2) A~F个人题解
    Dashboard-CodeforcesRound#566(Div.2)-CodeforcesA.FillingShapes题意:给你一个的表格,你 ... [详细]
  • MySQL DateTime 类型数据处理及.0 尾数去除方法
    本文介绍如何在 MySQL 中处理 DateTime 类型的数据,并解决获取数据时出现的.0尾数问题。同时,探讨了不同场景下的解决方案,确保数据格式的一致性和准确性。 ... [详细]
  • Scala与Java的数据类型对比及特性
    本文将深入探讨Scala和Java在数据类型上的差异与相似之处,重点介绍两种语言的基本类型、引用类型及其包装类,并分析它们在面向对象编程中的表现。 ... [详细]
  • 开发笔记:2020 BJDCTF Re encode
    开发笔记:2020 BJDCTF Re encode ... [详细]
  • 在软件开发过程中,MD5加密是一种常见的数据保护手段。本文将详细介绍如何在C#中使用两种不同的方式来实现MD5加密:字符串加密和流加密。 ... [详细]
  • 对象自省自省在计算机编程领域里,是指在运行时判断一个对象的类型和能力。dir能够返回一个列表,列举了一个对象所拥有的属性和方法。my_list[ ... [详细]
author-avatar
低调的小男2502928607
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有