2010-12-28 19:51:12 -04:00
|
|
|
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
|
|
|
//
|
|
|
|
// This is free software; you can redistribute it and/or modify it under
|
|
|
|
// the terms of the GNU Lesser General Public License as published by the
|
|
|
|
// Free Software Foundation; either version 2.1 of the License, or (at
|
|
|
|
// your option) any later version.
|
|
|
|
//
|
2010-12-29 04:26:21 -04:00
|
|
|
/// The AP variable interface. This allows different types
|
2010-12-28 19:51:12 -04:00
|
|
|
/// of variables to be passed to blocks for floating point
|
|
|
|
/// math, memory management, etc.
|
2010-12-31 02:20:28 -04:00
|
|
|
|
2010-12-29 04:26:21 -04:00
|
|
|
#ifndef AP_Var_H
|
|
|
|
#define AP_Var_H
|
2011-01-02 22:28:35 -04:00
|
|
|
#include <stddef.h>
|
2011-01-02 18:14:36 -04:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdint.h>
|
2011-01-02 22:28:35 -04:00
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
#include <avr/pgmspace.h>
|
|
|
|
#include <avr/eeprom.h>
|
|
|
|
|
2011-01-02 22:28:35 -04:00
|
|
|
#include "AP_MetaClass.h"
|
2011-01-02 18:14:36 -04:00
|
|
|
|
|
|
|
/// Nestable scopes for variable names.
|
|
|
|
///
|
|
|
|
/// This provides a mechanism for scoping variable names, and
|
|
|
|
/// may later be extended for other purposes.
|
|
|
|
///
|
2011-01-02 22:28:35 -04:00
|
|
|
/// When AP_Var is asked for the name of a variable, it will
|
2011-01-02 18:14:36 -04:00
|
|
|
/// prepend the names of all enclosing scopes. This provides a way
|
|
|
|
/// of grouping variables and saving memory when many share a large
|
|
|
|
/// common prefix.
|
|
|
|
///
|
|
|
|
class AP_VarScope
|
2010-12-28 19:51:12 -04:00
|
|
|
{
|
|
|
|
public:
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Constructor
|
|
|
|
///
|
|
|
|
/// @param scopeName The name of the scope.
|
|
|
|
///
|
|
|
|
AP_VarScope(const prog_char *name,
|
|
|
|
AP_VarScope *parent = NULL) :
|
|
|
|
_name(name),
|
|
|
|
_parent(parent)
|
|
|
|
{
|
|
|
|
}
|
2010-12-28 19:51:12 -04:00
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Copy the scope name, prefixed by any parent scope names, to a buffer.
|
|
|
|
///
|
|
|
|
/// Note that if the combination of names is larger than the buffer, the
|
|
|
|
/// result in the buffer will be truncated.
|
|
|
|
///
|
|
|
|
/// @param buffer The destination buffer
|
|
|
|
/// @param bufferSize Total size of the destination buffer.
|
|
|
|
///
|
|
|
|
void copy_name(char *buffer, size_t bufferSize) const
|
|
|
|
{
|
|
|
|
if (_parent)
|
|
|
|
_parent->copy_name(buffer, bufferSize);
|
|
|
|
strlcat_P(buffer, _name, bufferSize);
|
|
|
|
}
|
2010-12-28 19:51:12 -04:00
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
private:
|
|
|
|
const prog_char *_name; /// pointer to the scope name in program memory
|
|
|
|
AP_VarScope *_parent; /// pointer to a parent scope, if one exists
|
2010-12-28 19:51:12 -04:00
|
|
|
};
|
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Base class for variables.
|
|
|
|
///
|
|
|
|
/// Provides naming and lookup services for variables.
|
|
|
|
///
|
2011-01-02 22:28:35 -04:00
|
|
|
class AP_Var : public AP_MetaClass
|
2010-12-28 19:51:12 -04:00
|
|
|
{
|
|
|
|
public:
|
2011-01-02 18:14:36 -04:00
|
|
|
/// A unique identity for variables that can be saved to EEPROM
|
|
|
|
///
|
|
|
|
/// @todo This might be used as a token for mass serialisation,
|
|
|
|
/// but for now it's just the address of the variable's backing
|
|
|
|
/// store in EEPROM.
|
|
|
|
///
|
|
|
|
typedef uint16_t AP_VarIdentity;
|
2011-01-02 22:28:35 -04:00
|
|
|
static const AP_VarIdentity AP_VarUnsaved = ~(AP_VarIdentity)0;
|
2011-01-02 18:14:36 -04:00
|
|
|
|
|
|
|
/// The largest variable that will be saved to EEPROM
|
|
|
|
///
|
|
|
|
static const size_t AP_VarMaxSize = 16;
|
|
|
|
|
|
|
|
/// Constructor
|
|
|
|
///
|
|
|
|
/// @param name An optional name by which the variable may be known.
|
|
|
|
/// @param scope An optional scope that the variable may be a contained within.
|
|
|
|
///
|
2011-01-02 22:28:35 -04:00
|
|
|
AP_Var(AP_VarIdentity identity = AP_VarUnsaved,
|
|
|
|
const prog_char *name = NULL,
|
|
|
|
AP_VarScope *scope = NULL) :
|
|
|
|
_identity(identity),
|
|
|
|
_name(name),
|
|
|
|
_scope(scope)
|
2010-12-28 19:51:12 -04:00
|
|
|
{
|
2011-01-02 18:14:36 -04:00
|
|
|
if (name) {
|
|
|
|
_link = _variables;
|
|
|
|
_variables = this;
|
|
|
|
}
|
2010-12-28 19:51:12 -04:00
|
|
|
}
|
|
|
|
|
2011-01-02 22:28:35 -04:00
|
|
|
/// Destructor
|
|
|
|
///
|
|
|
|
/// This is largely a safety net, as the linked-list removal can be inefficient
|
|
|
|
/// when variables are destroyed in an order other than the reverse of the order
|
|
|
|
/// in which they are created.
|
|
|
|
///
|
|
|
|
~AP_Var(void);
|
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Copy the variable name, prefixed by any parent class names, to a buffer.
|
|
|
|
///
|
|
|
|
/// Note that if the combination of names is larger than the buffer, the
|
|
|
|
/// result in the buffer will be truncated.
|
|
|
|
///
|
|
|
|
/// @param buffer The destination buffer
|
|
|
|
/// @param bufferSize Total size of the destination buffer.
|
|
|
|
///
|
|
|
|
void copy_name(char *buffer, size_t bufferSize) const {
|
2011-01-02 22:28:35 -04:00
|
|
|
buffer[0] = '\0';
|
2011-01-02 18:14:36 -04:00
|
|
|
if (_scope)
|
|
|
|
_scope->copy_name(buffer, bufferSize);
|
|
|
|
strlcat_P(buffer, _name, bufferSize);
|
2010-12-29 04:26:21 -04:00
|
|
|
}
|
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Return a pointer to the n'th known variable
|
|
|
|
///
|
|
|
|
/// This interface is designed for the use of relatively low-rate clients;
|
|
|
|
/// GCS interfaces in particular. It may implement an acceleration scheme
|
|
|
|
/// for optimising the lookup of parameters.
|
|
|
|
///
|
|
|
|
/// Note that variable index numbers are not constant, they depend on the
|
|
|
|
/// the static construction order.
|
|
|
|
///
|
|
|
|
/// @param index enumerator for the variable to be returned
|
|
|
|
///
|
2011-01-02 22:28:35 -04:00
|
|
|
static AP_Var *lookup(int index);
|
2011-01-02 18:14:36 -04:00
|
|
|
|
|
|
|
/// Save the current value of the variable to EEPROM.
|
|
|
|
///
|
|
|
|
/// This interface works for any subclass that implements
|
|
|
|
/// serialize.
|
|
|
|
///
|
|
|
|
void save(void) {
|
|
|
|
if (_identity != AP_VarUnsaved) {
|
|
|
|
uint8_t vbuf[AP_VarMaxSize];
|
|
|
|
size_t size;
|
|
|
|
|
2011-01-02 22:28:35 -04:00
|
|
|
// serialize the variable into the buffer and work out how big it is
|
2011-01-02 18:14:36 -04:00
|
|
|
size = serialize(vbuf, sizeof(vbuf));
|
|
|
|
|
|
|
|
// if it fit in the buffer, save it to EEPROM
|
|
|
|
if (size <= sizeof(vbuf))
|
|
|
|
eeprom_write_block(vbuf, (void *)_identity, size);
|
|
|
|
}
|
2010-12-28 19:51:12 -04:00
|
|
|
}
|
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Load the variable from EEPROM.
|
|
|
|
///
|
|
|
|
/// This interface works for any subclass that implements
|
|
|
|
/// unserialize.
|
|
|
|
///
|
|
|
|
void load(void) {
|
|
|
|
if (_identity != AP_VarUnsaved) {
|
|
|
|
uint8_t vbuf[AP_VarMaxSize];
|
|
|
|
size_t size;
|
|
|
|
|
2011-01-02 22:28:35 -04:00
|
|
|
// ask the unserializer how big the variable is
|
2011-01-02 18:14:36 -04:00
|
|
|
size = unserialize(NULL, 0);
|
|
|
|
|
|
|
|
// read the buffer from EEPROM
|
|
|
|
if (size <= sizeof(vbuf)) {
|
|
|
|
eeprom_read_block(vbuf, (void *)_identity, size);
|
|
|
|
unserialize(vbuf, size);
|
|
|
|
}
|
|
|
|
}
|
2010-12-28 19:51:12 -04:00
|
|
|
}
|
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Save all variables to EEPROM
|
|
|
|
///
|
|
|
|
static void save_all(void);
|
2010-12-28 19:51:12 -04:00
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Load all variables from EEPROM
|
|
|
|
///
|
|
|
|
static void load_all(void);
|
2010-12-28 19:51:12 -04:00
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
private:
|
|
|
|
const AP_VarIdentity _identity;
|
|
|
|
const prog_char *_name;
|
|
|
|
AP_VarScope *_scope;
|
2011-01-02 22:28:35 -04:00
|
|
|
AP_Var *_link;
|
2010-12-29 04:26:21 -04:00
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
// static state used by ::lookup
|
2011-01-02 22:28:35 -04:00
|
|
|
static AP_Var *_variables;
|
|
|
|
static AP_Var *_lookupHint; /// pointer to the last variable that was looked up by ::lookup
|
2011-01-02 18:14:36 -04:00
|
|
|
static int _lookupHintIndex; /// index of the last variable that was looked up by ::lookup
|
|
|
|
};
|
2010-12-29 04:26:21 -04:00
|
|
|
|
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Template class for scalar variables.
|
|
|
|
///
|
|
|
|
/// Objects of this type have a value, and can be treated in many ways as though they
|
|
|
|
/// were the value.
|
|
|
|
///
|
|
|
|
/// @tparam T The scalar type of the variable
|
|
|
|
///
|
|
|
|
template<typename T>
|
2011-01-02 22:28:35 -04:00
|
|
|
class AP_VarT : public AP_Var
|
2011-01-02 18:14:36 -04:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
/// Constructor
|
|
|
|
///
|
|
|
|
/// @note Constructors for AP_Var are specialised so that they can
|
2011-01-02 22:28:35 -04:00
|
|
|
/// pass the correct typeCode argument to the AP_Var ctor.
|
2011-01-02 18:14:36 -04:00
|
|
|
///
|
|
|
|
/// @param initialValue Value the variable should have at startup.
|
|
|
|
/// @param identity A unique token used when saving the variable to EEPROM.
|
|
|
|
/// Note that this token must be unique, and should only be
|
|
|
|
/// changed for a variable if the EEPROM version is updated
|
|
|
|
/// to prevent the accidental unserialisation of old data
|
|
|
|
/// into the wrong variable.
|
|
|
|
/// @param name An optional name by which the variable may be known.
|
|
|
|
/// @param varClass An optional class that the variable may be a member of.
|
|
|
|
///
|
2011-01-02 22:28:35 -04:00
|
|
|
AP_VarT<T>(T initialValue = 0,
|
2011-01-02 18:14:36 -04:00
|
|
|
AP_VarIdentity identity = AP_VarUnsaved,
|
|
|
|
const prog_char *name = NULL,
|
|
|
|
AP_VarScope *scope = NULL) :
|
2011-01-02 22:28:35 -04:00
|
|
|
AP_Var(identity, name, scope),
|
2011-01-02 18:14:36 -04:00
|
|
|
_value(initialValue)
|
|
|
|
{
|
2010-12-29 04:26:21 -04:00
|
|
|
}
|
|
|
|
|
2011-01-02 22:28:35 -04:00
|
|
|
// serialize _value into the buffer, but only if it is big enough.
|
2011-01-02 18:14:36 -04:00
|
|
|
///
|
2011-01-02 22:28:35 -04:00
|
|
|
virtual size_t serialize(void *buf, size_t size) {
|
2011-01-02 18:14:36 -04:00
|
|
|
if (size >= sizeof(T))
|
|
|
|
*(T *)buf = _value;
|
|
|
|
return sizeof(T);
|
2010-12-28 19:51:12 -04:00
|
|
|
}
|
|
|
|
|
2011-01-02 22:28:35 -04:00
|
|
|
// Unserialize from the buffer, but only if it is big enough.
|
2011-01-02 18:14:36 -04:00
|
|
|
//
|
2011-01-02 22:28:35 -04:00
|
|
|
virtual size_t unserialize(void *buf, size_t size) {
|
2011-01-02 18:14:36 -04:00
|
|
|
if (size >= sizeof(T))
|
|
|
|
_value = *(T*)buf;
|
|
|
|
return sizeof(T);
|
2010-12-28 19:51:12 -04:00
|
|
|
}
|
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Value getter
|
|
|
|
///
|
|
|
|
T get(void) { return _value; }
|
|
|
|
|
|
|
|
/// Value setter
|
|
|
|
///
|
|
|
|
void set(T v) { _value = v; }
|
|
|
|
|
2011-01-02 22:28:35 -04:00
|
|
|
/// Conversion to T returns a reference to the value.
|
2011-01-02 18:14:36 -04:00
|
|
|
///
|
2011-01-02 22:28:35 -04:00
|
|
|
/// This allows the class to be used in many situations where the value would be legal.
|
2011-01-02 18:14:36 -04:00
|
|
|
///
|
2011-01-02 22:28:35 -04:00
|
|
|
operator T&() { return _value; }
|
2011-01-02 18:14:36 -04:00
|
|
|
|
|
|
|
private:
|
|
|
|
T _value;
|
|
|
|
};
|
2010-12-28 19:51:12 -04:00
|
|
|
|
2010-12-30 01:25:28 -04:00
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Convenience macro for defining instances of the AP_Var template
|
|
|
|
///
|
|
|
|
#define AP_VARDEF(_t, _n) \
|
2011-01-02 22:28:35 -04:00
|
|
|
typedef AP_VarT<_t> AP_##_n;
|
2010-12-28 19:51:12 -04:00
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
AP_VARDEF(float, Float); // defines AP_Float, AP_NamedFloat and AP_SavedFloat
|
|
|
|
AP_VARDEF(int8_t, Int8); // defines AP_UInt8, AP_NamedUInt8 and AP_SavedUInt8
|
|
|
|
AP_VARDEF(int16_t, Int16); // defines AP_UInt16, AP_NamedUInt16 and AP_SavedUInt16
|
|
|
|
AP_VARDEF(int32_t, Int32); // defines AP_UInt32, AP_NamedUInt32 and AP_SavedUInt32
|
2010-12-29 04:26:21 -04:00
|
|
|
|
2011-01-02 18:14:36 -04:00
|
|
|
/// Some convenient constant AP_Vars.
|
|
|
|
extern const AP_Float AP_FloatUnity;
|
|
|
|
extern const AP_Float AP_FloatNegativeUnity;
|
|
|
|
extern const AP_Float AP_FloatZero;
|
2010-12-31 02:20:28 -04:00
|
|
|
|
2010-12-29 04:26:21 -04:00
|
|
|
#endif // AP_Var_H
|