基于FPGA的五子棋游戏设计

基于FPGA的五子棋游戏设计
本文基于FPGA设计五子棋游戏,使用按键输入,使用VGA接口输出。五子棋的棋具与围棋相同,棋子分为黑白两色,棋盘为10×10,棋子放置于棋盘线交叉点上。两人对局,各执一色,轮流下一子,先将横、竖或斜线的5个或5个以上同色棋子连成不间断的一排者为胜。
VGA技术介绍:
VGA 接口就是显卡上输出模拟信号的接口,也叫 D-Sub 接口。VGA 接口是一种 D
型口,上面共有 15 针空,分成三排,每排五个。VGA 接口是目前中低端电脑配置上的主流口。
VGA显示中,FPGA需要产生5个信号:R、G、B三基色信号,行同步信号HS,场同步信号VS。以上接口的5个孔对应着我们FPGA中产生的5个重要的信号,其中R、G、B是数据信号;HS、VS是控制信号。下面是VGA 显示模块与CRT 显示器的控制框图:
在这里插入图片描述

像素是产生各种颜色的基本单元。根据物理学中的混色原理,三色发光的亮度比例适当,可呈现白色。适当的调整发光比例可以出现不同的颜色。三基色混色原理示意图如下图所示:
在这里插入图片描述

对于显示器来说,RGB三个信号其实是模拟信号,其电平的高低,可以表示颜色的深浅。利用这个原理,我们就可以产生丰富的色彩。为了控制电压的高低,我们就必须用到DA芯片。例如下图中,FPGA产生RGB三种信号,这时RGB都是多位的数字信号。DA芯片根据数字信号的值,产生不同电压的模拟信号rgb。

在这里插入图片描述

DE1-SoC板上有一个15针D-SUB连接器,用于VGA输出。VGA同步信号直接从Cyclone V SoC FPGA和Analog器件ADV7123三路10位高速视频DAC(仅使用较高的8位)转换从数字到模拟的信号代表三种基本颜色(红色,绿色和蓝色)。 它可以支持高达SXGA标准(1280 * 1024),信号以100MHz传输。 下图显示FPGA和VGA之间连接的信号。

在这里插入图片描述
在这里插入图片描述

显示器采用光栅扫描方式,即轰击荧光屏的电子束在 CRT 屏幕上从左到右(受水平同步信号 HSYNC 控制)、从上到下(受垂直同步信号 VSYNC 控制)做有规律的移动。电子束采用光栅扫描方式,从屏幕左上角一点开始,向右逐点进行扫描,形成一条水平线;到达最右端后,又回到下一条水平线的左端,重复上面的过程;当电子束完成右下角一点的扫描后,形成一帧。此后,电子束又回到左上方起点,开始下一帧的扫描。这种方法也就是常说的逐行扫描显示。
完成一行扫描的时间称为水平扫描时间,其倒数称为行频率;完成一帧(整屏)扫描的时间称为垂直扫描时间,其倒数称为场频率,即刷新一屏的频率,常见的有60Hz,75Hz等等。标准的VGA显示的场频60Hz。
行时序和场时序都需要同步脉冲(Sync a)、显示后沿(Back porch b)、显示时序段(Display interval c)和显示前沿(Front porch d)四部分。VGA工业标准显示模式要求:行同步,场同步都为负极性,即同步脉冲要求是负脉冲。

VGA支持的规格
在这里插入图片描述

在这里插入图片描述

我们以第一个分辨率640/480来分析,其刷新速率是60Hz,每幅图像有525行,每行有800个值。也就是说完成一幅图像约是1s/60=16.6ms,完成一行约为16.6ms/525=31.75us,完成一个像素传送约来31.75us /800=40ns。因此为了方便设计,接口的时候设为25MHz最方便,每个时钟送一个数据。

VGA时序设计

根据上节的介绍,VGA时序要求如下

在这里插入图片描述

VGA时序代码设计

/**********VGA驱动**********/
always@(posedge vga_clk or negedge rst_n)beginif(!rst_n)beginhcount <= 10'd0;endelse if(hcount == 10'd799)beginhcount <= 10'd0;endelse beginhcount <= hcount+1'b1;endendalways@(posedge vga_clk or negedge rst_n)beginif(!rst_n) beginvcount <= 10'd0;endelse if(hcount == 10'd799)beginif(vcount==524)vcount <= 0;elsevcount <= vcount+1;endendalways@(posedge vga_clk or negedge rst_n)beginif(!rst_n)beginhsync <= 1;endelse if(hcount==10'd95)beginhsync <= 1'b1;endelse if(hcount==10'd799)beginhsync <= 1'b0;end
endalways@(posedge vga_clk or negedge rst_n)beginif(!rst_n)beginvsync <= 1'b1;endelse if(vcount<10'd2)beginvsync <= 1'b0;endelse beginvsync <= 1'b1;end
end    

VGA时序仿真结果

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

五子棋设计
为了节约FPGA资源,本文设计并编写了10x10 五子棋的专用算法。棋子半径大小20个像素。光标当前的棋子所在位置使用青色表示,因为屏幕背景色是黑色,棋盘背景色是白色,所以黑白棋子使用紫色和黄色代替。
棋盘设计:
设10x10数组reg [9:0] PLAYED [9:0]; 数组中每一位代表棋盘的一个交叉点。数组的某位为0时,代表无子状态; 为l时,代表有棋子;
设10x10数组reg [9:0] CHESS_COLOR [9:0];表示棋子的颜色。
产生棋盘格的代码如下:

//产生竖长条
always@(posedge vga_clk )
beginif(hcount<=200||hcount>=640)beginv_flag <= 1'b1 ;    v_dat <= 12'h000;//heiend else if(hcount ==240||hcount ==280||hcount ==320||hcount ==360||hcount ==400||hcount ==440||hcount ==480||hcount ==520||hcount==560||hcount==600) beginv_flag <= 1'b1 ;    v_dat <= 12'h000;//heiend else beginv_flag <= 1'b0 ;    v_dat <= 12'hfff;//baiend
end//产生横长条
always@(posedge vga_clk )
beginif(vcount<=50||vcount>=490) beginh_flag <= 1'b1 ;h_dat <= 12'h000;//heiend else if(vcount ==90||vcount ==130||vcount ==170||vcount==210 ||vcount==250||vcount ==290||vcount ==330||vcount==370 ||vcount==410||vcount==450) beginh_flag <= 1'b1 ;    h_dat <= 12'h000;end else beginh_flag <= 1'b0 ;      h_dat <= 12'hfff;end
end  

2)当前棋子光标提示设计:
当前棋子位置提示显示数据为x_data,显示青色。V_dat和H_dat是横竖线的,显示为黑色,y_dat为棋子颜色。

/

/显示模块 
always@(posedge vga_clk)
begin
if(dat_act)beginif(over==0 && over2==0  &&   debug_flag==0    ) beginif(curchess_flag)disp_rgb <=x_dat;    // else if(v_flag)disp_rgb <=v_dat;    // else if(h_flag)disp_rgb <=h_dat;    // else if(chess_flag)disp_rgb <=y_dat;    //  elsedisp_rgb <=12'hfff;    //   end elsedisp_rgb <=z_dat;    			//游戏结束则显示另外画面
end
else begin
disp_rgb = 12'h000;  //VGA区域之外  
end   
end

3)按钮上、下、左、右、落子:

always@(posedge sysclk or negedge rst_n)beginif(!rst_n)beginPLAYED[0]<=10'd0;  PLAYED[1]<=10'd0;  PLAYED[2]<=10'd0;  PLAYED[3]<=10'd0;  PLAYED[4]<=10'd0;  PLAYED[5]<=10'd0;  PLAYED[6]<=10'd0;  PLAYED[7]<=10'd0;  PLAYED[8]<=10'd0;  PLAYED[9]<=10'd0;      CHESS_COLOR[0]<=10'd0;  CHESS_COLOR[1]<=10'd0;  CHESS_COLOR[2]<=10'd0;  CHESS_COLOR[3]<=10'd0;  CHESS_COLOR[4]<=10'd0;  CHESS_COLOR[5]<=10'd0;  CHESS_COLOR[6]<=10'd0;  CHESS_COLOR[7]<=10'd0;  CHESS_COLOR[8]<=10'd0;  CHESS_COLOR[9]<=10'd0;  endelse
begincase(key_en_tem)5'b00100:begin if (ball_x_pos== 10'd240)ball_x_pos<=10'd600;else ball_x_pos<=ball_x_pos-10'd40;end5'b01000:begin if (ball_x_pos==10'd600)ball_x_pos<=10'd240;else ball_x_pos<= ball_x_pos+10'd40;end 5'b00001: begin if (ball_y_pos== 10'd90)ball_y_pos<=10'd450;else ball_y_pos<=ball_y_pos-10'd40;                                                end5'b00010: beginif(ball_y_pos== 10'd450)ball_y_pos<=10'd90;else         ball_y_pos<=ball_y_pos+10'd40; end      5'b10000: beginif(PLAYED[current_y][current_x]==0)beginPLAYED[current_y][current_x]<=1;                                     
CHESS_COLOR[current_y][current_x]<=change_color;endend     

(4)判胜:
横竖正斜反斜四个方向出现连续落子,且颜色一样,输出胜利提示。


```c
//输赢状态判断1
always@(posedge vga_clk )if(!rst_n)beginover<=0; endelse
beginif (  (scan_x<=5&&CHESS_COLOR[scan_y][scan_x]==1&&CHESS_COLOR[scan_y][scan_x+1]==1&&CHESS_COLOR[scan_y][scan_x+2]==1&&CHESS_COLOR[scan_y][scan_x+3]==1&&CHESS_COLOR[scan_y][scan_x+4]==1)//heng&&   (PLAYED[scan_y][scan_x]==1&&PLAYED[scan_y][scan_x+1]==1&&PLAYED[scan_y][scan_x+2]==1&&PLAYED[scan_y][scan_x+3]==1&&PLAYED[scan_y][scan_x+4]==1)       )     beginover<=1;endelse if( (scan_y<=5&&CHESS_COLOR[scan_y][scan_x]==1&&CHESS_COLOR[scan_y+1][scan_x]==1&&CHESS_COLOR[scan_y+2][scan_x]==1&&CHESS_COLOR[scan_y+3][scan_x]==1&&CHESS_COLOR[scan_y+4][scan_x]==1)//shu&&     (PLAYED[scan_y][scan_x]==1&&PLAYED[scan_y+1][scan_x]==1&&PLAYED[scan_y+2][scan_x]==1&&PLAYED[scan_y+3][scan_x]==1&&PLAYED[scan_y+4][scan_x]==1) )beginover<=1;endelse if(  (scan_x<=5&&scan_y<=5&&CHESS_COLOR[scan_y][scan_x]==1&&CHESS_COLOR[scan_y+1][scan_x+1]==1&&CHESS_COLOR[scan_y+2][scan_x+2]==1&&CHESS_COLOR[scan_y+3][scan_x+3]==1&&CHESS_COLOR[scan_y+4][scan_x+4]==1)//chenggong&&  (PLAYED[scan_y][scan_x]==1&&PLAYED[scan_y+1][scan_x+1]==1&&PLAYED[scan_y+2][scan_x+2]==1&&PLAYED[scan_y+3][scan_x+3]==1&&PLAYED[scan_y+4][scan_x+4]==1) )over<=1;else if ( (scan_y<=5&&scan_x>=4&&CHESS_COLOR[scan_y][scan_x]==1&&CHESS_COLOR[scan_y+1][scan_x-1]==1&&CHESS_COLOR[scan_y+2][scan_x-2]==1&&CHESS_COLOR[scan_y+3][scan_x-3]==1&&CHESS_COLOR[scan_y+4][scan_x-4]==1)&& (PLAYED[scan_y][scan_x]==1&&PLAYED[scan_y+1][scan_x-1]==1&&PLAYED[scan_y+2][scan_x-2]==1&&PLAYED[scan_y+3][scan_x-3]==1&&PLAYED[scan_y+4][scan_x-4]==1) )over<=1;
end        

```c
//输赢状态判断2 yellow
always@(posedge vga_clk )if(!rst_n)beginover2<=0;    debug_flag<=0;   endelse
beginif( (scan_x<=5&&CHESS_COLOR[scan_y][scan_x]==0&&CHESS_COLOR[scan_y][scan_x+1]==0&&CHESS_COLOR[scan_y][scan_x+2]==0&&CHESS_COLOR[scan_y][scan_x+3]==0&&CHESS_COLOR[scan_y][scan_x+4]==0)//heng&&   (PLAYED[scan_y][scan_x]==1&&PLAYED[scan_y][scan_x+1]==1&&PLAYED[scan_y][scan_x+2]==1&&PLAYED[scan_y][scan_x+3]==1&&PLAYED[scan_y][scan_x+4]==1)       ) over2<=1;else if( (scan_y<=5&&CHESS_COLOR[scan_y][scan_x]==0&&CHESS_COLOR[scan_y+1][scan_x]==0&&CHESS_COLOR[scan_y+2][scan_x]==0&&CHESS_COLOR[scan_y+3][scan_x]==0&&CHESS_COLOR[scan_y+4][scan_x]==0)//shu&&     (PLAYED[scan_y][scan_x]==1&&PLAYED[scan_y+1][scan_x]==1&&PLAYED[scan_y+2][scan_x]==1&&PLAYED[scan_y+3][scan_x]==1&&PLAYED[scan_y+4][scan_x]==1) )beginover2<=1;debug_flag<=1;  endelse if(  (scan_x<=5&&scan_y<=5&&CHESS_COLOR[scan_y][scan_x]==0&&CHESS_COLOR[scan_y+1][scan_x+1]==0&&CHESS_COLOR[scan_y+2][scan_x+2]==0&&CHESS_COLOR[scan_y+3][scan_x+3]==0&&CHESS_COLOR[scan_y+4][scan_x+4]==0)//chenggong&&  (PLAYED[scan_y][scan_x]==1&&PLAYED[scan_y+1][scan_x+1]==1&&PLAYED[scan_y+2][scan_x+2]==1&&PLAYED[scan_y+3][scan_x+3]==1&&PLAYED[scan_y+4][scan_x+4]==1) )over2<=1;else if ( (scan_y<=5&&scan_x>=4&&CHESS_COLOR[scan_y][scan_x]==0&&CHESS_COLOR[scan_y+1][scan_x-1]==0&&CHESS_COLOR[scan_y+2][scan_x-2]==0&&CHESS_COLOR[scan_y+3][scan_x-3]==0&&CHESS_COLOR[scan_y+4][scan_x-4]==0)&& (PLAYED[scan_y][scan_x]==1&&PLAYED[scan_y+1][scan_x-1]==1&&PLAYED[scan_y+2][scan_x-2]==1&&PLAYED[scan_y+3][scan_x-3]==1&&PLAYED[scan_y+4][scan_x-4]==1) )over2<=1;  
end        

(5)win画面:在任一方胜利后VGA显示win三个字母画面输出胜利提示。根据棋子颜色win显示不同颜色以显示

//产生win画面1
always@(posedge vga_clk )
beginif(((hcount>=240&&hcount<=260)||(hcount>=300&&hcount<=320)||(hcount>=360&&hcount<=380)||(hcount>=420&&hcount<=440)||(hcount>=480&&hcount<=500)||(hcount>=560&&hcount<=580))&&(vcount >=200&&vcount <=360))beginz_flag <= 1'b1;if(over)z_dat <= 12'hf0f;elsez_dat <= 12'hff0;    end else if(hcount>=260&&hcount<=380&&vcount>=340&&vcount <=360) beginz_flag <= 1'b1;if(over)z_dat <= 12'hf0f;elsez_dat <= 12'hff0;    end else if(hcount>=500&&hcount<=560&&vcount>=200&&vcount <=220) beginz_flag <= 1'b1;if(over)z_dat <= 12'hf0f;elsez_dat <= 12'hff0;    end else beginz_flag <= 1'b0;z_dat <= 12'hfff;end
end       

遇到的问题:
1 棋子颜色显示不稳定,下棋时棋子颜色随机产生不受控制。原因怀疑是时钟一开始是分频产生的25M,后面换成pll锁相环后问题果然解决。
一开始的时钟产生方法:

always@(posedge sysclk)
beginvga_clk <= ~vga_clk;
end   
优化后的:pll pll(.inclk0(sysclk),.c0(vga_clk)
);

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

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

相关文章

牛客 算法题 golang语言实现

题目 HJ101 输入整型数组和排序标识&#xff0c;对其元素按照升序或降序进行排序 描述 输入整型数组和排序标识&#xff0c;对其元素按照升序或降序进行排序数据范围&#xff1a; 1 ≤ &#xfffd; ≤ 10001≤n≤1000 &#xff0c;元素大小满足 0 ≤ &#xfffd; &#…

有一种浪漫,叫接触Linux

大家好&#xff0c;我是五月。 嵌入式开发 嵌入式开发产品必须依赖硬件和软件。 硬件一般使用51单片机&#xff0c;STM32、ARM&#xff0c;做成的产品以平板&#xff0c;手机&#xff0c;智能机器人&#xff0c;智能小车居多。 软件用的当然是以linux系统为蓝本&#xff0c…

五分钟 k8s 实战-应用探针

Probe.png 今天进入 kubernetes 的运维部分&#xff08;并不是运维 kubernetes&#xff0c;而是运维应用&#xff09;&#xff0c;其实日常我们大部分使用 kubernetes 的功能就是以往运维的工作&#xff0c;现在云原生将运维和研发关系变得更紧密了。 今天主要讲解 Probe 探针相…

SpringCloud 微服务全栈体系(十七)

第十一章 分布式搜索引擎 elasticsearch 七、搜索结果处理 搜索的结果可以按照用户指定的方式去处理或展示。 1. 排序 elasticsearch 默认是根据相关度算分&#xff08;_score&#xff09;来排序&#xff0c;但是也支持自定义方式对搜索结果排序。可以排序字段类型有&#…

C语言进阶指南(11)(指针数组与二维数组)

*欢迎来到博主的专栏——C语言进阶指南 博主id&#xff1a;reverie_ly 文章目录 N级指针指针数组指针数组与二维数组数组指针作为函数的参数 N级指针 指针变量是一个存放地址的变量&#xff0c;在C语言中&#xff0c;每个变量都会有一个地址值。所以指针变量也有一个地址。 …

WIN10 x86环境部署ARM虚拟机(银河麒麟)

我们经常使用的是x86架构的cpu&#xff0c;而对于不同cpu架构的arm架构的操作系统&#xff0c;我们可以通过QEMU模拟器来进行模拟一个arm环境 1、部署前的准备 arm的镜像&#xff1a; 以此镜像为例&#xff1a;Kylin-Server-10-SP2-aarch64-Release-Build09-20210524.iso QE…

【C++ Primer Plus学习记录】while循环

while循环是没有初始化和更新部分的for循环&#xff0c;它只有测试条件和循环体&#xff1a; while(test-condition)dody 首先&#xff0c;程序计算圆括号内的测试条件表达式。如果该表达式为true&#xff0c;则执行循环体内的语句。与for循环一样&#xff0c;循环体也由一条…

IO和NIO的区别 BIO,NIO,AIO 有什么区别? Files的常用方法都有哪些?

文章目录 IO和NIO的区别BIO,NIO,AIO 有什么区别?Files的常用方法都有哪些&#xff1f; 今天来对java中的io, nio, bio, aio进行了解&#xff0c;有何区别。 IO和NIO的区别 NIO与IO区别 IO是面向流的&#xff0c;NIO是面向缓冲区的Java IO面向流意味着每次从流中读一个或多个字…

数据结构-单链表

文章目录 单链表概念链接存储方法头指针head和终端结点链接过程单链表的优缺点&#xff1a;实现代码一览 单链表概念 链表中的数据是以结点来表示的&#xff0c;每个结点的构成&#xff1a;元素(数据元素的映象) 指针(指示后继元素存储位置)&#xff0c;元素就是存储数据的存储…

经典滑动窗口试题(二)

&#x1f4d8;北尘_&#xff1a;个人主页 &#x1f30e;个人专栏:《Linux操作系统》《经典算法试题 》《C》 《数据结构与算法》 ☀️走在路上&#xff0c;不忘来时的初心 文章目录 一、水果成篮1、题目讲解2、讲解算法思路3、代码实现 二、找到字符串中所有字母异位词1、题目…

合并区间[中等]

一、题目 以数组intervals表示若干个区间的集合&#xff0c;其中单个区间为intervals[i] [starti, endi]。请你合并所有重叠的区间&#xff0c;并返回一个不重叠的区间数组&#xff0c;该数组需恰好覆盖输入中的所有区间。 示例 1&#xff1a; 输入&#xff1a;intervals […

初探HarmonyOS路由跳转

最近的鸿蒙新闻也是很大声势&#xff0c;鸿蒙的纯血版一出&#xff0c;各大互联网大厂都坐不住了&#xff0c;纷纷加入其中。这意味鸿蒙将来会取代大部分Android用户&#xff0c;这也是程序员的一篇大好前程。如今的Android开发行业已经夕阳西下了。 网上有关HarmonyOS的资料几…

SVD recommendation systems

SVD recommendation systems 为什么在推荐系统中使用SVD 一个好的推荐系统一定有小的RMSE R M S E 1 m ∑ i 1 m ( Y i − f ( x i ) 2 RMSE \sqrt{\frac{1}{m} \sum_{i1}^m(Y_i-f(x_i)^2} RMSEm1​i1∑m​(Yi​−f(xi​)2 ​ 希望模型能够在已知的ratings上有好的结果的…

springboot整合redis+自定义注解+反射+aop实现分布式锁

1.定义注解 import java.lang.annotation.*; import java.util.concurrent.TimeUnit;/** Author: best_liu* Description:* Date: 16:13 2023/9/4* Param * return **/ Retention(RetentionPolicy.RUNTIME) Target({ElementType.METHOD}) Documented public interface RedisLo…

Ubuntu系统Springboot项目Nginx安装(编译安装方式)

1.下载 nginx官网下载 Index of /download/ 2.解压 这里我下载的1.25.3版本&#xff0c;系统是ubuntu 解压 tar -zxvf nginx-1.25.3.tar.gz 3.编译安装 安装前需要执行安装一些系统依赖 3.1安装PCRE库 ubuntu&#xff1a;执行以下命令 sudo apt-get install libpcre…

MOSFET安全工作区域SOA

Safe Operating Area&#xff08;SOA&#xff09;即安全工作区&#xff1a;描述了当MOSFET工作在饱和区时可以处理的最大功率。超出安全工作区&#xff0c;则可能导致元件损坏。 SOA分为五个单独的界限&#xff0c;分别是RDS(on)限制 On Resistance&#xff08;RDS(on)&#…

Kafka系列 - Kafka一篇入门

Kafka是一个分布式流式处理平台。很多分布式处理系统&#xff0c;例如Spark&#xff0c;Flink等都支持与Kafka集成。 Kafka使用场景 消息系统&#xff1a;Kafka实现了消息顺序性保证和回溯消费。存储系统&#xff1a;Kafka把消息持久化到磁盘&#xff0c;相比于其他基于内存的…

⑨【Stream】Redis流是什么?怎么用?: Stream [使用手册]

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ ⑨Redis Stream基本操作命令汇总 一、Redis流 …

苹果mac屏幕投屏镜像工具AirServer2024

airserver 是什么软件&#xff1f;AirServer 是一款 Airplay Mac屏幕镜像应用&#xff0c;AirServer可以通过 mac 实时接收iPhone、iPad以及Android设备的实时屏幕画面。AirServer 可以将一个简单的大屏幕或投影仪变成一个通用的屏幕镜像接收器。在您的大屏幕上启用 AirServer …

八股文面试day6

什么是代理&#xff1f;为什么要用动态代理&#xff1f; 代理模式大概意思是&#xff1a;为其他对象提供一个代理项或者是占位符&#xff0c;以控制对这个对象的访问 代理模式核心思想&#xff1a;创建一个代理对象&#xff0c;在客户端和目标对象之间的一个中介&#xff0c;…