123456789_123456789_123456789_123456789_123456789_

Class: Mutex

Relationships & Source Files
Inherits: Object
Defined in: thread_sync.c,
thread_sync.c

Overview

Mutex implements a simple semaphore that can be used to coordinate access to shared data from multiple concurrent threads.

Example:

semaphore = Mutex.new

a = Thread.new {
  semaphore.synchronize {
    # access shared resource
  }
}

b = Thread.new {
  semaphore.synchronize {
    # access shared resource
  }
}

Class Method Summary

Instance Attribute Summary

Instance Method Summary

Constructor Details

.newMutex

Creates a new Mutex

[ GitHub ]

  
# File 'thread_sync.c', line 158

static VALUE
mutex_initialize(VALUE self)
{
    return self;
}

Instance Attribute Details

#locked?Boolean (readonly)

Returns true if this lock is currently held by some thread.

[ GitHub ]

  
# File 'thread_sync.c', line 176

VALUE
rb_mutex_locked_p(VALUE self)
{
    rb_mutex_t *mutex = mutex_ptr(self);

    return mutex->th ? Qtrue : Qfalse;
}

#owned?Boolean (readonly)

Returns true if this lock is currently held by current thread.

[ GitHub ]

  
# File 'thread_sync.c', line 328

VALUE
rb_mutex_owned_p(VALUE self)
{
    VALUE owned = Qfalse;
    rb_thread_t *th = GET_THREAD();
    rb_mutex_t *mutex = mutex_ptr(self);

    if (mutex->th == th)
	owned = Qtrue;

    return owned;
}

Instance Method Details

#lockself

Attempts to grab the lock and waits if it isn’t available. Raises ::ThreadError if mutex was locked by the current thread.

[ GitHub ]

  
# File 'thread_sync.c', line 316

VALUE
rb_mutex_lock(VALUE self)
{
    return do_mutex_lock(self, 1);
}

#sleep(timeout = nil) ⇒ Numeric

Releases the lock and sleeps timeout seconds if it is given and non-nil or forever. Raises ::ThreadError if mutex wasn’t locked by the current thread.

When the thread is next woken up, it will attempt to reacquire the lock.

Note that this method can wakeup without explicit Thread#wakeup call. For example, receiving signal and so on.

[ GitHub ]

  
# File 'thread_sync.c', line 492

static VALUE
mutex_sleep(int argc, VALUE *argv, VALUE self)
{
    VALUE timeout;

    timeout = rb_check_arity(argc, 0, 1) ? argv[0] : Qnil;
    return rb_mutex_sleep(self, timeout);
}

#synchronizeresult of the block

Obtains a lock, runs the block, and releases the lock when the block completes. See the example under Mutex.

[ GitHub ]

  
# File 'thread_sync.c', line 523

static VALUE
rb_mutex_synchronize_m(VALUE self, VALUE args)
{
    if (!rb_block_given_p()) {
	rb_raise(rb_eThreadError, "must be called with a block");
    }

    return rb_mutex_synchronize(self, rb_yield, Qundef);
}

#try_lockBoolean

Attempts to obtain the lock and returns immediately. Returns true if the lock was granted.

[ GitHub ]

  
# File 'thread_sync.c', line 202

VALUE
rb_mutex_trylock(VALUE self)
{
    rb_mutex_t *mutex = mutex_ptr(self);
    VALUE locked = Qfalse;

    if (mutex->th == 0) {
	rb_thread_t *th = GET_THREAD();
	mutex->th = th;
	locked = Qtrue;

	mutex_locked(th, self);
    }

    return locked;
}

#unlockself

Releases the lock. Raises ::ThreadError if mutex wasn’t locked by the current thread.

[ GitHub ]

  
# File 'thread_sync.c', line 390

VALUE
rb_mutex_unlock(VALUE self)
{
    const char *err;
    rb_mutex_t *mutex = mutex_ptr(self);

    err = rb_mutex_unlock_th(mutex, GET_THREAD());
    if (err) rb_raise(rb_eThreadError, "%s", err);

    return self;
}