# CS 1550 Week 5 – Lab 2 Synchronization with XV6 Teaching Assistant Xiaoyu (Veronica) Liang #### CS 1550 – Lab 2 • **Due**: Friday, October 4<sup>th</sup> @11:59pm #### Locks – Processes without sharing CPU • OS chooses another processes to execute once the first finishes #### Locks – Processes without sharing CPU What if P1 is a big process? Solution switch processes during their execution. - What happens in Parent-Child Process scenario? - How to keep integrity/correctness on race conditions? ``` struct list { int data; struct list *next; }; struct list *list = 0; void insert(int data) { struct list *1; l = malloc(sizeof *l); 1->data = data; 1->next = list; list = 1; ``` ``` struct list { int data; struct list *next; }; CPU P1 struct list *list = 0; void insert(int data) { struct list *1; P1 stops here the l = malloc(sizeof *1); OS switches to P2 l->data = data; 1->next = list; list = 1; ``` ``` struct list { int data; struct list *next; }; CP2 CPU P1 struct list *list = 0; void insert(int data) { struct list *1; P2 gets the same l = malloc(sizeof *1); reference to the P1 stopped 1->data = data; same block of 1->next = list; data of list and list = 1; overwrites it ``` ``` struct list { int data; struct list *next; }; struct list *list = 0; void insert(int data) { struct list *1; l = malloc(sizeof *1); l->data = data; CP2 stopped l->next = list; list = 1; ``` CPU P1 CP2 P1 When P1 comes back it will have written the wrong data **Race condition**: A race condition is an undesirable condition that happened when having multiple processes running on a piece of data which does not use any exclusive locks to control access. - Sharing CPU among processes - Ensuring data integrity/correctness - Ensure that a critical section of your code is only executed by one process ``` struct list *list = 0; struct lock listlock; void insert(int data) struct list *1; acquire(&listlock); 1 = malloc(sizeof *1); 1->data = data; l->next = list; list = 1; release(&listlock); ``` ``` struct list *list = 0; struct lock listlock; void insert(int data) CPU CP2 P1 P1 struct list *1; acquire(&listlock); 1 = malloc(sizeof *1); P1 gets locks the lock 1->data = data; 1->next = list; list = 1; release(&listlock); ``` ``` struct list *list = 0; struct lock listlock; void insert(int data) CPU CP2 P1 P1 struct list *1; acquire(&listlock); l = malloc(sizeof *1); P1 gets locks the lock l->data = data;✓ 1->next = list; list = 1; release(&listlock); ``` list = 1; release(&listlock); ``` struct list *list = 0; struct lock listlock; void insert(int data) CP2 CPU P1 struct list *1; acquire(&listlock); 1 = malloc(sizeof *1); When the OS schedule CP2 P1 stopped 1->data = data; 1->next = list; ``` ``` struct list *list = 0; struct lock listlock; void insert(int data) CPU CP2 P1 P1 struct list *1; acquire(&listlock); l = malloc(sizeof *l); It will try to get the lock but won't. P1 stopped 1->data = data; 1->next = list; It will be constantly try to get it (in a loop). list = 1; Until the OS switches back to P1 release(&listlock); ``` ``` struct list *list = 0; struct lock listlock; void insert(int data) CPU P1 CP2 P1 struct list *1; acquire(&listlock); CP2 stopped l = malloc(sizeof *l); 1->data = data; P1 release the lock P2 will finally be 1->next = list; able to execute, once scheduled list = 1; release(&listlock); ``` ``` struct lock listlock; void insert(int data) struct list *1; CP2 proceeds acquire(&listlock); l = malloc(sizeof *l); 1->data = data; 1->next = list; list = 1; release(&listlock); ``` struct list \*list = 0; P1 release the lock P2 will finally be able to execute, once scheduled #### SpinLock ``` Void acquire(struct spinlock *lk) { while(!lk->locked) ; /* busy wait */ lk->locked = 1; } ``` - Keep spinning until find lock is released - But we can have the same issue as before - We need to check and lock atomically - XV6 relies on a special 386 hardware instruction, xchg - Atomically check and change a register value - xchg(&lk->locked, 1) - Swap a word in memory with the contents of a register - In acquire function: - loop xchg instruction - Each round atomically read lock and set the lock to 1 ``` void acquire(struct spinlock *lk) pushcli(); // disable interrupts to avoid deadlock. ^{\prime}/ The xchg is atomic. while (xchg(\&lk -> locked, 1) != 0); // Record info about lock acquisition for debugging. 1k - cpu = mycpu(); getcallerpcs(&lk, lk->pcs); ``` - But the we have another issue - Busy waiting - Spin Lock - Busy waiting - Useful for short critical sections - E.g. increment a counter, access an array element, etc. - Not useful, when the period of wait is unpredictable or will take a long time - E.g. read page from disk - Sleep Locks - For code need to hold a lock for a long time (read/write to disk) - Avoids the schedule of "spin locked" processes - Sleep Locks - For code need to hold a lock for a long time (read/write to disk) - Avoids the schedule of "spin locked" processes ``` void acquiresleep(struct sleeplock *lk) { acquire(&lk->lk); while (lk->locked) { sleep(lk, &lk->lk); lk->locked = 0; sleep(lk, &lk->lk); lk->pid = 0; wakeup(lk); release(&lk->lk); release(&lk->lk); } release(&lk->lk); } ``` Put one process to sleep waiting for event Mark current process as sleeping Call sched() to release the processor ``` void sleep(void *chan, struct spinlock *lk) { struct proc *p = myproc(); ... p->state = SLEEPING; sched(); ... } ``` #### Sanity Checks - Must be a current process - Must have been passed a lock - Put one process to sleep waiting for event Mark current process as sleeping Call sched() to release the processor ``` void sleep(void *chan, struct spinlock *lk) struct proc *p = myproc(); if(p == 0) panic("sleep"); if(lk == 0) panic("sleep without lk"); if(lk != &ptable.lock) { acquire(&ptable.lock); release(lk); Hold the ptable·lock, it is safe to release Ik p->chan = chan; p->state = SLEEPING; sched(); p->chan = 0 if(lk != &ptable.lock) { release(&ptable.lock); acquire(lk); ``` - Wake up process when event happened - Mark a waiting process as runnable ``` static void wakeup(void *chan) { acquire(&ptable.lock); wakeup1(chan); release(&ptable.lock); } ``` # CS 1550 Week 5 – Lab 2 Synchronization with XV6 Teaching Assistant Xiaoyu (Veronica) Liang