1
0
mirror of https://bitbucket.org/librepilot/librepilot.git synced 2025-01-19 04:52:12 +01:00

253 lines
5.1 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__ 0x08
/**
* \file
* \brief Integer Object Type
*
* Integer object type operations.
*/
#include <stdint.h>
#include <limits.h>
#include "pm.h"
PmReturn_t
int_dup(pPmObj_t pint, pPmObj_t *r_pint)
{
PmReturn_t retval = PM_RET_OK;
/* Allocate new int */
retval = heap_getChunk(sizeof(PmInt_t), (uint8_t **)r_pint);
PM_RETURN_IF_ERROR(retval);
/* Copy value */
OBJ_SET_TYPE(*r_pint, OBJ_TYPE_INT);
((pPmInt_t)*r_pint)->val = ((pPmInt_t)pint)->val;
return retval;
}
PmReturn_t
int_new(int32_t n, pPmObj_t *r_pint)
{
PmReturn_t retval = PM_RET_OK;
/* If n is 0,1,-1, return static int objects from global struct */
if (n == 0)
{
*r_pint = PM_ZERO;
return PM_RET_OK;
}
if (n == 1)
{
*r_pint = PM_ONE;
return PM_RET_OK;
}
if (n == -1)
{
*r_pint = PM_NEGONE;
return PM_RET_OK;
}
/* Else create and return new int obj */
retval = heap_getChunk(sizeof(PmInt_t), (uint8_t **)r_pint);
PM_RETURN_IF_ERROR(retval);
OBJ_SET_TYPE(*r_pint, OBJ_TYPE_INT);
((pPmInt_t)*r_pint)->val = n;
return retval;
}
PmReturn_t
int_positive(pPmObj_t pobj, pPmObj_t *r_pint)
{
PmReturn_t retval;
/* Raise TypeError if obj is not an int */
if (OBJ_GET_TYPE(pobj) != OBJ_TYPE_INT)
{
PM_RAISE(retval, PM_RET_EX_TYPE);
return retval;
}
/* Create new int obj */
return int_new(((pPmInt_t)pobj)->val, r_pint);
}
PmReturn_t
int_negative(pPmObj_t pobj, pPmObj_t *r_pint)
{
PmReturn_t retval;
/* Raise TypeError if obj is not an int */
if (OBJ_GET_TYPE(pobj) != OBJ_TYPE_INT)
{
PM_RAISE(retval, PM_RET_EX_TYPE);
return retval;
}
/* Create new int obj */
return int_new(-((pPmInt_t)pobj)->val, r_pint);
}
PmReturn_t
int_bitInvert(pPmObj_t pobj, pPmObj_t *r_pint)
{
PmReturn_t retval;
/* Raise TypeError if obj is not an int */
if (OBJ_GET_TYPE(pobj) != OBJ_TYPE_INT)
{
PM_RAISE(retval, PM_RET_EX_TYPE);
return retval;
}
/* Create new int obj */
return int_new(~((pPmInt_t)pobj)->val, r_pint);
}
#ifdef HAVE_PRINT
PmReturn_t
int_print(pPmObj_t pint)
{
/* 2^31-1 has 10 decimal digits, plus sign and zero byte */
uint8_t tBuffer[10 + 1 + 1];
uint8_t bytesWritten;
uint8_t k;
PmReturn_t retval = PM_RET_OK;
C_ASSERT(pint != C_NULL);
/* Raise TypeError if obj is not an int */
if (OBJ_GET_TYPE(pint) != OBJ_TYPE_INT)
{
PM_RAISE(retval, PM_RET_EX_TYPE);
return retval;
}
/* #196: Changed to use snprintf */
bytesWritten =
snprintf((char *)&tBuffer, 12, "%li", (long int)((pPmInt_t)pint)->val);
/* Sanity check */
C_ASSERT(bytesWritten != 0);
C_ASSERT(bytesWritten < sizeof(tBuffer));
for (k = (uint8_t)0; k < bytesWritten; k++)
{
retval = plat_putByte(tBuffer[k]);
PM_RETURN_IF_ERROR(retval);
}
return PM_RET_OK;
}
PmReturn_t
int_printHexByte(uint8_t b)
{
uint8_t nibble;
PmReturn_t retval;
nibble = (b >> 4) + '0';
if (nibble > '9')
nibble += ('a' - '0' - 10);
retval = plat_putByte(nibble);
PM_RETURN_IF_ERROR(retval);
nibble = (b & (uint8_t)0x0F) + '0';
if (nibble > '9')
nibble += ('a' - '0' - (uint8_t)10);
retval = plat_putByte(nibble);
return retval;
}
PmReturn_t
_int_printHex(intptr_t n)
{
PmReturn_t retval;
int8_t i;
/* Print the hex value, most significant byte first */
for (i = CHAR_BIT * sizeof(intptr_t) - 8; i >= 0; i -= 8)
{
retval = int_printHexByte((n >> i) & 0xFF);
PM_BREAK_IF_ERROR(retval);
}
return retval;
}
PmReturn_t
int_printHex(pPmObj_t pint)
{
C_ASSERT(OBJ_GET_TYPE(pint) == OBJ_TYPE_INT);
/* Print the integer object */
return _int_printHex(((pPmInt_t)pint)->val);
}
#endif /* HAVE_PRINT */
PmReturn_t
int_pow(pPmObj_t px, pPmObj_t py, pPmObj_t *r_pn)
{
int32_t x;
int32_t y;
int32_t n;
PmReturn_t retval;
/* Raise TypeError if args aren't ints */
if ((OBJ_GET_TYPE(px) != OBJ_TYPE_INT)
|| (OBJ_GET_TYPE(py) != OBJ_TYPE_INT))
{
PM_RAISE(retval, PM_RET_EX_TYPE);
return retval;
}
x = ((pPmInt_t)px)->val;
y = ((pPmInt_t)py)->val;
/* Raise Value error if exponent is negative */
if (y < 0)
{
PM_RAISE(retval, PM_RET_EX_VAL);
return retval;
}
/* Calculate x raised to y */
n = 1;
while (y > 0)
{
n = n * x;
y--;
}
retval = int_new(n, r_pn);
return retval;
}