FPGA课设——交通信号控制器的VHDL设计

  • 1、设计任务及要求:
  • 2、参考设计方案
  • 3、采用 VHDL 语言输入的方式实现交通信号灯控制器
  • 4、设计方案
  • 5、自定义交通灯主从控制时间


1、设计任务及要求:

设计任务:模拟十字路口交通信号灯的工作过程,利用实验板上的两组红、黄、绿LED 作为交通信号灯,设计一个交通信号灯控制器。要求:

(1)交通灯从绿变红时,有4秒黄灯亮的间隔时间;

(2)交通灯红变绿是直接进行的,没有间隔时间;

(3)主干道上的绿灯时间为40秒,支干道的绿灯时间为20秒;

(4)在任意时间,显示每个状态到该状态结束所需的时间。

jquery 红绿灯 红绿灯vhdl设计_jquery 红绿灯


jquery 红绿灯 红绿灯vhdl设计_ide_02

设计要求:
(1)采用VHDL语言编写程序,并在QUARTUSII工具平台中进行仿真,下载到EDA实验箱进行验证。
(2)编写设计报告,要求包括方案选择、程序清单、调试过程、测试结果及心得体会。

2、参考设计方案

jquery 红绿灯 红绿灯vhdl设计_ide_03

3、采用 VHDL 语言输入的方式实现交通信号灯控制器

jquery 红绿灯 红绿灯vhdl设计_jquery 红绿灯_04


该程序由 7 个进程组成,进程 P1 和 P2 将 CLK 信号分频后产生 1 秒信号,进程 P3、P4、 P5构成两个带有预置数功能的十进制计数器,其中P4 产生允许十位计数器计数的控制信号。 进程P6实现状态转换和产生状态转换的控制信号,进程P7产生次态信号和信号灯输出信号, 以及每一个状态的时间值。

4、设计方案

LIBRARY IEEE;
USE IEEE.STD_LOGIC_1164.ALL;
USE IEEE.STD_LOGIC_UNSIGNED.ALL;

   ENTITY TRAFFIC_LIGHT IS
      PORT(CLK,RST: IN STD_LOGIC;
			LED: OUT STD_LOGIC_VECTOR(5 DOWNTO  0);	--定义6个LED灯
			WideSEGH: OUT STD_LOGIC_VECTOR(3 DOWNTO  0);  --主干倒计时的十位
			WideSEGL: OUT STD_LOGIC_VECTOR(3 DOWNTO  0);  --主干倒计时的个位
			NSEGH: OUT STD_LOGIC_VECTOR(3 DOWNTO  0);  --支干倒计时的十位
			NSEGL: OUT STD_LOGIC_VECTOR(3 DOWNTO  0));  --支干倒计时的个位
END ENTITY TRAFFIC_LIGHT;

ARCHITECTURE bhv OF TRAFFIC_LIGHT IS
    TYPE state IS(S0,S1,S2,S3);  --枚举状态类型
    SIGNAL current_s,next_s:state;
    SIGNAL COUNT68:STD_LOGIC_VECTOR(7 DOWNTO  0);
      BEGIN
COUNT: PROCESS(RST,CLK)                --辅助进程
           BEGIN
				IF RST= '1' THEN    --异步清零
					COUNT68<="00000000";
				ELSIF CLK'EVENT AND CLK='1' THEN
					IF COUNT68<"01000011" THEN    --B01000011=67设定68进制计数器
						COUNT68<=COUNT68+1;
					ELSE 
						COUNT68<="00000000"; 
					END IF;
				END IF;
		END PROCESS;

 REG: PROCESS (RST,CLK)                   --主控时序进程,用于机械地控制状态转换 
	BEGIN
		IF RST= '1' THEN   
			current_s <= s0;      
		ELSIF clk='1' AND clk'EVENT THEN   
			current_s <= next_s; 
		END IF;
	END PROCESS;     

COM:PROCESS(current_s, COUNT68)         --主控组合进程,用于设定状态转换的下一个状态
VARIABLE WESEG,NSSEG:STD_LOGIC_VECTOR(7 DOWNTO  0);
	BEGIN
		CASE current_s IS                     
			WHEN s0 => LED<= "010100";--主干绿灯亮,支干红灯亮
				WESEG:=39-COUNT68; --主干绿灯倒计时,用于数码管显示
				NSSEG:=43-COUNT68; --支干红灯倒计时 
				IF COUNT68 ="00100111" THEN   --B00100111=39  控制状态转变   
					next_s<=s1;                    
				ELSE 
					next_s<=s0;    
				END IF;
			WHEN s1 => LED <= "001100"; --主干黄灯,支干红灯
				WESEG:=43-COUNT68;
				NSSEG:=43-COUNT68; 
				IF COUNT68 = "00101011" THEN  --B00101011=43
					next_s<=s2;    
				ELSE
					next_s<=s1;   
				END IF;
			WHEN s2 => LED <= "100010";  --主干红灯,支干绿灯
				WESEG:=67-COUNT68;
				NSSEG:=63-COUNT68; 
				IF COUNT68 ="00111111" THEN  --B00111111=64
					next_s <= s3;
				ELSE  
					next_s <= s2; 
				END IF;
			WHEN s3 => LED <= "100001";  --主干红灯,支干黄灯
				WESEG:=67-COUNT68;
				NSSEG:=67-COUNT68; 
				IF COUNT68 ="01000011" THEN --B01000011=67 
					next_s<= s0; 
				ELSE  
					next_s <= s3;   
				END IF;
			WHEN OTHERS=> LED <="100100"; --否则都不能通行
    END case;
    
	IF WESEG>39 THEN --译码器的功能:将当前的倒计时转换成十进制,低四位和高四位分别是数码管的个位与十位,
						--38+18=56D=38H=00111000B
						--使数码管按照当前的计数(十进制)按二进制计数进行倒计时
		WESEG:=WESEG+24;
	ELSIF WESEG>29 THEN 
		WESEG:=WESEG+18;
	ELSIF WESEG>19 THEN
		WESEG:=WESEG+12;
	ELSIF WESEG>9 THEN
		WESEG:=WESEG+6;
	ELSE NULL;
	END IF;
	
	IF NSSEG>39 THEN
		NSSEG:=NSSEG+24;
	ELSIF NSSEG>29 THEN 
		NSSEG:=NSSEG+18;
	ELSIF NSSEG>19 THEN 
		NSSEG:=NSSEG+12;
	ELSIF NSSEG>9 THEN 
		NSSEG:=NSSEG+6;
	ELSE NULL;
	END IF;
	WideSEGH<=WESEG(7 DOWNTO 4);
	WideSEGL<=WESEG(3 DOWNTO 0);
	NSEGH<=NSSEG(7 DOWNTO 4);
	NSEGL<=NSSEG(3 DOWNTO 0);
 END PROCESS;  
END ARCHITECTURE bhv;

状态图:

jquery 红绿灯 红绿灯vhdl设计_ide_05

顶层文件图:

jquery 红绿灯 红绿灯vhdl设计_设计方案_06

波形仿真图:

jquery 红绿灯 红绿灯vhdl设计_ide_07

5、自定义交通灯主从控制时间

jquery 红绿灯 红绿灯vhdl设计_VHDL_08

LIBRARY IEEE;  
USE IEEE.STD_LOGIC_1164.ALL; 
USE IEEE.STD_LOGIC_UNSIGNED.ALL; 
  
ENTITY TRAFFIC_LAMP IS   
PORT(CLK,RST: IN STD_LOGIC; 
     LEDCTRL: OUT STD_LOGIC_VECTOR(5 DOWNTO 0); 
     WESEGH: OUT STD_LOGIC_VECTOR(3 DOWNTO 0); 
     WESEGL: OUT STD_LOGIC_VECTOR(3 DOWNTO 0); 
     NSSEGH: OUT STD_LOGIC_VECTOR(3 DOWNTO 0); 
     NSSEGL: OUT STD_LOGIC_VECTOR(3 DOWNTO 0)); 
END ENTITY TRAFFIC_LAMP; 
  
ARCHITECTURE behav OF TRAFFIC_LAMP IS 
  TYPE FSM_ST IS(S0,S1,S2,S3); 
  SIGNAL current_s,next_s:FSM_ST;  
  SIGNAL COUNT68:STD_LOGIC_VECTOR(7 DOWNTO 0); 
BEGIN  
  COUNT: PROCESS(RST,CLK) --辅助进程
  BEGIN  
  IF RST= '1' THEN COUNT68<="00000000";--检测异步复位信号
  ELSIF CLK'EVENT AND CLK='1' THEN 
  IF COUNT68<"(z-1变成二进制后代入)" THEN                                    -- 1 
    COUNT68<=COUNT68+1;  
  ELSE COUNT68<="00000000"; 
  END IF; 
END IF; 
END PROCESS; 
  
 
REG: PROCESS (RST,CLK) --主控时序进程
     BEGIN  
     IF RST= '1' THEN current_s <= s0; --检测异步复位信号
     ELSIF clk='1' AND clk'EVENT THEN 
  	current_s <= next_s; 
     END IF;  
     END PROCESS; 
 

COM:PROCESS(current_s, COUNT68) --主控组合进程
  VARIABLE WESEG,NSSEG:STD_LOGIC_VECTOR(7 DOWNTO 0); 
  BEGIN  
    CASE current_s IS  
       WHEN s0 => LEDCTRL<= "010100";  
	WESEG:=(x-1)-COUNT68;NSSEG:=(x+4-1)-COUNT68;                 --2,3
 	IF COUNT68 ="(x-1换算成二进制代入)" THEN next_s<=s1;               --4
	ELSE next_s<=s0; 
 	END IF;  
      WHEN s1 => LEDCTRL<= "001100";  
 	WESEG:=(x+4-1)-COUNT68;NSSEG:=(x+4-1)-COUNT68;                   --5,6
	IF COUNT68 = "(x+4-1换算成二进制)" THEN next_s<=s2; 		 --7
	ELSE next_s<=s1; 
	END IF;  
      WHEN s2 => LEDCTRL<= "100010"; 
 	WESEG:=(z-1)-COUNT68;NSSEG:=(z-5)-COUNT68;			 --8,9 
	IF COUNT68 ="(z-5换算成二进制)" THEN next_s <= s3; 
	ELSE next_s <= s2; 
	END IF;  
      WHEN s3 => LEDCTRL <= "100001";  
	WESEG:=(z-1)-COUNT68;NSSEG:=(z-1)-COUNT68; 
 	IF COUNT68 ="(z-1换算成二进制)" THEN next_s<= s0; 
	ELSE next_s <= s3; 
	END IF; 
    END case; 

 
--如果x>40的话,要改下面的内容
--例:x=67
--
--IF WESEG>59 THEN WESEG:=WESEG+36;		--后面依次加6
--	ELSIF WESEG>49 THEN WESEG:=WESEG+30;
--	ELSIF WESEG>39 THEN WESEG:=WESEG+24;
--	ELSIF WESEG>29 THEN WESEG:=WESEG+18; 
--	ELSIF WESEG>19 THEN WESEG:=WESEG+12; 
--	ELSIF WESEG>9 THEN WESEG:=WESEG+6; 
--	ELSE NULL;
	
IF WESEG>39 THEN WESEG:=WESEG+24;  
	ELSIF WESEG>29 THEN WESEG:=WESEG+18; 
	ELSIF WESEG>19 THEN WESEG:=WESEG+12; 
	ELSIF WESEG>9 THEN WESEG:=WESEG+6; 
	ELSE NULL; 
END IF;  

--这个修改方法跟上面一样
IF NSSEG>39 THEN NSSEG:=NSSEG+24;  
	ELSIF NSSEG>29 THEN NSSEG:=NSSEG+18; 
	ELSIF NSSEG>19 THEN NSSEG:=NSSEG+12; 
	ELSIF NSSEG>9 THEN NSSEG:=NSSEG+6; 
	ELSE NULL; 
END IF; 
 
WESEGH<=WESEG(7 DOWNTO 4);WESEGL<=WESEG(3 DOWNTO 0); 
NSSEGH<=NSSEG(7 DOWNTO 4);NSSEGL<=NSSEG(3 DOWNTO 0);  
END PROCESS; 
  
END ARCHITECTURE behav;

其他课设题目:
1、AC-DC数字稳压电源设计与仿真2、步进式增益调节语音放大器3、函数发生器课程设计(Multisim仿真+PCB实物)4、嵌入式+人工智能+物联网开发常用的开发工具5、数字交通灯设计(Multisim仿真+PCB实物)6、数字钟—VHDL设计