Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Store average accelerometer readings in Object Model #590

Open
wants to merge 6 commits into
base: 3.5-dev
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
59 changes: 48 additions & 11 deletions src/Accelerometers/Accelerometers.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
#include <RTOSIface/RTOSIface.h>
#include <Platform/TaskPriorities.h>
#include <Hardware/Spi/SharedSpiDevice.h>
#include <RRF3Common.h>

#if SUPPORT_CAN_EXPANSION
# include <CanMessageFormats.h>
Expand Down Expand Up @@ -68,6 +69,7 @@ static volatile uint32_t numSamplesRequested;
static uint8_t resolution = DefaultResolution;
static uint8_t orientation = 20; // +Z -> +Z, +X -> +X
static volatile uint8_t axesRequested;
static float lastRunAverages[NumAccelerometerAxes] = {0.0f, 0.0f, 0.0f};
static FileStore* volatile accelerometerFile = nullptr; // this is non-null when the accelerometer is running, null otherwise
static unsigned int numLocalRunsCompleted = 0;
static unsigned int lastRunNumSamplesReceived = 0;
Expand All @@ -80,17 +82,23 @@ static IoPort spiCsPort;
static IoPort irqPort;

// Add a local accelerometer run
static void AddLocalAccelerometerRun(unsigned int numDataPoints) noexcept
static void AddLocalAccelerometerRun(unsigned int numDataPoints, float averages[]) noexcept
{
lastRunNumSamplesReceived = numDataPoints;
++numLocalRunsCompleted;

for(unsigned int axis = 0;axis < NumAccelerometerAxes;++axis)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggest use memcpyf here instead of a loop

{
lastRunAverages[axis] = averages[axis];
}

reprap.BoardsUpdated();
}

static uint8_t TranslateAxes(uint8_t axes) noexcept
{
uint8_t rslt = 0;
for (unsigned int i = 0; i < 3; ++i)
for (unsigned int i = 0; i < NumAccelerometerAxes; ++i)
{
if (axes & (1u << i))
{
Expand All @@ -115,6 +123,7 @@ static uint8_t TranslateAxes(uint8_t axes) noexcept
const uint16_t mask = (1u << resolution) - 1;
const int decimalPlaces = GetDecimalPlaces(resolution);
bool recordFailedStart = false;
float accumulatedSamples[NumAccelerometerAxes] = {0.0f, 0.0f, 0.0f};

if (accelerometer->StartCollecting(TranslateAxes(axesRequested)))
{
Expand All @@ -133,7 +142,7 @@ static uint8_t TranslateAxes(uint8_t axes) noexcept
f->Truncate(); // truncate the file in case we didn't write all the preallocated space
f->Close();
f = nullptr;
AddLocalAccelerometerRun(0);
AddLocalAccelerometerRun(0, {0});
}
else
{
Expand All @@ -158,7 +167,7 @@ static uint8_t TranslateAxes(uint8_t axes) noexcept
String<StringLength50> temp;
temp.printf("%u", samplesWritten);

for (unsigned int axis = 0; axis < 3; ++axis)
for (unsigned int axis = 0; axis < NumAccelerometerAxes; ++axis)
{
if (axesRequested & (1u << axis))
{
Expand All @@ -180,6 +189,9 @@ static uint8_t TranslateAxes(uint8_t axes) noexcept

// Append it to the buffer
temp.catf(",%.*f", decimalPlaces, (double)fVal);

// accumulate the values so they can be averaged later
accumulatedSamples[axis] += fVal;
}
}

Expand Down Expand Up @@ -215,7 +227,15 @@ static uint8_t TranslateAxes(uint8_t axes) noexcept
{
f->Truncate(); // truncate the file in case we didn't write all the preallocated space
f->Close();
AddLocalAccelerometerRun(samplesWritten);

// find the average value for each axis
for(unsigned int axis = 0;axis < NumAccelerometerAxes;++axis)
{
accumulatedSamples[axis] /= float(samplesWritten);
}

AddLocalAccelerometerRun(samplesWritten, accumulatedSamples);

}

accelerometer->StopCollecting();
Expand Down Expand Up @@ -457,12 +477,12 @@ GCodeResult Accelerometers::StartAccelerometer(GCodeBuffer& gb, const StringRef&
# if SUPPORT_CAN_EXPANSION
if (device.IsRemote())
{
reprap.GetExpansion().AddAccelerometerRun(device.boardAddress, 0);
reprap.GetExpansion().AddAccelerometerRun(device.boardAddress, 0, {0});
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Need to pass an array of the correct size here because the compiler doesn't know how many values are expected

}
else
# endif
{
AddLocalAccelerometerRun(0);
AddLocalAccelerometerRun(0, {0});
Copy link
Collaborator

@dc42 dc42 Nov 5, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Need to pass an array of the correct size here because the compiler doesn't know how many values are expected, or add a new function AddFailedLocalAccelerometerRun that doesn't take the averages parameter

}
return GCodeResult::error;
}
Expand Down Expand Up @@ -493,7 +513,7 @@ GCodeResult Accelerometers::StartAccelerometer(GCodeBuffer& gb, const StringRef&
accelerometerFile->Close();
accelerometerFile = nullptr;
MassStorage::Delete(accelerometerFileName.c_str(), false);
reprap.GetExpansion().AddAccelerometerRun(device.boardAddress, 0);
reprap.GetExpansion().AddAccelerometerRun(device.boardAddress, 0, {0});
Copy link
Collaborator

@dc42 dc42 Nov 5, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Need to pass an array of the correct size here because the compiler doesn't know how many values are expected. Alternatively, add another function AddFailedAccelerometerRun that doesn't take the averages as a parameter and instead sets them to zero, since this appears to be done in several places.

}
return rslt;
}
Expand Down Expand Up @@ -532,6 +552,11 @@ bool Accelerometers::HasLocalAccelerometer() noexcept
return accelerometer != nullptr;
}

float Accelerometers::GetLocalAccelerometerLastRunAverage(const int &axis) noexcept
{
return lastRunAverages[axis];
}

unsigned int Accelerometers::GetLocalAccelerometerDataPoints() noexcept
{
return lastRunNumSamplesReceived;
Expand Down Expand Up @@ -572,15 +597,15 @@ void Accelerometers::ProcessReceivedData(CanAddress src, const CanMessageAcceler
f->Truncate(); // truncate the file in case we didn't write all the preallocated space
f->Close();
accelerometerFile = nullptr;
reprap.GetExpansion().AddAccelerometerRun(src, 0);
reprap.GetExpansion().AddAccelerometerRun(src, 0, {0});
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

See previous comment.

}
else if (msg.axes != expectedRemoteAxes || msg.firstSampleNumber != expectedRemoteSampleNumber || src != expectedRemoteBoardAddress)
{
f->Write("Received mismatched data\n");
f->Truncate(); // truncate the file in case we didn't write all the preallocated space
f->Close();
accelerometerFile = nullptr;
reprap.GetExpansion().AddAccelerometerRun(src, 0);
reprap.GetExpansion().AddAccelerometerRun(src, 0, {0});
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

See previous comment.

}
else
{
Expand All @@ -592,6 +617,8 @@ void Accelerometers::ProcessReceivedData(CanAddress src, const CanMessageAcceler
const unsigned int receivedResolution = msg.bitsPerSampleMinusOne + 1;
const uint16_t mask = (1u << receivedResolution) - 1;
const int decimalPlaces = GetDecimalPlaces(receivedResolution);
float accumulatedSamples[NumAccelerometerAxes] = {0.0f, 0.0f, 0.0f};

if (msg.overflowed)
{
++numRemoteOverflows;
Expand Down Expand Up @@ -632,6 +659,9 @@ void Accelerometers::ProcessReceivedData(CanAddress src, const CanMessageAcceler

// Append it to the buffer
temp.catf(",%.*f", decimalPlaces, (double)fVal);

// accumulate the values so they can be averaged later
accumulatedSamples[axis] += fVal;
}

temp.cat('\n');
Expand All @@ -647,7 +677,14 @@ void Accelerometers::ProcessReceivedData(CanAddress src, const CanMessageAcceler
f->Truncate(); // truncate the file in case we didn't write all the preallocated space
f->Close();
accelerometerFile = nullptr;
reprap.GetExpansion().AddAccelerometerRun(src, expectedRemoteSampleNumber);

// find the average value for each axis
for (unsigned int axis = 0; axis < NumAccelerometerAxes; ++axis)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Better to use a range-based for-loop here

{
accumulatedSamples[axis] /= float(msg.numSamples);
}

reprap.GetExpansion().AddAccelerometerRun(src, expectedRemoteSampleNumber, accumulatedSamples);
}
}
}
Expand Down
1 change: 1 addition & 0 deletions src/Accelerometers/Accelerometers.h
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ class CanMessageAccelerometerData;
namespace Accelerometers
{
bool HasLocalAccelerometer() noexcept;
float GetLocalAccelerometerLastRunAverage(const int& axis) noexcept;
unsigned int GetLocalAccelerometerRuns() noexcept;
unsigned int GetLocalAccelerometerDataPoints() noexcept;
GCodeResult ConfigureAccelerometer(GCodeBuffer& gb, const StringRef& reply) THROWS(GCodeException);
Expand Down
23 changes: 21 additions & 2 deletions src/CAN/ExpansionManager.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,18 @@
#define OBJECT_MODEL_FUNC(...) OBJECT_MODEL_FUNC_BODY(ExpansionManager, __VA_ARGS__)
#define OBJECT_MODEL_FUNC_IF(...) OBJECT_MODEL_FUNC_IF_BODY(ExpansionManager, __VA_ARGS__)

constexpr ObjectModelArrayTableEntry ExpansionManager::objectModelArrayTable[] =
{
{
nullptr, // no lock needed
[] (const ObjectModel *self, const ObjectExplorationContext& context) noexcept -> size_t { return NumAccelerometerAxes; },
[] (const ObjectModel *self, ObjectExplorationContext& context) noexcept -> ExpressionValue
{ return ExpressionValue(((const ExpansionManager*)self)->FindIndexedBoard(context.GetIndex(1)).accelerometerLastRunAverages[context.GetLastIndex()]); }
}
};

DEFINE_GET_OBJECT_MODEL_ARRAY_TABLE(ExpansionManager)

constexpr ObjectModelTableEntry ExpansionManager::objectModelTable[] =
{
// 0. boards[] members
Expand Down Expand Up @@ -59,6 +71,7 @@ constexpr ObjectModelTableEntry ExpansionManager::objectModelTable[] =
{ "min", OBJECT_MODEL_FUNC(self->FindIndexedBoard(context.GetLastIndex()).v12.minimum, 1), ObjectModelEntryFlags::none },

// 4. accelerometer members
{ "lastRunAverages", OBJECT_MODEL_FUNC_ARRAY(0), ObjectModelEntryFlags::none },
{ "points", OBJECT_MODEL_FUNC((int32_t)self->FindIndexedBoard(context.GetLastIndex()).accelerometerLastRunDataPoints), ObjectModelEntryFlags::none },
{ "runs", OBJECT_MODEL_FUNC((int32_t)self->FindIndexedBoard(context.GetLastIndex()).accelerometerRuns), ObjectModelEntryFlags::none },

Expand All @@ -74,7 +87,7 @@ constexpr uint8_t ExpansionManager::objectModelTableDescriptor[] =
3, // section 1: mcuTemp
3, // section 2: vIn
3, // section 3: v12
2, // section 4: accelerometer
3, // section 4: accelerometer
2 // section 5: closed loop
};

Expand Down Expand Up @@ -294,10 +307,16 @@ void ExpansionManager::UpdateFailed(CanAddress address) noexcept
UpdateBoardState(address, BoardState::flashFailed);
}

void ExpansionManager::AddAccelerometerRun(CanAddress address, unsigned int numDataPoints) noexcept
void ExpansionManager::AddAccelerometerRun(CanAddress address, unsigned int numDataPoints, float averages[]) noexcept
{
boards[address].accelerometerLastRunDataPoints = numDataPoints;
++boards[address].accelerometerRuns;

for(int32_t i = 0;i < NumAccelerometerAxes;++i)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could we use memcpyf here?

{
boards[address].accelerometerLastRunAverages[i] = averages[i];
}

reprap.BoardsUpdated();
}

Expand Down
5 changes: 3 additions & 2 deletions src/CAN/ExpansionManager.h
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ struct ExpansionBoardData

const char *_ecv_array typeName;
MinCurMax mcuTemp, vin, v12;
float accelerometerLastRunAverages[NumAccelerometerAxes];
uint32_t accelerometerLastRunDataPoints;
uint32_t closedLoopLastRunDataPoints;
UniqueId uniqueId;
Expand Down Expand Up @@ -58,14 +59,14 @@ class ExpansionManager INHERIT_OBJECT_MODEL

void UpdateFinished(CanAddress address) noexcept;
void UpdateFailed(CanAddress address) noexcept;
void AddAccelerometerRun(CanAddress address, unsigned int numDataPoints) noexcept;
void AddAccelerometerRun(CanAddress address, unsigned int numDataPoints, float averages[]) noexcept;
void AddClosedLoopRun(CanAddress address, unsigned int numDataPoints) noexcept;
bool IsFlashing() const noexcept { return numBoardsFlashing != 0; }

void EmergencyStop() noexcept;

protected:
DECLARE_OBJECT_MODEL
DECLARE_OBJECT_MODEL_WITH_ARRAYS

private:
const ExpansionBoardData& FindIndexedBoard(unsigned int index) const noexcept;
Expand Down
10 changes: 9 additions & 1 deletion src/Platform/Platform.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -219,6 +219,13 @@ constexpr ObjectModelArrayTableEntry Platform::objectModelArrayTable[] =
[] (const ObjectModel *self, const ObjectExplorationContext& context) noexcept -> size_t { return NumCoordinateSystems; },
[] (const ObjectModel *self, ObjectExplorationContext& context) noexcept -> ExpressionValue
{ return ExpressionValue(reprap.GetGCodes().GetWorkplaceOffset(context.GetIndex(1), context.GetLastIndex()), 3); }
},
// 2. Average readings from last accelerometer run
{
nullptr, // no lock needed
[] (const ObjectModel *self, const ObjectExplorationContext& context) noexcept -> size_t { return NumAccelerometerAxes; },
[] (const ObjectModel *self, ObjectExplorationContext& context) noexcept -> ExpressionValue
{ return ExpressionValue((float)Accelerometers::GetLocalAccelerometerLastRunAverage(context.GetLastIndex())); }
}
};

Expand Down Expand Up @@ -357,6 +364,7 @@ constexpr ObjectModelTableEntry Platform::objectModelTable[] =

#if SUPPORT_ACCELEROMETERS
// 9. boards[0].accelerometer members
{ "lastRunAverages", OBJECT_MODEL_FUNC_ARRAY(2), ObjectModelEntryFlags::none },
{ "points", OBJECT_MODEL_FUNC_NOSELF((int32_t)Accelerometers::GetLocalAccelerometerDataPoints()), ObjectModelEntryFlags::none },
{ "runs", OBJECT_MODEL_FUNC_NOSELF((int32_t)Accelerometers::GetLocalAccelerometerRuns()), ObjectModelEntryFlags::none },
#endif
Expand Down Expand Up @@ -393,7 +401,7 @@ constexpr uint8_t Platform::objectModelTableDescriptor[] =
2, // section 7: move.axes[].microstepping
2, // section 8: move.extruders[].microstepping
#if SUPPORT_ACCELEROMETERS
2, // section 9: boards[0].accelerometer
3, // section 9: boards[0].accelerometer
#else
0,
#endif
Expand Down