Lamport's bakery algorithm
Lamport's bakery algorithm is a computer algorithm, devised by computer scientist Dr Leslie Lamport, which is intended to improve the robustness of multiple thread-handling processes by means of mutual exclusion.
Nature of the problem
In computer science, it is common for multiple threads to simultaneously access the same resources. They either try to write into the same memory location, or one thread reads a location before the other has finished writing into it. This is undesirable as it can end in hazardous loss of data. Lamport's bakery algorithm is one of many mutual exclusion algorithms.
Algorithm
Analogy
Lamport envisaged a bakery with a numbering machine at its entrance so each customer is given a unique number. Numbers increase by one as customers enter the store. A global counter displays the number of the customer that is currently being served. All other customers must wait in a queue until the baker finishes serving the current customer and the next number is displayed. When done shopping, the customer loses their number and can then do whatever they want, except for shopping without getting a new number.
In the computer world, the 'customers' will be threads, identified by the letter i, obtained from a global variable.
Due to the limitations of computer architecture, some parts of the Lamport's analogy need slight modification. It is possible that more than one thread will get the same number when they request it; this cannot be avoided. Therefore, it is assumed that the thread identifier i is also a priority identifier. A lower value of i means a higher priority and threads with higher priority will enter the critical section first.
Critical section
The critical section is that part of code that requires exclusive access to resources and may only be executed by one thread at a time. In the bakery analogy, it is when the customer trades with the baker and others must wait.
When a thread wants to enter the critical section, it has to check whether it is its turn to do so. It should check the numbers of every other thread to make sure that it has the smallest one. In case another thread has the same number, the thread with the smallest i will enter the critical section first.
In pseudocode this comparison will be written in the form:
(a, b) < (c, d)
which is equivalent to:
(a < c) or ((a == c) and (b < d))
Once the thread ends its critical job, it gets rid of its number and enters the non-critical section.
Non-critical section
The non-critical section is the part of code that doesn't need exclusive access. It represents some thread-specific computation that doesn't interfere with other threads' resources and execution.
This part is analogous to actions that occur after shopping, such as like putting change back in the wallet.
Algorithm in pseudocode
// declaration and initial values of global variables LastNumber: integer = 0; Enter, Number: array [1..N] of integer = {0}; 1 Thread(i) { 2 while (1) { 3 Enter[i] = 1; 4 Number[i] = LastNumber + 1; 5 LastNumber = Number[i]; 6 Enter[i] = 0; 7 for (j = 1; j <= N; j++) { 8 while (Enter[j] == 1) { 9 // wait until thread j receives its number 10 } 11 while ((Number[j] != 0) && ((Number[j], j) < (Number[i], i))) { 12 // wait until threads with smaller numbers or with the same 13 // number, but with higher priority, finish their work 14 } 15 } 16 // critical section... 17 Number[i] = 0; 18 // non-critical section... 19 } 20 }
Note: The thread also checks itself before entering the critical section, but that doesn't cause any delays since every condition will be evaluated as false.