
This fixes occasional hangs we've been seeing in the past few days. I'm using rtc::Event instead of the EventWrapper, so I'll wait with landing this cl until I've made that change in a separate cl. BUG=2822,4282 R=pbos@webrtc.org Review URL: https://webrtc-codereview.appspot.com/38009004 Cr-Commit-Position: refs/heads/master@{#8293} git-svn-id: http://webrtc.googlecode.com/svn/trunk@8293 4adac7df-926f-26a2-2b94-8c16560cd09d
182 lines
5.1 KiB
C++
182 lines
5.1 KiB
C++
/*
|
|
* Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
#include "webrtc/system_wrappers/source/thread_posix.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
#ifdef WEBRTC_LINUX
|
|
#include <linux/unistd.h>
|
|
#include <sched.h>
|
|
#include <sys/prctl.h>
|
|
#include <sys/syscall.h>
|
|
#include <sys/types.h>
|
|
#endif
|
|
|
|
#include "webrtc/base/checks.h"
|
|
#include "webrtc/system_wrappers/interface/critical_section_wrapper.h"
|
|
#include "webrtc/system_wrappers/interface/event_wrapper.h"
|
|
#include "webrtc/system_wrappers/interface/sleep.h"
|
|
#include "webrtc/system_wrappers/interface/trace.h"
|
|
|
|
namespace webrtc {
|
|
namespace {
|
|
struct ThreadAttributes {
|
|
ThreadAttributes() { pthread_attr_init(&attr); }
|
|
~ThreadAttributes() { pthread_attr_destroy(&attr); }
|
|
pthread_attr_t* operator&() { return &attr; }
|
|
pthread_attr_t attr;
|
|
};
|
|
} // namespace
|
|
|
|
int ConvertToSystemPriority(ThreadPriority priority, int min_prio,
|
|
int max_prio) {
|
|
DCHECK(max_prio - min_prio > 2);
|
|
const int top_prio = max_prio - 1;
|
|
const int low_prio = min_prio + 1;
|
|
|
|
switch (priority) {
|
|
case kLowPriority:
|
|
return low_prio;
|
|
case kNormalPriority:
|
|
// The -1 ensures that the kHighPriority is always greater or equal to
|
|
// kNormalPriority.
|
|
return (low_prio + top_prio - 1) / 2;
|
|
case kHighPriority:
|
|
return std::max(top_prio - 2, low_prio);
|
|
case kHighestPriority:
|
|
return std::max(top_prio - 1, low_prio);
|
|
case kRealtimePriority:
|
|
return top_prio;
|
|
}
|
|
DCHECK(false);
|
|
return low_prio;
|
|
}
|
|
|
|
struct ThreadPosix::InitParams {
|
|
InitParams(ThreadPosix* thread)
|
|
: me(thread), started(EventWrapper::Create()) {
|
|
}
|
|
ThreadPosix* me;
|
|
rtc::scoped_ptr<EventWrapper> started;
|
|
};
|
|
|
|
// static
|
|
void* ThreadPosix::StartThread(void* param) {
|
|
auto params = static_cast<InitParams*>(param);
|
|
params->me->Run(params);
|
|
return 0;
|
|
}
|
|
|
|
ThreadPosix::ThreadPosix(ThreadRunFunction func, ThreadObj obj,
|
|
ThreadPriority prio, const char* thread_name)
|
|
: run_function_(func),
|
|
obj_(obj),
|
|
prio_(prio),
|
|
stop_event_(true, false),
|
|
name_(thread_name ? thread_name : "webrtc"),
|
|
thread_id_(0),
|
|
thread_(0) {
|
|
DCHECK(name_.length() < kThreadMaxNameLength);
|
|
}
|
|
|
|
uint32_t ThreadWrapper::GetThreadId() {
|
|
return rtc::CurrentThreadId();
|
|
}
|
|
|
|
ThreadPosix::~ThreadPosix() {
|
|
DCHECK(thread_checker_.CalledOnValidThread());
|
|
}
|
|
|
|
bool ThreadPosix::Start(unsigned int& thread_id) {
|
|
DCHECK(thread_checker_.CalledOnValidThread());
|
|
DCHECK(!thread_id_) << "Thread already started?";
|
|
|
|
ThreadAttributes attr;
|
|
// Set the stack stack size to 1M.
|
|
pthread_attr_setstacksize(&attr, 1024 * 1024);
|
|
|
|
InitParams params(this);
|
|
int result = pthread_create(&thread_, &attr, &StartThread, ¶ms);
|
|
if (result != 0)
|
|
return false;
|
|
|
|
// TODO(tommi): Change the timeout back to WEBRTC_EVENT_INFINITE when we've
|
|
// figured out where we're seeing tests hang.
|
|
auto state = params.started->Wait(30 * 1000);
|
|
CHECK_EQ(kEventSignaled, state);
|
|
DCHECK_NE(thread_id_, 0);
|
|
|
|
thread_id = thread_id_;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ThreadPosix::Stop() {
|
|
DCHECK(thread_checker_.CalledOnValidThread());
|
|
if (!thread_id_)
|
|
return true;
|
|
|
|
stop_event_.Set();
|
|
CHECK_EQ(0, pthread_join(thread_, nullptr));
|
|
thread_id_ = 0;
|
|
stop_event_.Reset();
|
|
|
|
return true;
|
|
}
|
|
|
|
void ThreadPosix::Run(ThreadPosix::InitParams* params) {
|
|
thread_id_ = rtc::CurrentThreadId();
|
|
params->started->Set();
|
|
|
|
if (!name_.empty()) {
|
|
// Setting the thread name may fail (harmlessly) if running inside a
|
|
// sandbox. Ignore failures if they happen.
|
|
#if defined(WEBRTC_LINUX) || defined(WEBRTC_ANDROID)
|
|
prctl(PR_SET_NAME, reinterpret_cast<unsigned long>(name_.c_str()));
|
|
#elif defined(WEBRTC_MAC) || defined(WEBRTC_IOS)
|
|
pthread_setname_np(name_.substr(0, 63).c_str());
|
|
#endif
|
|
}
|
|
|
|
#ifdef WEBRTC_THREAD_RR
|
|
const int policy = SCHED_RR;
|
|
#else
|
|
const int policy = SCHED_FIFO;
|
|
#endif
|
|
const int min_prio = sched_get_priority_min(policy);
|
|
const int max_prio = sched_get_priority_max(policy);
|
|
if ((min_prio == -1) || (max_prio == -1)) {
|
|
WEBRTC_TRACE(kTraceError, kTraceUtility, -1,
|
|
"unable to retreive min or max priority for threads");
|
|
}
|
|
|
|
if (max_prio - min_prio > 2) {
|
|
sched_param param;
|
|
param.sched_priority = ConvertToSystemPriority(prio_, min_prio, max_prio);
|
|
if (pthread_setschedparam(pthread_self(), policy, ¶m) != 0) {
|
|
WEBRTC_TRACE(
|
|
kTraceError, kTraceUtility, -1, "unable to set thread priority");
|
|
}
|
|
}
|
|
|
|
// It's a requirement that for successful thread creation that the run
|
|
// function be called at least once (see RunFunctionIsCalled unit test),
|
|
// so to fullfill that requirement, we use a |do| loop and not |while|.
|
|
do {
|
|
if (!run_function_(obj_))
|
|
break;
|
|
} while (!stop_event_.Wait(0));
|
|
}
|
|
|
|
} // namespace webrtc
|