2017-12-05 16:51:20 +00:00
|
|
|
/*
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
Copyright (c) 2014-2017 Chung, Hyung-Hwan. All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions
|
|
|
|
are met:
|
|
|
|
1. Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in the
|
|
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR
|
|
|
|
IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WAfRRANTIES
|
|
|
|
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include "moo-prv.h"
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------
|
|
|
|
* COMPARISON
|
|
|
|
* -------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_identical (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv, arg, b;
|
|
|
|
|
|
|
|
MOO_ASSERT (moo, nargs == 1);
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
arg = MOO_STACK_GETARG(moo, nargs, 0);
|
|
|
|
|
|
|
|
b = (rcv == arg)? moo->_true: moo->_false;
|
|
|
|
|
|
|
|
MOO_STACK_SETRET (moo, nargs, b);
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_not_identical (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv, arg, b;
|
|
|
|
|
|
|
|
MOO_ASSERT (moo, nargs == 1);
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
arg = MOO_STACK_GETARG(moo, nargs, 0);
|
|
|
|
|
|
|
|
b = (rcv != arg)? moo->_true: moo->_false;
|
|
|
|
|
|
|
|
MOO_STACK_SETRET (moo, nargs, b);
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _equal_objects (moo_t* moo, moo_oop_t rcv, moo_oop_t arg)
|
|
|
|
{
|
|
|
|
int rtag;
|
|
|
|
|
|
|
|
if (rcv == arg) return 1; /* identical. so equal */
|
|
|
|
|
|
|
|
rtag = MOO_OOP_GET_TAG(rcv);
|
|
|
|
if (rtag != MOO_OOP_GET_TAG(arg)) return 0;
|
|
|
|
|
|
|
|
switch (rtag)
|
|
|
|
{
|
|
|
|
case MOO_OOP_TAG_SMOOI:
|
|
|
|
return MOO_OOP_TO_SMOOI(rcv) == MOO_OOP_TO_SMOOI(arg)? 1: 0;
|
|
|
|
|
|
|
|
case MOO_OOP_TAG_SMPTR:
|
|
|
|
return MOO_OOP_TO_SMPTR(rcv) == MOO_OOP_TO_SMPTR(arg)? 1: 0;
|
|
|
|
|
|
|
|
case MOO_OOP_TAG_CHAR:
|
|
|
|
return MOO_OOP_TO_CHAR(rcv) == MOO_OOP_TO_CHAR(arg)? 1: 0;
|
|
|
|
|
|
|
|
case MOO_OOP_TAG_ERROR:
|
|
|
|
return MOO_OOP_TO_ERROR(rcv) == MOO_OOP_TO_ERROR(arg)? 1: 0;
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
MOO_ASSERT (moo, MOO_OOP_IS_POINTER(rcv));
|
|
|
|
|
|
|
|
if (MOO_OBJ_GET_CLASS(rcv) != MOO_OBJ_GET_CLASS(arg)) return 0; /* different class, not equal */
|
|
|
|
MOO_ASSERT (moo, MOO_OBJ_GET_FLAGS_TYPE(rcv) == MOO_OBJ_GET_FLAGS_TYPE(arg));
|
|
|
|
|
|
|
|
if (MOO_OBJ_GET_CLASS(rcv) == moo->_class && rcv != arg)
|
|
|
|
{
|
|
|
|
/* a class object are supposed to be unique */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (MOO_OBJ_GET_SIZE(rcv) != MOO_OBJ_GET_SIZE(arg)) return 0; /* different size, not equal */
|
|
|
|
|
|
|
|
switch (MOO_OBJ_GET_FLAGS_TYPE(rcv))
|
|
|
|
{
|
|
|
|
case MOO_OBJ_TYPE_BYTE:
|
|
|
|
case MOO_OBJ_TYPE_CHAR:
|
|
|
|
case MOO_OBJ_TYPE_HALFWORD:
|
|
|
|
case MOO_OBJ_TYPE_WORD:
|
2018-01-04 15:42:29 +00:00
|
|
|
return (MOO_MEMCMP(MOO_OBJ_GET_BYTE_SLOT(rcv), MOO_OBJ_GET_BYTE_SLOT(arg), MOO_BYTESOF(moo,rcv)) == 0)? 1: 0;
|
2017-12-05 16:51:20 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
2018-01-04 15:42:29 +00:00
|
|
|
moo_oow_t i, size;
|
|
|
|
|
2017-12-05 16:51:20 +00:00
|
|
|
if (rcv == moo->_nil) return arg == moo->_nil? 1: 0;
|
|
|
|
if (rcv == moo->_true) return arg == moo->_true? 1: 0;
|
|
|
|
if (rcv == moo->_false) return arg == moo->_false? 1: 0;
|
|
|
|
|
|
|
|
/* MOO_OBJ_TYPE_OOP, ... */
|
|
|
|
MOO_ASSERT (moo, MOO_OBJ_GET_FLAGS_TYPE(rcv) == MOO_OBJ_TYPE_OOP);
|
|
|
|
|
|
|
|
#if 1
|
|
|
|
moo_seterrbfmt (moo, MOO_ENOIMPL, "no builtin comparison implemented for %O and %O", rcv, arg); /* TODO: better error code */
|
|
|
|
return -1;
|
|
|
|
#else
|
2018-01-04 15:42:29 +00:00
|
|
|
|
|
|
|
if (moo->_process && MOO_OBJ_GET_CLASS(rcv) == moo->_process)
|
|
|
|
{
|
|
|
|
/* the stack in a process object doesn't need to be
|
|
|
|
* scanned in full. the slots above the stack pointer
|
|
|
|
* are garbages. */
|
|
|
|
size = MOO_PROCESS_NAMED_INSTVARS +
|
|
|
|
MOO_OOP_TO_SMOOI(((moo_oop_process_t)rcv)->sp) + 1;
|
|
|
|
MOO_ASSERT (moo, size <= MOO_OBJ_GET_SIZE(rcv));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
size = MOO_OBJ_GET_SIZE(rcv);
|
|
|
|
}
|
|
|
|
for (i = 0; i < size; i++)
|
2017-12-05 16:51:20 +00:00
|
|
|
{
|
|
|
|
/* TODO: remove recursion */
|
|
|
|
|
|
|
|
/* NOTE: even if the object implements the equality method,
|
|
|
|
* this primitive method doesn't honor it. */
|
|
|
|
if (!_equal_objects(moo, ((moo_oop_oop_t)rcv)->slot[i], ((moo_oop_oop_t)arg)->slot[i])) return 0;
|
|
|
|
}
|
2018-01-04 15:42:29 +00:00
|
|
|
|
|
|
|
/* the default implementation doesn't take the trailer space into account */
|
2017-12-05 16:51:20 +00:00
|
|
|
return 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_equal (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv, arg;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
arg = MOO_STACK_GETARG(moo, nargs, 0);
|
|
|
|
|
|
|
|
n = _equal_objects (moo, rcv, arg);
|
|
|
|
if (n <= -1) return MOO_PF_FAILURE;
|
|
|
|
|
|
|
|
MOO_STACK_SETRET (moo, nargs, (n? moo->_true: moo->_false));
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_not_equal (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv, arg;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
arg = MOO_STACK_GETARG(moo, nargs, 0);
|
|
|
|
|
|
|
|
n = _equal_objects (moo, rcv, arg);
|
|
|
|
if (n <= -1) return MOO_PF_FAILURE;
|
|
|
|
|
|
|
|
MOO_STACK_SETRET (moo, nargs, (n? moo->_false: moo->_true));
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------
|
|
|
|
* INSTANTIATION
|
|
|
|
* -------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_basic_new (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_class_t _class;
|
|
|
|
moo_oop_t szoop, obj;
|
|
|
|
moo_oow_t size = 0; /* size of the variable/indexed part */
|
|
|
|
|
|
|
|
_class = (moo_oop_class_t)MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
if (MOO_CLASSOF(moo, _class) != moo->_class)
|
|
|
|
{
|
|
|
|
/* the receiver is not a class object */
|
|
|
|
moo_seterrbfmt (moo, MOO_EMSGRCV, "non-class receiver - %O", _class);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check if #limited is set on the class */
|
|
|
|
if (MOO_CLASS_SELFSPEC_FLAGS(MOO_OOP_TO_SMOOI(_class->selfspec)) & MOO_CLASS_SELFSPEC_FLAG_LIMITED)
|
|
|
|
{
|
|
|
|
moo_seterrbfmt (moo, MOO_EPERM, "limited receiver - %O", _class);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nargs >= 1)
|
|
|
|
{
|
|
|
|
szoop = MOO_STACK_GETARG(moo, nargs, 0);
|
|
|
|
if (moo_inttooow (moo, szoop, &size) <= 0)
|
|
|
|
{
|
|
|
|
/* integer out of range or not integer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "size out of range or not integer - %O", szoop);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MOO_OOP_IS_SMOOI(((moo_oop_class_t)_class)->trsize))
|
|
|
|
{
|
|
|
|
obj = moo_instantiatewithtrailer (moo, _class, size, MOO_NULL, MOO_OOP_TO_SMOOI(((moo_oop_class_t)_class)->trsize));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* moo_instantiate() will ignore size if the instance specification
|
|
|
|
* disallows indexed(variable) parts. */
|
|
|
|
/* TODO: should i check the specification before calling
|
|
|
|
* moo_instantiate()? */
|
|
|
|
obj = moo_instantiate (moo, _class, MOO_NULL, size);
|
|
|
|
}
|
|
|
|
if (!obj) return MOO_PF_FAILURE;
|
|
|
|
|
|
|
|
MOO_STACK_SETRET (moo, nargs, obj);
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_shallow_copy (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv, obj;
|
|
|
|
|
|
|
|
MOO_ASSERT (moo, nargs == 0);
|
|
|
|
rcv = MOO_STACK_GETRCV (moo, nargs);
|
|
|
|
|
|
|
|
obj = moo_shallowcopy (moo, rcv);
|
|
|
|
if (!obj) return MOO_PF_FAILURE;
|
|
|
|
|
|
|
|
MOO_STACK_SETRET (moo, nargs, obj);
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------
|
|
|
|
* BASIC ACCESS
|
|
|
|
* -------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_class (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv;
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
MOO_STACK_SETRET (moo, nargs, (moo_oop_t)MOO_CLASSOF(moo, rcv));
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_basic_size (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
/* return the number of indexable fields */
|
|
|
|
|
|
|
|
moo_oop_t rcv, sz;
|
|
|
|
|
|
|
|
MOO_ASSERT (moo, nargs == 0);
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV (moo, nargs);
|
|
|
|
|
|
|
|
if (!MOO_OOP_IS_POINTER(rcv))
|
|
|
|
{
|
2017-12-31 03:19:50 +00:00
|
|
|
/* a non-pointer object has the size of 0.
|
|
|
|
* such an object doesn't consume object memory space except an OOP word */
|
2017-12-05 16:51:20 +00:00
|
|
|
sz = MOO_SMOOI_TO_OOP(0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-12-31 03:19:50 +00:00
|
|
|
sz = moo_oowtoint(moo, MOO_OBJ_GET_SIZE(rcv));
|
2017-12-05 16:51:20 +00:00
|
|
|
if (!sz) return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
MOO_STACK_SETRET(moo, nargs, sz);
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_basic_at (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv, pos, v;
|
|
|
|
moo_oow_t idx;
|
|
|
|
|
|
|
|
MOO_ASSERT (moo, nargs == 1);
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
if (!MOO_OOP_IS_POINTER(rcv))
|
|
|
|
{
|
|
|
|
/* the receiver is a special numeric object, not a normal pointer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EMSGRCV, "receiver not indexable - %O", rcv);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
pos = MOO_STACK_GETARG(moo, nargs, 0);
|
|
|
|
if (moo_inttooow (moo, pos, &idx) <= 0)
|
|
|
|
{
|
|
|
|
/* negative integer or not integer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "invalid position - %O", pos);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
if (idx >= MOO_OBJ_GET_SIZE(rcv))
|
|
|
|
{
|
|
|
|
/* index out of range */
|
2018-01-01 15:56:55 +00:00
|
|
|
moo_seterrbfmt (moo, MOO_ERANGE, "position out of bound - %zu", idx);
|
2017-12-05 16:51:20 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (MOO_OBJ_GET_FLAGS_TYPE(rcv))
|
|
|
|
{
|
|
|
|
case MOO_OBJ_TYPE_BYTE:
|
|
|
|
v = MOO_SMOOI_TO_OOP(((moo_oop_byte_t)rcv)->slot[idx]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_CHAR:
|
|
|
|
v = MOO_CHAR_TO_OOP(((moo_oop_char_t)rcv)->slot[idx]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_HALFWORD:
|
|
|
|
/* TODO: LargeInteger if the halfword is too large */
|
|
|
|
v = MOO_SMOOI_TO_OOP(((moo_oop_halfword_t)rcv)->slot[idx]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_WORD:
|
|
|
|
v = moo_oowtoint (moo, ((moo_oop_word_t)rcv)->slot[idx]);
|
|
|
|
if (!v) return MOO_PF_FAILURE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_OOP:
|
|
|
|
v = ((moo_oop_oop_t)rcv)->slot[idx];
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
moo_seterrnum (moo, MOO_EINTERN);
|
|
|
|
return MOO_PF_HARD_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
MOO_STACK_SETRET (moo, nargs, v);
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_basic_at_put (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv, pos, val;
|
|
|
|
moo_oow_t idx;
|
|
|
|
|
|
|
|
MOO_ASSERT (moo, nargs == 2);
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
if (!MOO_OOP_IS_POINTER(rcv))
|
|
|
|
{
|
|
|
|
/* the receiver is a special numeric object, not a normal pointer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EMSGRCV, "receiver not indexable - %O", rcv);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MOO_OBJ_GET_FLAGS_RDONLY(rcv))
|
|
|
|
{
|
|
|
|
moo_seterrbfmt (moo, MOO_EPERM, "now allowed to change a read-only object - %O", rcv);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
pos = MOO_STACK_GETARG(moo, nargs, 0);
|
|
|
|
val = MOO_STACK_GETARG(moo, nargs, 1);
|
|
|
|
|
|
|
|
if (moo_inttooow (moo, pos, &idx) <= 0)
|
|
|
|
{
|
|
|
|
/* negative integer or not integer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "invalid position - %O", pos);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
if (idx >= MOO_OBJ_GET_SIZE(rcv))
|
|
|
|
{
|
|
|
|
/* index out of range */
|
2018-01-01 15:56:55 +00:00
|
|
|
moo_seterrbfmt (moo, MOO_ERANGE, "position out of bound - %zu", idx);
|
2017-12-05 16:51:20 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (MOO_OBJ_GET_FLAGS_TYPE(rcv))
|
|
|
|
{
|
|
|
|
case MOO_OBJ_TYPE_BYTE:
|
|
|
|
if (!MOO_OOP_IS_SMOOI(val))
|
|
|
|
{
|
2017-12-31 03:19:50 +00:00
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "value not a byte - %O", val);
|
2017-12-05 16:51:20 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
/* TOOD: must I check the range of the value? */
|
2017-12-31 03:19:50 +00:00
|
|
|
((moo_oop_byte_t)rcv)->slot[idx] = MOO_OOP_TO_SMOOI(val);
|
2017-12-05 16:51:20 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_CHAR:
|
|
|
|
if (!MOO_OOP_IS_CHAR(val))
|
|
|
|
{
|
2017-12-31 03:19:50 +00:00
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "value not a character - %O", val);
|
2017-12-05 16:51:20 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
((moo_oop_char_t)rcv)->slot[idx] = MOO_OOP_TO_CHAR(val);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_HALFWORD:
|
|
|
|
if (!MOO_OOP_IS_SMOOI(val))
|
|
|
|
{
|
|
|
|
/* the value is not a number */
|
2017-12-31 03:19:50 +00:00
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "value not a half-word integer - %O", val);
|
2017-12-05 16:51:20 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if the small integer is too large, it will get truncated */
|
|
|
|
((moo_oop_halfword_t)rcv)->slot[idx] = MOO_OOP_TO_SMOOI(val);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_WORD:
|
|
|
|
{
|
|
|
|
moo_oow_t w;
|
|
|
|
|
|
|
|
if (moo_inttooow (moo, val, &w) <= 0)
|
|
|
|
{
|
|
|
|
/* the value is not a number, out of range, or negative */
|
2017-12-31 03:19:50 +00:00
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "value not a word integer - %O", val);
|
2017-12-05 16:51:20 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
((moo_oop_word_t)rcv)->slot[idx] = w;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_OOP:
|
|
|
|
((moo_oop_oop_t)rcv)->slot[idx] = val;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
moo_seterrnum (moo, MOO_EINTERN);
|
|
|
|
return MOO_PF_HARD_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: return receiver or value? */
|
|
|
|
MOO_STACK_SETRET (moo, nargs, val);
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-01-02 16:25:37 +00:00
|
|
|
moo_pfrc_t moo_pf_basic_fill (moo_t* moo, moo_ooi_t nargs)
|
2018-01-01 15:56:55 +00:00
|
|
|
{
|
2018-01-02 16:25:37 +00:00
|
|
|
moo_oop_t rcv, spos, dval, slen;
|
|
|
|
moo_oow_t sidx, ssz, maxlen, end, i;
|
2018-01-01 15:56:55 +00:00
|
|
|
|
|
|
|
MOO_ASSERT (moo, nargs == 3);
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
if (!MOO_OOP_IS_POINTER(rcv))
|
|
|
|
{
|
|
|
|
/* the receiver is a special numeric object, not a normal pointer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EMSGRCV, "receiver not indexable - %O", rcv);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MOO_OBJ_GET_FLAGS_RDONLY(rcv))
|
|
|
|
{
|
|
|
|
moo_seterrbfmt (moo, MOO_EPERM, "now allowed to change a read-only object - %O", rcv);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
spos = MOO_STACK_GETARG(moo, nargs, 0);
|
2018-01-02 16:25:37 +00:00
|
|
|
dval = MOO_STACK_GETARG(moo, nargs, 1);
|
|
|
|
slen = MOO_STACK_GETARG(moo, nargs, 2);
|
2018-01-01 15:56:55 +00:00
|
|
|
|
|
|
|
if (moo_inttooow(moo, spos, &sidx) <= 0)
|
|
|
|
{
|
|
|
|
/* negative integer or not integer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "invalid source position - %O", spos);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
2018-01-04 10:07:42 +00:00
|
|
|
if (moo_inttooow(moo, slen, &ssz) <= 0)
|
2018-01-01 15:56:55 +00:00
|
|
|
{
|
2018-01-04 10:07:42 +00:00
|
|
|
/* negative integer or not integer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "invalid fill count - %O", slen);
|
2018-01-01 15:56:55 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
2018-01-04 10:07:42 +00:00
|
|
|
if (ssz <= 0)
|
2018-01-01 15:56:55 +00:00
|
|
|
{
|
2018-01-04 10:07:42 +00:00
|
|
|
/* no filling is performed. also no validation about the range of
|
|
|
|
* source position is performed */
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sidx >= MOO_OBJ_GET_SIZE(rcv))
|
|
|
|
{
|
|
|
|
/* index out of range */
|
|
|
|
moo_seterrbfmt (moo, MOO_ERANGE, "source position out of bound - %zu", sidx);
|
2018-01-01 15:56:55 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
2018-01-02 16:25:37 +00:00
|
|
|
maxlen = MOO_OBJ_GET_SIZE(rcv) - sidx;
|
|
|
|
if (ssz > maxlen) ssz = maxlen;
|
|
|
|
end = sidx + ssz;
|
2018-01-01 15:56:55 +00:00
|
|
|
|
2018-01-02 16:25:37 +00:00
|
|
|
/* TODO: do i need to perform a range check? */
|
2018-01-01 15:56:55 +00:00
|
|
|
switch (MOO_OBJ_GET_FLAGS_TYPE(rcv))
|
|
|
|
{
|
|
|
|
case MOO_OBJ_TYPE_BYTE:
|
2018-01-02 16:25:37 +00:00
|
|
|
if (!MOO_OOP_IS_SMOOI(dval)) goto invalid_fill_value;
|
|
|
|
for (i = sidx; i < end; i++) ((moo_oop_byte_t)rcv)->slot[i] = MOO_OOP_TO_SMOOI(dval);
|
2018-01-01 15:56:55 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_CHAR:
|
2018-01-02 16:25:37 +00:00
|
|
|
if (!MOO_OOP_IS_CHAR(dval)) goto invalid_fill_value;
|
|
|
|
for (i = sidx; i < end; i++) ((moo_oop_char_t)rcv)->slot[i] = MOO_OOP_TO_CHAR(dval);
|
2018-01-01 15:56:55 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_HALFWORD:
|
2018-01-02 16:25:37 +00:00
|
|
|
if (!MOO_OOP_IS_SMOOI(dval)) goto invalid_fill_value;
|
|
|
|
for (i = sidx; i < end; i++) ((moo_oop_halfword_t)rcv)->slot[i] = MOO_OOP_TO_SMOOI(dval);
|
2018-01-01 15:56:55 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_WORD:
|
2018-01-02 16:25:37 +00:00
|
|
|
{
|
|
|
|
moo_oow_t dw;
|
|
|
|
if (moo_inttooow(moo, dval, &dw) <= 0) goto invalid_fill_value;
|
|
|
|
for (i = sidx; i < end; i++) ((moo_oop_word_t)rcv)->slot[i] = dw;
|
2018-01-01 15:56:55 +00:00
|
|
|
break;
|
2018-01-02 16:25:37 +00:00
|
|
|
}
|
2018-01-01 15:56:55 +00:00
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_OOP:
|
2018-01-02 16:25:37 +00:00
|
|
|
for (i = sidx; i < end; i++) ((moo_oop_oop_t)rcv)->slot[i] = dval;
|
2018-01-01 15:56:55 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
moo_seterrnum (moo, MOO_EINTERN);
|
|
|
|
return MOO_PF_HARD_FAILURE;
|
|
|
|
}
|
|
|
|
|
2018-01-02 16:25:37 +00:00
|
|
|
|
2018-01-04 10:07:42 +00:00
|
|
|
done:
|
2018-01-02 16:25:37 +00:00
|
|
|
#if defined(MOO_LIMIT_OBJ_SIZE)
|
|
|
|
MOO_ASSERT (moo, ssz <= MOO_SMOOI_MAX);
|
|
|
|
MOO_STACK_SETRET (moo, nargs, MOO_SMOOI_TO_OOP(ssz));
|
|
|
|
#else
|
|
|
|
if (ssz <= MOO_SMOOI_MAX)
|
|
|
|
{
|
|
|
|
MOO_STACK_SETRET (moo, nargs, MOO_SMOOI_TO_OOP(ssz));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
moo_oop_t v;
|
|
|
|
v = moo_oowtoint (moo, ssz);
|
|
|
|
if (!v) MOO_PF_FAILURE;
|
|
|
|
MOO_STACK_SETRET (moo, nargs, v);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
|
|
|
|
invalid_fill_value:
|
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "invalid fill value - %O", dval);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_basic_shift (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv, spos, dpos, slen;
|
|
|
|
moo_oow_t sidx, ssz, didx;
|
|
|
|
|
|
|
|
MOO_ASSERT (moo, nargs == 3);
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
if (!MOO_OOP_IS_POINTER(rcv))
|
|
|
|
{
|
|
|
|
/* the receiver is a special numeric object, not a normal pointer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EMSGRCV, "receiver not indexable - %O", rcv);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MOO_OBJ_GET_FLAGS_RDONLY(rcv))
|
|
|
|
{
|
|
|
|
moo_seterrbfmt (moo, MOO_EPERM, "now allowed to change a read-only object - %O", rcv);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
spos = MOO_STACK_GETARG(moo, nargs, 0);
|
|
|
|
dpos = MOO_STACK_GETARG(moo, nargs, 1);
|
|
|
|
slen = MOO_STACK_GETARG(moo, nargs, 2);
|
|
|
|
|
|
|
|
if (moo_inttooow(moo, spos, &sidx) <= 0)
|
|
|
|
{
|
|
|
|
/* negative integer or not integer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "invalid source position - %O", spos);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
2018-01-04 10:07:42 +00:00
|
|
|
if (moo_inttooow(moo, dpos, &didx) <= 0)
|
2018-01-02 16:25:37 +00:00
|
|
|
{
|
2018-01-04 10:07:42 +00:00
|
|
|
/* negative integer or not integer */
|
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "invalid destination position - %O", dpos);
|
2018-01-02 16:25:37 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
2018-01-04 10:07:42 +00:00
|
|
|
if (moo_inttooow(moo, slen, &ssz) <= 0)
|
2018-01-02 16:25:37 +00:00
|
|
|
{
|
|
|
|
/* negative integer or not integer */
|
2018-01-04 10:07:42 +00:00
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "invalid shift count - %O", slen);
|
2018-01-02 16:25:37 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
2018-01-04 10:07:42 +00:00
|
|
|
|
|
|
|
if (ssz <= 0)
|
|
|
|
{
|
|
|
|
/* no shifting is performed. also no validation about the range of
|
|
|
|
* source position and destionation position is performed */
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sidx >= MOO_OBJ_GET_SIZE(rcv))
|
2018-01-02 16:25:37 +00:00
|
|
|
{
|
|
|
|
/* index out of range */
|
2018-01-04 10:07:42 +00:00
|
|
|
moo_seterrbfmt (moo, MOO_ERANGE, "source position out of bound - %zu", sidx);
|
2018-01-02 16:25:37 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
2018-01-04 10:07:42 +00:00
|
|
|
if (didx >= MOO_OBJ_GET_SIZE(rcv))
|
2018-01-02 16:25:37 +00:00
|
|
|
{
|
2018-01-04 10:07:42 +00:00
|
|
|
/* index out of range */
|
|
|
|
moo_seterrbfmt (moo, MOO_ERANGE, "destination position out of bound - %zu", didx);
|
2018-01-02 16:25:37 +00:00
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
2018-01-04 10:07:42 +00:00
|
|
|
if (sidx != didx)
|
2018-01-02 16:25:37 +00:00
|
|
|
{
|
|
|
|
moo_oow_t maxlen;
|
|
|
|
|
|
|
|
maxlen = MOO_OBJ_GET_SIZE(rcv) - didx;
|
|
|
|
if (ssz > maxlen) ssz = maxlen;
|
|
|
|
|
|
|
|
switch (MOO_OBJ_GET_FLAGS_TYPE(rcv))
|
|
|
|
{
|
|
|
|
case MOO_OBJ_TYPE_BYTE:
|
|
|
|
MOO_MEMMOVE (&((moo_oop_byte_t)rcv)->slot[didx],
|
|
|
|
&((moo_oop_byte_t)rcv)->slot[sidx],
|
|
|
|
ssz * MOO_SIZEOF(((moo_oop_byte_t)rcv)->slot[0]));
|
|
|
|
|
|
|
|
if (didx > sidx)
|
|
|
|
{
|
|
|
|
MOO_MEMSET (&((moo_oop_byte_t)rcv)->slot[sidx], 0,
|
|
|
|
(didx - sidx) * MOO_SIZEOF(((moo_oop_byte_t)rcv)->slot[0]));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MOO_MEMSET (&((moo_oop_byte_t)rcv)->slot[didx + ssz - 1], 0,
|
|
|
|
(sidx - didx) * MOO_SIZEOF(((moo_oop_byte_t)rcv)->slot[0]));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_CHAR:
|
|
|
|
MOO_MEMMOVE (&((moo_oop_char_t)rcv)->slot[didx],
|
|
|
|
&((moo_oop_char_t)rcv)->slot[sidx],
|
|
|
|
ssz * MOO_SIZEOF(((moo_oop_char_t)rcv)->slot[0]));
|
|
|
|
|
|
|
|
if (didx > sidx)
|
|
|
|
{
|
|
|
|
MOO_MEMSET (&((moo_oop_char_t)rcv)->slot[sidx], 0,
|
|
|
|
(didx - sidx) * MOO_SIZEOF(((moo_oop_char_t)rcv)->slot[0]));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MOO_MEMSET (&((moo_oop_char_t)rcv)->slot[didx + ssz - 1], 0,
|
|
|
|
(sidx - didx) * MOO_SIZEOF(((moo_oop_char_t)rcv)->slot[0]));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_HALFWORD:
|
|
|
|
MOO_MEMMOVE (&((moo_oop_halfword_t)rcv)->slot[didx],
|
|
|
|
&((moo_oop_halfword_t)rcv)->slot[sidx],
|
|
|
|
ssz * MOO_SIZEOF(((moo_oop_halfword_t)rcv)->slot[0]));
|
|
|
|
|
|
|
|
if (didx > sidx)
|
|
|
|
{
|
|
|
|
MOO_MEMSET (&((moo_oop_halfword_t)rcv)->slot[sidx], 0,
|
|
|
|
(didx - sidx) * MOO_SIZEOF(((moo_oop_halfword_t)rcv)->slot[0]));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MOO_MEMSET (&((moo_oop_halfword_t)rcv)->slot[didx + ssz - 1], 0,
|
|
|
|
(sidx - didx) * MOO_SIZEOF(((moo_oop_halfword_t)rcv)->slot[0]));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_WORD:
|
|
|
|
MOO_MEMMOVE (&((moo_oop_word_t)rcv)->slot[didx],
|
|
|
|
&((moo_oop_word_t)rcv)->slot[sidx],
|
|
|
|
ssz * MOO_SIZEOF(((moo_oop_word_t)rcv)->slot[0]));
|
|
|
|
|
|
|
|
if (didx > sidx)
|
|
|
|
{
|
|
|
|
MOO_MEMSET (&((moo_oop_word_t)rcv)->slot[sidx], 0,
|
|
|
|
(didx - sidx) * MOO_SIZEOF(((moo_oop_word_t)rcv)->slot[0]));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MOO_MEMSET (&((moo_oop_word_t)rcv)->slot[didx + ssz - 1], 0,
|
|
|
|
(sidx - didx) * MOO_SIZEOF(((moo_oop_word_t)rcv)->slot[0]));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MOO_OBJ_TYPE_OOP:
|
|
|
|
MOO_MEMMOVE (&((moo_oop_oop_t)rcv)->slot[didx],
|
|
|
|
&((moo_oop_oop_t)rcv)->slot[sidx],
|
|
|
|
ssz * MOO_SIZEOF(((moo_oop_oop_t)rcv)->slot[0]));
|
|
|
|
|
|
|
|
if (didx > sidx)
|
|
|
|
{
|
|
|
|
while (sidx < didx) ((moo_oop_oop_t)rcv)->slot[sidx++] = moo->_nil;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (didx > sidx) ((moo_oop_oop_t)rcv)->slot[(didx++) + ssz] = moo->_nil;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
moo_seterrnum (moo, MOO_EINTERN);
|
|
|
|
return MOO_PF_HARD_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-04 10:07:42 +00:00
|
|
|
done:
|
2018-01-02 16:25:37 +00:00
|
|
|
#if defined(MOO_LIMIT_OBJ_SIZE)
|
|
|
|
MOO_ASSERT (moo, ssz <= MOO_SMOOI_MAX);
|
|
|
|
MOO_STACK_SETRET (moo, nargs, MOO_SMOOI_TO_OOP(ssz));
|
|
|
|
#else
|
|
|
|
if (ssz <= MOO_SMOOI_MAX)
|
|
|
|
{
|
|
|
|
MOO_STACK_SETRET (moo, nargs, MOO_SMOOI_TO_OOP(ssz));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
moo_oop_t v;
|
|
|
|
v = moo_oowtoint (moo, ssz);
|
|
|
|
if (!v) MOO_PF_FAILURE;
|
|
|
|
MOO_STACK_SETRET (moo, nargs, v);
|
|
|
|
}
|
|
|
|
#endif
|
2018-01-01 15:56:55 +00:00
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
2017-12-05 16:51:20 +00:00
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------
|
|
|
|
* BASIC QUERY
|
|
|
|
* -------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_is_kind_of (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv, _class;
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
_class = MOO_STACK_GETARG(moo, nargs, 0);
|
|
|
|
|
|
|
|
if (MOO_CLASSOF(moo, _class) == moo->_class &&
|
|
|
|
moo_iskindof(moo, rcv, (moo_oop_class_t)_class))
|
|
|
|
{
|
|
|
|
MOO_STACK_SETRET (moo, nargs, moo->_true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MOO_STACK_SETRET (moo, nargs, moo->_false);
|
|
|
|
}
|
|
|
|
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2017-12-08 15:28:51 +00:00
|
|
|
|
|
|
|
moo_pfrc_t moo_pf_responds_to (moo_t* moo, moo_ooi_t nargs)
|
|
|
|
{
|
|
|
|
moo_oop_t rcv, selector;
|
|
|
|
moo_oocs_t mthname;
|
|
|
|
|
|
|
|
rcv = MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
selector = MOO_STACK_GETARG(moo, nargs, 0);
|
|
|
|
|
|
|
|
/*if (MOO_CLASSOF(moo,selector) != moo->_symbol)*/
|
2017-12-09 15:47:43 +00:00
|
|
|
if (!MOO_OBJ_IS_CHAR_POINTER(selector))
|
2017-12-08 15:28:51 +00:00
|
|
|
{
|
|
|
|
moo_seterrbfmt (moo, MOO_EINVAL, "invalid argument - %O", selector);
|
|
|
|
return MOO_PF_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
mthname.ptr = MOO_OBJ_GET_CHAR_SLOT(selector);
|
|
|
|
mthname.len = MOO_OBJ_GET_SIZE(selector);
|
|
|
|
if (moo_findmethod (moo, rcv, &mthname, 0))
|
|
|
|
{
|
|
|
|
MOO_STACK_SETRET (moo, nargs, moo->_true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MOO_STACK_SETRET (moo, nargs, moo->_false);
|
|
|
|
}
|
|
|
|
|
|
|
|
return MOO_PF_SUCCESS;
|
|
|
|
}
|