blob: a5e1b69ea29020a8c98b33f5b6922483c7493c5f [file] [log] [blame]
Avi Drissmane4622aa2022-09-08 20:36:061// Copyright 2019 The Chromium Authors
Etienne Pierre-doray9e8b40f2019-10-09 16:00:242// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "base/task/post_job.h"
6
Arthur Sonzogni0844a992024-12-12 11:36:207#include <array>
Etienne Pierre-doray9e8b40f2019-10-09 16:00:248#include <atomic>
Etienne Pierre-doray2a67d292020-09-03 05:18:529#include <iterator>
10#include <numeric>
Etienne Pierre-doray9e8b40f2019-10-09 16:00:2411
Etienne Pierre-dorayd88e66be2022-08-03 01:37:4612#include "base/barrier_closure.h"
Guido Urdanetaef4e91942020-11-09 15:06:2413#include "base/test/bind.h"
Etienne Pierre-doray9e8b40f2019-10-09 16:00:2414#include "base/test/gtest_util.h"
15#include "base/test/task_environment.h"
Etienne Pierre-doraycfe0c452022-06-01 16:55:2416#include "base/test/test_timeouts.h"
Etienne Pierre-dorayd88e66be2022-08-03 01:37:4617#include "base/test/test_waitable_event.h"
Etienne Pierre-doraycfe0c452022-06-01 16:55:2418#include "base/threading/platform_thread.h"
Etienne Pierre-doray9e8b40f2019-10-09 16:00:2419#include "testing/gmock/include/gmock/gmock.h"
20#include "testing/gtest/include/gtest/gtest.h"
21
22namespace base {
23
24TEST(PostJobTest, PostJobSimple) {
25 test::TaskEnvironment task_environment;
26 std::atomic_size_t num_tasks_to_run(4);
Etienne Pierre-doray35b691e2020-01-31 14:36:1027 auto handle = PostJob(
Gabriel Charette7d5eac02020-01-28 13:37:1328 FROM_HERE, {},
Etienne Pierre-doray35b691e2020-01-31 14:36:1029 BindLambdaForTesting([&](JobDelegate* delegate) { --num_tasks_to_run; }),
Etienne Pierre-doray2f52b3512020-08-12 19:04:2930 BindLambdaForTesting(
31 [&](size_t /*worker_count*/) -> size_t { return num_tasks_to_run; }));
Etienne Pierre-doray9e8b40f2019-10-09 16:00:2432 handle.Join();
Etienne Pierre-doraycfe0c452022-06-01 16:55:2433 EXPECT_EQ(num_tasks_to_run, 0U);
34}
35
36TEST(PostJobTest, CreateJobSimple) {
37 test::TaskEnvironment task_environment;
38 std::atomic_size_t num_tasks_to_run(4);
Etienne Pierre-dorayd88e66be2022-08-03 01:37:4639 TestWaitableEvent threads_continue;
40 RepeatingClosure barrier = BarrierClosure(
Sorin Jianud01ad7f2024-10-09 03:45:4541 num_tasks_to_run,
42 BindLambdaForTesting([&threads_continue] { threads_continue.Signal(); }));
Etienne Pierre-doraycfe0c452022-06-01 16:55:2443 bool job_started = false;
44 auto handle =
45 CreateJob(FROM_HERE, {}, BindLambdaForTesting([&](JobDelegate* delegate) {
46 EXPECT_TRUE(job_started);
Etienne Pierre-dorayd88e66be2022-08-03 01:37:4647 barrier.Run();
48 threads_continue.Wait();
Etienne Pierre-doraycfe0c452022-06-01 16:55:2449 --num_tasks_to_run;
50 }),
51 BindLambdaForTesting([&](size_t /*worker_count*/) -> size_t {
52 EXPECT_TRUE(job_started);
53 return num_tasks_to_run;
54 }));
55
56 PlatformThread::Sleep(TestTimeouts::tiny_timeout());
57 EXPECT_EQ(num_tasks_to_run, 4U);
58 job_started = true;
59 handle.Join();
60 EXPECT_EQ(num_tasks_to_run, 0U);
Etienne Pierre-doray9e8b40f2019-10-09 16:00:2461}
62
Etienne Pierre-doray2a67d292020-09-03 05:18:5263// Verify that concurrent accesses with task_id as the only form of
64// synchronisation doesn't trigger a race.
65TEST(PostJobTest, TaskIds) {
66 static constexpr size_t kNumConcurrentThreads = 2;
67 static constexpr size_t kNumTasksToRun = 1000;
68 base::test::TaskEnvironment task_environment;
69
Arthur Sonzogni0844a992024-12-12 11:36:2070 std::array<size_t, kNumConcurrentThreads> concurrent_array = {};
Etienne Pierre-doray2a67d292020-09-03 05:18:5271 std::atomic_size_t remaining_tasks{kNumTasksToRun};
72 base::JobHandle handle = base::PostJob(
Javier Flores0d0d0a62021-03-24 12:07:0673 FROM_HERE, {}, BindLambdaForTesting([&](base::JobDelegate* job) {
Etienne Pierre-doray2a67d292020-09-03 05:18:5274 uint8_t id = job->GetTaskId();
75 size_t& slot = concurrent_array[id];
76 slot++;
77 --remaining_tasks;
78 }),
79 BindLambdaForTesting([&remaining_tasks](size_t) {
80 return std::min(remaining_tasks.load(), kNumConcurrentThreads);
81 }));
82 handle.Join();
83 EXPECT_EQ(kNumTasksToRun, std::accumulate(std::begin(concurrent_array),
84 std::end(concurrent_array), 0U));
85}
86
Gabriel Charette7d5eac02020-01-28 13:37:1387} // namespace base