Elevator control software project written for the course TTK4145
(Real-time Programming) at Norwegian University of Science and Technology (NTNU).
The elevator control software is to steer several (or one) elevators such that they service all requests regardless of single failures. Elevators also need some (sub-)optimal allocation algorithm to ensure that a good path is followed (least energy, least time,...).
- Elevators that are disconnected from the network will still need to serve their orders.
- An elevator that shuts down and restarts must restore internal (command) orders.
- Requests must always be handled in finite time.
The design is extremely data-oriented. We started contemplating the required information in order for elevators to make good decisions. We arrived at the following data structure:
(state (
id ; A UUID. It will be stored, so restarting will give you the same UUID. Reset it by clearing temporaries.
name ; A human-readable name given to the elevator (to make communication and detection easier).
position ; The last known position of the elevator (in-between floors has no position).
servicing-requests ; The requests that an elevator is currently servicing.
call-requests ; The available call requests (buttons from outside the elevator) for all elevators.
command-requests ; The available command requests (buttons from inside the elevator), only for the elevator holding it.
done-requests ; The call requests that have been finished.
opening-time)) ; The current open state of the doors. Decrements until zero and then continues servicing floors.
With the request
being (request (direction floor timestamp))
. The direction is either a 'up
, 'command
, or 'down
.
The floor is the floor for the request, and the timestamp is when the request was logged.
The timestamp serves as a uniqueness qualifier as well as a strict weak ordering of requests. This ordering
prevents excessive waiting times.
All elevator states are put in a hash-table of #hash((uuid . state))
. This way, we can process all states easily.
The entirety of the program bases itself on these data.
Control starts in main.rkt
which endlessly calls core.rkt
, which calls a boatload of functions from utility.rkt
to process the entire hash-table of elevators.
There is no effect system in standard racket so we've annotated all impure functions with #io
at the end
of their names.
Our solution exploits the fact that all elevators are connected to the same router. This allows us to broadcast. Broadcasting makes other elevators aware of existence. If an elevator has stopped broadcasting, the other elevators will detect this by not resetting the time to live counter for that elevator. For every frame where no packet was received from an elevator it's "time to live" value drops by one, and so, if the time to live reached a limit, the elevator is forgotten by the one holding the count.
In addition to broadcasting and receiving, we have serialization with an added sha256
check on a specific structure
so that stray packets sent to our ports get ignored. We use Racket's built-in fast-load serialization (FASL)
to serialize arbitrary prefabricated (#:prefab
) data and primitive types.
Message sizes don't get larger than about 1200 bytes. This fits well within the UDP size limits.
That pertains the entire networking module.
During each iteration of state, the system collects data from the network, the floor detectors, and the buttons. It then uses all this data and its previous state to compute where all elevators will be heading next.
This computation is done by each elevator, and serves as a method to make all elevators compute the same directions with the same knowledge. This allows elevators to be certain that their choice of floor is not already taken by another elevator.
If there is state lag due to the broadcast/receive latency, then everything is still fine.
The computation (when all elevators have updated all states) will converge on a single best-fit
elevator. So if one lags behind, that elevator may compute wrongly; say it decides it should not
service (request up 4 123)
. All other elevators think that the one lagging behind should take it.
During the next iteration. Because nobody has taken the task (because everyone thought the lagging
elevator was taking it), the lagging elevator may get an update that another elevator is no longer eligible
for (request up 4 123)
, so this elevator is able to acquire the request.
The elevators sleep before receiving, making it highly unlikely that an elevator lags behind. The sending is threaded and runs without interruptions (short sleeps only). These sleeps are more than two times shorter than the main sleep. This gives us a lot of confidence that a packet has arrived after main's sleep.
- Install Racket
https://racket-lang.org/download/
. - In a terminal:
raco pkg install --skip-installed lens libuuid reloadable sha threading
. git clone /~https://github.com/BourgondAries/elevators
.- Install
gcc
if you haven't already (for compiling the driver). - Optional: Install
dmd
(D compiler); only used by the simulator.
- Run the simulator
sim_server.d
, or connect the machine to the lab equipment. - Change the
simulated
variable insource/elevator-hardware/elevator-interface.rkt
to#t
or#f
. - Change the
floor-count
variable insource/data-structures.rkt
to9
for simulation or4
for lab equipment. - Run:
./main.rkt
orracket main.rkt
.
This is all that is needed to run the elevator control software.
The software supports hot-swappable code, so you can edit any rkt file and save it.
Edit any racket file using trce
, dbug
, info
, warn
, erro
, crit
, or ftal
(or any trce*
, dbug*
,... derivatives).
Save and observe the log statements.
Generated files will be stored in temporaries/
.
GPL-3.0
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
[at your option] any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.