/** * Copyright (C) 2018-present MongoDB, Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the Server Side Public License, version 1, * as published by MongoDB, Inc. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * Server Side Public License for more details. * * You should have received a copy of the Server Side Public License * along with this program. If not, see * . * * As a special exception, the copyright holders give permission to link the * code of portions of this program with the OpenSSL library under certain * conditions as described in each individual source file and distribute * linked combinations including the program with the OpenSSL library. You * must comply with the Server Side Public License in all respects for * all of the code used other than as permitted herein. If you modify file(s) * with this exception, you may extend this exception to your version of the * file(s), but you are not obligated to do so. If you do not wish to do so, * delete this exception statement from your version. If you delete this * exception statement from all source files in the program, then also delete * it in the license file. */ #include "mongo/platform/basic.h" #include #include "mongo/logv2/log.h" #include "mongo/stdx/thread.h" #include "mongo/unittest/unittest.h" #include "mongo/util/concurrency/spin_lock.h" #include "mongo/util/timer.h" #define MONGO_LOGV2_DEFAULT_COMPONENT ::mongo::logv2::LogComponent::kTest namespace { using mongo::SpinLock; using mongo::Timer; namespace stdx = mongo::stdx; class LockTester { public: LockTester(SpinLock* spin, int* counter) : _spin(spin), _counter(counter), _requests(0) {} virtual ~LockTester() { delete _t; } void start(int increments) { _t = new stdx::thread([this, increments] { test(increments); }); } void join() { if (_t) _t->join(); } int requests() const { return _requests; } private: SpinLock* _spin; // not owned here int* _counter; // not owned here int _requests; stdx::thread* _t; void test(int increments) { while (increments-- > 0) { acquireLock(); ++(*_counter); ++_requests; _spin->unlock(); } } LockTester(LockTester&); LockTester& operator=(LockTester&); protected: SpinLock* spin() const { return _spin; } virtual void acquireLock() { spin()->lock(); } }; class TryLockTester final : public LockTester { public: TryLockTester(SpinLock* spin, int* counter) : LockTester(spin, counter) {} protected: void acquireLock() override { while (!spin()->try_lock()) { } } }; class SpinLockTester { public: template void run(std::string testName) { SpinLock spin; int counter = 0; std::vector testers(_threads); Timer timer; for (int i = 0; i < _threads; i++) { testers[i] = new T(&spin, &counter); } for (int i = 0; i < _threads; i++) { testers[i]->start(_incs); } for (int i = 0; i < _threads; i++) { testers[i]->join(); ASSERT_EQUALS(testers[i]->requests(), _incs); delete testers[i]; } int ms = timer.millis(); using namespace mongo::literals; LOGV2(24149, "spinlock {testName} time: {ms}", "testName"_attr = testName, "ms"_attr = ms); ASSERT_EQUALS(counter, _threads * _incs); } private: const int _threads = 64; const int _incs = 50000; }; TEST(Concurrency, ConcurrentIncs) { SpinLockTester tester; tester.run("ConcurrentIncs"); } TEST(Concurrency, ConcurrentIncsWithTryLock) { SpinLockTester tester; tester.run("ConcurrentIncsWithTryLock"); } } // namespace