52
CMSC 330: Organization of Programming Languages Threads

CMSC 330: Organization of Programming Languages Threads

Embed Size (px)

Citation preview

Page 1: CMSC 330: Organization of Programming Languages Threads

CMSC 330: Organization of Programming Languages

Threads

Page 2: CMSC 330: Organization of Programming Languages Threads

CMSC 330 2

Computation Abstractions

CPU 1 CPU 2

p3p1 p2 p4

t1

t2

t1

t2

t3

t1

t4

t5

A computer

Processes(e.g., JVM’s)

Threads

Page 3: CMSC 330: Organization of Programming Languages Threads

CMSC 330 3

Processes vs. Threads

int x;foo() {…x…}

int x;foo() {…x…}

int x;

foo() { …x…}

foo() { …x…}

Processes do notshare data

Threads share datawithin a process

Page 4: CMSC 330: Organization of Programming Languages Threads

CMSC 330 4

So, What Is a Thread?

• Conceptually: it is a parallel computation occurring within a process

• Implementation view: it’s a program counter and a stack. The heap and static area are shared among all threads

• All processes have at least one thread (main)– Programs vs. processes

Page 5: CMSC 330: Organization of Programming Languages Threads

CMSC 330 5

Implementation View

• Per-thread stack and instruction pointer– Saved in memory when thread suspended– Put in hardware esp/eip when thread resumes

eip

eip

eipesp

esp

esp

Page 6: CMSC 330: Organization of Programming Languages Threads

CMSC 330 6

Tradeoffs

• Threads can increase performance– Parallelism on multiprocessors– Concurrency of computation and I/O

• Natural fit for some programming patterns– Event processing– Simulations

• But increased complexity– Need to worry about safety, liveness, composition

• And higher resource usage

Page 7: CMSC 330: Organization of Programming Languages Threads

CMSC 330 7

Programming Threads

• Threads are available in many languages– C, C++, Objective Caml, Java, SmallTalk …

• In many languages (e.g., C and C++), threads are a platform specific add-on– Not part of the language specification

• They're part of the Java language specification

Page 8: CMSC 330: Organization of Programming Languages Threads

CMSC 330 8

Java Threads

• Every application has at least one thread– The “main” thread, started by the JVM to run the

application’s main() method

• main() can create other threads– Explicitly, using the Thread class– Implicitly, by calling libraries that create threads as a

consequence• RMI, AWT/Swing, Applets, etc.

Page 9: CMSC 330: Organization of Programming Languages Threads

CMSC 330 9

Thread Creation

execution (time) main thread

thread starts

thread starts

thread endsthreadjoin

Page 10: CMSC 330: Organization of Programming Languages Threads

CMSC 330 10

Thread Creation in Java

• To explicitly create a thread:– Instantiate a Thread object

• An object of class Thread or a subclass of Thread

– Invoke the object’s start() method• This will start executing the Thread’s run() method

concurrently with the current thread

– Thread terminates when its run() method returns

Page 11: CMSC 330: Organization of Programming Languages Threads

CMSC 330 11

Running Example: Alarms

• Goal: To set alarms which will be triggered in the future– Input: time t (seconds) and message m– Result: we’ll see m printed after t seconds

Page 12: CMSC 330: Organization of Programming Languages Threads

CMSC 330 12

Example: Synchronous alarms

while (true) { System.out.print("Alarm> ");

// read user input String line = b.readLine(); parseInput(line); // sets timeout

// wait (in secs) try { Thread.sleep(timeout * 1000); } catch (InterruptedException e) { } System.out.println("("+timeout+") "+msg);}

like phone calls

thrown when another thread calls interrupt

Page 13: CMSC 330: Organization of Programming Languages Threads

CMSC 330 13

Making It Threaded (1)

public class AlarmThread extends Thread { private String msg = null; private int timeout = 0;

public AlarmThread(String msg, int time) { this.msg = msg; this.timeout = time; } public void run() { try { Thread.sleep(timeout * 1000); } catch (InterruptedException e) { } System.out.println("("+timeout+") "+msg); }}

Page 14: CMSC 330: Organization of Programming Languages Threads

CMSC 330 14

Making It Threaded (2)

while (true) { System.out.print("Alarm> "); // read user input String line = b.readLine(); parseInput(line); if (m != null) { // start alarm thread Thread t = new AlarmThread(m,tm); t.start(); }}

Page 15: CMSC 330: Organization of Programming Languages Threads

CMSC 330 15

Alternative: The Runnable Interface

• Extending Thread prohibits a different parent

• Instead implement Runnable– Declares that the class has a void run() method

• Construct a Thread from the Runnable– Constructor Thread(Runnable target)– Constructor Thread(Runnable target, String name)

Page 16: CMSC 330: Organization of Programming Languages Threads

CMSC 330 16

Thread Example Revisitedpublic class AlarmRunnable implements Runnable { private String msg = null; private int timeout = 0;

public AlarmRunnable(String msg, int time) { this.msg = msg; this.timeout = time; } public void run() { try { Thread.sleep(timeout * 1000); } catch (InterruptedException e) { } System.out.println("("+timeout+") "+msg); }}

Page 17: CMSC 330: Organization of Programming Languages Threads

CMSC 330 17

Thread Example Revisited (2)

while (true) { System.out.print("Alarm> "); // read user input String line = b.readLine(); parseInput(line); if (m != null) { // start alarm thread Thread t = new Thread( new AlarmRunnable(m,tm)); t.start(); }}

Page 18: CMSC 330: Organization of Programming Languages Threads

CMSC 330 18

Notes: Passing Parameters

• run() doesn’t take parameters

• We “pass parameters” to the new thread by storing them as private fields in the class that extends Runnable– Example: the time to wait and the message to print in

the AlarmThread class

Page 19: CMSC 330: Organization of Programming Languages Threads

CMSC 330 19

Concurrency• A concurrent program is one that has multiple

threads that may be active at the same time– Might run on one CPU

• The CPU alternates between running different threads• The scheduler takes care of the details

– Switching between threads might happen at any time

– Might run in parallel on a multiprocessor machine• One with more than one CPU• May have multiple threads per CPU

• Multiprocessor machines are becoming more common– Multi-CPU machines aren't that expensive any more– Dual-core CPUs are available now

Page 20: CMSC 330: Organization of Programming Languages Threads

CMSC 330 20

Scheduling Example (1)

CPU 1

CPU 2

p1

p2

p1

p2

One process per CPU

p2 threads: p1 threads:

Page 21: CMSC 330: Organization of Programming Languages Threads

CMSC 330 21

Scheduling Example (2)

CPU 1

CPU 2

p1

p2

p1

p2

Threads shared between CPUs

p2 threads: p1 threads:

Page 22: CMSC 330: Organization of Programming Languages Threads

CMSC 330 22

Concurrency and Shared Data

• Concurrency is easy if threads don’t interact– Each thread does its own thing, ignoring other threads– Typically, however, threads need to communicate with

each other

• Communication is done by sharing data– In Java, different threads may access the heap

simultaneously– But the scheduler might interleave threads arbitrarily– Problems can occur if we’re not careful.

Page 23: CMSC 330: Organization of Programming Languages Threads

CMSC 330 23

Data Race Examplepublic class Example extends Thread { private static int cnt = 0; // shared state public void run() { int y = cnt; cnt = y + 1; } public static void main(String args[]) { Thread t1 = new Example(); Thread t2 = new Example(); t1.start(); t2.start(); }}

Page 24: CMSC 330: Organization of Programming Languages Threads

CMSC 330 24

Data Race Example

static int cnt = 0;t1.run() { int y = cnt; cnt = y + 1;}t2.run() { int y = cnt; cnt = y + 1;}

cnt = 0

Start: both threads ready torun. Each will increment theglobal cnt.

Shared state

Page 25: CMSC 330: Organization of Programming Languages Threads

CMSC 330 25

Data Race Example

static int cnt = 0;t1.run() { int y = cnt; cnt = y + 1;}t2.run() { int y = cnt; cnt = y + 1;}

cnt = 0

T1 executes, grabbingthe global counter value intoits own y.

Shared state

y = 0

Page 26: CMSC 330: Organization of Programming Languages Threads

CMSC 330 26

Data Race Example

static int cnt = 0;t1.run() { int y = cnt; cnt = y + 1;}t2.run() { int y = cnt; cnt = y + 1;}

cnt = 1

T1 executes again, storing itsvalue of y + 1 into the counter.

Shared state

y = 0

Page 27: CMSC 330: Organization of Programming Languages Threads

CMSC 330 27

Data Race Example

static int cnt = 0;t1.run() { int y = cnt; cnt = y + 1;}t2.run() { int y = cnt; cnt = y + 1;}

cnt = 1

T1 finishes. T2 executes, grabbing the globalcounter value into its own y.

Shared state

y = 0

y = 1

Page 28: CMSC 330: Organization of Programming Languages Threads

CMSC 330 28

Data Race Example

static int cnt = 0;t1.run() { int y = cnt; cnt = y + 1;}t2.run() { int y = cnt; cnt = y + 1;}

cnt = 2

T2 executes, storing itsincremented cnt value intothe global counter.

Shared state

y = 0

y = 1

Page 29: CMSC 330: Organization of Programming Languages Threads

CMSC 330 29

But When it's Run Again?

Page 30: CMSC 330: Organization of Programming Languages Threads

CMSC 330 30

Data Race Example

static int cnt = 0;t1.run() { int y = cnt; cnt = y + 1;}t2.run() { int y = cnt; cnt = y + 1;}

cnt = 0

Start: both threads ready torun. Each will increment theglobal count.

Shared state

Page 31: CMSC 330: Organization of Programming Languages Threads

CMSC 330 31

Data Race Example

static int cnt = 0;t1.run() { int y = cnt; cnt = y + 1;}t2.run() { int y = cnt; cnt = y + 1;}

cnt = 0

T1 executes, grabbingthe global counter value intoits own y.

Shared state

y = 0

Page 32: CMSC 330: Organization of Programming Languages Threads

CMSC 330 32

Data Race Example

static int cnt = 0;t1.run() { int y = cnt; cnt = y + 1;}t2.run() { int y = cnt; cnt = y + 1;}

cnt = 0

T1 is preempted. T2executes, grabbing the globalcounter value into its own y.

Shared state

y = 0

y = 0

Page 33: CMSC 330: Organization of Programming Languages Threads

CMSC 330 33

Data Race Example

static int cnt = 0;t1.run() { int y = cnt; cnt = y + 1;}t2.run() { int y = cnt; cnt = y + 1;}

cnt = 1

T2 executes, storing theincremented cnt value.

Shared state

y = 0

y = 0

Page 34: CMSC 330: Organization of Programming Languages Threads

CMSC 330 34

Data Race Example

static int cnt = 0;t1.run() { int y = cnt; cnt = y + 1;}t2.run() { int y = cnt; cnt = y + 1;}

cnt = 1

T2 completes. T1executes again, storing theincremented original countervalue (1) rather than what theincremented updated valuewould have been (2)!

Shared state

y = 0

y = 0

Page 35: CMSC 330: Organization of Programming Languages Threads

CMSC 330 35

What Happened?

• Different schedules led to different outcomes– This is a data race or race condition

• A thread was preempted in the middle of an operation– Reading and writing cnt was supposed to be atomic-

to happen with no interference from other threads– But the schedule (interleaving of threads) which was

chosen allowed atomicity to be violated– These bugs can be extremely hard to reproduce, and

so hard to debug• Depends on what scheduler chose to do, which is hard to

predict

Page 36: CMSC 330: Organization of Programming Languages Threads

CMSC 330 36

Question

• If instead ofint y = cnt;

cnt = y+1;

• We had written– cnt++;

• Would the result be any different?• Answer: NO!

– Don’t depend on your intuition about atomicity

Page 37: CMSC 330: Organization of Programming Languages Threads

CMSC 330 37

Question

• If you run a program with a race condition, will you always get an unexpected result?– No! It depends on the scheduler, i.e., which JVM

you’re running, and on the other threads/processes/etc, that are running on the same CPU

• Race conditions are hard to find

Page 38: CMSC 330: Organization of Programming Languages Threads

CMSC 330 38

What’s Wrong with the Following?

• Threads may be interrupted after the while but before the assignment x = 1– Both may think they “hold” the lock!

• This is busy waiting– Consumes lots of processor cycles

Thread 1 while (x != 0); x = 1; cnt++; x = 0;

Thread 2 while (x != 0); x = 1; cnt++; x = 0;

static int cnt = 0;static int x = 0;

Page 39: CMSC 330: Organization of Programming Languages Threads

CMSC 330 39

Aside: Functional Programming

• No side effects• No memory access• No data races!• Easier to parallelize functional programs

Page 40: CMSC 330: Organization of Programming Languages Threads

CMSC 330 40

Synchronization

• Refers to mechanisms allowing a programmer to control the execution order of some operations across different threads in a concurrent program.

• Different languages have adopted different mechanisms to allow the programmer to synchronize threads.

• Java has several mechanisms; we'll look at locks first.

Page 41: CMSC 330: Organization of Programming Languages Threads

CMSC 330 41

Locks (Java 1.5)

• Only one thread can hold a lock at once– Other threads that try to acquire it block (or become

suspended) until the lock becomes available

• Reentrant lock can be reacquired by same thread– As many times as desired– No other thread may acquire a lock until has been

released same number of times it has been acquired

interface Lock { void lock(); void unlock(); ... /* Some more stuff, also */}class ReentrantLock implements Lock { ... }

Page 42: CMSC 330: Organization of Programming Languages Threads

CMSC 330 42

Avoiding Interference: Synchronization

public class Example extends Thread { private static int cnt = 0; static Lock foo = new ReentrantLock(); public void run() { foo.lock(); int y = cnt; cnt = y + 1; foo.unlock(); } } …}

Lock, for protecting the shared state

Acquires the lock;Only succeeds if notheld by anotherthread

Releases the lock

Page 43: CMSC 330: Organization of Programming Languages Threads

CMSC 330 43

Applying Synchronization

int cnt = 0;t1.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}t2.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}

cnt = 0Shared state

T1 acquires the lock

Page 44: CMSC 330: Organization of Programming Languages Threads

CMSC 330 44

Applying Synchronization

int cnt = 0;t1.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}t2.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}

cnt = 0Shared state

T1 reads cnt into y

y = 0

Page 45: CMSC 330: Organization of Programming Languages Threads

CMSC 330 45

Applying Synchronization

int cnt = 0;t1.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}t2.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}

cnt = 0Shared state

T1 is preempted.T2 attempts toacquire the lock but failsbecause it’s held byT1, so it blocks

y = 0

Page 46: CMSC 330: Organization of Programming Languages Threads

CMSC 330 46

Applying Synchronization

int cnt = 0;t1.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}t2.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}

cnt = 1Shared state

T1 runs, assigningto cnt

y = 0

Page 47: CMSC 330: Organization of Programming Languages Threads

CMSC 330 47

Applying Synchronization

int cnt = 0;t1.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}t2.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}

cnt = 1Shared state

T1 releases the lockand terminates

y = 0

Page 48: CMSC 330: Organization of Programming Languages Threads

CMSC 330 48

Applying Synchronization

int cnt = 0;t1.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}t2.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}

cnt = 1Shared state

T2 now can acquirethe lock.

y = 0

Page 49: CMSC 330: Organization of Programming Languages Threads

CMSC 330 49

Applying Synchronization

int cnt = 0;t1.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}t2.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}

cnt = 1Shared state

T2 reads cnt into y.

y = 0

y = 1

Page 50: CMSC 330: Organization of Programming Languages Threads

CMSC 330 50

Applying Synchronization

int cnt = 0;t1.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}t2.run() { lock.lock(); int y = cnt; cnt = y + 1; lock.unlock();}

cnt = 2Shared state

T2 assigns cnt, then releases the lock

y = 0

y = 1

Page 51: CMSC 330: Organization of Programming Languages Threads

CMSC 330 51

Different Locks Don’t Interact

• This program has a race condition– Threads only block if they try to acquire a lock held

by another thread

static int cnt = 0;static Lock l = new ReentrantLock();static Lock m = new ReentrantLock();

void inc() { l.lock(); cnt++; l.unlock();}

void inc() { m.lock(); cnt++; m.unlock();}

Page 52: CMSC 330: Organization of Programming Languages Threads

CMSC 330 52

Reentrant Lock Example

• Reentrancy is useful because each method can acquire/release locks as necessary– No need to worry about whether callers have locks– Discourages complicated coding practices

static int cnt = 0;static Lock l = new ReentrantLock();

void inc() { l.lock(); cnt++; l.unlock();}

void returnAndInc() { int temp;

l.lock(); temp = cnt; inc(); l.unlock();}