Presentation is loading. Please wait.

Presentation is loading. Please wait.

Andy Wang Operating Systems COP 4610 / CGS 5765

Similar presentations


Presentation on theme: "Andy Wang Operating Systems COP 4610 / CGS 5765"— Presentation transcript:

1 Andy Wang Operating Systems COP 4610 / CGS 5765
Deadlocks Andy Wang Operating Systems COP 4610 / CGS 5765

2 Deadlocks Deadlocks: Occur when threads are waiting for resources with circular dependencies Often involve nonpreemptable resources, which cannot be taken away from its current thread without failing the computation (e.g., storage allocated to a file)

3 Deadlocks Deadlocks that involve preemptable resources (e.g., CPU) can usually be resolved by reallocation of resources Starvation: a thread waits indefinitely A deadlock implies starvation

4 An Example of Deadlocks
Thread A Thread B P(x); P(y); P(y); P(x); A deadlock won’t always happen with this code, but it might

5 Deadlocks, Deadlocks, Everywhere…
Can happen with any kind of resource Among multiple resources Cannot be resolved for each resource independently A thread can grab all the memory The other grabs all the disk space Each thread may need to wait for the other to release

6 Deadlocks, Deadlocks, Everywhere…
Round-Robin CPU scheduling cannot prevent deadlocks (or starvation) from happening Can occur whenever there is waiting…

7 A Classic Example of Deadlocks
Dinning lawyers (philosophers) Each needs two chopsticks to eat

8 Dining Lawyers If each first grabs the chopstick on their right before the one on their left, and all grab at the same time, we have a deadlock (Personally, I prefer to starve than share chopsticks…)

9 A Dining Lawyer Implementation
semaphore chopstick[5] = {1, 1, 1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[j]); P(chopstick[(j + 1) % 5]; // eat V(chopstick[(j + 1) % 5]; V(chopstick[j]); }

10 A Dining Lawyer Implementation
// chopstick[5] = {0, 0, 0, 0, 0}; lawyer(int j) { while (TRUE) { P(chopstick[j]); P(chopstick[(j + 1) % 5]; // eat V(chopstick[(j + 1) % 5]; V(chopstick[j]); }

11 A Dining Lawyer Implementation
// chopstick[5] = {0, 0, 0, 0, 0}; lawyer(int j) { while (TRUE) { P(chopstick[j]); P(chopstick[(j + 1) % 5]; // eat V(chopstick[(j + 1) % 5]; V(chopstick[j]); }

12 Conditions for Deadlocks
Four necessary (but not sufficient) conditions Limited access (lock-protected resources) No preemption (if someone has the resource, it cannot be taken away) Wait while holding (holding a resource while requesting and waiting for the next resource) Circular chain of requests

13 Deadlock Prevention Techniques
All four conditions must be true If deadlock, then all four conditions are true All four conditions are true  Deadlock To prevent deadlocks Remove one of the four conditions

14 Deadlock Prevention Techniques
1. Infinite resources (buy a very large disk)

15 Deadlock Prevention Techniques
2. No sharing (independent threads)

16 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) easier said than done…

17 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) semaphore chopstick[5] = {1, 1, 1, 1, 1}, s = 1; lawyer(int j) { while (TRUE) { P(s); P(chopstick[j]); P(chopstick[(j + 1) % 5]; // eat V(chopstick[(j + 1) % 5]; V(chopstick[j]); V(s); }

18 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) semaphore chopstick[5] = {1, 1  0, 1  0, 1, 1}, s = 1  0; lawyer(int j) { while (TRUE) { P(s); P(chopstick[j]); P(chopstick[(j + 1) % 5]; // eat V(chopstick[(j + 1) % 5]; V(chopstick[j]); V(s); } 1

19 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) // chopstick[5] = {1, 0, 0, 1, 1}, s = 0; lawyer(int j) { while (TRUE) { P(s); P(chopstick[j]); P(chopstick[(j + 1) % 5]; // eat V(chopstick[(j + 1) % 5]; V(chopstick[j]); V(s); } 3 1

20 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) // chopstick[5] = {1, 1, 1, 1, 1}, s = 1; lawyer(int j) { while (TRUE) { P(s); P(chopstick[j]); P(chopstick[(j + 1) % 5]; V(s); // eat V(chopstick[(j + 1) % 5]; V(chopstick[j]); }

21 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) // chopstick[5] = {1, 1  0, 1  0, 1, 1}, s = 1  0  1; lawyer(int j) { while (TRUE) { P(s); P(chopstick[j]); P(chopstick[(j + 1) % 5]; V(s); // eat V(chopstick[(j + 1) % 5]; V(chopstick[j]); } 1

22 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) // chopstick[5] = {1, 0, 0, 1  0, 1  0}, s = 1  0  1; lawyer(int j) { while (TRUE) { P(s); P(chopstick[j]); P(chopstick[(j + 1) % 5]; V(s); // eat V(chopstick[(j + 1) % 5]; V(chopstick[j]); } 1 3

23 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) // chopstick[5] = {1, 0, 0, 0, 0}, s = 1  0; lawyer(int j) { while (TRUE) { P(s); P(chopstick[j]); // deadlock P(chopstick[(j + 1) % 5]; V(s); // eat V(chopstick[(j + 1) % 5]; V(chopstick[j]); } 2 1 3

24 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) // chopstick[5] = {1, 0, 0, 0, 0}, s = 0; lawyer(int j) { while (TRUE) { P(s); P(chopstick[j]); P(chopstick[(j + 1) % 5]; V(s); // eat P(s); // deadlock V(chopstick[(j + 1) % 5]; V(chopstick[j]); } 2 1 3

25 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) int counter[5] = {1, 1, 1, 1, 1}; semaphore chopstick[5] = {1, 1, 1, 1, 1}, s = 1; lawyer(int j) { while (TRUE) { P(s); // if both counters j and (j + 1) % 5 > 0, decrement counters // and grab chopstick[j] and chopstick[(j + 1) % 5] V(s); // if holding both chopsticks, eat // release chopsticks and increment counters as needed }

26 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) // counter[5] = {1, 0, 0, 1, 1}; chopstick[5] = {1, 0, 0, 1, 1}, s = 1; lawyer(int j) { while (TRUE) { P(s); // if both counters j and (j + 1) % 5 > 0, decrement counters // and grab chopstick[j] and chopstick[(j + 1) % 5] V(s); // if holding both chopsticks, eat // release chopsticks and increment counters as needed } 1 (1, 2)

27 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) // counter[5] = {1, 0, 0, 1, 1}; chopstick[5] = {1, 0, 0, 1, 1}, s = 1; lawyer(int j) { while (TRUE) { P(s); // if both counters j and (j + 1) % 5 > 0, decrement counters // and grab chopstick[j] and chopstick[(j + 1) % 5] V(s); // if holding both chopsticks, eat // release chopsticks and increment counters as needed } 1 (1, 2) 2 ()

28 Deadlock Prevention Techniques
3. Allocate all resources at the beginning (if you need 2 chopsticks, grab both at the same time) // counter[5] = {1, 0, 0, 0, 0}; chopstick[5] = {1, 0, 0, 0, 0}, s = 1; lawyer(int j) { while (TRUE) { P(s); // if both counters j and (j + 1) % 5 > 0, decrement counters // and grab chopstick[j] and chopstick[(j + 1) % 5] V(s); // if holding both chopsticks, eat // release chopsticks and increment counters as needed } 1 3 (1, 2) (3, 4)

29 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {1, 1, 1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); }

30 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {1  0, 1, 1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); }

31 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 1  0, 1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 1

32 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 1  0, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 1 2

33 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 1  0, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 1 2 3

34 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 0, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 1 2 3 4

35 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 0, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 1 2 3 4

36 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 0, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 2 3 4 1

37 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 0, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 3 4 1 2

38 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 0, 1  0}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

39 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 0, 0}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

40 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 0, 0  1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

41 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 0  1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

42 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 1  0, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

43 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 0, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

44 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 0  1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

45 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0  1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

46 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 1  0, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

47 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

48 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 0  1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

49 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0  1, 1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

50 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 1  0, 1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

51 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0, 1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

52 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 0  1, 1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

53 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0  1, 1, 1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

54 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {1  0, 1, 1, 1, 1}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

55 Deadlock Prevention Techniques
4. Make everyone use the same ordering in accessing resource (All threads must call P(x) before P(y) // chopstick[5] = {0, 1, 1, 1, 1  0}; lawyer(int j) { while (TRUE) { P(chopstick[min(j, (j + 1) % 5)]); P(chopstick[max(j, (j + 1) % 5)]); // eat V(chopstick[max(j, (j + 1) % 5)]); V(chopstick[min(j, (j + 1) % 5)]); } 4 1 2 3

56 More Deadlock Prevention Methods
5. No waiting (phone company) 6. Preempt resources (copying memory content to disk) 7. Banker’s algorithm 8. A combination of techniques

57 Banker’s Algorithm The idea of Banker’s algorithm:
Allows the sum of requested resources > total resources As long as, there is some way for all threads to finish without getting into any deadlocks

58 Banker’s Algorithm Banker’s algorithm:
A thread states its maximum resource needs in advance The OS allocates resource dynamically as needed. A thread waits if granting its request would lead to deadlocks A request can be granted if some sequential ordering of threads is deadlock free

59 Example 1 Total RAM 256 MB Allocated Still needed P1 80 MB 30 MB P2
free RAM (MB) 46 time

60 Example 1 Total RAM 256 MB Allocated Still needed P1 80 MB 30 MB P2
free RAM (MB) 46 36 P2 time

61 Example 1 Total RAM 256 MB Allocated Still needed P1 80 MB 30 MB P2
free RAM (MB) 56 46 36 P2 time

62 Example 1 Total RAM 256 MB Allocated Still needed P1 110 MB 0 MB P2 P3
free RAM (MB) 56 46 36 26 P2 P1 time

63 Example 1 Total RAM 256 MB Allocated Still needed P1 0 MB P2 P3 120 MB
free RAM (MB) 136 56 46 36 26 P2 P1 time

64 Example 1 Total RAM 256 MB Allocated Still needed P1 0 MB P2 P3 200 MB
free RAM (MB) 136 56 56 46 36 26 P2 P1 P3 time

65 Example 1 Total RAM 256 MB Allocated Still needed P1 0 MB P2 P3 256
free RAM (MB) 136 56 56 46 36 26 P2 P1 P3 time

66 Example 1 Total RAM 256 MB Allocated Still needed P1 80 MB 30 MB P2
free RAM (MB) 136 The system is initially in a safe state, since we can run P2, P1, and then P3 56 56 46 36 26 P2 P1 P3 time

67 Example 2 Total RAM 256 MB Allocated Still needed P1 80 MB 60 MB P2
free RAM (MB) 46 time

68 Example 2 Total RAM 256 MB Allocated Still needed P1 80 MB 60 MB P2
free RAM (MB) 46 36 P2 time

69 Example 2 Total RAM 256 MB Allocated Still needed P1 80 MB 60 MB P2
free RAM (MB) 56 46 36 P2 time

70 Example 2 Total RAM 256 MB Allocated Still needed P1 80 MB 60 MB P2
free RAM (MB) 56 46 36 P2 time

71 Example 2 Total RAM 256 MB Allocated Still needed P1 80 MB 60 MB P2
free RAM (MB) The system is initially in an unsafe state, since we cannot find an execution sequence for P1, P2, and P3 56 46 36 P2 time

72 Deadlock Detection and Recovery
Scan the resource allocation graph Detect circular chains of requests Recover from the deadlock

73 Resource Allocation Graph
Thread A owned by Thread B waiting for Resource Y Resource X

74 Once A Cycle is Detected…
Some possible actions Kill a thread and force it to give up resources Remaining system may be in an inconsistent state Rollback actions of a deadlocked thread Not always possible (a file maybe half-way through modifications) Need checkpointing, or taking snapshots of system states from time to time


Download ppt "Andy Wang Operating Systems COP 4610 / CGS 5765"

Similar presentations


Ads by Google