Skip to content

Latest commit

 

History

History
116 lines (86 loc) · 3.47 KB

README.md

File metadata and controls

116 lines (86 loc) · 3.47 KB

mult-version-smart-ptr

A safe and efficent way to share a mult-version immutable resource that needs to dispose whenever useless among multiple long-living goroutines.

Motivation

How to share an immutable resource(once made, never change) among goroutines?

The most intuitive way(but wrong)

writer:

  1. lock
  2. create a resource and save a reference to it in a global variable, called G
  3. set refcnt to 1
  4. unlock

reader:

  1. atomically read G
  2. atomically incr refcnt by 1
  3. make use of it
  4. atomically decr refcnt by 1, if it becomes zero, clean up the resource

Classical ABA[https://en.wikipedia.org/wiki/ABA_problem] problem: reader.4 races with reader.2 and may run into a situation where a reader decides to cleanup the resource and another one decides to use it. Side note: this is also a common misuse of cpp's std::shared_ptr<T>

Fix

reader:

  1. lock
  2. atomically reads G
  3. atomically incr refcnt by 1
  4. unlock
  5. make use of it
  6. lock
  7. atomically decr refcnt by 1
  8. unlock
  9. if refcnt becomes zero, clean up the resource

What's wrong? Not scalable because every read requires locking.

Improve by thread-localing and introducing a version number

writer:

  1. lock
  2. atomically drop the refcnt of G by 1, if it becomes 0, then delete the underlying resource
  3. create a resource and save a reference to it in a global variable, called G
  4. set refcnt to 1
  5. incr the global version number, called V by 1
  6. unlock

reader:

  1. compare the thread-local version number(called Vtl) with the global one
  2. if different
  • lock
  • atomically drop refcnt of the thread-local reference(called, Gtl) by 1, if it becomes 0, then delete the underlying resource
  • read G to Gtl
  • read V to thread-local store, Vtl
  • incr refcnt of Gtl by 1
  • unlock
  • make use of Gtl
  1. otherwise, make use of Gtl

What's wrong? Even it seems like every version is refcnt guarded and safe from leaking, a reader might hold a reference to an obsolute version forever and make no use of it.

Improve by disvalidating references in writers

Similar to Cache-Aside pattern: write disvalidate the thread-local handle, read checks for update and synchronize the thread-local with global if any

See the code for details

Features

  • Only one lock overhead for every version of the resource: a reader locks for the first query, and for subsequent quries, only a few atomic operations are involved if no newer version.
  • Like c++'s std::shared_ptr, if the refcnt drops to zero, the resource is automatically deleted.

Caveat

  • Not suitable for the common use of Go, that is, spawn goroutine without any limit.
  • It is profitable only when you have some long-living goroutines need to access an immutable resource concurrently.

Hello World

func getAndUse(gid) {
  handle := GetResouce(gid, false)
  use(handle)
  DoneUsingResource(gid, handle)
}

UpdateResouce(NewImmutableResourceThatHasDeleteMethod())
go func() { // mimic the behavior of writes
  for i := 0; i < 100; i ++ {
    UpdateResouce(NewImmutableResourceThatHasDeleteMethod())
    time.Sleep(1 * time.Second)
  }
}()

// shared among 10 goroutines
for i := 0; i < 10; i ++ {
  go func() {
    gid := AllocateGLocalImmRscHandle()
    for i := 0; i < 10; i ++ {
      getAndUse(gid)
    }
  }()
}

// Delete() of any shared resource will be automatically called when all 10+1 goroutines is done using it  

Acknowledgement

Inspired by RocksDB's MVCC implementation. https://rocksdb.org/blog/2014/06/27/avoid-expensive-locks-in-get.html