2010-06-09 00:52:24 +08:00
|
|
|
//===-- Mutex.cpp -----------------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "lldb/Host/Mutex.h"
|
2012-01-28 02:29:47 +08:00
|
|
|
#include "lldb/Host/Host.h"
|
|
|
|
|
2013-08-23 20:44:05 +08:00
|
|
|
#ifndef _WIN32
|
|
|
|
#include <pthread.h>
|
|
|
|
#endif
|
2012-01-28 02:29:47 +08:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
#if 0
|
2010-06-14 12:30:13 +08:00
|
|
|
// This logging is way too verbose to enable even for a log channel.
|
|
|
|
// This logging can be enabled by changing the "#if 0", but should be
|
|
|
|
// reverted prior to checking in.
|
|
|
|
#include <cstdio>
|
2010-06-09 00:52:24 +08:00
|
|
|
#define DEBUG_LOG(fmt, ...) printf(fmt, ## __VA_ARGS__)
|
|
|
|
#else
|
|
|
|
#define DEBUG_LOG(fmt, ...)
|
|
|
|
#endif
|
|
|
|
|
2012-01-28 02:29:47 +08:00
|
|
|
// Enable extra mutex error checking
|
|
|
|
#ifdef LLDB_CONFIGURATION_DEBUG
|
|
|
|
#define ENABLE_MUTEX_ERROR_CHECKING 1
|
2012-12-20 07:42:07 +08:00
|
|
|
#include <inttypes.h>
|
2012-01-28 02:29:47 +08:00
|
|
|
#endif
|
|
|
|
|
2012-04-13 08:21:53 +08:00
|
|
|
#if ENABLE_MUTEX_ERROR_CHECKING
|
|
|
|
#include <set>
|
|
|
|
|
|
|
|
enum MutexAction
|
|
|
|
{
|
|
|
|
eMutexActionInitialized,
|
|
|
|
eMutexActionDestroyed,
|
|
|
|
eMutexActionAssertInitialized
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool
|
|
|
|
error_check_mutex (pthread_mutex_t *m, MutexAction action)
|
|
|
|
{
|
|
|
|
typedef std::set<pthread_mutex_t *> mutex_set;
|
|
|
|
static pthread_mutex_t g_mutex_set_mutex = PTHREAD_MUTEX_INITIALIZER;
|
2012-04-14 09:05:29 +08:00
|
|
|
static mutex_set g_initialized_mutex_set;
|
|
|
|
static mutex_set g_destroyed_mutex_set;
|
2012-04-13 08:21:53 +08:00
|
|
|
|
|
|
|
bool success = true;
|
|
|
|
int err;
|
|
|
|
// Manually call lock so we don't to any of this error checking
|
|
|
|
err = ::pthread_mutex_lock (&g_mutex_set_mutex);
|
|
|
|
assert(err == 0);
|
|
|
|
switch (action)
|
|
|
|
{
|
|
|
|
case eMutexActionInitialized:
|
|
|
|
// Make sure this isn't already in our initialized mutex set...
|
|
|
|
assert (g_initialized_mutex_set.find(m) == g_initialized_mutex_set.end());
|
|
|
|
// Remove this from the destroyed set in case it was ever in there
|
|
|
|
g_destroyed_mutex_set.erase(m);
|
|
|
|
// Add the mutex to the initialized set
|
|
|
|
g_initialized_mutex_set.insert(m);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eMutexActionDestroyed:
|
|
|
|
// Make sure this isn't already in our destroyed mutex set...
|
|
|
|
assert (g_destroyed_mutex_set.find(m) == g_destroyed_mutex_set.end());
|
|
|
|
// Remove this from the initialized so we can put it into the destroyed set
|
|
|
|
g_initialized_mutex_set.erase(m);
|
|
|
|
// Add the mutex to the destroyed set
|
|
|
|
g_destroyed_mutex_set.insert(m);
|
|
|
|
break;
|
|
|
|
case eMutexActionAssertInitialized:
|
|
|
|
// This function will return true if "m" is in the initialized mutex set
|
|
|
|
success = g_initialized_mutex_set.find(m) != g_initialized_mutex_set.end();
|
2012-04-14 07:28:33 +08:00
|
|
|
assert (success);
|
2012-04-13 08:21:53 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Manually call unlock so we don't to any of this error checking
|
|
|
|
err = ::pthread_mutex_unlock (&g_mutex_set_mutex);
|
|
|
|
assert(err == 0);
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
using namespace lldb_private;
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Default constructor.
|
|
|
|
//
|
|
|
|
// This will create a scoped mutex locking object that doesn't have
|
|
|
|
// a mutex to lock. One will need to be provided using the Reset()
|
|
|
|
// method.
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
Mutex::Locker::Locker () :
|
|
|
|
m_mutex_ptr(NULL)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Constructor with a Mutex object.
|
|
|
|
//
|
|
|
|
// This will create a scoped mutex locking object that extracts the
|
|
|
|
// mutex owned by "m" and locks it.
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
Mutex::Locker::Locker (Mutex& m) :
|
2012-04-11 08:24:49 +08:00
|
|
|
m_mutex_ptr(NULL)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2012-05-05 07:02:50 +08:00
|
|
|
Lock (m);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Constructor with a Mutex object pointer.
|
|
|
|
//
|
|
|
|
// This will create a scoped mutex locking object that extracts the
|
|
|
|
// mutex owned by "m" and locks it.
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
Mutex::Locker::Locker (Mutex* m) :
|
2012-04-11 08:24:49 +08:00
|
|
|
m_mutex_ptr(NULL)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2012-04-11 08:24:49 +08:00
|
|
|
if (m)
|
2012-05-05 07:02:50 +08:00
|
|
|
Lock (m);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
2011-01-09 04:28:42 +08:00
|
|
|
// Destructor
|
2010-06-09 00:52:24 +08:00
|
|
|
//
|
|
|
|
// Unlocks any owned mutex object (if it is valid).
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
Mutex::Locker::~Locker ()
|
|
|
|
{
|
2012-04-11 08:24:49 +08:00
|
|
|
Unlock();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Unlock the current mutex in this object (if this owns a valid
|
|
|
|
// mutex) and lock the new "mutex" object if it is non-NULL.
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void
|
2012-05-05 07:02:50 +08:00
|
|
|
Mutex::Locker::Lock (Mutex &mutex)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-09-04 03:15:43 +08:00
|
|
|
// We already have this mutex locked or both are NULL...
|
2012-06-09 06:50:40 +08:00
|
|
|
if (m_mutex_ptr == &mutex)
|
2010-09-04 03:15:43 +08:00
|
|
|
return;
|
|
|
|
|
2012-04-11 08:24:49 +08:00
|
|
|
Unlock ();
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2012-06-09 06:50:40 +08:00
|
|
|
m_mutex_ptr = &mutex;
|
|
|
|
m_mutex_ptr->Lock();
|
2012-04-11 08:24:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Mutex::Locker::Unlock ()
|
|
|
|
{
|
|
|
|
if (m_mutex_ptr)
|
|
|
|
{
|
2012-06-09 06:50:40 +08:00
|
|
|
m_mutex_ptr->Unlock ();
|
2012-04-11 08:24:49 +08:00
|
|
|
m_mutex_ptr = NULL;
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2012-06-09 06:50:40 +08:00
|
|
|
Mutex::Locker::TryLock (Mutex &mutex, const char *failure_message)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-09-04 03:15:43 +08:00
|
|
|
// We already have this mutex locked!
|
2012-06-09 06:50:40 +08:00
|
|
|
if (m_mutex_ptr == &mutex)
|
|
|
|
return true;
|
2010-09-04 03:15:43 +08:00
|
|
|
|
2012-04-11 08:24:49 +08:00
|
|
|
Unlock ();
|
2010-09-04 03:15:43 +08:00
|
|
|
|
2012-06-09 06:50:40 +08:00
|
|
|
if (mutex.TryLock(failure_message) == 0)
|
|
|
|
m_mutex_ptr = &mutex;
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
return m_mutex_ptr != NULL;
|
|
|
|
}
|
|
|
|
|
2013-08-23 20:44:05 +08:00
|
|
|
#ifndef _WIN32
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Default constructor.
|
|
|
|
//
|
|
|
|
// Creates a pthread mutex with no attributes.
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
Mutex::Mutex () :
|
|
|
|
m_mutex()
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
err = ::pthread_mutex_init (&m_mutex, NULL);
|
2012-04-13 08:21:53 +08:00
|
|
|
#if ENABLE_MUTEX_ERROR_CHECKING
|
|
|
|
if (err == 0)
|
|
|
|
error_check_mutex (&m_mutex, eMutexActionInitialized);
|
|
|
|
#endif
|
2010-06-09 00:52:24 +08:00
|
|
|
assert(err == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Default constructor.
|
|
|
|
//
|
|
|
|
// Creates a pthread mutex with "type" as the mutex type.
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
Mutex::Mutex (Mutex::Type type) :
|
|
|
|
m_mutex()
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
::pthread_mutexattr_t attr;
|
|
|
|
err = ::pthread_mutexattr_init (&attr);
|
|
|
|
assert(err == 0);
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case eMutexTypeNormal:
|
2012-01-28 02:29:47 +08:00
|
|
|
#if ENABLE_MUTEX_ERROR_CHECKING
|
|
|
|
err = ::pthread_mutexattr_settype (&attr, PTHREAD_MUTEX_ERRORCHECK);
|
|
|
|
#else
|
2010-06-09 00:52:24 +08:00
|
|
|
err = ::pthread_mutexattr_settype (&attr, PTHREAD_MUTEX_NORMAL);
|
2012-01-28 02:29:47 +08:00
|
|
|
#endif
|
2010-06-09 00:52:24 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case eMutexTypeRecursive:
|
|
|
|
err = ::pthread_mutexattr_settype (&attr, PTHREAD_MUTEX_RECURSIVE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
assert(err == 0);
|
|
|
|
err = ::pthread_mutex_init (&m_mutex, &attr);
|
2012-04-13 08:21:53 +08:00
|
|
|
#if ENABLE_MUTEX_ERROR_CHECKING
|
|
|
|
if (err == 0)
|
|
|
|
error_check_mutex (&m_mutex, eMutexActionInitialized);
|
|
|
|
#endif
|
2010-06-09 00:52:24 +08:00
|
|
|
assert(err == 0);
|
|
|
|
err = ::pthread_mutexattr_destroy (&attr);
|
|
|
|
assert(err == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Destructor.
|
|
|
|
//
|
|
|
|
// Destroys the mutex owned by this object.
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
Mutex::~Mutex()
|
|
|
|
{
|
2014-07-19 02:32:45 +08:00
|
|
|
#if ENABLE_MUTEX_ERROR_CHECKING
|
2013-04-20 05:31:16 +08:00
|
|
|
int err = ::pthread_mutex_destroy (&m_mutex);
|
2013-06-04 02:00:07 +08:00
|
|
|
assert(err == 0);
|
2012-04-13 08:21:53 +08:00
|
|
|
if (err == 0)
|
|
|
|
error_check_mutex (&m_mutex, eMutexActionDestroyed);
|
|
|
|
else
|
2012-01-28 02:29:47 +08:00
|
|
|
{
|
|
|
|
Host::SetCrashDescriptionWithFormat ("%s error: pthread_mutex_destroy() => err = %i (%s)", __PRETTY_FUNCTION__, err, strerror(err));
|
|
|
|
assert(err == 0);
|
|
|
|
}
|
|
|
|
memset (&m_mutex, '\xba', sizeof(m_mutex));
|
2014-07-19 02:32:45 +08:00
|
|
|
#else
|
|
|
|
::pthread_mutex_destroy (&m_mutex);
|
2012-01-28 02:29:47 +08:00
|
|
|
#endif
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2012-06-09 06:50:40 +08:00
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Locks the mutex owned by this object, if the mutex is already
|
|
|
|
// locked, the calling thread will block until the mutex becomes
|
|
|
|
// available.
|
|
|
|
//
|
|
|
|
// RETURNS
|
|
|
|
// The error code from the pthread_mutex_lock() function call.
|
|
|
|
//----------------------------------------------------------------------
|
2010-06-09 00:52:24 +08:00
|
|
|
int
|
2012-06-09 06:50:40 +08:00
|
|
|
Mutex::Lock()
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2012-11-30 05:49:15 +08:00
|
|
|
DEBUG_LOG ("[%4.4" PRIx64 "/%4.4" PRIx64 "] pthread_mutex_lock (%p)...\n", Host::GetCurrentProcessID(), Host::GetCurrentThreadID(), &m_mutex);
|
2012-04-13 08:21:53 +08:00
|
|
|
|
|
|
|
#if ENABLE_MUTEX_ERROR_CHECKING
|
2012-06-09 06:50:40 +08:00
|
|
|
error_check_mutex (&m_mutex, eMutexActionAssertInitialized);
|
2012-04-13 08:21:53 +08:00
|
|
|
#endif
|
|
|
|
|
2012-06-09 06:50:40 +08:00
|
|
|
int err = ::pthread_mutex_lock (&m_mutex);
|
2012-04-13 08:21:53 +08:00
|
|
|
|
|
|
|
|
2012-01-28 02:29:47 +08:00
|
|
|
#if ENABLE_MUTEX_ERROR_CHECKING
|
|
|
|
if (err)
|
|
|
|
{
|
2012-06-09 06:50:40 +08:00
|
|
|
Host::SetCrashDescriptionWithFormat ("%s error: pthread_mutex_lock(%p) => err = %i (%s)", __PRETTY_FUNCTION__, &m_mutex, err, strerror(err));
|
2012-01-28 02:29:47 +08:00
|
|
|
assert(err == 0);
|
|
|
|
}
|
|
|
|
#endif
|
2012-11-30 05:49:15 +08:00
|
|
|
DEBUG_LOG ("[%4.4" PRIx64 "/%4.4" PRIx64 "] pthread_mutex_lock (%p) => %i\n", Host::GetCurrentProcessID(), Host::GetCurrentThreadID(), &m_mutex, err);
|
2010-06-09 00:52:24 +08:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2012-06-09 06:50:40 +08:00
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Attempts to lock the mutex owned by this object without blocking.
|
|
|
|
// If the mutex is already locked, TryLock() will not block waiting
|
|
|
|
// for the mutex, but will return an error condition.
|
|
|
|
//
|
|
|
|
// RETURNS
|
|
|
|
// The error code from the pthread_mutex_trylock() function call.
|
|
|
|
//----------------------------------------------------------------------
|
2010-06-09 00:52:24 +08:00
|
|
|
int
|
2012-06-09 06:50:40 +08:00
|
|
|
Mutex::TryLock(const char *failure_message)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2012-04-13 08:21:53 +08:00
|
|
|
#if ENABLE_MUTEX_ERROR_CHECKING
|
2012-06-09 06:50:40 +08:00
|
|
|
error_check_mutex (&m_mutex, eMutexActionAssertInitialized);
|
2012-04-13 08:21:53 +08:00
|
|
|
#endif
|
|
|
|
|
2012-06-09 06:50:40 +08:00
|
|
|
int err = ::pthread_mutex_trylock (&m_mutex);
|
2012-11-30 05:49:15 +08:00
|
|
|
DEBUG_LOG ("[%4.4" PRIx64 "/%4.4" PRIx64 "] pthread_mutex_trylock (%p) => %i\n", Host::GetCurrentProcessID(), Host::GetCurrentThreadID(), &m_mutex, err);
|
2010-06-09 00:52:24 +08:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2012-06-09 06:50:40 +08:00
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// If the current thread holds the lock on the owned mutex, then
|
|
|
|
// Unlock() will unlock the mutex. Calling Unlock() on this object
|
|
|
|
// that the calling thread does not hold will result in undefined
|
|
|
|
// behavior.
|
|
|
|
//
|
|
|
|
// RETURNS
|
|
|
|
// The error code from the pthread_mutex_unlock() function call.
|
|
|
|
//----------------------------------------------------------------------
|
2010-06-09 00:52:24 +08:00
|
|
|
int
|
2012-06-09 06:50:40 +08:00
|
|
|
Mutex::Unlock()
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2012-04-13 08:21:53 +08:00
|
|
|
#if ENABLE_MUTEX_ERROR_CHECKING
|
2012-06-09 06:50:40 +08:00
|
|
|
error_check_mutex (&m_mutex, eMutexActionAssertInitialized);
|
2012-04-13 08:21:53 +08:00
|
|
|
#endif
|
|
|
|
|
2012-06-09 06:50:40 +08:00
|
|
|
int err = ::pthread_mutex_unlock (&m_mutex);
|
2012-04-13 08:21:53 +08:00
|
|
|
|
2012-01-28 02:29:47 +08:00
|
|
|
#if ENABLE_MUTEX_ERROR_CHECKING
|
|
|
|
if (err)
|
|
|
|
{
|
2012-06-09 06:50:40 +08:00
|
|
|
Host::SetCrashDescriptionWithFormat ("%s error: pthread_mutex_unlock(%p) => err = %i (%s)", __PRETTY_FUNCTION__, &m_mutex, err, strerror(err));
|
2012-01-28 02:29:47 +08:00
|
|
|
assert(err == 0);
|
|
|
|
}
|
|
|
|
#endif
|
2012-11-30 05:49:15 +08:00
|
|
|
DEBUG_LOG ("[%4.4" PRIx64 "/%4.4" PRIx64 "] pthread_mutex_unlock (%p) => %i\n", Host::GetCurrentProcessID(), Host::GetCurrentThreadID(), &m_mutex, err);
|
2010-06-09 00:52:24 +08:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2013-08-23 20:44:05 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Mutex get accessor.
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
lldb::mutex_t *
|
|
|
|
Mutex::GetMutex()
|
|
|
|
{
|
|
|
|
return &m_mutex;
|
|
|
|
}
|
|
|
|
|
2012-06-09 06:50:40 +08:00
|
|
|
#ifdef LLDB_CONFIGURATION_DEBUG
|
2010-06-09 00:52:24 +08:00
|
|
|
int
|
2012-06-09 06:50:40 +08:00
|
|
|
TrackingMutex::Unlock ()
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2012-06-09 06:50:40 +08:00
|
|
|
if (!m_failure_message.empty())
|
|
|
|
Host::SetCrashDescriptionWithFormat ("Unlocking lock (on thread %p) that thread: %p failed to get: %s",
|
|
|
|
pthread_self(),
|
|
|
|
m_thread_that_tried,
|
|
|
|
m_failure_message.c_str());
|
|
|
|
assert (m_failure_message.empty());
|
|
|
|
return Mutex::Unlock();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2012-12-20 07:42:07 +08:00
|
|
|
|
|
|
|
int
|
|
|
|
LoggingMutex::Lock ()
|
|
|
|
{
|
|
|
|
printf("locking mutex %p by [%4.4" PRIx64 "/%4.4" PRIx64 "]...", this, Host::GetCurrentProcessID(), Host::GetCurrentThreadID());
|
|
|
|
int x = Mutex::Lock();
|
|
|
|
m_locked = true;
|
|
|
|
printf("%d\n",x);
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
LoggingMutex::Unlock ()
|
|
|
|
{
|
|
|
|
printf("unlocking mutex %p by [%4.4" PRIx64 "/%4.4" PRIx64 "]...", this, Host::GetCurrentProcessID(), Host::GetCurrentThreadID());
|
|
|
|
int x = Mutex::Unlock();
|
|
|
|
m_locked = false;
|
|
|
|
printf("%d\n",x);
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
LoggingMutex::TryLock (const char *failure_message)
|
|
|
|
{
|
|
|
|
printf("trylocking mutex %p by [%4.4" PRIx64 "/%4.4" PRIx64 "]...", this, Host::GetCurrentProcessID(), Host::GetCurrentThreadID());
|
|
|
|
int x = Mutex::TryLock(failure_message);
|
|
|
|
if (x == 0)
|
|
|
|
m_locked = true;
|
|
|
|
printf("%d\n",x);
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
2012-06-09 06:50:40 +08:00
|
|
|
#endif
|
2012-12-20 07:42:07 +08:00
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
|