blob: 36d63d7f906327c4c2718f39fe6781f5b3e990ec [file] [log] [blame]
/*
* Copyright (c) 2020, The OpenThread Authors.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of the copyright holder nor the
* names of its contributors may be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
/**
* @file
* This file implements Dataset Updater.
*
*/
#include "dataset_updater.hpp"
#if (OPENTHREAD_CONFIG_DATASET_UPDATER_ENABLE || OPENTHREAD_CONFIG_CHANNEL_MANAGER_ENABLE) && OPENTHREAD_FTD
#include "common/code_utils.hpp"
#include "common/instance.hpp"
#include "common/locator_getters.hpp"
#include "common/log.hpp"
#include "common/random.hpp"
#include "meshcop/timestamp.hpp"
namespace ot {
namespace MeshCoP {
DatasetUpdater::DatasetUpdater(Instance &aInstance)
: InstanceLocator(aInstance)
, mCallback(nullptr)
, mCallbackContext(nullptr)
, mTimer(aInstance, DatasetUpdater::HandleTimer)
, mDataset(nullptr)
{
}
Error DatasetUpdater::RequestUpdate(const Dataset::Info &aDataset, Callback aCallback, void *aContext)
{
Error error = kErrorNone;
Message *message = nullptr;
VerifyOrExit(!Get<Mle::Mle>().IsDisabled(), error = kErrorInvalidState);
VerifyOrExit(mDataset == nullptr, error = kErrorBusy);
VerifyOrExit(!aDataset.IsActiveTimestampPresent() && !aDataset.IsPendingTimestampPresent(),
error = kErrorInvalidArgs);
message = Get<MessagePool>().Allocate(Message::kTypeOther);
VerifyOrExit(message != nullptr, error = kErrorNoBufs);
SuccessOrExit(error = message->Append(aDataset));
mCallback = aCallback;
mCallbackContext = aContext;
mDataset = message;
mTimer.Start(1);
exit:
FreeMessageOnError(message, error);
return error;
}
void DatasetUpdater::CancelUpdate(void)
{
VerifyOrExit(mDataset != nullptr);
FreeMessage(mDataset);
mDataset = nullptr;
mTimer.Stop();
exit:
return;
}
void DatasetUpdater::HandleTimer(Timer &aTimer)
{
aTimer.Get<DatasetUpdater>().HandleTimer();
}
void DatasetUpdater::HandleTimer(void)
{
PreparePendingDataset();
}
void DatasetUpdater::PreparePendingDataset(void)
{
Dataset dataset;
Dataset::Info requestedDataset;
Error error;
VerifyOrExit(!Get<Mle::Mle>().IsDisabled(), error = kErrorInvalidState);
IgnoreError(mDataset->Read(0, requestedDataset));
error = Get<ActiveDatasetManager>().Read(dataset);
if (error != kErrorNone)
{
// If there is no valid Active Dataset but MLE is not disabled,
// set the timer to try again after the retry interval. This
// handles the situation where a dataset update request comes
// right after the network is formed but before the active
// dataset is created.
mTimer.Start(kRetryInterval);
ExitNow(error = kErrorNone);
}
IgnoreError(dataset.SetFrom(requestedDataset));
if (!requestedDataset.IsDelayPresent())
{
uint32_t delay = kDefaultDelay;
SuccessOrExit(error = dataset.SetTlv(Tlv::kDelayTimer, delay));
}
{
Timestamp timestamp;
if (Get<PendingDatasetManager>().GetTimestamp() != nullptr)
{
timestamp = *Get<PendingDatasetManager>().GetTimestamp();
}
timestamp.AdvanceRandomTicks();
dataset.SetTimestamp(Dataset::kPending, timestamp);
}
{
ActiveTimestampTlv *tlv = dataset.GetTlv<ActiveTimestampTlv>();
tlv->GetTimestamp().AdvanceRandomTicks();
}
SuccessOrExit(error = Get<PendingDatasetManager>().Save(dataset));
exit:
if (error != kErrorNone)
{
Finish(error);
}
}
void DatasetUpdater::Finish(Error aError)
{
OT_ASSERT(mDataset != nullptr);
FreeMessage(mDataset);
mDataset = nullptr;
if (mCallback != nullptr)
{
mCallback(aError, mCallbackContext);
}
}
void DatasetUpdater::HandleNotifierEvents(Events aEvents)
{
Dataset::Info requestedDataset;
Dataset::Info dataset;
VerifyOrExit(mDataset != nullptr);
VerifyOrExit(aEvents.ContainsAny(kEventActiveDatasetChanged | kEventPendingDatasetChanged));
IgnoreError(mDataset->Read(0, requestedDataset));
if (aEvents.Contains(kEventActiveDatasetChanged) && Get<ActiveDatasetManager>().Read(dataset) == kErrorNone)
{
if (requestedDataset.IsSubsetOf(dataset))
{
Finish(kErrorNone);
}
else
{
Timestamp requestedDatasetTimestamp;
Timestamp activeDatasetTimestamp;
requestedDataset.GetActiveTimestamp(requestedDatasetTimestamp);
dataset.GetActiveTimestamp(activeDatasetTimestamp);
if (Timestamp::Compare(requestedDatasetTimestamp, activeDatasetTimestamp) <= 0)
{
Finish(kErrorAlready);
}
}
}
if (aEvents.Contains(kEventPendingDatasetChanged) && Get<PendingDatasetManager>().Read(dataset) == kErrorNone)
{
if (!requestedDataset.IsSubsetOf(dataset))
{
Finish(kErrorAlready);
}
}
exit:
return;
}
} // namespace MeshCoP
} // namespace ot
#endif // #if (OPENTHREAD_CONFIG_DATASET_UPDATER_ENABLE || OPENTHREAD_CONFIG_CHANNEL_MANAGER_ENABLE) && OPENTHREAD_FTD