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

fpga实操训练(利用fpga实现pwm)

【声明:版权所有,欢迎转载,请勿用于商业用途。联系信箱:feixiaoxing163.com】pwm,其实就是

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing @163.com】

        pwm,其实就是方波。它的本质就是通过方波中占空比的调节,实现对外部设备的控制。简单如台灯,复杂如电机都是这么做的。fpga输入的时钟信号是50M,每个时钟信号中高低电平的比率是50%对50%。那方波是什么样的呢?它的控制频率可能只有100,高低电平是的比率很有可能是10%对90%,那么这又该如何实现呢?

module pwm_test(clk, rst, pwm);input clk;
input rst;
output pwm;wire clk;
wire rst;
reg pwm;reg[31:0] start;
reg[31:0] stop;// pwm determinate the basic frequency of pwmalways@(posedge clk or negedge rst)if(!rst)start <= 32&#39;d0;elsestart <= start + 32&#39;d8590;always@(posedge clk or negedge rst)if(!rst)stop <= 32&#39;d429_496_730;elsestop <= stop;always@(posedge clk or negedge rst)if(!rst)pwm <= 1&#39;b1;else if(start

        和之前的做法不同,今天我们首先给出verilog代码,再进一步分析。代码不长,主要的寄存器是start、stop和pwm。其中pwm是输出信号。

        首先来看start信号,rst复位的时候被设置为0,随后开始每次自增8590。至于为什么每次增加这个数字,目前未知。

        其次来看stop信号,rst复位的时候被设置位429_496_730,大约是4亿多。看样子没有越界。因为32位整数最大能表示的数值是40多亿。等rst被复位之后,这个数据就没有发生变化了。

        最后来看pwm信号,这也是最重要的部分。因为pwm连接的是一个蜂鸣器,低电平有效,所以复位后默认输出是1。接着,在start小于stop这段时间内,蜂鸣器有效,发出声音;但是start大于stop的这段时间,蜂鸣器无效,停止发出声音。看到这里,很多同学都以为verilog分析完了,这就是一个开始发声、停止发声的verilog代码,和pwm没有关系。但是,请大家重新看下这段代码,

always@(posedge clk or negedge rst)if(!rst)start <= 32&#39;d0;elsestart <= start + 32&#39;d8590;

        前面我们说过,32位数据表达的数值是有范围的。start不可能这样一直增加下去,等到了临界值之后,start又会重新小于stop,蜂鸣器又会发生,循环又开始了。这才是最终我们想要的效果。

        分析完了之后,就是分析8590、429496730这些数字是怎么来的。首先假设输入的clk频率是50M,假设最终pwm的控制频率是100,则单次pwm内的时钟数应该是50000000/100=500000。

        利用32位整数表达范围有限制的特定,对于2^32位整数来说,单次pwm里面的每一个clk相当于自增多少呢?那就是2^32/500000 = 8589.9,差不多就是8590,这正是start每次递增的数据。

        而429496730代表什么呢?因为每次递增8589.9,差不多需要迭代500000次才能越界,而429496730/8589.9=50000左右,差不多占了500000次的10%所有,因此这个数据主要是调节单次pwm调节内的占空比的。实际的计算公式应该是,target = 500000*ratio*8589.9。

        弄懂了原理之后,下面就是编译和pin绑定了。

        实验的时候因为没有示波器,所以用了蜂鸣器做了实验。同样的频率内,ratio越大,蜂鸣器越响;反之则越闷,大家可以试试。


推荐阅读
  • 前言--页数多了以后需要指定到某一页(只做了功能,样式没有细调)html ... [详细]
  • 本文深入探讨了 Java 中的 Serializable 接口,解释了其实现机制、用途及注意事项,帮助开发者更好地理解和使用序列化功能。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
  • 深度学习理论解析与理解
    梯度方向指示函数值增加的方向,由各轴方向的偏导数综合而成,其模长表示函数值变化的速率。本文详细探讨了导数、偏导数、梯度等概念,并结合Softmax函数、卷积神经网络(CNN)中的卷积计算、权值共享及池化操作进行了深入分析。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 本文介绍了在Windows环境下使用pydoc工具的方法,并详细解释了如何通过命令行和浏览器查看Python内置函数的文档。此外,还提供了关于raw_input和open函数的具体用法和功能说明。 ... [详细]
  • 使用Vultr云服务器和Namesilo域名搭建个人网站
    本文详细介绍了如何通过Vultr云服务器和Namesilo域名搭建一个功能齐全的个人网站,包括购买、配置服务器以及绑定域名的具体步骤。文章还提供了详细的命令行操作指南,帮助读者顺利完成建站过程。 ... [详细]
  • 本文讨论了如何根据特定条件动态显示或隐藏文件上传控件中的默认文本(如“未选择文件”)。通过结合CSS和JavaScript,可以实现更灵活的用户界面。 ... [详细]
  • 本文详细介绍了C语言中链表的两种动态创建方法——头插法和尾插法,包括具体的实现代码和运行示例。通过这些内容,读者可以更好地理解和掌握链表的基本操作。 ... [详细]
  • 在维护公司项目时,发现按下手机的某个物理按键后会激活相应的服务,并在屏幕上模拟点击特定坐标点。本文详细介绍了如何使用ADB Shell Input命令来模拟各种输入事件,包括滑动、按键和点击等。 ... [详细]
  • Hadoop入门与核心组件详解
    本文详细介绍了Hadoop的基础知识及其核心组件,包括HDFS、MapReduce和YARN。通过本文,读者可以全面了解Hadoop的生态系统及应用场景。 ... [详细]
  • 尽管使用TensorFlow和PyTorch等成熟框架可以显著降低实现递归神经网络(RNN)的门槛,但对于初学者来说,理解其底层原理至关重要。本文将引导您使用NumPy从头构建一个用于自然语言处理(NLP)的RNN模型。 ... [详细]
  • Struts2 深度解析:第八章 输入验证与内建验证机制
    本章将深入探讨 Struts2 中的输入验证机制,重点介绍基于 XWork 验证框架的内建验证程序,如 required、requiredstring 和 stringlength。这些工具简化了开发者的工作,使得验证逻辑更加高效和易于管理。 ... [详细]
  • 本题探讨如何通过最大流算法解决农场排水系统的设计问题。题目要求计算从水源点到汇合点的最大水流速率,使用经典的EK(Edmonds-Karp)和Dinic算法进行求解。 ... [详细]
  • 毕业设计:基于机器学习与深度学习的垃圾邮件(短信)分类算法实现
    本文详细介绍了如何使用机器学习和深度学习技术对垃圾邮件和短信进行分类。内容涵盖从数据集介绍、预处理、特征提取到模型训练与评估的完整流程,并提供了具体的代码示例和实验结果。 ... [详细]
author-avatar
我爱伟华
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有