如何用stm32使用LWIP网络协议栈实现DHCP客户端

发布者:温馨的家庭最新更新时间:2020-02-07 来源: elecfans关键字:stm32  LWIP网络协议栈  DHCP客户端 手机看文章 扫描二维码
随时随地手机看文章

LWIP是一款开源的嵌入式网络协议栈,支持的功能很多,而且能在多任务环境下和单任务裸机环境下跑,今天说说他的移植过程,芯片为STM32,网卡为ENC28J60,无操作系统

首先下载LWIP的源代码,我下载的是1.4.1的源码,下载后解压,文件结构如图

如何用stm32使用LWIP网络协议栈实现DHCP客户端

将这四个目录中的文件全部拷贝到工程中,API是一些socket通讯的接口,需要在多任务的环境下实现,core里面存放的内核源码,我们主要使用IPV4,include目录下是需要包含的目录,lwip只要求我们包含include目录,里面的内层目录会自动找到,最后建立的工程目录如下

如何用stm32使用LWIP网络协议栈实现DHCP客户端

如何用stm32使用LWIP网络协议栈实现DHCP客户端

如何用stm32使用LWIP网络协议栈实现DHCP客户端

好了,此时源码已经做好,还有需要做的,在include目录下新建一个文件夹,必须叫arch,里面存放这几个文件,自己新建

如何用stm32使用LWIP网络协议栈实现DHCP客户端

文件的具体内容如下

cc.h

/*

* Copyright (c) 2001-2003 Swedish Institute of Computer Science.

* All rights reserved.

*

* RedistribuTIon and use in source and binary forms, with or without modificaTIon,

* are permitted provided that the following condiTIons are met:

*

* 1. RedistribuTIons of source code must retain the above copyright notice,

* this list of conditions and the following disclaimer.

* 2. Redistributions in binary form must reproduce the above copyright notice,

* this list of conditions and the following disclaimer in the documentation

* and/or other materials provided with the distribution.

* 3. The name of the author may not be used to endorse or promote products

* derived from this software without specific prior written permission.

*

* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS‘’ AND ANY EXPRESS OR IMPLIED

* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF

* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT

* SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,

* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT

* OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS

* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN

* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING

* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY

* OF SUCH DAMAGE.

*

* This file is part of the lwIP TCP/IP stack.

*

* Author: Adam Dunkels

*

*/

#ifndef __CC_H__

#define __CC_H__

#include “cpu.h”

//编译器无关的数据类型定义

typedef unsigned char u8_t;

typedef signed char s8_t;

typedef unsigned short u16_t;

typedef signed short s16_t;

typedef unsigned long u32_t;

typedef signed long s32_t;

typedef u32_t mem_ptr_t;

typedef int sys_prot_t;

//lwip调试的时候数据类型定义

#define U16_F “hu”

#define S16_F “d”

#define X16_F “hx”

#define U32_F “u”

#define S32_F “d”

#define X32_F “x”

#define SZT_F “uz”

//根据不同的编译器的符号定义

#if defined (__ICCARM__)

#define PACK_STRUCT_BEGIN

#define PACK_STRUCT_STRUCT

#define PACK_STRUCT_END

#define PACK_STRUCT_FIELD(x) x

#define PACK_STRUCT_USE_INCLUDES

#elif defined (__CC_ARM)

#define PACK_STRUCT_BEGIN __packed

#define PACK_STRUCT_STRUCT

#define PACK_STRUCT_END

#define PACK_STRUCT_FIELD(x) x

#elif defined (__GNUC__)

#define PACK_STRUCT_BEGIN

#define PACK_STRUCT_STRUCT __attribute__ ((__packed__))

#define PACK_STRUCT_END

#define PACK_STRUCT_FIELD(x) x

#elif defined (__TASKING__)

#define PACK_STRUCT_BEGIN

#define PACK_STRUCT_STRUCT

#define PACK_STRUCT_END

#define PACK_STRUCT_FIELD(x) x

#endif

#define LWIP_PLATFORM_ASSERT(x) //do { if(!(x)) while(1); } while(0)

#endif /* __CC_H__ */

cpu.h

/*

* Copyright (c) 2001-2003 Swedish Institute of Computer Science.

* All rights reserved.

*

* Redistribution and use in source and binary forms, with or without modification,

* are permitted provided that the following conditions are met:

*

* 1. Redistributions of source code must retain the above copyright notice,

* this list of conditions and the following disclaimer.

* 2. Redistributions in binary form must reproduce the above copyright notice,

* this list of conditions and the following disclaimer in the documentation

* and/or other materials provided with the distribution.

* 3. The name of the author may not be used to endorse or promote products

* derived from this software without specific prior written permission.

*

* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS‘’ AND ANY EXPRESS OR IMPLIED

* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF

* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT

* SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,

* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT

* OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS

* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN

* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING

* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY

* OF SUCH DAMAGE.

*

* This file is part of the lwIP TCP/IP stack.

*

* Author: Adam Dunkels

*

*/

#ifndef __CPU_H__

#define __CPU_H__

//定义cpu的数据模式,大端小端

#define BYTE_ORDER LITTLE_ENDIAN

#endif /* __CPU_H__ */

perf.h

#ifndef __PERF_H__

#define __PERF_H__

//用于lwip内置的统计功能

//不使能定义为空就可以了

#define PERF_START /* null definition */

#define PERF_STOP(x) /* null definition */

#endif /* __PERF_H__ */

sys_arch.h

#ifndef __SYS_RTXC_H__

#define __SYS_RTXC_H__

void init_lwip_timer(void); //初始化LWIP定时器

u8_t timer_expired(u32_t *last_time,u32_t tmr_interval); //定时器超时判断

#endif /* __SYS_RTXC_H__ */

sya_arch.c--注意该文件要加入源文件列表中,这是c文件哦

#include “lwip/debug.h”

#include “lwip/def.h”

#include “lwip/sys.h”

#include “lwip/mem.h”

#include “timerx.h”

//初始化LWIP定时器

void init_lwip_timer(void)

{

TIM6_Int_Init(1000,719);//100Khz计数频率,计数到100为10ms

}

//为LWIP提供计时

extern u32_t lwip_timer;//lwip 计时器,每10ms增加1.

u32_t sys_now(void)

{

return lwip_timer;

}

//定时器超时判断

//last_time:最近时间

//tmr_interval:定时器溢出周期

u8_t timer_expired(u32_t *last_time,u32_t tmr_interval)

{

u32_t time;

time = *last_time;

if((lwip_timer-time)》=tmr_interval){

*last_time = lwip_timer;

return 1;

}

return 0;

}

可以看到我们定义了定时器,那么就要修改相关的定时器文件,文件如下

timerx.c

#include “timerx.h”

u32 lwip_timer=0;//lwip 计时器,每10ms增加1.

//定时器6中断服务程序

void TIM6_IRQHandler(void)

{

if (TIM_GetITStatus(TIM6, TIM_IT_Update) != RESET) //检查指定的TIM中断发生与否:TIM 中断源

{

TIM_ClearITPendingBit(TIM6, TIM_IT_Update ); //清除TIMx的中断待处理位:TIM 中断源

lwip_timer++;//lwip计时器增加1

}

}

//基本定时器6中断初始化

//这里时钟选择为APB1的2倍,而APB1为36M

//arr:自动重装值。

//psc:时钟预分频数

//这里使用的是定时器3!

void TIM6_Int_Init(u16 arr,u16 psc)

{

TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;

NVIC_InitTypeDef NVIC_InitStructure;

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM6, ENABLE); //时钟使能

TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值 计数到5000为500ms

TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值 10Khz的计数频率

TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim

TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式

TIM_TimeBaseInit(TIM6, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位

TIM_ITConfig( TIM6,TIM_IT_Update“TIM_IT_Trigger,ENABLE);//使能定时器6更新触发中断

TIM_Cmd(TIM6, ENABLE); //使能TIMx外设

NVIC_InitStructure.NVIC_IRQChannel = TIM6_IRQn; //TIM3中断

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; //先占优先级0级

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2; //从优先级3级

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能

NVIC_Init(&NVIC_InitStructure); //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器

关键字:stm32  LWIP网络协议栈  DHCP客户端 引用地址:如何用stm32使用LWIP网络协议栈实现DHCP客户端

上一篇:融性能、集成度和能效于一身,STM32H7系列问市
下一篇:STM32单片机的复用端口初始化的步骤及方法

推荐阅读最新更新时间:2024-11-09 13:45

如何实现stm32单片机每次接上电源就进行一次重启动?
如何实现stm32单片机每次接上电源就进行一次重启动? 在嵌入式系统中,单片机的重启有时是十分必要的,它可以解决一些系统启动过程中的问题、恢复系统的正常运行、使系统更加稳定等。 然而,有时候我们需要每次接上电源就进行一次重启,也就是实现系统的自动重启。本文将给出一个思路,来实现stm32单片机每次接上电源就进行一次重启动。 一、背景知识 在嵌入式系统中,有两类重启,分别是软重启和硬重启。 1.1 软重启 软重启是指通过程序控制单片机重新开始执行,实现系统的重启。在stm32单片机中,一般采用如下代码进行软重启: ```c NVIC_SystemReset(); //调用此函数进行软重启 ``` 1.2 硬重启 硬重启是指通过硬件控
[单片机]
基于STM32设计的酒精检测仪
一、需求分析 随着社会的发展和生活水平的提高,人们对于行车安全、家庭安全的要求越来越高,而酒驾等问题也日渐突出,为此,开发一款基于STM32的酒精检测仪,通过检测酒精浓度,实时显示结果并进行报警,可以有效避免因酒后驾车带来的安全隐患。 二、设计思路 2.1 硬件设计 1、主控芯片采用STM32F103RCT6,该芯片具有较高的性能和稳定性,能够满足本设计的各项需求。 2、酒精传感器采用MQ-3模块,该模块具有高精度、响应速度快等特点,能够准确检测酒精浓度。 3、OLED显示屏,用于实时显示酒精浓度等信息。 4、蜂鸣器,用于进行声音报警。 5、按键,用于设定报警阈值。 2.2 软件设计 1、IO口配置:将相应
[单片机]
STM32片内RTC亚秒特性的应用示例(上)
绝大多数STM32系列里的RTC都具有亚秒【或称子秒】计数单元。为了了解亚秒特性及功能,不妨先看RTC的功能框图。本文中的有关截图若无特别说明均来自STM32L4系列参考手册。 RTC的时钟源【RTCCLK】可以是LSE、LSI或者HSE/32,由RTCCLK最终变成日历的秒脉冲驱动信号经过了2次分频。先经过上图中A处的异步分频单元,默认分频系数是128,形成ck_apre时钟,默认情况下该时钟频率为256Hz;然后该时钟脉冲来到图中B处的同步分频单元,默认分频系数为256,最终形成1Hz的秒脉冲【ck_spre】到日历单元。关于两分频单元分频系数的配置,通过对RTC_PRER寄存器的相关位编程实现。 其中异步分频系数配
[单片机]
<font color='red'>STM32</font>片内RTC亚秒特性的应用示例(上)
使用FlyMCU对STM32串口下载程序出错的解决方法
1,问题:使用FlyMCU对STM32进行串口下载程序时出现0KB出错,如下图所示: 2,分析:出现上述所示错误是因为勾选 了 编程到FLASH是写选项字节 3,解决方法: (1)安装 STM32 ST-LINK Utility 软件(百度 搜索 “STM32 ST-LINK Utility” 即可获得下载) (2)安装软件 (3)使用管理员身份运行该软件 (4)操作: 步骤如下:Target -- Connect -- Option Bytes... 进入Option Bytes后 直接点击Apply (5)使用STM32 ST-LINK Utility下载自己提供的 xxx.hex 文件 步骤如下:File -
[单片机]
使用FlyMCU对<font color='red'>STM32</font>串口下载程序出错的解决方法
stm32处理器调试模式下运行正常,上电自启动后运行不正常的故障排查分析
最近负责一个项目,用到stm32f4的一款高性能芯片。研发过程中遇到一个很诡异的现象,前前后后折腾了两三天,最后才搞定。由于是新手,经验不足,排故过程很纠结~~ 现象如下: 1.采用JLINK下载程序后,断电让其上电重新启动,发现有时可以正常运行,有时候无法正常运行,大约每两、三次就有一次无法正常上电启动。 2.通过JLINK调试程序,每次均正常运行。太诡异了! 发现问题后开始定位原因。首先考虑是BOOT启动出问题了,stm32f4启动方式分三种:User FLash、SystemFlash和EmbeddedSRAM,通过BOOT0和BOOT1管脚配置。程序正常运行时从User FLash启动,如果BOOT0和BOOT1配
[单片机]
STM32固件解密步骤
方法1:代码解密 FLASH_OBProgramInitTypeDef OBInit; __HAL_FLASH_PREFETCH_BUFFER_DISABLE(); HAL_FLASHEx_OBGetConfig(&OBInit); if(OBInit.RDPLevel != OB_RDP_LEVEL_0) { OBInit.OptionType = OPTIONBYTE_RDP; OBInit.RDPLevel = OB_RDP_LEVEL_0; (void)HAL_FLASH_Unlock(); (void)HAL_FLASH_OB_Unlock(); (void)HAL_FLASHEx_OBProgram(&OBInit
[单片机]
STM32HAL库ADC实验(一)——阻塞查询法
实验现象 生成代码: int main(void) { /* USER CODE BEGIN 1 */ uint16_t adcData; float voltage; /* USER CODE END 1 */ /* MCU Configuration--------------------------------------------------------*/ /* Reset of all peripherals, Initializes the Flash interface and the Systick. */ HAL_Init(); /* USER CODE BE
[单片机]
STM32HAL库ADC实验(一)——阻塞查询法
STM32使用外部晶振并设置主频为48M
因为项目上对时钟要求不高,只需要用到一个9600波特率的串口和一些普通IO口,下面是简单的使用方法。 void MYRCC_DeInit(void) { RCC- APB1RSTR = 0x00000000;//复位结束 RCC- APB2RSTR = 0x00000000; RCC- AHBENR = 0x00000014; //睡眠模式闪存和SRAM时钟使能.其他关闭. RCC- APB2ENR = 0x00000000; //外设时钟关闭. RCC- APB1ENR = 0x00000000; RCC- CR |= 0x00000001; //使能内部高速时钟HSION RCC- CF
[单片机]
<font color='red'>STM32</font>使用外部晶振并设置主频为48M
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved