集成电路技术分享

 找回密码
 我要注册

QQ登录

只需一步,快速开始

搜索
查看: 1094|回复: 0

RISC CPU结构

[复制链接]
fpga_feixiang 发表于 2019-10-5 14:52:10 | 显示全部楼层 |阅读模式
RISC_CPU是一个复杂的数字逻辑电路,但是它的基本部件的逻辑并不复杂。可把它分成八个基本部件:
时钟发生器
指令寄存器
累加器
RISC CPU算术逻辑运算单元
数据控制器
状态控制器
程序计数器
地址多路器
其中时钟发生器利用外来时钟信号进行分频生成一系列时钟信号,送往其他部件用作时钟信号。各部件之间的相互操作关系则由状态控制器来控制。下面逐一介绍各部件的具体结构和逻辑关系。
1 时钟发生器

         时钟发生器 clkgen 利用外来时钟信号clk 来生成一系列时钟信号clk1、fetch、alu_clk 送往CPU的其他部件。其中fetch是外来时钟 clk 的八分频信号。利用fetch的上升沿来触发CPU控制器开始执行一条指令,同时fetch信号还将控制地址多路器输出指令地址和数据地址。clk1信号用作指令寄存器、累加器、状态控制器的时钟信号。alu_clk 则用于触发算术逻辑运算单元。时钟发生器clkgen的波形见下图所示:

其VerilogHDL 程序见下面的模块:
复制代码
1 module clk_gen (clk,reset,clk1,clk2,clk4,fetch,alu_clk);
2 input clk,reset;
3 output clk1,clk2,clk4,fetch,alu_clk;
4 wire clk,reset;
5 reg clk2,clk4,fetch,alu_clk;
6 reg[7:0] state;
7 parameter S1 = 8'b00000001,
8     S2 = 8'b00000010,
9     S3 = 8'b00000100,
10     S4 = 8'b00001000,
11     S5 = 8'b00010000,
12     S6 = 8'b00100000,
13     S7 = 8'b01000000,
14     S8 = 8'b10000000,
15     idle = 8'b00000000;
16 assign clk1 = ~clk;
17
18 always @(negedge clk)
19     if(reset)
20         begin
21             clk2 <= 0;
22             clk4 <= 1;
23             fetch <= 0;
24             alu_clk <= 0;
25             state <= idle;
26         end
27     else
28         begin
29             case(state)
30             S1:
31                 begin
32                     clk2 <= ~clk2;
33                     alu_clk <= ~alu_clk;
34                     state <= S2;
35                 end
36             S2:
37                 begin
38                     clk2 <= ~clk2;
39                     clk4 <= ~clk4;
40                     alu_clk <= ~alu_clk;
41                     state <= S3;
42                 end
43             S3:
44                 begin
45                     clk2 <= ~clk2;
46                     state <= S4;
47                 end
48             S4:
49                 begin
50                     clk2 <= ~clk2;
51                     clk4 <= ~clk4;
52                     fetch <= ~fetch;
53                     state <= S5;
54                 end
55             S5:
56                 begin
57                     clk2 <= ~clk2;
58                     state <= S6;
59                 end
60             S6:
61                 begin
62                     clk2 <= ~clk2;
63                     clk4 <= ~clk4;
64                     state <= S7;
65                 end
66             S7:
67                 begin
68                     clk2 <= ~clk2;
69                     state <= S8;
70                 end
71             S8:
72                 begin
73                     clk2 <= ~clk2;
74                     clk4 <= ~clk4;
75                     fetch <= ~fetch;
76                     state <= S1;
77                 end
78             idle: state <= S1;
79             default: state <= idle;
80         endcase
81     end
82 endmodule
83 //--------------------------------------------------------------------------------
复制代码
          由于在时钟发生器的设计中采用了同步状态机的设计方法,不但使clk_gen模块的源程序可以被各种综合器综合,也使得由其生成的clk1、clk2、clk4、fetch、alu_clk 在跳变时间同步性能上有明显的提高,为整个系统的性能提高打下了良好的基础。
2 指令寄存器

       顾名思义,指令寄存器用于寄存指令。
       指令寄存器的触发时钟是clk1,在clk1的正沿触发下,寄存器将数据总线送来的指令存入高8位或低8位寄存器中。但并不是每个clk1的上升沿都寄存数据总线的数据,因为数据总线上有时传输指令,有时传输数据。什么时候寄存,什么时候不寄存由CPU状态控制器的load_ir信号控制。load_ir信号通过ena 口输入到指令寄存器。复位后,指令寄存器被清为零。
       每条指令为2个字节,即16位。高3位是操作码,低13位是地址。(CPU的地址总线为13位,寻址空间为8K字节。)本设计的数据总线为8位,所以每条指令需取两次。先取高8位,后取低8位。而当前取的是高8位还是低8位,由变量state记录。state为零表示取的高8位,存入高8位寄存器,同时将变量state置为1。下次再寄存时,由于state为1,可知取的是低8位,存入低8位寄存器中。
        其VerilogHDL 程序见下面的模块:
复制代码
1 //---------------------------------------------------------------
2 module register(opc_iraddr,data,ena,clk1,rst);
3 output [15:0] opc_iraddr;
4 input [7:0] data;
5 input ena, clk1, rst;
6 reg [15:0] opc_iraddr;
7 reg state;
8
9 always @(posedge clk1)
10 begin
11     if(rst)
12         begin
13             opc_iraddr<=16'b0000_0000_0000_0000;
14             state<=1'b0;
15         end
16     else
17         begin
18             if(ena) //如果加载指令寄存器信号load_ir到来,
19                 begin //分两个时钟每次8位加载指令寄存器
20                     casex(state) //先高字节,后低字节
21                         1’b0: begin
22                             opc_iraddr[15:8]<=data;
23                             state<=1;
24                         end
25                         1’b1: begin
26                             opc_iraddr[7:0]<=data;
27                             state<=0;
28                         end
29                         default: begin
30                             opc_iraddr[15:0]<=16'bxxxxxxxxxxxxxxxx;
31                             state<=1'bx;
32                         end
33                     endcase
34                 end
35             else
36                 state<=1'b0;
37         end
38 end
39 endmodule
40 //--------------------------------------------------------
复制代码
3.累加器

           累加器用于存放当前的结果,它也是双目运算其中一个数据来源。复位后,累加器的值是零。当累加器通过ena口收到来自CPU状态控制器load_acc信号时,在clk1时钟正跳沿时就收到来自于数据总线的数据。
           其VerilogHDL 程序见下面的模块:
复制代码
1 //--------------------------------------------------------------
2 module accum( accum, data, ena, clk1, rst);
3 output[7:0]accum;
4 input[7:0]data;
5 input ena,clk1,rst;
6 reg[7:0]accum;
7
8 always@(posedge clk1)
9     begin
10         if(rst)
11             accum<=8'b0000_0000; //Reset
12         else
13             if(ena) //当CPU状态控制器发出load_acc信号
14                 accum<=data; //Accumulate
15     end
16 endmodule
复制代码
4.算术运算器


       算术逻辑运算单元 根据输入的8种不同操作码分别实现相应的加、与、异或、跳转等8种基本操作运算。利用这几种基本运算可以实现很多种其它运算以及逻辑判断等操作。
       其VerilogHDL 程序见下面的模块:
复制代码
1 //------------------------------------------------------------------------------
2 module alu (alu_out, zero, data, accum, alu_clk, opcode);
3 output [7:0]alu_out;
4 output zero;
5 input [7:0] data, accum;
6 input [2:0] opcode;
7 input alu_clk;
8 reg [7:0] alu_out;
9
10 parameter HLT =3’b000,
11     SKZ =3’b001,
12     ADD =3’b010,
13     ANDD =3’b011,
14     XORR =3’b100,
15     LDA =3’b101,
16     STO =3’b110,
17     JMP =3’b111;
18
19 assign zero = !accum;
20 always @(posedgealu_clk)
21     begin //操作码来自指令寄存器的输出opc_iaddr<15..0>的低3位
22         casex (opcode)
23             HLT: alu_out<=accum;
24             SKZ: alu_out<=accum;
25             ADD: alu_out<=data+accum;
26             ANDD: alu_out<=data&accum;
27             XORR: alu_out<=data^accum;
28             LDA: alu_out<=data;
29             STO: alu_out<=accum;
30             JMP: alu_out<=accum;
31             default: alu_out<=8'bxxxx_xxxx;
32         endcase
33     end
34 endmodule
35 //----------------------------------------------------------------------------
复制代码
5.数据控制器


         数据控制器的作用是控制累加器数据输出,由于数据总线是各种操作时传送数据的公共通道,不同的情况下传送不同的内容。有时要传输指令,有时要传送RAM区或接口的数据。累加器的数据只有在需要往RAM区或端口写时才允许输出,否则应呈现高阻态,以允许其它部件使用数据总线。 所以任何部件往总线上输出数据时,都需要一控制信号。而此控制信号的启、停,则由CPU状态控制器输出的各信号控制决定。数据控制器何时输出累加器的数据则由状态控制器输出的控制信号datactl_ena决定。
        其VerilogHDL 程序见下面的模块:
复制代码
1 //--------------------------------------------------------------------
2 module datactl (data,in,data_ena);
3 output [7:0]data;
4 input [7:0]in;
5 input data_ena;
6
7 assign data = (data_ena)? In : 8'bzzzz_zzzz;
8
9 endmodule
10 //--------------------------------------------------------------------
复制代码
6.地址多路器


        地址多路器用于选择输出的地址是PC(程序计数)地址还是数据/端口地址。每个指令周期的前4个时钟周期用于从ROM中读取指令,输出的应是PC地址。后4个时钟周期用于对RAM或端口的读写,该地址由指令中给出。地址的选择输出信号由时钟信号的8分频信号fetch提供。
        其VerilogHDL 程序见下面的模块:
复制代码
1 //------------------------------------------------------------------------------
2 module adr(addr,fetch,ir_addr,pc_addr);
3 output [12:0] addr;
4 input [12:0] ir_addr, pc_addr;
5 input fetch;
6
7
8 assign addr = fetch? pc_addr : ir_addr;
9
10
11 endmodule
12 //------------------------------------------------------------------------------
复制代码
7.程序计数器


       程序计数器用于提供指令地址。以便读取指令,指令按地址顺序存放在存储器中。有两种途径可形成指令地址:其一是顺序执行的情况,其二是遇到要改变顺序执行程序的情况,例如执行JMP指令后,需要形成新的指令地址。复位后,指令指针为零,即每次CPU重新启动将从ROM的零地址开始读取指令并执行。每条指令执行完需2个时钟,这时pc_addr已被增2,指向下一条指令。(因为每条指令占两个字节。)如果正执行的指令是跳转语句,这时CPU状态控制器将会输出load_pc信号,通过load口进入程序计数器。程序计数器(pc_addr)将装入目标地址(ir_addr),而不是增2。
       其VerilogHDL 程序见下面的模块:
复制代码
1 //------------------------------------------------------------------------------
2 module counter ( pc_addr, ir_addr, load, clock, rst);
3 output [12:0] pc_addr;
4 input [12:0] ir_addr;
5 input load, clock, rst;
6 reg [12:0] pc_addr;
7
8 always @( posedge clock or posedge rst )
9     begin
10         if(rst)
11             pc_addr<=13'b0_0000_0000_0000;
12         else
13             if(load)
14                 pc_addr<=ir_addr;
15             else
16                 pc_addr <= pc_addr + 1;
17     end
18 endmodule
19 //------------------------------------------------------------------------------
复制代码
8.状态控制器


     状态控制器由两部分组成:
状态机(上图中的MACHINE部分)
状态控制器(上图中的MACHINECTL部分)
状态机控制器接受复位信号RST,当RST有效时通过信号ena使其为0,输入到状态机中停止状态机的工作。
         状态控制器的VerilogHDL程序见下面模块:
复制代码
1 //------------------------------------------------------------------------------
2 module machinectl( ena, fetch, rst);
3 output ena;
4 input fetch, rst;
5 reg ena;
6 always @(posedge fetch or posedge rst)
7
8
9 begin
10     if(rst)
11         ena<=0;
12     else
13         ena<=1;
14 end
15 endmodule
16 //------------------------------------------------------------------------------
复制代码
       状态机是CPU的控制核心,用于产生一系列的控制信号,启动或停止某些部件。CPU何时进行读指令读写I/O端口,RAM区等操作,都是由状态机来控制的。状态机的当前状态,由变量state记录,state的值就是当前这个指令周期中已经过的时钟数(从零计起)。
指令周期是由8个时钟周期组成,每个时钟周期都要完成固定的操作。
第0个时钟,因为CPU状态控制器的输出:rd和load_ir为高电平,其余均为低电平。指令寄存器寄存由ROM送来的高8位指令代码。
第1个时钟,与上一时钟相比只是inc_pc从0变为1故PC增1,ROM送来低8位指令代码,指令寄存器寄存该8位代码。
第2个时钟,空操作。
第3个时钟,PC增1,指向下一条指令。若操作符为HLT,则输出信号HLT为高。如果操作符不为HLT,除了PC增一外(指向下一条指令),其它各控制线输出为零。
第4个时钟,若操作符为AND、ADD、XOR或LDA,读相应地址的数据;若为JMP,将目的地址送给程序计数器;若为STO,输出累加器数据。
第5个时钟,若操作符为ANDD、ADD或XORR,算术运算器就进行相应的运算;若为LDA,就把数据通过算术运算器送给累加器;若为SKZ,先判断累加器的值是否为0,如果为0,PC就增1,否则保持原值;若为JMP,锁存目的地址;若为STO,将数据写入地址处。
第6个时钟,空操作。
第7个时钟,若操作符为SKZ且累加器值为0,则PC值再增1,跳过一条指令,否则PC无变化。
状态机的VerilogHDL 程序见下面模块:
复制代码
  1 //------------------------------------------------------------------------------
  2 module machine( inc_pc, load_acc, load_pc, rd,wr, load_ir,
  3 datactl_ena, halt, clk1, zero, ena, opcode );
  4 output inc_pc, load_acc, load_pc, rd, wr, load_ir;
  5 output datactl_ena, halt;
  6 input clk1, zero, ena;
  7 input [2:0] opcode;
  8 reg inc_pc, load_acc, load_pc, rd, wr, load_ir;
  9 reg datactl_ena, halt;
10 reg [2:0] state;
11
12 parameter HLT = 3 'b000,
13     SKZ = 3 'b001,
14     ADD = 3 'b010,
15     ANDD = 3 'b011,
16     XORR = 3 'b100,
17     LDA = 3 'b101,
18     STO = 3 'b110,
19     JMP = 3 'b111;
20
21 always @( negedge clk1 )
22     begin
23         if ( !ena ) //接收到复位信号RST,进行复位操作
24             begin
25                 state<=3'b000;
26                 {inc_pc,load_acc,load_pc,rd}<=4'b0000;
27                 {wr,load_ir,datactl_ena,halt}<=4'b0000;
28             end
29         else
30             ctl_cycle;
31     end
32 //-----------------begin of task ctl_cycle---------
33 task ctl_cycle;
34 begin
35     casex(state)
36         3’b000: //load high 8bits in struction
37             begin
38                 {inc_pc,load_acc,load_pc,rd}<=4'b0001;
39                 {wr,load_ir,datactl_ena,halt}<=4'b0100;
40                 state<=3’b001;
41             end
42         3’b001: //pc increased by one then load low 8bits instruction
43             begin
44                 {inc_pc,load_acc,load_pc,rd}<=4'b1001;
45                 {wr,load_ir,datactl_ena,halt}<=4'b0100;
46                 state<=3’b010;
47             end
48         3’b010: //idle
49             begin
50                 {inc_pc,load_acc,load_pc,rd}<=4'b0000;
51                 {wr,load_ir,datactl_ena,halt}<=4'b0000;
52                 state<=3’b011;
53             end
54         3’b011: //next instruction address setup 分析指令从这里开始
55             begin
56                 if(opcode==HLT) //指令为暂停HLT
57                     begin
58                         {inc_pc,load_acc,load_pc,rd}<=4'b1000;
59                         {wr,load_ir,datactl_ena,halt}<=4'b0001;
60                     end
61                 else
62                     begin
63                         {inc_pc,load_acc,load_pc,rd}<=4'b1000;
64                         {wr,load_ir,datactl_ena,halt}<=4'b0000;
65                     end
66                 state<=3’b100;
67             end
68         3’b100: //fetch oprand
69             begin
70                 if(opcode==JMP)
71                     begin
72                         {inc_pc,load_acc,load_pc,rd}<=4'b0010;
73                         {wr,load_ir,datactl_ena,halt}<=4'b0000;
74                     end
75                 else
76                 if( opcode==ADD || opcode==ANDD ||
77                     opcode==XORR || opcode==LDA)
78                         begin
79                             {inc_pc,load_acc,load_pc,rd}<=4'b0001;
80                             {wr,load_ir,datactl_ena,halt}<=4'b0000;
81                         end
82                 else
83                     if(opcode==STO)
84                         begin
85                             {inc_pc,load_acc,load_pc,rd}<=4'b0000;
86                             {wr,load_ir,datactl_ena,halt}<=4'b0010;
87                         end
88                 else
89                     begin
90                         {inc_pc,load_acc,load_pc,rd}<=4'b0000;
91                         {wr,load_ir,datactl_ena,halt}<=4'b0000;
92                 end
93                 state<=3’b101;
94             end
95         3’b101: //operation
96             begin
97                 if ( opcode==ADD||opcode==ANDD||opcode==XORR||opcode==LDA )
98                     begin //过一个时钟后与累加器的内容进行运算
99                         {inc_pc,load_acc,load_pc,rd}<=4'b0101;
100                         {wr,load_ir,datactl_ena,halt}<=4'b0000;
101                     end
102                 else
103                     if( opcode==SKZ && zero==1)
104                         begin
105                             {inc_pc,load_acc,load_pc,rd}<=4'b1000;
106                             {wr,load_ir,datactl_ena,halt}<=4'b0000;
107                         end
108                 else
109                     if(opcode==JMP)
110                         begin
111                             {inc_pc,load_acc,load_pc,rd}<=4'b1010;
112                             {wr,load_ir,datactl_ena,halt}<=4'b0000;
113                         end
114                 else
115                     if(opcode==STO)
116                         begin
117                             //过一个时钟后把wr变1就可写到RAM中
118                             {inc_pc,load_acc,load_pc,rd}<=4'b0000;
119                             {wr,load_ir,datactl_ena,halt}<=4'b1010;
120                         end
121                 else
122                     begin
123                         {inc_pc,load_acc,load_pc,rd}<=4'b0000;
124                         {wr,load_ir,datactl_ena,halt}<=4'b0000;
125                     end
126                 state<=3’b110;
127             end
128     3’b110: //idle
129         begin
130             if ( opcode==STO )
131                 begin
132                     {inc_pc,load_acc,load_pc,rd}<=4'b0000;
133                     {wr,load_ir,datactl_ena,halt}<=4'b0010;
134                 end
135             else
136                 if ( opcode==ADD||opcode==ANDD||opcode==XORR||opcode==LDA)
137                     begin
138                         {inc_pc,load_acc,load_pc,rd}<=4'b0001;
139                         {wr,load_ir,datactl_ena,halt}<=4'b0000;
140                     end
141             else
142                 begin
143                     {inc_pc,load_acc,load_pc,rd}<=4'b0000;
144                     {wr,load_ir,datactl_ena,halt}<=4'b0000;
145                 end
146         state<=3’b111;
147         end
148     3’b111: //
149         begin
150             if( opcode==SKZ && zero==1 )
151                 begin
152                     {inc_pc,load_acc,load_pc,rd}<=4'b1000;
153                     {wr,load_ir,datactl_ena,halt}<=4'b0000;
154                 end
155             else
156                 begin
157                     {inc_pc,load_acc,load_pc,rd}<=4'b0000;
158                     {wr,load_ir,datactl_ena,halt}<=4'b0000;
159                 end
160             state<=3’b000;
161         end
162     default:
163         begin
164             {inc_pc,load_acc,load_pc,rd}<=4'b0000;
165             {wr,load_ir,datactl_ena,halt}<=4'b0000;
166             state<=3’b000;
167         end
168     endcase
169 end
170 endtask
171 //-----------------end of task ctl_cycle---------
172 endmodule
173 //------------------------------------------------------------------------------
复制代码
9.外围模块

为了对RISC_CPU进行测试,需要有存储测试程序的ROM和装载数据的RAM、地址译码器。下面来简单介绍一下:
地址译码器

复制代码
1 module addr_decode( addr, rom_sel, ram_sel);
2 output rom_sel, ram_sel;
3 input [12:0] addr;
4 reg rom_sel, ram_sel;
5 always @( addr )
6
7 begin
8     casex(addr)
9         13'b1_1xxx_xxxx_xxxx:{rom_sel,ram_sel}<=2'b01;
10         13'b0_xxxx_xxxx_xxxx:{rom_sel,ram_sel}<=2'b10;
11         13'b1_0xxx_xxxx_xxxx:{rom_sel,ram_sel}<=2'b10;
12         default:{rom_sel,ram_sel}<=2'b00;
13     endcase
14 end
15 endmodule
复制代码


地址译码器用于产生选通信号,选通ROM或RAM。
FFFFH---1800H RAM
1800H---0000H ROM
RAM

复制代码
1 module ram( data, addr, ena, read, write );
2 inout [7:0] data;
3 input [9:0] addr;
4 input ena;
5 input read, write;
6 reg [7:0] ram [10'h3ff:0];
7
8 assign data = ( read && ena )? ram[addr] : 8'hzz;
9
10 always @(posedge write)
11 begin
12     ram[addr]<=data;
13 end
14 endmodule
复制代码
ROM

复制代码
1 module rom( data, addr, read, ena );
2 output [7:0] data;
3 input [12:0] addr;
4 input read, ena;
5 reg [7:0] memory [13'h1fff:0];
6 wire [7:0] data;
7
8 assign data= ( read && ena )? memory[addr] : 8'bzzzzzzzz;
9
10 endmodule
复制代码
ROM用于装载测试程序,可读不可写。RAM用于存放数据,可读可写。

三、RISC CPU中各部件的相互连接关系

四、RISC CPU和它的外围电路

五、RISC CPU的寻址方式和指令系统
RISC_CPU的指令格式一律为:

指令系统仅由8条指令组成:
1) HLT 停机操作。该操作将空一个指令周期,即8个时钟周期。
2) SKZ 为零跳过下一条语句。该操作先判断当前alu中的结果是否为零,若是零就跳过下一条语句,否则继续执行。
3) ADD 相加。该操作将累加器中的值与地址所指的存储器或端口的数据相加,结果仍送回累加器中。
4) AND 相与。该操作将累加器的值与地址所指的存储器或端口的数据相与,结果仍送回累加器中。
5) XOR 异或。该操作将累加器的值与指令中给出地址的数据异或,结果仍送回累加器中。
6) LDA 读数据。该操作将指令中给出地址的数据放入累加器。
7) STO 写数据。该操作将累加器的数据放入指令中给出的地址。
8) JMP 无条件跳转语句。该操作将跳转至指令给出的目的地址,继续执行。
RISC_CPU是8位微处理器,一律采用直接寻址方式,即数据总是放在存储器中,寻址单元的地址由指令直接给出。这是最简单的寻址方式。
六、RISC CPU的操作和时序
一个微机系统为了完成自身的功能,需要CPU执行许多操作。以下是RISC_CPU的主要操作:
1.系统的复位和启动操作
2.总线读操作
3.总线写操作
下面详细介绍一下每个操作:
1.系统的复位和启动操作

        RISC_CPU的复位和启动操作是通过rst引脚的信号触发执行的。当rst信号一进入高电平,RISC_CPU就会结束现行操作,并且只要rst停留在高电平状态,CPU就维持在复位状态。在复位状态,CPU各内部寄存器都被设为初值,全部为零。数据总线为高阻态,地址总线为0000H,所有控制信号均为无效状态。rst回到低电平后,接着到来的第一个fetch上升沿将启动RISC_CPU开始工作,从ROM的000处开始读取指令并执行相应操作。波形图如下图所示。虚线标志处为RISC_CPU启动工作的时刻。

RISC_CPU的复位和启动操作波形
2.总线读操作
       每个指令周期的前0--3个时钟周期用于读指令,在状态控制器一节中已详细讲述,这里就不再重复。第3.5个周期处,存储器或端口地址就输出到地址总线上,第4--6个时钟周期,读信号rd有效,数据送到数据总线上,以备累加器锁存,或参与算术、逻辑运算。第7个时钟周期,读信号无效,第7.5个周期,地址总线输出PC地址,为下一个指令做好准备。

CPU从存储器或端口读取数据的时序
3写总线操作
每个指令周期的第3.5个时钟周期处,写的地址就建立了,第4个时钟周期输出数据,第5个时钟周期输出写信号。至第6个时钟结束,数据无效,第7.5时钟地址输出为PC地址,为下一个指令周期做好准备。
您需要登录后才可以回帖 登录 | 我要注册

本版积分规则

关闭

站长推荐上一条 /1 下一条

QQ|小黑屋|手机版|Archiver|fpga论坛|fpga设计论坛 ( 京ICP备20003123号-1 )

GMT+8, 2025-4-30 12:27 , Processed in 0.065858 second(s), 20 queries .

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表