From: James Sedgwick Date: Wed, 6 Sep 2017 06:09:06 +0000 (-0700) Subject: move wangle/concurrent to folly/executors X-Git-Tag: v2017.09.11.00~6 X-Git-Url: http://demsky.eecs.uci.edu/git/?a=commitdiff_plain;h=af4a335238f0ccefdf8a8e6a2689d3a1d44b584a;p=folly.git move wangle/concurrent to folly/executors Summary: For the initial cutover, just pull the copies in folly into the wangle namespace The main problem with that approach is that forward declarations of wangle components no longer work, so I fixed those manually ALSO, IMPORTANT: This is a great, once in a lifetime opportunity to rename/restructure these components. I have a few ideas that I'll noodle and share eventually, e.g. changing LifoSemMPMCQueue so it's just descriptive and not an enumeration of implementation details. Please chip in with yr ideas! Reviewed By: yfeldblum Differential Revision: D5694213 fbshipit-source-id: 4fc0ea9359d1216191676fc9729fb53a3f06339f --- diff --git a/CMakeLists.txt b/CMakeLists.txt index c067fb7c..6c72796f 100755 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -296,6 +296,15 @@ if (BUILD_TESTS) folly_define_tests( DIRECTORY concurrency/test/ TEST cache_locality_test SOURCES CacheLocalityTest.cpp + DIRECTORY executors/test/ + TEST async_test SOURCES AsyncTest.cpp + TEST codel_test SOURCES CodelTest.cpp + TEST fiber_io_executor_test SOURCES FiberIOExecutorTest.cpp + TEST global_executor_test SOURCES GlobalExecutorTest.cpp + TEST serial_executor_test SOURCES SerialExecutorTest.cpp + TEST thread_pool_executor_test SOURCES ThreadPoolExecutorTest.cpp + TEST threaded_executor_test SOURCES ThreadedExecutorTest.cpp + TEST unbounded_blocking_queue_test SOURCES UnboundedBlockingQueueTest.cpp DIRECTORY experimental/test/ TEST autotimer_test SOURCES AutoTimerTest.cpp TEST bits_test_2 SOURCES BitsTest.cpp diff --git a/folly/Makefile.am b/folly/Makefile.am index 442c1e1c..cb6c1074 100644 --- a/folly/Makefile.am +++ b/folly/Makefile.am @@ -85,6 +85,25 @@ nobase_follyinclude_HEADERS = \ detail/ThreadLocalDetail.h \ detail/TurnSequencer.h \ detail/UncaughtExceptionCounter.h \ + executors/Async.h \ + executors/BlockingQueue.h \ + executors/CPUThreadPoolExecutor.h \ + executors/Codel.h \ + executors/FiberIOExecutor.h \ + executors/FutureExecutor.h \ + executors/GlobalExecutor.h \ + executors/IOExecutor.h \ + executors/IOObjectCache.h \ + executors/IOThreadPoolExecutor.h \ + executors/LifoSemMPMCQueue.h \ + executors/NamedThreadFactory.h \ + executors/PriorityLifoSemMPMCQueue.h \ + executors/PriorityThreadFactory.h \ + executors/SerialExecutor.h \ + executors/ThreadFactory.h \ + executors/ThreadPoolExecutor.h \ + executors/ThreadedExecutor.h \ + executors/UnboundedBlockingQueue.h \ Demangle.h \ DiscriminatedPtr.h \ DynamicConverter.h \ @@ -487,6 +506,13 @@ libfolly_la_SOURCES = \ futures/QueuedImmediateExecutor.cpp \ futures/ThreadWheelTimekeeper.cpp \ futures/test/TestExecutor.cpp \ + executors/CPUThreadPoolExecutor.cpp \ + executors/Codel.cpp \ + executors/GlobalExecutor.cpp \ + executors/IOThreadPoolExecutor.cpp \ + executors/SerialExecutor.cpp \ + executors/ThreadPoolExecutor.cpp \ + executors/ThreadedExecutor.cpp \ experimental/hazptr/hazptr.cpp \ experimental/hazptr/memory_resource.cpp \ GlobalThreadPoolList.cpp \ diff --git a/folly/Singleton.h b/folly/Singleton.h index 975bab57..028cf7f1 100644 --- a/folly/Singleton.h +++ b/folly/Singleton.h @@ -405,7 +405,7 @@ class SingletonVault { * * Sample usage: * - * wangle::IOThreadPoolExecutor executor(max_concurrency_level); + * folly::IOThreadPoolExecutor executor(max_concurrency_level); * folly::Baton<> done; * doEagerInitVia(executor, &done); * done.wait(); // or 'timed_wait', or spin with 'try_wait' diff --git a/folly/docs/Executors.md b/folly/docs/Executors.md new file mode 100644 index 00000000..8491bcf3 --- /dev/null +++ b/folly/docs/Executors.md @@ -0,0 +1,63 @@ +

Thread pools & Executors

Run your concurrent code in a performant way

All about thread pools #

+ +

How do I use the thread pools? #

+ +

Wangle provides two concrete thread pools (IOThreadPoolExecutor, CPUThreadPoolExecutor) as well as building them in as part of a complete async framework. Generally you might want to grab the global executor, and use it with a future, like this:

+ +
auto f = someFutureFunction().via(getCPUExecutor()).then(...)
+ +

Or maybe you need to construct a thrift/memcache client, and need an event base:

+ +
auto f = getClient(getIOExecutor()->getEventBase())->callSomeFunction(args...)
+         .via(getCPUExecutor())
+         .then([](Result r){ .... do something with result});
+ +

vs. C++11's std::launch #

+ +

The current C++11 std::launch only has two modes: async or deferred. In a production system, neither is what you want: async will launch a new thread for every launch without limit, while deferred will defer the work until it is needed lazily, but then do the work in the current thread synchronously when it is needed.

+ +

Wangle's thread pools always launch work as soon as possible, have limits to the maximum number of tasks / threads allowed, so we will never use more threads than absolutely needed. See implementation details below about each type of executor.

+ +

Why do we need yet another set of thread pools? #

+ +

Unfortunately none of the existing thread pools had every feature needed - things based on pipes are too slow. Several older ones didn't support std::function.

+ +

Why do we need several different types of thread pools? #

+ +

If you want epoll support, you need an fd - event_fd is the latest notification hotness. Unfortunately, an active fd triggers all the epoll loops it is in, leading to thundering herd - so if you want a fair queue (one queue total vs. one queue per worker thread), you need to use some kind of semaphore. Unfortunately semaphores can't be put in epoll loops, so they are incompatible with IO. Fortunately, you usually want to separate the IO and CPU bound work anyway to give stronger tail latency guarantees on IO.

+ +

IOThreadPoolExecutor #

+ + + +

CPUThreadPoolExecutor #

+ + + +

ThreadPoolExecutor #

+ +

Base class that contains the thread startup/shutdown/stats logic, since this is pretty disjoint from how tasks are actually run

+ +

Observers #

+ +

An observer interface is provided to listen for thread start/stop events. This is useful to create objects that should be one-per-thread, but also have them work correctly if threads are added/removed from the thread pool.

+ +

Stats #

+ +

PoolStats are provided to get task count, running time, waiting time, etc.

+
+ diff --git a/folly/executors/Async.h b/folly/executors/Async.h new file mode 100644 index 00000000..a2a4eae6 --- /dev/null +++ b/folly/executors/Async.h @@ -0,0 +1,28 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once +#include +#include + +namespace folly { + +template +auto async(F&& fn) { + return folly::via(getCPUExecutor().get(), std::forward(fn)); +} + +} // namespace folly diff --git a/folly/executors/BlockingQueue.h b/folly/executors/BlockingQueue.h new file mode 100644 index 00000000..49280954 --- /dev/null +++ b/folly/executors/BlockingQueue.h @@ -0,0 +1,50 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include +#include + +#include + +namespace folly { + +// Some queue implementations (for example, LifoSemMPMCQueue or +// PriorityLifoSemMPMCQueue) support both blocking (BLOCK) and +// non-blocking (THROW) behaviors. +enum class QueueBehaviorIfFull { THROW, BLOCK }; + +class QueueFullException : public std::runtime_error { + using std::runtime_error::runtime_error; // Inherit constructors. +}; + +template +class BlockingQueue { + public: + virtual ~BlockingQueue() = default; + virtual void add(T item) = 0; + virtual void addWithPriority(T item, int8_t /* priority */) { + add(std::move(item)); + } + virtual uint8_t getNumPriorities() { + return 1; + } + virtual T take() = 0; + virtual size_t size() = 0; +}; + +} // namespace folly diff --git a/folly/executors/CPUThreadPoolExecutor.cpp b/folly/executors/CPUThreadPoolExecutor.cpp new file mode 100644 index 00000000..37fd4441 --- /dev/null +++ b/folly/executors/CPUThreadPoolExecutor.cpp @@ -0,0 +1,157 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include +#include + +namespace folly { + +const size_t CPUThreadPoolExecutor::kDefaultMaxQueueSize = 1 << 14; + +CPUThreadPoolExecutor::CPUThreadPoolExecutor( + size_t numThreads, + std::unique_ptr> taskQueue, + std::shared_ptr threadFactory) + : ThreadPoolExecutor(numThreads, std::move(threadFactory)), + taskQueue_(std::move(taskQueue)) { + setNumThreads(numThreads); +} + +CPUThreadPoolExecutor::CPUThreadPoolExecutor( + size_t numThreads, + std::shared_ptr threadFactory) + : CPUThreadPoolExecutor( + numThreads, + std::make_unique>( + CPUThreadPoolExecutor::kDefaultMaxQueueSize), + std::move(threadFactory)) {} + +CPUThreadPoolExecutor::CPUThreadPoolExecutor(size_t numThreads) + : CPUThreadPoolExecutor( + numThreads, + std::make_shared("CPUThreadPool")) {} + +CPUThreadPoolExecutor::CPUThreadPoolExecutor( + size_t numThreads, + int8_t numPriorities, + std::shared_ptr threadFactory) + : CPUThreadPoolExecutor( + numThreads, + std::make_unique>( + numPriorities, + CPUThreadPoolExecutor::kDefaultMaxQueueSize), + std::move(threadFactory)) {} + +CPUThreadPoolExecutor::CPUThreadPoolExecutor( + size_t numThreads, + int8_t numPriorities, + size_t maxQueueSize, + std::shared_ptr threadFactory) + : CPUThreadPoolExecutor( + numThreads, + std::make_unique>( + numPriorities, + maxQueueSize), + std::move(threadFactory)) {} + +CPUThreadPoolExecutor::~CPUThreadPoolExecutor() { + stop(); + CHECK(threadsToStop_ == 0); +} + +void CPUThreadPoolExecutor::add(Func func) { + add(std::move(func), std::chrono::milliseconds(0)); +} + +void CPUThreadPoolExecutor::add( + Func func, + std::chrono::milliseconds expiration, + Func expireCallback) { + // TODO handle enqueue failure, here and in other add() callsites + taskQueue_->add( + CPUTask(std::move(func), expiration, std::move(expireCallback))); +} + +void CPUThreadPoolExecutor::addWithPriority(Func func, int8_t priority) { + add(std::move(func), priority, std::chrono::milliseconds(0)); +} + +void CPUThreadPoolExecutor::add( + Func func, + int8_t priority, + std::chrono::milliseconds expiration, + Func expireCallback) { + CHECK(getNumPriorities() > 0); + taskQueue_->addWithPriority( + CPUTask(std::move(func), expiration, std::move(expireCallback)), + priority); +} + +uint8_t CPUThreadPoolExecutor::getNumPriorities() const { + return taskQueue_->getNumPriorities(); +} + +BlockingQueue* +CPUThreadPoolExecutor::getTaskQueue() { + return taskQueue_.get(); +} + +void CPUThreadPoolExecutor::threadRun(std::shared_ptr thread) { + this->threadPoolHook_.registerThread(); + + thread->startupBaton.post(); + while (1) { + auto task = taskQueue_->take(); + if (UNLIKELY(task.poison)) { + CHECK(threadsToStop_-- > 0); + for (auto& o : observers_) { + o->threadStopped(thread.get()); + } + folly::RWSpinLock::WriteHolder w{&threadListLock_}; + threadList_.remove(thread); + stoppedThreads_.add(thread); + return; + } else { + runTask(thread, std::move(task)); + } + + if (UNLIKELY(threadsToStop_ > 0 && !isJoin_)) { + if (--threadsToStop_ >= 0) { + folly::RWSpinLock::WriteHolder w{&threadListLock_}; + threadList_.remove(thread); + stoppedThreads_.add(thread); + return; + } else { + threadsToStop_++; + } + } + } +} + +void CPUThreadPoolExecutor::stopThreads(size_t n) { + threadsToStop_ += n; + for (size_t i = 0; i < n; i++) { + taskQueue_->addWithPriority(CPUTask(), Executor::LO_PRI); + } +} + +// threadListLock_ is readlocked +uint64_t CPUThreadPoolExecutor::getPendingTaskCountImpl( + const folly::RWSpinLock::ReadHolder&) { + return taskQueue_->size(); +} + +} // namespace folly diff --git a/folly/executors/CPUThreadPoolExecutor.h b/folly/executors/CPUThreadPoolExecutor.h new file mode 100644 index 00000000..64f1b73c --- /dev/null +++ b/folly/executors/CPUThreadPoolExecutor.h @@ -0,0 +1,130 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include + +namespace folly { + +/** + * A Thread pool for CPU bound tasks. + * + * @note A single queue backed by folly/LifoSem and folly/MPMC queue. + * Because of this contention can be quite high, + * since all the worker threads and all the producer threads hit + * the same queue. MPMC queue excels in this situation but dictates a max queue + * size. + * + * @note If a blocking queue (folly::QueueBehaviorIfFull::BLOCK) is used, and + * tasks executing on a given thread pool schedule more tasks, deadlock is + * possible if the queue becomes full. Deadlock is also possible if there is + * a circular dependency among multiple thread pools with blocking queues. + * To avoid this situation, use non-blocking queue(s), or schedule tasks only + * from threads not belonging to the given thread pool(s), or use + * folly::IOThreadPoolExecutor. + * + * @note LifoSem wakes up threads in Lifo order - i.e. there are only few + * threads as necessary running, and we always try to reuse the same few threads + * for better cache locality. + * Inactive threads have their stack madvised away. This works quite well in + * combination with Lifosem - it almost doesn't matter if more threads than are + * necessary are specified at startup. + * + * @note stop() will finish all outstanding tasks at exit. + * + * @note Supports priorities - priorities are implemented as multiple queues - + * each worker thread checks the highest priority queue first. Threads + * themselves don't have priorities set, so a series of long running low + * priority tasks could still hog all the threads. (at last check pthreads + * thread priorities didn't work very well). + */ +class CPUThreadPoolExecutor : public ThreadPoolExecutor { + public: + struct CPUTask; + + CPUThreadPoolExecutor( + size_t numThreads, + std::unique_ptr> taskQueue, + std::shared_ptr threadFactory = + std::make_shared("CPUThreadPool")); + + explicit CPUThreadPoolExecutor(size_t numThreads); + + CPUThreadPoolExecutor( + size_t numThreads, + std::shared_ptr threadFactory); + + CPUThreadPoolExecutor( + size_t numThreads, + int8_t numPriorities, + std::shared_ptr threadFactory = + std::make_shared("CPUThreadPool")); + + CPUThreadPoolExecutor( + size_t numThreads, + int8_t numPriorities, + size_t maxQueueSize, + std::shared_ptr threadFactory = + std::make_shared("CPUThreadPool")); + + ~CPUThreadPoolExecutor() override; + + void add(Func func) override; + void add( + Func func, + std::chrono::milliseconds expiration, + Func expireCallback = nullptr) override; + + void addWithPriority(Func func, int8_t priority) override; + void add( + Func func, + int8_t priority, + std::chrono::milliseconds expiration, + Func expireCallback = nullptr); + + uint8_t getNumPriorities() const override; + + struct CPUTask : public ThreadPoolExecutor::Task { + // Must be noexcept move constructible so it can be used in MPMCQueue + + explicit CPUTask( + Func&& f, + std::chrono::milliseconds expiration, + Func&& expireCallback) + : Task(std::move(f), expiration, std::move(expireCallback)), + poison(false) {} + CPUTask() + : Task(nullptr, std::chrono::milliseconds(0), nullptr), poison(true) {} + + bool poison; + }; + + static const size_t kDefaultMaxQueueSize; + + protected: + BlockingQueue* getTaskQueue(); + + private: + void threadRun(ThreadPtr thread) override; + void stopThreads(size_t n) override; + uint64_t getPendingTaskCountImpl(const RWSpinLock::ReadHolder&) override; + + std::unique_ptr> taskQueue_; + std::atomic threadsToStop_{0}; +}; + +} // namespace folly diff --git a/folly/executors/Codel.cpp b/folly/executors/Codel.cpp new file mode 100644 index 00000000..c84b9be3 --- /dev/null +++ b/folly/executors/Codel.cpp @@ -0,0 +1,102 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include + +#include +#include + +DEFINE_int32(codel_interval, 100, "Codel default interval time in ms"); +DEFINE_int32(codel_target_delay, 5, "Target codel queueing delay in ms"); + +using std::chrono::nanoseconds; +using std::chrono::milliseconds; + +namespace folly { + +Codel::Codel() + : codelMinDelay_(0), + codelIntervalTime_(std::chrono::steady_clock::now()), + codelResetDelay_(true), + overloaded_(false) {} + +bool Codel::overloaded(std::chrono::nanoseconds delay) { + bool ret = false; + auto now = std::chrono::steady_clock::now(); + + // Avoid another thread updating the value at the same time we are using it + // to calculate the overloaded state + auto minDelay = codelMinDelay_; + + if (now > codelIntervalTime_ && + // testing before exchanging is more cacheline-friendly + (!codelResetDelay_.load(std::memory_order_acquire) && + !codelResetDelay_.exchange(true))) { + codelIntervalTime_ = now + getInterval(); + + if (minDelay > getTargetDelay()) { + overloaded_ = true; + } else { + overloaded_ = false; + } + } + // Care must be taken that only a single thread resets codelMinDelay_, + // and that it happens after the interval reset above + if (codelResetDelay_.load(std::memory_order_acquire) && + codelResetDelay_.exchange(false)) { + codelMinDelay_ = delay; + // More than one request must come in during an interval before codel + // starts dropping requests + return false; + } else if (delay < codelMinDelay_) { + codelMinDelay_ = delay; + } + + // Here is where we apply different logic than codel proper. Instead of + // adapting the interval until the next drop, we slough off requests with + // queueing delay > 2*target_delay while in the overloaded regime. This + // empirically works better for our services than the codel approach of + // increasingly often dropping packets. + if (overloaded_ && delay > getSloughTimeout()) { + ret = true; + } + + return ret; +} + +int Codel::getLoad() { + // it might be better to use the average delay instead of minDelay, but we'd + // have to track it. aspiring bootcamper? + return std::min(100, 100 * getMinDelay() / getSloughTimeout()); +} + +nanoseconds Codel::getMinDelay() { + return codelMinDelay_; +} + +milliseconds Codel::getInterval() { + return milliseconds(FLAGS_codel_interval); +} + +milliseconds Codel::getTargetDelay() { + return milliseconds(FLAGS_codel_target_delay); +} + +milliseconds Codel::getSloughTimeout() { + return getTargetDelay() * 2; +} + +} // namespace folly diff --git a/folly/executors/Codel.h b/folly/executors/Codel.h new file mode 100644 index 00000000..98718bb8 --- /dev/null +++ b/folly/executors/Codel.h @@ -0,0 +1,93 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include +#include + +#include + +DECLARE_int32(codel_interval); +DECLARE_int32(codel_target_delay); + +namespace folly { + +/// CoDel (controlled delay) is an active queue management algorithm from +/// networking for battling bufferbloat. +/// +/// Services also have queues (of requests, not packets) and suffer from +/// queueing delay when overloaded. This class adapts the codel algorithm for +/// services. +/// +/// Codel is discussed in depth on the web [1,2], but a basic sketch of the +/// algorithm is this: if every request has experienced queueing delay greater +/// than the target (5ms) during the past interval (100ms), then we shed load. +/// +/// We have adapted the codel algorithm. TCP sheds load by changing windows in +/// reaction to dropped packets. Codel in a network setting drops packets at +/// increasingly shorter intervals (100 / sqrt(n)) to achieve a linear change +/// in throughput. In our experience a different scheme works better for +/// services: when overloaded slough off requests that we dequeue which have +/// exceeded an alternate timeout (2 * target_delay). +/// +/// So in summary, to use this class, calculate the time each request spent in +/// the queue and feed that delay to overloaded(), which will tell you whether +/// to expire this request. +/// +/// You can also ask for an instantaneous load estimate and the minimum delay +/// observed during this interval. +/// +/// +/// 1. http://queue.acm.org/detail.cfm?id=2209336 +/// 2. https://en.wikipedia.org/wiki/CoDel +class Codel { + public: + Codel(); + + /// Returns true if this request should be expired to reduce overload. + /// In detail, this returns true if min_delay > target_delay for the + /// interval, and this delay > 2 * target_delay. + /// + /// As you may guess, we observe the clock so this is time sensitive. Call + /// it promptly after calculating queueing delay. + bool overloaded(std::chrono::nanoseconds delay); + + /// Get the queue load, as seen by the codel algorithm + /// Gives a rough guess at how bad the queue delay is. + /// + /// min(100%, min_delay / (2 * target_delay)) + /// + /// Return: 0 = no delay, 100 = At the queueing limit + int getLoad(); + + std::chrono::nanoseconds getMinDelay(); + std::chrono::milliseconds getInterval(); + std::chrono::milliseconds getTargetDelay(); + std::chrono::milliseconds getSloughTimeout(); + + private: + std::chrono::nanoseconds codelMinDelay_; + std::chrono::time_point codelIntervalTime_; + + // flag to make overloaded() thread-safe, since we only want + // to reset the delay once per time period + std::atomic codelResetDelay_; + + bool overloaded_; +}; + +} // namespace folly diff --git a/folly/executors/FiberIOExecutor.h b/folly/executors/FiberIOExecutor.h new file mode 100644 index 00000000..18c36181 --- /dev/null +++ b/folly/executors/FiberIOExecutor.h @@ -0,0 +1,49 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include +#include + +namespace folly { + +/** + * @class FiberIOExecutor + * @brief An IOExecutor that executes funcs under mapped fiber context + * + * A FiberIOExecutor wraps an IOExecutor, but executes funcs on the FiberManager + * mapped to the underlying IOExector's event base. + */ +class FiberIOExecutor : public IOExecutor { + public: + explicit FiberIOExecutor(const std::shared_ptr& ioExecutor) + : ioExecutor_(ioExecutor) {} + + virtual void add(folly::Function f) override { + auto eventBase = ioExecutor_->getEventBase(); + folly::fibers::getFiberManager(*eventBase).add(std::move(f)); + } + + virtual folly::EventBase* getEventBase() override { + return ioExecutor_->getEventBase(); + } + + private: + std::shared_ptr ioExecutor_; +}; + +} // namespace folly diff --git a/folly/executors/FutureExecutor.h b/folly/executors/FutureExecutor.h new file mode 100644 index 00000000..705a82fd --- /dev/null +++ b/folly/executors/FutureExecutor.h @@ -0,0 +1,80 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once +#include + +namespace folly { + +template +class FutureExecutor : public ExecutorImpl { + public: + template + explicit FutureExecutor(Args&&... args) + : ExecutorImpl(std::forward(args)...) {} + + /* + * Given a function func that returns a Future, adds that function to the + * contained Executor and returns a Future which will be fulfilled with + * func's result once it has been executed. + * + * For example: auto f = futureExecutor.addFuture([](){ + * return doAsyncWorkAndReturnAFuture(); + * }); + */ + template + typename std::enable_if< + folly::isFuture::type>::value, + typename std::result_of::type>::type + addFuture(F func) { + typedef typename std::result_of::type::value_type T; + folly::Promise promise; + auto future = promise.getFuture(); + ExecutorImpl::add( + [ promise = std::move(promise), func = std::move(func) ]() mutable { + func().then([promise = std::move(promise)]( + folly::Try && t) mutable { promise.setTry(std::move(t)); }); + }); + return future; + } + + /* + * Similar to addFuture above, but takes a func that returns some non-Future + * type T. + * + * For example: auto f = futureExecutor.addFuture([]() { + * return 42; + * }); + */ + template + typename std::enable_if< + !folly::isFuture::type>::value, + folly::Future::type>::type>>::type + addFuture(F func) { + using T = + typename folly::Unit::Lift::type>::type; + folly::Promise promise; + auto future = promise.getFuture(); + ExecutorImpl::add( + [ promise = std::move(promise), func = std::move(func) ]() mutable { + promise.setWith(std::move(func)); + }); + return future; + } +}; + +} // namespace folly diff --git a/folly/executors/GlobalExecutor.cpp b/folly/executors/GlobalExecutor.cpp new file mode 100644 index 00000000..d2de51d7 --- /dev/null +++ b/folly/executors/GlobalExecutor.cpp @@ -0,0 +1,112 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include +#include +#include +#include + +using namespace folly; + +namespace { + +// lock protecting global CPU executor +struct CPUExecutorLock {}; +Singleton globalCPUExecutorLock; +// global CPU executor +Singleton> globalCPUExecutor; +// default global CPU executor is an InlineExecutor +Singleton> globalInlineExecutor([] { + return new std::shared_ptr( + std::make_shared()); +}); + +// lock protecting global IO executor +struct IOExecutorLock {}; +Singleton globalIOExecutorLock; +// global IO executor +Singleton> globalIOExecutor; +// default global IO executor is an IOThreadPoolExecutor +Singleton> globalIOThreadPool([] { + return new std::shared_ptr( + std::make_shared( + sysconf(_SC_NPROCESSORS_ONLN), + std::make_shared("GlobalIOThreadPool"))); +}); +} + +namespace folly { + +template +std::shared_ptr getExecutor( + Singleton>& sExecutor, + Singleton>& sDefaultExecutor, + Singleton& sExecutorLock) { + std::shared_ptr executor; + auto singleton = sExecutor.try_get(); + auto lock = sExecutorLock.try_get(); + + { + RWSpinLock::ReadHolder guard(lock.get()); + if ((executor = sExecutor.try_get()->lock())) { + return executor; + } + } + + RWSpinLock::WriteHolder guard(lock.get()); + executor = singleton->lock(); + if (!executor) { + std::weak_ptr defaultExecutor = *sDefaultExecutor.try_get().get(); + executor = defaultExecutor.lock(); + sExecutor.try_get().get()->swap(defaultExecutor); + } + return executor; +} + +template +void setExecutor( + std::weak_ptr executor, + Singleton>& sExecutor, + Singleton& sExecutorLock) { + auto lock = sExecutorLock.try_get(); + RWSpinLock::WriteHolder guard(*lock); + std::weak_ptr executor_weak = std::move(executor); + sExecutor.try_get().get()->swap(executor_weak); +} + +std::shared_ptr getCPUExecutor() { + return getExecutor( + globalCPUExecutor, globalInlineExecutor, globalCPUExecutorLock); +} + +void setCPUExecutor(std::weak_ptr executor) { + setExecutor(std::move(executor), globalCPUExecutor, globalCPUExecutorLock); +} + +std::shared_ptr getIOExecutor() { + return getExecutor( + globalIOExecutor, globalIOThreadPool, globalIOExecutorLock); +} + +EventBase* getEventBase() { + return getIOExecutor()->getEventBase(); +} + +void setIOExecutor(std::weak_ptr executor) { + setExecutor(std::move(executor), globalIOExecutor, globalIOExecutorLock); +} + +} // namespace folly diff --git a/folly/executors/GlobalExecutor.h b/folly/executors/GlobalExecutor.h new file mode 100644 index 00000000..79f5dce6 --- /dev/null +++ b/folly/executors/GlobalExecutor.h @@ -0,0 +1,49 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include + +#include +#include + +namespace folly { + +// Retrieve the global Executor. If there is none, a default InlineExecutor +// will be constructed and returned. This is named CPUExecutor to distinguish +// it from IOExecutor below and to hint that it's intended for CPU-bound tasks. +std::shared_ptr getCPUExecutor(); + +// Set an Executor to be the global Executor which will be returned by +// subsequent calls to getCPUExecutor(). +void setCPUExecutor(std::weak_ptr executor); + +// Retrieve the global IOExecutor. If there is none, a default +// IOThreadPoolExecutor will be constructed and returned. +// +// IOExecutors differ from Executors in that they drive and provide access to +// one or more EventBases. +std::shared_ptr getIOExecutor(); + +// Retrieve an event base from the global IOExecutor +folly::EventBase* getEventBase(); + +// Set an IOExecutor to be the global IOExecutor which will be returned by +// subsequent calls to getIOExecutor(). +void setIOExecutor(std::weak_ptr executor); + +} // namespace folly diff --git a/folly/executors/IOExecutor.h b/folly/executors/IOExecutor.h new file mode 100644 index 00000000..223c0496 --- /dev/null +++ b/folly/executors/IOExecutor.h @@ -0,0 +1,46 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include + +namespace folly { +class EventBase; +} + +namespace folly { + +// An IOExecutor is an executor that operates on at least one EventBase. One of +// these EventBases should be accessible via getEventBase(). The event base +// returned by a call to getEventBase() is implementation dependent. +// +// Note that IOExecutors don't necessarily loop on the base themselves - for +// instance, EventBase itself is an IOExecutor but doesn't drive itself. +// +// Implementations of IOExecutor are eligible to become the global IO executor, +// returned on every call to getIOExecutor(), via setIOExecutor(). +// These functions are declared in GlobalExecutor.h +// +// If getIOExecutor is called and none has been set, a default global +// IOThreadPoolExecutor will be created and returned. +class IOExecutor : public virtual folly::Executor { + public: + ~IOExecutor() override = default; + virtual folly::EventBase* getEventBase() = 0; +}; + +} // namespace folly diff --git a/folly/executors/IOObjectCache.h b/folly/executors/IOObjectCache.h new file mode 100644 index 00000000..a271d6e7 --- /dev/null +++ b/folly/executors/IOObjectCache.h @@ -0,0 +1,70 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include +#include +#include + +namespace folly { + +/* + * IOObjectCache manages objects of type T that are dependent on an EventBase + * provided by the global IOExecutor. + * + * Provide a factory that creates T objects given an EventBase, and get() will + * lazily create T objects based on an EventBase from the global IOExecutor. + * These are stored thread locally - for a given pair of event base and calling + * thread there will only be one T object created. + * + * The primary use case is for managing objects that need to do async IO on an + * event base (e.g. thrift clients) that can be used outside the IO thread + * without much hassle. For instance, you could use this to manage Thrift + * clients that are only ever called from within other threads without the + * calling thread needing to know anything about the IO threads that the clients + * will do their work on. + */ +template +class IOObjectCache { + public: + typedef std::function(folly::EventBase*)> TFactory; + + IOObjectCache() = default; + explicit IOObjectCache(TFactory factory) : factory_(std::move(factory)) {} + + std::shared_ptr get() { + CHECK(factory_); + auto eb = getIOExecutor()->getEventBase(); + CHECK(eb); + auto it = cache_->find(eb); + if (it == cache_->end()) { + auto p = cache_->insert(std::make_pair(eb, factory_(eb))); + it = p.first; + } + return it->second; + }; + + void setFactory(TFactory factory) { + factory_ = std::move(factory); + } + + private: + folly::ThreadLocal>> cache_; + TFactory factory_; +}; + +} // namespace folly diff --git a/folly/executors/IOThreadPoolExecutor.cpp b/folly/executors/IOThreadPoolExecutor.cpp new file mode 100644 index 00000000..1cf9bc5e --- /dev/null +++ b/folly/executors/IOThreadPoolExecutor.cpp @@ -0,0 +1,219 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include + +#include + +#include + +namespace folly { + +using folly::detail::MemoryIdler; + +/* Class that will free jemalloc caches and madvise the stack away + * if the event loop is unused for some period of time + */ +class MemoryIdlerTimeout : public AsyncTimeout, public EventBase::LoopCallback { + public: + explicit MemoryIdlerTimeout(EventBase* b) : AsyncTimeout(b), base_(b) {} + + void timeoutExpired() noexcept override { + idled = true; + } + + void runLoopCallback() noexcept override { + if (idled) { + MemoryIdler::flushLocalMallocCaches(); + MemoryIdler::unmapUnusedStack(MemoryIdler::kDefaultStackToRetain); + + idled = false; + } else { + std::chrono::steady_clock::duration idleTimeout = + MemoryIdler::defaultIdleTimeout.load(std::memory_order_acquire); + + idleTimeout = MemoryIdler::getVariationTimeout(idleTimeout); + + scheduleTimeout( + std::chrono::duration_cast(idleTimeout) + .count()); + } + + // reschedule this callback for the next event loop. + base_->runBeforeLoop(this); + } + + private: + EventBase* base_; + bool idled{false}; +}; + +IOThreadPoolExecutor::IOThreadPoolExecutor( + size_t numThreads, + std::shared_ptr threadFactory, + EventBaseManager* ebm, + bool waitForAll) + : ThreadPoolExecutor(numThreads, std::move(threadFactory), waitForAll), + nextThread_(0), + eventBaseManager_(ebm) { + setNumThreads(numThreads); +} + +IOThreadPoolExecutor::~IOThreadPoolExecutor() { + stop(); +} + +void IOThreadPoolExecutor::add(Func func) { + add(std::move(func), std::chrono::milliseconds(0)); +} + +void IOThreadPoolExecutor::add( + Func func, + std::chrono::milliseconds expiration, + Func expireCallback) { + RWSpinLock::ReadHolder r{&threadListLock_}; + if (threadList_.get().empty()) { + throw std::runtime_error("No threads available"); + } + auto ioThread = pickThread(); + + auto task = Task(std::move(func), expiration, std::move(expireCallback)); + auto wrappedFunc = [ ioThread, task = std::move(task) ]() mutable { + runTask(ioThread, std::move(task)); + ioThread->pendingTasks--; + }; + + ioThread->pendingTasks++; + if (!ioThread->eventBase->runInEventBaseThread(std::move(wrappedFunc))) { + ioThread->pendingTasks--; + throw std::runtime_error("Unable to run func in event base thread"); + } +} + +std::shared_ptr +IOThreadPoolExecutor::pickThread() { + auto& me = *thisThread_; + auto& ths = threadList_.get(); + // When new task is added to IOThreadPoolExecutor, a thread is chosen for it + // to be executed on, thisThread_ is by default chosen, however, if the new + // task is added by the clean up operations on thread destruction, thisThread_ + // is not an available thread anymore, thus, always check whether or not + // thisThread_ is an available thread before choosing it. + if (me && std::find(ths.cbegin(), ths.cend(), me) != ths.cend()) { + return me; + } + auto n = ths.size(); + if (n == 0) { + return me; + } + auto thread = ths[nextThread_++ % n]; + return std::static_pointer_cast(thread); +} + +EventBase* IOThreadPoolExecutor::getEventBase() { + return pickThread()->eventBase; +} + +EventBase* IOThreadPoolExecutor::getEventBase( + ThreadPoolExecutor::ThreadHandle* h) { + auto thread = dynamic_cast(h); + + if (thread) { + return thread->eventBase; + } + + return nullptr; +} + +EventBaseManager* IOThreadPoolExecutor::getEventBaseManager() { + return eventBaseManager_; +} + +std::shared_ptr IOThreadPoolExecutor::makeThread() { + return std::make_shared(this); +} + +void IOThreadPoolExecutor::threadRun(ThreadPtr thread) { + this->threadPoolHook_.registerThread(); + + const auto ioThread = std::static_pointer_cast(thread); + ioThread->eventBase = eventBaseManager_->getEventBase(); + thisThread_.reset(new std::shared_ptr(ioThread)); + + auto idler = std::make_unique(ioThread->eventBase); + ioThread->eventBase->runBeforeLoop(idler.get()); + + ioThread->eventBase->runInEventBaseThread( + [thread] { thread->startupBaton.post(); }); + while (ioThread->shouldRun) { + ioThread->eventBase->loopForever(); + } + if (isJoin_) { + while (ioThread->pendingTasks > 0) { + ioThread->eventBase->loopOnce(); + } + } + idler.reset(); + if (isWaitForAll_) { + // some tasks, like thrift asynchronous calls, create additional + // event base hookups, let's wait till all of them complete. + ioThread->eventBase->loop(); + } + + std::lock_guard guard(ioThread->eventBaseShutdownMutex_); + ioThread->eventBase = nullptr; + eventBaseManager_->clearEventBase(); +} + +// threadListLock_ is writelocked +void IOThreadPoolExecutor::stopThreads(size_t n) { + std::vector stoppedThreads; + stoppedThreads.reserve(n); + for (size_t i = 0; i < n; i++) { + const auto ioThread = + std::static_pointer_cast(threadList_.get()[i]); + for (auto& o : observers_) { + o->threadStopped(ioThread.get()); + } + ioThread->shouldRun = false; + stoppedThreads.push_back(ioThread); + std::lock_guard guard(ioThread->eventBaseShutdownMutex_); + if (ioThread->eventBase) { + ioThread->eventBase->terminateLoopSoon(); + } + } + for (auto thread : stoppedThreads) { + stoppedThreads_.add(thread); + threadList_.remove(thread); + } +} + +// threadListLock_ is readlocked +uint64_t IOThreadPoolExecutor::getPendingTaskCountImpl( + const folly::RWSpinLock::ReadHolder&) { + uint64_t count = 0; + for (const auto& thread : threadList_.get()) { + auto ioThread = std::static_pointer_cast(thread); + size_t pendingTasks = ioThread->pendingTasks; + if (pendingTasks > 0 && !ioThread->idle) { + pendingTasks--; + } + count += pendingTasks; + } + return count; +} + +} // namespace folly diff --git a/folly/executors/IOThreadPoolExecutor.h b/folly/executors/IOThreadPoolExecutor.h new file mode 100644 index 00000000..cf52822c --- /dev/null +++ b/folly/executors/IOThreadPoolExecutor.h @@ -0,0 +1,94 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include +#include +#include + +namespace folly { + +/** + * A Thread Pool for IO bound tasks + * + * @note Uses event_fd for notification, and waking an epoll loop. + * There is one queue (NotificationQueue specifically) per thread/epoll. + * If the thread is already running and not waiting on epoll, + * we don't make any additional syscalls to wake up the loop, + * just put the new task in the queue. + * If any thread has been waiting for more than a few seconds, + * its stack is madvised away. Currently however tasks are scheduled round + * robin on the queues, so unless there is no work going on, + * this isn't very effective. + * Since there is one queue per thread, there is hardly any contention + * on the queues - so a simple spinlock around an std::deque is used for + * the tasks. There is no max queue size. + * By default, there is one thread per core - it usually doesn't make sense to + * have more IO threads than this, assuming they don't block. + * + * @note ::getEventBase() will return an EventBase you can schedule IO work on + * directly, chosen round-robin. + * + * @note N.B. For this thread pool, stop() behaves like join() because + * outstanding tasks belong to the event base and will be executed upon its + * destruction. + */ +class IOThreadPoolExecutor : public ThreadPoolExecutor, public IOExecutor { + public: + explicit IOThreadPoolExecutor( + size_t numThreads, + std::shared_ptr threadFactory = + std::make_shared("IOThreadPool"), + folly::EventBaseManager* ebm = folly::EventBaseManager::get(), + bool waitForAll = false); + + ~IOThreadPoolExecutor() override; + + void add(Func func) override; + void add( + Func func, + std::chrono::milliseconds expiration, + Func expireCallback = nullptr) override; + + folly::EventBase* getEventBase() override; + + static folly::EventBase* getEventBase(ThreadPoolExecutor::ThreadHandle*); + + folly::EventBaseManager* getEventBaseManager(); + + private: + struct FOLLY_ALIGN_TO_AVOID_FALSE_SHARING IOThread : public Thread { + IOThread(IOThreadPoolExecutor* pool) + : Thread(pool), shouldRun(true), pendingTasks(0) {} + std::atomic shouldRun; + std::atomic pendingTasks; + folly::EventBase* eventBase; + std::mutex eventBaseShutdownMutex_; + }; + + ThreadPtr makeThread() override; + std::shared_ptr pickThread(); + void threadRun(ThreadPtr thread) override; + void stopThreads(size_t n) override; + uint64_t getPendingTaskCountImpl(const RWSpinLock::ReadHolder&) override; + + size_t nextThread_; + folly::ThreadLocal> thisThread_; + folly::EventBaseManager* eventBaseManager_; +}; + +} // namespace folly diff --git a/folly/executors/LifoSemMPMCQueue.h b/folly/executors/LifoSemMPMCQueue.h new file mode 100644 index 00000000..3a16da27 --- /dev/null +++ b/folly/executors/LifoSemMPMCQueue.h @@ -0,0 +1,66 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include +#include +#include + +namespace folly { + +template +class LifoSemMPMCQueue : public BlockingQueue { + public: + // Note: The queue pre-allocates all memory for max_capacity + explicit LifoSemMPMCQueue(size_t max_capacity) : queue_(max_capacity) {} + + void add(T item) override { + switch (kBehavior) { // static + case QueueBehaviorIfFull::THROW: + if (!queue_.write(std::move(item))) { + throw QueueFullException("LifoSemMPMCQueue full, can't add item"); + } + break; + case QueueBehaviorIfFull::BLOCK: + queue_.blockingWrite(std::move(item)); + break; + } + sem_.post(); + } + + T take() override { + T item; + while (!queue_.readIfNotEmpty(item)) { + sem_.wait(); + } + return item; + } + + size_t capacity() { + return queue_.capacity(); + } + + size_t size() override { + return queue_.size(); + } + + private: + folly::LifoSem sem_; + folly::MPMCQueue queue_; +}; + +} // namespace folly diff --git a/folly/executors/NamedThreadFactory.h b/folly/executors/NamedThreadFactory.h new file mode 100644 index 00000000..bcd4a6bb --- /dev/null +++ b/folly/executors/NamedThreadFactory.h @@ -0,0 +1,55 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include +#include +#include + +#include +#include +#include +#include + +namespace folly { + +class NamedThreadFactory : public ThreadFactory { + public: + explicit NamedThreadFactory(folly::StringPiece prefix) + : prefix_(prefix.str()), suffix_(0) {} + + std::thread newThread(Func&& func) override { + auto thread = std::thread(std::move(func)); + folly::setThreadName( + thread.native_handle(), folly::to(prefix_, suffix_++)); + return thread; + } + + void setNamePrefix(folly::StringPiece prefix) { + prefix_ = prefix.str(); + } + + std::string getNamePrefix() { + return prefix_; + } + + private: + std::string prefix_; + std::atomic suffix_; +}; + +} // namespace folly diff --git a/folly/executors/PriorityLifoSemMPMCQueue.h b/folly/executors/PriorityLifoSemMPMCQueue.h new file mode 100644 index 00000000..797287c0 --- /dev/null +++ b/folly/executors/PriorityLifoSemMPMCQueue.h @@ -0,0 +1,107 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include +#include +#include +#include + +namespace folly { + +template +class PriorityLifoSemMPMCQueue : public BlockingQueue { + public: + // Note A: The queue pre-allocates all memory for max_capacity + // Note B: To use folly::Executor::*_PRI, for numPriorities == 2 + // MID_PRI and HI_PRI are treated at the same priority level. + PriorityLifoSemMPMCQueue(uint8_t numPriorities, size_t max_capacity) { + queues_.reserve(numPriorities); + for (int8_t i = 0; i < numPriorities; i++) { + queues_.emplace_back(max_capacity); + } + } + + uint8_t getNumPriorities() override { + return queues_.size(); + } + + // Add at medium priority by default + void add(T item) override { + addWithPriority(std::move(item), folly::Executor::MID_PRI); + } + + void addWithPriority(T item, int8_t priority) override { + int mid = getNumPriorities() / 2; + size_t queue = priority < 0 + ? std::max(0, mid + priority) + : std::min(getNumPriorities() - 1, mid + priority); + CHECK_LT(queue, queues_.size()); + switch (kBehavior) { // static + case QueueBehaviorIfFull::THROW: + if (!queues_[queue].write(std::move(item))) { + throw QueueFullException("LifoSemMPMCQueue full, can't add item"); + } + break; + case QueueBehaviorIfFull::BLOCK: + queues_[queue].blockingWrite(std::move(item)); + break; + } + sem_.post(); + } + + T take() override { + T item; + while (true) { + if (nonBlockingTake(item)) { + return item; + } + sem_.wait(); + } + } + + bool nonBlockingTake(T& item) { + for (auto it = queues_.rbegin(); it != queues_.rend(); it++) { + if (it->readIfNotEmpty(item)) { + return true; + } + } + return false; + } + + size_t size() override { + size_t size = 0; + for (auto& q : queues_) { + size += q.size(); + } + return size; + } + + size_t sizeGuess() const { + size_t size = 0; + for (auto& q : queues_) { + size += q.sizeGuess(); + } + return size; + } + + private: + folly::LifoSem sem_; + std::vector> queues_; +}; + +} // namespace folly diff --git a/folly/executors/PriorityThreadFactory.h b/folly/executors/PriorityThreadFactory.h new file mode 100644 index 00000000..ed46dd35 --- /dev/null +++ b/folly/executors/PriorityThreadFactory.h @@ -0,0 +1,60 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include + +#include +#include + +namespace folly { + +/** + * A ThreadFactory that sets nice values for each thread. The main + * use case for this class is if there are multiple + * CPUThreadPoolExecutors in a single process, or between multiple + * processes, where some should have a higher priority than the others. + * + * Note that per-thread nice values are not POSIX standard, but both + * pthreads and linux support per-thread nice. The default linux + * scheduler uses these values to do smart thread prioritization. + * sched_priority function calls only affect real-time schedulers. + */ +class PriorityThreadFactory : public ThreadFactory { + public: + explicit PriorityThreadFactory( + std::shared_ptr factory, + int priority) + : factory_(std::move(factory)), priority_(priority) {} + + std::thread newThread(Func&& func) override { + int priority = priority_; + return factory_->newThread([ priority, func = std::move(func) ]() mutable { + if (setpriority(PRIO_PROCESS, 0, priority) != 0) { + LOG(ERROR) << "setpriority failed (are you root?) with error " << errno, + strerror(errno); + } + func(); + }); + } + + private: + std::shared_ptr factory_; + int priority_; +}; + +} // folly diff --git a/folly/executors/SerialExecutor.cpp b/folly/executors/SerialExecutor.cpp new file mode 100644 index 00000000..9380d685 --- /dev/null +++ b/folly/executors/SerialExecutor.cpp @@ -0,0 +1,89 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include "SerialExecutor.h" + +#include +#include + +#include + +#include + +namespace folly { + +class SerialExecutor::TaskQueueImpl { + public: + void add(Func&& func) { + std::lock_guard lock(mutex_); + queue_.push(std::move(func)); + } + + void run() { + std::unique_lock lock(mutex_); + + ++scheduled_; + + if (scheduled_ > 1) { + return; + } + + do { + DCHECK(!queue_.empty()); + Func func = std::move(queue_.front()); + queue_.pop(); + lock.unlock(); + + try { + func(); + } catch (std::exception const& ex) { + LOG(ERROR) << "SerialExecutor: func threw unhandled exception " + << folly::exceptionStr(ex); + } catch (...) { + LOG(ERROR) << "SerialExecutor: func threw unhandled non-exception " + "object"; + } + + // Destroy the function (and the data it captures) before we acquire the + // lock again. + func = {}; + + lock.lock(); + --scheduled_; + } while (scheduled_); + } + + private: + std::mutex mutex_; + std::size_t scheduled_{0}; + std::queue queue_; +}; + +SerialExecutor::SerialExecutor(std::shared_ptr parent) + : parent_(std::move(parent)), + taskQueueImpl_(std::make_shared()) {} + +void SerialExecutor::add(Func func) { + taskQueueImpl_->add(std::move(func)); + parent_->add([impl = taskQueueImpl_] { impl->run(); }); +} + +void SerialExecutor::addWithPriority(Func func, int8_t priority) { + taskQueueImpl_->add(std::move(func)); + parent_->addWithPriority([impl = taskQueueImpl_] { impl->run(); }, priority); +} + +} // namespace folly diff --git a/folly/executors/SerialExecutor.h b/folly/executors/SerialExecutor.h new file mode 100644 index 00000000..c6c1caec --- /dev/null +++ b/folly/executors/SerialExecutor.h @@ -0,0 +1,87 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include + +#include +#include + +namespace folly { + +/** + * @class SerialExecutor + * + * @brief Executor that guarantees serial non-concurrent execution of added + * tasks + * + * SerialExecutor is similar to boost asio's strand concept. A SerialExecutor + * has a parent executor which is given at construction time (defaults to + * folly's global CPUExecutor). Tasks added to SerialExecutor are executed + * in the parent executor, however strictly non-concurrently and in the order + * they were added. + * + * SerialExecutor tries to schedule its tasks fairly. Every task submitted to + * it results in one task submitted to the parent executor. Whenever the parent + * executor executes one of those, one of the tasks submitted to SerialExecutor + * is marked for execution, which means it will either be executed at once, + * or if a task is currently being executed already, after that. + * + * The SerialExecutor may be deleted at any time. All tasks that have been + * submitted will still be executed with the same guarantees, as long as the + * parent executor is executing tasks. + */ + +class SerialExecutor : public folly::Executor { + public: + ~SerialExecutor() override = default; + SerialExecutor(SerialExecutor const&) = delete; + SerialExecutor& operator=(SerialExecutor const&) = delete; + SerialExecutor(SerialExecutor&&) = default; + SerialExecutor& operator=(SerialExecutor&&) = default; + + explicit SerialExecutor( + std::shared_ptr parent = folly::getCPUExecutor()); + + /** + * Add one task for execution in the parent executor + */ + void add(Func func) override; + + /** + * Add one task for execution in the parent executor, and use the given + * priority for one task submission to parent executor. + * + * Since in-order execution of tasks submitted to SerialExecutor is + * guaranteed, the priority given here does not necessarily reflect the + * execution priority of the task submitted with this call to + * `addWithPriority`. The given priority is passed on to the parent executor + * for the execution of one of the SerialExecutor's tasks. + */ + void addWithPriority(Func func, int8_t priority) override; + uint8_t getNumPriorities() const override { + return parent_->getNumPriorities(); + } + + private: + class TaskQueueImpl; + + std::shared_ptr parent_; + std::shared_ptr taskQueueImpl_; +}; + +} // namespace folly diff --git a/folly/executors/ThreadFactory.h b/folly/executors/ThreadFactory.h new file mode 100644 index 00000000..0af86322 --- /dev/null +++ b/folly/executors/ThreadFactory.h @@ -0,0 +1,30 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once +#include + +#include + +namespace folly { + +class ThreadFactory { + public: + virtual ~ThreadFactory() = default; + virtual std::thread newThread(Func&& func) = 0; +}; + +} // namespace folly diff --git a/folly/executors/ThreadPoolExecutor.cpp b/folly/executors/ThreadPoolExecutor.cpp new file mode 100644 index 00000000..552d981c --- /dev/null +++ b/folly/executors/ThreadPoolExecutor.cpp @@ -0,0 +1,258 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include + +#include + +namespace folly { + +ThreadPoolExecutor::ThreadPoolExecutor( + size_t /* numThreads */, + std::shared_ptr threadFactory, + bool isWaitForAll) + : threadFactory_(std::move(threadFactory)), + isWaitForAll_(isWaitForAll), + taskStatsCallbacks_(std::make_shared()), + threadPoolHook_("Wangle::ThreadPoolExecutor") {} + +ThreadPoolExecutor::~ThreadPoolExecutor() { + CHECK_EQ(0, threadList_.get().size()); +} + +ThreadPoolExecutor::Task::Task( + Func&& func, + std::chrono::milliseconds expiration, + Func&& expireCallback) + : func_(std::move(func)), + expiration_(expiration), + expireCallback_(std::move(expireCallback)), + context_(folly::RequestContext::saveContext()) { + // Assume that the task in enqueued on creation + enqueueTime_ = std::chrono::steady_clock::now(); +} + +void ThreadPoolExecutor::runTask(const ThreadPtr& thread, Task&& task) { + thread->idle = false; + auto startTime = std::chrono::steady_clock::now(); + task.stats_.waitTime = startTime - task.enqueueTime_; + if (task.expiration_ > std::chrono::milliseconds(0) && + task.stats_.waitTime >= task.expiration_) { + task.stats_.expired = true; + if (task.expireCallback_ != nullptr) { + task.expireCallback_(); + } + } else { + folly::RequestContextScopeGuard rctx(task.context_); + try { + task.func_(); + } catch (const std::exception& e) { + LOG(ERROR) << "ThreadPoolExecutor: func threw unhandled " + << typeid(e).name() << " exception: " << e.what(); + } catch (...) { + LOG(ERROR) << "ThreadPoolExecutor: func threw unhandled non-exception " + "object"; + } + task.stats_.runTime = std::chrono::steady_clock::now() - startTime; + } + thread->idle = true; + thread->lastActiveTime = std::chrono::steady_clock::now(); + thread->taskStatsCallbacks->callbackList.withRLock([&](auto& callbacks) { + *thread->taskStatsCallbacks->inCallback = true; + SCOPE_EXIT { + *thread->taskStatsCallbacks->inCallback = false; + }; + try { + for (auto& callback : callbacks) { + callback(task.stats_); + } + } catch (const std::exception& e) { + LOG(ERROR) << "ThreadPoolExecutor: task stats callback threw " + "unhandled " + << typeid(e).name() << " exception: " << e.what(); + } catch (...) { + LOG(ERROR) << "ThreadPoolExecutor: task stats callback threw " + "unhandled non-exception object"; + } + }); +} + +size_t ThreadPoolExecutor::numThreads() { + RWSpinLock::ReadHolder r{&threadListLock_}; + return threadList_.get().size(); +} + +void ThreadPoolExecutor::setNumThreads(size_t n) { + size_t numThreadsToJoin = 0; + { + RWSpinLock::WriteHolder w{&threadListLock_}; + const auto current = threadList_.get().size(); + if (n > current) { + addThreads(n - current); + } else if (n < current) { + numThreadsToJoin = current - n; + removeThreads(numThreadsToJoin, true); + } + } + joinStoppedThreads(numThreadsToJoin); + CHECK_EQ(n, threadList_.get().size()); + CHECK_EQ(0, stoppedThreads_.size()); +} + +// threadListLock_ is writelocked +void ThreadPoolExecutor::addThreads(size_t n) { + std::vector newThreads; + for (size_t i = 0; i < n; i++) { + newThreads.push_back(makeThread()); + } + for (auto& thread : newThreads) { + // TODO need a notion of failing to create the thread + // and then handling for that case + thread->handle = threadFactory_->newThread( + std::bind(&ThreadPoolExecutor::threadRun, this, thread)); + threadList_.add(thread); + } + for (auto& thread : newThreads) { + thread->startupBaton.wait(); + } + for (auto& o : observers_) { + for (auto& thread : newThreads) { + o->threadStarted(thread.get()); + } + } +} + +// threadListLock_ is writelocked +void ThreadPoolExecutor::removeThreads(size_t n, bool isJoin) { + CHECK_LE(n, threadList_.get().size()); + isJoin_ = isJoin; + stopThreads(n); +} + +void ThreadPoolExecutor::joinStoppedThreads(size_t n) { + for (size_t i = 0; i < n; i++) { + auto thread = stoppedThreads_.take(); + thread->handle.join(); + } +} + +void ThreadPoolExecutor::stop() { + size_t n = 0; + { + RWSpinLock::WriteHolder w{&threadListLock_}; + n = threadList_.get().size(); + removeThreads(n, false); + } + joinStoppedThreads(n); + CHECK_EQ(0, threadList_.get().size()); + CHECK_EQ(0, stoppedThreads_.size()); +} + +void ThreadPoolExecutor::join() { + size_t n = 0; + { + RWSpinLock::WriteHolder w{&threadListLock_}; + n = threadList_.get().size(); + removeThreads(n, true); + } + joinStoppedThreads(n); + CHECK_EQ(0, threadList_.get().size()); + CHECK_EQ(0, stoppedThreads_.size()); +} + +ThreadPoolExecutor::PoolStats ThreadPoolExecutor::getPoolStats() { + const auto now = std::chrono::steady_clock::now(); + RWSpinLock::ReadHolder r{&threadListLock_}; + ThreadPoolExecutor::PoolStats stats; + stats.threadCount = threadList_.get().size(); + for (auto thread : threadList_.get()) { + if (thread->idle) { + stats.idleThreadCount++; + const std::chrono::nanoseconds idleTime = now - thread->lastActiveTime; + stats.maxIdleTime = std::max(stats.maxIdleTime, idleTime); + } else { + stats.activeThreadCount++; + } + } + stats.pendingTaskCount = getPendingTaskCountImpl(r); + stats.totalTaskCount = stats.pendingTaskCount + stats.activeThreadCount; + return stats; +} + +uint64_t ThreadPoolExecutor::getPendingTaskCount() { + RWSpinLock::ReadHolder r{&threadListLock_}; + return getPendingTaskCountImpl(r); +} + +std::atomic ThreadPoolExecutor::Thread::nextId(0); + +void ThreadPoolExecutor::subscribeToTaskStats(TaskStatsCallback cb) { + if (*taskStatsCallbacks_->inCallback) { + throw std::runtime_error("cannot subscribe in task stats callback"); + } + taskStatsCallbacks_->callbackList.wlock()->push_back(std::move(cb)); +} + +void ThreadPoolExecutor::StoppedThreadQueue::add( + ThreadPoolExecutor::ThreadPtr item) { + std::lock_guard guard(mutex_); + queue_.push(std::move(item)); + sem_.post(); +} + +ThreadPoolExecutor::ThreadPtr ThreadPoolExecutor::StoppedThreadQueue::take() { + while (1) { + { + std::lock_guard guard(mutex_); + if (queue_.size() > 0) { + auto item = std::move(queue_.front()); + queue_.pop(); + return item; + } + } + sem_.wait(); + } +} + +size_t ThreadPoolExecutor::StoppedThreadQueue::size() { + std::lock_guard guard(mutex_); + return queue_.size(); +} + +void ThreadPoolExecutor::addObserver(std::shared_ptr o) { + RWSpinLock::ReadHolder r{&threadListLock_}; + observers_.push_back(o); + for (auto& thread : threadList_.get()) { + o->threadPreviouslyStarted(thread.get()); + } +} + +void ThreadPoolExecutor::removeObserver(std::shared_ptr o) { + RWSpinLock::ReadHolder r{&threadListLock_}; + for (auto& thread : threadList_.get()) { + o->threadNotYetStopped(thread.get()); + } + + for (auto it = observers_.begin(); it != observers_.end(); it++) { + if (*it == o) { + observers_.erase(it); + return; + } + } + DCHECK(false); +} + +} // namespace folly diff --git a/folly/executors/ThreadPoolExecutor.h b/folly/executors/ThreadPoolExecutor.h new file mode 100644 index 00000000..1e270b4c --- /dev/null +++ b/folly/executors/ThreadPoolExecutor.h @@ -0,0 +1,264 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +#include + +namespace folly { + +class ThreadPoolExecutor : public virtual folly::Executor { + public: + explicit ThreadPoolExecutor( + size_t numThreads, + std::shared_ptr threadFactory, + bool isWaitForAll = false); + + ~ThreadPoolExecutor() override; + + void add(Func func) override = 0; + virtual void + add(Func func, std::chrono::milliseconds expiration, Func expireCallback) = 0; + + void setThreadFactory(std::shared_ptr threadFactory) { + CHECK(numThreads() == 0); + threadFactory_ = std::move(threadFactory); + } + + std::shared_ptr getThreadFactory(void) { + return threadFactory_; + } + + size_t numThreads(); + void setNumThreads(size_t numThreads); + /* + * stop() is best effort - there is no guarantee that unexecuted tasks won't + * be executed before it returns. Specifically, IOThreadPoolExecutor's stop() + * behaves like join(). + */ + void stop(); + void join(); + + struct PoolStats { + PoolStats() + : threadCount(0), + idleThreadCount(0), + activeThreadCount(0), + pendingTaskCount(0), + totalTaskCount(0), + maxIdleTime(0) {} + size_t threadCount, idleThreadCount, activeThreadCount; + uint64_t pendingTaskCount, totalTaskCount; + std::chrono::nanoseconds maxIdleTime; + }; + + PoolStats getPoolStats(); + uint64_t getPendingTaskCount(); + + struct TaskStats { + TaskStats() : expired(false), waitTime(0), runTime(0) {} + bool expired; + std::chrono::nanoseconds waitTime; + std::chrono::nanoseconds runTime; + }; + + using TaskStatsCallback = std::function; + void subscribeToTaskStats(TaskStatsCallback cb); + + /** + * Base class for threads created with ThreadPoolExecutor. + * Some subclasses have methods that operate on these + * handles. + */ + class ThreadHandle { + public: + virtual ~ThreadHandle() = default; + }; + + /** + * Observer interface for thread start/stop. + * Provides hooks so actions can be taken when + * threads are created + */ + class Observer { + public: + virtual void threadStarted(ThreadHandle*) = 0; + virtual void threadStopped(ThreadHandle*) = 0; + virtual void threadPreviouslyStarted(ThreadHandle* h) { + threadStarted(h); + } + virtual void threadNotYetStopped(ThreadHandle* h) { + threadStopped(h); + } + virtual ~Observer() = default; + }; + + void addObserver(std::shared_ptr); + void removeObserver(std::shared_ptr); + + protected: + // Prerequisite: threadListLock_ writelocked + void addThreads(size_t n); + // Prerequisite: threadListLock_ writelocked + void removeThreads(size_t n, bool isJoin); + + struct TaskStatsCallbackRegistry; + + struct FOLLY_ALIGN_TO_AVOID_FALSE_SHARING Thread : public ThreadHandle { + explicit Thread(ThreadPoolExecutor* pool) + : id(nextId++), + handle(), + idle(true), + lastActiveTime(std::chrono::steady_clock::now()), + taskStatsCallbacks(pool->taskStatsCallbacks_) {} + + ~Thread() override = default; + + static std::atomic nextId; + uint64_t id; + std::thread handle; + bool idle; + std::chrono::steady_clock::time_point lastActiveTime; + folly::Baton<> startupBaton; + std::shared_ptr taskStatsCallbacks; + }; + + typedef std::shared_ptr ThreadPtr; + + struct Task { + explicit Task( + Func&& func, + std::chrono::milliseconds expiration, + Func&& expireCallback); + Func func_; + TaskStats stats_; + std::chrono::steady_clock::time_point enqueueTime_; + std::chrono::milliseconds expiration_; + Func expireCallback_; + std::shared_ptr context_; + }; + + static void runTask(const ThreadPtr& thread, Task&& task); + + // The function that will be bound to pool threads. It must call + // thread->startupBaton.post() when it's ready to consume work. + virtual void threadRun(ThreadPtr thread) = 0; + + // Stop n threads and put their ThreadPtrs in the stoppedThreads_ queue + // and remove them from threadList_, either synchronize or asynchronize + // Prerequisite: threadListLock_ writelocked + virtual void stopThreads(size_t n) = 0; + + // Join n stopped threads and remove them from waitingForJoinThreads_ queue. + // Should not hold a lock because joining thread operation may invoke some + // cleanup operations on the thread, and those cleanup operations may + // require a lock on ThreadPoolExecutor. + void joinStoppedThreads(size_t n); + + // Create a suitable Thread struct + virtual ThreadPtr makeThread() { + return std::make_shared(this); + } + + // Prerequisite: threadListLock_ readlocked + virtual uint64_t getPendingTaskCountImpl(const RWSpinLock::ReadHolder&) = 0; + + class ThreadList { + public: + void add(const ThreadPtr& state) { + auto it = std::lower_bound( + vec_.begin(), + vec_.end(), + state, + // compare method is a static method of class + // and therefore cannot be inlined by compiler + // as a template predicate of the STL algorithm + // but wrapped up with the lambda function (lambda will be inlined) + // compiler can inline compare method as well + [&](const ThreadPtr& ts1, const ThreadPtr& ts2) -> bool { // inline + return compare(ts1, ts2); + }); + vec_.insert(it, state); + } + + void remove(const ThreadPtr& state) { + auto itPair = std::equal_range( + vec_.begin(), + vec_.end(), + state, + // the same as above + [&](const ThreadPtr& ts1, const ThreadPtr& ts2) -> bool { // inline + return compare(ts1, ts2); + }); + CHECK(itPair.first != vec_.end()); + CHECK(std::next(itPair.first) == itPair.second); + vec_.erase(itPair.first); + } + + const std::vector& get() const { + return vec_; + } + + private: + static bool compare(const ThreadPtr& ts1, const ThreadPtr& ts2) { + return ts1->id < ts2->id; + } + + std::vector vec_; + }; + + class StoppedThreadQueue : public BlockingQueue { + public: + void add(ThreadPtr item) override; + ThreadPtr take() override; + size_t size() override; + + private: + folly::LifoSem sem_; + std::mutex mutex_; + std::queue queue_; + }; + + std::shared_ptr threadFactory_; + const bool isWaitForAll_; // whether to wait till event base loop exits + + ThreadList threadList_; + folly::RWSpinLock threadListLock_; + StoppedThreadQueue stoppedThreads_; + std::atomic isJoin_; // whether the current downsizing is a join + + struct TaskStatsCallbackRegistry { + folly::ThreadLocal inCallback; + folly::Synchronized> callbackList; + }; + std::shared_ptr taskStatsCallbacks_; + std::vector> observers_; + folly::ThreadPoolListHook threadPoolHook_; +}; + +} // namespace folly diff --git a/folly/executors/ThreadedExecutor.cpp b/folly/executors/ThreadedExecutor.cpp new file mode 100644 index 00000000..d75e82b6 --- /dev/null +++ b/folly/executors/ThreadedExecutor.cpp @@ -0,0 +1,111 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include + +#include + +#include + +#include +#include + +namespace folly { + +template +static auto with_unique_lock(std::mutex& m, F&& f) -> decltype(f()) { + std::unique_lock lock(m); + return f(); +} + +ThreadedExecutor::ThreadedExecutor(std::shared_ptr threadFactory) + : threadFactory_(std::move(threadFactory)) { + controlt_ = std::thread([this] { control(); }); +} + +ThreadedExecutor::~ThreadedExecutor() { + stopping_.store(true, std::memory_order_release); + notify(); + controlt_.join(); + CHECK(running_.empty()); + CHECK(finished_.empty()); +} + +void ThreadedExecutor::add(Func func) { + CHECK(!stopping_.load(std::memory_order_acquire)); + with_unique_lock(enqueuedm_, [&] { enqueued_.push_back(std::move(func)); }); + notify(); +} + +std::shared_ptr ThreadedExecutor::newDefaultThreadFactory() { + return std::make_shared("Threaded"); +} + +void ThreadedExecutor::notify() { + with_unique_lock(controlm_, [&] { controls_ = true; }); + controlc_.notify_one(); +} + +void ThreadedExecutor::control() { + folly::setThreadName("ThreadedCtrl"); + auto looping = true; + while (looping) { + controlWait(); + looping = controlPerformAll(); + } +} + +void ThreadedExecutor::controlWait() { + constexpr auto kMaxWait = std::chrono::seconds(10); + std::unique_lock lock(controlm_); + controlc_.wait_for(lock, kMaxWait, [&] { return controls_; }); + controls_ = false; +} + +void ThreadedExecutor::work(Func& func) { + func(); + auto id = std::this_thread::get_id(); + with_unique_lock(finishedm_, [&] { finished_.push_back(id); }); + notify(); +} + +void ThreadedExecutor::controlJoinFinishedThreads() { + std::deque finishedt; + with_unique_lock(finishedm_, [&] { std::swap(finishedt, finished_); }); + for (auto id : finishedt) { + running_[id].join(); + running_.erase(id); + } +} + +void ThreadedExecutor::controlLaunchEnqueuedTasks() { + std::deque enqueuedt; + with_unique_lock(enqueuedm_, [&] { std::swap(enqueuedt, enqueued_); }); + for (auto& f : enqueuedt) { + auto th = threadFactory_->newThread( + [ this, f = std::move(f) ]() mutable { work(f); }); + auto id = th.get_id(); + running_[id] = std::move(th); + } +} + +bool ThreadedExecutor::controlPerformAll() { + auto stopping = stopping_.load(std::memory_order_acquire); + controlJoinFinishedThreads(); + controlLaunchEnqueuedTasks(); + return !stopping || !running_.empty(); +} +} diff --git a/folly/executors/ThreadedExecutor.h b/folly/executors/ThreadedExecutor.h new file mode 100644 index 00000000..81c5fca8 --- /dev/null +++ b/folly/executors/ThreadedExecutor.h @@ -0,0 +1,99 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +namespace folly { + +/*** + * ThreadedExecutor + * + * An executor for blocking tasks. + * + * This executor runs each task in its own thread. It works well for tasks + * which mostly sleep, but works poorly for tasks which mostly compute. + * + * For each task given to the executor with `add`, the executor spawns a new + * thread for that task, runs the task in that thread, and joins the thread + * after the task has completed. + * + * Spawning and joining task threads are done in the executor's internal + * control thread. Calls to `add` put the tasks to be run into a queue, where + * the control thread will find them. + * + * There is currently no limitation on, or throttling of, concurrency. + * + * This executor is not currently optimized for performance. For example, it + * makes no attempt to re-use task threads. Rather, it exists primarily to + * offload sleep-heavy tasks from the CPU executor, where they might otherwise + * be run. + */ +class ThreadedExecutor : public virtual folly::Executor { + public: + explicit ThreadedExecutor( + std::shared_ptr threadFactory = newDefaultThreadFactory()); + ~ThreadedExecutor() override; + + ThreadedExecutor(ThreadedExecutor const&) = delete; + ThreadedExecutor(ThreadedExecutor&&) = delete; + + ThreadedExecutor& operator=(ThreadedExecutor const&) = delete; + ThreadedExecutor& operator=(ThreadedExecutor&&) = delete; + + void add(Func func) override; + + private: + static std::shared_ptr newDefaultThreadFactory(); + + void notify(); + void control(); + void controlWait(); + bool controlPerformAll(); + void controlJoinFinishedThreads(); + void controlLaunchEnqueuedTasks(); + + void work(Func& func); + + std::shared_ptr threadFactory_; + + std::atomic stopping_{false}; + + std::mutex controlm_; + std::condition_variable controlc_; + bool controls_ = false; + std::thread controlt_; + + std::mutex enqueuedm_; + std::deque enqueued_; + + // Accessed only by the control thread, so no synchronization. + std::map running_; + + std::mutex finishedm_; + std::deque finished_; +}; +} diff --git a/folly/executors/UnboundedBlockingQueue.h b/folly/executors/UnboundedBlockingQueue.h new file mode 100644 index 00000000..3fb09b3f --- /dev/null +++ b/folly/executors/UnboundedBlockingQueue.h @@ -0,0 +1,64 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include +#include +#include +#include + +namespace folly { + +// Warning: this is effectively just a std::deque wrapped in a single mutex +// We are aiming to add a more performant concurrent unbounded queue in the +// future, but this class is available if you must have an unbounded queue +// and can tolerate any contention. +template +class UnboundedBlockingQueue : public BlockingQueue { + public: + virtual ~UnboundedBlockingQueue() {} + + void add(T item) override { + queue_.wlock()->push(std::move(item)); + sem_.post(); + } + + T take() override { + while (true) { + { + auto ulockedQueue = queue_.ulock(); + if (!ulockedQueue->empty()) { + auto wlockedQueue = ulockedQueue.moveFromUpgradeToWrite(); + T item = std::move(wlockedQueue->front()); + wlockedQueue->pop(); + return item; + } + } + sem_.wait(); + } + } + + size_t size() override { + return queue_.rlock()->size(); + } + + private: + LifoSem sem_; + Synchronized> queue_; +}; + +} // namespace folly diff --git a/folly/executors/test/AsyncTest.cpp b/folly/executors/test/AsyncTest.cpp new file mode 100644 index 00000000..60f913f7 --- /dev/null +++ b/folly/executors/test/AsyncTest.cpp @@ -0,0 +1,51 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include +#include +#include + +using namespace folly; + +TEST(AsyncFunc, manual_executor) { + auto x = std::make_shared(); + auto oldX = getCPUExecutor(); + setCPUExecutor(x); + auto f = async([] { return 42; }); + EXPECT_FALSE(f.isReady()); + x->run(); + EXPECT_EQ(42, f.value()); + setCPUExecutor(oldX); +} + +TEST(AsyncFunc, value_lambda) { + auto lambda = [] { return 42; }; + auto future = async(lambda); + EXPECT_EQ(42, future.get()); +} + +TEST(AsyncFunc, void_lambda) { + auto lambda = [] { /*do something*/ return; }; + auto future = async(lambda); + // Futures with a void returning function, return Unit type + EXPECT_EQ(typeid(Unit), typeid(future.get())); +} + +TEST(AsyncFunc, moveonly_lambda) { + auto lambda = [] { return std::unique_ptr(new int(42)); }; + auto future = async(lambda); + EXPECT_EQ(42, *future.get()); +} diff --git a/folly/executors/test/CodelTest.cpp b/folly/executors/test/CodelTest.cpp new file mode 100644 index 00000000..71450011 --- /dev/null +++ b/folly/executors/test/CodelTest.cpp @@ -0,0 +1,89 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include +#include +#include +#include + +using std::chrono::milliseconds; +using std::this_thread::sleep_for; + +TEST(CodelTest, Basic) { + folly::Codel c; + std::this_thread::sleep_for(milliseconds(110)); + // This interval is overloaded + EXPECT_FALSE(c.overloaded(milliseconds(100))); + std::this_thread::sleep_for(milliseconds(90)); + // At least two requests must happen in an interval before they will fail + EXPECT_FALSE(c.overloaded(milliseconds(50))); + EXPECT_TRUE(c.overloaded(milliseconds(50))); + std::this_thread::sleep_for(milliseconds(110)); + // Previous interval is overloaded, but 2ms isn't enough to fail + EXPECT_FALSE(c.overloaded(milliseconds(2))); + std::this_thread::sleep_for(milliseconds(90)); + // 20 ms > target interval * 2 + EXPECT_TRUE(c.overloaded(milliseconds(20))); +} + +TEST(CodelTest, highLoad) { + folly::Codel c; + c.overloaded(milliseconds(40)); + EXPECT_EQ(100, c.getLoad()); +} + +TEST(CodelTest, mediumLoad) { + folly::Codel c; + c.overloaded(milliseconds(20)); + sleep_for(milliseconds(90)); + // this is overloaded but this request shouldn't drop because it's not > + // slough timeout + EXPECT_FALSE(c.overloaded(milliseconds(8))); + EXPECT_GT(100, c.getLoad()); +} + +TEST(CodelTest, reducingLoad) { + folly::Codel c; + c.overloaded(milliseconds(20)); + sleep_for(milliseconds(90)); + EXPECT_FALSE(c.overloaded(milliseconds(4))); +} + +TEST(CodelTest, oneRequestNoDrop) { + folly::Codel c; + EXPECT_FALSE(c.overloaded(milliseconds(20))); +} + +TEST(CodelTest, getLoadSanity) { + folly::Codel c; + // should be 100% but leave a litte wiggle room. + c.overloaded(milliseconds(10)); + EXPECT_LT(99, c.getLoad()); + EXPECT_GT(101, c.getLoad()); + + // should be 70% but leave a litte wiggle room. + c.overloaded(milliseconds(7)); + EXPECT_LT(60, c.getLoad()); + EXPECT_GT(80, c.getLoad()); + + // should be 20% but leave a litte wiggle room. + c.overloaded(milliseconds(2)); + EXPECT_LT(10, c.getLoad()); + EXPECT_GT(30, c.getLoad()); + + // this test demonstrates how silly getLoad() is, but silly isn't + // necessarily useless +} diff --git a/folly/executors/test/FiberIOExecutorTest.cpp b/folly/executors/test/FiberIOExecutorTest.cpp new file mode 100644 index 00000000..87d60c7e --- /dev/null +++ b/folly/executors/test/FiberIOExecutorTest.cpp @@ -0,0 +1,58 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include + +#include +#include + +#include + +namespace { + +class FiberIOExecutorTest : public testing::Test {}; +} + +TEST_F(FiberIOExecutorTest, event_base) { + auto tpe = std::make_shared(1); + folly::FiberIOExecutor e(tpe); + + ASSERT_NE(e.getEventBase(), nullptr); + ASSERT_EQ(e.getEventBase(), tpe->getEventBase()); +} + +TEST_F(FiberIOExecutorTest, basic_execution) { + auto tpe = std::make_shared(1); + folly::FiberIOExecutor e(tpe); + + // FiberIOExecutor should add tasks using the FiberManager mapped to the + // IOThreadPoolExecutor's event base. + folly::Baton<> baton; + bool inContext = false; + + e.add([&]() { + inContext = folly::fibers::onFiber(); + auto& lc = dynamic_cast( + folly::fibers::getFiberManager(*e.getEventBase()).loopController()); + auto& eb = lc.getEventBase()->getEventBase(); + inContext = + inContext && &eb == folly::EventBaseManager::get()->getEventBase(); + baton.post(); + }); + baton.wait(); + + ASSERT_TRUE(inContext); +} diff --git a/folly/executors/test/GlobalExecutorTest.cpp b/folly/executors/test/GlobalExecutorTest.cpp new file mode 100644 index 00000000..06484cfc --- /dev/null +++ b/folly/executors/test/GlobalExecutorTest.cpp @@ -0,0 +1,85 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include +#include +#include + +using namespace folly; + +TEST(GlobalExecutorTest, GlobalCPUExecutor) { + class DummyExecutor : public folly::Executor { + public: + void add(Func f) override { + f(); + count++; + } + int count{0}; + }; + + // The default CPU executor is a synchronous inline executor, lets verify + // that work we add is executed + auto count = 0; + auto f = [&]() { count++; }; + + // Don't explode, we should create the default global CPUExecutor lazily here. + getCPUExecutor()->add(f); + EXPECT_EQ(1, count); + + { + auto dummy = std::make_shared(); + setCPUExecutor(dummy); + getCPUExecutor()->add(f); + // Make sure we were properly installed. + EXPECT_EQ(1, dummy->count); + EXPECT_EQ(2, count); + } + + // Don't explode, we should restore the default global CPUExecutor because our + // weak reference to dummy has expired + getCPUExecutor()->add(f); + EXPECT_EQ(3, count); +} + +TEST(GlobalExecutorTest, GlobalIOExecutor) { + class DummyExecutor : public IOExecutor { + public: + void add(Func) override { + count++; + } + folly::EventBase* getEventBase() override { + return nullptr; + } + int count{0}; + }; + + auto f = []() {}; + + // Don't explode, we should create the default global IOExecutor lazily here. + getIOExecutor()->add(f); + + { + auto dummy = std::make_shared(); + setIOExecutor(dummy); + getIOExecutor()->add(f); + // Make sure we were properly installed. + EXPECT_EQ(1, dummy->count); + } + + // Don't explode, we should restore the default global IOExecutor because our + // weak reference to dummy has expired + getIOExecutor()->add(f); +} diff --git a/folly/executors/test/SerialExecutorTest.cpp b/folly/executors/test/SerialExecutorTest.cpp new file mode 100644 index 00000000..1a132ef1 --- /dev/null +++ b/folly/executors/test/SerialExecutorTest.cpp @@ -0,0 +1,155 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include + +#include + +#include +#include +#include +#include + +using namespace std::chrono; +using folly::SerialExecutor; + +namespace { +void burnMs(uint64_t ms) { + /* sleep override */ std::this_thread::sleep_for(milliseconds(ms)); +} +} // namespace + +void SimpleTest(std::shared_ptr const& parent) { + SerialExecutor executor(parent); + + std::vector values; + std::vector expected; + + for (int i = 0; i < 20; ++i) { + executor.add([i, &values] { + // make this extra vulnerable to concurrent execution + values.push_back(0); + burnMs(10); + values.back() = i; + }); + expected.push_back(i); + } + + // wait until last task has executed + folly::Baton<> finished_baton; + executor.add([&finished_baton] { finished_baton.post(); }); + finished_baton.wait(); + + EXPECT_EQ(expected, values); +} + +TEST(SerialExecutor, Simple) { + SimpleTest(std::make_shared(4)); +} +TEST(SerialExecutor, SimpleInline) { + SimpleTest(std::make_shared()); +} + +// The Afterlife test only works with an asynchronous executor (not the +// InlineExecutor), because we want execution of tasks to happen after we +// destroy the SerialExecutor +TEST(SerialExecutor, Afterlife) { + auto cpu_executor = std::make_shared(4); + auto executor = std::make_unique(cpu_executor); + + // block executor until we call start_baton.post() + folly::Baton<> start_baton; + executor->add([&start_baton] { start_baton.wait(); }); + + std::vector values; + std::vector expected; + + for (int i = 0; i < 20; ++i) { + executor->add([i, &values] { + // make this extra vulnerable to concurrent execution + values.push_back(0); + burnMs(10); + values.back() = i; + }); + expected.push_back(i); + } + + folly::Baton<> finished_baton; + executor->add([&finished_baton] { finished_baton.post(); }); + + // destroy SerialExecutor + executor.reset(); + + // now kick off the tasks + start_baton.post(); + + // wait until last task has executed + finished_baton.wait(); + + EXPECT_EQ(expected, values); +} + +void RecursiveAddTest(std::shared_ptr const& parent) { + SerialExecutor executor(parent); + + folly::Baton<> finished_baton; + + std::vector values; + std::vector expected = {{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}}; + + int i = 0; + std::function lambda = [&] { + if (i < 10) { + // make this extra vulnerable to concurrent execution + values.push_back(0); + burnMs(10); + values.back() = i; + executor.add(lambda); + } else if (i < 12) { + // Below we will post this lambda three times to the executor. When + // executed, the lambda will re-post itself during the first ten + // executions. Afterwards we do nothing twice (this else-branch), and + // then on the 13th execution signal that we are finished. + } else { + finished_baton.post(); + } + ++i; + }; + + executor.add(lambda); + executor.add(lambda); + executor.add(lambda); + + // wait until last task has executed + finished_baton.wait(); + + EXPECT_EQ(expected, values); +} + +TEST(SerialExecutor, RecursiveAdd) { + RecursiveAddTest(std::make_shared(4)); +} +TEST(SerialExecutor, RecursiveAddInline) { + RecursiveAddTest(std::make_shared()); +} + +TEST(SerialExecutor, ExecutionThrows) { + SerialExecutor executor(std::make_shared()); + + // an empty Func will throw std::bad_function_call when invoked, + // but SerialExecutor should catch that exception + executor.add(folly::Func{}); +} diff --git a/folly/executors/test/ThreadPoolExecutorTest.cpp b/folly/executors/test/ThreadPoolExecutorTest.cpp new file mode 100644 index 00000000..66388f9b --- /dev/null +++ b/folly/executors/test/ThreadPoolExecutorTest.cpp @@ -0,0 +1,596 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include + +#include +#include +#include +#include +#include +#include +#include + +using namespace folly; +using namespace std::chrono; + +static Func burnMs(uint64_t ms) { + return [ms]() { std::this_thread::sleep_for(milliseconds(ms)); }; +} + +template +static void basic() { + // Create and destroy + TPE tpe(10); +} + +TEST(ThreadPoolExecutorTest, CPUBasic) { + basic(); +} + +TEST(IOThreadPoolExecutorTest, IOBasic) { + basic(); +} + +template +static void resize() { + TPE tpe(100); + EXPECT_EQ(100, tpe.numThreads()); + tpe.setNumThreads(50); + EXPECT_EQ(50, tpe.numThreads()); + tpe.setNumThreads(150); + EXPECT_EQ(150, tpe.numThreads()); +} + +TEST(ThreadPoolExecutorTest, CPUResize) { + resize(); +} + +TEST(ThreadPoolExecutorTest, IOResize) { + resize(); +} + +template +static void stop() { + TPE tpe(1); + std::atomic completed(0); + auto f = [&]() { + burnMs(10)(); + completed++; + }; + for (int i = 0; i < 1000; i++) { + tpe.add(f); + } + tpe.stop(); + EXPECT_GT(1000, completed); +} + +// IOThreadPoolExecutor's stop() behaves like join(). Outstanding tasks belong +// to the event base, will be executed upon its destruction, and cannot be +// taken back. +template <> +void stop() { + IOThreadPoolExecutor tpe(1); + std::atomic completed(0); + auto f = [&]() { + burnMs(10)(); + completed++; + }; + for (int i = 0; i < 10; i++) { + tpe.add(f); + } + tpe.stop(); + EXPECT_EQ(10, completed); +} + +TEST(ThreadPoolExecutorTest, CPUStop) { + stop(); +} + +TEST(ThreadPoolExecutorTest, IOStop) { + stop(); +} + +template +static void join() { + TPE tpe(10); + std::atomic completed(0); + auto f = [&]() { + burnMs(1)(); + completed++; + }; + for (int i = 0; i < 1000; i++) { + tpe.add(f); + } + tpe.join(); + EXPECT_EQ(1000, completed); +} + +TEST(ThreadPoolExecutorTest, CPUJoin) { + join(); +} + +TEST(ThreadPoolExecutorTest, IOJoin) { + join(); +} + +template +static void resizeUnderLoad() { + TPE tpe(10); + std::atomic completed(0); + auto f = [&]() { + burnMs(1)(); + completed++; + }; + for (int i = 0; i < 1000; i++) { + tpe.add(f); + } + tpe.setNumThreads(5); + tpe.setNumThreads(15); + tpe.join(); + EXPECT_EQ(1000, completed); +} + +TEST(ThreadPoolExecutorTest, CPUResizeUnderLoad) { + resizeUnderLoad(); +} + +TEST(ThreadPoolExecutorTest, IOResizeUnderLoad) { + resizeUnderLoad(); +} + +template +static void poolStats() { + folly::Baton<> startBaton, endBaton; + TPE tpe(1); + auto stats = tpe.getPoolStats(); + EXPECT_EQ(1, stats.threadCount); + EXPECT_EQ(1, stats.idleThreadCount); + EXPECT_EQ(0, stats.activeThreadCount); + EXPECT_EQ(0, stats.pendingTaskCount); + EXPECT_EQ(0, tpe.getPendingTaskCount()); + EXPECT_EQ(0, stats.totalTaskCount); + tpe.add([&]() { + startBaton.post(); + endBaton.wait(); + }); + tpe.add([&]() {}); + startBaton.wait(); + stats = tpe.getPoolStats(); + EXPECT_EQ(1, stats.threadCount); + EXPECT_EQ(0, stats.idleThreadCount); + EXPECT_EQ(1, stats.activeThreadCount); + EXPECT_EQ(1, stats.pendingTaskCount); + EXPECT_EQ(1, tpe.getPendingTaskCount()); + EXPECT_EQ(2, stats.totalTaskCount); + endBaton.post(); +} + +TEST(ThreadPoolExecutorTest, CPUPoolStats) { + poolStats(); +} + +TEST(ThreadPoolExecutorTest, IOPoolStats) { + poolStats(); +} + +template +static void taskStats() { + TPE tpe(1); + std::atomic c(0); + tpe.subscribeToTaskStats([&](ThreadPoolExecutor::TaskStats stats) { + int i = c++; + EXPECT_LT(milliseconds(0), stats.runTime); + if (i == 1) { + EXPECT_LT(milliseconds(0), stats.waitTime); + } + }); + tpe.add(burnMs(10)); + tpe.add(burnMs(10)); + tpe.join(); + EXPECT_EQ(2, c); +} + +TEST(ThreadPoolExecutorTest, CPUTaskStats) { + taskStats(); +} + +TEST(ThreadPoolExecutorTest, IOTaskStats) { + taskStats(); +} + +template +static void expiration() { + TPE tpe(1); + std::atomic statCbCount(0); + tpe.subscribeToTaskStats([&](ThreadPoolExecutor::TaskStats stats) { + int i = statCbCount++; + if (i == 0) { + EXPECT_FALSE(stats.expired); + } else if (i == 1) { + EXPECT_TRUE(stats.expired); + } else { + FAIL(); + } + }); + std::atomic expireCbCount(0); + auto expireCb = [&]() { expireCbCount++; }; + tpe.add(burnMs(10), seconds(60), expireCb); + tpe.add(burnMs(10), milliseconds(10), expireCb); + tpe.join(); + EXPECT_EQ(2, statCbCount); + EXPECT_EQ(1, expireCbCount); +} + +TEST(ThreadPoolExecutorTest, CPUExpiration) { + expiration(); +} + +TEST(ThreadPoolExecutorTest, IOExpiration) { + expiration(); +} + +template +static void futureExecutor() { + FutureExecutor fe(2); + std::atomic c{0}; + fe.addFuture([]() { return makeFuture(42); }).then([&](Try&& t) { + c++; + EXPECT_EQ(42, t.value()); + }); + fe.addFuture([]() { return 100; }).then([&](Try&& t) { + c++; + EXPECT_EQ(100, t.value()); + }); + fe.addFuture([]() { return makeFuture(); }).then([&](Try&& t) { + c++; + EXPECT_NO_THROW(t.value()); + }); + fe.addFuture([]() { return; }).then([&](Try&& t) { + c++; + EXPECT_NO_THROW(t.value()); + }); + fe.addFuture([]() { throw std::runtime_error("oops"); }) + .then([&](Try&& t) { + c++; + EXPECT_THROW(t.value(), std::runtime_error); + }); + // Test doing actual async work + folly::Baton<> baton; + fe.addFuture([&]() { + auto p = std::make_shared>(); + std::thread t([p]() { + burnMs(10)(); + p->setValue(42); + }); + t.detach(); + return p->getFuture(); + }) + .then([&](Try&& t) { + EXPECT_EQ(42, t.value()); + c++; + baton.post(); + }); + baton.wait(); + fe.join(); + EXPECT_EQ(6, c); +} + +TEST(ThreadPoolExecutorTest, CPUFuturePool) { + futureExecutor(); +} + +TEST(ThreadPoolExecutorTest, IOFuturePool) { + futureExecutor(); +} + +TEST(ThreadPoolExecutorTest, PriorityPreemptionTest) { + bool tookLopri = false; + auto completed = 0; + auto hipri = [&] { + EXPECT_FALSE(tookLopri); + completed++; + }; + auto lopri = [&] { + tookLopri = true; + completed++; + }; + CPUThreadPoolExecutor pool(0, 2); + for (int i = 0; i < 50; i++) { + pool.addWithPriority(lopri, Executor::LO_PRI); + } + for (int i = 0; i < 50; i++) { + pool.addWithPriority(hipri, Executor::HI_PRI); + } + pool.setNumThreads(1); + pool.join(); + EXPECT_EQ(100, completed); +} + +class TestObserver : public ThreadPoolExecutor::Observer { + public: + void threadStarted(ThreadPoolExecutor::ThreadHandle*) override { + threads_++; + } + void threadStopped(ThreadPoolExecutor::ThreadHandle*) override { + threads_--; + } + void threadPreviouslyStarted(ThreadPoolExecutor::ThreadHandle*) override { + threads_++; + } + void threadNotYetStopped(ThreadPoolExecutor::ThreadHandle*) override { + threads_--; + } + void checkCalls() { + ASSERT_EQ(threads_, 0); + } + + private: + std::atomic threads_{0}; +}; + +TEST(ThreadPoolExecutorTest, IOObserver) { + auto observer = std::make_shared(); + + { + IOThreadPoolExecutor exe(10); + exe.addObserver(observer); + exe.setNumThreads(3); + exe.setNumThreads(0); + exe.setNumThreads(7); + exe.removeObserver(observer); + exe.setNumThreads(10); + } + + observer->checkCalls(); +} + +TEST(ThreadPoolExecutorTest, CPUObserver) { + auto observer = std::make_shared(); + + { + CPUThreadPoolExecutor exe(10); + exe.addObserver(observer); + exe.setNumThreads(3); + exe.setNumThreads(0); + exe.setNumThreads(7); + exe.removeObserver(observer); + exe.setNumThreads(10); + } + + observer->checkCalls(); +} + +TEST(ThreadPoolExecutorTest, AddWithPriority) { + std::atomic_int c{0}; + auto f = [&] { c++; }; + + // IO exe doesn't support priorities + IOThreadPoolExecutor ioExe(10); + EXPECT_THROW(ioExe.addWithPriority(f, 0), std::runtime_error); + + CPUThreadPoolExecutor cpuExe(10, 3); + cpuExe.addWithPriority(f, -1); + cpuExe.addWithPriority(f, 0); + cpuExe.addWithPriority(f, 1); + cpuExe.addWithPriority(f, -2); // will add at the lowest priority + cpuExe.addWithPriority(f, 2); // will add at the highest priority + cpuExe.addWithPriority(f, Executor::LO_PRI); + cpuExe.addWithPriority(f, Executor::HI_PRI); + cpuExe.join(); + + EXPECT_EQ(7, c); +} + +TEST(ThreadPoolExecutorTest, BlockingQueue) { + std::atomic_int c{0}; + auto f = [&] { + burnMs(1)(); + c++; + }; + const int kQueueCapacity = 1; + const int kThreads = 1; + + auto queue = std::make_unique>(kQueueCapacity); + + CPUThreadPoolExecutor cpuExe( + kThreads, + std::move(queue), + std::make_shared("CPUThreadPool")); + + // Add `f` five times. It sleeps for 1ms every time. Calling + // `cppExec.add()` is *almost* guaranteed to block because there's + // only 1 cpu worker thread. + for (int i = 0; i < 5; i++) { + EXPECT_NO_THROW(cpuExe.add(f)); + } + cpuExe.join(); + + EXPECT_EQ(5, c); +} + +TEST(PriorityThreadFactoryTest, ThreadPriority) { + PriorityThreadFactory factory( + std::make_shared("stuff"), 1); + int actualPriority = -21; + factory.newThread([&]() { actualPriority = getpriority(PRIO_PROCESS, 0); }) + .join(); + EXPECT_EQ(1, actualPriority); +} + +class TestData : public folly::RequestData { + public: + explicit TestData(int data) : data_(data) {} + ~TestData() override {} + int data_; +}; + +TEST(ThreadPoolExecutorTest, RequestContext) { + CPUThreadPoolExecutor executor(1); + + RequestContextScopeGuard rctx; // create new request context for this scope + EXPECT_EQ(nullptr, RequestContext::get()->getContextData("test")); + RequestContext::get()->setContextData( + "test", std::unique_ptr(new TestData(42))); + auto data = RequestContext::get()->getContextData("test"); + EXPECT_EQ(42, dynamic_cast(data)->data_); + + executor.add([] { + auto data = RequestContext::get()->getContextData("test"); + ASSERT_TRUE(data != nullptr); + EXPECT_EQ(42, dynamic_cast(data)->data_); + }); +} + +struct SlowMover { + explicit SlowMover(bool slow = false) : slow(slow) {} + SlowMover(SlowMover&& other) noexcept { + *this = std::move(other); + } + SlowMover& operator=(SlowMover&& other) noexcept { + slow = other.slow; + if (slow) { + /* sleep override */ std::this_thread::sleep_for(milliseconds(50)); + } + return *this; + } + + bool slow; +}; + +TEST(ThreadPoolExecutorTest, BugD3527722) { + // Test that the queue does not get stuck if writes are completed in + // order opposite to how they are initiated. + LifoSemMPMCQueue q(1024); + std::atomic turn{}; + + std::thread consumer1([&] { + ++turn; + q.take(); + }); + std::thread consumer2([&] { + ++turn; + q.take(); + }); + + std::thread producer1([&] { + ++turn; + while (turn < 4) + ; + ++turn; + q.add(SlowMover(true)); + }); + std::thread producer2([&] { + ++turn; + while (turn < 5) + ; + q.add(SlowMover(false)); + }); + + producer1.join(); + producer2.join(); + consumer1.join(); + consumer2.join(); +} + +template +static void ShutdownTest() { + // test that adding a .then() after we have + // started shutting down does not deadlock + folly::Optional> f; + { + TPE fe(1); + f = folly::makeFuture().via(&fe).then([]() { burnMs(100)(); }).then([]() { + return 77; + }); + } + EXPECT_THROW(f->get(), ERR_T); +} + +TEST(ThreadPoolExecutorTest, ShutdownTestIO) { + ShutdownTest(); +} + +TEST(ThreadPoolExecutorTest, ShutdownTestCPU) { + ShutdownTest(); +} + +template +static void removeThreadTest() { + // test that adding a .then() after we have removed some threads + // doesn't cause deadlock and they are executed on different threads + folly::Optional> f; + std::thread::id id1, id2; + TPE fe(2); + f = folly::makeFuture() + .via(&fe) + .then([&id1]() { + burnMs(100)(); + id1 = std::this_thread::get_id(); + }) + .then([&id2]() { + return 77; + id2 = std::this_thread::get_id(); + }); + fe.setNumThreads(1); + + // future::then should be fulfilled because there is other thread available + EXPECT_EQ(77, f->get()); + // two thread should be different because then part should be rescheduled to + // the other thread + EXPECT_NE(id1, id2); +} + +TEST(ThreadPoolExecutorTest, RemoveThreadTestIO) { + removeThreadTest(); +} + +TEST(ThreadPoolExecutorTest, RemoveThreadTestCPU) { + removeThreadTest(); +} + +template +static void resizeThreadWhileExecutingTest() { + TPE tpe(10); + EXPECT_EQ(10, tpe.numThreads()); + + std::atomic completed(0); + auto f = [&]() { + burnMs(10)(); + completed++; + }; + for (int i = 0; i < 1000; i++) { + tpe.add(f); + } + tpe.setNumThreads(8); + EXPECT_EQ(8, tpe.numThreads()); + tpe.setNumThreads(5); + EXPECT_EQ(5, tpe.numThreads()); + tpe.setNumThreads(15); + EXPECT_EQ(15, tpe.numThreads()); + tpe.stop(); + EXPECT_EQ(1000, completed); +} + +TEST(ThreadPoolExecutorTest, resizeThreadWhileExecutingTestIO) { + resizeThreadWhileExecutingTest(); +} + +TEST(ThreadPoolExecutorTest, resizeThreadWhileExecutingTestCPU) { + resizeThreadWhileExecutingTest(); +} diff --git a/folly/executors/test/ThreadedExecutorTest.cpp b/folly/executors/test/ThreadedExecutorTest.cpp new file mode 100644 index 00000000..7bdf4c0e --- /dev/null +++ b/folly/executors/test/ThreadedExecutorTest.cpp @@ -0,0 +1,109 @@ +/* + * Copyright 2017 Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include + +#include + +#include +#include +#include + +namespace { + +class ThreadedExecutorTest : public testing::Test {}; +} + +TEST_F(ThreadedExecutorTest, example) { + folly::ThreadedExecutor x; + auto ret = folly::via(&x) + .then([&] { return 42; }) + .then([&](int n) { return folly::to(n); }) + .get(); + + EXPECT_EQ("42", ret); +} + +TEST_F(ThreadedExecutorTest, dtor_waits) { + constexpr auto kDelay = std::chrono::milliseconds(100); + auto x = std::make_unique(); + auto fut = folly::via(&*x, [&] { /* sleep override */ + std::this_thread::sleep_for(kDelay); + }); + x = nullptr; + + EXPECT_TRUE(fut.isReady()); +} + +TEST_F(ThreadedExecutorTest, many) { + constexpr auto kNumTasks = 1024; + folly::ThreadedExecutor x; + auto rets = + folly::collect( + folly::gen::range(0, kNumTasks) | + folly::gen::map([&](size_t i) { + return folly::via(&x).then([=] { return i; }).then([](size_t k) { + return folly::to(k); + }); + }) | + folly::gen::as()) + .get(); + + EXPECT_EQ("42", rets[42]); +} + +TEST_F(ThreadedExecutorTest, many_sleeping_constant_time) { + constexpr auto kNumTasks = 256; + constexpr auto kDelay = std::chrono::milliseconds(100); + folly::ThreadedExecutor x; + auto rets = + folly::collect( + folly::gen::range(0, kNumTasks) | + folly::gen::map([&](size_t i) { + return folly::via(&x) + .then([=] { + /* sleep override */ std::this_thread::sleep_for(kDelay); + }) + .then([=] { return i; }) + .then([](size_t k) { return folly::to(k); }); + }) | + folly::gen::as()) + .get(); + + EXPECT_EQ("42", rets[42]); +} + +TEST_F(ThreadedExecutorTest, many_sleeping_decreasing_time) { + constexpr auto kNumTasks = 256; + constexpr auto kDelay = std::chrono::milliseconds(100); + folly::ThreadedExecutor x; + auto rets = + folly::collect( + folly::gen::range(0, kNumTasks) | + folly::gen::map([&](size_t i) { + return folly::via(&x) + .then([=] { + auto delay = kDelay * (kNumTasks - i) / kNumTasks; + /* sleep override */ std::this_thread::sleep_for(delay); + }) + .then([=] { return i; }) + .then([](size_t k) { return folly::to(k); }); + }) | + folly::gen::as()) + .get(); + + EXPECT_EQ("42", rets[42]); +} diff --git a/folly/executors/test/UnboundedBlockingQueueTest.cpp b/folly/executors/test/UnboundedBlockingQueueTest.cpp new file mode 100644 index 00000000..203c1eb5 --- /dev/null +++ b/folly/executors/test/UnboundedBlockingQueueTest.cpp @@ -0,0 +1,51 @@ +/* + * Copyright 2017-present Facebook, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +#include +#include +#include +#include + +using namespace folly; + +TEST(UnboundedQueuee, push_pop) { + UnboundedBlockingQueue q; + q.add(42); + EXPECT_EQ(42, q.take()); +} +TEST(UnboundedBlockingQueue, size) { + UnboundedBlockingQueue q; + EXPECT_EQ(0, q.size()); + q.add(42); + EXPECT_EQ(1, q.size()); + q.take(); + EXPECT_EQ(0, q.size()); +} + +TEST(UnboundedBlockingQueue, concurrent_push_pop) { + UnboundedBlockingQueue q; + Baton<> b1, b2; + std::thread t([&] { + b1.post(); + EXPECT_EQ(42, q.take()); + EXPECT_EQ(0, q.size()); + b2.post(); + }); + b1.wait(); + q.add(42); + b2.wait(); + EXPECT_EQ(0, q.size()); + t.join(); +} diff --git a/folly/futures/README.md b/folly/futures/README.md index ec50fa8d..d0e2693d 100644 --- a/folly/futures/README.md +++ b/folly/futures/README.md @@ -722,9 +722,9 @@ Although inspired by the C++11 std::future interface, it is not a drop-in replac

via() wouldn't be of much use without practical implementations around. We have a handful, and here's a (possibly incomplete) list.

    -
  • ThreadPoolExecutor is an abstract thread pool implementation that supports resizing, custom thread factories, pool and per-task stats, NUMA awareness, user-defined task expiration, and Codel task expiration. It and its subclasses are under active development. It currently has two implementations:
      -
    • CPUThreadPoolExecutor is a general purpose thread pool. In addition to the above features, it also supports task priorities.
    • -
    • IOThreadPoolExecutor is similar to CPUThreadPoolExecutor, but each thread spins on an EventBase (accessible to callbacks via EventBaseManager)
    • +
    • ThreadPoolExecutor is an abstract thread pool implementation that supports resizing, custom thread factories, pool and per-task stats, NUMA awareness, user-defined task expiration, and Codel task expiration. It and its subclasses are under active development. It currently has two implementations:
    • folly's EventBase is an Executor and executes work as a callback in the event loop
    • ManualExecutor only executes work when manually cranked. This is useful for testing.
    • @@ -732,7 +732,7 @@ Although inspired by the C++11 std::future interface, it is not a drop-in replac
    • QueuedImmediateExecutor is similar to InlineExecutor, but work added during callback execution will be queued instead of immediately executed
    • ScheduledExecutor is a subinterface of Executor that supports scheduled (i.e. delayed) execution. There aren't many implementations yet, see T5924392
    • Thrift's ThreadManager is an Executor but we aim to deprecate it in favor of the aforementioned CPUThreadPoolExecutor
    • -
    • FutureExecutor wraps another Executor and provides Future<T> addFuture(F func) which returns a Future representing the result of func. This is equivalent to futures::async(executor, func) and the latter should probably be preferred.
    • +
    • FutureExecutor wraps another Executor and provides Future<T> addFuture(F func) which returns a Future representing the result of func. This is equivalent to futures::async(executor, func) and the latter should probably be preferred.

    Timeouts and related features

    Futures provide a number of timing-related features. Here's an overview.

    Timing implementation #

    Timing resolution #

    @@ -1179,4 +1179,4 @@ The three laws refer to a different formulation of the axioms, in terms of the K

    The tradeoff is memory. Each continuation has a stack, and that stack is usually fixed-size and has to be big enough to support whatever ordinary computation you might want to do on it. So each living continuation requires a relatively large amount of memory. If you know the number of continuations will be small, this might be a good fit. In particular, it might be faster, the code might read cleaner, and debugging stack traces might be much easier.

    -

    Futures takes the middle road between callback hell and continuations, one which has been trodden and proved useful in other languages. It doesn't claim to be the best model for all situations. Use your tools wisely.

    \ No newline at end of file +

    Futures takes the middle road between callback hell and continuations, one which has been trodden and proved useful in other languages. It doesn't claim to be the best model for all situations. Use your tools wisely.