RK3568平台(网络篇)添加网络交换芯片RTL8306M

一.硬件原理图

分析:
该交换芯片支持I2C、SPI、mdio通信,但是看ast1520的uboot代码采用的是mdio去通信phy芯片的,所以暂时也先采用mdio的方式,需要配置相应的引脚才可以配置成mdio通信模式,具体的配置硬件工程师解决。

背景:

RTL8306M芯片上可能没有提供MDC/MDIO接口,可以通过GPIO(General Purpose Input/Output)来模拟,GPIO可实现串行输入输出,且一般CPU上会提供很多GPIO接口供用户自定义使用。每组SMI需要两个GPIO口分别来模拟MDC和MDIO,首先需要保证这两个GPIO口不作其他用途,且相应的复用模式设置为GPIO模式。

二.SMI[MDC/MDIO]协议

SMI:串行管理接口(Serial Management Interface),通常直接被称为MDIO接口(Management Data Input/Output Interface)。MDIO最早在IEEE 802.3的第22卷定义,后来在第45卷又定义了增强版本的MDIO,其主要被应用于以太网的MAC和PHY层之间,用于MAC层器件通过读写寄存器来实现对PHY层器件的操作与管理。

MDIO接口包括两条线,MDIO和MDC,其中MDIO是双向数据线,而MDC是由STA驱动的时钟线。MDC时钟的最高速率一般为2.5MHz,MDC也可以是非固定频率,甚至可以是非周期的。MDIO接口只是会在MDC时钟的上升沿进行采样,而并不在意MDC时钟的频率(类似于I2C接口)。MDIO是一个PHY的管理接口,用来读/写PHY的寄存器,以控制PHY的行为或获取PHY的状态,MDC为MDIO提供时钟。

MAC读取PHY的寄存器:

MAC向PHY的寄存器写入数据:

三.代码实现

原厂提供了芯片RTL8306M的读写逻辑,需要自己实现gpio模拟MDC/MDIO和驱动入口。

Kconfig:

config RTL8309tristate "RTL8309 driver"default mhelpEnable this driver will support network switch control

Makefile:

# SPDX-License-Identifier: GPL-2.0-only
#
# Makefile for the Realtek network device drivers.
#obj-$(CONFIG_RTL8309)	+= rtl8309.o
rtl8309-objs := rtl8309_main.o mdcmdio.o rtk_api.o rtl8309n_asicdrv.

DTS配置:

rtl_8309: rtl-8309{status ="okay";compatible = "rtl8309";pinctrl-names = "default";pinctrl-0 = <&rtl8309_mdio_pin&rtl8309_mdc_pin>;rtl8309-mdio-gpios = <&gpio3 RK_PC3 GPIO_ACTIVE_HIGH>;rtl8309-mdc-gpios = <&gpio3 RK_PC2 GPIO_ACTIVE_HIGH>;};rtl-8309 {rtl8309_mdio_pin:rtl8309-mdio-gpios{rockchip,pins = <3 RK_PC3 RK_FUNC_GPIO &pcfg_pull_up>;};rtl8309_mdc_pin:rtl8309-mdc-gpios{rockchip,pins = <3 RK_PC2 RK_FUNC_GPIO &pcfg_pull_up>;};};

 实现一个杂散类设备RTL8306M:

#include <linux/miscdevice.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/fs.h>
#include <linux/i2c-dev.h>
#include <linux/i2c.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/notifier.h>
#include <linux/slab.h>
#include <linux/uaccess.h>
#include <linux/compat.h>
#include <linux/printk.h>
#include <linux/kobject.h>
#include <linux/version.h>
#include <linux/kthread.h>
#include <linux/gpio.h>
#include <linux/of_gpio.h>
#include <linux/of.h>
#include <linux/of_device.h>
#include <linux/platform_device.h>#include "rtk_api.h"
#include "rtl8309_main.h"#include <linux/delay.h>#define DRIVER_NAME "RTL8309"#define RTL8309_DEBUG	1
#if RTL8309_DEBUG
#define DBG(format, args...)					\printk(KERN_DEBUG "%s: " format, DRIVER_NAME, ##args)
#define ERR(format, args...)					\printk(KERN_ERR "%s: " format, DRIVER_NAME, ##args)
#define WARNING(format, args...)				\printk(KERN_WARN "%s: " format, DRIVER_NAME, ##args)
#define INFO(format, args...)					\printk(KERN_INFO "%s: " format, DRIVER_NAME, ##args)
#else
#define DBG(format, args...)
#define ERR(format, args...)
#define WARNING(format, args...)
#define INFO(format, args...)
#endif#define LINK_UP			1		
#define LINK_DOWN		2
#define LINK_UNKNOW		3int PRE_PORT_STATUS[5]={0,0,0,0,0};struct rtl8309_dev {struct device *dev;struct device sys_dev;struct gpio_desc *rtl8309_mdio;struct gpio_desc *rtl8309_mdc;struct delayed_work check_status_work;struct mutex status_lock;u32 link_status;
};struct rtl8309_dev *g_rtl8309;
struct rtl8309_dev *rtl8309;void setGpioDirection(int gpio, uint32_t  dir)
{if(gpio == IST_GPIO_RTL8309_MDIO){if(dir){gpiod_direction_output(g_rtl8309->rtl8309_mdio,1);}else{gpiod_direction_input(g_rtl8309->rtl8309_mdio);}}else if(gpio == IST_GPIO_RTL8309_MDC){if(dir){gpiod_direction_output(g_rtl8309->rtl8309_mdc,1);}else{gpiod_direction_input(g_rtl8309->rtl8309_mdc);}}
}void setGpioOutput(int gpio, uint32_t out )
{if(gpio == IST_GPIO_RTL8309_MDIO){gpiod_set_value(g_rtl8309->rtl8309_mdio, out);}else if(gpio == IST_GPIO_RTL8309_MDC){gpiod_set_value(g_rtl8309->rtl8309_mdc, out);}
}void getGpioInput(int gpio,uint32_t *in)
{if(gpio == IST_GPIO_RTL8309_MDIO){*in = gpiod_get_value(g_rtl8309->rtl8309_mdio);}else if(gpio == IST_GPIO_RTL8309_MDC){*in = gpiod_get_value(g_rtl8309->rtl8309_mdc);}
}static long rtl8309_ioctl(struct file *file, uint32_t cmd, unsigned long arg)
{return 0;
}static ssize_t rtl8309_write(struct file *file, const char __user *buf,size_t size, loff_t *ppos)
{return 1;
}static ssize_t rtl8309_read(struct file *file, char __user *buf, size_t size,loff_t *ppos)
{return 1;
}static void check_link_status(struct work_struct *work)
{struct delayed_work *dwork = to_delayed_work(work);struct rtl8309_dev *rtl8309 =container_of(dwork, struct rtl8309_dev, check_status_work);int PORT_ENABLE[5]={0,1,0,1,0};int PORT_STATUS[5]={0,0,0,0,0};int status = 0;int i = 0;rtk_port_linkStatus_t pLinkStatus = PORT_LINKDOWN;rtk_port_speed_t pSpeed = PORT_SPEED_10M;rtk_port_duplex_t pDuplex = PORT_HALF_DUPLEX;mutex_lock(&rtl8309->status_lock);for(i=0;i<5;i++){if(PORT_ENABLE[i] == 1){if(rtk_port_phyStatus_get((rtk_port_t)i, &pLinkStatus, &pSpeed, &pDuplex) !=0 ){printk("rtk_port_phyStatus_get error:%d",i);break;}printk("GetEthStatus, port= %d, pLinkStatus= %d, pSpeed= %d, pDuplex= %d \n",i, pLinkStatus, pSpeed, pDuplex);if(pLinkStatus == PORT_LINKUP){PORT_STATUS[i] = 1;printk("port_status\n");if(PRE_PORT_STATUS[i] != PORT_STATUS[i]){printk("port[%d] status changed, set led mode\n",i);if(pSpeed == PORT_SPEED_100M){printk("port_speed\n");}else{printk("port_speed error\n");}}}else{PORT_STATUS[i] = 0;printk("port_status\n");}PRE_PORT_STATUS[i] = PORT_STATUS[i];msleep(10);}}mutex_unlock(&rtl8309->status_lock);for(i=0;i<5;i++){status = status | PORT_STATUS[i];}if(status == 0)rtl8309->link_status =  LINK_DOWN;else if(status == 1)rtl8309->link_status =  LINK_UP;else rtl8309->link_status =  LINK_UNKNOW;schedule_delayed_work(&rtl8309->check_status_work, msecs_to_jiffies(5000));
}static ssize_t rtl8309_status_read(struct device *dev,struct device_attribute *attr, char *buf)
{struct rtl8309_dev *rtl8309 = g_rtl8309;printk("rtl8309_status_read rtl8309_status_read:%d\n",rtl8309->link_status);return sprintf(buf, "%d\n", rtl8309->link_status);
}static ssize_t rtl8309_status_write(struct device *dev,struct device_attribute *attr,const char *buf, size_t count)
{printk("rtl8309_status_write rtl8309_status_write");return count;
}static DEVICE_ATTR(linkstatus, 0644,rtl8309_status_read, rtl8309_status_write);static const struct file_operations rtl8309_fops = {.owner = THIS_MODULE,.read = rtl8309_read,.write = rtl8309_write,.unlocked_ioctl = rtl8309_ioctl,
};struct miscdevice rtl8309_miscdev = {.minor = MISC_DYNAMIC_MINOR,.name = "rtl8309_dev",.fops = &rtl8309_fops,
};static int rtl8309_probe(struct platform_device *pdev)
{printk("rtl8309_probe");struct rtl8309_dev *rtl8309;int ret = 0;rtl8309 = devm_kzalloc(&pdev->dev, sizeof(*rtl8309), GFP_KERNEL);if (!rtl8309)return -ENOMEM;rtl8309->dev = &pdev->dev;rtl8309->rtl8309_mdio = devm_gpiod_get_optional(rtl8309->dev,"rtl8309-mdio", GPIOD_OUT_HIGH);if (IS_ERR(rtl8309->rtl8309_mdio)) {printk("Could not get rtl8367-mdio");rtl8309->rtl8309_mdio = NULL;}rtl8309->rtl8309_mdc = devm_gpiod_get_optional(rtl8309->dev,"rtl8309-mdc", GPIOD_OUT_HIGH);if (IS_ERR(rtl8309->rtl8309_mdc)) {printk("Could not get rtl8367-mdc ");rtl8309->rtl8309_mdc = NULL;}g_rtl8309 = rtl8309;ret = misc_register(&rtl8309_miscdev);if (ret) {ERR("rtl8309_miscdev ERROR: could not register rtl8309_miscdev device\n");return ret;}ret = device_create_file(rtl8309_miscdev.this_device,&dev_attr_linkstatus);if (ret) {printk("failed to create attr linkstatus");return ret;}mutex_init(&rtl8309->status_lock);INIT_DELAYED_WORK(&rtl8309->check_status_work, check_link_status);schedule_delayed_work(&rtl8309->check_status_work, msecs_to_jiffies(5000));return 0;
}static int rtl8309_remove(struct platform_device *client)
{return 0;
}static const struct of_device_id rtl8309_of_match[] = {{ .compatible = "rtl8309" },{}
};
MODULE_DEVICE_TABLE(of, rtl8309_of_match);static struct platform_driver rtl8309_driver = {.probe = rtl8309_probe,.remove = rtl8309_remove,.driver = {.owner = THIS_MODULE,.name = DRIVER_NAME,.of_match_table = of_match_ptr(rtl8309_of_match),},
};static int __init rtl8309_init(void)
{int ret = platform_driver_register(&rtl8309_driver);if (ret != 0) {pr_err("Failed to register example driver: %d\n", ret);return ret;}pr_info("Example driver initialized\n");return 0;
}static void __exit rtl8309_exit(void)
{platform_driver_unregister(&rtl8309_driver);pr_info("Example driver exited\n");
}module_init(rtl8309_init);
module_exit(rtl8309_exit);MODULE_DESCRIPTION("rtl8309 GPIO Switch");
MODULE_AUTHOR("Zewei Ye <yezw@ist.com.hk>");
MODULE_LICENSE("GPL v2");

函数说明:rtk_port_phyStatus_get为原厂提供的读取RTL8306M芯片状态的接口,根据返回的状态判断是否有百兆网。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/282291.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux 基本语句_16_Udp网络聊天室

代码&#xff1a; 服务端代码&#xff1a; #include <stdio.h> #include <arpa/inet.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <stdlib.h> #include <unistd.h> #include <string…

探索 Coinbase 二层链 Base 的潜力与风险

作者&#xff1a;lesleyfootprint.network 在不断变化的加密货币领域&#xff0c;Coinbase 已经确立了自己领先中心化交易所&#xff08;CEX&#xff09;的地位。然而&#xff0c;Coinbase 坚信去中心化是创造一个开放、全球范围内对每个人都可访问的加密经济的关键&#xff0…

从0到1:志愿者小程序开发心得

背景调研 志愿者服务小程序可以使志愿服务更加有序和高效&#xff0c;方便志愿者参与&#xff0c;也方便组织方管理和组织志愿服务活动;志愿者可以方便的在手机上报名&#xff0c;查看记录&#xff0c;获取积分&#xff0c;了解积分排行&#xff1b; 同时提供积分商城模块&…

用23种设计模式打造一个cocos creator的游戏框架----(十九)备忘录模式

1、模式标准 模式名称&#xff1a;备忘录模式 模式分类&#xff1a;行为型 模式意图&#xff1a;在不破坏封装性的前提下捕获一个对象的内部状态&#xff0c;并在对象之外保存这个状态。这样以后就可以将对象恢复到原先保存的状态 结构图&#xff1a; 适用于&#xff1a; …

Freemarker基本语法与案例讲解

&#x1f389;&#x1f389;欢迎来到我的CSDN主页&#xff01;&#x1f389;&#x1f389; &#x1f3c5;我是Java方文山&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; &#x1f31f;推荐给大家我的专栏《SpringBoot》。&#x1f3af;&#x1f3af; &…

Azure Machine Learning - 提示工程简介

OpenAI的GPT-3、GPT-3.5和GPT-4模型基于用户输入的文本提示工作。有效的提示构造是使用这些模型的关键技能&#xff0c;涉及到配置模型权重以执行特定任务。这不仅是技术操作&#xff0c;更像是一种艺术&#xff0c;需要经验和直觉。本文旨在介绍适用于所有GPT模型的提示概念和…

网络(十)ACL和NAT

前言 网络管理在生产环境和生活中&#xff0c;如何实现拒绝不希望的访问连接&#xff0c;同时又要允许正常的访问连接&#xff1f;当下公网地址消耗殆尽&#xff0c;且公网IP地址费用昂贵&#xff0c;企业访问Internet全部使用公网IP地址不够现实&#xff0c;如何让私网地址也…

前端面试(5)

1、移动端适配 1.1、设置meta缩放比例&#xff0c;将设备窗口调整为设计图大小。 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width,initial-sc…

基于ssm的简单学校课程管理系统的设计与实现(源码+调试)

项目描述 临近学期结束&#xff0c;还是毕业设计&#xff0c;你还在做java程序网络编程&#xff0c;期末作业&#xff0c;老师的作业要求觉得大了吗?不知道毕业设计该怎么办?网页功能的数量是否太多?没有合适的类型或系统?等等。今天给大家介绍一篇基于ssm的简单学校课程管…

C++之程序生成

一、C的发展史 截止到2023年12月&#xff0c;C已经更新了很多版本&#xff0c;并在每个版本中修复了bug和添加了新的特性&#xff0c;ISO C委员会每三年会对C进行一次更新&#xff1a; C98&#xff1a;于1998年发布&#xff0c;是最早的国际标准化版本。它包含了面向对象编程…

可替代LM5145,5.5V-100V Vin同步降压控制器_SCT82A30

SCT82A30是一款100V电压模式控制同步降压控制器&#xff0c;具有线路前馈。40ns受控高压侧MOSFET的最小导通时间支持高转换比&#xff0c;实现从48V输入到低压轨的直接降压转换&#xff0c;降低了系统复杂性和解决方案成本。如果需要&#xff0c;在低至6V的输入电压下降期间&am…

GMP原理与调度

GMP是Go语言运行时(runtime)层面的实现,是go语言自己实现的一套调度系统,区别于操作系统调度OS线程 Golang"调度器"的由来 单进程时代不需要调度器 我们知道,一切的软件都是跑在操作系统上,真正用来干活(计算)的是CPU,早期的操作系统每个程序就是一个进程,直到一个…