联系方式

  • QQ:99515681
  • 邮箱:99515681@qq.com
  • 工作时间:8:00-21:00
  • 微信:codinghelp

您当前位置:首页 >> C/C++编程C/C++编程

日期:2020-03-21 11:07

COMP2401 - Assignment #5

In this assignment, you will make a simulator for a Dispatch Center that dispatches taxis to different

areas of a city to pick up customers and drop them off to other areas of the city. Your code will make

use of threads to handle incoming requests and display. The code will also fork multiple processes

to represent multiple taxis. You will then run separate processes as customers that connect to the

dispatch center to be picked up and dropped off by available taxis.

To begin this assignment, you should download the following files:

• simulator.h – contains definitions and structs that will be used throughout your code

• simulator.c – contains a template for the code that will run the simulator server

• dispatchCenter.c – contains a template for the code for a process that will run the dispatch

• display.c – contains the window/drawing code that you will use to display the city and taxis

• taxi.c – contains a template for the code for a process that will run a single taxi

• customer.c – contains a template for the code for a process that runs one customer request

• generator.c – contains a template for the code that will continuously generate customers

• stop.c – a program used to shut down the server

• makefile – the file that you will use to compile everything

You will generate (and use) five executables in this assignment:

1. simulator – this will open a widow to display everything as well as set up the dispatch server

2. stop – this will shut down the server

3. customer – this will run a process that will simulate a single customer request

4. generator – this will continuously generate random customers for testing

When compiling the files, you will need to include the -lm -lpthread and -lX11 libraries.

(but the -lX11 is only needed for the simulator program since it uses a window and graphics.)

(1) Examine the simulator.h file. It contains the following definitions and constants that represent

the city that we will be testing:

CITY_WIDTH 720 // The width & height of the city’s display window

CITY_HEIGHT 600

NUM_CITY_AREAS 6 // Number of main areas of the city

UNKNOWN_AREA -1 // An indicator that the city area is unknown

const char *AREA_NAMES[NUM_CITY_AREAS] = {"North", "East", "Airport",

"South", "Center", "West"};

const short AREA_X_LOCATIONS[6] = {360, 670, 670, 260, 400, 50};

const short AREA_Y_LOCATIONS[6] = { 60, 150, 380, 520, 280, 240};

The next page shows a screenshot of what the city will look like when printed. Notice that each of the

6 city areas has a name as well as a fixed (x,y) location and is displayed as a large circle of a certain

color. The smaller circles are moving taxis. A taxi will have the same color as a city area if it is on

its way there to drop off a customer. A taxi will be black if it is picking up a customer and white if it is

available and waiting for a customer. Each taxi has a 3-digit plate number as its ID. Each city area is

actually just an index from 0 to 5. An index of -1 will represent an UNKNOWN_AREA.

Ideally, it would be nice to see a map of the city, but we kept it simple and just chose 6 circles to

represent pickup and dropoff areas of the city.

As the program runs, the taxis

will travel back and forth from

one city area to another, as

customers are picked up and

dropped off. The travel time

estimates from one area of the

city to another area is defined

by this figure. Note that taxis

may pickup and drop off a

customer in the same area, but

this still takes 10 minutes. This

diagram has been hard-coded

into a constant 2D array which

is also defined in the

simulator.h file as follows.

Make sure that you understand

it.

const char TIME_ESTIMATES[NUM_CITY_AREAS][NUM_CITY_AREAS] =

{{10, 20, 40, 40, 20, 20},

{20, 10, 20, 20, 20, 40},

{40, 20, 10, 40, 40, 60},

{40, 20, 40, 10, 20, 20},

{20, 20, 40, 20, 10, 20},

{20, 40, 60, 20, 20, 10}};

A dispatch center is defined as follows:

typedef struct {

char online; // 0 = no, 1 = yes

Taxi *taxis[MAX_TAXIS]; // the taxis belonging to this dispatch center

int numTaxis; // the number of taxis in the dispatch center

Request requests[MAX_REQUESTS]; // customer requests

int numRequests; // number of customer requests in the system

} DispatchCenter;

Notice that it is basically two arrays … one for taxis and one for pending customer requests. Each of

these is set to be a maximum of 100 in size, but the numTaxis and numRequests should indicate

the actual number at any moment. numTaxis will remain constant in our program, whereas

numRequests will vary over time.

Each Request is just two small numbers indicating where the customer wants

to be picked up and where he/she wants to be dropped off as follows:

typedef struct {

char pickupLocation; // City area to be picked up in (0 to NUM_CTY_AREAS)

char dropoffLocation; // City area to be dropped off in (0 to NUM_CTY_AREAS)

} Request;

Each Taxi has a unique pID (i.e., process id, since each will run as a separate process). A Taxi also

has a unique plate number (0 to numTaxis) and their current (x,y) location in the city at any time. A

taxi has a status of being either AVAILABLE, PICKUP_UP someone or DROPPING_OFF someone.

The currentArea is the area that the taxi is currently in. The pickupArea is the area that the taxi is

on the way to in order to get the customer and the dropoffArea is the area that the customer is to be

dropped off at. Each of these areas is a number from 0 to NUM_CITY_AREAS or set to

UNKNOWN_AREA. Finally, the eta is the estimated time of arrival to the pickup or destination area:

typedef struct {

int pID; // process ID for this taxi

int x; // x location of the taxi

int y; // y location of the taxi

short plateNumber; // unique plate id for taxi

char status; // AVAILABLE, PICKING_UP, or DROPPING_OFF

char currentArea; // index of city area currently in

char pickupArea; // index of city area to pick up person in

char dropoffArea; // index of city area to drop off person in

short eta; // time left until reaching destination

} Taxi;

(2) The simulator.c file contains the code for the main application. You need to add code so that the

function does the following:

• It should dynamically-allocate a single Taxi and store it in the dispatch center. The new taxi

should be placed in a random area of the city (i.e., in the center of one of the city area

circles) and have plate number 0. The currentArea of the taxi should be set to that area,

while the pickup/dropoff areas should be UNKNOWN_AREA. New taxis should start off

being available for customers.

• It must use fork() to start a new process for that taxi … calling the runTaxi() function in the

taxi.c file … passing newly created taxi as its parameter. Upon returning from the

runTaxi() function, the taxi process should simply exit.

• It should spawn a thread to handle incoming requests. This thread should call the

handleIncomingRequests() function in the dispatchCenter.c file and pass in a pointer to

the ottawaDispatch dispatch center.

• It should spawn a thread to handle the displaying of the city. This thread should call the

showSimulation() function in the display.c file and pass in a pointer to the

ottawaDispatch dispatch center.

• The code should then wait for the request-handling thread to complete and it should then

kill all running taxi processes.

• The code should free all allocated memory so that valgrind shows no leaks.

The showSimulation() function (in the display.c file) has been completed for you. You MUST

NOT alter any code in the display.c file. Use the makefile that was given to you to compile

everything. Once you have this step completed, run the simulator in the background (i.e., use &).

Make sure that you see the screen snapshot (on the previous page) appear … showing the city

areas and a single taxi in one of the areas. Once you see this, you can be sure that the display

thread is working properly. Type ps in the terminal window. You should see two processes

labelled simulator. That is because you have a taxi process running with the same name. When

you close the window, you should see some XIO error (don’t worry about this). After closing the

window, you can type ps again … you should see that one simulator process is no longer running,

but you will likely see the taxi process (also called simulator) still running. You should kill this

process in the terminal using kill followed by the process id.

(3) Write code in the handleIncomingRequests() function (from dispatchCenter.c) so that it first

starts up a server and then goes into an infinite loop that repeatedly waits for incoming user

requests and handles them. The idea is that taxis will communicate with the server to ask for a

customer to pickup. When the server receives a REQUEST_CUSTOMER command from a taxi,

it should check to see if there is a customer request available in its requests buffer. If not, the taxi

should be told that no request is available. Otherwise, the taxi should be given the pickup and

dropoff location for the customer … and this customer request should be removed from the

request buffer so that no other taxis get that request. While on route to pickup or drop off the

customer … the taxi will send an UPDATE command to the server along with its plate number,

current x and y location, status and the dropoffArea. These should be recorded at the dispatch

center so that the dispatch center always has each taxi’s latest information. Customers will send

a REQUEST_TAXI command to the server which will contain the requested pickup and dropoff

locations as city area indices. If the maximum number of requests has been reached, the

customer should be told that the request cannot be handled. Otherwise, it should be told that all is

ok and the request should be added to the buffer. Requests should always be handled on a firstcome-first-served

basis. Finally, a SHUTDOWN command should cause the server to go off line

(gracefully) when it is received. Make sure the code compiles, but you cannot test it until you do

the next step.

(4) Complete the code in the stop.c file so that it attempts to shut down the dispatch center server by

sending the SHUTDOWN command to it. Test your code by running the simulator server in the

background and then running the stop program. If all works well, the simulator window should

close and should shut down gracefully. Use ps to make sure that the simulator has indeed shut

down properly as well as the stop program. Make sure that you don’t have any segmentation

faults.

(5) Complete the runTaxi() function in the taxi.c file so that it runs in an infinite loop. If the taxi is

available, it should contact the dispatch center to get a request. If it is busy picking up or

dropping off a customer, it should send updates to the dispatch center. A delay has been added

to the loop so that it does not overwhelm the server with many requests. Here is how to do these

two steps (you may want them done as helper functions):

a. To request a customer from the dispatch center, the taxi should first attempt to connect to

the dispatch center and then if successful, it should send the REQUEST_CUSTOMER

command. If the server responds with a YES, then the taxi should handle the customer

request as follows:

i. If the pickup area is the same as the area the taxi is currently in, assume that the

pickup is immediate. Then the taxi just needs to head to the dropoff area. The

taxi’s currentArea, pickupArea and dropoffArea should all be set properly and the

taxi should be in DROPPING_OFF mode. The estimated time of arrival (eta) should

be set accordingly based on the 2D TIME_ESTIMATES array from the pickupArea

to the dropoffArea.

ii. If the pickup area is different from the area that the taxi is currently in, then the taxi

should go into PICKING_UP mode. Make sure to set the pickupArea and

dropoffArea accordingly. The eta should be set according to the time it takes to get

from the currentArea to the pickupArea.

b. To send an update to the dispatch center, the taxi should first attempt to connect to the

dispatch center and then if successful, it should send the UPDATE command along with its

(x,y) location, plate number, status and dropoffArea.

You will also need to make the taxi move ahead a little bit each time in the loop. You may want to

make a helper function for this as well. To do this … follow this logic:

IF (taxi is PICKING_UP) {

deltaX = (x of pickup area – x of current area) / (time estimate from current area to pickup area)

deltaY= (y of pickup area – y of current area) / (time estimate from current area to pickup area)

}

ELSE {

deltaX = (x of dropoff area – x of pickup area) / (time estimate from pickup area to dropoff area)

deltaY= (y of dropoff area – y of pickup area) / (time estimate from pickup area to dropoff area)

}

New taxi position = (x + deltaX, y + dektaY)

New eta should now be one second less.

You will need to check for when the taxi arrives at its pickup or destination locations. If it is

DROPPING_OFF a customer and the eta has decreased to 0, then the taxi’s (x,y) location should

be set to the dropoff area’s center, the taxi should be AVAILABLE now, the currentArea should be

the dropoffArea and the pickup and dropoff areas should be unknown now. If the taxi is

PICKING_UP a customer and reaches the pickup location (i.e., eta down to 0), the taxi’s (x,y)

location should be the pickup area’s center, the taxi should go into DROPPING_OFF mode, the

current Area should be unknown now (as it is travelling) and the eta should be set to the

TIME_ESTIMATE from the pickup to dropoff areas.

(6) Complete the customer.c program so that it takes two command line arguments. The first being

the pickup area index and the 2nd being the dropoff area index. It should connect to the dispatch

center server and send a single request according to the command line arguments. There is no

loop … just a single request should be sent and the response received. If the request was

denied, a message should be printed to the screen to indicate this.

(7) As a final step, complete the generator.c program so that it keeps generating customer requests

at random pickup and dropoff areas in the city by using the system() command. Once you get it

working… run the simulator in the background. Then run the generator in the background as

well. You should see the taxis being busy picking up and dropping off clients. Finally, stop the

generator (you can use fg to get it back into the foreground and then use CNTRL+C to stop it).

You should see the requests stop generating and eventually all the taxis will lie still at one of the

city area centers. Use ps to make sure that all customer processes are completed, with the

exception of the simulator. Run the generator again for a bit. Then stop it again … and while

there are still some customers being shown … run the stop program. It should shut down the

server gracefully. Run ps again. All processes should be stopped and there should be no

simulator nor customer processes running.

(8) Modify the simulator.c code so that 10 taxis are running. Make sure that the code still works.

________________________________________________________________________________

IMPORTANT SUBMISSION INSTRUCTIONS:

Submit all of your c source code files (and makefile) as a single tar file containing:

1. A Readme text file containing

• your name and studentNumber

• a list of source files submitted

• any specific instructions for compiling and/or running your code

2. All of your .c source files and all other files needed for testing/running your programs.

3. Any output files required, if there are any.

The code MUST compile and run on the course VM.

• If your internet connection at home is down or does not work, we will not accept this as a reason for

handing in an assignment late ... so make sure to submit the assignment WELL BEFORE it is due !

• You WILL lose marks on this assignment if any of your files are missing. So, make sure that you hand

in the correct files and version of your assignment. You will also lose marks if your code is not written

neatly with proper indentation and containing a reasonable number of comments. See course

notes for examples of what is proper indentation, writing style and reasonable commenting).

________________________________________________________________________________


版权所有:编程辅导网 2021 All Rights Reserved 联系方式:QQ:99515681 微信:codinghelp 电子信箱:99515681@qq.com
免责声明:本站部分内容从网络整理而来,只供参考!如有版权问题可联系本站删除。 站长地图

python代写
微信客服:codinghelp