blob: 143b1f69d9fa7b1c2ed46e1d51a5268fc6978fca [file] [log] [blame]
// Copyright 2019 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/task/post_job.h"
#include <atomic>
#include <iterator>
#include <numeric>
#include "base/barrier_closure.h"
#include "base/test/bind.h"
#include "base/test/gtest_util.h"
#include "base/test/task_environment.h"
#include "base/test/test_timeouts.h"
#include "base/test/test_waitable_event.h"
#include "base/threading/platform_thread.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace base {
TEST(PostJobTest, PostJobSimple) {
test::TaskEnvironment task_environment;
std::atomic_size_t num_tasks_to_run(4);
auto handle = PostJob(
FROM_HERE, {},
BindLambdaForTesting([&](JobDelegate* delegate) { --num_tasks_to_run; }),
BindLambdaForTesting(
[&](size_t /*worker_count*/) -> size_t { return num_tasks_to_run; }));
handle.Join();
EXPECT_EQ(num_tasks_to_run, 0U);
}
TEST(PostJobTest, CreateJobSimple) {
test::TaskEnvironment task_environment;
std::atomic_size_t num_tasks_to_run(4);
TestWaitableEvent threads_continue;
RepeatingClosure barrier = BarrierClosure(
num_tasks_to_run, BindLambdaForTesting([&threads_continue]() {
threads_continue.Signal();
}));
bool job_started = false;
auto handle =
CreateJob(FROM_HERE, {}, BindLambdaForTesting([&](JobDelegate* delegate) {
EXPECT_TRUE(job_started);
barrier.Run();
threads_continue.Wait();
--num_tasks_to_run;
}),
BindLambdaForTesting([&](size_t /*worker_count*/) -> size_t {
EXPECT_TRUE(job_started);
return num_tasks_to_run;
}));
PlatformThread::Sleep(TestTimeouts::tiny_timeout());
EXPECT_EQ(num_tasks_to_run, 4U);
job_started = true;
handle.Join();
EXPECT_EQ(num_tasks_to_run, 0U);
}
// Verify that concurrent accesses with task_id as the only form of
// synchronisation doesn't trigger a race.
TEST(PostJobTest, TaskIds) {
static constexpr size_t kNumConcurrentThreads = 2;
static constexpr size_t kNumTasksToRun = 1000;
base::test::TaskEnvironment task_environment;
size_t concurrent_array[kNumConcurrentThreads] = {0};
std::atomic_size_t remaining_tasks{kNumTasksToRun};
base::JobHandle handle = base::PostJob(
FROM_HERE, {}, BindLambdaForTesting([&](base::JobDelegate* job) {
uint8_t id = job->GetTaskId();
size_t& slot = concurrent_array[id];
slot++;
--remaining_tasks;
}),
BindLambdaForTesting([&remaining_tasks](size_t) {
return std::min(remaining_tasks.load(), kNumConcurrentThreads);
}));
handle.Join();
EXPECT_EQ(kNumTasksToRun, std::accumulate(std::begin(concurrent_array),
std::end(concurrent_array), 0U));
}
} // namespace base