2018-09-20 04:40:58 +02:00
|
|
|
// OpenVPN -- An application to securely tunnel IP networks
|
|
|
|
// over a single port, with support for SSL/TLS-based
|
|
|
|
// session authentication and key exchange,
|
|
|
|
// packet encryption, packet authentication, and
|
|
|
|
// packet compression.
|
|
|
|
//
|
2022-09-29 11:41:13 +02:00
|
|
|
// Copyright (C) 2012-2022 OpenVPN Inc.
|
2018-09-20 04:40:58 +02:00
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License Version 3
|
|
|
|
// as published by the Free Software Foundation.
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with this program in the COPYING file.
|
|
|
|
// If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2019-08-29 19:56:00 +02:00
|
|
|
#include <openvpn/common/rc.hpp>
|
2018-09-20 04:40:58 +02:00
|
|
|
#include <openvpn/time/asiotimer.hpp>
|
|
|
|
|
2019-08-29 19:56:00 +02:00
|
|
|
// AsioTimerSafe is like AsioTimer but with strict cancellation
|
|
|
|
// semantics that guarantees that a handler will never be called
|
|
|
|
// with a non-error status after the timer is cancelled.
|
2018-09-20 04:40:58 +02:00
|
|
|
|
|
|
|
namespace openvpn {
|
|
|
|
class AsioTimerSafe
|
|
|
|
{
|
|
|
|
public:
|
2022-03-31 20:22:23 +02:00
|
|
|
typedef std::unique_ptr<AsioTimerSafe> UPtr;
|
|
|
|
|
2018-09-20 04:40:58 +02:00
|
|
|
AsioTimerSafe(openvpn_io::io_context &io_context)
|
2019-08-29 19:56:00 +02:00
|
|
|
: timer_(io_context),
|
|
|
|
epoch_(new Epoch)
|
2018-09-20 04:40:58 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
std::size_t expires_at(const Time &t)
|
|
|
|
{
|
2019-08-29 19:56:00 +02:00
|
|
|
inc_epoch();
|
2018-09-20 04:40:58 +02:00
|
|
|
return timer_.expires_at(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::size_t expires_after(const Time::Duration &d)
|
|
|
|
{
|
2019-08-29 19:56:00 +02:00
|
|
|
inc_epoch();
|
2018-09-20 04:40:58 +02:00
|
|
|
return timer_.expires_after(d);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::size_t cancel()
|
|
|
|
{
|
2019-08-29 19:56:00 +02:00
|
|
|
inc_epoch();
|
2018-09-20 04:40:58 +02:00
|
|
|
return timer_.cancel();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename F>
|
|
|
|
void async_wait(F &&func)
|
|
|
|
{
|
2019-08-29 19:56:00 +02:00
|
|
|
inc_epoch();
|
|
|
|
timer_.async_wait([func = std::move(func), epoch = epoch(), eptr = epoch_](const openvpn_io::error_code &error)
|
|
|
|
{ func(epoch == eptr->epoch ? error : openvpn_io::error::operation_aborted); });
|
2018-09-20 04:40:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2019-08-29 19:56:00 +02:00
|
|
|
typedef std::size_t epoch_t;
|
|
|
|
|
|
|
|
struct Epoch : public RC<thread_unsafe_refcount>
|
|
|
|
{
|
|
|
|
typedef RCPtr<Epoch> Ptr;
|
|
|
|
epoch_t epoch = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
epoch_t epoch() const
|
|
|
|
{
|
|
|
|
return epoch_->epoch;
|
|
|
|
}
|
|
|
|
|
|
|
|
void inc_epoch()
|
|
|
|
{
|
|
|
|
++epoch_->epoch;
|
|
|
|
}
|
|
|
|
|
2018-09-20 04:40:58 +02:00
|
|
|
AsioTimer timer_;
|
2019-08-29 19:56:00 +02:00
|
|
|
Epoch::Ptr epoch_;
|
2018-09-20 04:40:58 +02:00
|
|
|
};
|
|
|
|
} // namespace openvpn
|