mirror of
https://bitbucket.org/librepilot/librepilot.git
synced 2025-01-19 04:52:12 +01:00
192 lines
4.4 KiB
C
192 lines
4.4 KiB
C
|
/*
|
||
|
# This file is Copyright 2003, 2006, 2007, 2009, 2010 Dean Hall.
|
||
|
#
|
||
|
# This file is part of the PyMite VM.
|
||
|
# The PyMite VM is free software: you can redistribute it and/or modify
|
||
|
# it under the terms of the GNU GENERAL PUBLIC LICENSE Version 2.
|
||
|
#
|
||
|
# The PyMite VM is distributed in the hope that it will be useful,
|
||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||
|
# A copy of the GNU GENERAL PUBLIC LICENSE Version 2
|
||
|
# is seen in the file COPYING in this directory.
|
||
|
*/
|
||
|
|
||
|
|
||
|
#undef __FILE_ID__
|
||
|
#define __FILE_ID__ 0x13
|
||
|
|
||
|
|
||
|
/**
|
||
|
* \file
|
||
|
* \brief Tuple Object Type
|
||
|
*
|
||
|
* Tuple object type operations.
|
||
|
*/
|
||
|
|
||
|
|
||
|
#include "pm.h"
|
||
|
|
||
|
|
||
|
/* The follwing value should match that in pmImgCreator.py */
|
||
|
#define MAX_TUPLE_LEN 253
|
||
|
|
||
|
|
||
|
PmReturn_t
|
||
|
tuple_loadFromImg(PmMemSpace_t memspace,
|
||
|
uint8_t const **paddr, pPmObj_t *r_ptuple)
|
||
|
{
|
||
|
PmReturn_t retval = PM_RET_OK;
|
||
|
uint8_t i = (uint8_t)0;
|
||
|
uint8_t n = (uint8_t)0;
|
||
|
uint8_t objid;
|
||
|
|
||
|
/* Get num objs in tuple */
|
||
|
n = mem_getByte(memspace, paddr);
|
||
|
|
||
|
/* Create empty tuple */
|
||
|
retval = tuple_new(n, r_ptuple);
|
||
|
PM_RETURN_IF_ERROR(retval);
|
||
|
((pPmTuple_t)*r_ptuple)->length = 0;
|
||
|
|
||
|
/* Load the next n objs into tuple */
|
||
|
heap_gcPushTempRoot((pPmObj_t)*r_ptuple, &objid);
|
||
|
for (i = (uint8_t)0; i < n; i++)
|
||
|
{
|
||
|
retval = obj_loadFromImg(memspace,
|
||
|
paddr,
|
||
|
(pPmObj_t *)&(((pPmTuple_t)*r_ptuple)->
|
||
|
val[i]));
|
||
|
if (retval != PM_RET_OK)
|
||
|
{
|
||
|
heap_gcPopTempRoot(objid);
|
||
|
return retval;
|
||
|
}
|
||
|
((pPmTuple_t)*r_ptuple)->length++;
|
||
|
}
|
||
|
heap_gcPopTempRoot(objid);
|
||
|
return PM_RET_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
PmReturn_t
|
||
|
tuple_new(uint16_t n, pPmObj_t *r_ptuple)
|
||
|
{
|
||
|
PmReturn_t retval = PM_RET_OK;
|
||
|
uint16_t size = 0;
|
||
|
|
||
|
/* Raise a SystemError for a Tuple that is too large */
|
||
|
if (n > MAX_TUPLE_LEN)
|
||
|
{
|
||
|
PM_RAISE(retval, PM_RET_EX_SYS);
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
/* Calc size of struct to hold tuple; (n-1) because PmTuple_t has val[1] */
|
||
|
size = sizeof(PmTuple_t) + ((n - 1) * sizeof(pPmObj_t));
|
||
|
|
||
|
/* Allocate a tuple */
|
||
|
retval = heap_getChunk(size, (uint8_t **)r_ptuple);
|
||
|
PM_RETURN_IF_ERROR(retval);
|
||
|
OBJ_SET_TYPE(*r_ptuple, OBJ_TYPE_TUP);
|
||
|
|
||
|
/* Set the number of objs in the tuple */
|
||
|
((pPmTuple_t)*r_ptuple)->length = n;
|
||
|
|
||
|
/* No need to null the ptrs because they are set by the caller */
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
|
||
|
PmReturn_t
|
||
|
tuple_replicate(pPmObj_t ptup, int16_t n, pPmObj_t *r_ptuple)
|
||
|
{
|
||
|
PmReturn_t retval = PM_RET_OK;
|
||
|
int16_t length;
|
||
|
int16_t i;
|
||
|
int16_t j;
|
||
|
|
||
|
/* Raise TypeError if object is not a Tuple */
|
||
|
if (OBJ_GET_TYPE(ptup) != OBJ_TYPE_TUP)
|
||
|
{
|
||
|
PM_RAISE(retval, PM_RET_EX_SYS);
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
C_ASSERT(n >= 0);
|
||
|
|
||
|
/* Allocate the new tuple */
|
||
|
length = ((pPmTuple_t)ptup)->length;
|
||
|
retval = tuple_new(length * n, r_ptuple);
|
||
|
PM_RETURN_IF_ERROR(retval);
|
||
|
|
||
|
/* Copy src tuple the designated number of times */
|
||
|
for (i = 0; i < n; i++)
|
||
|
{
|
||
|
for (j = 0; j < length; j++)
|
||
|
{
|
||
|
((pPmTuple_t)*r_ptuple)->val[length * i + j] =
|
||
|
((pPmTuple_t)ptup)->val[j];
|
||
|
}
|
||
|
}
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
|
||
|
PmReturn_t
|
||
|
tuple_getItem(pPmObj_t ptup, int16_t index, pPmObj_t *r_pobj)
|
||
|
{
|
||
|
PmReturn_t retval = PM_RET_OK;
|
||
|
|
||
|
/* Adjust for negative index */
|
||
|
if (index < 0)
|
||
|
{
|
||
|
index += ((pPmTuple_t)ptup)->length;
|
||
|
}
|
||
|
|
||
|
/* Raise IndexError if index is out of bounds */
|
||
|
if ((index < 0) || (index > ((pPmTuple_t)ptup)->length))
|
||
|
{
|
||
|
PM_RAISE(retval, PM_RET_EX_INDX);
|
||
|
}
|
||
|
|
||
|
/* Get the tuple item */
|
||
|
*r_pobj = ((pPmTuple_t)ptup)->val[index];
|
||
|
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
|
||
|
#ifdef HAVE_PRINT
|
||
|
PmReturn_t
|
||
|
tuple_print(pPmObj_t ptup)
|
||
|
{
|
||
|
PmReturn_t retval = PM_RET_OK;
|
||
|
int16_t index;
|
||
|
|
||
|
C_ASSERT(ptup != C_NULL);
|
||
|
|
||
|
/* If it's not a tuple, raise TypeError */
|
||
|
if (OBJ_GET_TYPE(ptup) != OBJ_TYPE_TUP)
|
||
|
{
|
||
|
PM_RAISE(retval, PM_RET_EX_TYPE);
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
plat_putByte('(');
|
||
|
|
||
|
for (index = 0; index < ((pPmTuple_t)ptup)->length; index++)
|
||
|
{
|
||
|
if (index != 0)
|
||
|
{
|
||
|
plat_putByte(',');
|
||
|
plat_putByte(' ');
|
||
|
}
|
||
|
retval = obj_print(((pPmTuple_t)ptup)->val[index], C_FALSE, C_TRUE);
|
||
|
PM_RETURN_IF_ERROR(retval);
|
||
|
}
|
||
|
|
||
|
return plat_putByte(')');
|
||
|
}
|
||
|
#endif /* HAVE_PRINT */
|