blob: 5f8ddacf22f3cf88a1d0cf350dc0cfbb6662f214 [file] [log] [blame] [edit]
#include "bej_decoder_json.hpp"
namespace libbej
{
/**
* @brief This structure is used to pass additional data to callback functions.
*/
struct BejJsonParam
{
bool* isPrevAnnotated;
std::string* output;
};
/**
* @brief Add a property name to output buffer.
*
* @param[in] params - a valid BejJsonParam struct.
* @param[in] propertyName - a NULL terminated string.
*/
static void addPropertyNameToOutput(struct BejJsonParam* params,
const char* propertyName)
{
if (propertyName[0] == '\0')
{
return;
}
if (!(*params->isPrevAnnotated))
{
params->output->push_back('\"');
}
params->output->append(propertyName);
params->output->append("\":");
}
/**
* @brief Callback for bejSet start.
*
* @param[in] propertyName - a NULL terminated string.
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackSetStart(const char* propertyName, void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
addPropertyNameToOutput(params, propertyName);
params->output->push_back('{');
*params->isPrevAnnotated = false;
return 0;
}
/**
* @brief Callback for bejSet end.
*
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackSetEnd(void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
params->output->push_back('}');
return 0;
}
/**
* @brief Callback for bejArray start.
*
* @param[in] propertyName - a NULL terminated string.
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackArrayStart(const char* propertyName, void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
addPropertyNameToOutput(params, propertyName);
params->output->push_back('[');
*params->isPrevAnnotated = false;
return 0;
}
/**
* @brief Callback for bejArray end.
*
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackArrayEnd(void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
params->output->push_back(']');
return 0;
}
/**
* @brief Callback when an end of a property is detected.
*
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackPropertyEnd(void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
// Not a section ending. So add a comma.
params->output->push_back(',');
return 0;
}
/**
* @brief Callback for bejNull type.
*
* @param[in] propertyName - a NULL terminated string.
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackNull(const char* propertyName, void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
addPropertyNameToOutput(params, propertyName);
params->output->append("null");
*params->isPrevAnnotated = false;
return 0;
}
/**
* @brief Callback for bejInteger type.
*
* @param[in] propertyName - a NULL terminated string.
* @param[in] value - integer value.
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackInteger(const char* propertyName, int64_t value,
void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
addPropertyNameToOutput(params, propertyName);
params->output->append(std::to_string(value));
*params->isPrevAnnotated = false;
return 0;
}
/**
* @brief Callback for bejEnum type.
*
* @param[in] propertyName - a NULL terminated string.
* @param[in] value - a NULL terminated string.
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackEnum(const char* propertyName, const char* value,
void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
addPropertyNameToOutput(params, propertyName);
params->output->push_back('\"');
params->output->append(value);
params->output->push_back('\"');
*params->isPrevAnnotated = false;
return 0;
}
/**
* @brief Callback for bejString type.
*
* @param[in] propertyName - a NULL terminated string.
* @param[in] value - a NULL terminated string.
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackString(const char* propertyName, const char* value,
void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
addPropertyNameToOutput(params, propertyName);
params->output->push_back('\"');
params->output->append(value);
params->output->push_back('\"');
*params->isPrevAnnotated = false;
return 0;
}
/**
* @brief Callback for bejReal type.
*
* @param[in] propertyName - a NULL terminated string.
* @param[in] value - pointing to a valid BejReal.
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackReal(const char* propertyName, const struct BejReal* value,
void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
addPropertyNameToOutput(params, propertyName);
params->output->append(std::to_string(value->whole));
params->output->push_back('.');
params->output->insert(params->output->cend(), value->zeroCount, '0');
params->output->append(std::to_string(value->fract));
if (value->expLen != 0)
{
params->output->push_back('e');
params->output->append(std::to_string(value->exp));
}
*params->isPrevAnnotated = false;
return 0;
}
/**
* @brief Callback for bejBoolean type.
*
* @param[in] propertyName - a NULL terminated string.
* @param[in] value - boolean value.
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackBool(const char* propertyName, bool value, void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
addPropertyNameToOutput(params, propertyName);
params->output->append(value ? "true" : "false");
*params->isPrevAnnotated = false;
return 0;
}
/**
* @brief Callback for bejPropertyAnnotation type.
*
* @param[in] propertyName - a NULL terminated string.
* @param[in] dataPtr - pointing to a valid BejJsonParam struct.
* @return 0 if successful.
*/
static int callbackAnnotation(const char* propertyName, void* dataPtr)
{
struct BejJsonParam* params =
reinterpret_cast<struct BejJsonParam*>(dataPtr);
params->output->push_back('\"');
params->output->append(propertyName);
// bejPropertyAnnotation type has the form "Status@Message.ExtendedInfo".
// First the decoder will see "Status" part of the annotated property. This
// will be in its own SFLV tuple. The remainder of the property name,
// @Message.ExtendedInfo will be contained in the next bej SFLV tuple.
// Therefore to add the inverted commas to the complete property name,
// Status@Message.ExtendedInfo, we need to know that the previous property
// we processed is a start to an annotation property. We can use
// isPrevAnnotated to pass this information.
// Here we are adding: "propertyName
// If isPrevAnnotated is true, next property should add: propertyNameNext"
*params->isPrevAnnotated = true;
return 0;
}
/**
* @brief Callback for stackEmpty.
*
* @param[in] dataPtr - pointer to a valid std::vector<BejStackProperty>
* @return true if the stack is empty.
*/
static bool stackEmpty(void* dataPtr)
{
std::vector<BejStackProperty>* stack =
reinterpret_cast<std::vector<BejStackProperty>*>(dataPtr);
return stack->empty();
}
/**
* @brief Callback for stackPeek.
*
* @param[in] dataPtr - pointer to a valid std::vector<BejStackProperty>
* @return a const reference to the stack top.
*/
static const struct BejStackProperty* stackPeek(void* dataPtr)
{
std::vector<BejStackProperty>* stack =
reinterpret_cast<std::vector<BejStackProperty>*>(dataPtr);
if (stack->empty())
{
return nullptr;
}
return &(stack->back());
}
/**
* @brief Callback for stackPop. Remove the top element from the stack.
*
* @param[in] dataPtr - pointer to a valid std::vector<BejStackProperty>
*/
static void stackPop(void* dataPtr)
{
std::vector<BejStackProperty>* stack =
reinterpret_cast<std::vector<BejStackProperty>*>(dataPtr);
if (stack->empty())
{
return;
}
stack->pop_back();
}
/**
* @brief Callback for stackPush. Push a new element to the top of the stack.
*
* @param[in] property - property to push.
* @param[in] dataPtr - pointer to a valid std::vector<BejStackProperty>
* @return 0 if successful.
*/
static int stackPush(const struct BejStackProperty* const property,
void* dataPtr)
{
std::vector<BejStackProperty>* stack =
reinterpret_cast<std::vector<BejStackProperty>*>(dataPtr);
stack->push_back(*property);
return 0;
}
int BejDecoderJson::decode(const BejDictionaries& dictionaries,
const std::span<const uint8_t> encodedPldmBlock)
{
// Clear the previous output if any.
output.clear();
// The dictionaries have to be traversed in a depth first manner. This is
// using a stack to implement it non-recursively. Going into a set or an
// array or a property annotation section means that we have to jump to the
// child dictionary offset start point but needs to retrieve the parent
// dictionary offset start once all the children are processed. This stack
// will hold the parent dictionary offsets and endings for each section.
stack.clear();
struct BejStackCallback stackCallback = {
.stackEmpty = stackEmpty,
.stackPeek = stackPeek,
.stackPop = stackPop,
.stackPush = stackPush,
};
struct BejDecodedCallback decodedCallback = {
.callbackSetStart = callbackSetStart,
.callbackSetEnd = callbackSetEnd,
.callbackArrayStart = callbackArrayStart,
.callbackArrayEnd = callbackArrayEnd,
.callbackPropertyEnd = callbackPropertyEnd,
.callbackNull = callbackNull,
.callbackInteger = callbackInteger,
.callbackEnum = callbackEnum,
.callbackString = callbackString,
.callbackReal = callbackReal,
.callbackBool = callbackBool,
.callbackAnnotation = callbackAnnotation,
.callbackReadonlyProperty = nullptr,
};
isPrevAnnotated = false;
struct BejJsonParam callbackData = {
.isPrevAnnotated = &isPrevAnnotated,
.output = &output,
};
return bejDecodePldmBlock(
&dictionaries, encodedPldmBlock.data(), encodedPldmBlock.size_bytes(),
&stackCallback, &decodedCallback, (void*)(&callbackData),
(void*)(&stack));
}
std::string BejDecoderJson::getOutput()
{
return output;
}
} // namespace libbej