You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
movesense_exercise/MovesenseExerciseService.cpp

289 lines
8.6 KiB

#include "MovesenseExerciseService.h"
#include "app-resources/resources.h"
#include "common/core/debug.h"
#include "meas_acc/resources.h"
#include "whiteboard/builtinTypes/UnknownStructure.h"
#include "ui_ind/resources.h"
#include <float.h>
#include <math.h>
const char* const MovesenseExerciseService::LAUNCHABLE_NAME = "MExercise";
#define SAMPLE_RATE 13
const size_t BLINK_PERIOD_MS = 1000;
static const whiteboard::ExecutionContextId sExecutionContextId =
WB_RES::LOCAL::EXERCISE_SUMVECTOR_DENES::EXECUTION_CONTEXT;
static const whiteboard::LocalResourceId sProviderResources[] = {
WB_RES::LOCAL::EXERCISE_SUMVECTOR_DENES::LID,
};
namespace {
whiteboard::FloatVector3D SumVectorAverage(const std::vector<whiteboard::FloatVector3D>& v)
{
whiteboard::FloatVector3D sum;
for (size_t i = 0; i < v.size(); i++)
{
whiteboard::FloatVector3D fv = v[i];
sum += fv;
}
return sum / (float)v.size();
}
}
MovesenseExerciseService::MovesenseExerciseService()
: ResourceClient(WBDEBUG_NAME(__FUNCTION__), sExecutionContextId),
ResourceProvider(WBDEBUG_NAME(__FUNCTION__), sExecutionContextId),
LaunchableModule(LAUNCHABLE_NAME, sExecutionContextId),
isRunning(false),
mLastTimeStamp(0),
mTimer(whiteboard::ID_INVALID_TIMER),
mAccelerationData(SAMPLE_RATE)
{
}
MovesenseExerciseService::~MovesenseExerciseService()
{
}
bool MovesenseExerciseService::initModule()
{
if (registerProviderResources(sProviderResources) != whiteboard::HTTP_CODE_OK)
{
return false;
}
mModuleState = WB_RES::ModuleStateValues::INITIALIZED;
return true;
}
void MovesenseExerciseService::deinitModule()
{
unregisterProviderResources(sProviderResources);
mModuleState = WB_RES::ModuleStateValues::UNINITIALIZED;
}
/** @see whiteboard::ILaunchableModule::startModule */
bool MovesenseExerciseService::startModule()
{
mModuleState = WB_RES::ModuleStateValues::STARTED;
return true;
}
void MovesenseExerciseService::stopModule()
{
mModuleState = WB_RES::ModuleStateValues::STOPPED;
}
void MovesenseExerciseService::onUnsubscribeResult(whiteboard::RequestId requestId,
whiteboard::ResourceId resourceId,
whiteboard::Result resultCode,
const whiteboard::Value& rResultData)
{
DEBUGLOG("MovesenseExerciseService::onUnsubscribeResult() called.");
}
void MovesenseExerciseService::onSubscribeResult(whiteboard::RequestId requestId,
whiteboard::ResourceId resourceId,
whiteboard::Result resultCode,
const whiteboard::Value& rResultData)
{
DEBUGLOG("MovesenseExerciseService::onSubscribeResult() called. resourceId: %u, result: %d", resourceId.localResourceId, (uint32_t)resultCode);
whiteboard::Request relatedIncomingRequest;
bool relatedRequestFound = mOngoingRequests.get(requestId, relatedIncomingRequest);
if (relatedRequestFound)
{
returnResult(relatedIncomingRequest, wb::HTTP_CODE_OK);
}
}
whiteboard::Result MovesenseExerciseService::startRunning(whiteboard::RequestId& remoteRequestId)
{
if (isRunning)
{
return whiteboard::HTTP_CODE_OK;
}
DEBUGLOG("MovesenseExerciseService::startRunning()");
wb::Result result = getResource("Meas/Acc/13", mMeasAccResourceId);
if (!wb::RETURN_OKC(result))
{
return result;
}
result = asyncSubscribe(mMeasAccResourceId, AsyncRequestOptions(&remoteRequestId, 0, true));
if (!wb::RETURN_OKC(result))
{
DEBUGLOG("asyncSubscribe threw error: %u", result);
return whiteboard::HTTP_CODE_BAD_REQUEST;
}
// Start LED timer. true = trigger repeatedly
mTimer = ResourceClient::startTimer(BLINK_PERIOD_MS, true);
isRunning = true;
return whiteboard::HTTP_CODE_OK;
}
whiteboard::Result MovesenseExerciseService::stopRunning()
{
if (!isRunning)
{
return whiteboard::HTTP_CODE_OK;
}
if (isResourceSubscribed(WB_RES::LOCAL::EXERCISE_SUMVECTOR_DENES::ID) == wb::HTTP_CODE_OK)
{
DEBUGLOG("MovesenseExerciseService::stopRunning() skipping. Subscribers still exist.");
return whiteboard::HTTP_CODE_OK;
}
DEBUGLOG("MovesenseExerciseService::stopRunning()");
// Unsubscribe the LinearAcceleration resource, when unsubscribe is done, we get callback
wb::Result result = asyncUnsubscribe(mMeasAccResourceId, NULL);
if (!wb::RETURN_OKC(result))
{
DEBUGLOG("asyncUnsubscribe threw error: %u", result);
}
// Stop LED timer
ResourceClient::stopTimer(mTimer);
mTimer = whiteboard::ID_INVALID_TIMER;
isRunning = false;
releaseResource(mMeasAccResourceId);
return whiteboard::HTTP_CODE_OK;
}
// This callback is called when the resource we have subscribed notifies us
void MovesenseExerciseService::onNotify(whiteboard::ResourceId resourceId, const whiteboard::Value& value,
const whiteboard::ParameterList& parameters)
{
DEBUGLOG("onNotify() called.");
// Confirm that it is the correct resource
switch (resourceId.localResourceId)
{
case WB_RES::LOCAL::MEAS_ACC_SAMPLERATE::LID:
{
const WB_RES::AccData& linearAccelerationValue =
value.convertTo<const WB_RES::AccData&>();
if (linearAccelerationValue.arrayAcc.size() <= 0)
{
// No value, do nothing...
return;
}
const whiteboard::Array<whiteboard::FloatVector3D>& arrayData = linearAccelerationValue.arrayAcc;
mLastTimeStamp = linearAccelerationValue.timestamp;
for (size_t i = 0; i < arrayData.size(); i++)
{
mAccelerationData.push_back(arrayData[i]);
}
}
break;
}
}
void MovesenseExerciseService::onSubscribe(const whiteboard::Request& request,
const whiteboard::ParameterList& parameters)
{
switch (request.getResourceConstId())
{
case WB_RES::LOCAL::EXERCISE_SUMVECTOR_DENES::ID:
{
// Someone subscribed to our service. Start collecting data
// and publish calculated sumvector average (every 1 second)
whiteboard::RequestId remoteRequestId;
whiteboard::Result result = startRunning(remoteRequestId);
if (isRunning)
{
return returnResult(request, whiteboard::HTTP_CODE_OK);
}
if (!whiteboard::RETURN_OK(result))
{
return returnResult(request, result);
}
bool queueResult = mOngoingRequests.put(remoteRequestId, request);
(void)queueResult;
WB_ASSERT(queueResult);
break;
}
default:
ASSERT(0); // Should not happen
}
}
void MovesenseExerciseService::onUnsubscribe(const whiteboard::Request& request,
const whiteboard::ParameterList& parameters)
{
switch (request.getResourceConstId())
{
case WB_RES::LOCAL::EXERCISE_SUMVECTOR_DENES::ID:
stopRunning();
returnResult(request, wb::HTTP_CODE_OK);
break;
default:
returnResult(request, wb::HTTP_CODE_BAD_REQUEST);
ASSERT(0); // Should not happen
}
}
void MovesenseExerciseService::onRemoteWhiteboardDisconnected(whiteboard::WhiteboardId whiteboardId)
{
DEBUGLOG("MovesenseExerciseService::onRemoteWhiteboardDisconnected()");
stopRunning();
}
void MovesenseExerciseService::onClientUnavailable(whiteboard::ClientId clientId)
{
DEBUGLOG("MovesenseExerciseService::onClientUnavailable()");
stopRunning();
}
void MovesenseExerciseService::onTimer(whiteboard::TimerId timerId)
{
DEBUGLOG("MovesenseExerciseService::onTimer()");
if (timerId != mTimer)
{
return;
}
uint16_t indicationType = 2; // SHORT_VISUAL_INDICATION, defined in ui/ind.yaml
// Make PUT request to trigger led blink
asyncPut(WB_RES::LOCAL::UI_IND_VISUAL::ID, AsyncRequestOptions::Empty, indicationType);
// Notify our subscribers
/// @todo The timestamp is the last measurement's timestamp and not the current time.
WB_RES::SampleDataValue sampleDataValue;
sampleDataValue.relativeTime = mLastTimeStamp;
sampleDataValue.value = SumVectorAverage(mAccelerationData);
// Reset members
mLastTimeStamp = 0;
mAccelerationData.clear();
// and update our WB resource. This causes notification to be fired to our subscribers
updateResource(WB_RES::LOCAL::EXERCISE_SUMVECTOR_DENES(),
ResponseOptions::Empty, sampleDataValue);
}