Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

sync: implement futex-based Mutex #4650

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
69 changes: 69 additions & 0 deletions src/internal/task/mutex.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
package task

// Futex-based mutex.
// This is largely based on the paper "Futexes are Tricky" by Ulrich Drepper.
// It describes a few ways to implement mutexes using a futex, and how some
// seemingly-obvious implementations don't exactly work as intended.
// Unfortunately, Go atomic operations work slightly differently so we can't
// copy the algorithm verbatim.
//
// The implementation works like this. The futex can have 3 different values,
// depending on the state:
//
// - 0: the futex is currently unlocked.
// - 1: the futex is locked, but is uncontended. There is one special case: if
// a contended futex is unlocked, it is set to 0. It is possible for another
// thread to lock the futex before the next waiter is woken. But because a
// waiter will be woken (if there is one), it will always change to 2
// regardless. So this is not a problem.
// - 2: the futex is locked, and is contended. At least one thread is trying
// to obtain the lock (and is in the contended loop, see below).
//
// For the paper, see:
// https://dept-info.labri.fr/~denis/Enseignement/2008-IR/Articles/01-futex.pdf)

type Mutex struct {
futex Futex
}

func (m *Mutex) Lock() {
// Fast path: try to take an uncontended lock.
if m.futex.CompareAndSwap(0, 1) {
// We obtained the mutex.
return
}

// The futex is contended, so we enter the contended loop.
// If we manage to change the futex from 0 to 2, we managed to take the
// look. Else, we have to wait until a call to Unlock unlocks this mutex.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit

Suggested change
// look. Else, we have to wait until a call to Unlock unlocks this mutex.
// lock. Else, we have to wait until a call to Unlock unlocks this mutex.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks, fixed in #4559!

// (Unlock will wake one waiter when it finds the futex is set to 2 when
// unlocking).
for m.futex.Swap(2) != 0 {
// Wait until we get resumed in Unlock.
m.futex.Wait(2)
}
}

func (m *Mutex) Unlock() {
if old := m.futex.Swap(0); old == 0 {
// Mutex wasn't locked before.
panic("sync: unlock of unlocked Mutex")
} else if old == 2 {
// Mutex was a contended lock, so we need to wake the next waiter.
m.futex.Wake()
}
}

// TryLock tries to lock m and reports whether it succeeded.
//
// Note that while correct uses of TryLock do exist, they are rare,
// and use of TryLock is often a sign of a deeper problem
// in a particular use of mutexes.
func (m *Mutex) TryLock() bool {
// Fast path: try to take an uncontended lock.
if m.futex.CompareAndSwap(0, 1) {
// We obtained the mutex.
return true
}
return false
}
3 changes: 3 additions & 0 deletions src/sync/cond.go
Original file line number Diff line number Diff line change
Expand Up @@ -89,3 +89,6 @@ func (c *Cond) Wait() {
// signal.
task.Pause()
}

//go:linkname scheduleTask runtime.scheduleTask
func scheduleTask(*task.Task)
45 changes: 1 addition & 44 deletions src/sync/mutex.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,50 +5,7 @@ import (
_ "unsafe"
)

type Mutex struct {
locked bool
blocked task.Stack
}

//go:linkname scheduleTask runtime.scheduleTask
func scheduleTask(*task.Task)

func (m *Mutex) Lock() {
if m.locked {
// Push self onto stack of blocked tasks, and wait to be resumed.
m.blocked.Push(task.Current())
task.Pause()
return
}

m.locked = true
}

func (m *Mutex) Unlock() {
if !m.locked {
panic("sync: unlock of unlocked Mutex")
}

// Wake up a blocked task, if applicable.
if t := m.blocked.Pop(); t != nil {
scheduleTask(t)
} else {
m.locked = false
}
}

// TryLock tries to lock m and reports whether it succeeded.
//
// Note that while correct uses of TryLock do exist, they are rare,
// and use of TryLock is often a sign of a deeper problem
// in a particular use of mutexes.
func (m *Mutex) TryLock() bool {
if m.locked {
return false
}
m.Lock()
return true
}
type Mutex = task.Mutex

type RWMutex struct {
// waitingWriters are all of the tasks waiting for write locks.
Expand Down
Loading