打印
[单片机芯片]

【CH32V317W-R0开发板】以太网测评之DHCP

[复制链接]
139|1
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
本帖最后由 abner_ma 于 2025-7-31 17:52 编辑

     CH32V *单片机是基于青稞 RISC-V 内核设计的工业级通用微控制器。青稞(C-SKY)RISC-V4F 内核是由中天微(后被阿里巴巴收购,并入平头哥半导体)开发的基于RISC-V架构的嵌入式处理器内核,支持单精度浮点运算(F扩展),主要面向IoT、边缘计算和低功耗嵌入式市场。包括 CH32V305 连接型 MCU,CH32V307/CH32V317 互联型 MCU、CH32V208 无线型 MCU 等。CH32V30x 和 CH32V31x 系列基于青稞 V4F微处理器设计,支持单精度浮点指令和快速中断响应,支持 144MHz 主频零等待运行,提供 8 组串口、4 组电机 PWM 高级定时器、SDIO、DVP 数字图像接口、4 组模拟运放、双 ADC 单元、双 DAC 单元,内置USB2.0 高速 PHY 收发器(480Mbps)、千兆以太网 MAC 控制器及 10 兆物理层收发器、10/100 兆物理层收发器(仅适用于 CH32V317)等。
    CH32V317 支持用户选择字配置为(192K FLASH+128K SRAM)、(224K FLASH+96K SRAM)、(256KFLASH+64K SRAM)、(288K FLASH+32K SRAM)、(128K FLASH+192K SRAM)几种组合中的一种。FLASH 闪存表示的是零等待运行区域 R0WAIT,CH32V317 支持(480K-R0WAIT)字节的非零等待区域。
   串口提供了 3 组通用同步/异步串口收发器(USART1、USART2、USART3),以及 5 组通用异步收发器(UART4、UART5、UART6、UART7、UART8)。支持全双工异步通信、同步单向通信以及半双工单线通信,也支持 LIN(局部互连网),兼容 ISO7816 的智能卡协议和 IrDA SIR ENDEC 传输编解码规范,以及调制解调器(CTS/RTS 硬件流控)操作。还允许多处理器通信。其采用分数波特率发生器系统,并支持 DMA 操作连续通讯。       CH32V3xx 基于 RISC-V 指令集设计,其架构中将内核、仲裁单元、DMA 模块、SRAM 存储等部分通过多组总线实现交互。设计中集成通用 DMA 控制器以减轻 CPU 负担、提高访问效率,应用多级时钟管理机制降低了外设的运行功耗,同时兼有数据保护机制,时钟自动切换保护等措施增加了系统稳定性。
    RISC-V4F 支持 RISC-V 指令集 IMAFC 子集,增加了单精度浮点运算。处理器内部以模块化管理,包含快速可编程中断控制器(PFIC)、内存保护、分支预测模式、扩展指令支持等单元。对外多组总线与外部单元模块相连,实现外部功能模块和内核的交互。青稞微处理器以其极简指令集、多种工作模式、模块化定制扩展等特点可以灵活应用不同场景微控制器设计,例如小面积低功耗嵌入式场景、高性能应用操作系统场景等。
1.支持机器和用户特权模式
2. 快速可编程中断控制器(PFIC)
3.多级硬件中断堆栈
4..串行2线调试接口
5.标准内存保护设计
6. 静态或动态分支预测、高效跳转、冲突检测机制
7. 自定义扩展指

总体架构框图。

   
    以太网收发器(ETH)
     CH32V307、CH32F207 芯片的 MAC 支持 MII/RMII/RGMII 接口且内置一个 10Mbps PHY;CH32V317 芯片支持以太网控制器 MAC 和内置 10Mbps/100Mbps PHY。以太网 MAC 通过 SMI 接口控制 PHY,接口的时序由 MAC 自动实现,无需用户通过软件生成。RGMII接口支持发送时钟相位翻转和相对数据延迟,最大延迟 4 纳秒。以太网 MAC 支持标准 IEEE802.3 协议的以太网,支持魔法帧和特定唤醒帧。以太网收发器搭配的DMA 控制器以描述符的形式进行数据的收发管理和内存搬运,描述符的数量由用户根据沟通密集程度自行确定,DMA 控制器能以 32 位宽的速度向描述符指定的内存空间写入接收到的数据或从取出将发送的数据。此外,以太网收发器的 MAC 还支持 IEEE1588 精确时间协议。   CH32V3X7作为一款集成RISC-V内核的微控制器,其以太网模块支持动态主机配置协议(DHCP),可实现自动获取IP地址、子网掩码、网关等网络参数,简化网络设备配置流程。沁恒官方提供WCHNET协议栈库,封装了DHCP客户端功能,用户仅需调用WCHNET_DHCPStart()函数即可启动DHCP服务,无需深入理解协议细节。  
   

   

#ifndef __WCHNET_H__
#define __WCHNET_H__

#include "stdint.h"
#ifndef NET_LIB
#include "net_config.h"
#endif
#ifdef __cplusplus
extern "C" {
#endif

#define WCHNET_LIB_VER                  0x11 /* 库版本号 */
#define WCHNET_CFG_VALID                0x12345678 /* 配置值有效标志 */

/* LED状态 @LED_STAT */
#define  LED_ON                         0
#define  LED_OFF                        1

/* PHY 状态 @PHY_STAT */
#define PHY_LINK_SUCCESS                (1 << 2) /* PHY建立连接 */
#define PHY_AUTO_SUCCESS                (1 << 5) /* PHY自动协商完成 */

/* 库初始化状态 @CFG_INIT_STAT */
#define  INIT_OK                        0x00
#define  INIT_ERR_RX_BUF_SIZE           0x01
#define  INIT_ERR_TCP_MSS               0x02
#define  INIT_ERR_HEAP_SIZE             0x03
#define  INIT_ERR_ARP_TABLE_NEM         0x04
#define  INIT_ERR_MISC_CONFIG0          0x05
#define  INIT_ERR_MISC_CONFIG1          0x06
#define  INIT_ERR_FUNC_SEND             0x09
#define  INIT_ERR_CHECK_VALID           0xFF

/* Socket 工作模式定义,协议类型 */
#define PROTO_TYPE_IP_RAW               0     /* IP层原始数据 */
#define PROTO_TYPE_UDP                  2     /* UDP协议类型 */
#define PROTO_TYPE_TCP                  3     /* TCP协议类型 */

/* 中断状态 */
/* 以下为 GLOB_INT 会产生的状态 */
#define GINT_STAT_UNREACH               (1 << 0) /* 不可达中断*/
#define GINT_STAT_IP_CONFLI             (1 << 1) /* IP冲突*/
#define GINT_STAT_PHY_CHANGE            (1 << 2) /* PHY状态改变 */
#define GINT_STAT_SOCKET                (1 << 4) /* socket 产生中断 */

/*以下为 Sn_INT 会产生的状态*/
#define SINT_STAT_RECV                  (1 << 2) /* socket端口接收到数据或者接收缓冲区不为空 */
#define SINT_STAT_CONNECT               (1 << 3) /* 连接成功,TCP模式下产生此中断 */
#define SINT_STAT_DISCONNECT            (1 << 4) /* 连接断开,TCP模式下产生此中断 */
#define SINT_STAT_TIM_OUT               (1 << 6) /* ARP和TCP模式下会发生此中断 */


/* Definitions for error constants. @ERR_T */
#define ERR_T
#define WCHNET_ERR_SUCCESS              0x00  /* No error, everything OK */
#define WCHNET_ERR_BUSY                 0x10  /* busy */
#define WCHNET_ERR_MEM                  0x11  /* Out of memory error */
#define WCHNET_ERR_BUF                  0x12  /* Buffer error */
#define WCHNET_ERR_TIMEOUT              0x13  /* Timeout */
#define WCHNET_ERR_RTE                  0x14  /* Routing problem */
#define WCHNET_ERR_ABRT                 0x15  /* Connection aborted */
#define WCHNET_ERR_RST                  0x16  /* Connection reset */
#define WCHNET_ERR_CLSD                 0x17  /* Connection closed */
#define WCHNET_ERR_CONN                 0x18  /* Not connected */
#define WCHNET_ERR_VAL                  0x19  /* Illegal value */
#define WCHNET_ERR_ARG                  0x1a  /* Illegal argument */
#define WCHNET_ERR_USE                  0x1b  /* Address in use */
#define WCHNET_ERR_IF                   0x1c  /* Low-level netif error  */
#define WCHNET_ERR_ISCONN               0x1d  /* Already connected */
#define WCHNET_ERR_INPROGRESS           0x1e  /* Operation in progress */
#define WCHNET_ERR_SOCKET_MEM           0X20  /* Socket information error */
#define WCHNET_ERR_UNSUPPORT_PROTO      0X21  /* unsupported protocol type */
#define WCHNET_RET_ABORT                0x5F  /* command process fail */
#define WCHNET_ERR_UNKNOW               0xFA  /* unknow */

/* 不可达代码 */
#define UNREACH_CODE_HOST               0     /* 主机不可达 */
#define UNREACH_CODE_NET                1     /* 网络不可达 */
#define UNREACH_CODE_PROTOCOL           2     /* 协议不可达 */
#define UNREACH_CODE_PROT               3     /* 端口不可达 */
/*其他值请参考RFC792文档*/

/* TCP关闭参数 */
#define TCP_CLOSE_NORMAL                0     /* 正常关闭,进行4此握手 */
#define TCP_CLOSE_RST                   1     /* 复位连接,并关闭  */
#define TCP_CLOSE_ABANDON               2     /* 内部丢弃连接,不会发送任何终止报文 */

/* socket状态 */
#define SOCK_STAT_CLOSED                0X00  /* socket关闭 */
#define SOCK_STAT_OPEN                  0X05  /* socket打开 */

/* TCP状态 */
#define TCP_CLOSED                      0     /* TCP关闭 */
#define TCP_LISTEN                      1     /* TCP监听 */
#define TCP_SYN_SENT                    2     /* SYN发送,连接请求 */
#define TCP_SYN_RCVD                    3     /* SYN接收,接收到连接请求 */
#define TCP_ESTABLISHED                 4     /* TCP连接建立 */
#define TCP_FIN_WAIT_1                  5     /* WAIT_1状态 */
#define TCP_FIN_WAIT_2                  6     /* WAIT_2状态 */
#define TCP_CLOSE_WAIT                  7     /* 等待关闭 */
#define TCP_CLOSING                     8     /* 正在关闭 */
#define TCP_LAST_ACK                    9     /* LAST_ACK*/
#define TCP_TIME_WAIT                   10    /* 2MSL等待 */

/* 以下值为固定值不可以更改 */
#define WCHNET_MEM_ALIGN_SIZE(size)    (((size) + WCHNET_MEM_ALIGNMENT - 1) & ~(WCHNET_MEM_ALIGNMENT - 1))
#define WCHNET_SIZE_IPRAW_PCB          0xFF   /* IPRAW PCB大小 */
#define WCHNET_SIZE_UDP_PCB            0x30   /* UDP PCB大小 */
#define WCHNET_SIZE_TCP_PCB            0xFF   /* TCP PCB大小 */
#define WCHNET_SIZE_TCP_PCB_LISTEN     0x40   /* TCP LISTEN PCB大小 */
#define WCHNET_SIZE_IP_REASSDATA       0x40   /* IP分片管理  */
#define WCHNET_SIZE_PBUF               0x40   /* Packet Buf */
#define WCHNET_SIZE_TCP_SEG            0x60   /* TCP SEG结构 */
#define WCHNET_SIZE_MEM                0x0c   /* sizeof(struct mem) */
#define WCHNET_SIZE_ARP_TABLE          0x20   /* sizeof ARP table */

#define WCHNET_SIZE_POOL_BUF           WCHNET_MEM_ALIGN_SIZE(WCHNET_TCP_MSS + 40 + 14) /* pbuf池大小 */
#define WCHNET_MEMP_SIZE               ((WCHNET_MEM_ALIGNMENT - 1) +                                                 \
                          (WCHNET_NUM_IPRAW * WCHNET_MEM_ALIGN_SIZE(WCHNET_SIZE_IPRAW_PCB)) +                        \
                          (WCHNET_NUM_UDP * WCHNET_MEM_ALIGN_SIZE(WCHNET_SIZE_UDP_PCB)) +                            \
                          (WCHNET_NUM_TCP * WCHNET_MEM_ALIGN_SIZE(WCHNET_SIZE_TCP_PCB)) +                            \
                          (WCHNET_NUM_TCP_LISTEN * WCHNET_MEM_ALIGN_SIZE(WCHNET_SIZE_TCP_PCB_LISTEN)) +              \
                          (WCHNET_NUM_TCP_SEG * WCHNET_MEM_ALIGN_SIZE(WCHNET_SIZE_TCP_SEG)) +                        \
                          (WCHNET_NUM_IP_REASSDATA * WCHNET_MEM_ALIGN_SIZE(WCHNET_SIZE_IP_REASSDATA)) +              \
                          (WCHNET_NUM_PBUF * (WCHNET_MEM_ALIGN_SIZE(WCHNET_SIZE_PBUF) + WCHNET_MEM_ALIGN_SIZE(0))) + \
                          (WCHNET_NUM_POOL_BUF * (WCHNET_MEM_ALIGN_SIZE(WCHNET_SIZE_PBUF) + WCHNET_MEM_ALIGN_SIZE(WCHNET_SIZE_POOL_BUF))))

#define HEAP_MEM_ALIGN_SIZE          (WCHNET_MEM_ALIGN_SIZE(WCHNET_SIZE_MEM))
#define WCHNET_RAM_HEAP_SIZE         (WCHNET_MEM_HEAP_SIZE + (2 * HEAP_MEM_ALIGN_SIZE) + WCHNET_MEM_ALIGNMENT)
#define WCHNET_RAM_ARP_TABLE_SIZE    (WCHNET_SIZE_ARP_TABLE * WCHNET_NUM_ARP_TABLE)

/* LED回调结构体 */
typedef void (*led_callback)( uint8_t setbit );

/* LED回调结构体 */
typedef uint32_t (*eth_tx_set )( uint16_t len, uint32_t *pBuff );

/* DNS回调结构体 */
typedef void (*dns_callback)( const char *name, uint8_t *ipaddr, void *callback_arg );

/* DHCP回调结构体 */
typedef uint8_t (*dhcp_callback)( uint8_t status, void * );

// 接收回调函数
struct _SCOK_INF;
typedef void (*pSockRecv)( struct _SCOK_INF *, uint32_t, uint16_t, uint8_t *, uint32_t);

/* sokcet信息表 */
typedef struct _SCOK_INF
{
    uint32_t IntStatus;                       /* 中断状态 */
    uint32_t SockIndex;                       /* Socket索引值 */
    uint32_t RecvStartPoint;                  /* 接收缓冲区的开始指针 */
    uint32_t RecvBufLen;                      /* 接收缓冲区长度 */
    uint32_t RecvCurPoint;                    /* 接收缓冲区的当前指针 */
    uint32_t RecvReadPoint;                   /* 接收缓冲区的读指针 */
    uint32_t RecvRemLen;                      /* 接收缓冲区的剩余长度 */
    uint32_t ProtoType;                       /* 协议类型 */
    uint32_t ScokStatus;                      /* 低字节Socket状态,次低字节为TCP状态,仅TCP模式下有意义 */
    uint32_t DesPort;                         /* 目的端口 */
    uint32_t SourPort;                        /* 源端口在IPRAW模式下为协议类型 */
    uint8_t  IPAddr[4];                       /* Socket目标IP地址 32bit*/
    void *Resv1;                              /* 保留,内部使用,用于保存各个PCB */
    void *Resv2;                              /* 保留,内部使用,TCP Server使用 */
    pSockRecv AppCallBack;                    /* 接收回调函数*/
} SOCK_INF;

struct _WCH_CFG
{
  uint32_t TxBufSize;   /* MAC发送缓冲区大小,保留使用 */
  uint32_t TCPMss;      /* TCP MSS大小 */
  uint32_t HeapSize;    /* 堆分配内存大小 */
  uint32_t ARPTableNum; /* ARP列表个数 */
  uint32_t MiscConfig0; /* 杂项配置0 */
  /* 位0     TCP发送缓冲区复制 1:复制,0:不复制 */
  /* 位1     TCP接收复制优化,内部调试使用 */
  /* 位2     删除最早的TCP连接 1:启用,0:禁用  */
  /* 位3-7   IP分片的PBUF个数  */
  uint32_t MiscConfig1; /* 杂项配置1 */
  /* 位0-7   Socket的个数 */
  /* 位8-12  保留 */
  /* 位13    PING使能,1:开启  0:关闭  */
  /* 位14-18 TCP重传次数  */
  /* 位19-23 TCP重传周期,单位为50毫秒  */
  /* 位25    发送失败重试,1:启用,0:禁用 */
  /* 位26-31 保留 */
  led_callback led_link; /* PHY Link状态指示灯 */
  led_callback led_data; /* 以太网通信指示灯 */
  eth_tx_set net_send;
  uint32_t   CheckValid;/* 配置值有效标志,固定值 @WCHNET_CFG_VALID */
};

struct _NET_SYS
{
  uint8_t  IPAddr[4];        /* IP地址 32bit */
  uint8_t  GWIPAddr[4];      /* 网关地址 32bit */
  uint8_t  MASKAddr[4];      /* 子网掩码 32bit */
  uint8_t  MacAddr[8];       /* MAC地址 48bit */
  uint8_t  UnreachIPAddr[4]; /* 不可到达IP */
  uint32_t RetranCount;      /* 重试次数 默认为10次 */
  uint32_t RetranPeriod;     /* 重试周期,单位MS,默认500MS */
  uint32_t PHYStat;          /* PHY状态码 8bit */
  uint32_t NetStat;          /* 以太网的状态 ,包含是否打开等 */
  uint32_t MackFilt;         /*  MAC过滤,默认为接收广播,接收本机MAC 8bit */
  uint32_t GlobIntStatus;    /* 全局中断 */
  uint32_t UnreachCode;      /* 不可达 */
  uint32_t UnreachProto;     /* 不可达协议 */
  uint32_t UnreachPort;      /* 不可到达端口 */
  uint32_t SendFlag;
  uint32_t Flags;
};

/* KEEP LIVE配置结构体 */
struct _KEEP_CFG
{
  uint32_t KLIdle;  /* KEEPLIVE空闲时间,单位ms */
  uint32_t KLIntvl; /* KEEPLIVE周期,单位ms */
  uint32_t KLCount; /* KEEPLIVE次数 */
};

/**
* [url=home.php?mod=space&uid=247401]@brief[/url]   库初始化.
*
* @param   ip - IP地址
* @param   gwip - 网关
* @param   mask - 子网掩码
* @param   macaddr - MAC地址
*
* [url=home.php?mod=space&uid=266161]@return[/url]  @ERR_T
*/
uint8_t WCHNET_Init(const uint8_t *ip, const uint8_t *gwip, const uint8_t *mask, const uint8_t *macaddr);

/**
* [url=home.php?mod=space&uid=247401]@brief[/url]   获取版本号
*
* @param   None
*
* [url=home.php?mod=space&uid=266161]@return[/url]  库版本
*/
uint8_t WCHNET_GetVer(void);

/**
* @brief   获取MAC地址
*
* @param(in)   macaddr - 内存地址
*
* @param(out)  mac地址
*
* @return  None
*/
void WCHNET_GetMacAddr(uint8_t *macaddr);

/**
* @brief   库参数配置
*
* @param   cfg -配置参数 @_WCH_CFG
*
* @return  库配置初始化状态 @CFG_INIT_STAT
*/
uint8_t WCHNET_ConfigLIB(struct _WCH_CFG *cfg);

/**
* @brief   库主任务函数,需要一直不断调用
*
* @param   None
*
* @return  None
*/
void WCHNET_MainTask(void);

/**
* @brief   时钟中断服务函数,需配置时钟周期
*
* @param   timperiod - 时钟周期,单位ms
*
* @return  None
*/
void WCHNET_TimeIsr( uint16_t timperiod);

/**
* @brief   以太网中断服务函数,产生以太网中断后调用
*
* @param   None
*
* @return  None
*/
void WCHNET_ETHIsr(void);

/**
* @brief   Get PHY status
*
* @param   None
*
* @return  PHY 状态 @PHY_STAT
*/
uint8_t WCHNET_GetPHYStatus(void);

/**
* @brief   查询全局中断
*
* @param   None
*
* @return  GLOB_INT
*/
uint8_t WCHNET_QueryGlobalInt(void);

/**
* @brief   读全局中断并将全局中断清零
*
* @param   None
*
* @return  GLOB_INT
*/
uint8_t WCHNET_GetGlobalInt(void);

/**
* @brief   创建socket
*
* @param(in)   *socketid - 内存地址
* @param       socinf - 创建socket的配置参数 @SOCK_INF
*
* @param(out)  *socketid - socketID值
*
* @return  @ERR_T
*/
uint8_t WCHNET_SocketCreat( uint8_t *socketid, SOCK_INF *socinf);

/**
* @brief   Socket发送数据
*
* @param       socketid - socketID值
* @param       *buf - 发送数据首地址
* @param(in)   *len - 内存地址及期望发送的数据长度
*
* @param(out)  *len - 实际发送的数据长度
*
* @return  @ERR_T
*/
uint8_t WCHNET_SocketSend( uint8_t socketid, uint8_t *buf, uint32_t *len);

/**
* @brief   Socket接收数据
*
* @param       socketid - socketID值
* @param       *buf - 接收数据首地址
* @param(in)   *len - 内存地址及期望读取的数据长度
*
* @param(out)  *buf - 写入读取到的数据内容
* @param(out)  *len - 实际读取的数据长度
*
* @return  @ERR_T
*/
uint8_t WCHNET_SocketRecv( uint8_t socketid, uint8_t *buf, uint32_t *len);

/**
* @brief   获取socket中断并清零
*
* @param   socketid - socketID值
*
* @return  Sn_INT
*/
uint8_t WCHNET_GetSocketInt( uint8_t socketid );

/**
* @brief   获取socket接收长度
*
* @param       socketid - socketID值
* @param(in)   *bufaddr - 内存地址
*
* @param(out)  *bufaddr - socket数据的首地址
*
* @return  数据长度
*/
uint32_t WCHNET_SocketRecvLen( uint8_t socketid, uint32_t *bufaddr);

/**
* @brief   TCP连接
*
* @param   socketid - socketID值
*
* @return  @ERR_T
*/
uint8_t WCHNET_SocketConnect( uint8_t socketid);

/**
* @brief   TCP监听
*
* @param   socketid - socketID值
*
* @return  @ERR_T
*/
uint8_t WCHNET_SocketListen( uint8_t socketid);

/**
* @brief   关闭socket
*
* @param   socketid - socketID值
* @param   mode - socket为TCP连接,参数为关闭的方式 @TCP关闭参数
*
* @return  @ERR_T
*/
uint8_t WCHNET_SocketClose( uint8_t socketid, uint8_t mode );

/**
* @brief   修改接收缓冲区
*
* @param   socketid - socketID值
* @param   bufaddr - 接收缓冲区地址
* @param   bufsize - 接收缓冲区大小
*
* @return  None
*/
void WCHNET_ModifyRecvBuf( uint8_t socketid, uint32_t bufaddr, uint32_t bufsize);

/**
* @brief   UDP发送,指定目的IP、目的端口
*
* @param       socketid - socketID值
* @param       *buf - 发送数据的地址
* @param(in)   *slen - 期望发送的长度地址
* @param       *sip - 目的IP地址
* @param       port - 目的端口号
*
* @param(out)  *slen - 实际发送的长度
*
* @return  @ERR_T
*/
uint8_t WCHNET_SocketUdpSendTo( uint8_t socketid, uint8_t *buf, uint32_t *slen, uint8_t *sip, uint16_t port);

/**
* @brief   ASCII码地址转网络地址
*
* @param       *cp - 需转换的ASCII码地址
* @param(in)   *addr - 转换后网络地址存放的内存首地址
*
* @param(out)  *addr - 转换后的网络地址
*
* @return  0:转换成功  1:转换失败
*/
uint8_t WCHNET_Aton(const char *cp, uint8_t *addr);

/**
* @brief   网络地址转ASCII地址
*
* @param   *ipaddr - socketID值
*
* @return  转换后的ASCII地址
*/
uint8_t *WCHNET_Ntoa( uint8_t *ipaddr);

/**
* @brief   设置socket的TTL
*
* @param   socketid - socketID值
* @param   ttl - TTL值
*
* @return  @ERR_T
*/
uint8_t WCHNET_SetSocketTTL( uint8_t socketid, uint8_t ttl);

/**
* @brief   立即启动TCP重传
*
* @param   socketid - socketID值
*
* @return  None
*/
void WCHNET_RetrySendUnack( uint8_t socketid);

/**
* @brief   查询未发送成功的数据包
*
* @param       socketid - socketID值
* @param(in)   *addrlist - 存放的内存首地址
* @param       lislen - 存放的内存大小
*
* @param(out)  *addrlist - 未发送成功的数据包地址列表
*
* @return  未发送和未应答的数据段的个数
*/
uint8_t WCHNET_QueryUnack( uint8_t socketid, uint32_t *addrlist, uint16_t lislen );

/**
* @brief   DHCP启动
*
* @param   dhcp - 应用层回调函数
*
* @return  @ERR_T
*/
uint8_t WCHNET_DHCPStart( dhcp_callback dhcp );

/**
* @brief   DHCP停止
*
* @param   None
*
* @return  @ERR_T
*/
uint8_t WCHNET_DHCPStop( void );

/**
* @brief   配置DHCP主机名
*
* @param   *name - DHCP主机名首地址
*
* @return  0成功,否则失败
*/
uint8_t WCHNET_DHCPSetHostname(char *name);

/**
* @brief   Initialize the resolver: set up the UDP pcb and configure the default server
*
* @param   *dnsip - dns服务器IP地址
* @param   port - dns服务器端口号
*
* @return  None
*/
void WCHNET_InitDNS( uint8_t *dnsip, uint16_t port);

/**
* Resolve a hostname (string) into an IP address.
*
* @param   hostname - the hostname that is to be queried
* @param   addr     - pointer to a struct ip_addr where to store the address if it is already
*                     cached in the dns_table (only valid if ERR_OK is returned!)
* @param   found    - a callback function to be called on success, failure or timeout (only if
*                     ERR_INPROGRESS is returned!)
* @param   arg      - argument to pass to the callback function
*
* @return  @ERR_T
*   WCHNET_ERR_SUCCESS if hostname is a valid IP address string or the host name is already in the local names table.
*   ERR_INPROGRESS     enqueue a request to be sent to the DNS server for resolution if no errors are present.
*/
uint8_t WCHNET_HostNameGetIp( const char *hostname, uint8_t *addr, dns_callback found, void *arg );

/**
* @brief   配置库KEEP LIVE参数
*
* @param   *cfg - KEEPLIVE配置参数
*
* @return  None
*/
void WCHNET_ConfigKeepLive( struct _KEEP_CFG *cfg );

/**
* @brief   配置socket KEEP LIVE使能
*
* @param   socketid - socketID值
* @param   enable - 1:启用  0:关闭
*
* @return  @ERR_T
*/
uint8_t WCHNET_SocketSetKeepLive( uint8_t socketid, uint8_t enable );

/**
* @brief   PHY状态改变配置
*
* @param   phy_stat - PHY状态值
*
* @return  None
*/
void WCHNET_PhyStatus( uint32_t phy_stat );


#ifdef __cplusplus
}
#endif
#endif
main.c
void WCHNET_HandleGlobalInt(void)
{
    u8  initstat;
    u16 i;
    u8  socketinit;

    initstat = WCHNET_GetGlobalInt(); /* 获取全局中断标志*/
    if(initstat & GINT_STAT_UNREACH){/* 不可达中断 */
        debug_printf("GINT_STAT_UNREACH\r\n");
    }
    if(initstat & GINT_STAT_IP_CONFLI){/* IP冲突中断 */
        debug_printf("GINT_STAT_IP_CONFLI\r\n");
    }
    if(initstat & GINT_STAT_PHY_CHANGE){/* PHY状态变化中断 */
        i = WCHNET_GetPHYStatus(); /* 获取PHY连接状态*/
        if(i & PHY_Linked_Status) {
            debug_printf("PHY Link connected\r\n");
        }
        else {
            debug_printf("PHY Link disconnected\r\n");
        }
    }
    if(initstat & GINT_STAT_SOCKET){//socket通信
        for(i = 0; i < WCHNET_MAX_SOCKET_NUM; i++){
            socketinit = WCHNET_GetSocketInt(i);
            if(socketinit){//有事件需要处理

            }
        }
    }
}

/*********************************************************************
* @fn      WCHNET_DHCPCallBack
*
* @brief   DHCPCallBack
*
* @return  DHCP status
*/
u8 WCHNET_DHCPCallBack(u8 status, void *arg)
{
    u8 *p;
    if(!status){
        p = arg;
        printf("DHCP Success\r\n");
        memcpy(eth_ip_addr, p, 4);
        memcpy(eth_gwip_addr, &p[4], 4);
        memcpy(eth_ip_mask, &p[8], 4);
        debug_printf("eth_ip_addr: %d.%d.%d.%d\r\n",eth_ip_addr[0],eth_ip_addr[1],eth_ip_addr[2],eth_ip_addr[3]);
        debug_printf("eth_gwip_addr: %d.%d.%d.%d\r\n",eth_gwip_addr[0],eth_gwip_addr[1],eth_gwip_addr[2],eth_gwip_addr[3]);
        debug_printf("eth_ip_mask: %d.%d.%d.%d\r\n",eth_ip_mask[0],eth_ip_mask[1],eth_ip_mask[2],eth_ip_mask[3]);
        debug_printf("DNS1: %d.%d.%d.%d \r\n", p[12], p[13], p[14], p[15]);
        debug_printf("DNS2: %d.%d.%d.%d \r\n", p[16], p[17], p[18], p[19]);
        return 0;
    }
    else{
        debug_printf("DHCP Fail %02x \r\n", status);
        return 1;
    }
}

int main(void)
{
        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
        delay_init();
        uart_init(115200, 115200);

        //初始化以太网
        if (eth_init()!=0) {
            debug_printf("eth_init err\r\n");
            delay_ms(100);
        __disable_irq();
        NVIC_SystemReset();//重启
    }
        WCHNET_DHCPStart(WCHNET_DHCPCallBack);                 /*开始DHCP,返回结果在WCHNET_DHCPCallBack函数中*/

        printf("start\r\n");
        while(1)
    {
        WCHNET_MainTask();           /*以太网库主任务函数,需要循环调用*/
        if(WCHNET_QueryGlobalInt())  /*查询以太网全局中断,如果有中断,调用全局中断处理函数*/
        {
            WCHNET_HandleGlobalInt();
        }
        }
}

  MAC地址:需唯一,可通过WCHNET_GetMacAddr()获取芯片内置地址或手动设置。
  回调函数:通过WCHNET_DHCPStart()注册,用于接收DHCP事件(成功/失败/超时)。
  主机名:可选配置,便于在路由器设备列表中识别设备。


应用场景:
  • 物联网设备
    CH32V307作为边缘节点,通过DHCP快速接入局域网,实现数据上传至云端(如MQTT协议)。例如,智能家居中的温湿度传感器,通电后自动获取IP并连接服务器。
  • 工业自动化
    在工厂网络中,设备需频繁更换位置或IP地址。DHCP功能可避免手动配置错误,提升部署效率。例如,PLC模块通过DHCP获取临时IP,完成调试后恢复静态IP。
  • 开发调试阶段
    使用DHCP可快速验证网络功能,无需预先规划IP地址。结合ping命令或网络调试助手,可快速定位连接问题。




使用特权

评论回复
沙发
风之呢喃| | 2025-8-1 13:46 | 只看该作者
能ping通对吧。楼主是否可以把工程文件打包一份上传一下。

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

认证:项目经理
简介:资深嵌入式开发工程师

95

主题

181

帖子

3

粉丝