

# CS303 Term Project: Elevator Module

Team Members:

Görkem Topçu 28862

Can Zunal 29453

January 19, 2023

|                                       |           |
|---------------------------------------|-----------|
| <b>Introduction</b>                   | <b>3</b>  |
| Inputs and Outputs                    | 3         |
| <b>Implementation</b>                 | <b>4</b>  |
| Clock Divider                         | 4         |
| Debouncer                             | 5         |
| Elevator Control Module               | 6         |
| Inputs                                | 6         |
| Outputs                               | 7         |
| Sequential Part - State Transitions   | 7         |
| Next Floor Definitions                | 8         |
| Status State Machine                  | 9         |
| Elevator Calling                      | 10        |
| Status Machine to Display the Outputs | 12        |
| SSD Module                            | 16        |
| Top Module                            | 20        |
| <b>Simulations</b>                    | <b>21</b> |
| First Case                            | 22        |
| Second case                           | 22        |
| Third case                            | 23        |
| <b>Constraint File</b>                | <b>23</b> |
| <b>Conclusion</b>                     | <b>24</b> |

# Introduction

Implementation of a module that operates an elevator that has 5 floors in verilog programming language and used in an FPGA board for demonstration. The implementation of the elevator can be separated into 4 different modules namely: debouncer which controls external inputs, clock divider which is used to derive another clock using the internal clock of the FPGA, SSD module which manages 7 segment displays to observe the floor numbers in real time and the main elevator control module which operates the floor transitions using external inputs. These modules are combined in a top module for the implementation. This report is divided into 3 parts: implementation, which will explain the individual modules; simulation that shows how the modules work in waveform and constraint file to show how we mapped the inputs and outputs into the FPGA board. The actual input and output names that are used in the code will be in parentheses for improved comprehension.

## Inputs and Outputs

The circuit has 15 inputs and 4 different types of outputs for controlling LEDs and a 7-segment displays for our elevator controller design. Some of the outputs will be used as inputs in other modules, like clock dividers output divided clock(clk\_50hz) will be used as input for the rest of the modules. Each input and output used in a module will be explained in more detail in the corresponding module. You can observe the input and outputs of the overall circuit of the top module in the following graph.



# Implementation

In this section each module's input, outputs and what they do will be explained. Code snippets will be used to demonstrate how the code works. Implementation is divided into 5 parts: clock divider, debouncer, SSD Modul elevator control module and the top module that connects all the individual parts.

## Clock Divider

For the clock divider we have a common reset(rst) and a 50 MHz clock(clk) generated by the FPGA board as input . The goal for this module is to change the frequency of the internal clock so that the display works in real time, each transition taking 5 seconds. This is achieved by using a counter that counts to 2000000. When the counter hits 2000000, the divided clock which is the output of circuit, gets complemented.

```
1  `timescale 1ns / 1ps
2  /*
3   clk_divider module reduces 100MHz (10ns) clock
4   to 50Hz (20ms) by dividing the clock with 2000000
5   It is possible to change the frequency of the output clock
6   by changing the toggle_value
7  */
8  module clk_divider(
9    input clk_in,
10   input rst,
11   output reg divided_clk
12 );
13 parameter toggle_value = 2000000;
14 reg [24:0] cnt;
15 always@(posedge clk_in or negedge rst)
16 begin
17   if (~rst) begin
18     cnt <= 0;
19     divided_clk <= 0;
20   end
21   else begin
22     if (cnt==toggle_value) begin
23       cnt <= 0;
24       divided_clk <= ~divided_clk;
25     end
26     else begin
27       cnt <= cnt +1;
28       divided_clk <= divided_clk;
29     end
30   end
31 end
32 endmodule
```

## Debouncer

There are a total of 5 debouncer modules created for this project as we have 5 floors with 5 different buttons in each floor. This module has several inputs including the 50Hz (20ms) clock input generated by the clock divider module, common reset (rst) and an input received directly from the FPGA board's push buttons which is used to call an elevator from the outside. The goal is to filter out any noise present on the digital signal caused by multiple transitions when you press or release the button and output the cleaned signal. The cleaned outputs are then used as input for the elevator control module.



```
Q | H | ← | → | X | E | D | X | // | E | ? |
1 `timescale 1ns / 1ps
2 module debouncer(
3     input      clk,
4     input      rst,
5     input      noisy_in, // port from the push button
6     output reg  clean_out // port into the circuit
7 );
8 reg noisy_in_reg;
9 reg clean_out_tmp1; // will be used to detect rising edge
10 reg clean_out_tmp2; // will be used to detect rising edge
11 always@(posedge clk or negedge rst)
12 begin
13     if (~rst) begin
14         noisy_in_reg <= 0;
15         clean_out_tmp1 <= 0;
16         clean_out_tmp2 <= 0;
17
18         clean_out <= 0;
19     end
20     else begin
21         // store the input
22         noisy_in_reg <= noisy_in;
23         clean_out_tmp1 <= noisy_in_reg;
24
25         // rising edge detect
26         clean_out_tmp2 <= clean_out_tmp1;
27         clean_out <= ~clean_out_tmp2 & clean_out_tmp1; // it produce a single pulse during a risingedge
28     end
29 end
30 endmodule
31
```

# Elevator Control Module

This is the main module that governs the state transition of the elevator. It uses different status machines, with combinational and sequential parts to achieve the behavior of an elevator. The module takes inputs and outputs directly from the FPGA board, the debouncer module and the clock divider module. Module checks if the calls from the different floors are valid, and changes the status of the elevator depending on the operation criteria. There are 2 different statuses that we consider and display in the 7 segment decoders: which floor the elevator currently is and its status(UP, DOWN or IDLE). Parameters are used for easier understanding in the code snippets.

```
1  `timescale lns / 1ps
2
3  module elevator_control_module(
4      input rst,
5      input clk,
6      input floor_0_p, floor_1_p, floor_2_p, floor_3_p, floor_4_p,
7      input direction_1, direction_2, direction_3,
8      input floor_0_d ,floor_1_d, floor_2_d,floor_3_d, floor_4_d,
9      output reg led_inside_0, led_inside_1, led_inside_2, led_inside_3, led_inside_4,
10     output reg led_outside_0, led_outside_1, led_outside_2, led_outside_3, led_outside_4,
11     output reg led_busy,
12     output reg [7:0] a, b, c, d, e, f, g, p
13 );
14
15 parameter FL_IDX0 = 3'b000, FL_IDX1 = 3'b001, FL_IDX2 = 3'b010, FL_IDX3 = 3'b011, FL_IDX4 = 3'b100; // floor indices
16 parameter FLOOR0 = 5'b000001, FLOOR1 = 5'b000010, FLOOR2 = 5'b00100, FLOOR3 = 5'b01000, FLOOR4 = 5'b10000; // floors
17 parameter IDLE = 2'b00, UP = 2'b01, DOWN = 2'b10; // states
18
19 reg [4:0] current_floor;
20 reg [4:0] next_floor;
21 reg [2:0] current_fl_idx, next_fl_idx;
22
23 reg [1:0] current_state;
24 reg [1:0] next_state;
25 reg [1:0] intermediate_state;
26
27 //counter to slow the input clock
28 reg [7:0] counter;
29
30 // floor states
31 reg [4:0] call_floor;
```

## Inputs

This module has 15 inputs in total, a common clock (clk\_50hz) from the clock divider module, a common reset (rst) that resets the circuit, 5 inputs for calling the elevator from outside from each floor (floor\_x\_p), 3 directional switches used when calling the elevator from outside for floors 1 to 3(direction\_x ) and 5 inputs for calling the elevator from outside from each floor(floor\_x\_d). Consider x as the corresponding floor number.

## Outputs

This module has 4 different types of output, 5 LEDs to show which switches are used to go to destination floors for the people inside the elevators (led\_inside\_x). Another 5 LEDs to show which push buttons are pressed to call the elevator for the people outside the elevator (led\_outside\_x). These outputs should only go high when the call from the outside and inside inputs are valid according to the operation criteria which will be checked by a status machine. There is also another output used to show that the elevator is busy after reaching and stopping on a floor(led\_busy). The last output type of this module is used to govern the 7 segment decoder used for observing which floor the elevator is in real life and its status.

## Sequential Part - State Transitions

In this part we have an always block that is used to govern state transitions. Common reset signal resets the states to the initial state, in this case floor 0 and idle. Next state and next floor are assigned to the current state and current floor respectively. Also we used a counter to increase the time period between transitions as the elevator needs to change floors and states every 5 seconds in the FPGA implementation.

```
33 //busy state
34 reg busy_state;
35
36 //sequential part - state transitions
37 always@(posedge clk or negedge rst)
38 begin
39 if (~rst)begin
40     current_floor <= FLOOR0;
41     current_f1_idx <= FL_IDX0;
42     current_state <= IDLE;
43     led_busy <= 0;
44     counter <= 0;
45 end
46 else begin
47     if(counter == 8'b11111010) begin      // if equal to 250
48         // status change
49         current_state <= next_state;
50         // led busy change
51         led_busy <= busy_state;
52         // floor change
53         current_floor <= next_floor;
54         current_f1_idx <= next_f1_idx;
55         // restart counter
56         counter <= 0;
57     end
58     else counter <= counter + 1; // if not 250 add 1
59 end
60 end
..
```

## Next Floor Definitions

In this part depending on the current floor, current state and a register we hold (call\_floor) we decide on the next floor of the elevator. Different cases are checked for each floor. For instance, if the elevator is on floor 0 and the current state is IDLE or there is a call from the same floor, the next floor should stay the same. Another example is if the elevator is on the 2nd floor and the next state is up elevator needs to go to the 3rd floor. Depending on the current state and floor next\_floor changes. Next\_fl\_idx, which is used for comparisons in another state machine, is also changed in this part.

```
62 : // next floor definitions
63 ⊕ always @(*)
64 ⊕ begin
65 ⊕   case(current_floor)
66 ⊕     FLOOR0:begin
67 ⊕       if(current_state == IDLE || call_floor[current_fl_idx])begin
68 ⊕         next_floor <= FLOOR0;
69 ⊕         next_fl_idx <= FL_IDX0;
70 ⊕       end
71 ⊕     else if(current_state == UP)begin
72 ⊕       next_floor <= FLOOR1;
73 ⊕       next_fl_idx <= FL_IDX1;
74 ⊕     end
75 ⊕     else begin
76 ⊕       next_floor <= FLOOR0;
77 ⊕       next_fl_idx <= FL_IDX0;
78 ⊕     end
79 ⊕   end
80 ⊕   FLOOR1:begin
81 ⊕     if(current_state == IDLE || call_floor[current_fl_idx])begin
82 ⊕       next_floor <= FLOOR1;
83 ⊕       next_fl_idx <= FL_IDX1;
84 ⊕     end
85 ⊕     else if(current_state == UP)begin
86 ⊕       next_floor <= FLOOR2;
87 ⊕       next_fl_idx <= FL_IDX2;
88 ⊕     end
89 ⊕     else if(current_state == DOWN)begin
90 ⊕       next_floor <= FLOOR0;
91 ⊕       next_fl_idx <= FL_IDX0;
92 ⊕     end
93 ⊕     else begin
94 ⊕       next_floor <= FLOOR1;
95 ⊕       next_fl_idx <= FL_IDX1;
96 ⊕     end
97 ⊕   end
98 ⊕   FLOOR2:begin
99 ⊕     if(current_state == IDLE || call_floor[current_fl_idx])begin
100 ⊕       next_floor <= FLOOR2;
101 ⊕       next_fl_idx <= FL_IDX2;
102 ⊕     end
103 ⊕     else if(current_state == UP)begin
104 ⊕       next_floor <= FLOOR3;
105 ⊕       next_fl_idx <= FL_IDX3;
106 ⊕     end
107 ⊕     else if(current_state == DOWN)begin
108 ⊕       next_floor <= FLOOR1;
109 ⊕       next_fl_idx <= FL_IDX1;
110 ⊕     end
111 ⊕     else begin
112 ⊕       next_floor <= FLOOR3;
113 ⊕       next_fl_idx <= FL_IDX3;
114 ⊕     end
115 ⊕   end
116 ⊕   FLOOR3:begin
117 ⊕     if(current_state == IDLE || call_floor[current_fl_idx])begin
118 ⊕       next_floor <= FLOOR3;
119 ⊕       next_fl_idx <= FL_IDX3;
120 ⊕     end
121 ⊕     else if(current_state == UP)begin
122 ⊕       next_floor <= FLOOR4;
123 ⊕       next_fl_idx <= FL_IDX4;
124 ⊕     end
125 ⊕     else if(current_state == DOWN)begin
```

```

126      next_floor <= FLOOR2;
127      next_fl_idx <= FL_IDX2;
128    end
129  else begin
130    next_floor <= FLOOR3;
131    next_fl_idx <= FL_IDX3;
132  end
133 end
134 FLOOR4:begin
135   if(current_state == IDLE || call_floor[current_fl_idx])begin // if idle state or there is a call here
136     next_floor <= FLOOR4;
137     next_fl_idx <= FL_IDX4;
138   end
139  else if(current_state == DOWN)begin
140    next_floor <= FLOOR3;
141    next_fl_idx <= FL_IDX3;
142  end
143  else begin
144    next_floor <= FLOOR4;
145    next_fl_idx <= FL_IDX4;
146  end
147 end
148 default: begin
149   next_floor <= FLOOR0;
150   next_fl_idx <= FL_IDX0;
151 end
152 endcase
153 end

```

## Status State Machine

In this part we change the state of the elevator depending on an array that we hold called `call_floor` that stores valid calls made from different floors and current state of the elevator. In order to change floors the following logic is used. If the current state is IDLE and there is a call from the current floor the next state should be IDLE. If the current state is IDLE and there is a call from a floor that is higher than the current floor, the current state should change to up. If the current state is IDLE and there is a call from a floor that is lower than the current floor, the current state should change to DOWN. Notice that while the current state is UP or DOWN and the next state is IDLE meaning that the elevator stops on a floor the `busy_state` changes to 1. Busy state is used to light led\_busy.

```

155 // Status state machine
156 always# (*)begin
157   case (current_state)
158     IDLE:begin
159       busy_state <= 0;
160       if(call_floor)begin // there is a call
161         if(call_floor[current_fl_idx]) next_state = IDLE; // call is at current floor
162         else if(call_floor > current_floor) next_state = UP; // call is at upper floors
163         else if (call_floor < current_floor) next_state = DOWN; // call is at lower floors
164         else next_state = IDLE;
165       end
166       else next_state <= IDLE;
167     end
168     UP:begin
169       if(call_floor[current_fl_idx+1])begin
170         next_state <= IDLE; // there is a call at one upper floor
171         busy_state <= 1;
172       end
173       else next_state <= UP;
174     end
175     DOWN:begin
176       if(call_floor[current_fl_idx-1])begin
177         next_state <= IDLE; // there is a call at one lower floor
178         busy_state <= 1;
179       end
180       else next_state <= DOWN;
181     end
182     default: next_state <= IDLE;
183   endcase
184 end

```

## Elevator Calling

In this status machine the goal is to change an array called call\_floor that is used in another status machine that decides state changes and it also changes inside(led\_inside\_x) and outside LEDs (led\_outside\_x) depending on inputs. Common reset(rst) is used to reset the LEDS and call\_floor array. The operation is fairly simple, we just need to check which input is high. After that, depending on the current floor, call\_floor's status as well as direction inputs it decides whether the call is viable and adds it into the call\_floor array which is then used for operating status state machines.

```
186 : // ELEVATOR CALLING
187 : always #8 (posedge clk or negedge rst)begin
188 :   if(~rst)begin
189 :     call_floor <= 0;
190 :     intermediate_state <= IDLE; // NEW
191 :     led_outside_0 <= 0;
192 :     led_outside_1 <= 0;
193 :     led_outside_2 <= 0;
194 :     led_outside_3 <= 0;
195 :     led_outside_4 <= 0;
196 :     led_inside_0 <= 0;
197 :     led_inside_1 <= 0;
198 :     led_inside_2 <= 0;
199 :     led_inside_3 <= 0;
200 :     led_inside_4 <= 0;
201 :   end
202 :   else if((floor_0_p || floor_0_d) && current_floor != FLOOR0 && ~call_floor[0] && intermediate_state != DOWN)begin // floor_0_p (lowermost floor)
203 :     if(current_state != UP)begin
204 :       intermediate_state <= UP;
205 :       call_floor[0] <= 1;
206 :       if(floor_0_p)led_outside_0 <= 1;
207 :       else if (floor_0_d) led_inside_0 <= 1;
208 :     end
209 :     else begin
210 :       call_floor[0] <= 0;
211 :       if(floor_0_p)led_outside_0 <= 0;
212 :       else if(floor_0_d) led_inside_0 <= 0;
213 :     end
214 :   end
215 :   else if((floor_1_p || floor_1_d) && current_floor != FLOOR1 && ~call_floor[1])begin // floor_1_p
216 :     if(current_state == IDLE)begin
217 :       if(~direction_1 && intermediate_state!=UP) || (~direction_1 && intermediate_state != DOWN))begin
218 :         if(~direction_1) intermediate_state <= DOWN;
219 :         else intermediate_state <= UP;
220 :         call_floor[1] <= 1;
221 :         if(floor_1_p)led_outside_1 <= 1;
222 :         else if(floor_1_d) led_inside_1 <= 1;
223 :       end
224 :       else begin
225 :         call_floor[1] <= 0;
226 :         if(floor_1_p)led_outside_1 <= 0;
227 :         else if(floor_1_d) led_inside_1 <= 0;
228 :       end
229 :     end
230 :     else if(~direction_1)begin // go down
231 :       if(current_state == UP)begin
232 :         call_floor[1] <= 0;
233 :         if(floor_1_p)led_outside_1 <= 0;
234 :         else if(floor_1_d) led_inside_1 <= 0;
235 :       end
236 :       else begin
237 :         call_floor[1] <= 1;
238 :         if(floor_1_p)led_outside_1 <= 1;
239 :         else if(floor_1_d) led_inside_1 <= 1;
240 :       end
241 :     end
242 :     else if(direction_1)begin // go up
243 :       if(FLOOR1 < current_floor)begin
244 :         call_floor[1] <= 0;
245 :         if(floor_1_p)led_outside_1 <= 0;
246 :         else if(floor_1_d) led_inside_1 <= 0;
247 :       end
248 :       else begin
249 :         call_floor[1] <= 1;
```

```

250      if(floor_1_p)led_outside_1 <= 1;
251      else if(floor_1_d) led_inside_1 <= 1;
252    end
253  end
254
255  else if((floor_2_p && floor_2_d) && current_floor != FLOOR2 && ~call_floor[2])begin // floor_2_p
256    if(current_state == IDEL) begin
257      if(~(direction_2 && intermediate_state!=DOWN)) || (~direction_2 && intermediate_state != UP))begin
258        if(~(direction_1) intermediate_state <= DOWN;
259        else intermediate_state <= UP;
260        call_floor[2] <= 0;
261        if(floor_2_p)led_outside_2 <= 1;
262        else if(floor_2_d) led_inside_2 <= 1;
263      end
264    end
265    else begin
266      call_floor[2] <= 0;
267      if(floor_2_p)led_outside_2 <= 0;
268      else if(floor_2_d) led_inside_2 <= 0;
269    end
270  end
271  else if(~direction_2)begin // go down
272    if(FLOOR2 > current_floor)begin
273      call_floor[2] <= 0;
274      if(floor_2_p)led_outside_2 <= 0;
275      else if(floor_2_d) led_inside_2 <= 0;
276    end
277    else if(current_state == UP)begin
278      call_floor[2] <= 0;
279      if(floor_2_p)led_outside_2 <= 0;
280      else if(floor_2_d) led_inside_2 <= 0;
281    end
282  end
283
284    else begin
285      call_floor[2] <= 1;
286      if(floor_2_p)led_outside_2 <= 1;
287      else if(floor_2_d) led_inside_2 <= 1;
288    end
289  end
290  else if(direction_2)begin // go up
291    if(FLOOR2 < current_floor)begin
292      call_floor[2] <= 0;
293      if(floor_2_p)led_outside_2 <= 0;
294      else if(floor_2_d) led_inside_2 <= 0;
295    end
296  end
297  else if(current_state == DOWN)begin
298    call_floor[2] <= 0;
299    if(floor_2_p)led_outside_2 <= 0;
300    else if(floor_2_d) led_inside_2 <= 0;
301  end
302
303  end
304
305  end if((floor_3_p || floor_3_d) && current_floor != FLOOR3 && ~call_floor[3])begin// floor_3_p
306  if(current_state == IDEL) begin
307    if((direction_3 && intermediate_state!=DOWN) || (~direction_3 && intermediate_state!=UP))begin
308      if(~(direction_1) intermediate_state <= DOWN;
309      else intermediate_state <= UP;
310      call_floor[3] <= 1;
311      if(floor_3_p)led_outside_3 <= 1;
312      else if(floor_3_d) led_inside_3 <= 1;
313    end
314
315    else begin
316      call_floor[3] <= 0;
317      if(floor_3_p)led_outside_3 <= 0;
318      else if(floor_3_d) led_inside_3 <= 0;
319    end
320  end
321  else if(~direction_3)begin // go down
322    if(FLOOR3 > current_floor)begin
323      call_floor[3] <= 0;
324      if(floor_3_p)led_outside_3 <= 0;
325      else if(floor_3_d) led_inside_3 <= 0;
326    end
327    else begin
328      call_floor[3] <= 1;
329      if(floor_3_p)led_outside_3 <= 1;
330      else if(floor_3_d) led_inside_3 <= 1;
331    end
332  end
333  else if(direction_3)begin // go up
334    if(current_state == DOWN)begin
335      call_floor[3] <= 0;
336      if(floor_3_p)led_outside_3 <= 0;
337      else if(floor_3_d) led_inside_3 <= 0;
338    end
339    else begin
340      call_floor[3] <= 1;
341      if(floor_3_p)led_outside_3 <= 1;
342    end
343  end
344
345  end if((floor_4_p || floor_4_d) && current_floor != FLOOR4 && ~call_floor[4] && intermediate_state != UP)begin // floor_4_p (uppermost floor)

```

```

347         intermediate_state <= DOWN;
348         call_floor[4] <= 1;
349         if(floor_4_p)led_outside_4 <= 1;
350         else if(floor_4_d) led_inside_4 <= 1;
351     end
352     else if(current_state != DOWN)begin
353         call_floor[4] <= 1;
354         if(floor_4_p)led_outside_4 <= 1;
355         else if(floor_4_d) led_inside_4 <= 1;
356     end
357     else begin
358         call_floor[4] <= 0;
359         if(floor_4_p)led_outside_4 <= 0;
360         else if(floor_4_d) led_inside_4 <= 0;
361     end
362 end
363 else begin
364     if(current_floor == FLOOR1)begin // floor is 1
365         call_floor[1] <= 0;
366         led_outside_1 <= 0;
367         led_inside_1 <= 0;
368     end
369     else if(current_floor == FLOOR2)begin // floor is 2
370         call_floor[2] <= 0;
371         led_outside_2 <= 0;
372         led_inside_2 <= 0;
373     end
374     else if(current_floor == FLOOR3)begin // floor is 3
375         call_floor[3] <= 0;
376         led_outside_3 <= 0;
377         led_inside_3 <= 0;
378     end
379     else if(current_floor == FLOOR4)begin // floor is 4
380         call_floor[4] <= 0;
381         led_outside_4 <= 0;
382         led_inside_4 <= 0;
383     end
384     else begin // floor is 0
385         call_floor[0] <= 0;
386         led_outside_0 <= 0;
387         led_inside_0 <= 0;
388     end
389     if(~call_floor) intermediate_state <= IDLE;
390     else begin end
391 end
392 end

```

## Status Machine to Display the Outputs

In this part we give outputs to the SSD module to observe the status of the elevator in real time. The outputs are 8 bit arrays that map the 7 segment decoder. The right and left block have 8 SSD in total. The 4 leftmost SSD's show the status and 4 right most SSD's show current floor number. There are a total of 13 cases as there are 5 floors, and 3 states for floors 1 to 3 and 2 states for 0th and 4th floor. Depending on these 13 cases, the status machine changes the displays. SSD module's operation will be explained in the corresponding part.

```

394 : // Seven segment display
395 : always @ (posedge clk or negedge rst) begin
396 :   if(~rst) begin
397 :     a <= 8'b11111111;
398 :     b <= 8'b11111111;
399 :     c <= 8'b11111111;
400 :     d <= 8'b11111111;
401 :     e <= 8'b11111111;
402 :     f <= 8'b11111111;
403 :     g <= 8'b11111111;
404 :     p <= 8'b11111111;
405 :
406 :
407 :   else begin
408 :     case(current_floor)
409 :
410 :       FLOOR0: begin // FL-0
411 :         case(current_state)
412 :           IDLE: begin //--Id
413 :             a <= 8'b11110110;
414 :             b <= 8'b11001110;
415 :             c <= 8'b11001110;
416 :             d <= 8'b11101010;
417 :             e <= 8'b11100010;
418 :             f <= 8'b11110010;
419 :             g <= 8'b11100111;
420 :             p <= 8'b11111111;
421 :
422 :
423 :           end
424 :           UP: begin //--UP
425 :             a <= 8'b11100110;
426 :             b <= 8'b11001110;
427 :             c <= 8'b11011110;
428 :             d <= 8'b110111010;
429 :             e <= 8'b11000010;
430 :
431 :
432 :             f <= 8'b11000010;
433 :             g <= 8'b11100111;
434 :             p <= 8'b11111111;
435 :
436 :
437 :           end
438 :           DOWN: begin //--d0
439 :             a <= 8'b11100110;
440 :             b <= 8'b11001110;
441 :             c <= 8'b11001110;
442 :             d <= 8'b111001010;
443 :
444 :           end
445 :
446 :
447 :           IDLE: begin //--Id
448 :             a <= 8'b11110111;
449 :             b <= 8'b11001110;
450 :             c <= 8'b11001110;
451 :             d <= 8'b11101011;
452 :             e <= 8'b11100011;
453 :
454 :
455 :           end
456 :           UP: begin //--UP
457 :             a <= 8'b11100111;

```

```

458         d <= 8'b11011011;
459         e <= 8'b11000011;
460         f <= 8'b11000011;
461         g <= 8'b11100111;
462         p <= 8'b11111111;
463     end
464     DOWN: begin //--d0
465         a <= 8'b11100111;
466         b <= 8'b11001110;
467         c <= 8'b11001110;
468         d <= 8'b11001011;
469         e <= 8'b11000011;
470         f <= 8'b11100011;
471         g <= 8'b11010111;
472         p <= 8'b11111111;
473     end
474     endcase
475 end
476 FLOOR2: begin // FL-2
477     case(current_state)
478         IDLE: begin //--Id
479             a <= 8'b11110110;
480             b <= 8'b11001110;
481             c <= 8'b11001111;
482             d <= 8'b11101010;
483             e <= 8'b11100010;
484             f <= 8'b11110011;
485             g <= 8'b11100111;
486             p <= 8'b11111111;
487     end
488         UP: begin //--UP
489             a <= 8'b11100110;
490             b <= 8'b11001110;
491             c <= 8'b11011111;
492             d <= 8'b11011010;
493             e <= 8'b11000010;
494             f <= 8'b11000011;
495             g <= 8'b11100111;
496             p <= 8'b11111111;
497     end
498     DOWN: begin //--d0
499         a <= 8'b11100110;
500         b <= 8'b11001110;
501         c <= 8'b11001111;
502         d <= 8'b11001010;
503         e <= 8'b11000010;
504         f <= 8'b11100011;
505         g <= 8'b11010111;
506         p <= 8'b11111111;
507     end
508     endcase
509 end
510 FLOOR3: begin // FL-3
511     case(current_state)
512         IDLE: begin //--Id
513             a <= 8'b11110110;
514             b <= 8'b11001110;
515             c <= 8'b11001110;
516             d <= 8'b11101010;
517             e <= 8'b11100011;
518             f <= 8'b11110011;
519             g <= 8'b11100110;

```

```

520           p <= 8'b11111111;
521     end
522   begin //--UP
523     a <= 8'b11100110;
524     b <= 8'b11001110;
525     c <= 8'b11011110;
526     d <= 8'b11011010;
527     e <= 8'b11000011;
528     f <= 8'b11000011;
529     g <= 8'b11100110;
530     p <= 8'b11111111;
531   end
532   begin //--d0
533     a <= 8'b11100110;
534     b <= 8'b11001110;
535     c <= 8'b11001110;
536     d <= 8'b11001010;
537     e <= 8'b11000011;
538     f <= 8'b11100011;
539     g <= 8'b11010110;
540     p <= 8'b11111111;
541   end
542 endcase
543 begin
544   begin // FL-4
545     case(current_state)
546       IDLE: begin //--Id
547         a <= 8'b11110111;
548         b <= 8'b11001110;
549         c <= 8'b11001110;
550         d <= 8'b11101011;
551         e <= 8'b11100011;

552         f <= 8'b11110010;
553         g <= 8'b11100110;
554         p <= 8'b11111111;
555       end
556       begin //--UP
557         a <= 8'b11100111;
558         b <= 8'b11001110;
559         c <= 8'b11011110;
560         d <= 8'b11011011;
561         e <= 8'b11000011;
562         f <= 8'b11000010;
563         g <= 8'b11100110;
564         p <= 8'b11111111;
565       end
566       begin //--d0
567         a <= 8'b11100111;
568         b <= 8'b11001110;
569         c <= 8'b11001110;
570         d <= 8'b11001011;
571         e <= 8'b11000011;
572         f <= 8'b11100010;
573         g <= 8'b11010110;
574         p <= 8'b11111111;
575       end
576     endcase
577   end
578 endcase
579 end
580 end
581 endmodule

```

## SSD Module

Each SSD (digit) has 8 diodes (A, B, C, D, E, F, G, P). The anodes of all 8 diodes are connected, so they have a common anode. Their cathodes are operated by the module. To activate anything you need to activate the common anode of that SSD. It works active low. Common reset input reset all the displays as usual.



```
1 `timescale 1ns / 1ps
2 module ssd(clk,reset,a,b,c,d,e,f,g,p,
3   a_out,b_out,c_out,d_out,e_out,f_out,g_out,p_out,
4   an);
5   input clk, reset;// set clock and reset as input(1 bit)
6   input [7:0] a,b,c,d,e,f,g,p;
7   output reg a_out,b_out,c_out,d_out,e_out,f_out,g_out,p_out;
8   output reg [7:0] an;
9
10  reg [3:0] state;//holds state number (3 bit)
11  reg [19:0] counter;//counter to slow the input clock
12
13 // in this always block the speed of the clock reduced by 100000
14 // times so that display works properly
15 always @ (posedge clk or negedge reset) begin //state counter
16   if(~reset) begin //synchronous reset
17     state <= 0; //if reset set state and counter to zero
18     counter <= 0;
19   end
20   else begin //else the counter until 100000
21     if(counter == 20'h186A0) begin //if equal to 100000
22       if (state == 4'b1000)
23         state <= 1;
24       else
25         state <= state + 1;
26       counter <= 0;
27     end
28   else
29     counter <= counter + 1; //if not 100000 add 1
30   end
31 end
32 end
```

```

33 //in this always block we give the inputs to the leds by choosing
34 //different display segment in each time
35 :
36 always@(posedge clk or negedge reset)
37 begin
38 if(~reset)// if reset initialize the outputs
39 begin
40 an[7:0] <= 8'b11111111;
41 a_out <= 1;
42 b_out <= 1;
43 c_out <= 1;
44 d_out <= 1;
45 e_out <= 1;
46 f_out <= 1;
47 g_out <= 1;
48 p_out <= 1;
49 end
50 else if(state == 4'b0001)
51 //state 1 gives the led outputs to the AN0
52 begin
53 an[7:0] <= 8'b11111110;
54 a_out <= a[0];
55 b_out <= b[0];
56 c_out <= c[0];
57 d_out <= d[0];
58 e_out <= e[0];
59 f_out <= f[0];
60 g_out <= g[0];
61 p_out <= p[0];
62 end
63 else if(state == 4'b0010)
64 //state 2 gives the led outputs to the AN1
65 begin
66 an[7:0] <= 8'b11111101;
67 a_out <= a[1];
68 b_out <= b[1];
69 c_out <= c[1];
70 d_out <= d[1];
71 e_out <= e[1];
72 f_out <= f[1];
73 g_out <= g[1];
74 p_out <= p[1];
75 end
76 else if(state == 4'b0011)
77 //state 3 gives the led outputs to the AN2
78 begin
79 an[7:0] <= 8'b11111011;
80 a_out <= a[2];
81 b_out <= b[2];
82 c_out <= c[2];
83 d_out <= d[2];
84 e_out <= e[2];
85 f_out <= f[2];
86 g_out <= g[2];
87 p_out <= p[2];
88 end
89
90 end

```

```

77      else if(state == 4'b0011)
78 //state 3 gives the led outputs to the AN2
79 begin
80     an[7:0] <= 8'b11111011;
81     a_out  <= a[2];
82     b_out  <= b[2];
83     c_out  <= c[2];
84     d_out  <= d[2];
85     e_out  <= e[2];
86     f_out  <= f[2];
87     g_out  <= g[2];
88     p_out  <= p[2];
89
90 end
91 else if(state == 4'b0100)
92 //state 4 gives the led outputs to the AN3
93 begin
94     an[7:0] <= 8'b11110111;
95     a_out  <= a[3];
96     b_out  <= b[3];
97     c_out  <= c[3];
98     d_out  <= d[3];
99     e_out  <= e[3];
100    f_out  <= f[3];
101    g_out  <= g[3];
102    p_out  <= p[3];
103
104 end
105 else if(state == 4'b0101)
106 //state 4 gives the led outputs to the AN4
107 begin
108     an[7:0] <= 8'b11101111;
109     a_out  <= a[4];
110     b_out  <= b[4];
111     c_out  <= c[4];
112     d_out  <= d[4];
113     e_out  <= e[4];
114     f_out  <= f[4];
115     g_out  <= g[4];
116     p_out  <= p[4];
117
118 end
119
120 else if(state == 4'b0110)
121 //state 4 gives the led outputs to the AN5
122 begin
123     an[7:0] <= 8'b11011111;
124     a_out  <= a[5];
125     b_out  <= b[5];
126     c_out  <= c[5];
127     d_out  <= d[5];
128     e_out  <= e[5];
129     f_out  <= f[5];
130     g_out  <= g[5];
131     p_out  <= p[5];
132
133 end

```

```

135      else if(state == 4'b0111)
136      //state 4 gives the led outputs to the AN6
137      begin
138          an[7:0] <= 8'b10111111;
139          a_out <= a[6];
140          b_out <= b[6];
141          c_out <= c[6];
142          d_out <= d[6];
143          e_out <= e[6];
144          f_out <= f[6];
145          g_out <= g[6];
146          p_out <= p[6];
147
148      end
149
150      else if(state == 4'b1000)
151      //state 4 gives the led outputs to the AN7
152      begin
153          an[7:0] <= 8'b01111111;
154          a_out <= a[7];
155          b_out <= b[7];
156          c_out <= c[7];
157          d_out <= d[7];
158          e_out <= e[7];
159          f_out <= f[7];
160          g_out <= g[7];
161          p_out <= p[7];
162
163      end
164
165      else //For other states default inputs and outputs
166      begin
167          an[7:0] <= 8'b11111111;
168          a_out <= 1;
169          b_out <= 1;
170          c_out <= 1;
171          d_out <= 1;
172          e_out <= 1;
173          f_out <= 1;
174          g_out <= 1;
175          p_out <= 1;
176
177      end
178
179 endmodule
```

```

## Top Module

In this part all the modules explained thus far are combined into a top module. Clock divider module is used to create a new clock(clk50hz) that is distributed to other modules. 5 debouncers are used to filter unwanted inputs, These inputs are then fed into the elevator control module which then creates input for the SSD module.

```
1  `timescale 1ns / 1ps
2
3  module top_module(
4      input clk,
5      rst,
6
7      floor_0_p,
8      floor_1_p,
9      floor_2_p,
10     floor_3_p,
11     floor_4_p,
12
13    direction_1,
14    direction_2,
15    direction_3,
16
17    floor_0_d,
18    floor_1_d,
19    floor_2_d,
20    floor_3_d,
21    floor_4_d,
22
23    output led_inside_0,
24    led_inside_1,
25    led_inside_2,
26    led_inside_3,
27    led_inside_4,
28
29    led_outside_0,
30    led_outside_1,
31    led_outside_2,
32    led_outside_3,
```

```

66     .direction_2(direction_2),
67     .direction_3(direction_3),
68     .floor_0_d(floor_0_d),
69     .floor_1_d(floor_1_d),
70     .floor_2_d(floor_2_d),
71     .floor_3_d(floor_3_d),
72     .floor_4_d(floor_4_d),
73     .led_inside_0(led_inside_0),
74     .led_inside_1(led_inside_1),
75     .led_inside_2(led_inside_2),
76     .led_inside_3(led_inside_3),
77     .led_inside_4(led_inside_4),
78     .led_outside_0(led_outside_0),
79     .led_outside_1(led_outside_1),
80     .led_outside_2(led_outside_2),
81     .led_outside_3(led_outside_3),
82     .led_outside_4(led_outside_4),
83     .led_busy(led_busy),
84     .a(a),
85     .b(b),
86     .c(c),
87     .d(d),
88     .e(e),
89     .f(f),
90     .g(g),
91     .p(p)
92   );
93 endmodule
94

34     led_busy,
35
36     a_out,b_out,c_out,d_out,e_out,f_out,g_out,p_out,
37     [7:0]an
38   );
39
40     wire [7:0] a,b,c,d,e,f,g,p;
41     wire clk50hz;
42     wire floor_0_p_clean, floor_1_p_clean, floor_2_p_clean, floor_3_p_clean, floor_4_p_clean;
43
44     clk_divider divider(.clk_in(clk), .rst(rst), .divided_clk(clk50hz));
45
46     debouncer pb0(.clk(clk50hz), .rst(~rst), .noisy_in(floor_0_p), .clean_out(floor_0_p_clean));
47     debouncer pb1(.clk(clk50hz), .rst(~rst), .noisy_in(floor_1_p), .clean_out(floor_1_p_clean));
48     debouncer pb2(.clk(clk50hz), .rst(~rst), .noisy_in(floor_2_p), .clean_out(floor_2_p_clean));
49     debouncer pb3(.clk(clk50hz), .rst(~rst), .noisy_in(floor_3_p), .clean_out(floor_3_p_clean));
50     debouncer pb4(.clk(clk50hz), .rst(~rst), .noisy_in(floor_4_p), .clean_out(floor_4_p_clean));
51
52     ssd_seven_segment_display(clk,rst,a,b,c,d,e,f,g,p,
53       a_out,b_out,c_out,d_out,e_out,f_out,g_out,p_out,
54       an);
55
56     elevator_control_module elevator(
57       .clk(clk50hz),
58       .rst(rst),
59       .floor_0_p(floor_0_p),
60       .floor_1_p(floor_1_p),
61       .floor_2_p(floor_2_p),
62       .floor_3_p(floor_3_p),
63       .floor_4_p(floor_4_p),
64       .direction_1(direction_1),
65

```

## Simulations

Simulations are conducted in order to see if the circuit is working as expected for the operation criteria. As the original code is used for FPGA board implementation, current state and floor names are slightly different in simulation. For the current state 0 is for IDLE, 1 is for UP

and 2 is for DOWN. For the current floor 01 is 0th floor, 02 is 1st floor, 04 is 2nd floor, 08 is for 3rd floor and 10 is for the 4th floor. As you can see floor numbers are currently represented in hexadecimal which is not the case for the original FPGA implementation. Unnecessary inputs and outputs are omitted for improved clarity.

## First Case



As you might have observed there is a call from outside of the 4th floor and 3rd floor respectively. As it can be seen on floors where an elevator call was made previously, elevator stops and led busy starts to light for one clock cycle. Since there were two calls from outside, led busy lights twice. After reaching the final destination which is the 4th floor, there is a call to the first floor from inside. Calls from inside of the elevator operate exactly the same as previous situations.

## Second case



In this case there is a call from the 4th floor. While the elevator is going up reset input is asserted which resets the entire circuit.

## Third case



In this case, to observe the difference between using direction switches during the call, we can divide the case into 2 parts before and after 70 ns. As you might observe in the first case (before 70ns) as direction\_2 is off which means the caller wants to go down, which will cause the input to be discarded since it is not viable according to the operation criteria. In the other case as the direction\_2 is high it works as intended.

## Constraint File

We used the following constraint file in order to map the implementation to the FPGA Board. We created a clock to use in the FPGA board. Directions and floor\_x\_d are switches in the FPGA board. Floor\_x\_p and reset(rst) are push buttons. Outputs of the top module are mapped to the SSD's while led\_inside\_x, led\_outside\_x and led\_busy are mapped to different leds. In the following code snippets specific pins can be observed.

```

1 : set_property -dict { PACKAGE_PIN E3 IOSTANDARD LVCMS33 } [get_ports { clk }];
2 : create_clock -add -name sys_clk_pin -period 10.00 -waveform { 0 5 } [get_ports { clk }];
3 :
4 : set_property -dict { PACKAGE_PIN C12 IOSTANDARD LVCMS33 } [get_ports { rst }];
5 : set_property -dict { PACKAGE_PIN M18 IOSTANDARD LVCMS33 } [get_ports { floor_0_p }];
6 : set_property -dict { PACKAGE_PIN P17 IOSTANDARD LVCMS33 } [get_ports { floor_1_p }];
7 : set_property -dict { PACKAGE_PIN N17 IOSTANDARD LVCMS33 } [get_ports { floor_2_p }];
8 : set_property -dict { PACKAGE_PIN M17 IOSTANDARD LVCMS33 } [get_ports { floor_3_p }];
9 : set_property -dict { PACKAGE_PIN P18 IOSTANDARD LVCMS33 } [get_ports { floor_4_p }];
10 :
11 : set_property -dict { PACKAGE_PIN L16 IOSTANDARD LVCMS33 } [get_ports { direction_1 }];
12 : set_property -dict { PACKAGE_PIN M13 IOSTANDARD LVCMS33 } [get_ports { direction_2 }];
13 : set_property -dict { PACKAGE_PIN R15 IOSTANDARD LVCMS33 } [get_ports { direction_3 }];
14 :
15 : set_property -dict { PACKAGE_PIN V10 IOSTANDARD LVCMS33 } [get_ports { floor_0_d }];
16 : set_property -dict { PACKAGE_PIN U11 IOSTANDARD LVCMS33 } [get_ports { floor_1_d }];
17 : set_property -dict { PACKAGE_PIN U12 IOSTANDARD LVCMS33 } [get_ports { floor_2_d }];
18 : set_property -dict { PACKAGE_PIN H6 IOSTANDARD LVCMS33 } [get_ports { floor_3_d }];
19 : set_property -dict { PACKAGE_PIN T13 IOSTANDARD LVCMS33 } [get_ports { floor_4_d }];
20 :
21 : set_property -dict { PACKAGE_PIN V11 IOSTANDARD LVCMS33 } [get_ports { led_inside_0 }];
22 : set_property -dict { PACKAGE_PIN V12 IOSTANDARD LVCMS33 } [get_ports { led_inside_1 }];
23 : set_property -dict { PACKAGE_PIN V14 IOSTANDARD LVCMS33 } [get_ports { led_inside_2 }];
24 : set_property -dict { PACKAGE_PIN V15 IOSTANDARD LVCMS33 } [get_ports { led_inside_3 }];
25 : set_property -dict { PACKAGE_PIN T16 IOSTANDARD LVCMS33 } [get_ports { led_inside_4 }];
26 :
27 : set_property -dict { PACKAGE_PIN H17 IOSTANDARD LVCMS33 } [get_ports { led_outside_0 }];
28 : set_property -dict { PACKAGE_PIN K15 IOSTANDARD LVCMS33 } [get_ports { led_outside_1 }];
29 : set_property -dict { PACKAGE_PIN J13 IOSTANDARD LVCMS33 } [get_ports { led_outside_2 }];
30 : set_property -dict { PACKAGE_PIN N14 IOSTANDARD LVCMS33 } [get_ports { led_outside_3 }];
31 : set_property -dict { PACKAGE_PIN R18 IOSTANDARD LVCMS33 } [get_ports { led_outside_4 }];

```

```

33 set_property -dict { PACKAGE_PIN N16 IOSTANDARD LVCMOS33 } [get_ports { led_busy }];
34
35 # outputs to seven-segment display
36 set_property -dict { PACKAGE_PIN H15 IOSTANDARD LVCMOS33 } [get_ports { p_out }]
37 set_property -dict { PACKAGE_PIN L18 IOSTANDARD LVCMOS33 } [get_ports { g_out }]
38 set_property -dict { PACKAGE_PIN T11 IOSTANDARD LVCMOS33 } [get_ports { f_out }]
39 set_property -dict { PACKAGE_PIN P15 IOSTANDARD LVCMOS33 } [get_ports { e_out }]
40 set_property -dict { PACKAGE_PIN K13 IOSTANDARD LVCMOS33 } [get_ports { d_out }]
41 set_property -dict { PACKAGE_PIN K16 IOSTANDARD LVCMOS33 } [get_ports { c_out }]
42 set_property -dict { PACKAGE_PIN R10 IOSTANDARD LVCMOS33 } [get_ports { b_out }]
43 set_property -dict { PACKAGE_PIN T10 IOSTANDARD LVCMOS33 } [get_ports { a_out }]
44
45 # outputs to seven-segment display segment select
46 set_property -dict { PACKAGE_PIN U13 IOSTANDARD LVCMOS33 } [get_ports { an[7] }]
47 set_property -dict { PACKAGE_PIN K2 IOSTANDARD LVCMOS33 } [get_ports { an[6] }]
48 set_property -dict { PACKAGE_PIN T14 IOSTANDARD LVCMOS33 } [get_ports { an[5] }]
49 set_property -dict { PACKAGE_PIN P14 IOSTANDARD LVCMOS33 } [get_ports { an[4] }]
50 set_property -dict { PACKAGE_PIN J14 IOSTANDARD LVCMOS33 } [get_ports { an[3] }]
51 set_property -dict { PACKAGE_PIN T9 IOSTANDARD LVCMOS33 } [get_ports { an[2] }]
52 set_property -dict { PACKAGE_PIN J18 IOSTANDARD LVCMOS33 } [get_ports { an[1] }]
53 set_property -dict { PACKAGE_PIN J17 IOSTANDARD LVCMOS33 } [get_ports { an[0] }]

```

## Conclusion

In this project we implement an elevator controller in verilog programming language for a 5 floor elevator that has buttons and displays both inside and outside. It had both combinational and sequential parts. Different modules are combined into a top module. Test cases are performed in the simulations to check whether the module works as intended.