f1589f6ef1d7a52dce8530c493006459c9705927
[WebKit-https.git] / Source / WTF / wtf / cocoa / WorkQueueCocoa.cpp
1 /*
2  * Copyright (C) 2010 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27 #include "WorkQueue.h"
28 #include "BlockPtr.h"
29 #include "Ref.h"
30
31 namespace WTF {
32
33 void WorkQueue::dispatch(Function<void()>&& function)
34 {
35     dispatch_async(m_dispatchQueue, BlockPtr<void()>::fromCallable([protectedThis = makeRef(*this), function = WTFMove(function)] {
36         function();
37     }).get());
38 }
39
40 void WorkQueue::dispatchAfter(Seconds duration, Function<void()>&& function)
41 {
42     dispatch_after(dispatch_time(DISPATCH_TIME_NOW, duration.nanosecondsAs<int64_t>()), m_dispatchQueue, BlockPtr<void()>::fromCallable([protectedThis = makeRef(*this), function = WTFMove(function)] {
43         function();
44     }).get());
45 }
46
47 #if HAVE(QOS_CLASSES)
48 static dispatch_qos_class_t dispatchQOSClass(WorkQueue::QOS qos)
49 {
50     switch (qos) {
51     case WorkQueue::QOS::UserInteractive:
52         return Thread::adjustedQOSClass(QOS_CLASS_USER_INTERACTIVE);
53     case WorkQueue::QOS::UserInitiated:
54         return Thread::adjustedQOSClass(QOS_CLASS_USER_INITIATED);
55     case WorkQueue::QOS::Default:
56         return Thread::adjustedQOSClass(QOS_CLASS_DEFAULT);
57     case WorkQueue::QOS::Utility:
58         return Thread::adjustedQOSClass(QOS_CLASS_UTILITY);
59     case WorkQueue::QOS::Background:
60         return Thread::adjustedQOSClass(QOS_CLASS_BACKGROUND);
61     }
62 }
63 #else
64 static dispatch_queue_t targetQueueForQOSClass(WorkQueue::QOS qos)
65 {
66     switch (qos) {
67     case WorkQueue::QOS::UserInteractive:
68     case WorkQueue::QOS::UserInitiated:
69         return dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0);
70     case WorkQueue::QOS::Utility:
71         return dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0);
72     case WorkQueue::QOS::Background:
73         return dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0);
74     case WorkQueue::QOS::Default:
75         ASSERT_NOT_REACHED();
76         return nullptr;
77     }
78     ASSERT_NOT_REACHED();
79 }
80 #endif
81
82 void WorkQueue::platformInitialize(const char* name, Type type, QOS qos)
83 {
84     dispatch_queue_attr_t attr = type == Type::Concurrent ? DISPATCH_QUEUE_CONCURRENT : DISPATCH_QUEUE_SERIAL;
85 #if HAVE(QOS_CLASSES)
86     attr = dispatch_queue_attr_make_with_qos_class(attr, dispatchQOSClass(qos), 0);
87 #else
88     UNUSED_PARAM(qos);
89 #endif
90     m_dispatchQueue = dispatch_queue_create(name, attr);
91 #if !HAVE(QOS_CLASSES)
92     if (qos != WorkQueue::QOS::Default)
93         dispatch_set_target_queue(m_dispatchQueue, targetQueueForQOSClass(qos));
94 #endif
95     dispatch_set_context(m_dispatchQueue, this);
96 }
97
98 void WorkQueue::platformInvalidate()
99 {
100     dispatch_release(m_dispatchQueue);
101 }
102
103 void WorkQueue::concurrentApply(size_t iterations, WTF::Function<void(size_t index)>&& function)
104 {
105     dispatch_apply(iterations, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), BlockPtr<void(size_t index)>::fromCallable([function = WTFMove(function)](size_t index) {
106         function(index);
107     }).get());
108 }
109
110 }