2015-06-04 18:34:37 +00:00
/*
* $ Id $
*
2017-01-10 14:27:31 +00:00
Copyright ( c ) 2014 - 2017 Chung , Hyung - Hwan . All rights reserved .
2015-06-04 18:34:37 +00:00
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 .
*/
2017-01-09 09:54:49 +00:00
# include "moo-prv.h"
2015-06-04 18:34:37 +00:00
2016-03-22 14:18:07 +00:00
# define PROC_STATE_RUNNING 3
# define PROC_STATE_WAITING 2
# define PROC_STATE_RUNNABLE 1
# define PROC_STATE_SUSPENDED 0
# define PROC_STATE_TERMINATED -1
2017-08-22 13:45:37 +00:00
static MOO_INLINE const char * proc_state_to_string ( int state )
2017-08-17 18:10:29 +00:00
{
static const char * str [ ] =
{
" TERMINATED " ,
" SUSPENDED " ,
" RUNNABLE " ,
" WAITING " ,
" RUNNING "
} ;
2017-07-27 17:29:45 +00:00
2017-08-17 18:10:29 +00:00
return str [ state + 1 ] ;
}
2017-08-22 13:45:37 +00:00
2017-07-30 15:15:47 +00:00
/* TODO: adjust this process map increment value */
# define PROC_MAP_INC 64
2017-07-27 17:29:45 +00:00
2017-02-12 18:59:03 +00:00
/* TODO: adjust these max semaphore pointer buffer capacity,
* proably depending on the object memory size ? */
2016-03-22 14:18:07 +00:00
# define SEM_LIST_INC 256
# define SEM_HEAP_INC 256
2017-02-12 18:59:03 +00:00
# define SEM_IO_INC 256
2016-03-22 14:18:07 +00:00
# define SEM_LIST_MAX (SEM_LIST_INC * 1000)
# define SEM_HEAP_MAX (SEM_HEAP_INC * 1000)
2017-02-12 18:59:03 +00:00
# define SEM_IO_MAX (SEM_IO_INC * 1000)
2016-03-22 14:18:07 +00:00
# define SEM_HEAP_PARENT(x) (((x) - 1) / 2)
# define SEM_HEAP_LEFT(x) ((x) * 2 + 1)
# define SEM_HEAP_RIGHT(x) ((x) * 2 + 2)
# define SEM_HEAP_EARLIER_THAN(stx,x,y) ( \
2017-01-09 09:54:49 +00:00
( MOO_OOP_TO_SMOOI ( ( x ) - > heap_ftime_sec ) < MOO_OOP_TO_SMOOI ( ( y ) - > heap_ftime_sec ) ) | | \
( MOO_OOP_TO_SMOOI ( ( x ) - > heap_ftime_sec ) = = MOO_OOP_TO_SMOOI ( ( y ) - > heap_ftime_sec ) & & MOO_OOP_TO_SMOOI ( ( x ) - > heap_ftime_nsec ) < MOO_OOP_TO_SMOOI ( ( y ) - > heap_ftime_nsec ) ) \
2016-03-22 14:18:07 +00:00
)
2016-02-15 18:07:20 +00:00
2017-01-09 09:54:49 +00:00
# define LOAD_IP(moo, v_ctx) ((moo)->ip = MOO_OOP_TO_SMOOI((v_ctx)->ip))
# define STORE_IP(moo, v_ctx) ((v_ctx)->ip = MOO_SMOOI_TO_OOP((moo)->ip))
2015-06-12 13:52:30 +00:00
2017-01-09 09:54:49 +00:00
# define LOAD_SP(moo, v_ctx) ((moo)->sp = MOO_OOP_TO_SMOOI((v_ctx)->sp))
# define STORE_SP(moo, v_ctx) ((v_ctx)->sp = MOO_SMOOI_TO_OOP((moo)->sp))
2015-06-16 04:31:28 +00:00
2017-01-09 09:54:49 +00:00
# define LOAD_ACTIVE_IP(moo) LOAD_IP(moo, (moo)->active_context)
# define STORE_ACTIVE_IP(moo) STORE_IP(moo, (moo)->active_context)
2015-06-16 04:31:28 +00:00
2017-01-09 09:54:49 +00:00
# define LOAD_ACTIVE_SP(moo) LOAD_SP(moo, (moo)->processor->active)
# define STORE_ACTIVE_SP(moo) STORE_SP(moo, (moo)->processor->active)
2015-06-04 18:34:37 +00:00
2017-01-09 09:54:49 +00:00
# define SWITCH_ACTIVE_CONTEXT(moo,v_ctx) \
2017-02-12 18:59:03 +00:00
do { \
2017-01-09 09:54:49 +00:00
STORE_ACTIVE_IP ( moo ) ; \
( moo ) - > active_context = ( v_ctx ) ; \
( moo ) - > active_method = ( moo_oop_method_t ) ( moo ) - > active_context - > origin - > method_or_nargs ; \
SET_ACTIVE_METHOD_CODE ( moo ) ; \
LOAD_ACTIVE_IP ( moo ) ; \
( moo ) - > processor - > active - > current_context = ( moo ) - > active_context ; \
2016-06-24 15:01:51 +00:00
} while ( 0 )
2015-06-20 03:07:11 +00:00
2017-01-09 09:54:49 +00:00
# define FETCH_BYTE_CODE(moo) ((moo)->active_code[(moo)->ip++])
# if (MOO_BCODE_LONG_PARAM_SIZE == 2)
# define FETCH_PARAM_CODE_TO(moo, v_ooi) \
2016-02-12 16:23:26 +00:00
do { \
2017-01-09 09:54:49 +00:00
v_ooi = FETCH_BYTE_CODE ( moo ) ; \
v_ooi = ( v_ooi < < 8 ) | FETCH_BYTE_CODE ( moo ) ; \
2016-02-12 16:23:26 +00:00
} while ( 0 )
2015-07-01 15:01:39 +00:00
# else
2017-01-09 09:54:49 +00:00
# define FETCH_PARAM_CODE_TO(moo, v_ooi) (v_ooi = FETCH_BYTE_CODE(moo))
2015-07-01 15:01:39 +00:00
# endif
2017-01-09 09:54:49 +00:00
# if defined(MOO_DEBUG_VM_EXEC)
# define LOG_MASK_INST (MOO_LOG_IC | MOO_LOG_MNEMONIC)
2016-06-05 18:01:35 +00:00
2016-10-06 14:17:24 +00:00
/* TODO: for send_message, display the method name. or include the method name before 'ip' */
2017-07-20 16:33:53 +00:00
# define LOG_INST_0(moo,fmt) MOO_LOG1(moo, LOG_MASK_INST, " %06zd " fmt "\n", (moo)->last_inst_pointer)
# define LOG_INST_1(moo,fmt,a1) MOO_LOG2(moo, LOG_MASK_INST, " %06zd " fmt "\n",(moo)->last_inst_pointer, a1)
# define LOG_INST_2(moo,fmt,a1,a2) MOO_LOG3(moo, LOG_MASK_INST, " %06zd " fmt "\n", (moo)->last_inst_pointer, a1, a2)
# define LOG_INST_3(moo,fmt,a1,a2,a3) MOO_LOG4(moo, LOG_MASK_INST, " %06zd " fmt "\n", (moo)->last_inst_pointer, a1, a2, a3)
2015-10-08 14:26:04 +00:00
# else
2017-01-09 09:54:49 +00:00
# define LOG_INST_0(moo,fmt)
# define LOG_INST_1(moo,fmt,a1)
# define LOG_INST_2(moo,fmt,a1,a2)
# define LOG_INST_3(moo,fmt,a1,a2,a3)
2015-10-08 14:26:04 +00:00
# endif
2017-01-15 17:53:37 +00:00
# if defined(__DOS__) && (defined(_INTELC32_) || (defined(__WATCOMC__) && (__WATCOMC__ <= 1000)))
2017-01-09 14:52:15 +00:00
/* the old intel c code builder doesn't support __FUNCTION__ */
# define __PRIMITIVE_NAME__ "<<primitive>>"
# else
2017-04-24 09:20:27 +00:00
# define __PRIMITIVE_NAME__ (&__FUNCTION__[0])
2017-01-09 14:52:15 +00:00
# endif
2016-06-26 15:03:12 +00:00
2017-04-08 07:23:31 +00:00
static void signal_io_semaphore ( moo_t * moo , moo_ooi_t mask , void * ctx ) ;
2017-03-19 14:18:37 +00:00
static int send_message ( moo_t * moo , moo_oop_char_t selector , int to_super , moo_ooi_t nargs ) ;
2017-05-11 14:59:20 +00:00
static int send_message_with_str ( moo_t * moo , const moo_ooch_t * nameptr , moo_oow_t namelen , int to_super , moo_ooi_t nargs ) ;
2017-02-15 11:57:24 +00:00
2016-05-12 05:53:35 +00:00
/* ------------------------------------------------------------------------- */
2017-02-12 18:59:03 +00:00
static MOO_INLINE int vm_startup ( moo_t * moo )
{
2017-07-20 16:33:53 +00:00
MOO_DEBUG0 ( moo , " VM started up \n " ) ;
2017-02-12 18:59:03 +00:00
if ( moo - > vmprim . vm_startup ( moo ) < = - 1 ) return - 1 ;
moo - > vmprim . vm_gettime ( moo , & moo - > exec_start_time ) ; /* raw time. no adjustment */
return 0 ;
}
static MOO_INLINE void vm_cleanup ( moo_t * moo )
{
moo - > vmprim . vm_gettime ( moo , & moo - > exec_end_time ) ; /* raw time. no adjustment */
moo - > vmprim . vm_cleanup ( moo ) ;
2017-07-20 16:33:53 +00:00
MOO_DEBUG0 ( moo , " VM cleaned up \n " ) ;
2017-02-12 18:59:03 +00:00
}
2017-01-09 09:54:49 +00:00
static MOO_INLINE void vm_gettime ( moo_t * moo , moo_ntime_t * now )
2016-03-24 14:58:47 +00:00
{
2017-02-10 09:06:53 +00:00
moo - > vmprim . vm_gettime ( moo , now ) ;
/* in vm_startup(), moo->exec_start_time has been set to the time of
2017-02-12 18:59:03 +00:00
* that moment . time returned here get offset by moo - > exec_start_time and
* thus becomes relative to it . this way , it is kept small such that it
* can be represented in a small integer with leaving almost zero chance
* of overflow . */
2017-02-10 09:06:53 +00:00
MOO_SUBNTIME ( now , now , & moo - > exec_start_time ) ; /* now = now - exec_start_time */
2016-03-24 14:58:47 +00:00
}
2017-01-09 09:54:49 +00:00
static MOO_INLINE void vm_sleep ( moo_t * moo , const moo_ntime_t * dur )
2016-03-24 14:58:47 +00:00
{
2017-02-10 09:06:53 +00:00
moo - > vmprim . vm_sleep ( moo , dur ) ;
2016-03-24 14:58:47 +00:00
}
2017-03-25 05:16:18 +00:00
static MOO_INLINE void vm_muxwait ( moo_t * moo , const moo_ntime_t * dur )
2016-06-24 15:53:00 +00:00
{
2017-02-22 09:48:58 +00:00
moo - > vmprim . vm_muxwait ( moo , dur , signal_io_semaphore ) ;
2016-06-24 15:53:00 +00:00
}
2016-05-12 05:53:35 +00:00
/* ------------------------------------------------------------------------- */
2017-07-30 15:15:47 +00:00
static MOO_INLINE int prepare_to_alloc_pid ( moo_t * moo )
{
moo_oow_t new_capa ;
moo_ooi_t i , j ;
moo_oop_t * tmp ;
MOO_ASSERT ( moo , moo - > proc_map_free_first < = - 1 ) ;
MOO_ASSERT ( moo , moo - > proc_map_free_last < = - 1 ) ;
new_capa = moo - > proc_map_capa + PROC_MAP_INC ;
if ( new_capa > MOO_SMOOI_MAX )
{
if ( moo - > proc_map_capa > = MOO_SMOOI_MAX )
{
# if defined(MOO_DEBUG_VM_PROCESSOR)
MOO_LOG0 ( moo , MOO_LOG_IC | MOO_LOG_FATAL , " Processor - too many processes \n " ) ;
# endif
moo_seterrnum ( moo , MOO_EPFULL ) ;
return - 1 ;
}
new_capa = MOO_SMOOI_MAX ;
}
tmp = moo_reallocmem ( moo , moo - > proc_map , MOO_SIZEOF ( moo_oop_t ) * new_capa ) ;
if ( ! tmp ) return - 1 ;
moo - > proc_map_free_first = moo - > proc_map_capa ;
for ( i = moo - > proc_map_capa , j = moo - > proc_map_capa + 1 ; j < new_capa ; i + + , j + + )
{
tmp [ i ] = MOO_SMOOI_TO_OOP ( j ) ;
}
tmp [ i ] = MOO_SMOOI_TO_OOP ( - 1 ) ;
moo - > proc_map_free_last = i ;
moo - > proc_map = tmp ;
moo - > proc_map_capa = new_capa ;
return 0 ;
}
static MOO_INLINE void alloc_pid ( moo_t * moo , moo_oop_process_t proc )
{
moo_ooi_t pid ;
pid = moo - > proc_map_free_first ;
proc - > id = MOO_SMOOI_TO_OOP ( pid ) ;
MOO_ASSERT ( moo , MOO_OOP_IS_SMOOI ( moo - > proc_map [ pid ] ) ) ;
moo - > proc_map_free_first = MOO_OOP_TO_SMOOI ( moo - > proc_map [ pid ] ) ;
if ( moo - > proc_map_free_first < = - 1 ) moo - > proc_map_free_last = - 1 ;
moo - > proc_map [ pid ] = ( moo_oop_t ) proc ;
}
static MOO_INLINE void free_pid ( moo_t * moo , moo_oop_process_t proc )
{
moo_ooi_t pid ;
pid = MOO_OOP_TO_SMOOI ( proc - > id ) ;
MOO_ASSERT ( moo , pid < moo - > proc_map_capa ) ;
moo - > proc_map [ pid ] = MOO_SMOOI_TO_OOP ( - 1 ) ;
if ( moo - > proc_map_free_last < = - 1 )
{
MOO_ASSERT ( moo , moo - > proc_map_free_first < = - 1 ) ;
moo - > proc_map_free_first = pid ;
}
else
{
moo - > proc_map [ moo - > proc_map_free_last ] = MOO_SMOOI_TO_OOP ( pid ) ;
}
moo - > proc_map_free_last = pid ;
}
2017-01-09 09:54:49 +00:00
static moo_oop_process_t make_process ( moo_t * moo , moo_oop_context_t c )
2015-10-18 15:06:17 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_process_t proc ;
2017-07-25 15:26:04 +00:00
moo_ooi_t total_count ;
moo_ooi_t suspended_count ;
total_count = MOO_OOP_TO_SMOOI ( moo - > processor - > total_count ) ;
if ( total_count > = MOO_SMOOI_MAX )
{
# if defined(MOO_DEBUG_VM_PROCESSOR)
MOO_LOG0 ( moo , MOO_LOG_IC | MOO_LOG_FATAL , " Processor - too many processes \n " ) ;
# endif
moo_seterrnum ( moo , MOO_EPFULL ) ;
return MOO_NULL ;
}
2015-10-18 15:06:17 +00:00
2017-07-30 15:15:47 +00:00
if ( moo - > proc_map_free_first < = - 1 & & prepare_to_alloc_pid ( moo ) < = - 1 ) return MOO_NULL ;
2017-07-27 17:29:45 +00:00
2017-01-09 09:54:49 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & c ) ;
proc = ( moo_oop_process_t ) moo_instantiate ( moo , moo - > _process , MOO_NULL , moo - > option . dfl_procstk_size ) ;
moo_poptmp ( moo ) ;
if ( ! proc ) return MOO_NULL ;
2015-10-18 15:06:17 +00:00
2017-01-09 09:54:49 +00:00
proc - > state = MOO_SMOOI_TO_OOP ( PROC_STATE_SUSPENDED ) ;
2017-07-24 13:25:25 +00:00
2017-07-27 17:29:45 +00:00
/* assign a process id to the process */
2017-07-30 15:15:47 +00:00
alloc_pid ( moo , proc ) ;
2017-07-24 13:25:25 +00:00
2015-10-19 06:16:43 +00:00
proc - > initial_context = c ;
2016-02-19 15:52:56 +00:00
proc - > current_context = c ;
2017-01-09 09:54:49 +00:00
proc - > sp = MOO_SMOOI_TO_OOP ( - 1 ) ;
2017-04-25 15:20:58 +00:00
proc - > perr = MOO_ERROR_TO_OOP ( MOO_ENOERR ) ;
2017-05-16 02:04:18 +00:00
proc - > perrmsg = moo - > _nil ;
2015-10-18 15:06:17 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , ( moo_oop_t ) c - > sender = = moo - > _nil ) ;
2016-05-02 13:50:42 +00:00
2017-01-09 09:54:49 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR)
2017-08-20 14:43:34 +00:00
MOO_LOG2 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - process[%zd] **CREATED**->%hs \n " , MOO_OOP_TO_SMOOI ( proc - > id ) , proc_state_to_string ( MOO_OOP_TO_SMOOI ( proc - > state ) ) ) ;
2016-02-12 16:23:26 +00:00
# endif
2017-07-21 16:54:43 +00:00
2017-07-25 15:26:04 +00:00
/* a process is created in the SUSPENDED state. chain it to the suspended process list */
suspended_count = MOO_OOP_TO_SMOOI ( moo - > processor - > suspended . count ) ;
MOO_APPEND_TO_OOP_LIST ( moo , & moo - > processor - > suspended , moo_oop_process_t , proc , ps ) ;
suspended_count + + ;
moo - > processor - > suspended . count = MOO_SMOOI_TO_OOP ( suspended_count ) ;
total_count + + ;
moo - > processor - > total_count = MOO_SMOOI_TO_OOP ( total_count ) ;
2015-10-18 15:06:17 +00:00
return proc ;
}
2017-01-09 09:54:49 +00:00
static MOO_INLINE void sleep_active_process ( moo_t * moo , int state )
2015-10-19 06:16:43 +00:00
{
2017-01-09 09:54:49 +00:00
STORE_ACTIVE_SP ( moo ) ;
2016-02-12 16:23:26 +00:00
2016-02-19 15:52:56 +00:00
/* store the current active context to the current process.
* it is the suspended context of the process to be suspended */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > processor - > active ! = moo - > nil_process ) ;
2017-08-17 18:10:29 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR)
2017-08-20 14:43:34 +00:00
MOO_LOG3 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - process [%zd] %hs->%hs in sleep_active_process \n " , MOO_OOP_TO_SMOOI ( moo - > processor - > active - > id ) , proc_state_to_string ( MOO_OOP_TO_SMOOI ( moo - > processor - > active - > state ) ) , proc_state_to_string ( state ) ) ;
2017-08-17 18:10:29 +00:00
# endif
2017-01-09 09:54:49 +00:00
moo - > processor - > active - > current_context = moo - > active_context ;
moo - > processor - > active - > state = MOO_SMOOI_TO_OOP ( state ) ;
2016-03-23 02:34:13 +00:00
}
2016-02-12 16:23:26 +00:00
2017-07-25 15:26:04 +00:00
static MOO_INLINE void wake_process ( moo_t * moo , moo_oop_process_t proc )
2016-03-23 02:34:13 +00:00
{
2016-02-19 15:52:56 +00:00
/* activate the given process */
2017-08-17 18:10:29 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR)
2017-08-20 14:43:34 +00:00
MOO_LOG2 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - process [%zd] %hs->RUNNING in wake_process \n " , MOO_OOP_TO_SMOOI ( proc - > id ) , proc_state_to_string ( MOO_OOP_TO_SMOOI ( proc - > state ) ) ) ;
2017-08-17 18:10:29 +00:00
# endif
MOO_ASSERT ( moo , proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNABLE ) ) ;
2017-01-09 09:54:49 +00:00
proc - > state = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNING ) ;
moo - > processor - > active = proc ;
2016-02-12 16:23:26 +00:00
2017-09-25 15:16:19 +00:00
/* load the stack pointer from 'proc'.
* moo - > processor - > active points to ' proc ' now . */
LOAD_ACTIVE_SP ( moo ) ;
2015-10-19 06:16:43 +00:00
2016-02-19 15:52:56 +00:00
/* activate the suspended context of the new process */
2017-01-09 09:54:49 +00:00
SWITCH_ACTIVE_CONTEXT ( moo , proc - > current_context ) ;
2016-02-12 16:23:26 +00:00
2017-08-20 14:43:34 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR) && (MOO_DEBUG_VM_PROCESSOR >= 2)
2017-07-25 15:26:04 +00:00
MOO_LOG3 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - woke up process [%zd] context %O ip=%zd \n " , MOO_OOP_TO_SMOOI ( moo - > processor - > active - > id ) , moo - > active_context , moo - > ip ) ;
2015-12-02 16:14:37 +00:00
# endif
2016-03-23 02:34:13 +00:00
}
2017-01-09 09:54:49 +00:00
static void switch_to_process ( moo_t * moo , moo_oop_process_t proc , int new_state_for_old_active )
2016-03-23 02:34:13 +00:00
{
/* the new process must not be the currently active process */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > processor - > active ! = proc ) ;
2016-03-23 02:34:13 +00:00
/* the new process must be in the runnable state */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNABLE ) | |
2017-07-24 13:25:25 +00:00
proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_WAITING ) ) ;
2016-03-23 02:34:13 +00:00
2017-01-09 09:54:49 +00:00
sleep_active_process ( moo , new_state_for_old_active ) ;
2017-07-25 15:26:04 +00:00
wake_process ( moo , proc ) ;
2016-02-19 15:52:56 +00:00
2017-01-09 09:54:49 +00:00
moo - > proc_switched = 1 ;
2015-10-19 06:16:43 +00:00
}
2017-07-25 15:26:04 +00:00
static MOO_INLINE void switch_to_process_from_nil ( moo_t * moo , moo_oop_process_t proc )
{
MOO_ASSERT ( moo , moo - > processor - > active = = moo - > nil_process ) ;
wake_process ( moo , proc ) ;
moo - > proc_switched = 1 ;
}
2017-01-09 09:54:49 +00:00
static MOO_INLINE moo_oop_process_t find_next_runnable_process ( moo_t * moo )
2015-10-19 06:16:43 +00:00
{
2017-07-24 13:25:25 +00:00
moo_oop_process_t nrp ;
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > processor - > active - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNING ) ) ;
2017-07-24 13:25:25 +00:00
nrp = moo - > processor - > active - > ps . next ;
if ( ( moo_oop_t ) nrp = = moo - > _nil ) nrp = moo - > processor - > runnable . first ;
return nrp ;
2016-02-19 15:52:56 +00:00
}
2017-01-09 09:54:49 +00:00
static MOO_INLINE void switch_to_next_runnable_process ( moo_t * moo )
2016-02-19 15:52:56 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_process_t nrp ;
nrp = find_next_runnable_process ( moo ) ;
if ( nrp ! = moo - > processor - > active ) switch_to_process ( moo , nrp , PROC_STATE_RUNNABLE ) ;
2015-10-19 06:16:43 +00:00
}
2017-07-25 15:26:04 +00:00
static MOO_INLINE void chain_into_processor ( moo_t * moo , moo_oop_process_t proc , int new_state )
2015-10-18 15:06:17 +00:00
{
2015-12-02 15:24:13 +00:00
/* the process is not scheduled at all.
* link it to the processor ' s process list . */
2017-07-24 13:25:25 +00:00
moo_ooi_t runnable_count ;
2017-07-25 15:26:04 +00:00
moo_ooi_t suspended_count ;
2015-10-18 15:06:17 +00:00
2017-07-24 13:25:25 +00:00
/*MOO_ASSERT (moo, (moo_oop_t)proc->ps.prev == moo->_nil);
MOO_ASSERT ( moo , ( moo_oop_t ) proc - > ps . next = = moo - > _nil ) ; */
2016-02-15 18:07:20 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_SUSPENDED ) ) ;
2017-07-25 15:26:04 +00:00
MOO_ASSERT ( moo , new_state = = PROC_STATE_RUNNABLE | | new_state = = PROC_STATE_RUNNING ) ;
2015-12-02 15:24:13 +00:00
2017-08-20 14:43:34 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR)
2017-09-25 15:16:19 +00:00
MOO_LOG3 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG ,
" Processor - process [%zd] %hs->%hs in chain_into_processor \n " ,
MOO_OOP_TO_SMOOI ( proc - > id ) ,
proc_state_to_string ( MOO_OOP_TO_SMOOI ( proc - > state ) ) ,
proc_state_to_string ( new_state ) ) ;
2017-08-20 14:43:34 +00:00
# endif
2017-07-24 13:25:25 +00:00
runnable_count = MOO_OOP_TO_SMOOI ( moo - > processor - > runnable . count ) ;
2016-02-12 16:23:26 +00:00
2017-07-24 13:25:25 +00:00
MOO_ASSERT ( moo , runnable_count > = 0 ) ;
2016-02-18 17:49:56 +00:00
2017-07-25 15:26:04 +00:00
suspended_count = MOO_OOP_TO_SMOOI ( moo - > processor - > suspended . count ) ;
MOO_DELETE_FROM_OOP_LIST ( moo , & moo - > processor - > suspended , proc , ps ) ;
suspended_count - - ;
moo - > processor - > suspended . count = MOO_SMOOI_TO_OOP ( suspended_count ) ;
2016-02-19 15:52:56 +00:00
2017-07-24 13:25:25 +00:00
/* append to the runnable list */
MOO_APPEND_TO_OOP_LIST ( moo , & moo - > processor - > runnable , moo_oop_process_t , proc , ps ) ;
2017-07-25 15:26:04 +00:00
proc - > state = MOO_SMOOI_TO_OOP ( new_state ) ;
2017-07-24 13:25:25 +00:00
runnable_count + + ;
moo - > processor - > runnable . count = MOO_SMOOI_TO_OOP ( runnable_count ) ;
2015-12-02 15:24:13 +00:00
}
2015-10-19 06:16:43 +00:00
2017-07-25 15:26:04 +00:00
static MOO_INLINE void unchain_from_processor ( moo_t * moo , moo_oop_process_t proc , int new_state )
2016-02-18 17:49:56 +00:00
{
2017-07-24 13:25:25 +00:00
moo_ooi_t runnable_count ;
2017-07-25 15:26:04 +00:00
moo_ooi_t suspended_count ;
moo_ooi_t total_count ;
2016-02-18 17:49:56 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNING ) | |
2017-08-17 18:10:29 +00:00
proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNABLE ) | |
proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_SUSPENDED ) ) ;
2017-07-24 13:25:25 +00:00
2017-08-17 18:10:29 +00:00
MOO_ASSERT ( moo , proc - > state ! = MOO_SMOOI_TO_OOP ( new_state ) ) ;
2017-07-24 13:25:25 +00:00
2017-08-20 14:43:34 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR)
MOO_LOG3 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - process [%zd] %hs->%hs in unchain_from_processor \n " , MOO_OOP_TO_SMOOI ( proc - > id ) , proc_state_to_string ( MOO_OOP_TO_SMOOI ( proc - > state ) ) , proc_state_to_string ( MOO_OOP_TO_SMOOI ( new_state ) ) ) ;
# endif
2017-08-17 18:10:29 +00:00
if ( proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_SUSPENDED ) )
{
suspended_count = MOO_OOP_TO_SMOOI ( moo - > processor - > suspended . count ) ;
MOO_ASSERT ( moo , suspended_count > 0 ) ;
MOO_DELETE_FROM_OOP_LIST ( moo , & moo - > processor - > suspended , proc , ps ) ;
suspended_count - - ;
moo - > processor - > suspended . count = MOO_SMOOI_TO_OOP ( suspended_count ) ;
}
else
{
runnable_count = MOO_OOP_TO_SMOOI ( moo - > processor - > runnable . count ) ;
MOO_ASSERT ( moo , runnable_count > 0 ) ;
MOO_DELETE_FROM_OOP_LIST ( moo , & moo - > processor - > runnable , proc , ps ) ;
runnable_count - - ;
moo - > processor - > runnable . count = MOO_SMOOI_TO_OOP ( runnable_count ) ;
if ( runnable_count = = 0 ) moo - > processor - > active = moo - > nil_process ;
}
2017-07-24 13:25:25 +00:00
2017-07-25 15:26:04 +00:00
if ( new_state = = PROC_STATE_TERMINATED )
2017-07-24 13:25:25 +00:00
{
2017-07-25 15:26:04 +00:00
/* do not chain it to the suspended process list as it's being terminated */
proc - > ps . prev = ( moo_oop_process_t ) moo - > _nil ;
proc - > ps . next = ( moo_oop_process_t ) moo - > _nil ;
2016-02-18 17:49:56 +00:00
2017-07-25 15:26:04 +00:00
total_count = MOO_OOP_TO_SMOOI ( moo - > processor - > total_count ) ;
total_count - - ;
moo - > processor - > total_count = MOO_SMOOI_TO_OOP ( total_count ) ;
}
else
{
/* append to the suspended process list */
MOO_ASSERT ( moo , new_state = = PROC_STATE_SUSPENDED ) ;
2016-02-18 17:49:56 +00:00
2017-07-25 15:26:04 +00:00
suspended_count = MOO_OOP_TO_SMOOI ( moo - > processor - > suspended . count ) ;
2017-07-24 13:25:25 +00:00
MOO_APPEND_TO_OOP_LIST ( moo , & moo - > processor - > suspended , moo_oop_process_t , proc , ps ) ;
suspended_count + + ;
moo - > processor - > suspended . count = MOO_SMOOI_TO_OOP ( suspended_count ) ;
}
2016-02-29 17:26:40 +00:00
2017-07-25 15:26:04 +00:00
proc - > state = MOO_SMOOI_TO_OOP ( new_state ) ;
2016-02-18 17:49:56 +00:00
}
2017-01-09 09:54:49 +00:00
static MOO_INLINE void chain_into_semaphore ( moo_t * moo , moo_oop_process_t proc , moo_oop_semaphore_t sem )
2016-02-29 17:26:40 +00:00
{
2017-09-21 07:56:51 +00:00
/* append a process to the process list of a semaphore or a semaphore group */
2016-02-29 17:26:40 +00:00
2017-09-21 07:56:51 +00:00
/* a process chained to a semaphore cannot get chained to
* a semaphore again . a process can get chained to a single semaphore
* or a single semaphore group only */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , ( moo_oop_t ) proc - > sem = = moo - > _nil ) ;
2017-07-24 13:25:25 +00:00
MOO_ASSERT ( moo , ( moo_oop_t ) proc - > sem_wait . prev = = moo - > _nil ) ;
MOO_ASSERT ( moo , ( moo_oop_t ) proc - > sem_wait . next = = moo - > _nil ) ;
2016-02-29 17:26:40 +00:00
2017-09-21 14:22:36 +00:00
/* a semaphore or a semaphore group must be given for process chaining */
2017-10-02 01:22:49 +00:00
MOO_ASSERT ( moo , moo_iskindof ( moo , ( moo_oop_t ) sem , moo - > _semaphore ) | |
moo_iskindof ( moo , ( moo_oop_t ) sem , moo - > _semaphore_group ) ) ;
2017-09-21 07:56:51 +00:00
/* i assume the head part of the semaphore has the same layout as
* the semaphore group */
MOO_ASSERT ( moo , MOO_OFFSETOF ( moo_semaphore_t , waiting ) = =
MOO_OFFSETOF ( moo_semaphore_group_t , waiting ) ) ;
2017-07-24 13:25:25 +00:00
MOO_APPEND_TO_OOP_LIST ( moo , & sem - > waiting , moo_oop_process_t , proc , sem_wait ) ;
2016-02-29 17:26:40 +00:00
2017-09-21 07:56:51 +00:00
proc - > sem = ( moo_oop_t ) sem ;
2016-02-29 17:26:40 +00:00
}
2017-01-09 09:54:49 +00:00
static MOO_INLINE void unchain_from_semaphore ( moo_t * moo , moo_oop_process_t proc )
2016-02-29 17:26:40 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_semaphore_t sem ;
2016-02-29 17:26:40 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , ( moo_oop_t ) proc - > sem ! = moo - > _nil ) ;
2016-02-29 17:26:40 +00:00
2017-10-02 01:22:49 +00:00
MOO_ASSERT ( moo , moo_iskindof ( moo , proc - > sem , moo - > _semaphore ) | |
moo_iskindof ( moo , proc - > sem , moo - > _semaphore_group ) ) ;
2017-09-21 07:56:51 +00:00
MOO_ASSERT ( moo , MOO_OFFSETOF ( moo_semaphore_t , waiting ) = =
MOO_OFFSETOF ( moo_semaphore_group_t , waiting ) ) ;
/* proc->sem may be one of a semaphore or a semaphore group.
* i assume that ' waiting ' is defined to the same position
* in both Semaphore and SemaphoreGroup . there is no need to
* write different code for each class . */
sem = ( moo_oop_semaphore_t ) proc - > sem ; /* semgrp = (moo_oop_semaphore_group_t)proc->sem */
MOO_DELETE_FROM_OOP_LIST ( moo , & sem - > waiting , proc , sem_wait ) ;
2016-03-22 14:18:07 +00:00
2017-07-24 13:25:25 +00:00
proc - > sem_wait . prev = ( moo_oop_process_t ) moo - > _nil ;
proc - > sem_wait . next = ( moo_oop_process_t ) moo - > _nil ;
2017-09-21 07:56:51 +00:00
proc - > sem = moo - > _nil ;
2016-02-29 17:26:40 +00:00
}
2017-01-09 09:54:49 +00:00
static void terminate_process ( moo_t * moo , moo_oop_process_t proc )
2016-02-13 17:39:11 +00:00
{
2017-01-09 09:54:49 +00:00
if ( proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNING ) | |
proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNABLE ) )
2016-02-13 17:39:11 +00:00
{
2016-02-19 15:52:56 +00:00
/* RUNNING/RUNNABLE ---> TERMINATED */
2017-01-09 09:54:49 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR)
2017-08-20 14:43:34 +00:00
MOO_LOG2 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - process [%zd] %hs->TERMINATED in terminate_process \n " , MOO_OOP_TO_SMOOI ( proc - > id ) , proc_state_to_string ( MOO_OOP_TO_SMOOI ( proc - > state ) ) ) ;
2016-06-05 18:01:35 +00:00
# endif
2017-01-09 09:54:49 +00:00
if ( proc = = moo - > processor - > active )
2016-02-19 15:52:56 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_process_t nrp ;
2016-02-13 17:39:11 +00:00
2017-01-09 09:54:49 +00:00
nrp = find_next_runnable_process ( moo ) ;
2016-02-13 17:39:11 +00:00
2017-01-09 09:54:49 +00:00
unchain_from_processor ( moo , proc , PROC_STATE_TERMINATED ) ;
proc - > sp = MOO_SMOOI_TO_OOP ( - 1 ) ; /* invalidate the process stack */
2016-02-19 15:52:56 +00:00
proc - > current_context = proc - > initial_context ; /* not needed but just in case */
2016-02-13 17:39:11 +00:00
2016-02-29 17:26:40 +00:00
/* a runnable or running process must not be chanined to the
* process list of a semaphore */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , ( moo_oop_t ) proc - > sem = = moo - > _nil ) ;
2016-02-13 17:39:11 +00:00
2016-02-19 15:52:56 +00:00
if ( nrp = = proc )
{
/* no runnable process after termination */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > processor - > active = = moo - > nil_process ) ;
2017-11-21 09:15:22 +00:00
MOO_LOG5 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG ,
" No runnable process after termination of process %zd - total %zd runnable/running %zd suspended %zd - sem_io_wait_count %zu \n " ,
MOO_OOP_TO_SMOOI ( proc - > id ) ,
MOO_OOP_TO_SMOOI ( moo - > processor - > total_count ) ,
MOO_OOP_TO_SMOOI ( moo - > processor - > runnable . count ) ,
MOO_OOP_TO_SMOOI ( moo - > processor - > suspended . count ) ,
moo - > sem_io_wait_count
) ;
2016-02-19 15:52:56 +00:00
}
else
{
2017-01-09 09:54:49 +00:00
switch_to_process ( moo , nrp , PROC_STATE_TERMINATED ) ;
2016-02-19 15:52:56 +00:00
}
2016-02-13 17:39:11 +00:00
}
else
{
2017-01-09 09:54:49 +00:00
unchain_from_processor ( moo , proc , PROC_STATE_TERMINATED ) ;
proc - > sp = MOO_SMOOI_TO_OOP ( - 1 ) ; /* invalidate the process stack */
2016-02-13 17:39:11 +00:00
}
2017-07-27 17:29:45 +00:00
2017-07-30 15:15:47 +00:00
/* when terminated, clear it from the pid table and set the process id to a negative number */
free_pid ( moo , proc ) ;
2016-02-19 15:52:56 +00:00
}
2017-01-09 09:54:49 +00:00
else if ( proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_SUSPENDED ) )
2016-02-19 15:52:56 +00:00
{
/* SUSPENDED ---> TERMINATED */
2017-01-09 09:54:49 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR)
2017-08-20 14:43:34 +00:00
MOO_LOG2 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - process [%zd] %hs->TERMINATED in terminate_process \n " , MOO_OOP_TO_SMOOI ( proc - > id ) , proc_state_to_string ( MOO_OOP_TO_SMOOI ( proc - > state ) ) ) ;
2016-06-05 18:01:35 +00:00
# endif
2017-07-25 15:26:04 +00:00
/*proc->state = MOO_SMOOI_TO_OOP(PROC_STATE_TERMINATED);*/
unchain_from_processor ( moo , proc , PROC_STATE_TERMINATED ) ;
2017-01-09 09:54:49 +00:00
proc - > sp = MOO_SMOOI_TO_OOP ( - 1 ) ; /* invalidate the proce stack */
2016-02-29 17:26:40 +00:00
2017-01-09 09:54:49 +00:00
if ( ( moo_oop_t ) proc - > sem ! = moo - > _nil )
2016-02-29 17:26:40 +00:00
{
2017-01-09 09:54:49 +00:00
unchain_from_semaphore ( moo , proc ) ;
2016-02-29 17:26:40 +00:00
}
2017-07-27 17:29:45 +00:00
2017-07-30 15:15:47 +00:00
/* when terminated, clear it from the pid table and set the process id to a negative number */
free_pid ( moo , proc ) ;
2016-02-19 15:52:56 +00:00
}
2017-07-25 15:26:04 +00:00
#if 0
2017-01-09 09:54:49 +00:00
else if ( proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_WAITING ) )
2016-02-19 15:52:56 +00:00
{
/* WAITING ---> TERMINATED */
/* TODO: */
2016-02-13 17:39:11 +00:00
}
2017-07-25 15:26:04 +00:00
# endif
2016-02-13 17:39:11 +00:00
}
2017-01-09 09:54:49 +00:00
static void resume_process ( moo_t * moo , moo_oop_process_t proc )
2015-12-02 15:24:13 +00:00
{
2017-01-09 09:54:49 +00:00
if ( proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_SUSPENDED ) )
2016-02-13 17:39:11 +00:00
{
2017-08-17 18:10:29 +00:00
/* SUSPENDED ---> RUNNABLE */
2017-07-24 13:25:25 +00:00
/*MOO_ASSERT (moo, (moo_oop_t)proc->ps.prev == moo->_nil);
MOO_ASSERT ( moo , ( moo_oop_t ) proc - > ps . next = = moo - > _nil ) ; */
2016-02-19 15:52:56 +00:00
2017-01-09 09:54:49 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR)
2017-08-20 14:43:34 +00:00
MOO_LOG2 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - process [%zd] %hs->RUNNABLE in resume_process \n " , MOO_OOP_TO_SMOOI ( proc - > id ) , proc_state_to_string ( MOO_OOP_TO_SMOOI ( proc - > state ) ) ) ;
2016-06-05 18:01:35 +00:00
# endif
2017-07-25 15:26:04 +00:00
/* don't switch to this process. just change the state to RUNNABLE.
* process switching should be triggerd by the process scheduler . */
chain_into_processor ( moo , proc , PROC_STATE_RUNNABLE ) ;
2016-02-29 15:27:10 +00:00
/*proc->current_context = proc->initial_context;*/
2015-10-19 06:16:43 +00:00
}
2016-02-19 15:52:56 +00:00
#if 0
2017-01-09 09:54:49 +00:00
else if ( proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNABLE ) )
2015-10-19 06:16:43 +00:00
{
2016-02-19 15:52:56 +00:00
/* RUNNABLE ---> RUNNING */
/* TODO: should i allow this? */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > processor - > active ! = proc ) ;
switch_to_process ( moo , proc , PROC_STATE_RUNNABLE ) ;
2015-10-19 06:16:43 +00:00
}
2016-02-19 15:52:56 +00:00
# endif
2015-10-18 15:06:17 +00:00
}
2017-01-09 09:54:49 +00:00
static void suspend_process ( moo_t * moo , moo_oop_process_t proc )
2016-02-18 17:49:56 +00:00
{
2017-01-09 09:54:49 +00:00
if ( proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNING ) | |
proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNABLE ) )
2016-02-18 17:49:56 +00:00
{
2016-02-19 15:52:56 +00:00
/* RUNNING/RUNNABLE ---> SUSPENDED */
2017-01-09 09:54:49 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR)
2017-08-20 14:43:34 +00:00
MOO_LOG2 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - process [%zd] %hs->SUSPENDED in suspend_process \n " , MOO_OOP_TO_SMOOI ( proc - > id ) , proc_state_to_string ( MOO_OOP_TO_SMOOI ( proc - > state ) ) ) ;
2016-06-05 18:01:35 +00:00
# endif
2017-01-09 09:54:49 +00:00
if ( proc = = moo - > processor - > active )
2016-02-19 15:52:56 +00:00
{
2017-08-17 18:10:29 +00:00
/* suspend the active process */
2017-01-09 09:54:49 +00:00
moo_oop_process_t nrp ;
2016-02-19 15:52:56 +00:00
2017-01-09 09:54:49 +00:00
nrp = find_next_runnable_process ( moo ) ;
2016-02-19 15:52:56 +00:00
if ( nrp = = proc )
{
/* no runnable process after suspension */
2017-01-09 09:54:49 +00:00
sleep_active_process ( moo , PROC_STATE_RUNNABLE ) ;
unchain_from_processor ( moo , proc , PROC_STATE_SUSPENDED ) ;
2016-02-29 15:27:10 +00:00
2016-03-23 12:58:30 +00:00
/* the last running/runnable process has been unchained
* from the processor and set to SUSPENDED . the active
2016-06-05 18:01:35 +00:00
* process must be the nil process */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > processor - > active = = moo - > nil_process ) ;
2016-02-19 15:52:56 +00:00
}
else
{
2017-07-25 15:26:04 +00:00
/* unchain_from_processor moves the process to the suspended
* process and sets its state to the given state ( SUSPENDED here ) .
* it doesn ' t change the active process . we switch the active
* process with switch_to_process ( ) . setting the state of the
* old active process to SUSPENDED is redundant because it ' s
* done in unchain_from_processor ( ) . the state of the active
* process is somewhat wrong for a short period of time until
* switch_to_process ( ) has changed the active process . */
unchain_from_processor ( moo , proc , PROC_STATE_SUSPENDED ) ;
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > processor - > active ! = moo - > nil_process ) ;
2017-07-24 13:25:25 +00:00
switch_to_process ( moo , nrp , PROC_STATE_SUSPENDED ) ;
2016-02-19 15:52:56 +00:00
}
}
else
{
2017-01-09 09:54:49 +00:00
unchain_from_processor ( moo , proc , PROC_STATE_SUSPENDED ) ;
2016-02-19 15:52:56 +00:00
}
2016-02-18 17:49:56 +00:00
}
}
2017-01-09 09:54:49 +00:00
static void yield_process ( moo_t * moo , moo_oop_process_t proc )
2016-02-18 17:49:56 +00:00
{
2017-01-09 09:54:49 +00:00
if ( proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNING ) )
2016-02-19 15:52:56 +00:00
{
/* RUNNING --> RUNNABLE */
2017-01-09 09:54:49 +00:00
moo_oop_process_t nrp ;
2016-02-19 15:52:56 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , proc = = moo - > processor - > active ) ;
2016-02-19 15:52:56 +00:00
2017-01-09 09:54:49 +00:00
nrp = find_next_runnable_process ( moo ) ;
2016-02-19 15:52:56 +00:00
/* if there are more than 1 runnable processes, the next
* runnable process must be different from proc */
2016-02-29 15:27:10 +00:00
if ( nrp ! = proc )
{
2017-01-09 09:54:49 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR)
2017-08-20 14:43:34 +00:00
MOO_LOG2 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - process [%zd] %hs->RUNNABLE in yield_process \n " , MOO_OOP_TO_SMOOI ( proc - > id ) , proc_state_to_string ( MOO_OOP_TO_SMOOI ( proc - > state ) ) ) ;
2016-06-05 18:01:35 +00:00
# endif
2017-01-09 09:54:49 +00:00
switch_to_process ( moo , nrp , PROC_STATE_RUNNABLE ) ;
2016-02-29 15:27:10 +00:00
}
}
}
2017-09-21 14:22:36 +00:00
#if 0
2017-01-09 09:54:49 +00:00
static int async_signal_semaphore ( moo_t * moo , moo_oop_semaphore_t sem )
2016-02-29 15:27:10 +00:00
{
2017-01-09 09:54:49 +00:00
if ( moo - > sem_list_count > = SEM_LIST_MAX )
2016-03-22 14:18:07 +00:00
{
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_ESLFULL ) ;
2016-03-22 14:18:07 +00:00
return - 1 ;
}
2017-01-09 09:54:49 +00:00
if ( moo - > sem_list_count > = moo - > sem_list_capa )
2016-02-29 15:27:10 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oow_t new_capa ;
moo_oop_semaphore_t * tmp ;
2016-03-16 02:27:18 +00:00
2017-01-09 09:54:49 +00:00
new_capa = moo - > sem_list_capa + SEM_LIST_INC ; /* TODO: overflow check.. */
tmp = moo_reallocmem ( moo , moo - > sem_list , MOO_SIZEOF ( moo_oop_semaphore_t ) * new_capa ) ;
2016-03-16 02:27:18 +00:00
if ( ! tmp ) return - 1 ;
2017-01-09 09:54:49 +00:00
moo - > sem_list = tmp ;
moo - > sem_list_capa = new_capa ;
2016-02-19 15:52:56 +00:00
}
2016-02-29 15:27:10 +00:00
2017-01-09 09:54:49 +00:00
moo - > sem_list [ moo - > sem_list_count ] = sem ;
moo - > sem_list_count + + ;
2016-02-29 15:27:10 +00:00
return 0 ;
2016-02-18 17:49:56 +00:00
}
2017-09-21 14:22:36 +00:00
# endif
2016-02-18 17:49:56 +00:00
2017-01-09 09:54:49 +00:00
static moo_oop_process_t signal_semaphore ( moo_t * moo , moo_oop_semaphore_t sem )
2016-02-18 17:49:56 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_process_t proc ;
moo_ooi_t count ;
2017-10-08 15:40:32 +00:00
moo_oop_semaphore_group_t semgrp ;
2016-02-18 17:49:56 +00:00
2017-10-08 15:40:32 +00:00
semgrp = sem - > group ;
if ( ( moo_oop_t ) semgrp ! = moo - > _nil )
2017-09-21 07:56:51 +00:00
{
2017-09-21 14:22:36 +00:00
/* the semaphore belongs to a semaphore group */
2017-09-21 07:56:51 +00:00
if ( ( moo_oop_t ) semgrp - > waiting . first ! = moo - > _nil )
{
/* there is a process waiting on the process group */
proc = semgrp - > waiting . first ;
unchain_from_semaphore ( moo , proc ) ;
resume_process ( moo , proc ) ;
2017-10-08 15:40:32 +00:00
/* [IMPORTANT] RETURN VALUE of SemaphoreGroup's wait.
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* the waiting process has been suspended after a waiting
2017-09-26 09:01:42 +00:00
* primitive function in Semaphore or SemaphoreGroup .
* the top of the stack of the process must hold the temporary
* return value set by await_semaphore ( ) or await_semaphore_group ( ) .
* change the return value forcibly to the actual signaled
* semaphore */
MOO_ASSERT ( moo , MOO_OOP_TO_SMOOI ( proc - > sp ) < ( moo_ooi_t ) ( MOO_OBJ_GET_SIZE ( proc ) - MOO_PROCESS_NAMED_INSTVARS ) ) ;
proc - > slot [ MOO_OOP_TO_SMOOI ( proc - > sp ) ] = ( moo_oop_t ) sem ;
2017-11-03 08:10:52 +00:00
if ( MOO_OOP_TO_SMOOI ( sem - > io_index ) > = 0 ) moo - > sem_io_wait_count - - ;
2017-09-21 07:56:51 +00:00
return proc ;
}
}
2017-09-25 15:16:19 +00:00
/* if the semaphore belongs to a semaphore group and the control reaches
* here , no process is waiting on the semaphore group . however , a process
* may still be waiting on the semaphore . If a process waits on a semaphore
2017-10-08 18:13:10 +00:00
* group and another process wait on a semaphore that belongs to the
2017-09-25 15:16:19 +00:00
* semaphore group , the process waiting on the group always wins .
2017-09-21 14:22:36 +00:00
*
* TODO : implement a fair scheduling policy . or do i simply have to disallow individual wait on a semaphore belonging to a group ?
*
2017-10-08 15:40:32 +00:00
* if it doesn ' t belong to a sempahore group , i ' m free from the starvation issue .
2017-09-21 14:22:36 +00:00
*/
2017-07-24 13:25:25 +00:00
if ( ( moo_oop_t ) sem - > waiting . first = = moo - > _nil )
2016-02-18 17:49:56 +00:00
{
2016-02-29 15:27:10 +00:00
/* no process is waiting on this semaphore */
2017-01-09 09:54:49 +00:00
count = MOO_OOP_TO_SMOOI ( sem - > count ) ;
2017-10-08 18:13:10 +00:00
count + + ;
sem - > count = MOO_SMOOI_TO_OOP ( count ) ;
MOO_ASSERT ( moo , count > = 1 ) ;
if ( count = = 1 & & ( moo_oop_t ) semgrp ! = moo - > _nil )
2017-10-08 15:40:32 +00:00
{
/* move the semaphore from the unsignaled list to the signaled list
2017-10-08 18:13:10 +00:00
* if the semaphore count has changed from 0 to 1 in a group */
MOO_DELETE_FROM_OOP_LIST ( moo , & semgrp - > sems [ MOO_SEMAPHORE_GROUP_SEMS_UNSIG ] , sem , grm ) ;
MOO_APPEND_TO_OOP_LIST ( moo , & semgrp - > sems [ MOO_SEMAPHORE_GROUP_SEMS_SIG ] , moo_oop_semaphore_t , sem , grm ) ;
2017-10-08 15:40:32 +00:00
}
2016-03-24 14:58:47 +00:00
/* no process has been resumed */
2017-01-09 09:54:49 +00:00
return ( moo_oop_process_t ) moo - > _nil ;
2016-02-18 17:49:56 +00:00
}
else
{
2017-07-24 13:25:25 +00:00
proc = sem - > waiting . first ;
2016-03-24 14:58:47 +00:00
2017-01-09 09:54:49 +00:00
/* [NOTE] no GC must occur as 'proc' isn't protected with moo_pushtmp(). */
2016-03-24 14:58:47 +00:00
2017-07-20 16:33:53 +00:00
/* detach a process from a semaphore's waiting list and
2017-02-12 18:59:03 +00:00
* make it runnable */
2017-01-09 09:54:49 +00:00
unchain_from_semaphore ( moo , proc ) ;
2017-07-25 15:26:04 +00:00
resume_process ( moo , proc ) ;
2016-03-24 14:58:47 +00:00
2017-02-18 13:31:47 +00:00
if ( MOO_OOP_TO_SMOOI ( sem - > io_index ) > = 0 ) moo - > sem_io_wait_count - - ;
2017-02-12 18:59:03 +00:00
/* return the resumed(runnable) process */
2016-03-24 14:58:47 +00:00
return proc ;
2016-02-18 17:49:56 +00:00
}
}
2017-09-26 09:01:42 +00:00
static MOO_INLINE void await_semaphore ( moo_t * moo , moo_oop_semaphore_t sem )
2016-02-18 17:49:56 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_process_t proc ;
moo_ooi_t count ;
2017-10-08 15:40:32 +00:00
moo_oop_semaphore_group_t semgrp ;
2016-02-18 17:49:56 +00:00
2017-10-08 15:40:32 +00:00
semgrp = sem - > group ;
2017-09-21 14:22:36 +00:00
#if 0
/* TODO: do i have to disallow?? */
2017-10-08 15:40:32 +00:00
if ( ( moo_oop_t ) semgrp ! = moo - > _nil )
2017-09-21 14:22:36 +00:00
{
/* disallow a semaphore in a semaphore group to be waited on */
moo_seterrnum ( moo , MOO_EPERM ) ;
return - 1 ;
}
# endif
2017-01-09 09:54:49 +00:00
count = MOO_OOP_TO_SMOOI ( sem - > count ) ;
2016-02-18 17:49:56 +00:00
if ( count > 0 )
{
2017-11-03 16:26:55 +00:00
/* it's already signaled */
2016-02-18 17:49:56 +00:00
count - - ;
2017-01-09 09:54:49 +00:00
sem - > count = MOO_SMOOI_TO_OOP ( count ) ;
2017-10-08 15:40:32 +00:00
if ( ( moo_oop_t ) semgrp ! = moo - > _nil & & count = = 0 )
{
2017-10-08 18:13:10 +00:00
int sems_idx ;
2017-10-08 15:40:32 +00:00
/* TODO: if i disallow individual wait on a semaphore in a group,
* this membership manipulation is redundant */
2017-10-08 18:13:10 +00:00
MOO_DELETE_FROM_OOP_LIST ( moo , & semgrp - > sems [ MOO_SEMAPHORE_GROUP_SEMS_SIG ] , sem , grm ) ;
sems_idx = count > 0 ? MOO_SEMAPHORE_GROUP_SEMS_SIG : MOO_SEMAPHORE_GROUP_SEMS_UNSIG ;
MOO_APPEND_TO_OOP_LIST ( moo , & semgrp - > sems [ sems_idx ] , moo_oop_semaphore_t , sem , grm ) ;
2017-10-08 15:40:32 +00:00
}
2016-02-18 17:49:56 +00:00
}
else
{
2016-02-29 15:27:10 +00:00
/* not signaled. need to wait */
2017-01-09 09:54:49 +00:00
proc = moo - > processor - > active ;
2016-02-18 17:49:56 +00:00
2016-02-29 17:26:40 +00:00
/* suspend the active process */
2017-01-09 09:54:49 +00:00
suspend_process ( moo , proc ) ;
2016-02-29 17:26:40 +00:00
2016-03-24 14:58:47 +00:00
/* link the suspended process to the semaphore's process list */
2017-01-09 09:54:49 +00:00
chain_into_semaphore ( moo , proc , sem ) ;
2016-02-18 17:49:56 +00:00
2017-07-24 13:25:25 +00:00
MOO_ASSERT ( moo , sem - > waiting . last = = proc ) ;
2016-03-24 14:58:47 +00:00
2017-02-18 13:31:47 +00:00
if ( MOO_OOP_TO_SMOOI ( sem - > io_index ) > = 0 ) moo - > sem_io_wait_count + + ;
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > processor - > active ! = proc ) ;
2016-02-18 17:49:56 +00:00
}
2017-09-21 14:22:36 +00:00
2017-09-26 09:01:42 +00:00
#if 0
2017-09-21 14:22:36 +00:00
return 0 ;
2017-09-26 09:01:42 +00:00
# endif
2016-02-18 17:49:56 +00:00
}
2017-10-08 15:40:32 +00:00
static MOO_INLINE moo_oop_t await_semaphore_group ( moo_t * moo , moo_oop_semaphore_group_t semgrp )
2017-08-22 13:45:37 +00:00
{
/* wait for one of semaphores in the group to be signaled */
2017-09-21 07:56:51 +00:00
moo_oop_process_t proc ;
moo_oop_semaphore_t sem ;
2017-10-02 01:22:49 +00:00
MOO_ASSERT ( moo , moo_iskindof ( moo , ( moo_oop_t ) semgrp , moo - > _semaphore_group ) ) ;
2017-09-21 07:56:51 +00:00
2017-10-08 18:13:10 +00:00
sem = semgrp - > sems [ MOO_SEMAPHORE_GROUP_SEMS_SIG ] . first ;
2017-10-08 15:40:32 +00:00
if ( ( moo_oop_t ) sem ! = moo - > _nil )
2017-09-21 07:56:51 +00:00
{
2017-10-08 15:40:32 +00:00
moo_ooi_t count ;
2017-10-08 18:13:10 +00:00
int sems_idx ;
2017-09-21 07:56:51 +00:00
count = MOO_OOP_TO_SMOOI ( sem - > count ) ;
2017-10-08 15:40:32 +00:00
MOO_ASSERT ( moo , count > 0 ) ;
count - - ;
sem - > count = MOO_SMOOI_TO_OOP ( count ) ;
2017-10-08 18:13:10 +00:00
MOO_DELETE_FROM_OOP_LIST ( moo , & semgrp - > sems [ MOO_SEMAPHORE_GROUP_SEMS_SIG ] , sem , grm ) ;
sems_idx = count > 0 ? MOO_SEMAPHORE_GROUP_SEMS_SIG : MOO_SEMAPHORE_GROUP_SEMS_UNSIG ;
MOO_APPEND_TO_OOP_LIST ( moo , & semgrp - > sems [ sems_idx ] , moo_oop_semaphore_t , sem , grm ) ;
2017-10-08 15:40:32 +00:00
return ( moo_oop_t ) sem ;
2017-09-21 07:56:51 +00:00
}
/* no semaphores have been signaled. suspend the current process
2017-11-03 16:26:55 +00:00
* until at least one of them is signaled */
2017-09-21 07:56:51 +00:00
proc = moo - > processor - > active ;
/* suspend the active process */
suspend_process ( moo , proc ) ;
2017-09-21 14:22:36 +00:00
/* link the suspended process to the semaphore group's process list */
2017-09-21 07:56:51 +00:00
chain_into_semaphore ( moo , proc , ( moo_oop_semaphore_t ) semgrp ) ;
MOO_ASSERT ( moo , semgrp - > waiting . last = = proc ) ;
2017-10-31 07:13:22 +00:00
2017-11-21 09:15:22 +00:00
if ( MOO_OOP_TO_SMOOI ( semgrp - > sem_io_count ) > 0 )
2017-10-31 07:13:22 +00:00
{
/* there might be more than 1 IO semaphores in the group
* but i increment moo - > sem_io_wait_count by 1 only */
moo - > sem_io_wait_count + + ;
}
2017-08-22 13:45:37 +00:00
2017-09-25 15:16:19 +00:00
/* the current process will get suspended after the caller (mostly a
* a primitive function handler ) is over as it ' s added to a suspened
* process list above */
2017-09-21 07:56:51 +00:00
MOO_ASSERT ( moo , moo - > processor - > active ! = proc ) ;
2017-09-25 15:16:19 +00:00
return moo - > _nil ;
2017-08-22 13:45:37 +00:00
}
2017-01-09 09:54:49 +00:00
static void sift_up_sem_heap ( moo_t * moo , moo_ooi_t index )
2016-03-22 14:18:07 +00:00
{
if ( index > 0 )
{
2017-01-09 09:54:49 +00:00
moo_ooi_t parent ;
moo_oop_semaphore_t sem , parsem ;
2016-03-22 14:18:07 +00:00
parent = SEM_HEAP_PARENT ( index ) ;
2017-01-09 09:54:49 +00:00
sem = moo - > sem_heap [ index ] ;
parsem = moo - > sem_heap [ parent ] ;
if ( SEM_HEAP_EARLIER_THAN ( moo , sem , parsem ) )
2016-03-22 14:18:07 +00:00
{
do
{
/* move down the parent to the current position */
2017-01-09 09:54:49 +00:00
parsem - > heap_index = MOO_SMOOI_TO_OOP ( index ) ;
moo - > sem_heap [ index ] = parsem ;
2016-03-22 14:18:07 +00:00
/* traverse up */
index = parent ;
if ( index < = 0 ) break ;
parent = SEM_HEAP_PARENT ( parent ) ;
2017-01-09 09:54:49 +00:00
parsem = moo - > sem_heap [ parent ] ;
2016-03-22 14:18:07 +00:00
}
2017-01-09 09:54:49 +00:00
while ( SEM_HEAP_EARLIER_THAN ( moo , sem , parsem ) ) ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
sem - > heap_index = MOO_SMOOI_TO_OOP ( index ) ;
moo - > sem_heap [ index ] = sem ;
2016-03-22 14:18:07 +00:00
}
}
}
2017-01-09 09:54:49 +00:00
static void sift_down_sem_heap ( moo_t * moo , moo_ooi_t index )
2016-03-22 14:18:07 +00:00
{
2017-01-09 09:54:49 +00:00
moo_ooi_t base = moo - > sem_heap_count / 2 ;
2016-03-22 14:18:07 +00:00
if ( index < base ) /* at least 1 child is under the 'index' position */
{
2017-01-09 09:54:49 +00:00
moo_ooi_t left , right , child ;
moo_oop_semaphore_t sem , chisem ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
sem = moo - > sem_heap [ index ] ;
2016-03-22 14:18:07 +00:00
do
{
left = SEM_HEAP_LEFT ( index ) ;
right = SEM_HEAP_RIGHT ( index ) ;
2017-08-19 15:54:03 +00:00
if ( right < moo - > sem_heap_count & & SEM_HEAP_EARLIER_THAN ( moo , moo - > sem_heap [ right ] , moo - > sem_heap [ left ] ) )
2016-03-22 14:18:07 +00:00
{
child = right ;
}
else
{
child = left ;
}
2017-01-09 09:54:49 +00:00
chisem = moo - > sem_heap [ child ] ;
if ( SEM_HEAP_EARLIER_THAN ( moo , sem , chisem ) ) break ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
chisem - > heap_index = MOO_SMOOI_TO_OOP ( index ) ;
2017-08-19 15:54:03 +00:00
moo - > sem_heap [ index ] = chisem ;
2016-03-22 14:18:07 +00:00
index = child ;
}
while ( index < base ) ;
2017-01-09 09:54:49 +00:00
sem - > heap_index = MOO_SMOOI_TO_OOP ( index ) ;
moo - > sem_heap [ index ] = sem ;
2016-03-22 14:18:07 +00:00
}
}
2017-01-09 09:54:49 +00:00
static int add_to_sem_heap ( moo_t * moo , moo_oop_semaphore_t sem )
2016-03-22 14:18:07 +00:00
{
2017-01-09 09:54:49 +00:00
moo_ooi_t index ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
if ( moo - > sem_heap_count > = SEM_HEAP_MAX )
2016-03-22 14:18:07 +00:00
{
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_ESHFULL ) ;
2016-03-22 14:18:07 +00:00
return - 1 ;
}
2017-01-09 09:54:49 +00:00
if ( moo - > sem_heap_count > = moo - > sem_heap_capa )
2016-03-22 14:18:07 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oow_t new_capa ;
moo_oop_semaphore_t * tmp ;
2016-03-22 14:18:07 +00:00
/* no overflow check when calculating the new capacity
* owing to SEM_HEAP_MAX check above */
2017-01-09 09:54:49 +00:00
new_capa = moo - > sem_heap_capa + SEM_HEAP_INC ;
tmp = moo_reallocmem ( moo , moo - > sem_heap , MOO_SIZEOF ( moo_oop_semaphore_t ) * new_capa ) ;
2016-03-22 14:18:07 +00:00
if ( ! tmp ) return - 1 ;
2017-01-09 09:54:49 +00:00
moo - > sem_heap = tmp ;
moo - > sem_heap_capa = new_capa ;
2016-03-22 14:18:07 +00:00
}
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > sem_heap_count < = MOO_SMOOI_MAX ) ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
index = moo - > sem_heap_count ;
moo - > sem_heap [ index ] = sem ;
sem - > heap_index = MOO_SMOOI_TO_OOP ( index ) ;
moo - > sem_heap_count + + ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
sift_up_sem_heap ( moo , index ) ;
2016-03-22 14:18:07 +00:00
return 0 ;
}
2017-01-09 09:54:49 +00:00
static void delete_from_sem_heap ( moo_t * moo , moo_ooi_t index )
2016-03-22 14:18:07 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_semaphore_t sem , lastsem ;
2016-03-22 14:18:07 +00:00
2017-07-21 16:54:43 +00:00
MOO_ASSERT ( moo , index > = 0 & & index < moo - > sem_heap_count ) ;
2017-01-09 09:54:49 +00:00
sem = moo - > sem_heap [ index ] ;
sem - > heap_index = MOO_SMOOI_TO_OOP ( - 1 ) ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
moo - > sem_heap_count - - ;
2017-07-21 16:54:43 +00:00
if ( /*moo->sem_heap_count > 0 &&*/ index ! = moo - > sem_heap_count )
2016-03-22 14:18:07 +00:00
{
/* move the last item to the deletion position */
2017-01-09 09:54:49 +00:00
lastsem = moo - > sem_heap [ moo - > sem_heap_count ] ;
lastsem - > heap_index = MOO_SMOOI_TO_OOP ( index ) ;
moo - > sem_heap [ index ] = lastsem ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
if ( SEM_HEAP_EARLIER_THAN ( moo , lastsem , sem ) )
sift_up_sem_heap ( moo , index ) ;
2016-03-22 14:18:07 +00:00
else
2017-01-09 09:54:49 +00:00
sift_down_sem_heap ( moo , index ) ;
2016-03-22 14:18:07 +00:00
}
}
2017-02-12 18:59:03 +00:00
#if 0
/* unused */
2017-01-09 09:54:49 +00:00
static void update_sem_heap ( moo_t * moo , moo_ooi_t index , moo_oop_semaphore_t newsem )
2016-03-22 14:18:07 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_semaphore_t sem ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
sem = moo - > sem_heap [ index ] ;
sem - > heap_index = MOO_SMOOI_TO_OOP ( - 1 ) ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
newsem - > heap_index = MOO_SMOOI_TO_OOP ( index ) ;
moo - > sem_heap [ index ] = newsem ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
if ( SEM_HEAP_EARLIER_THAN ( moo , newsem , sem ) )
sift_up_sem_heap ( moo , index ) ;
2016-03-22 14:18:07 +00:00
else
2017-01-09 09:54:49 +00:00
sift_down_sem_heap ( moo , index ) ;
2016-03-22 14:18:07 +00:00
}
2017-02-12 18:59:03 +00:00
# endif
static int add_to_sem_io ( moo_t * moo , moo_oop_semaphore_t sem )
{
moo_ooi_t index ;
2017-02-15 11:57:24 +00:00
int n ;
2017-02-12 18:59:03 +00:00
if ( moo - > sem_io_count > = SEM_IO_MAX )
{
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_ESHFULL ) ;
2017-02-12 18:59:03 +00:00
return - 1 ;
}
if ( moo - > sem_io_count > = moo - > sem_io_capa )
{
moo_oow_t new_capa ;
moo_oop_semaphore_t * tmp ;
/* no overflow check when calculating the new capacity
* owing to SEM_IO_MAX check above */
new_capa = moo - > sem_io_capa + SEM_IO_INC ;
tmp = moo_reallocmem ( moo , moo - > sem_io , MOO_SIZEOF ( moo_oop_semaphore_t ) * new_capa ) ;
if ( ! tmp ) return - 1 ;
moo - > sem_io = tmp ;
moo - > sem_io_capa = new_capa ;
}
MOO_ASSERT ( moo , moo - > sem_io_count < = MOO_SMOOI_MAX ) ;
2017-02-15 11:57:24 +00:00
MOO_ASSERT ( moo , sem - > io_index = = MOO_SMOOI_TO_OOP ( - 1 ) ) ;
2017-02-12 18:59:03 +00:00
index = moo - > sem_io_count ;
moo - > sem_io [ index ] = sem ;
2017-02-15 11:57:24 +00:00
sem - > io_index = MOO_SMOOI_TO_OOP ( index ) ;
2017-02-12 18:59:03 +00:00
moo - > sem_io_count + + ;
2017-02-15 11:57:24 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & sem ) ;
2017-02-22 09:48:58 +00:00
n = moo - > vmprim . vm_muxadd ( moo , sem ) ;
2017-02-15 11:57:24 +00:00
moo_poptmp ( moo ) ;
if ( n < = - 1 )
{
/* roll back */
sem - > io_index = MOO_SMOOI_TO_OOP ( - 1 ) ;
moo - > sem_io_count - - ;
}
2017-10-31 07:13:22 +00:00
else
{
/* update the number of IO semaphores in a group if necessary */
if ( ( moo_oop_t ) sem - > group ! = moo - > _nil )
{
moo_ooi_t count ;
count = MOO_OOP_TO_SMOOI ( sem - > group - > sem_io_count ) ;
count + + ;
sem - > group - > sem_io_count = MOO_SMOOI_TO_OOP ( count ) ;
}
}
2017-04-08 07:23:31 +00:00
2017-02-15 11:57:24 +00:00
return n ;
2017-02-12 18:59:03 +00:00
}
2017-09-25 15:16:19 +00:00
static MOO_INLINE int modify_in_sem_io ( moo_t * moo , moo_oop_semaphore_t sem )
2017-02-12 18:59:03 +00:00
{
2017-04-08 07:23:31 +00:00
return moo - > vmprim . vm_muxmod ( moo , sem ) ;
}
static int delete_from_sem_io ( moo_t * moo , moo_ooi_t index )
{
moo_oop_semaphore_t sem ;
int x ;
2017-02-12 18:59:03 +00:00
2017-07-21 16:54:43 +00:00
MOO_ASSERT ( moo , index > = 0 & & index < moo - > sem_io_count ) ;
2017-02-12 18:59:03 +00:00
sem = moo - > sem_io [ index ] ;
2017-04-08 07:23:31 +00:00
MOO_ASSERT ( moo , index = = MOO_OOP_TO_SMOOI ( sem - > io_index ) ) ;
2017-02-15 11:57:24 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & sem ) ;
2017-04-08 07:23:31 +00:00
x = moo - > vmprim . vm_muxdel ( moo , sem ) ;
2017-02-15 11:57:24 +00:00
moo_poptmp ( moo ) ;
2017-04-08 07:23:31 +00:00
if ( x < = - 1 )
{
MOO_DEBUG2 ( moo , " Failed to delete IO semaphore at index %zd on handle %zd \n " , index , MOO_OOP_TO_SMOOI ( sem - > io_handle ) ) ;
return - 1 ;
}
MOO_DEBUG2 ( moo , " Deleted IO semaphore at index %zd on handle %zd \n " , index , MOO_OOP_TO_SMOOI ( sem - > io_handle ) ) ;
2017-02-12 18:59:03 +00:00
sem - > io_index = MOO_SMOOI_TO_OOP ( - 1 ) ;
moo - > sem_io_count - - ;
2017-10-31 07:13:22 +00:00
if ( ( moo_oop_t ) sem - > group ! = moo - > _nil )
{
moo_ooi_t count ;
count = MOO_OOP_TO_SMOOI ( sem - > group - > sem_io_count ) ;
MOO_ASSERT ( moo , count > 0 ) ;
count - - ;
sem - > group - > sem_io_count = MOO_SMOOI_TO_OOP ( count ) ;
}
2017-07-21 16:54:43 +00:00
if ( /*moo->sem_io_count > 0 &&*/ index ! = moo - > sem_io_count )
2017-02-12 18:59:03 +00:00
{
2017-04-08 07:23:31 +00:00
moo_oop_semaphore_t lastsem ;
/* move the last item to the deletion position for compaction */
2017-02-12 18:59:03 +00:00
lastsem = moo - > sem_io [ moo - > sem_io_count ] ;
lastsem - > io_index = MOO_SMOOI_TO_OOP ( index ) ;
moo - > sem_io [ index ] = lastsem ;
2017-04-08 07:23:31 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & lastsem ) ;
x = moo - > vmprim . vm_muxmod ( moo , lastsem ) ;
moo_poptmp ( moo ) ;
if ( x < = - 1 )
{
2017-04-08 07:33:06 +00:00
/* unfortunately, i can't roll back gracefully. i nullify the delete slot instead of compaction */
MOO_LOG3 ( moo , MOO_LOG_WARN , " Warning - IO sempahore migration failure from %zd to %zd on handle %zd - expect VM memory waste \n " , moo - > sem_io_count , MOO_OOP_TO_SMOOI ( lastsem - > io_index ) , MOO_OOP_TO_SMOOI ( lastsem - > io_handle ) ) ;
2017-04-08 07:23:31 +00:00
lastsem - > io_index = MOO_SMOOI_TO_OOP ( moo - > sem_io_count ) ;
moo - > sem_io [ moo - > sem_io_count ] = lastsem ;
moo - > sem_io_count + + ;
moo - > sem_io [ index ] = ( moo_oop_semaphore_t ) moo - > _nil ;
}
else
{
MOO_DEBUG3 ( moo , " Migrated IO semaphore from index %zd to %zd on handle %zd \n " , moo - > sem_io_count , MOO_OOP_TO_SMOOI ( lastsem - > io_index ) , MOO_OOP_TO_SMOOI ( lastsem - > io_handle ) ) ;
}
2017-02-12 18:59:03 +00:00
}
2016-03-22 14:18:07 +00:00
2017-04-08 07:23:31 +00:00
return 0 ;
}
2017-02-15 11:57:24 +00:00
2017-04-08 07:23:31 +00:00
static void signal_io_semaphore ( moo_t * moo , moo_ooi_t mask , void * ctx )
2017-02-15 11:57:24 +00:00
{
moo_oow_t sem_io_index = ( moo_oow_t ) ctx ;
/* TODO: sanity check on the index. conditional handling on mask */
if ( sem_io_index < moo - > sem_io_count )
{
moo_oop_semaphore_t sem ;
moo_oop_process_t proc ;
sem = moo - > sem_io [ sem_io_index ] ;
2017-04-08 07:33:06 +00:00
if ( ( moo_oop_t ) sem = = moo - > _nil )
{
/* it's a nullified slot for migration failure in delete_from_sem_io() */
goto invalid_semaphore ;
}
2017-02-15 11:57:24 +00:00
proc = signal_semaphore ( moo , sem ) ;
if ( moo - > processor - > active = = moo - > nil_process & & ( moo_oop_t ) proc ! = moo - > _nil )
{
/* this is the only runnable process.
* switch the process to the running state .
2017-07-25 15:26:04 +00:00
* it uses wake_process ( ) instead of
2017-02-15 11:57:24 +00:00
* switch_to_process ( ) as there is no running
* process at this moment */
MOO_ASSERT ( moo , proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNABLE ) ) ;
2017-07-24 13:25:25 +00:00
MOO_ASSERT ( moo , proc = = moo - > processor - > runnable . first ) ;
2017-02-15 11:57:24 +00:00
2017-07-25 15:26:04 +00:00
#if 0
wake_process ( moo , proc ) ; /* switch to running */
2017-02-15 11:57:24 +00:00
moo - > proc_switched = 1 ;
2017-07-25 15:26:04 +00:00
# else
switch_to_process_from_nil ( moo , proc ) ;
# endif
2017-02-15 11:57:24 +00:00
}
}
else
{
2017-04-08 07:33:06 +00:00
invalid_semaphore :
2017-02-15 11:57:24 +00:00
MOO_LOG1 ( moo , MOO_LOG_WARN , " Warning - Invalid semaphore index %zu \n " , sem_io_index ) ;
}
}
/* ------------------------------------------------------------------------- */
2017-01-09 09:54:49 +00:00
static moo_oop_process_t start_initial_process ( moo_t * moo , moo_oop_context_t c )
2015-12-02 15:24:13 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_process_t proc ;
2015-12-02 15:24:13 +00:00
2016-02-12 16:23:26 +00:00
/* there must be no active process when this function is called */
2017-07-24 13:25:25 +00:00
MOO_ASSERT ( moo , moo - > processor - > runnable . count = = MOO_SMOOI_TO_OOP ( 0 ) ) ;
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > processor - > active = = moo - > nil_process ) ;
2016-02-12 16:23:26 +00:00
2017-01-09 09:54:49 +00:00
proc = make_process ( moo , c ) ;
if ( ! proc ) return MOO_NULL ;
2015-10-18 15:06:17 +00:00
2017-07-25 15:26:04 +00:00
chain_into_processor ( moo , proc , PROC_STATE_RUNNING ) ;
2017-01-09 09:54:49 +00:00
moo - > processor - > active = proc ;
2016-02-12 16:23:26 +00:00
2016-02-18 17:49:56 +00:00
/* do somthing that resume_process() would do with less overhead */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , ( moo_oop_t ) proc - > current_context ! = moo - > _nil ) ;
MOO_ASSERT ( moo , proc - > current_context = = proc - > initial_context ) ;
SWITCH_ACTIVE_CONTEXT ( moo , proc - > current_context ) ;
2016-02-12 16:23:26 +00:00
2015-12-02 15:24:13 +00:00
return proc ;
}
2015-10-13 14:51:04 +00:00
2017-01-09 09:54:49 +00:00
static MOO_INLINE int activate_new_method ( moo_t * moo , moo_oop_method_t mth , moo_ooi_t actual_nargs )
2015-06-04 18:34:37 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_context_t ctx ;
moo_ooi_t i , j ;
moo_ooi_t ntmprs , nargs , actual_ntmprs ;
2015-06-04 18:34:37 +00:00
2017-01-09 09:54:49 +00:00
ntmprs = MOO_OOP_TO_SMOOI ( mth - > tmpr_count ) ;
nargs = MOO_OOP_TO_SMOOI ( mth - > tmpr_nargs ) ;
2016-05-02 13:50:42 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , ntmprs > = 0 ) ;
MOO_ASSERT ( moo , nargs < = ntmprs ) ;
2015-06-04 18:34:37 +00:00
2016-12-14 07:18:01 +00:00
if ( actual_nargs > nargs )
{
/* more arguments than the method specification have been passed in.
2017-04-01 06:54:27 +00:00
* it must be a variadic or liberal unary method . othewise , the compiler is buggy */
MOO_ASSERT ( moo , MOO_METHOD_GET_PREAMBLE_FLAGS ( MOO_OOP_TO_SMOOI ( mth - > preamble ) ) & ( MOO_METHOD_PREAMBLE_FLAG_VARIADIC | MOO_METHOD_PREAMBLE_FLAG_LIBERAL ) ) ;
2016-12-14 07:18:01 +00:00
actual_ntmprs = ntmprs + ( actual_nargs - nargs ) ;
}
else actual_ntmprs = ntmprs ;
2017-01-09 09:54:49 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & mth ) ;
ctx = ( moo_oop_context_t ) moo_instantiate ( moo , moo - > _method_context , MOO_NULL , actual_ntmprs ) ;
moo_poptmp ( moo ) ;
2015-06-26 15:49:08 +00:00
if ( ! ctx ) return - 1 ;
2017-01-09 09:54:49 +00:00
ctx - > sender = moo - > active_context ;
ctx - > ip = MOO_SMOOI_TO_OOP ( 0 ) ;
2016-05-02 13:50:42 +00:00
/* ctx->sp will be set further down */
2016-06-24 15:01:51 +00:00
/* A context is compose of a fixed part and a variable part.
2016-12-13 15:18:19 +00:00
* the variable part holds temporary varibles including arguments .
2015-06-06 07:24:35 +00:00
*
2016-06-24 15:01:51 +00:00
* Assuming a method context with 2 arguments and 3 local temporary
* variables , the context will look like this .
2015-06-06 07:24:35 +00:00
* + - - - - - - - - - - - - - - - - - - - - - +
* | fixed part |
* | |
* | |
* | |
* + - - - - - - - - - - - - - - - - - - - - - +
* | tmp1 ( arg1 ) | slot [ 0 ]
* | tmp2 ( arg2 ) | slot [ 1 ]
2016-06-24 15:01:51 +00:00
* | tmp3 | slot [ 2 ]
* | tmp4 | slot [ 3 ]
* | tmp5 | slot [ 4 ]
2015-06-06 07:24:35 +00:00
* + - - - - - - - - - - - - - - - - - - - - - +
*/
2016-06-24 15:01:51 +00:00
2017-01-09 09:54:49 +00:00
ctx - > ntmprs = MOO_SMOOI_TO_OOP ( ntmprs ) ;
ctx - > method_or_nargs = ( moo_oop_t ) mth ;
/* the 'home' field of a method context is always moo->_nil.
ctx - > home = moo - > _nil ; */
2015-06-21 14:45:45 +00:00
ctx - > origin = ctx ; /* point to self */
2015-06-04 18:34:37 +00:00
/*
2015-06-06 07:24:35 +00:00
* Assume this message sending expression :
2015-06-04 18:34:37 +00:00
* obj1 do : # this with : # that with : # it
*
2015-06-06 07:24:35 +00:00
* It would be compiled to these logical byte - code sequences shown below :
2015-06-04 18:34:37 +00:00
* push obj1
* push # this
* push # that
* push # it
* send # do : with :
*
* After three pushes , the stack looks like this .
*
* | # it | < - sp
* | # that | sp - 1
* | # this | sp - 2
* | obj1 | sp - nargs
*
* Since the number of arguments is 3 , stack [ sp - 3 ] points to
* the receiver . When the stack is empty , sp is - 1.
*/
2016-12-14 07:18:01 +00:00
if ( actual_nargs > = nargs )
2015-06-04 18:34:37 +00:00
{
2016-12-14 07:18:01 +00:00
for ( i = actual_nargs , j = ntmprs + ( actual_nargs - nargs ) ; i > nargs ; i - - )
{
/* place variadic arguments after local temporaries */
2017-01-09 09:54:49 +00:00
ctx - > slot [ - - j ] = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_POP ( moo ) ;
2016-12-14 07:18:01 +00:00
}
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , i = = nargs ) ;
2016-12-14 07:18:01 +00:00
while ( i > 0 )
{
/* place normal argument before local temporaries */
2017-01-09 09:54:49 +00:00
ctx - > slot [ - - i ] = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_POP ( moo ) ;
2016-12-14 07:18:01 +00:00
}
}
else
{
for ( i = actual_nargs ; i > 0 ; )
{
/* place normal argument before local temporaries */
2017-01-09 09:54:49 +00:00
ctx - > slot [ - - i ] = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_POP ( moo ) ;
2016-12-14 07:18:01 +00:00
}
2015-06-04 18:34:37 +00:00
}
2015-06-09 12:14:18 +00:00
/* copy receiver */
2017-01-09 09:54:49 +00:00
ctx - > receiver_or_source = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_POP ( moo ) ;
2015-06-04 18:34:37 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > sp > = - 1 ) ;
2015-06-04 18:34:37 +00:00
2016-06-24 15:01:51 +00:00
/* the stack pointer in a context is a stack pointer of a process
* before it is activated . this stack pointer is stored to the context
* so that it is used to restore the process stack pointer upon returning
* from a method context . */
2017-01-09 09:54:49 +00:00
ctx - > sp = MOO_SMOOI_TO_OOP ( moo - > sp ) ;
2016-02-16 17:00:10 +00:00
2016-06-24 15:01:51 +00:00
/* switch the active context to the newly instantiated one*/
2017-01-09 09:54:49 +00:00
SWITCH_ACTIVE_CONTEXT ( moo , ctx ) ;
2015-06-08 13:24:02 +00:00
2015-06-04 18:34:37 +00:00
return 0 ;
}
2017-12-08 15:28:51 +00:00
moo_oop_method_t moo_findmethod ( moo_t * moo , moo_oop_t receiver , const moo_oocs_t * message , int super )
2015-06-04 18:34:37 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_class_t cls ;
moo_oop_association_t ass ;
moo_oop_t c ;
2017-05-20 02:27:48 +00:00
moo_oop_dic_t mthdic ;
2015-06-06 07:24:35 +00:00
int dic_no ;
2015-06-08 13:24:02 +00:00
/* TODO: implement method lookup cache */
2015-06-04 18:34:37 +00:00
2017-02-15 11:57:24 +00:00
cls = MOO_CLASSOF ( moo , receiver ) ;
if ( cls = = moo - > _class )
2015-06-04 18:34:37 +00:00
{
2016-05-03 10:10:28 +00:00
/* receiver is a class object (an instance of Class) */
2015-06-08 13:24:02 +00:00
c = receiver ;
2017-01-09 09:54:49 +00:00
dic_no = MOO_METHOD_CLASS ;
2015-06-08 13:24:02 +00:00
}
2015-06-04 18:34:37 +00:00
else
{
2016-11-21 13:56:20 +00:00
/* receiver is not a class object. so take its class */
2017-01-09 09:54:49 +00:00
c = ( moo_oop_t ) cls ;
dic_no = MOO_METHOD_INSTANCE ;
2015-06-06 07:24:35 +00:00
}
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , c ! = moo - > _nil ) ;
2016-11-21 13:56:20 +00:00
if ( super )
2015-06-06 07:24:35 +00:00
{
2016-11-21 13:56:20 +00:00
/*
2017-01-09 09:54:49 +00:00
moo_oop_method_t m ;
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , moo - > active_context - > origin ) = = moo - > _method_context ) ;
m = ( moo_oop_method_t ) moo - > active_context - > origin - > method_or_nargs ;
c = ( ( moo_oop_class_t ) m - > owner ) - > superclass ;
2016-11-21 13:56:20 +00:00
*/
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > active_method ) ;
MOO_ASSERT ( moo , moo - > active_method - > owner ) ;
c = ( ( moo_oop_class_t ) moo - > active_method - > owner ) - > superclass ;
if ( c = = moo - > _nil ) goto not_found ; /* reached the top of the hierarchy */
2016-11-21 13:56:20 +00:00
}
2015-06-06 07:24:35 +00:00
2016-11-21 13:56:20 +00:00
do
{
2017-01-09 09:54:49 +00:00
mthdic = ( ( moo_oop_class_t ) c ) - > mthdic [ dic_no ] ;
2017-04-02 13:13:33 +00:00
/* if a kernel class is not defined in the bootstrapping code,
* the method dictionary is still nil . the initial kernel classes
* must all be defined properly */
MOO_ASSERT ( moo , ( moo_oop_t ) mthdic ! = moo - > _nil ) ;
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , mthdic ) = = moo - > _method_dictionary ) ;
2015-06-09 12:14:18 +00:00
2017-01-09 09:54:49 +00:00
ass = ( moo_oop_association_t ) moo_lookupdic ( moo , mthdic , message ) ;
2016-11-21 13:56:20 +00:00
if ( ass )
{
/* found the method */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , ass - > value ) = = moo - > _method ) ;
return ( moo_oop_method_t ) ass - > value ;
2015-06-06 07:24:35 +00:00
}
2017-01-09 09:54:49 +00:00
c = ( ( moo_oop_class_t ) c ) - > superclass ;
2015-06-04 18:34:37 +00:00
}
2017-01-09 09:54:49 +00:00
while ( c ! = moo - > _nil ) ;
2015-06-04 18:34:37 +00:00
2015-06-09 12:14:18 +00:00
not_found :
2017-02-15 11:57:24 +00:00
if ( cls = = moo - > _class )
2016-05-03 10:10:28 +00:00
{
/* the object is an instance of Class. find the method
* in an instance method dictionary of Class also */
2017-01-09 09:54:49 +00:00
mthdic = ( ( moo_oop_class_t ) cls ) - > mthdic [ MOO_METHOD_INSTANCE ] ;
MOO_ASSERT ( moo , ( moo_oop_t ) mthdic ! = moo - > _nil ) ;
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , mthdic ) = = moo - > _method_dictionary ) ;
2016-05-03 10:10:28 +00:00
2017-01-09 09:54:49 +00:00
ass = ( moo_oop_association_t ) moo_lookupdic ( moo , mthdic , message ) ;
2016-05-03 10:10:28 +00:00
if ( ass )
{
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , ass - > value ) = = moo - > _method ) ;
return ( moo_oop_method_t ) ass - > value ;
2016-05-03 10:10:28 +00:00
}
}
2017-01-09 09:54:49 +00:00
MOO_DEBUG3 ( moo , " Method [%.*js] not found for %O \n " , message - > len , message - > ptr , receiver ) ;
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_ENOENT ) ;
2017-01-09 09:54:49 +00:00
return MOO_NULL ;
2015-06-04 18:34:37 +00:00
}
2017-01-09 09:54:49 +00:00
static int start_initial_process_and_context ( moo_t * moo , const moo_oocs_t * objname , const moo_oocs_t * mthname )
2015-06-04 18:34:37 +00:00
{
2015-06-08 13:24:02 +00:00
/* the initial context is a fake context. if objname is 'Stix' and
* mthname is ' main ' , this function emulates message sending ' Stix main ' .
* it should emulate the following logical byte - code sequences :
*
* push Stix
* send # main
*/
2017-01-09 09:54:49 +00:00
moo_oop_context_t ctx ;
moo_oop_method_t mth ;
moo_oop_process_t proc ;
2017-07-20 16:33:53 +00:00
# if defined(INVOKE_DIRECTLY)
moo_oop_association_t ass ;
# else
moo_oop_t s1 , s2 ;
# endif
moo_oow_t tmp_count = 0 ;
2015-06-08 13:24:02 +00:00
2017-07-20 16:33:53 +00:00
moo_oocs_t startup ;
static moo_ooch_t str_startup [ ] = { ' s ' , ' t ' , ' a ' , ' r ' , ' t ' , ' u ' , ' p ' } ;
# if defined(INVOKE_DIRECTLY)
2015-06-04 18:34:37 +00:00
2017-01-09 09:54:49 +00:00
ass = moo_lookupsysdic ( moo , objname ) ;
2017-07-20 16:33:53 +00:00
if ( ! ass | | MOO_CLASSOF ( moo , ass - > value ) ! = moo - > _class )
{
MOO_DEBUG2 ( moo , " Cannot find a class - %.*js " , objname - > len , objname - > ptr ) ;
return - 1 ;
}
2015-06-04 18:34:37 +00:00
2017-12-08 15:28:51 +00:00
mth = moo_findmethod ( moo , ass - > value , mthname , 0 ) ;
2017-07-20 16:33:53 +00:00
if ( ! mth )
{
MOO_DEBUG4 ( moo , " Cannot find a method in %.*js - %.*js " , objname - > len , objname - > ptr , mthname - > len , mthname - > ptr ) ;
return - 1 ;
}
2015-06-04 18:34:37 +00:00
2017-01-09 09:54:49 +00:00
if ( MOO_OOP_TO_SMOOI ( mth - > tmpr_nargs ) > 0 )
2015-06-06 07:24:35 +00:00
{
/* this method expects more than 0 arguments.
* i can ' t use it as a start - up method .
2017-07-20 16:33:53 +00:00
TODO : overcome this problem - accept parameters . . . .
2015-06-06 07:24:35 +00:00
*/
2017-07-20 16:33:53 +00:00
MOO_DEBUG4 ( moo , " Arguments not supported for a startup method - %.*js>>%.*js " , objname - > len , objname - > ptr , mthname - > len , mthname - > ptr ) ;
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_EINVAL ) ;
2015-06-06 07:24:35 +00:00
return - 1 ;
}
2017-07-20 16:33:53 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & mth ) ; tmp_count + + ;
moo_pushtmp ( moo , ( moo_oop_t * ) & ass ) ; tmp_count + + ;
# else
startup . ptr = str_startup ;
startup . len = 7 ;
2017-12-08 15:28:51 +00:00
mth = moo_findmethod ( moo , ( moo_oop_t ) moo - > _system , & startup , 0 ) ;
2017-07-20 16:33:53 +00:00
if ( ! mth )
{
MOO_DEBUG0 ( moo , " Cannot find the startup method in the system class " ) ;
2017-07-25 15:26:04 +00:00
goto oops ;
2017-07-20 16:33:53 +00:00
}
if ( MOO_OOP_TO_SMOOI ( mth - > tmpr_nargs ) ! = 2 )
{
MOO_DEBUG1 ( moo , " Weird argument count %zd for a startup method - should be 2 " , MOO_OOP_TO_SMOOI ( mth - > tmpr_nargs ) ) ;
moo_seterrnum ( moo , MOO_EINVAL ) ;
2017-07-25 15:26:04 +00:00
goto oops ;
2017-07-20 16:33:53 +00:00
}
/* TODO: check if it's variadic.... it should be. and accept more than 2... */
moo_pushtmp ( moo , ( moo_oop_t * ) & mth ) ; tmp_count + + ;
s1 = moo_makesymbol ( moo , objname - > ptr , objname - > len ) ;
2017-07-25 15:26:04 +00:00
if ( ! s1 ) goto oops ;
2017-07-20 16:33:53 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & s1 ) ;
s2 = moo_makesymbol ( moo , mthname - > ptr , mthname - > len ) ;
2017-07-25 15:26:04 +00:00
if ( ! s2 ) goto oops ;
2017-07-20 16:33:53 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & s2 ) ;
# endif
/* create a fake initial context. */
ctx = ( moo_oop_context_t ) moo_instantiate ( moo , moo - > _method_context , MOO_NULL , MOO_OOP_TO_SMOOI ( mth - > tmpr_nargs ) ) ;
2017-07-25 15:26:04 +00:00
if ( ! ctx ) goto oops ;
2017-07-20 16:33:53 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & ctx ) ; tmp_count + + ;
2015-06-14 07:15:53 +00:00
/* TODO: handle preamble */
2015-06-06 07:24:35 +00:00
/* the initial context starts the life of the entire VM
* and is not really worked on except that it is used to call the
2016-02-11 14:26:26 +00:00
* initial method . so it doesn ' t really require any extra stack space . */
/* TODO: verify this theory of mine. */
2017-01-09 09:54:49 +00:00
moo - > ip = 0 ;
moo - > sp = - 1 ;
2015-06-25 13:37:50 +00:00
2017-01-09 09:54:49 +00:00
ctx - > ip = MOO_SMOOI_TO_OOP ( 0 ) ; /* point to the beginning */
ctx - > sp = MOO_SMOOI_TO_OOP ( - 1 ) ; /* pointer to -1 below the bottom */
2016-02-11 14:26:26 +00:00
ctx - > origin = ctx ; /* point to self */
2017-01-09 09:54:49 +00:00
ctx - > method_or_nargs = ( moo_oop_t ) mth ; /* fake. help SWITCH_ACTIVE_CONTEXT() not fail. TODO: create a static fake method and use it... instead of 'mth' */
2015-10-18 15:06:17 +00:00
/* [NOTE]
* the receiver field and the sender field of ctx are nils .
* especially , the fact that the sender field is nil is used by
* the main execution loop for breaking out of the loop */
2015-06-04 18:34:37 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > active_context = = MOO_NULL ) ;
MOO_ASSERT ( moo , moo - > active_method = = MOO_NULL ) ;
2015-06-20 03:07:11 +00:00
2017-01-09 09:54:49 +00:00
/* moo_gc() uses moo->processor when moo->active_context
* is not NULL . at this poinst , moo - > processor should point to
2016-02-12 16:23:26 +00:00
* an instance of ProcessScheduler . */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , ( moo_oop_t ) moo - > processor ! = moo - > _nil ) ;
2017-07-24 13:25:25 +00:00
MOO_ASSERT ( moo , moo - > processor - > runnable . count = = MOO_SMOOI_TO_OOP ( 0 ) ) ;
2015-06-16 04:31:28 +00:00
2016-02-12 16:23:26 +00:00
/* start_initial_process() calls the SWITCH_ACTIVE_CONTEXT() macro.
2017-01-09 09:54:49 +00:00
* the macro assumes a non - null value in moo - > active_context .
2017-07-20 16:33:53 +00:00
* let ' s forcefully set active_context to ctx directly . */
2017-01-09 09:54:49 +00:00
moo - > active_context = ctx ;
2016-02-12 16:23:26 +00:00
2017-01-09 09:54:49 +00:00
proc = start_initial_process ( moo , ctx ) ;
2017-07-20 16:33:53 +00:00
moo_poptmps ( moo , tmp_count ) ; tmp_count = 0 ;
2017-07-25 15:26:04 +00:00
if ( ! proc ) goto oops ;
2015-10-18 15:06:17 +00:00
2017-07-20 16:33:53 +00:00
# if defined(INVOKE_DIRECTLY)
2017-01-09 09:54:49 +00:00
MOO_STACK_PUSH ( moo , ass - > value ) ; /* push the receiver - the object referenced by 'objname' */
2017-07-20 16:33:53 +00:00
# else
MOO_STACK_PUSH ( moo , ( moo_oop_t ) moo - > _system ) ;
MOO_STACK_PUSH ( moo , s1 ) ;
MOO_STACK_PUSH ( moo , s2 ) ;
# endif
2017-01-09 09:54:49 +00:00
STORE_ACTIVE_SP ( moo ) ; /* moo->active_context->sp = MOO_SMOOI_TO_OOP(moo->sp) */
2016-02-12 16:23:26 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > processor - > active = = proc ) ;
MOO_ASSERT ( moo , moo - > processor - > active - > initial_context = = ctx ) ;
MOO_ASSERT ( moo , moo - > processor - > active - > current_context = = ctx ) ;
MOO_ASSERT ( moo , moo - > active_context = = ctx ) ;
2016-02-12 16:23:26 +00:00
/* emulate the message sending */
2017-07-20 16:33:53 +00:00
# if defined(INVOKE_DIRECTLY)
2017-01-09 09:54:49 +00:00
return activate_new_method ( moo , mth , 0 ) ;
2017-07-20 16:33:53 +00:00
# else
return activate_new_method ( moo , mth , 2 ) ;
# endif
2017-07-25 15:26:04 +00:00
oops :
if ( tmp_count > 0 ) moo_poptmps ( moo , tmp_count ) ;
return - 1 ;
2015-06-04 18:34:37 +00:00
}
2015-10-03 15:29:03 +00:00
/* ------------------------------------------------------------------------- */
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_dump ( moo_t * moo , moo_ooi_t nargs )
2015-06-11 09:11:18 +00:00
{
2017-01-09 09:54:49 +00:00
moo_ooi_t i ;
2015-06-11 09:11:18 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs > = 0 ) ;
2015-06-14 07:15:53 +00:00
2017-09-25 15:16:19 +00:00
/*moo_logbfmt (moo, 0, "RECEIVER: %O IN PID %d SP %d XSP %d\n", MOO_STACK_GET(moo, moo->sp - nargs), (int)MOO_OOP_TO_SMOOI(moo->processor->active->id), (int)moo->sp, (int)MOO_OOP_TO_SMOOI(moo->processor->active->sp));*/
2017-11-24 17:40:20 +00:00
moo_logbfmt ( moo , MOO_LOG_FATAL | MOO_LOG_APP , " RECEIVER: %O IN PID %d \n " , MOO_STACK_GET ( moo , moo - > sp - nargs ) , ( int ) MOO_OOP_TO_SMOOI ( moo - > processor - > active - > id ) ) ;
2015-06-14 07:15:53 +00:00
for ( i = nargs ; i > 0 ; )
2015-06-09 12:14:18 +00:00
{
2015-06-14 07:15:53 +00:00
- - i ;
2017-11-24 17:40:20 +00:00
moo_logbfmt ( moo , MOO_LOG_FATAL | MOO_LOG_APP , " ARGUMENT %zd: %O \n " , i , MOO_STACK_GET ( moo , moo - > sp - i ) ) ;
2015-06-14 07:15:53 +00:00
}
2015-06-11 09:11:18 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRETTORCV ( moo , nargs ) ; /* ^self */
return MOO_PF_SUCCESS ;
2015-06-14 07:15:53 +00:00
}
2015-06-09 12:14:18 +00:00
2017-01-09 09:54:49 +00:00
static void log_char_object ( moo_t * moo , moo_oow_t mask , moo_oop_char_t msg )
2016-07-01 16:31:47 +00:00
{
2017-01-09 09:54:49 +00:00
moo_ooi_t n ;
moo_oow_t rem ;
const moo_ooch_t * ptr ;
2016-07-01 16:31:47 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_OBJ_GET_FLAGS_TYPE ( msg ) = = MOO_OBJ_TYPE_CHAR ) ;
2016-07-01 16:31:47 +00:00
2017-01-09 09:54:49 +00:00
rem = MOO_OBJ_GET_SIZE ( msg ) ;
2016-07-01 16:31:47 +00:00
ptr = msg - > slot ;
start_over :
while ( rem > 0 )
{
if ( * ptr = = ' \0 ' )
{
2017-01-09 09:54:49 +00:00
n = moo_logbfmt ( moo , mask , " %jc " , * ptr ) ;
MOO_ASSERT ( moo , n = = 1 ) ;
2016-07-01 16:31:47 +00:00
rem - = n ;
ptr + = n ;
goto start_over ;
}
2017-01-09 09:54:49 +00:00
n = moo_logbfmt ( moo , mask , " %.*js " , rem , ptr ) ;
2016-07-01 16:31:47 +00:00
if ( n < = - 1 ) break ;
if ( n = = 0 )
{
/* to skip the unprinted character.
2017-01-06 10:42:59 +00:00
* actually , this check is not needed because of ' \0 ' skipped
2016-07-01 16:31:47 +00:00
* at the beginning of the loop */
2017-01-09 09:54:49 +00:00
n = moo_logbfmt ( moo , mask , " %jc " , * ptr ) ;
MOO_ASSERT ( moo , n = = 1 ) ;
2016-07-01 16:31:47 +00:00
}
rem - = n ;
ptr + = n ;
}
}
2017-07-20 16:33:53 +00:00
static moo_pfrc_t pf_add_to_be_finalized ( moo_t * moo , moo_ooi_t nargs )
{
/* TODO: check if it has already been added */
moo_regfinalizable ( moo , MOO_STACK_GETRCV ( moo , nargs ) ) ;
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
return MOO_PF_SUCCESS ;
}
static moo_pfrc_t pf_remove_to_be_finalized ( moo_t * moo , moo_ooi_t nargs )
{
/* TODO: check if it has already been added */
moo_deregfinalizable ( moo , MOO_STACK_GETRCV ( moo , nargs ) ) ;
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
return MOO_PF_SUCCESS ;
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_hash ( moo_t * moo , moo_ooi_t nargs )
2017-01-05 10:16:04 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
moo_oow_t hv ;
2017-01-05 10:16:04 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 0 ) ;
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-01-05 10:16:04 +00:00
2017-01-09 09:54:49 +00:00
switch ( MOO_OOP_GET_TAG ( rcv ) )
2017-01-05 10:16:04 +00:00
{
2017-01-10 13:56:19 +00:00
case MOO_OOP_TAG_SMOOI :
2017-01-09 09:54:49 +00:00
hv = MOO_OOP_TO_SMOOI ( rcv ) ;
2017-01-05 10:16:04 +00:00
break ;
2017-04-02 13:13:33 +00:00
case MOO_OOP_TAG_SMPTR :
hv = ( moo_oow_t ) MOO_OOP_TO_SMPTR ( rcv ) ;
break ;
2017-01-09 09:54:49 +00:00
case MOO_OOP_TAG_CHAR :
hv = MOO_OOP_TO_CHAR ( rcv ) ;
2017-01-05 10:16:04 +00:00
break ;
2017-01-09 09:54:49 +00:00
case MOO_OOP_TAG_ERROR :
hv = MOO_OOP_TO_ERROR ( rcv ) ;
2017-01-05 10:16:04 +00:00
break ;
default :
{
int type ;
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_OOP_IS_POINTER ( rcv ) ) ;
type = MOO_OBJ_GET_FLAGS_TYPE ( rcv ) ;
2017-01-05 10:16:04 +00:00
switch ( type )
{
2017-01-09 09:54:49 +00:00
case MOO_OBJ_TYPE_BYTE :
hv = moo_hashbytes ( ( ( moo_oop_byte_t ) rcv ) - > slot , MOO_OBJ_GET_SIZE ( rcv ) ) ;
2017-01-05 10:16:04 +00:00
break ;
2017-01-09 09:54:49 +00:00
case MOO_OBJ_TYPE_CHAR :
hv = moo_hashoochars ( ( ( moo_oop_char_t ) rcv ) - > slot , MOO_OBJ_GET_SIZE ( rcv ) ) ;
2017-01-05 10:16:04 +00:00
break ;
2017-01-09 09:54:49 +00:00
case MOO_OBJ_TYPE_HALFWORD :
hv = moo_hashhalfwords ( ( ( moo_oop_halfword_t ) rcv ) - > slot , MOO_OBJ_GET_SIZE ( rcv ) ) ;
2017-01-05 10:16:04 +00:00
break ;
2017-01-09 09:54:49 +00:00
case MOO_OBJ_TYPE_WORD :
hv = moo_hashwords ( ( ( moo_oop_word_t ) rcv ) - > slot , MOO_OBJ_GET_SIZE ( rcv ) ) ;
2017-01-05 10:16:04 +00:00
break ;
default :
2017-01-09 09:54:49 +00:00
/* MOO_OBJ_TYPE_OOP, ... */
2017-12-03 17:08:04 +00:00
moo_seterrbfmt ( moo , MOO_ENOIMPL , " no builtin hash implemented for %O " , rcv ) ; /* TODO: better error code? */
2017-01-09 09:54:49 +00:00
return MOO_PF_FAILURE ;
2017-01-05 10:16:04 +00:00
}
break ;
}
}
2017-01-09 09:54:49 +00:00
/* moo_hashxxx() functions should limit the return value to fall
* in the range between 0 and MOO_SMOOI_MAX inclusive */
MOO_ASSERT ( moo , hv > = 0 & & hv < = MOO_SMOOI_MAX ) ;
2017-01-06 13:27:49 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , MOO_SMOOI_TO_OOP ( hv ) ) ;
return MOO_PF_SUCCESS ;
2017-01-05 10:16:04 +00:00
}
2017-09-29 15:03:54 +00:00
2017-03-19 14:18:37 +00:00
static moo_pfrc_t pf_perform ( moo_t * moo , moo_ooi_t nargs )
{
moo_oop_t /*rcv,*/ selector ;
moo_oow_t ssp , esp , i ;
2017-04-01 04:58:02 +00:00
MOO_ASSERT ( moo , nargs > = 1 ) ; /* at least, a selector must be specified */
2017-03-19 14:18:37 +00:00
/*rcv = MOO_STACK_GETRCV(moo, nargs);*/
selector = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
if ( MOO_CLASSOF ( moo , selector ) ! = moo - > _symbol )
{
2017-12-03 17:08:04 +00:00
moo_seterrbfmt ( moo , MOO_EINVAL , " invalid argument - %O " , selector ) ;
2017-03-19 14:18:37 +00:00
return MOO_PF_FAILURE ;
}
/* remove the selector from the stack */
ssp = MOO_STACK_GETARGSP ( moo , nargs , 0 ) ;
esp = MOO_STACK_GETARGSP ( moo , nargs , nargs - 1 ) ;
2017-03-24 18:09:34 +00:00
for ( i = ssp ; i < esp ; i + + )
2017-03-19 14:18:37 +00:00
{
moo_oop_t t ;
2017-03-24 18:09:34 +00:00
t = MOO_STACK_GET ( moo , i + 1 ) ;
2017-03-19 14:18:37 +00:00
MOO_STACK_SET ( moo , i , t ) ;
}
MOO_STACK_POP ( moo ) ;
/* emulate message sending */
if ( send_message ( moo , ( moo_oop_char_t ) selector , 0 , nargs - 1 ) < = - 1 ) return MOO_PF_HARD_FAILURE ;
return MOO_PF_SUCCESS ;
}
2017-12-03 17:08:04 +00:00
/* ------------------------------------------------------------------ */
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_context_goto ( moo_t * moo , moo_ooi_t nargs )
2016-05-18 11:10:54 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
moo_oop_t pc ;
2016-05-18 11:10:54 +00:00
/* this primivie provides the similar functionality to MethodContext>>pc:
* except that it pops the receiver and arguments and doesn ' t push a
* return value . it ' s useful when you want to change the instruction
* pointer while maintaining the stack level before the call */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2016-05-18 11:10:54 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_CLASSOF ( moo , rcv ) = = moo - > _method_context ) ;
2016-05-18 11:10:54 +00:00
2017-01-09 09:54:49 +00:00
pc = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
if ( ! MOO_OOP_IS_SMOOI ( pc ) | | MOO_OOP_TO_SMOOI ( pc ) < 0 )
2016-05-18 11:10:54 +00:00
{
2017-12-03 17:08:04 +00:00
moo_seterrbfmt ( moo , MOO_EINVAL , " invalid pc - %O " , pc ) ;
2017-01-09 09:54:49 +00:00
return MOO_PF_FAILURE ;
2016-05-18 11:10:54 +00:00
}
2017-01-09 09:54:49 +00:00
( ( moo_oop_context_t ) rcv ) - > ip = pc ;
LOAD_ACTIVE_IP ( moo ) ;
2016-05-18 11:10:54 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs + 1 = = 2 ) ;
MOO_STACK_POPS ( moo , 2 ) ; /* pop both the argument and the receiver */
return MOO_PF_SUCCESS ;
2016-05-18 11:10:54 +00:00
}
2017-12-03 17:08:04 +00:00
/* ------------------------------------------------------------------ */
2017-01-09 09:54:49 +00:00
static moo_pfrc_t __block_value ( moo_t * moo , moo_oop_context_t rcv_blkctx , moo_ooi_t nargs , moo_ooi_t num_first_arg_elems , moo_oop_context_t * pblkctx )
2015-06-17 13:46:16 +00:00
{
2016-02-11 14:26:26 +00:00
/* prepare a new block context for activation.
* the receiver must be a block context which becomes the base
* for a new block context . */
2017-01-09 09:54:49 +00:00
moo_oop_context_t blkctx ;
moo_ooi_t local_ntmprs , i ;
moo_ooi_t actual_arg_count ;
2015-10-22 02:47:25 +00:00
actual_arg_count = ( num_first_arg_elems > 0 ) ? num_first_arg_elems : nargs ;
2015-06-17 13:46:16 +00:00
2015-06-22 14:21:46 +00:00
/* TODO: find a better way to support a reentrant block context. */
2015-06-17 13:46:16 +00:00
2015-06-22 14:21:46 +00:00
/* | sum |
* sum : = [ : n | ( n < 2 ) ifTrue : [ 1 ] ifFalse : [ n + ( sum value : ( n - 1 ) ) ] ] .
* ( sum value : 10 ) .
*
* For the code above , sum is a block context and it is sent value : inside
* itself . Let me simply clone a block context to allow reentrancy like this
* while the block context is active
*/
2016-08-10 14:41:45 +00:00
/* the receiver must be a block context */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , rcv_blkctx ) = = moo - > _block_context ) ;
if ( rcv_blkctx - > receiver_or_source ! = moo - > _nil )
2015-06-25 13:37:50 +00:00
{
2015-06-26 15:49:08 +00:00
/* the 'source' field is not nil.
* this block context has already been activated once .
2015-06-25 13:37:50 +00:00
* you can ' t send ' value ' again to reactivate it .
2016-03-26 06:39:56 +00:00
* For example , [ thisContext value ] value . */
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_OBJ_GET_SIZE ( rcv_blkctx ) > MOO_CONTEXT_NAMED_INSTVARS ) ;
2017-12-03 17:08:04 +00:00
moo_seterrbfmt ( moo , MOO_EPERM , " re-valuing of a block context - %O " , rcv_blkctx ) ;
2017-01-09 09:54:49 +00:00
return MOO_PF_FAILURE ;
2015-06-25 13:37:50 +00:00
}
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_OBJ_GET_SIZE ( rcv_blkctx ) = = MOO_CONTEXT_NAMED_INSTVARS ) ;
2015-06-25 13:37:50 +00:00
2017-01-09 09:54:49 +00:00
if ( MOO_OOP_TO_SMOOI ( rcv_blkctx - > method_or_nargs ) ! = actual_arg_count /* nargs */ )
2015-06-17 13:46:16 +00:00
{
2017-12-03 17:08:04 +00:00
moo_seterrbfmt ( moo , MOO_ENUMARGS ,
" wrong number of arguments to a block context %O - %zd expected, %zd given " ,
rcv_blkctx , MOO_OOP_TO_SMOOI ( rcv_blkctx - > method_or_nargs ) , actual_arg_count ) ;
2017-01-09 09:54:49 +00:00
return MOO_PF_FAILURE ;
2015-06-17 13:46:16 +00:00
}
2016-02-12 16:23:26 +00:00
/* the number of temporaries stored in the block context
* accumulates the number of temporaries starting from the origin .
* simple calculation is needed to find the number of local temporaries */
2017-01-09 09:54:49 +00:00
local_ntmprs = MOO_OOP_TO_SMOOI ( rcv_blkctx - > ntmprs ) -
MOO_OOP_TO_SMOOI ( ( ( moo_oop_context_t ) rcv_blkctx - > home ) - > ntmprs ) ;
MOO_ASSERT ( moo , local_ntmprs > = actual_arg_count ) ;
2016-02-12 16:23:26 +00:00
2016-03-26 06:39:56 +00:00
/* create a new block context to clone rcv_blkctx */
2017-01-09 09:54:49 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & rcv_blkctx ) ;
blkctx = ( moo_oop_context_t ) moo_instantiate ( moo , moo - > _block_context , MOO_NULL , local_ntmprs ) ;
moo_poptmp ( moo ) ;
2017-12-03 17:08:04 +00:00
if ( ! blkctx ) return MOO_PF_FAILURE ;
2015-06-22 14:21:46 +00:00
2015-06-25 13:37:50 +00:00
#if 0
/* shallow-copy the named part including home, origin, etc. */
2017-01-09 09:54:49 +00:00
for ( i = 0 ; i < MOO_CONTEXT_NAMED_INSTVARS ; i + + )
2015-06-22 14:21:46 +00:00
{
2017-01-09 09:54:49 +00:00
( ( moo_oop_oop_t ) blkctx ) - > slot [ i ] = ( ( moo_oop_oop_t ) rcv_blkctx ) - > slot [ i ] ;
2015-06-22 14:21:46 +00:00
}
2015-06-25 13:37:50 +00:00
# else
2016-03-26 06:39:56 +00:00
blkctx - > ip = rcv_blkctx - > ip ;
blkctx - > ntmprs = rcv_blkctx - > ntmprs ;
blkctx - > method_or_nargs = rcv_blkctx - > method_or_nargs ;
2017-01-09 09:54:49 +00:00
blkctx - > receiver_or_source = ( moo_oop_t ) rcv_blkctx ;
2016-03-26 06:39:56 +00:00
blkctx - > home = rcv_blkctx - > home ;
blkctx - > origin = rcv_blkctx - > origin ;
2015-10-08 14:26:04 +00:00
# endif
2015-06-22 14:21:46 +00:00
/* TODO: check the stack size of a block context to see if it's large enough to hold arguments */
2015-10-22 02:47:25 +00:00
if ( num_first_arg_elems > 0 )
2015-06-22 14:21:46 +00:00
{
2016-01-29 04:04:39 +00:00
/* the first argument should be an array. this function is ordered
* to pass array elements to the new block */
2017-01-09 09:54:49 +00:00
moo_oop_oop_t xarg ;
MOO_ASSERT ( moo , nargs = = 1 ) ;
xarg = ( moo_oop_oop_t ) MOO_STACK_GETTOP ( moo ) ;
2017-02-14 08:29:30 +00:00
MOO_ASSERT ( moo , MOO_OBJ_IS_OOP_POINTER ( xarg ) ) ;
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_OBJ_GET_SIZE ( xarg ) = = num_first_arg_elems ) ;
2015-10-22 02:47:25 +00:00
for ( i = 0 ; i < num_first_arg_elems ; i + + )
{
blkctx - > slot [ i ] = xarg - > slot [ i ] ;
}
}
else
{
/* copy the arguments to the stack */
for ( i = 0 ; i < nargs ; i + + )
{
2017-01-09 09:54:49 +00:00
blkctx - > slot [ i ] = MOO_STACK_GETARG ( moo , nargs , i ) ;
2015-10-22 02:47:25 +00:00
}
2015-06-22 14:21:46 +00:00
}
2017-01-09 09:54:49 +00:00
MOO_STACK_POPS ( moo , nargs + 1 ) ; /* pop arguments and receiver */
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , blkctx - > home ! = moo - > _nil ) ;
blkctx - > sp = MOO_SMOOI_TO_OOP ( - 1 ) ; /* not important at all */
blkctx - > sender = moo - > active_context ;
2015-06-17 13:46:16 +00:00
2015-10-22 02:47:25 +00:00
* pblkctx = blkctx ;
2017-01-09 09:54:49 +00:00
return MOO_PF_SUCCESS ;
2015-10-22 02:47:25 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_block_value ( moo_t * moo , moo_ooi_t nargs )
2015-10-22 02:47:25 +00:00
{
2017-01-09 09:54:49 +00:00
moo_pfrc_t x ;
moo_oop_context_t rcv_blkctx , blkctx ;
2015-10-22 02:47:25 +00:00
2017-01-09 09:54:49 +00:00
rcv_blkctx = ( moo_oop_context_t ) MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_CLASSOF ( moo , rcv_blkctx ) = = moo - > _block_context ) ;
2016-08-10 14:41:45 +00:00
2017-01-09 09:54:49 +00:00
x = __block_value ( moo , rcv_blkctx , nargs , 0 , & blkctx ) ;
if ( x < = MOO_PF_FAILURE ) return x ; /* hard failure and soft failure */
2015-10-22 02:47:25 +00:00
2017-01-09 09:54:49 +00:00
SWITCH_ACTIVE_CONTEXT ( moo , ( moo_oop_context_t ) blkctx ) ;
return MOO_PF_SUCCESS ;
2015-06-17 13:46:16 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_block_new_process ( moo_t * moo , moo_ooi_t nargs )
2015-10-18 15:06:17 +00:00
{
2016-02-11 14:26:26 +00:00
/* create a new process from a block context.
* the receiver must be be a block .
* [ 1 + 2 ] newProcess .
2017-12-03 17:08:04 +00:00
* [ : a : b | a + b ] newProcess ( 1 , 2 )
2016-02-11 14:26:26 +00:00
*/
2015-10-22 02:47:25 +00:00
int x ;
2017-01-09 09:54:49 +00:00
moo_oop_context_t rcv_blkctx , blkctx ;
moo_oop_process_t proc ;
2017-12-03 17:08:04 +00:00
#if 0
2017-01-09 09:54:49 +00:00
moo_ooi_t num_first_arg_elems = 0 ;
2015-10-22 02:47:25 +00:00
2017-07-24 13:25:25 +00:00
MOO_ASSERT ( moo , nargs < = 1 ) ;
2015-10-22 02:47:25 +00:00
if ( nargs = = 1 )
{
2017-01-09 09:54:49 +00:00
moo_oop_t xarg ;
2015-10-22 02:47:25 +00:00
2017-01-09 09:54:49 +00:00
xarg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2017-02-14 08:29:30 +00:00
if ( ! MOO_OBJ_IS_OOP_POINTER ( xarg ) )
2015-10-22 02:47:25 +00:00
{
2016-02-11 14:26:26 +00:00
/* the only optional argument must be an OOP-indexable
* object like an array */
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_EINVAL ) ;
2017-01-09 09:54:49 +00:00
return MOO_PF_FAILURE ;
2015-10-22 02:47:25 +00:00
}
2017-01-09 09:54:49 +00:00
num_first_arg_elems = MOO_OBJ_GET_SIZE ( xarg ) ;
2015-10-22 02:47:25 +00:00
}
2017-12-03 17:08:04 +00:00
# endif
2015-10-22 02:47:25 +00:00
2017-01-09 09:54:49 +00:00
rcv_blkctx = ( moo_oop_context_t ) MOO_STACK_GETRCV ( moo , nargs ) ;
2017-12-03 17:08:04 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_CLASSOF ( moo , rcv_blkctx ) = = moo - > _block_context ) ;
2016-03-26 06:39:56 +00:00
2015-10-22 02:47:25 +00:00
/* this primitive creates a new process with a block as if the block
* is sent the value message */
2017-12-03 17:08:04 +00:00
#if 0
2017-01-09 09:54:49 +00:00
x = __block_value ( moo , rcv_blkctx , nargs , num_first_arg_elems , & blkctx ) ;
2017-12-03 17:08:04 +00:00
# else
x = __block_value ( moo , rcv_blkctx , nargs , 0 , & blkctx ) ;
# endif
2016-02-11 14:26:26 +00:00
if ( x < = 0 ) return x ; /* both hard failure and soft failure */
2015-10-22 02:47:25 +00:00
2017-01-09 09:54:49 +00:00
/* reset the sender field to moo->_nil because this block context
2016-05-02 13:50:42 +00:00
* will be the initial context of a new process . you can simply
* inspect the sender field to see if a context is an initial
* context of a process . */
2017-01-09 09:54:49 +00:00
blkctx - > sender = ( moo_oop_context_t ) moo - > _nil ;
2016-05-02 13:50:42 +00:00
2017-01-09 09:54:49 +00:00
proc = make_process ( moo , blkctx ) ;
2017-12-03 17:08:04 +00:00
if ( ! proc ) return MOO_PF_FAILURE ; /* hard failure */ /* TOOD: can't this be treated as a soft failure? throw an exception instead?? */
2015-10-22 02:47:25 +00:00
/* __block_value() has popped all arguments and the receiver.
* PUSH the return value instead of changing the stack top */
2017-01-09 09:54:49 +00:00
MOO_STACK_PUSH ( moo , ( moo_oop_t ) proc ) ;
return MOO_PF_SUCCESS ;
2015-10-18 15:06:17 +00:00
}
2017-05-16 02:04:18 +00:00
/* ------------------------------------------------------------------ */
2017-09-25 15:16:19 +00:00
static moo_pfrc_t pf_process_sp ( moo_t * moo , moo_ooi_t nargs )
{
moo_oop_t rcv ;
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
MOO_PF_CHECK_RCV ( moo , MOO_CLASSOF ( moo , rcv ) = = moo - > _process ) ;
/* commit the SP register of the VM to the active process for accuracy */
STORE_ACTIVE_SP ( moo ) ;
MOO_STACK_SETRET ( moo , nargs , moo - > processor - > active - > sp ) ;
return MOO_PF_SUCCESS ;
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_process_resume ( moo_t * moo , moo_ooi_t nargs )
2016-02-19 15:52:56 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
2016-02-19 15:52:56 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_CLASSOF ( moo , rcv ) = = moo - > _process ) ;
2016-02-19 15:52:56 +00:00
2017-09-25 15:16:19 +00:00
/* set the return value before resume_process() in case it changes
* the active process . */
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
2017-07-25 15:26:04 +00:00
resume_process ( moo , ( moo_oop_process_t ) rcv ) ;
2016-02-19 15:52:56 +00:00
2017-01-09 09:54:49 +00:00
return MOO_PF_SUCCESS ;
2016-02-19 15:52:56 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_process_terminate ( moo_t * moo , moo_ooi_t nargs )
2016-02-14 06:35:18 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
2016-02-14 06:35:18 +00:00
2016-06-30 13:44:37 +00:00
/* TODO: need to run ensure blocks here..
* when it ' s executed here . it does ' t have to be in Exception > > handleException when there is no exception handler */
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_CLASSOF ( moo , rcv ) = = moo - > _process ) ;
2016-02-14 06:35:18 +00:00
2017-09-25 15:16:19 +00:00
/* set the return value before terminate_process() in case it changes
* the active process . */
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
2017-01-09 09:54:49 +00:00
terminate_process ( moo , ( moo_oop_process_t ) rcv ) ;
2016-02-14 06:35:18 +00:00
2017-01-09 09:54:49 +00:00
return MOO_PF_SUCCESS ;
2016-02-14 06:35:18 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_process_yield ( moo_t * moo , moo_ooi_t nargs )
2016-02-19 15:52:56 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
2016-02-19 15:52:56 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_CLASSOF ( moo , rcv ) = = moo - > _process ) ;
2016-02-19 15:52:56 +00:00
2017-09-25 15:16:19 +00:00
/* set the return value before yield_process() in case it changes
* the active process . */
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
2017-01-09 09:54:49 +00:00
yield_process ( moo , ( moo_oop_process_t ) rcv ) ;
2016-02-19 15:52:56 +00:00
2017-01-09 09:54:49 +00:00
return MOO_PF_SUCCESS ;
2016-02-19 15:52:56 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_process_suspend ( moo_t * moo , moo_ooi_t nargs )
2016-07-04 15:36:10 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
2016-07-04 15:36:10 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_CLASSOF ( moo , rcv ) = = moo - > _process ) ;
2016-07-04 15:36:10 +00:00
2017-09-25 15:16:19 +00:00
/* set the return value before suspend_process() in case it changes
* the active process . */
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
2017-01-09 09:54:49 +00:00
suspend_process ( moo , ( moo_oop_process_t ) rcv ) ;
2016-07-04 15:36:10 +00:00
2017-01-09 09:54:49 +00:00
return MOO_PF_SUCCESS ;
2016-07-04 15:36:10 +00:00
}
2017-05-16 02:04:18 +00:00
/* ------------------------------------------------------------------ */
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_semaphore_signal ( moo_t * moo , moo_ooi_t nargs )
2016-02-18 17:49:56 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
2016-02-18 17:49:56 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-30 04:49:54 +00:00
MOO_PF_CHECK_RCV ( moo , moo_iskindof ( moo , rcv , moo - > _semaphore ) ) ;
2017-09-25 15:16:19 +00:00
/* signal_semaphore() may change the active process though the
* implementation as of this writing makes runnable the process waiting
* on the signal to be processed . it is safer to set the return value
* before calling signal_sempahore ( ) */
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
2016-02-18 17:49:56 +00:00
2017-01-09 09:54:49 +00:00
signal_semaphore ( moo , ( moo_oop_semaphore_t ) rcv ) ;
2016-02-18 17:49:56 +00:00
2017-01-09 09:54:49 +00:00
return MOO_PF_SUCCESS ;
2016-02-18 17:49:56 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_semaphore_wait ( moo_t * moo , moo_ooi_t nargs )
2016-02-18 17:49:56 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
2016-02-18 17:49:56 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-30 04:49:54 +00:00
MOO_PF_CHECK_RCV ( moo , moo_iskindof ( moo , rcv , moo - > _semaphore ) ) ;
2017-09-25 15:16:19 +00:00
/* i must set the return value before calling await_semaphore().
* await_semaphore ( ) may switch the active process and the stack
* manipulation macros target at the active process . i ' m not supposed
* to change the return value of a new active process . */
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
2016-02-18 17:49:56 +00:00
2017-09-26 09:01:42 +00:00
#if 0
2017-09-21 14:22:36 +00:00
if ( await_semaphore ( moo , ( moo_oop_semaphore_t ) rcv ) < = - 1 )
{
2017-09-25 15:16:19 +00:00
/* i must switch the top because the return value has been set already */
MOO_STACK_SETTOP ( moo , MOO_ERROR_TO_OOP ( moo - > errnum ) ) ;
return MOO_PF_SUCCESS ;
2017-09-21 14:22:36 +00:00
}
2017-09-26 09:01:42 +00:00
# else
await_semaphore ( moo , ( moo_oop_semaphore_t ) rcv ) ;
# endif
2016-02-18 17:49:56 +00:00
2017-01-09 09:54:49 +00:00
return MOO_PF_SUCCESS ;
2016-02-18 17:49:56 +00:00
}
2017-10-08 15:40:32 +00:00
static moo_pfrc_t pf_semaphore_group_add_semaphore ( moo_t * moo , moo_ooi_t nargs )
{
moo_oop_semaphore_group_t rcv ;
moo_oop_semaphore_t sem ;
rcv = ( moo_oop_semaphore_group_t ) MOO_STACK_GETRCV ( moo , nargs ) ;
MOO_PF_CHECK_RCV ( moo , moo_iskindof ( moo , ( moo_oop_t ) rcv , moo - > _semaphore_group ) ) ;
sem = ( moo_oop_semaphore_t ) MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2017-11-24 17:40:20 +00:00
MOO_PF_CHECK_ARGS_STRICT ( moo , nargs , moo_iskindof ( moo , ( moo_oop_t ) sem , moo - > _semaphore ) ) ;
2017-10-08 15:40:32 +00:00
if ( ( moo_oop_t ) sem - > group = = moo - > _nil )
{
2017-10-08 18:13:10 +00:00
int sems_idx ;
sems_idx = MOO_OOP_TO_SMOOI ( sem - > count ) > 0 ? MOO_SEMAPHORE_GROUP_SEMS_SIG : MOO_SEMAPHORE_GROUP_SEMS_UNSIG ;
MOO_APPEND_TO_OOP_LIST ( moo , & rcv - > sems [ sems_idx ] , moo_oop_semaphore_t , sem , grm ) ;
2017-10-08 15:40:32 +00:00
sem - > group = rcv ;
2017-10-31 07:13:22 +00:00
2017-11-04 03:41:08 +00:00
if ( MOO_OOP_TO_SMOOI ( sem - > io_index ) > = 0 )
2017-10-31 07:13:22 +00:00
{
moo_ooi_t count ;
count = MOO_OOP_TO_SMOOI ( rcv - > sem_io_count ) ;
2017-11-04 03:41:08 +00:00
MOO_ASSERT ( moo , count > = 0 ) ;
2017-10-31 07:13:22 +00:00
count + + ;
rcv - > sem_io_count = MOO_SMOOI_TO_OOP ( count ) ;
2017-11-04 03:41:08 +00:00
if ( count = = 1 )
{
moo_oop_process_t wp ;
/* TODO: add sem_wait_count to process. no traversal... */
for ( wp = rcv - > waiting . first ; ( moo_oop_t ) wp ! = moo - > _nil ; wp = wp - > sem_wait . next )
2017-11-21 09:15:22 +00:00
{
2017-11-04 03:41:08 +00:00
moo - > sem_io_wait_count + + ;
2017-11-21 09:15:22 +00:00
}
2017-11-04 03:41:08 +00:00
}
2017-10-31 07:13:22 +00:00
}
2017-10-08 15:40:32 +00:00
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
}
2017-10-08 18:13:10 +00:00
else if ( sem - > group = = rcv )
{
/* do nothing. don't change the group of the semaphore */
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
}
2017-10-08 15:40:32 +00:00
else
{
/* the semaphore belongs to a group already */
2017-10-08 18:13:10 +00:00
/* TODO: is it better to move this semaphore to the new group? */
2017-10-08 15:40:32 +00:00
MOO_STACK_SETRETTOERROR ( moo , nargs , MOO_EPERM ) ;
}
return MOO_PF_SUCCESS ;
}
static moo_pfrc_t pf_semaphore_group_remove_semaphore ( moo_t * moo , moo_ooi_t nargs )
{
moo_oop_semaphore_group_t rcv ;
moo_oop_semaphore_t sem ;
rcv = ( moo_oop_semaphore_group_t ) MOO_STACK_GETRCV ( moo , nargs ) ;
MOO_PF_CHECK_RCV ( moo , moo_iskindof ( moo , ( moo_oop_t ) rcv , moo - > _semaphore_group ) ) ;
sem = ( moo_oop_semaphore_t ) MOO_STACK_GETARG ( moo , nargs , 0 ) ;
MOO_PF_CHECK_ARGS ( moo , nargs , moo_iskindof ( moo , ( moo_oop_t ) sem , moo - > _semaphore ) ) ;
2017-10-08 18:13:10 +00:00
if ( sem - > group = = rcv )
2017-10-08 15:40:32 +00:00
{
2017-10-08 18:13:10 +00:00
int sems_idx ;
2017-11-03 16:26:55 +00:00
2017-11-04 03:41:08 +00:00
#if 0
2017-11-03 16:26:55 +00:00
if ( ( moo_oop_t ) rcv - > waiting . first ! = moo - > _nil )
{
/* there is a process waiting on this semaphore group.
* i don ' t allow a semaphore to be removed from the group .
* i want to dodge potential problems arising when removal is allowed .
*
* for instance , consider this psuedo code .
* sg addSemaphore : s
* [ sg wait ] fork .
* [ sg wait ] fork .
* [ sg wait ] fork .
* sg removeSemaphore : s .
*
*/
MOO_STACK_SETRETTOERROR ( moo , nargs , MOO_EPERM ) ;
goto done ;
}
2017-11-04 03:41:08 +00:00
# endif
2017-11-03 16:26:55 +00:00
2017-10-08 18:13:10 +00:00
sems_idx = MOO_OOP_TO_SMOOI ( sem - > count ) > 0 ? MOO_SEMAPHORE_GROUP_SEMS_SIG : MOO_SEMAPHORE_GROUP_SEMS_UNSIG ;
MOO_DELETE_FROM_OOP_LIST ( moo , & rcv - > sems [ sems_idx ] , sem , grm ) ;
sem - > grm . prev = ( moo_oop_semaphore_t ) moo - > _nil ;
sem - > grm . next = ( moo_oop_semaphore_t ) moo - > _nil ;
sem - > group = ( moo_oop_semaphore_group_t ) moo - > _nil ;
2017-10-31 07:13:22 +00:00
if ( MOO_OOP_TO_SMOOI ( sem - > io_index ) > = 0 )
{
moo_ooi_t count ;
count = MOO_OOP_TO_SMOOI ( rcv - > sem_io_count ) ;
2017-11-03 16:26:55 +00:00
MOO_ASSERT ( moo , count > 0 ) ;
2017-10-31 07:13:22 +00:00
count - - ;
rcv - > sem_io_count = MOO_SMOOI_TO_OOP ( count ) ;
2017-11-04 03:41:08 +00:00
if ( count = = 0 )
{
moo_oop_process_t wp ;
/* TODO: add sem_wait_count to process. no traversal... */
for ( wp = rcv - > waiting . first ; ( moo_oop_t ) wp ! = moo - > _nil ; wp = wp - > sem_wait . next )
2017-11-21 09:15:22 +00:00
{
2017-11-04 03:41:08 +00:00
moo - > sem_io_wait_count - - ;
2017-11-21 09:15:22 +00:00
}
2017-11-04 03:41:08 +00:00
}
2017-10-31 07:13:22 +00:00
}
2017-10-08 18:13:10 +00:00
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
2017-10-08 15:40:32 +00:00
}
else
{
2017-10-08 18:13:10 +00:00
/* it doesn't belong to a group or belongs to a different group */
MOO_STACK_SETRETTOERROR ( moo , nargs , MOO_EPERM ) ;
2017-10-08 15:40:32 +00:00
}
2017-11-03 16:26:55 +00:00
done :
2017-10-08 15:40:32 +00:00
return MOO_PF_SUCCESS ;
}
2017-08-22 13:45:37 +00:00
static moo_pfrc_t pf_semaphore_group_wait ( moo_t * moo , moo_ooi_t nargs )
{
2017-09-26 09:01:42 +00:00
moo_oop_t rcv , sem ;
2017-08-22 13:45:37 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-30 04:49:54 +00:00
MOO_PF_CHECK_RCV ( moo , moo_iskindof ( moo , rcv , moo - > _semaphore_group ) ) ;
2017-09-25 15:16:19 +00:00
/* i must set the return value before calling await_semaphore_group().
* MOO_STACK_SETRETTORCV ( ) manipulates the stack of the currently active
* process ( moo - > processor - > active ) . moo - > processor - > active may become
* moo - > nil_process if the current active process must get suspended .
* it is safer to set the return value of the calling method here .
* but the arguments and the receiver information will be lost from
* the stack from this moment on . */
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
2017-08-22 13:45:37 +00:00
2017-10-08 15:40:32 +00:00
sem = await_semaphore_group ( moo , ( moo_oop_semaphore_group_t ) rcv ) ;
2017-09-26 09:01:42 +00:00
if ( sem ! = moo - > _nil )
{
2017-11-03 16:26:55 +00:00
/* there was a signaled semaphore. the active process won't get
2017-09-26 09:01:42 +00:00
* suspended . change the return value of the current process
* forcibly to the signaled semaphore */
MOO_STACK_SETTOP ( moo , sem ) ;
}
/* the return value will get changed to an actual semaphore signaled
* when the semaphore is signaled . see signal_semaphore ( ) */
2017-08-22 13:45:37 +00:00
return MOO_PF_SUCCESS ;
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_processor_schedule ( moo_t * moo , moo_ooi_t nargs )
2016-02-14 06:35:18 +00:00
{
2017-09-25 15:16:19 +00:00
moo_oop_t arg ;
2016-02-14 06:35:18 +00:00
2017-09-25 15:16:19 +00:00
/*MOO_PF_CHECK_RCV (moo, MOO_STACK_GETRCV(moo, nargs) == (moo_oop_t)moo->processor);*/
2016-02-14 06:35:18 +00:00
2017-09-25 15:16:19 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2017-01-09 09:54:49 +00:00
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_ARGS ( moo , nargs , MOO_CLASSOF ( moo , arg ) = = moo - > _process ) ;
2016-02-14 06:35:18 +00:00
2017-09-25 15:16:19 +00:00
/* set the return value before resume_process() in case it changes the active process */
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
2016-02-14 06:35:18 +00:00
2017-07-25 15:26:04 +00:00
resume_process ( moo , ( moo_oop_process_t ) arg ) ;
2017-09-25 15:16:19 +00:00
2017-07-25 15:26:04 +00:00
return MOO_PF_SUCCESS ;
}
2017-11-05 16:47:13 +00:00
/* ------------------------------------------------------------------ */
static moo_pfrc_t pf_system_add_gcfin_semaphore ( moo_t * moo , moo_ooi_t nargs )
2017-07-25 15:26:04 +00:00
{
moo_oop_semaphore_t sem ;
2017-09-25 15:16:19 +00:00
/*MOO_PF_CHECK_RCV (moo, MOO_STACK_GETRCV(moo, nargs) == (moo_oop_t)moo->processor);*/
2017-07-25 15:26:04 +00:00
2017-09-25 15:16:19 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2017-07-25 15:26:04 +00:00
sem = ( moo_oop_semaphore_t ) MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2017-10-02 01:22:49 +00:00
MOO_PF_CHECK_ARGS ( moo , nargs , moo_iskindof ( moo , ( moo_oop_t ) sem , moo - > _semaphore ) ) ;
2017-07-25 15:26:04 +00:00
/* TODO: no overwriting.. */
moo - > sem_gcfin = sem ;
MOO_STACK_SETRETTORCV ( moo , nargs ) ; /* ^self */
2017-01-09 09:54:49 +00:00
return MOO_PF_SUCCESS ;
2016-02-14 06:35:18 +00:00
}
2017-11-05 16:47:13 +00:00
static moo_pfrc_t pf_system_add_timed_semaphore ( moo_t * moo , moo_ooi_t nargs )
2016-03-22 14:18:07 +00:00
{
2017-09-25 15:16:19 +00:00
moo_oop_t sec , nsec ;
2017-01-09 09:54:49 +00:00
moo_oop_semaphore_t sem ;
moo_ntime_t now , ft ;
2016-03-22 14:18:07 +00:00
2017-11-05 16:47:13 +00:00
/* don't care about the receiver much as the receiver is not used at all.
* however , it ' s inteded to be called from the System class . */
2017-09-25 15:16:19 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs > = 2 | | nargs < = 3 ) ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
sem = ( moo_oop_semaphore_t ) MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2017-11-05 16:47:13 +00:00
sec = MOO_STACK_GETARG ( moo , nargs , 1 ) ;
nsec = ( nargs = = 3 ? MOO_STACK_GETARG ( moo , nargs , 2 ) : MOO_SMOOI_TO_OOP ( 0 ) ) ;
2016-03-22 14:18:07 +00:00
2017-11-05 16:47:13 +00:00
MOO_PF_CHECK_ARGS ( moo , nargs ,
moo_iskindof ( moo , ( moo_oop_t ) sem , moo - > _semaphore ) & &
MOO_OOP_IS_SMOOI ( sec ) & & MOO_OOP_IS_SMOOI ( nsec ) ) ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
if ( MOO_OOP_IS_SMOOI ( sem - > heap_index ) & &
sem - > heap_index ! = MOO_SMOOI_TO_OOP ( - 1 ) )
2016-03-22 14:18:07 +00:00
{
2017-02-12 18:59:03 +00:00
/* if the semaphore is already been added. remove it first */
2017-01-09 09:54:49 +00:00
delete_from_sem_heap ( moo , MOO_OOP_TO_SMOOI ( sem - > heap_index ) ) ;
MOO_ASSERT ( moo , sem - > heap_index = = MOO_SMOOI_TO_OOP ( - 1 ) ) ;
2016-03-22 14:18:07 +00:00
/*
Is this more desired ? ? ?
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , moo - > _false ) ;
return MOO_PF_SUCCESS ;
2016-03-22 14:18:07 +00:00
*/
}
2017-02-10 09:06:53 +00:00
/* this code assumes that the monotonic clock returns a small value
2017-02-12 18:59:03 +00:00
* that can fit into a SmallInteger , even after some additions . */
2017-01-09 09:54:49 +00:00
vm_gettime ( moo , & now ) ;
MOO_ADDNTIMESNS ( & ft , & now , MOO_OOP_TO_SMOOI ( sec ) , MOO_OOP_TO_SMOOI ( nsec ) ) ;
if ( ft . sec < 0 | | ft . sec > MOO_SMOOI_MAX )
2016-03-22 14:18:07 +00:00
{
2017-09-25 15:16:19 +00:00
/* soft error - cannot represent the expiry time in a small integer. */
2017-01-09 09:54:49 +00:00
MOO_LOG3 ( moo , MOO_LOG_PRIMITIVE | MOO_LOG_ERROR ,
2016-06-26 15:03:12 +00:00
" Error(%hs) - time (%ld) out of range(0 - %zd) when adding a timed semaphore \n " ,
2017-01-09 09:54:49 +00:00
__PRIMITIVE_NAME__ , ( unsigned long int ) ft . sec , ( moo_ooi_t ) MOO_SMOOI_MAX ) ;
2017-09-25 15:16:19 +00:00
moo_seterrnum ( moo , MOO_ERANGE ) ;
2017-01-09 09:54:49 +00:00
return MOO_PF_FAILURE ;
2016-03-22 14:18:07 +00:00
}
2017-01-09 09:54:49 +00:00
sem - > heap_ftime_sec = MOO_SMOOI_TO_OOP ( ft . sec ) ;
sem - > heap_ftime_nsec = MOO_SMOOI_TO_OOP ( ft . nsec ) ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
if ( add_to_sem_heap ( moo , sem ) < = - 1 ) return MOO_PF_HARD_FAILURE ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRETTORCV ( moo , nargs ) ; /* ^self */
return MOO_PF_SUCCESS ;
2016-03-22 14:18:07 +00:00
}
2017-11-05 16:47:13 +00:00
static moo_pfrc_t __system_add_io_semaphore ( moo_t * moo , moo_ooi_t nargs , moo_ooi_t mask )
2017-02-12 18:59:03 +00:00
{
2017-09-25 15:16:19 +00:00
moo_oop_t fd ;
2017-02-12 18:59:03 +00:00
moo_oop_semaphore_t sem ;
2017-09-25 15:16:19 +00:00
/*MOO_PF_CHECK_RCV (moo, MOO_STACK_GETRCV(moo, nargs) == (moo_oop_t)moo->processor);*/
2017-02-15 11:57:24 +00:00
MOO_ASSERT ( moo , nargs = = 2 ) ;
2017-02-12 18:59:03 +00:00
2017-02-15 11:57:24 +00:00
fd = MOO_STACK_GETARG ( moo , nargs , 1 ) ;
2017-02-12 18:59:03 +00:00
sem = ( moo_oop_semaphore_t ) MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2017-04-08 07:23:31 +00:00
if ( MOO_CLASSOF ( moo , sem ) ! = moo - > _semaphore | | ! MOO_OOP_IS_SMOOI ( fd ) )
2017-02-12 18:59:03 +00:00
{
2017-04-08 07:23:31 +00:00
MOO_STACK_SETRETTOERROR ( moo , nargs , MOO_EINVAL ) ;
return MOO_PF_SUCCESS ;
}
if ( MOO_OOP_IS_SMOOI ( sem - > io_index ) & & sem - > io_index ! = MOO_SMOOI_TO_OOP ( - 1 ) & & sem - > io_handle = = fd )
{
moo_ooi_t old_mask ;
2017-09-25 15:16:19 +00:00
/* the semaphore is already linked with the requested IO handle */
2017-04-08 07:23:31 +00:00
old_mask = MOO_OOP_TO_SMOOI ( sem - > io_mask ) ;
if ( old_mask ! = mask )
{
sem - > io_mask = MOO_SMOOI_TO_OOP ( mask ) ;
2017-09-25 15:16:19 +00:00
if ( modify_in_sem_io ( moo , sem ) < = - 1 )
2017-04-08 07:23:31 +00:00
{
sem - > io_mask = MOO_SMOOI_TO_OOP ( old_mask ) ;
MOO_STACK_SETRETTOERRNUM ( moo , nargs ) ;
return MOO_PF_SUCCESS ;
}
}
2017-02-12 18:59:03 +00:00
}
2017-04-08 07:23:31 +00:00
else
{
if ( MOO_OOP_IS_SMOOI ( sem - > io_index ) & & sem - > io_index ! = MOO_SMOOI_TO_OOP ( - 1 ) )
{
/* remove it if it's already added for IO */
2017-09-25 15:16:19 +00:00
if ( delete_from_sem_io ( moo , MOO_OOP_TO_SMOOI ( sem - > io_index ) ) < = - 1 )
2017-04-08 07:23:31 +00:00
{
MOO_STACK_SETRETTOERRNUM ( moo , nargs ) ;
return MOO_PF_SUCCESS ;
}
MOO_ASSERT ( moo , sem - > io_index = = MOO_SMOOI_TO_OOP ( - 1 ) ) ;
}
2017-02-12 18:59:03 +00:00
2017-04-08 07:23:31 +00:00
sem - > io_handle = fd ;
sem - > io_mask = MOO_SMOOI_TO_OOP ( mask ) ;
2017-09-25 15:16:19 +00:00
if ( add_to_sem_io ( moo , sem ) < = - 1 )
2017-04-08 07:23:31 +00:00
{
MOO_STACK_SETRETTOERRNUM ( moo , nargs ) ;
return MOO_PF_SUCCESS ;
}
}
2017-02-12 18:59:03 +00:00
MOO_STACK_SETRETTORCV ( moo , nargs ) ; /* ^self */
return MOO_PF_SUCCESS ;
}
2017-11-05 16:47:13 +00:00
static moo_pfrc_t pf_system_add_input_semaphore ( moo_t * moo , moo_ooi_t nargs )
2017-02-15 11:57:24 +00:00
{
2017-11-05 16:47:13 +00:00
return __system_add_io_semaphore ( moo , nargs , MOO_SEMAPHORE_IO_MASK_INPUT ) ;
2017-02-15 11:57:24 +00:00
}
2017-11-05 16:47:13 +00:00
static moo_pfrc_t pf_system_add_output_semaphore ( moo_t * moo , moo_ooi_t nargs )
2017-02-15 11:57:24 +00:00
{
2017-11-05 16:47:13 +00:00
return __system_add_io_semaphore ( moo , nargs , MOO_SEMAPHORE_IO_MASK_OUTPUT ) ;
2017-02-15 11:57:24 +00:00
}
2017-11-05 16:47:13 +00:00
static moo_pfrc_t pf_system_add_inoutput_semaphore ( moo_t * moo , moo_ooi_t nargs )
2017-02-15 11:57:24 +00:00
{
2017-11-05 16:47:13 +00:00
return __system_add_io_semaphore ( moo , nargs , MOO_SEMAPHORE_IO_MASK_INPUT | MOO_SEMAPHORE_IO_MASK_OUTPUT ) ;
2017-02-15 11:57:24 +00:00
}
2017-11-05 16:47:13 +00:00
static moo_pfrc_t pf_system_remove_semaphore ( moo_t * moo , moo_ooi_t nargs )
2016-03-22 14:18:07 +00:00
{
2016-03-24 14:58:47 +00:00
/* remove a semaphore from processor's signal scheduling */
2017-01-09 09:54:49 +00:00
moo_oop_semaphore_t sem ;
2016-03-22 14:18:07 +00:00
2017-09-25 15:16:19 +00:00
/*MOO_PF_CHECK_RCV (moo, MOO_STACK_GETRCV(moo, nargs) == (moo_oop_t)moo->processor);*/
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2016-03-22 14:18:07 +00:00
2017-01-09 09:54:49 +00:00
sem = ( moo_oop_semaphore_t ) MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2016-03-24 14:58:47 +00:00
/* TODO: remove a semaphore from IO handler if it's registered...
2017-07-25 15:26:04 +00:00
* remove a semaphore from elsewhere registered too */
2016-03-22 14:18:07 +00:00
2017-04-08 07:23:31 +00:00
if ( MOO_CLASSOF ( moo , sem ) ! = moo - > _semaphore )
{
MOO_STACK_SETRETTOERROR ( moo , nargs , MOO_EINVAL ) ;
return MOO_PF_SUCCESS ;
}
2016-03-22 14:18:07 +00:00
2017-07-25 15:26:04 +00:00
if ( sem = = moo - > sem_gcfin )
{
moo - > sem_gcfin = ( moo_oop_semaphore_t ) moo - > _nil ;
}
2017-01-09 09:54:49 +00:00
if ( MOO_OOP_IS_SMOOI ( sem - > heap_index ) & &
sem - > heap_index ! = MOO_SMOOI_TO_OOP ( - 1 ) )
2016-03-22 14:18:07 +00:00
{
2016-03-24 14:58:47 +00:00
/* the semaphore is in the timed semaphore heap */
2017-01-09 09:54:49 +00:00
delete_from_sem_heap ( moo , MOO_OOP_TO_SMOOI ( sem - > heap_index ) ) ;
MOO_ASSERT ( moo , sem - > heap_index = = MOO_SMOOI_TO_OOP ( - 1 ) ) ;
2016-03-22 14:18:07 +00:00
}
2017-02-12 18:59:03 +00:00
if ( MOO_OOP_IS_SMOOI ( sem - > io_index ) & &
sem - > io_index ! = MOO_SMOOI_TO_OOP ( - 1 ) )
{
2017-09-25 15:16:19 +00:00
/* the semaphore is associated with IO */
2017-04-08 07:23:31 +00:00
if ( delete_from_sem_io ( moo , MOO_OOP_TO_SMOOI ( sem - > io_index ) ) < = - 1 )
{
MOO_STACK_SETRETTOERRNUM ( moo , nargs ) ;
return MOO_PF_SUCCESS ;
}
2017-02-12 18:59:03 +00:00
MOO_ASSERT ( moo , sem - > io_index = = MOO_SMOOI_TO_OOP ( - 1 ) ) ;
}
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRETTORCV ( moo , nargs ) ; /* ^self */
return MOO_PF_SUCCESS ;
2016-03-22 14:18:07 +00:00
}
2017-11-05 16:47:13 +00:00
/* ------------------------------------------------------------------ */
2017-11-21 09:15:22 +00:00
static moo_pfrc_t pf_system_return_value_to_context ( moo_t * moo , moo_ooi_t nargs )
2016-03-28 13:25:36 +00:00
{
2017-09-25 15:16:19 +00:00
moo_oop_t ret , ctx ;
/*MOO_PF_CHECK_RCV (moo, MOO_STACK_GETRCV(moo, nargs) == (moo_oop_t)moo->processor);*/
2016-03-28 13:25:36 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 2 ) ;
2016-03-28 13:25:36 +00:00
2017-01-09 09:54:49 +00:00
ret = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
ctx = MOO_STACK_GETARG ( moo , nargs , 1 ) ;
2016-03-28 13:25:36 +00:00
2017-11-21 09:15:22 +00:00
MOO_PF_CHECK_ARGS_STRICT ( moo , nargs , MOO_CLASSOF ( moo , ctx ) = = moo - > _block_context | |
MOO_CLASSOF ( moo , ctx ) = = moo - > _method_context ) ;
2016-04-30 05:56:35 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_POPS ( moo , nargs + 1 ) ; /* pop arguments and receiver */
2017-11-21 09:15:22 +00:00
/* TODO: verify if this is correct? does't it correctly restore the stack pointer?
2016-06-24 14:29:43 +00:00
* test complex chains of method contexts and block contexts */
2017-01-09 09:54:49 +00:00
if ( MOO_CLASSOF ( moo , ctx ) = = moo - > _method_context )
2016-06-24 14:29:43 +00:00
{
/* when returning to a method context, load the sp register with
* the value stored in the context */
2017-01-09 09:54:49 +00:00
moo - > sp = MOO_OOP_TO_SMOOI ( ( ( moo_oop_context_t ) ctx ) - > sp ) ;
2016-06-24 14:29:43 +00:00
}
2017-01-09 09:54:49 +00:00
MOO_STACK_PUSH ( moo , ret ) ;
2016-05-02 13:50:42 +00:00
2017-01-09 09:54:49 +00:00
SWITCH_ACTIVE_CONTEXT ( moo , ( moo_oop_context_t ) ctx ) ;
return MOO_PF_SUCCESS ;
2016-05-02 13:50:42 +00:00
}
2017-05-16 02:04:18 +00:00
/* ------------------------------------------------------------------ */
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_add ( moo_t * moo , moo_ooi_t nargs )
2015-06-20 03:07:11 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-06-20 03:07:11 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-06-20 03:07:11 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-06-20 03:07:11 +00:00
2017-01-09 09:54:49 +00:00
res = moo_addints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-10-30 15:36:37 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-06-22 14:21:46 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_sub ( moo_t * moo , moo_ooi_t nargs )
2015-06-22 14:21:46 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
res = moo_subints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-10-30 15:36:37 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-06-22 14:21:46 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_mul ( moo_t * moo , moo_ooi_t nargs )
2015-07-01 07:21:54 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-07-01 07:21:54 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-07-01 07:21:54 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-07-01 07:21:54 +00:00
2017-01-09 09:54:49 +00:00
res = moo_mulints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-11-11 13:31:05 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-11-22 13:32:06 +00:00
}
2015-07-01 07:21:54 +00:00
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_quo ( moo_t * moo , moo_ooi_t nargs )
2015-11-22 13:32:06 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , quo ;
2015-07-01 07:21:54 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-11-22 13:32:06 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-11-22 13:32:06 +00:00
2017-01-09 09:54:49 +00:00
quo = moo_divints ( moo , rcv , arg , 0 , MOO_NULL ) ;
if ( ! quo ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
/* TODO: MOO_EDIVBY0 soft or hard failure? */
2015-11-11 13:31:05 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , quo ) ;
return MOO_PF_SUCCESS ;
2015-11-22 13:32:06 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_rem ( moo_t * moo , moo_ooi_t nargs )
2015-11-22 13:32:06 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , quo , rem ;
2015-11-22 13:32:06 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-11-22 13:32:06 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-11-22 13:32:06 +00:00
2017-01-09 09:54:49 +00:00
quo = moo_divints ( moo , rcv , arg , 0 , & rem ) ;
if ( ! quo ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
/* TODO: MOO_EDIVBY0 soft or hard failure? */
2015-11-22 14:48:09 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , rem ) ;
return MOO_PF_SUCCESS ;
2015-11-22 14:48:09 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_quo2 ( moo_t * moo , moo_ooi_t nargs )
2015-11-22 14:48:09 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , quo ;
2015-11-22 14:48:09 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-11-22 14:48:09 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-11-22 14:48:09 +00:00
2017-01-09 09:54:49 +00:00
quo = moo_divints ( moo , rcv , arg , 1 , MOO_NULL ) ;
if ( ! quo ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
/* TODO: MOO_EDIVBY0 soft or hard failure? */
2015-11-22 14:48:09 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , quo ) ;
return MOO_PF_SUCCESS ;
2015-11-22 14:48:09 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_rem2 ( moo_t * moo , moo_ooi_t nargs )
2015-11-22 14:48:09 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , quo , rem ;
2015-11-22 14:48:09 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-11-22 14:48:09 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-11-22 14:48:09 +00:00
2017-01-09 09:54:49 +00:00
quo = moo_divints ( moo , rcv , arg , 1 , & rem ) ;
if ( ! quo ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
/* TODO: MOO_EDIVBY0 soft or hard failure? */
2015-11-22 13:32:06 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , rem ) ;
return MOO_PF_SUCCESS ;
2015-07-01 07:21:54 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_negated ( moo_t * moo , moo_ooi_t nargs )
2015-12-25 05:09:17 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , res ;
2015-12-25 05:09:17 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 0 ) ;
2015-12-25 05:09:17 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2015-12-25 05:09:17 +00:00
2017-01-09 09:54:49 +00:00
res = moo_negateint ( moo , rcv ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-12-25 05:09:17 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-12-25 05:09:17 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_bitat ( moo_t * moo , moo_ooi_t nargs )
2015-12-25 05:09:17 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-12-25 05:09:17 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-12-25 05:09:17 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-12-25 05:09:17 +00:00
2017-01-09 09:54:49 +00:00
res = moo_bitatint ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-12-25 05:09:17 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-12-25 05:09:17 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_bitand ( moo_t * moo , moo_ooi_t nargs )
2015-12-10 14:23:09 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
res = moo_bitandints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-12-10 14:23:09 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_bitor ( moo_t * moo , moo_ooi_t nargs )
2015-12-10 14:23:09 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
res = moo_bitorints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-12-10 14:23:09 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_bitxor ( moo_t * moo , moo_ooi_t nargs )
2015-12-10 14:23:09 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
res = moo_bitxorints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-12-10 14:23:09 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-12-10 14:23:09 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_bitinv ( moo_t * moo , moo_ooi_t nargs )
2015-12-17 16:37:26 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , res ;
2015-12-17 16:37:26 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 0 ) ;
2015-12-17 16:37:26 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2015-12-17 16:37:26 +00:00
2017-01-09 09:54:49 +00:00
res = moo_bitinvint ( moo , rcv ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-12-17 16:37:26 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-12-17 16:37:26 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_bitshift ( moo_t * moo , moo_ooi_t nargs )
2015-12-18 15:58:45 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-12-18 15:58:45 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-12-18 15:58:45 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-12-18 15:58:45 +00:00
2017-01-09 09:54:49 +00:00
res = moo_bitshiftint ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-12-18 15:58:45 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-12-18 15:58:45 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_eq ( moo_t * moo , moo_ooi_t nargs )
2015-07-01 07:21:54 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-07-01 07:21:54 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-07-01 07:21:54 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-07-01 07:21:54 +00:00
2017-01-09 09:54:49 +00:00
res = moo_eqints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-07-01 07:21:54 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-07-01 07:21:54 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_ne ( moo_t * moo , moo_ooi_t nargs )
2015-10-04 16:21:31 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
res = moo_neints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-10-04 16:21:31 +00:00
}
2015-12-27 18:02:59 +00:00
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_lt ( moo_t * moo , moo_ooi_t nargs )
2015-06-22 14:21:46 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
res = moo_ltints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-06-20 03:07:11 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-06-20 03:07:11 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_gt ( moo_t * moo , moo_ooi_t nargs )
2015-06-22 14:21:46 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
res = moo_gtints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-06-22 14:21:46 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-06-22 14:21:46 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_le ( moo_t * moo , moo_ooi_t nargs )
2015-10-04 16:21:31 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
res = moo_leints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-10-04 16:21:31 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_ge ( moo_t * moo , moo_ooi_t nargs )
2015-10-04 16:21:31 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , res ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
res = moo_geints ( moo , rcv , arg ) ;
if ( ! res ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-10-04 16:21:31 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , res ) ;
return MOO_PF_SUCCESS ;
2015-10-04 16:21:31 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_integer_inttostr ( moo_t * moo , moo_ooi_t nargs )
2015-12-02 15:24:13 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , arg , str ;
moo_ooi_t radix ;
2015-12-02 15:24:13 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 1 ) ;
2015-12-13 16:08:05 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
arg = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
2015-12-02 15:24:13 +00:00
2017-01-09 09:54:49 +00:00
if ( ! MOO_OOP_IS_SMOOI ( arg ) ) return MOO_PF_FAILURE ;
radix = MOO_OOP_TO_SMOOI ( arg ) ;
2015-12-02 15:24:13 +00:00
2017-01-09 09:54:49 +00:00
if ( radix < 2 | | radix > 36 ) return MOO_PF_FAILURE ;
str = moo_inttostr ( moo , rcv , radix ) ;
2017-04-03 13:24:18 +00:00
if ( ! str ) return ( moo - > errnum = = MOO_EINVAL ? MOO_PF_FAILURE : MOO_PF_HARD_FAILURE ) ;
2015-12-02 15:24:13 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , str ) ;
return MOO_PF_SUCCESS ;
2015-12-02 15:24:13 +00:00
}
2017-05-16 02:04:18 +00:00
/* ------------------------------------------------------------------ */
2017-04-03 13:24:18 +00:00
static moo_pfrc_t pf_character_as_smooi ( moo_t * moo , moo_ooi_t nargs )
{
moo_oop_t rcv ;
moo_ooi_t c ;
MOO_ASSERT ( moo , nargs = = 0 ) ;
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_OOP_IS_CHAR ( rcv ) ) ;
2017-04-03 13:24:18 +00:00
c = MOO_OOP_TO_CHAR ( rcv ) ;
MOO_STACK_SETRET ( moo , nargs , MOO_SMOOI_TO_OOP ( c ) ) ;
return MOO_PF_SUCCESS ;
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_smooi_as_character ( moo_t * moo , moo_ooi_t nargs )
2016-12-28 19:12:14 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
moo_ooi_t ec ;
2016-12-28 19:12:14 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 0 ) ;
2016-12-28 19:12:14 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_OOP_IS_SMOOI ( rcv ) ) ;
2016-12-28 19:12:14 +00:00
2017-01-09 09:54:49 +00:00
ec = MOO_OOP_TO_SMOOI ( rcv ) ;
2016-12-28 19:12:14 +00:00
if ( ec < 0 ) ec = 0 ;
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , MOO_CHAR_TO_OOP ( ec ) ) ;
return MOO_PF_SUCCESS ;
2016-12-28 19:12:14 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_smooi_as_error ( moo_t * moo , moo_ooi_t nargs )
2016-12-28 19:12:14 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
moo_ooi_t ec ;
2016-12-28 19:12:14 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 0 ) ;
2016-12-28 19:12:14 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_OOP_IS_SMOOI ( rcv ) ) ;
2016-12-28 19:12:14 +00:00
2017-01-09 09:54:49 +00:00
ec = MOO_OOP_TO_SMOOI ( rcv ) ;
2017-04-02 13:13:33 +00:00
if ( ec < MOO_ERROR_MIN ) ec = MOO_ERROR_MIN ;
else if ( ec > MOO_ERROR_MAX ) ec = MOO_ERROR_MAX ;
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , MOO_ERROR_TO_OOP ( ec ) ) ;
return MOO_PF_SUCCESS ;
2016-12-28 19:12:14 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_error_as_character ( moo_t * moo , moo_ooi_t nargs )
2016-12-28 13:42:12 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
moo_ooi_t ec ;
2016-12-28 13:42:12 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 0 ) ;
2016-12-28 13:42:12 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_OOP_IS_ERROR ( rcv ) ) ;
2016-12-28 13:42:12 +00:00
2017-01-09 09:54:49 +00:00
ec = MOO_OOP_TO_ERROR ( rcv ) ;
2017-04-02 13:13:33 +00:00
MOO_ASSERT ( moo , ec > = MOO_CHAR_MIN & & ec < = MOO_CHAR_MAX ) ;
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , MOO_CHAR_TO_OOP ( ec ) ) ;
return MOO_PF_SUCCESS ;
2016-12-28 13:42:12 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_error_as_integer ( moo_t * moo , moo_ooi_t nargs )
2016-12-26 18:44:47 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv ;
moo_ooi_t ec ;
2016-12-26 18:44:47 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 0 ) ;
2016-12-26 18:44:47 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_OOP_IS_ERROR ( rcv ) ) ;
2016-12-26 18:44:47 +00:00
2017-01-09 09:54:49 +00:00
ec = MOO_OOP_TO_ERROR ( rcv ) ;
MOO_ASSERT ( moo , MOO_IN_SMOOI_RANGE ( ec ) ) ;
MOO_STACK_SETRET ( moo , nargs , MOO_SMOOI_TO_OOP ( ec ) ) ;
return MOO_PF_SUCCESS ;
2016-12-26 18:44:47 +00:00
}
2017-01-09 09:54:49 +00:00
static moo_pfrc_t pf_error_as_string ( moo_t * moo , moo_ooi_t nargs )
2016-12-26 18:44:47 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_t rcv , ss ;
moo_ooi_t ec ;
const moo_ooch_t * s ;
2016-12-26 18:44:47 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , nargs = = 0 ) ;
2016-12-27 18:15:35 +00:00
2017-01-09 09:54:49 +00:00
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_OOP_IS_ERROR ( rcv ) ) ;
2016-12-27 18:15:35 +00:00
2017-01-09 09:54:49 +00:00
ec = MOO_OOP_TO_ERROR ( rcv ) ;
MOO_ASSERT ( moo , MOO_IN_SMOOI_RANGE ( ec ) ) ;
2016-12-27 18:15:35 +00:00
/* TODO: error string will be mostly the same.. do i really have to call makestring every time? */
2017-10-18 16:15:51 +00:00
s = moo_errnum_to_errstr ( ec ) ;
2017-01-09 09:54:49 +00:00
ss = moo_makestring ( moo , s , moo_countoocstr ( s ) ) ;
2017-04-03 13:24:18 +00:00
if ( ! ss )
{
MOO_STACK_SETRETTOERRNUM ( moo , nargs ) ;
return MOO_PF_SUCCESS ;
}
2016-12-27 18:15:35 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_SETRET ( moo , nargs , ss ) ;
return MOO_PF_SUCCESS ;
2016-12-26 18:44:47 +00:00
}
2017-05-16 02:04:18 +00:00
static moo_pfrc_t pf_strlen ( moo_t * moo , moo_ooi_t nargs )
{
moo_oop_t rcv , ret ;
moo_oow_t i , limit ;
moo_ooch_t * ptr ;
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_OBJ_IS_CHAR_POINTER ( rcv ) ) ;
2017-05-16 02:04:18 +00:00
/* [NOTE] the length check loop is directly implemented
* here to be able to handle character objects
* regardless of the existence of the EXTRA flag */
limit = MOO_OBJ_GET_SIZE ( rcv ) ;
ptr = MOO_OBJ_GET_CHAR_SLOT ( rcv ) ;
for ( i = 0 ; i < limit ; i + + )
{
if ( * ptr = = ' \0 ' ) break ;
ptr + + ;
}
ret = moo_oowtoint ( moo , i ) ;
if ( ! ret ) return MOO_PF_FAILURE ;
MOO_STACK_SETRET ( moo , nargs , ret ) ;
return MOO_PF_SUCCESS ;
}
2017-05-21 16:57:21 +00:00
/* ------------------------------------------------------------------ */
static moo_pfrc_t pf_system_log ( moo_t * moo , moo_ooi_t nargs )
{
moo_oop_t msg , level ;
moo_oow_t mask ;
moo_ooi_t k ;
MOO_ASSERT ( moo , nargs > = 2 ) ;
/* TODO: enhance this primitive */
level = MOO_STACK_GETARG ( moo , nargs , 0 ) ;
if ( ! MOO_OOP_IS_SMOOI ( level ) ) mask = MOO_LOG_APP | MOO_LOG_INFO ;
else mask = MOO_LOG_APP | MOO_OOP_TO_SMOOI ( level ) ;
for ( k = 1 ; k < nargs ; k + + )
{
msg = MOO_STACK_GETARG ( moo , nargs , k ) ;
if ( msg = = moo - > _nil | | msg = = moo - > _true | | msg = = moo - > _false )
{
goto dump_object ;
}
else if ( MOO_OOP_IS_POINTER ( msg ) )
{
if ( MOO_OBJ_GET_FLAGS_TYPE ( msg ) = = MOO_OBJ_TYPE_CHAR )
{
log_char_object ( moo , mask , ( moo_oop_char_t ) msg ) ;
}
else if ( MOO_OBJ_GET_FLAGS_TYPE ( msg ) = = MOO_OBJ_TYPE_OOP )
{
/* visit only 1-level down into an array-like object */
moo_oop_t inner ;
moo_oop_class_t _class ;
moo_oow_t i , spec ;
_class = MOO_CLASSOF ( moo , msg ) ;
spec = MOO_OOP_TO_SMOOI ( ( ( moo_oop_class_t ) _class ) - > spec ) ;
if ( MOO_CLASS_SPEC_NAMED_INSTVARS ( spec ) > 0 | | ! MOO_CLASS_SPEC_IS_INDEXED ( spec ) ) goto dump_object ;
for ( i = 0 ; i < MOO_OBJ_GET_SIZE ( msg ) ; i + + )
{
inner = ( ( moo_oop_oop_t ) msg ) - > slot [ i ] ;
if ( i > 0 ) moo_logbfmt ( moo , mask , " " ) ;
if ( MOO_OOP_IS_POINTER ( inner ) & &
MOO_OBJ_GET_FLAGS_TYPE ( inner ) = = MOO_OBJ_TYPE_CHAR )
{
log_char_object ( moo , mask , ( moo_oop_char_t ) inner ) ;
}
else
{
moo_logbfmt ( moo , mask , " %O " , inner ) ;
}
}
}
else goto dump_object ;
}
else
{
dump_object :
moo_logbfmt ( moo , mask , " %O " , msg ) ;
}
}
MOO_STACK_SETRETTORCV ( moo , nargs ) ; /* ^self */
return MOO_PF_SUCCESS ;
}
2017-07-21 16:54:43 +00:00
static moo_pfrc_t pf_system_collect_garbage ( moo_t * moo , moo_ooi_t nargs )
{
moo_gc ( moo ) ;
MOO_STACK_SETRETTORCV ( moo , nargs ) ;
return MOO_PF_SUCCESS ;
}
2017-07-20 16:33:53 +00:00
static moo_pfrc_t pf_system_pop_collectable ( moo_t * moo , moo_ooi_t nargs )
{
if ( moo - > collectable . first )
{
2017-07-21 16:54:43 +00:00
moo_finalizable_t * first ;
2017-07-20 16:33:53 +00:00
first = moo - > collectable . first ;
/* TODO: if it's already fininalized, delete it from collectable */
MOO_ASSERT ( moo , MOO_OOP_IS_POINTER ( first - > oop ) ) ;
MOO_ASSERT ( moo , MOO_OBJ_GET_FLAGS_GCFIN ( first - > oop ) & MOO_GCFIN_FINALIZABLE ) ;
MOO_STACK_SETRET ( moo , nargs , first - > oop ) ;
MOO_OBJ_SET_FLAGS_GCFIN ( first - > oop , MOO_OBJ_GET_FLAGS_GCFIN ( first - > oop ) | MOO_GCFIN_FINALIZED ) ;
MOO_DELETE_FROM_LIST ( & moo - > collectable , first ) ;
2017-07-21 16:54:43 +00:00
moo_freemem ( moo , first ) ; /* TODO: move it to the free list instead... */
2017-07-20 16:33:53 +00:00
}
else
{
MOO_STACK_SETRETTOERROR ( moo , nargs , MOO_ENOENT ) ;
}
return MOO_PF_SUCCESS ;
}
2017-04-03 13:24:18 +00:00
2017-04-02 17:13:04 +00:00
static void sprintptr ( moo_ooch_t * nbuf , moo_oow_t num , moo_oow_t * lenp )
{
static const moo_ooch_t hex2ascii_upper [ ] =
{
' 0 ' , ' 1 ' , ' 2 ' , ' 3 ' , ' 4 ' , ' 5 ' , ' 6 ' , ' 7 ' , ' 8 ' , ' 9 ' ,
' A ' , ' B ' , ' C ' , ' D ' , ' E ' , ' F ' , ' G ' , ' H ' , ' I ' , ' J ' , ' K ' , ' L ' , ' M ' ,
' N ' , ' O ' , ' P ' , ' Q ' , ' R ' , ' S ' , ' T ' , ' U ' , ' V ' , ' W ' , ' X ' , ' H ' , ' Z '
} ;
moo_ooch_t * p , * end , ch ;
p = nbuf ;
* p = ' \0 ' ;
do { * + + p = hex2ascii_upper [ num % 16 ] ; } while ( num / = 16 ) ;
* + + p = ' r ' ;
* + + p = ' 6 ' ;
* + + p = ' 1 ' ;
* lenp = p - nbuf ;
end = p ;
p = nbuf ;
while ( p < = end )
{
ch = * p ;
* p + + = * end ;
* end - - = ch ;
}
}
static moo_pfrc_t pf_smptr_as_string ( moo_t * moo , moo_ooi_t nargs )
{
moo_oop_t rcv ;
void * ptr ;
moo_ooch_t buf [ MOO_SIZEOF_OOW_T * 2 + 4 ] ;
moo_oow_t len ;
moo_oop_t ss ;
MOO_ASSERT ( moo , nargs = = 0 ) ;
rcv = MOO_STACK_GETRCV ( moo , nargs ) ;
2017-09-25 15:16:19 +00:00
MOO_PF_CHECK_RCV ( moo , MOO_OOP_IS_SMPTR ( rcv ) ) ;
2017-04-02 17:13:04 +00:00
ptr = MOO_OOP_TO_SMPTR ( rcv ) ;
sprintptr ( buf , ( moo_oow_t ) ptr , & len ) ;
ss = moo_makestring ( moo , buf , len ) ;
2017-04-03 13:24:18 +00:00
if ( ! ss )
{
MOO_STACK_SETRETTOERRNUM ( moo , nargs ) ;
return MOO_PF_SUCCESS ;
}
2017-04-02 17:13:04 +00:00
MOO_STACK_SETRET ( moo , nargs , ss ) ;
return MOO_PF_SUCCESS ;
}
2017-04-01 04:58:02 +00:00
# define MA MOO_TYPE_MAX(moo_oow_t)
2016-11-29 05:25:08 +00:00
struct pf_t
2015-06-14 07:15:53 +00:00
{
2017-02-14 10:25:26 +00:00
const char * name ; /* the name is supposed to be 7-bit ascii only */
2017-04-01 04:58:02 +00:00
moo_pfbase_t pfbase ;
2015-06-14 07:15:53 +00:00
} ;
2016-11-29 05:25:08 +00:00
typedef struct pf_t pf_t ;
static pf_t pftab [ ] =
{
2017-04-01 04:58:02 +00:00
{ " _dump " , { pf_dump , 0 , MA } } ,
2017-07-25 15:26:04 +00:00
{ " _processor_schedule " , { pf_processor_schedule , 1 , 1 } } ,
2017-04-01 04:58:02 +00:00
{ " _integer_add " , { pf_integer_add , 1 , 1 } } ,
{ " _integer_sub " , { pf_integer_sub , 1 , 1 } } ,
{ " _integer_mul " , { pf_integer_mul , 1 , 1 } } ,
{ " _integer_quo " , { pf_integer_quo , 1 , 1 } } ,
{ " _integer_rem " , { pf_integer_rem , 1 , 1 } } ,
{ " _integer_quo2 " , { pf_integer_quo2 , 1 , 1 } } ,
{ " _integer_rem2 " , { pf_integer_rem2 , 1 , 1 } } ,
{ " _integer_negated " , { pf_integer_negated , 0 , 0 } } ,
{ " _integer_bitat " , { pf_integer_bitat , 1 , 1 } } ,
{ " _integer_bitand " , { pf_integer_bitand , 1 , 1 } } ,
{ " _integer_bitor " , { pf_integer_bitor , 1 , 1 } } ,
{ " _integer_bitxor " , { pf_integer_bitxor , 1 , 1 } } ,
{ " _integer_bitinv " , { pf_integer_bitinv , 0 , 0 } } ,
{ " _integer_bitshift " , { pf_integer_bitshift , 1 , 1 } } ,
{ " _integer_eq " , { pf_integer_eq , 1 , 1 } } ,
{ " _integer_ne " , { pf_integer_ne , 1 , 1 } } ,
{ " _integer_lt " , { pf_integer_lt , 1 , 1 } } ,
{ " _integer_gt " , { pf_integer_gt , 1 , 1 } } ,
{ " _integer_le " , { pf_integer_le , 1 , 1 } } ,
{ " _integer_ge " , { pf_integer_ge , 1 , 1 } } ,
{ " _integer_inttostr " , { pf_integer_inttostr , 1 , 1 } } ,
2017-07-20 16:33:53 +00:00
{ " Apex_addToBeFinalized " , { pf_add_to_be_finalized , 0 , 0 } } ,
2017-12-05 16:51:20 +00:00
{ " Apex_basicAt: " , { moo_pf_basic_at , 1 , 1 } } ,
{ " Apex_basicAt:put: " , { moo_pf_basic_at_put , 2 , 2 } } ,
{ " Apex_basicNew " , { moo_pf_basic_new , 0 , 0 } } ,
{ " Apex_basicNew: " , { moo_pf_basic_new , 1 , 1 } } ,
{ " Apex_basicSize " , { moo_pf_basic_size , 0 , 0 } } ,
{ " Apex_class " , { moo_pf_class , 0 , 0 } } ,
2017-12-03 17:08:04 +00:00
{ " Apex_hash " , { pf_hash , 0 , 0 } } ,
2017-12-08 15:28:51 +00:00
{ " Apex_isKindOf: " , { moo_pf_is_kind_of , 1 , 1 , } } ,
2017-12-03 17:08:04 +00:00
{ " Apex_perform " , { pf_perform , 1 , MA } } ,
2017-07-20 16:33:53 +00:00
{ " Apex_removeToBeFinalized " , { pf_remove_to_be_finalized , 0 , 0 } } ,
2017-12-08 15:28:51 +00:00
{ " Apex_respondsTo: " , { moo_pf_responds_to , 1 , 1 } } ,
2017-12-05 16:51:20 +00:00
{ " Apex_shallowCopy " , { moo_pf_shallow_copy , 0 , 0 } } ,
2017-12-03 17:08:04 +00:00
2017-12-05 16:51:20 +00:00
{ " Apex_== " , { moo_pf_identical , 1 , 1 } } ,
{ " Apex_~~ " , { moo_pf_not_identical , 1 , 1 } } ,
{ " Apex_= " , { moo_pf_equal , 1 , 1 } } ,
{ " Apex_~= " , { moo_pf_not_equal , 1 , 1 } } ,
2017-12-03 17:08:04 +00:00
{ " BlockContext_value " , { pf_block_value , 0 , MA } } ,
{ " BlockContext_newProcess " , { pf_block_new_process , 0 , MA } } ,
2017-05-08 16:00:55 +00:00
2017-04-03 13:24:18 +00:00
{ " Character_asInteger " , { pf_character_as_smooi , 0 , 0 } } ,
2017-04-01 04:58:02 +00:00
{ " Error_asCharacter " , { pf_error_as_character , 0 , 0 } } ,
{ " Error_asInteger " , { pf_error_as_integer , 0 , 0 } } ,
{ " Error_asString " , { pf_error_as_string , 0 , 0 } } ,
2017-12-03 17:08:04 +00:00
{ " MethodContext_goto: " , { pf_context_goto , 1 , 1 } } ,
2017-05-16 02:04:18 +00:00
{ " Process_resume " , { pf_process_resume , 0 , 0 } } ,
2017-09-25 15:16:19 +00:00
{ " Process_sp " , { pf_process_sp , 0 , 0 } } ,
{ " Process_suspend " , { pf_process_suspend , 0 , 0 } } ,
2017-05-16 02:04:18 +00:00
{ " Process_yield " , { pf_process_yield , 0 , 0 } } ,
2017-11-24 17:40:20 +00:00
{ " Process_terminate " , { pf_process_terminate , 0 , 0 } } ,
2017-05-16 02:04:18 +00:00
2017-04-03 05:43:50 +00:00
{ " Semaphore_signal " , { pf_semaphore_signal , 0 , 0 } } ,
2017-11-24 17:40:20 +00:00
{ " Semaphore_wait " , { pf_semaphore_wait , 0 , 0 } } ,
{ " SemaphoreGroup_addSemaphore: " , { pf_semaphore_group_add_semaphore , 1 , 1 } } ,
{ " SemaphoreGroup_removeSemaphore: " , { pf_semaphore_group_remove_semaphore , 1 , 1 } } ,
{ " SemaphoreGroup_wait " , { pf_semaphore_group_wait , 0 , 0 } } ,
2017-04-03 05:43:50 +00:00
2017-04-01 04:58:02 +00:00
{ " SmallInteger_asCharacter " , { pf_smooi_as_character , 0 , 0 } } ,
2017-04-02 17:08:03 +00:00
{ " SmallInteger_asError " , { pf_smooi_as_error , 0 , 0 } } ,
{ " SmallPointer_asString " , { pf_smptr_as_string , 0 , 0 } } ,
2017-12-08 15:28:51 +00:00
{ " SmallPointer_free " , { moo_pf_smptr_free , 0 , 0 } } ,
{ " SmallPointer_getInt16 " , { moo_pf_smptr_get_int16 , 1 , 1 } } ,
{ " SmallPointer_getInt32 " , { moo_pf_smptr_get_int32 , 1 , 1 } } ,
{ " SmallPointer_getInt64 " , { moo_pf_smptr_get_int64 , 1 , 1 } } ,
{ " SmallPointer_getInt8 " , { moo_pf_smptr_get_int8 , 1 , 1 } } ,
{ " SmallPointer_getUint16 " , { moo_pf_smptr_get_uint16 , 1 , 1 } } ,
{ " SmallPointer_getUint32 " , { moo_pf_smptr_get_uint32 , 1 , 1 } } ,
{ " SmallPointer_getUint64 " , { moo_pf_smptr_get_uint64 , 1 , 1 } } ,
{ " SmallPointer_getUint8 " , { moo_pf_smptr_get_uint8 , 1 , 1 } } ,
{ " SmallPointer_putInt8 " , { moo_pf_smptr_put_int8 , 2 , 2 } } ,
{ " SmallPointer_putInt16 " , { moo_pf_smptr_put_int16 , 2 , 2 } } ,
{ " SmallPointer_putInt32 " , { moo_pf_smptr_put_int32 , 2 , 2 } } ,
{ " SmallPointer_putInt64 " , { moo_pf_smptr_put_int64 , 2 , 2 } } ,
{ " SmallPointer_putUint8 " , { moo_pf_smptr_put_uint8 , 2 , 2 } } ,
{ " SmallPointer_putUint16 " , { moo_pf_smptr_put_uint16 , 2 , 2 } } ,
{ " SmallPointer_putUint32 " , { moo_pf_smptr_put_uint32 , 2 , 2 } } ,
{ " SmallPointer_putUint64 " , { moo_pf_smptr_put_uint64 , 2 , 2 } } ,
2017-04-02 17:13:04 +00:00
2017-05-16 02:04:18 +00:00
{ " String_strlen " , { pf_strlen , 0 , 0 } } ,
2017-12-08 15:28:51 +00:00
{ " System_calloc " , { moo_pf_system_calloc , 1 , 1 } } ,
{ " System_calloc: " , { moo_pf_system_calloc , 1 , 1 } } ,
{ " System_free " , { moo_pf_system_free , 1 , 1 } } ,
{ " System_free: " , { moo_pf_system_free , 1 , 1 } } ,
2017-12-09 15:47:43 +00:00
{ " System_getBytes " , { moo_pf_system_get_bytes , 5 , 5 } } ,
2017-12-08 15:28:51 +00:00
{ " System_getInt16 " , { moo_pf_system_get_int16 , 2 , 2 } } ,
{ " System_getInt32 " , { moo_pf_system_get_int32 , 2 , 2 } } ,
{ " System_getInt64 " , { moo_pf_system_get_int64 , 2 , 2 } } ,
{ " System_getInt8 " , { moo_pf_system_get_int8 , 2 , 2 } } ,
{ " System_getUint16 " , { moo_pf_system_get_uint16 , 2 , 2 } } ,
{ " System_getUint32 " , { moo_pf_system_get_uint32 , 2 , 2 } } ,
{ " System_getUint64 " , { moo_pf_system_get_uint64 , 2 , 2 } } ,
{ " System_getUint8 " , { moo_pf_system_get_uint8 , 2 , 2 } } ,
{ " System_malloc " , { moo_pf_system_malloc , 1 , 1 } } ,
{ " System_malloc: " , { moo_pf_system_malloc , 1 , 1 } } ,
2017-12-03 17:08:04 +00:00
{ " System_popCollectable " , { pf_system_pop_collectable , 0 , 0 } } ,
2017-12-09 15:47:43 +00:00
{ " System_putBytes " , { moo_pf_system_put_bytes , 5 , 5 } } ,
2017-12-08 15:28:51 +00:00
{ " System_putInt8 " , { moo_pf_system_put_int8 , 3 , 3 } } ,
{ " System_putInt16 " , { moo_pf_system_put_int16 , 3 , 3 } } ,
{ " System_putInt32 " , { moo_pf_system_put_int32 , 3 , 3 } } ,
{ " System_putInt64 " , { moo_pf_system_put_int64 , 3 , 3 } } ,
{ " System_putUint8 " , { moo_pf_system_put_uint8 , 3 , 3 } } ,
{ " System_putUint16 " , { moo_pf_system_put_uint16 , 3 , 3 } } ,
{ " System_putUint32 " , { moo_pf_system_put_uint32 , 3 , 3 } } ,
{ " System_putUint64 " , { moo_pf_system_put_uint64 , 3 , 3 } } ,
2017-12-03 17:08:04 +00:00
{ " System_signal:afterSecs: " , { pf_system_add_timed_semaphore , 2 , 2 } } ,
{ " System_signal:afterSecs:nanosecs: " , { pf_system_add_timed_semaphore , 3 , 3 } } ,
{ " System_signal:onInput: " , { pf_system_add_input_semaphore , 2 , 2 } } ,
{ " System_signal:onInOutput: " , { pf_system_add_inoutput_semaphore , 2 , 2 } } ,
{ " System_signal:onOutput: " , { pf_system_add_output_semaphore , 2 , 2 } } ,
{ " System_signalOnGCFin: " , { pf_system_add_gcfin_semaphore , 1 , 1 } } ,
{ " System_unsignal: " , { pf_system_remove_semaphore , 1 , 1 } } ,
2017-11-05 16:47:13 +00:00
2017-07-21 16:54:43 +00:00
{ " System_collectGarbage " , { pf_system_collect_garbage , 0 , 0 } } ,
2017-11-21 09:15:22 +00:00
{ " System_log " , { pf_system_log , 2 , MA } } ,
{ " System_return:to: " , { pf_system_return_value_to_context , 2 , 2 } }
2015-06-14 07:15:53 +00:00
} ;
2015-06-11 09:11:18 +00:00
2017-04-01 04:58:02 +00:00
moo_pfbase_t * moo_getpfnum ( moo_t * moo , const moo_ooch_t * ptr , moo_oow_t len , moo_ooi_t * pfnum )
2015-10-03 15:29:03 +00:00
{
2017-04-01 04:58:02 +00:00
moo_ooi_t i ;
2015-10-03 15:29:03 +00:00
2017-02-15 11:57:24 +00:00
/* TODO: have the pftable sorted alphabetically and do binary search */
2017-04-01 04:58:02 +00:00
/* TODO: sort pftab and utilize moo_findpfbase? */
2017-01-09 09:54:49 +00:00
for ( i = 0 ; i < MOO_COUNTOF ( pftab ) ; i + + )
2015-10-03 15:29:03 +00:00
{
2017-03-31 14:21:22 +00:00
/* moo_compoocharsbcstr() is not aware of multibyte encoding.
* so the names above should be composed of the single byte
* characters only */
2017-04-01 04:58:02 +00:00
if ( moo_compoocharsbcstr ( ptr , len , pftab [ i ] . name ) = = 0 )
{
MOO_ASSERT ( moo , MOO_OOI_IN_METHOD_PREAMBLE_INDEX_RANGE ( i ) ) ; /* this must never be so big */
* pfnum = i ;
return & pftab [ i ] . pfbase ;
}
2015-10-03 15:29:03 +00:00
}
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_ENOENT ) ;
2017-04-01 04:58:02 +00:00
return MOO_NULL ;
2015-10-03 15:29:03 +00:00
}
2015-12-27 10:19:36 +00:00
/* ------------------------------------------------------------------------- */
2017-01-09 09:54:49 +00:00
static int start_method ( moo_t * moo , moo_oop_method_t method , moo_oow_t nargs )
2015-12-27 10:19:36 +00:00
{
2017-04-01 06:54:27 +00:00
moo_ooi_t preamble , preamble_code , preamble_flags ;
2017-03-23 16:14:22 +00:00
moo_ooi_t /*sp,*/ stack_base ;
2015-12-27 10:19:36 +00:00
2017-01-09 09:54:49 +00:00
# if defined(MOO_DEBUG_VM_EXEC)
2017-07-20 16:33:53 +00:00
/* set it to a fake value */
moo - > last_instruction_pointer = 0 ;
2016-10-06 14:17:24 +00:00
# endif
2017-01-09 09:54:49 +00:00
preamble = MOO_OOP_TO_SMOOI ( method - > preamble ) ;
2017-04-01 06:54:27 +00:00
preamble_flags = MOO_METHOD_GET_PREAMBLE_FLAGS ( preamble ) ;
2016-12-14 07:18:01 +00:00
2017-04-01 06:54:27 +00:00
if ( preamble_flags & MOO_METHOD_PREAMBLE_FLAG_LIBERAL )
2016-12-13 15:18:19 +00:00
{
2017-04-01 06:54:27 +00:00
/* do nothing - no argument check */
}
else if ( preamble_flags & MOO_METHOD_PREAMBLE_FLAG_VARIADIC )
{
if ( nargs < MOO_OOP_TO_SMOOI ( method - > tmpr_nargs ) ) goto arg_count_mismatch ;
}
else
{
if ( nargs ! = MOO_OOP_TO_SMOOI ( method - > tmpr_nargs ) )
2016-12-14 07:18:01 +00:00
{
2017-01-09 09:54:49 +00:00
/* TODO: better to throw a moo exception so that the caller can catch it??? */
2017-04-01 06:54:27 +00:00
arg_count_mismatch :
2017-01-09 09:54:49 +00:00
MOO_LOG3 ( moo , MOO_LOG_IC | MOO_LOG_FATAL ,
2016-12-14 07:18:01 +00:00
" Fatal error - Argument count mismatch for a non-variadic method [%O] - %zd expected, %zu given \n " ,
2017-01-09 09:54:49 +00:00
method - > name , MOO_OOP_TO_SMOOI ( method - > tmpr_nargs ) , nargs ) ;
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_EINVAL ) ;
2016-12-14 07:18:01 +00:00
return - 1 ;
}
2016-12-13 15:18:19 +00:00
}
2015-12-27 10:19:36 +00:00
2017-01-09 09:54:49 +00:00
preamble_code = MOO_METHOD_GET_PREAMBLE_CODE ( preamble ) ;
2015-12-27 10:19:36 +00:00
switch ( preamble_code )
{
2017-01-09 09:54:49 +00:00
case MOO_METHOD_PREAMBLE_RETURN_RECEIVER :
LOG_INST_0 ( moo , " preamble_return_receiver " ) ;
MOO_STACK_POPS ( moo , nargs ) ; /* pop arguments only*/
2015-12-27 10:19:36 +00:00
break ;
2017-02-14 08:29:30 +00:00
/* [NOTE] this is useless becuase it returns a caller's context
* as the callee ' s context has not been created yet .
case MOO_METHOD_PREAMBLE_RETURN_CONTEXT :
LOG_INST_0 ( moo , " preamble_return_context " ) ;
MOO_STACK_POPS ( moo , nargs ) ;
MOO_STACK_SETTOP ( moo , ( moo_oop_t ) moo - > active_context ) ;
break ;
*/
case MOO_METHOD_PREAMBLE_RETURN_PROCESS :
LOG_INST_0 ( moo , " preamble_return_process " ) ;
MOO_STACK_POPS ( moo , nargs ) ;
MOO_STACK_SETTOP ( moo , ( moo_oop_t ) moo - > processor - > active ) ;
break ;
2017-06-01 15:42:05 +00:00
case MOO_METHOD_PREAMBLE_RETURN_RECEIVER_NS :
{
moo_oop_t c ;
LOG_INST_0 ( moo , " preamble_return_receiver_ns " ) ;
MOO_STACK_POPS ( moo , nargs ) ; /* pop arguments only*/
c = MOO_STACK_GETTOP ( moo ) ; /* get receiver */
c = ( moo_oop_t ) MOO_CLASSOF ( moo , c ) ;
if ( c = = ( moo_oop_t ) moo - > _class ) c = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_SETTOP ( moo , ( moo_oop_t ) ( ( moo_oop_class_t ) c ) - > nsup ) ;
break ;
}
2017-01-09 09:54:49 +00:00
case MOO_METHOD_PREAMBLE_RETURN_NIL :
LOG_INST_0 ( moo , " preamble_return_nil " ) ;
MOO_STACK_POPS ( moo , nargs ) ;
MOO_STACK_SETTOP ( moo , moo - > _nil ) ;
2015-12-27 10:19:36 +00:00
break ;
2017-01-09 09:54:49 +00:00
case MOO_METHOD_PREAMBLE_RETURN_TRUE :
LOG_INST_0 ( moo , " preamble_return_true " ) ;
MOO_STACK_POPS ( moo , nargs ) ;
MOO_STACK_SETTOP ( moo , moo - > _true ) ;
2015-12-27 10:19:36 +00:00
break ;
2017-01-09 09:54:49 +00:00
case MOO_METHOD_PREAMBLE_RETURN_FALSE :
LOG_INST_0 ( moo , " preamble_return_false " ) ;
MOO_STACK_POPS ( moo , nargs ) ;
MOO_STACK_SETTOP ( moo , moo - > _false ) ;
2015-12-27 10:19:36 +00:00
break ;
2017-01-09 09:54:49 +00:00
case MOO_METHOD_PREAMBLE_RETURN_INDEX :
2016-06-05 18:01:35 +00:00
/* preamble_index field is used to store a positive integer */
2017-01-09 09:54:49 +00:00
LOG_INST_1 ( moo , " preamble_return_index %zd " , MOO_METHOD_GET_PREAMBLE_INDEX ( preamble ) ) ;
MOO_STACK_POPS ( moo , nargs ) ;
MOO_STACK_SETTOP ( moo , MOO_SMOOI_TO_OOP ( MOO_METHOD_GET_PREAMBLE_INDEX ( preamble ) ) ) ;
2015-12-27 10:19:36 +00:00
break ;
2017-01-09 09:54:49 +00:00
case MOO_METHOD_PREAMBLE_RETURN_NEGINDEX :
2016-06-05 18:01:35 +00:00
/* preamble_index field is used to store a negative integer */
2017-01-09 09:54:49 +00:00
LOG_INST_1 ( moo , " preamble_return_negindex %zd " , MOO_METHOD_GET_PREAMBLE_INDEX ( preamble ) ) ;
MOO_STACK_POPS ( moo , nargs ) ;
MOO_STACK_SETTOP ( moo , MOO_SMOOI_TO_OOP ( - MOO_METHOD_GET_PREAMBLE_INDEX ( preamble ) ) ) ;
2015-12-27 10:19:36 +00:00
break ;
2017-01-09 09:54:49 +00:00
case MOO_METHOD_PREAMBLE_RETURN_INSTVAR :
2015-12-27 10:19:36 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oop_oop_t rcv ;
2015-12-27 10:19:36 +00:00
2017-01-09 09:54:49 +00:00
MOO_STACK_POPS ( moo , nargs ) ; /* pop arguments only */
2015-12-27 10:19:36 +00:00
2017-01-09 09:54:49 +00:00
LOG_INST_1 ( moo , " preamble_return_instvar %zd " , MOO_METHOD_GET_PREAMBLE_INDEX ( preamble ) ) ;
2015-12-27 10:19:36 +00:00
/* replace the receiver by an instance variable of the receiver */
2017-01-09 09:54:49 +00:00
rcv = ( moo_oop_oop_t ) MOO_STACK_GETTOP ( moo ) ;
MOO_ASSERT ( moo , MOO_OBJ_GET_FLAGS_TYPE ( rcv ) = = MOO_OBJ_TYPE_OOP ) ;
MOO_ASSERT ( moo , MOO_OBJ_GET_SIZE ( rcv ) > MOO_METHOD_GET_PREAMBLE_INDEX ( preamble ) ) ;
2015-12-27 10:19:36 +00:00
2017-01-09 09:54:49 +00:00
if ( rcv = = ( moo_oop_oop_t ) moo - > active_context )
2015-12-27 10:19:36 +00:00
{
/* the active context object doesn't keep
* the most up - to - date information in the
* ' ip ' and ' sp ' field . commit these fields
* when the object to be accessed is
* the active context . this manual commit
* is required because this premable handling
* skips activation of a new method context
* that would commit these fields .
*/
2017-01-09 09:54:49 +00:00
STORE_ACTIVE_IP ( moo ) ;
STORE_ACTIVE_SP ( moo ) ;
2015-12-27 10:19:36 +00:00
}
/* this accesses the instance variable of the receiver */
2017-01-09 09:54:49 +00:00
MOO_STACK_SET ( moo , moo - > sp , rcv - > slot [ MOO_METHOD_GET_PREAMBLE_INDEX ( preamble ) ] ) ;
2015-12-27 10:19:36 +00:00
break ;
}
2017-01-09 09:54:49 +00:00
case MOO_METHOD_PREAMBLE_PRIMITIVE :
2015-12-27 10:19:36 +00:00
{
2017-01-09 09:54:49 +00:00
moo_ooi_t pfnum ;
2015-12-27 10:19:36 +00:00
2017-03-23 16:14:22 +00:00
stack_base = moo - > sp - nargs - 1 ; /* stack base before receiver and arguments */
2017-01-09 09:54:49 +00:00
pfnum = MOO_METHOD_GET_PREAMBLE_INDEX ( preamble ) ;
LOG_INST_1 ( moo , " preamble_primitive %zd " , pf_no ) ;
2015-12-27 10:19:36 +00:00
2017-03-23 16:14:22 +00:00
if ( pfnum > = 0 & & pfnum < MOO_COUNTOF ( pftab ) )
2015-12-27 10:19:36 +00:00
{
int n ;
2017-04-01 04:58:02 +00:00
if ( ( nargs < pftab [ pfnum ] . pfbase . minargs | | nargs > pftab [ pfnum ] . pfbase . maxargs ) )
2017-03-23 16:14:22 +00:00
{
MOO_DEBUG4 ( moo , " Soft failure due to argument count mismatch for primitive function %hs - %zu-%zu expected, %zu given \n " ,
2017-04-01 04:58:02 +00:00
pftab [ pfnum ] . name , pftab [ pfnum ] . pfbase . minargs , pftab [ pfnum ] . pfbase . maxargs , nargs ) ;
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_ENUMARGS ) ;
2017-03-23 16:14:22 +00:00
goto activate_primitive_method_body ;
}
2017-01-09 09:54:49 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & method ) ;
2017-04-01 04:58:02 +00:00
n = pftab [ pfnum ] . pfbase . handler ( moo , nargs ) ;
2017-01-09 09:54:49 +00:00
moo_poptmp ( moo ) ;
2017-05-07 16:45:27 +00:00
if ( n < = MOO_PF_HARD_FAILURE )
{
MOO_DEBUG3 ( moo , " Hard failure indicated by primitive function %p - %hs - return code %d \n " , pftab [ pfnum ] . pfbase . handler , pftab [ pfnum ] . name , n ) ;
return - 1 ;
}
2017-01-09 09:54:49 +00:00
if ( n > = MOO_PF_SUCCESS ) break ;
2017-03-23 16:14:22 +00:00
2017-04-01 04:58:02 +00:00
MOO_DEBUG2 ( moo , " Soft failure indicated by primitive function %p - %hs \n " , pftab [ pfnum ] . pfbase . handler , pftab [ pfnum ] . name ) ;
2017-03-23 16:14:22 +00:00
}
else
{
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_ENOENT ) ;
2017-05-08 16:00:55 +00:00
MOO_DEBUG1 ( moo , " Cannot call primitive function numbered %zd - unknown primitive function number \n " , pfnum ) ;
2015-12-27 10:19:36 +00:00
}
2017-03-23 16:14:22 +00:00
goto activate_primitive_method_body ;
2015-12-27 10:19:36 +00:00
}
2017-01-09 09:54:49 +00:00
case MOO_METHOD_PREAMBLE_NAMED_PRIMITIVE :
2015-12-27 10:19:36 +00:00
{
2017-01-09 09:54:49 +00:00
moo_ooi_t pf_name_index ;
2017-03-30 14:59:55 +00:00
moo_pfbase_t * pfbase ;
2017-05-07 16:45:27 +00:00
moo_oop_t pfname ;
2017-01-09 09:54:49 +00:00
moo_oow_t w ;
2016-11-12 03:02:12 +00:00
2017-03-23 16:14:22 +00:00
stack_base = moo - > sp - nargs - 1 ; /* stack base before receiver and arguments */
2015-12-27 10:19:36 +00:00
2017-05-07 16:45:27 +00:00
/* index to the primitive function identifier in the literal frame */
2017-01-09 09:54:49 +00:00
pf_name_index = MOO_METHOD_GET_PREAMBLE_INDEX ( preamble ) ;
2017-02-13 13:25:42 +00:00
MOO_ASSERT ( moo , pf_name_index > = 0 ) ;
2017-05-07 16:45:27 +00:00
pfname = method - > slot [ pf_name_index ] ;
2015-12-27 10:19:36 +00:00
2017-05-07 16:45:27 +00:00
# if !defined(NDEBUG)
LOG_INST_1 ( moo , " preamble_named_primitive %zd " , pf_name_index ) ;
2017-05-08 16:00:55 +00:00
# endif
2017-05-07 16:45:27 +00:00
MOO_ASSERT ( moo , MOO_OBJ_IS_CHAR_POINTER ( pfname ) ) ;
MOO_ASSERT ( moo , MOO_OBJ_GET_FLAGS_EXTRA ( pfname ) ) ;
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , pfname ) = = moo - > _symbol ) ;
2017-02-13 13:25:42 +00:00
/* merge two SmallIntegers to get a full pointer from the cached data */
2017-01-09 09:54:49 +00:00
w = ( moo_oow_t ) MOO_OOP_TO_SMOOI ( method - > preamble_data [ 0 ] ) < < ( MOO_OOW_BITS / 2 ) |
( moo_oow_t ) MOO_OOP_TO_SMOOI ( method - > preamble_data [ 1 ] ) ;
2017-03-30 14:59:55 +00:00
pfbase = ( moo_pfbase_t * ) w ;
2017-05-07 16:45:27 +00:00
if ( pfbase ) goto exec_handler ; /* skip moo_querymod() */
2016-11-12 03:02:12 +00:00
2017-05-07 16:45:27 +00:00
pfbase = moo_querymod ( moo , MOO_OBJ_GET_CHAR_SLOT ( pfname ) , MOO_OBJ_GET_SIZE ( pfname ) ) ;
2017-03-30 14:59:55 +00:00
if ( pfbase )
2015-12-27 10:19:36 +00:00
{
int n ;
2017-05-07 16:45:27 +00:00
/* split a pointer to two OOP fields as SmallIntegers for storing/caching. */
method - > preamble_data [ 0 ] = MOO_SMOOI_TO_OOP ( ( moo_oow_t ) pfbase > > ( MOO_OOW_BITS / 2 ) ) ;
method - > preamble_data [ 1 ] = MOO_SMOOI_TO_OOP ( ( moo_oow_t ) pfbase & MOO_LBMASK ( moo_oow_t , MOO_OOW_BITS / 2 ) ) ;
exec_handler :
2017-03-31 14:21:22 +00:00
if ( nargs < pfbase - > minargs | | nargs > pfbase - > maxargs )
{
MOO_DEBUG5 ( moo , " Soft failure due to argument count mismatch for primitive function %.*js - %zu-%zu expected, %zu given \n " ,
2017-05-07 16:45:27 +00:00
MOO_OBJ_GET_SIZE ( pfname ) , MOO_OBJ_GET_CHAR_SLOT ( pfname ) , pfbase - > minargs , pfbase - > maxargs , nargs ) ;
2017-05-08 16:00:55 +00:00
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_ENUMARGS ) ;
2017-03-31 14:21:22 +00:00
goto activate_primitive_method_body ;
}
2017-01-09 09:54:49 +00:00
moo_pushtmp ( moo , ( moo_oop_t * ) & method ) ;
2016-12-14 07:18:01 +00:00
/* the primitive handler is executed without activating the method itself.
* one major difference between the primitive function and the normal method
* invocation is that the primitive function handler should access arguments
* directly in the stack unlik a normal activated method context where the
* arguments are copied to the back . */
2016-12-14 09:40:41 +00:00
2017-05-16 02:04:18 +00:00
moo_seterrnum ( moo , MOO_ENOERR ) ;
2017-03-30 14:59:55 +00:00
n = pfbase - > handler ( moo , nargs ) ;
2016-12-13 15:18:19 +00:00
2017-01-09 09:54:49 +00:00
moo_poptmp ( moo ) ;
if ( n < = MOO_PF_HARD_FAILURE )
2016-11-12 03:02:12 +00:00
{
2017-05-07 16:45:27 +00:00
MOO_DEBUG4 ( moo , " Hard failure indicated by primitive function %p - %.*js - return code %d \n " , pfbase - > handler , MOO_OBJ_GET_SIZE ( pfname ) , MOO_OBJ_GET_CHAR_SLOT ( pfname ) , n ) ;
2016-11-12 03:02:12 +00:00
return - 1 ; /* hard primitive failure */
}
2017-01-09 09:54:49 +00:00
if ( n > = MOO_PF_SUCCESS ) break ; /* primitive ok*/
2015-12-27 10:19:36 +00:00
2016-12-03 18:08:31 +00:00
/* soft primitive failure */
2017-05-07 16:45:27 +00:00
MOO_DEBUG3 ( moo , " Soft failure indicated by primitive function %p - %.*js \n " , pfbase - > handler , MOO_OBJ_GET_SIZE ( pfname ) , MOO_OBJ_GET_CHAR_SLOT ( pfname ) ) ;
2016-12-03 18:08:31 +00:00
}
else
{
/* no handler found */
2017-05-07 16:45:27 +00:00
MOO_DEBUG2 ( moo , " Soft failure for non-existent primitive function - %.*js \n " , MOO_OBJ_GET_SIZE ( pfname ) , MOO_OBJ_GET_CHAR_SLOT ( pfname ) ) ;
2016-12-03 18:08:31 +00:00
}
2016-11-12 03:02:12 +00:00
2017-03-23 16:14:22 +00:00
activate_primitive_method_body :
2017-05-07 16:45:27 +00:00
if ( MOO_METHOD_GET_PREAMBLE_FLAGS ( preamble ) & MOO_METHOD_PREAMBLE_FLAG_LENIENT )
{
2017-05-08 16:00:55 +00:00
/* convert soft failure to error return */
2017-05-07 16:45:27 +00:00
moo - > sp = stack_base ;
MOO_STACK_PUSH ( moo , MOO_ERROR_TO_OOP ( moo - > errnum ) ) ;
break ;
}
2017-05-08 16:00:55 +00:00
/* set the error number in the current process for 'thisProcess primError' */
2017-05-07 16:45:27 +00:00
moo - > processor - > active - > perr = MOO_ERROR_TO_OOP ( moo - > errnum ) ;
2017-05-16 02:04:18 +00:00
if ( moo - > errmsg . len > 0 )
{
/* compose an error message string. */
/* TODO: i don't like to do this here.
* is it really a good idea to compose a string here which
* is not really failure safe without losing integrity ? ? ? ? */
moo_oop_t tmp ;
moo_pushtmp ( moo , ( moo_oop_t * ) & method ) ;
tmp = moo_makestring ( moo , moo - > errmsg . buf , moo - > errmsg . len ) ;
moo_poptmp ( moo ) ;
/* [NOTE] carry on even if instantiation fails */
2017-10-18 16:15:51 +00:00
moo - > processor - > active - > perrmsg = tmp ? tmp : moo - > _nil ; /* TODO: set to nil or set to an empty string if instantiation fails? */
2017-05-16 02:04:18 +00:00
}
else
{
moo - > processor - > active - > perrmsg = moo - > _nil ;
}
2017-05-07 16:45:27 +00:00
2017-02-14 08:29:30 +00:00
# if defined(MOO_USE_METHOD_TRAILER)
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_OBJ_GET_FLAGS_TRAILER ( method ) ) ;
2017-05-08 16:00:55 +00:00
if ( MOO_METHOD_GET_CODE_SIZE ( method ) = = 0 ) /* this trailer size field is not a small integer */
2016-11-12 03:02:12 +00:00
# else
2017-01-09 09:54:49 +00:00
if ( method - > code = = moo - > _nil )
2016-11-12 03:02:12 +00:00
# endif
{
2017-11-24 17:40:20 +00:00
/* no byte code to execute - invoke 'self primitiveFailed' */
2017-05-07 16:45:27 +00:00
2017-05-08 16:00:55 +00:00
static moo_ooch_t prim_fail_msg [ ] = {
' p ' , ' r ' , ' i ' , ' m ' , ' i ' , ' t ' , ' i ' , ' v ' , ' e ' ,
' F ' , ' a ' , ' i ' , ' l ' , ' e ' , ' d '
} ;
moo_oow_t i ;
2017-05-09 15:48:44 +00:00
if ( stack_base ! = moo - > sp - nargs - 1 )
{
/* a primitive function handler must not touch the stack when it returns soft failure */
2017-05-12 16:38:16 +00:00
MOO_DEBUG3 ( moo , " Stack seems to get corrupted by a primitive handler function - %O>>%.*js \n " , method - > owner , MOO_OBJ_GET_SIZE ( method - > name ) , MOO_OBJ_GET_CHAR_SLOT ( method - > name ) ) ;
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_EINTERN ) ;
2017-05-09 15:48:44 +00:00
return - 1 ;
}
2017-05-08 16:00:55 +00:00
2017-08-22 13:45:37 +00:00
MOO_DEBUG3 ( moo , " Sending primitiveFailed - %O>>%.*js \n " , method - > owner , MOO_OBJ_GET_SIZE ( method - > name ) , MOO_OBJ_GET_CHAR_SLOT ( method - > name ) ) ;
2017-05-09 15:48:44 +00:00
/*
* | arg1 | < - - - - stack_base + 3
* | arg0 | < - - - - stack_base + 2
* | receiver | < - - - - stack_base + 1
* | | < - - - - stack_base
*/
2017-05-08 16:00:55 +00:00
2017-05-09 15:48:44 +00:00
/* push out arguments by one slot */
2017-05-08 16:00:55 +00:00
MOO_STACK_PUSH ( moo , moo - > _nil ) ; /* fake */
for ( i = moo - > sp ; i > stack_base + 2 ; i - - ) MOO_STACK_SET ( moo , i , MOO_STACK_GET ( moo , i - 1 ) ) ;
2017-05-09 15:48:44 +00:00
/* inject the method as the first argument */
MOO_STACK_SET ( moo , stack_base + 2 , ( moo_oop_t ) method ) ;
/* send primitiveFailed to self */
2017-05-11 14:59:20 +00:00
if ( send_message_with_str ( moo , prim_fail_msg , 15 , 0 , nargs + 1 ) < = - 1 ) return - 1 ;
2016-11-12 03:02:12 +00:00
}
2017-05-12 16:38:16 +00:00
else
{
/* arrange to execute the method body */
if ( activate_new_method ( moo , method , nargs ) < = - 1 ) return - 1 ;
}
2015-12-27 10:19:36 +00:00
break ;
}
default :
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , preamble_code = = MOO_METHOD_PREAMBLE_NONE | |
2017-02-14 08:29:30 +00:00
preamble_code = = MOO_METHOD_PREAMBLE_RETURN_CONTEXT | |
preamble_code = = MOO_METHOD_PREAMBLE_EXCEPTION | |
preamble_code = = MOO_METHOD_PREAMBLE_ENSURE ) ;
2017-01-09 09:54:49 +00:00
if ( activate_new_method ( moo , method , nargs ) < = - 1 ) return - 1 ;
2015-12-27 10:19:36 +00:00
break ;
}
return 0 ;
}
2017-01-09 09:54:49 +00:00
static int send_message ( moo_t * moo , moo_oop_char_t selector , int to_super , moo_ooi_t nargs )
2016-06-20 15:42:51 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oocs_t mthname ;
moo_oop_t receiver ;
moo_oop_method_t method ;
2016-06-20 15:42:51 +00:00
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , MOO_OOP_IS_POINTER ( selector ) ) ;
MOO_ASSERT ( moo , MOO_OBJ_GET_FLAGS_TYPE ( selector ) = = MOO_OBJ_TYPE_CHAR ) ;
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , selector ) = = moo - > _symbol ) ;
2016-06-20 15:42:51 +00:00
2017-01-09 09:54:49 +00:00
receiver = MOO_STACK_GET ( moo , moo - > sp - nargs ) ;
2016-06-20 15:42:51 +00:00
2017-03-19 14:18:37 +00:00
mthname . ptr = MOO_OBJ_GET_CHAR_SLOT ( selector ) ;
2017-01-09 09:54:49 +00:00
mthname . len = MOO_OBJ_GET_SIZE ( selector ) ;
2017-12-08 15:28:51 +00:00
method = moo_findmethod ( moo , receiver , & mthname , to_super ) ;
2016-06-20 15:42:51 +00:00
if ( ! method )
{
2017-01-09 09:54:49 +00:00
static moo_ooch_t fbm [ ] = {
2016-06-20 15:42:51 +00:00
' d ' , ' o ' , ' e ' , ' s ' ,
' N ' , ' o ' , ' t ' ,
' U ' , ' n ' , ' d ' , ' e ' , ' r ' , ' s ' , ' t ' , ' a ' , ' n ' , ' d ' , ' : '
} ;
mthname . ptr = fbm ;
mthname . len = 18 ;
2017-12-08 15:28:51 +00:00
method = moo_findmethod ( moo , receiver , & mthname , 0 ) ;
2016-06-20 15:42:51 +00:00
if ( ! method )
{
/* this must not happen as long as doesNotUnderstand: is implemented under Apex.
* this check should indicate a very serious internal problem */
2017-01-09 09:54:49 +00:00
MOO_LOG4 ( moo , MOO_LOG_IC | MOO_LOG_FATAL ,
2017-01-06 10:42:59 +00:00
" Fatal error - receiver [%O] of class [%O] does not understand a message [%.*js] \n " ,
2017-01-09 09:54:49 +00:00
receiver , MOO_CLASSOF ( moo , receiver ) , mthname . len , mthname . ptr ) ;
2016-06-20 15:42:51 +00:00
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_EMSGSND ) ;
2016-06-20 15:42:51 +00:00
return - 1 ;
}
else
{
2016-08-13 06:46:14 +00:00
/* manipulate the stack as if 'receier doesNotUnderstand: selector'
2016-06-20 15:42:51 +00:00
* has been called . */
2016-08-13 06:46:14 +00:00
/* TODO: if i manipulate the stack this way here, the stack trace for the last call is kind of lost.
2016-06-20 15:42:51 +00:00
* how can i preserve it gracefully ? */
2017-01-09 09:54:49 +00:00
MOO_STACK_POPS ( moo , nargs ) ;
2016-06-20 15:42:51 +00:00
nargs = 1 ;
2017-01-09 09:54:49 +00:00
MOO_STACK_PUSH ( moo , ( moo_oop_t ) selector ) ;
2016-06-20 15:42:51 +00:00
}
}
2017-01-09 09:54:49 +00:00
return start_method ( moo , method , nargs ) ;
2016-06-20 15:42:51 +00:00
}
2017-05-11 14:59:20 +00:00
static int send_message_with_str ( moo_t * moo , const moo_ooch_t * nameptr , moo_oow_t namelen , int to_super , moo_ooi_t nargs )
2016-06-20 15:42:51 +00:00
{
2017-01-09 09:54:49 +00:00
moo_oocs_t mthname ;
moo_oop_t receiver ;
moo_oop_method_t method ;
2016-06-20 15:42:51 +00:00
2017-01-09 09:54:49 +00:00
receiver = MOO_STACK_GET ( moo , moo - > sp - nargs ) ;
2016-06-20 15:42:51 +00:00
2017-01-09 09:54:49 +00:00
mthname . ptr = ( moo_ooch_t * ) nameptr ;
2016-06-20 15:42:51 +00:00
mthname . len = namelen ;
2017-12-08 15:28:51 +00:00
method = moo_findmethod ( moo , receiver , & mthname , to_super ) ;
2016-06-20 15:42:51 +00:00
if ( ! method )
{
2017-01-09 09:54:49 +00:00
MOO_LOG4 ( moo , MOO_LOG_IC | MOO_LOG_FATAL ,
2017-01-06 10:42:59 +00:00
" Fatal error - receiver [%O] of class [%O] does not understand a private message [%.*js] \n " ,
2017-01-09 09:54:49 +00:00
receiver , MOO_CLASSOF ( moo , receiver ) , mthname . len , mthname . ptr ) ;
2017-05-11 14:59:20 +00:00
moo_seterrnum ( moo , MOO_EMSGSND ) ;
2016-06-20 15:42:51 +00:00
return - 1 ;
}
2017-01-09 09:54:49 +00:00
return start_method ( moo , method , nargs ) ;
2016-06-20 15:42:51 +00:00
}
2017-05-08 16:00:55 +00:00
2015-10-03 15:29:03 +00:00
/* ------------------------------------------------------------------------- */
2017-02-11 05:48:30 +00:00
static MOO_INLINE int switch_process_if_needed ( moo_t * moo )
{
if ( moo - > sem_heap_count > 0 )
{
2017-02-12 18:59:03 +00:00
/* handle timed semaphores */
2017-02-11 05:48:30 +00:00
moo_ntime_t ft , now ;
2017-02-12 18:59:03 +00:00
2017-02-11 05:48:30 +00:00
vm_gettime ( moo , & now ) ;
do
{
MOO_ASSERT ( moo , MOO_OOP_IS_SMOOI ( moo - > sem_heap [ 0 ] - > heap_ftime_sec ) ) ;
MOO_ASSERT ( moo , MOO_OOP_IS_SMOOI ( moo - > sem_heap [ 0 ] - > heap_ftime_nsec ) ) ;
MOO_INITNTIME ( & ft ,
MOO_OOP_TO_SMOOI ( moo - > sem_heap [ 0 ] - > heap_ftime_sec ) ,
MOO_OOP_TO_SMOOI ( moo - > sem_heap [ 0 ] - > heap_ftime_nsec )
) ;
if ( MOO_CMPNTIME ( & ft , ( moo_ntime_t * ) & now ) < = 0 )
{
moo_oop_process_t proc ;
/* waited long enough. signal the semaphore */
proc = signal_semaphore ( moo , moo - > sem_heap [ 0 ] ) ;
/* [NOTE] no moo_pushtmp() on proc. no GC must occur
* in the following line until it ' s used for
2017-07-25 15:26:04 +00:00
* wake_process ( ) below . */
2017-03-25 05:16:18 +00:00
delete_from_sem_heap ( moo , 0 ) ; /* moo->sem_heap_count is decremented */
2017-02-11 05:48:30 +00:00
/* if no process is waiting on the semaphore,
* signal_semaphore ( ) returns moo - > _nil . */
if ( moo - > processor - > active = = moo - > nil_process & & ( moo_oop_t ) proc ! = moo - > _nil )
{
/* this is the only runnable process.
* switch the process to the running state .
2017-07-25 15:26:04 +00:00
* it uses wake_process ( ) instead of
2017-02-11 05:48:30 +00:00
* switch_to_process ( ) as there is no running
* process at this moment */
2017-08-17 18:10:29 +00:00
2017-08-20 14:43:34 +00:00
# if defined(MOO_DEBUG_VM_PROCESSOR) && (MOO_DEBUG_VM_PROCESSOR >= 2)
2017-08-17 18:10:29 +00:00
MOO_LOG2 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Processor - switching to a process [%zd] while no process is active - total runnables %zd \n " , MOO_OOP_TO_SMOOI ( proc - > id ) , MOO_OOP_TO_SMOOI ( moo - > processor - > runnable . count ) ) ;
# endif
2017-02-11 05:48:30 +00:00
MOO_ASSERT ( moo , proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNABLE ) ) ;
2017-08-17 18:10:29 +00:00
MOO_ASSERT ( moo , proc = = moo - > processor - > runnable . last ) ; /* resume_process() appends to the runnable list */
2017-07-25 15:26:04 +00:00
#if 0
wake_process ( moo , proc ) ; /* switch to running */
2017-02-11 05:48:30 +00:00
moo - > proc_switched = 1 ;
2017-07-25 15:26:04 +00:00
# else
switch_to_process_from_nil ( moo , proc ) ;
# endif
2017-02-11 05:48:30 +00:00
}
}
else if ( moo - > processor - > active = = moo - > nil_process )
{
2017-02-12 18:59:03 +00:00
/* no running process */
2017-02-11 05:48:30 +00:00
MOO_SUBNTIME ( & ft , & ft , ( moo_ntime_t * ) & now ) ;
2017-02-12 18:59:03 +00:00
2017-02-18 13:31:47 +00:00
if ( moo - > sem_io_wait_count > 0 )
2017-02-15 11:57:24 +00:00
{
2017-03-25 05:16:18 +00:00
/* no running process but io semaphore being waited on */
vm_muxwait ( moo , & ft ) ;
2017-03-26 17:15:25 +00:00
/* exit early if a process has been woken up.
* the break in the else part further down will get hit
* eventually even if the following line doesn ' t exist .
2017-07-25 15:26:04 +00:00
* having the following line causes to skip firing the
2017-03-26 17:15:25 +00:00
* timed semaphore that would expire between now and the
* moment the next inspection occurs . */
2017-07-27 08:32:16 +00:00
if ( moo - > processor - > active ! = moo - > nil_process ) goto switch_to_next ;
2017-02-15 11:57:24 +00:00
}
2017-02-12 18:59:03 +00:00
else
2017-02-15 11:57:24 +00:00
{
2017-03-25 05:16:18 +00:00
/* no running process, no io semaphore */
2017-07-30 15:15:47 +00:00
if ( ( moo_oop_t ) moo - > sem_gcfin ! = moo - > _nil & & moo - > sem_gcfin_sigreq ) goto signal_sem_gcfin ;
2017-02-15 11:57:24 +00:00
vm_sleep ( moo , & ft ) ;
}
2017-02-11 05:48:30 +00:00
vm_gettime ( moo , & now ) ;
}
else
{
2017-02-12 18:59:03 +00:00
/* there is a running process. go on */
2017-02-11 05:48:30 +00:00
break ;
}
2017-02-15 11:57:24 +00:00
}
2017-02-12 18:59:03 +00:00
while ( moo - > sem_heap_count > 0 & & ! moo - > abort_req ) ;
2017-02-11 05:48:30 +00:00
}
2017-02-18 13:31:47 +00:00
if ( moo - > sem_io_wait_count > 0 )
{
2017-03-25 05:16:18 +00:00
moo_ntime_t now ;
if ( moo - > processor - > active = = moo - > nil_process )
{
/* no runnable process while there is an io semaphore being waited */
2017-07-30 15:15:47 +00:00
if ( ( moo_oop_t ) moo - > sem_gcfin ! = moo - > _nil & & moo - > sem_gcfin_sigreq ) goto signal_sem_gcfin ;
2017-03-25 05:16:18 +00:00
do
{
vm_gettime ( moo , & now ) ;
2017-09-26 09:01:42 +00:00
now . sec + = 3 ; /* TODO: use a configured value? */
2017-03-25 05:16:18 +00:00
vm_muxwait ( moo , & now ) ;
}
while ( moo - > processor - > active = = moo - > nil_process & & ! moo - > abort_req ) ;
}
else
{
vm_muxwait ( moo , MOO_NULL ) ;
}
2017-02-18 13:31:47 +00:00
}
2017-02-12 18:59:03 +00:00
2017-07-27 08:32:16 +00:00
if ( ( moo_oop_t ) moo - > sem_gcfin ! = moo - > _nil )
2017-07-25 15:26:04 +00:00
{
2017-07-27 08:32:16 +00:00
moo_oop_process_t proc ;
if ( moo - > sem_gcfin_sigreq )
2017-07-25 15:26:04 +00:00
{
2017-07-27 08:32:16 +00:00
signal_sem_gcfin :
2017-11-03 16:26:55 +00:00
MOO_LOG0 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " Signaled GCFIN semaphore \n " ) ;
2017-07-25 15:26:04 +00:00
proc = signal_semaphore ( moo , moo - > sem_gcfin ) ;
if ( moo - > processor - > active = = moo - > nil_process & & ( moo_oop_t ) proc ! = moo - > _nil )
{
MOO_ASSERT ( moo , proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNABLE ) ) ;
MOO_ASSERT ( moo , proc = = moo - > processor - > runnable . first ) ;
switch_to_process_from_nil ( moo , proc ) ;
}
2017-07-27 08:32:16 +00:00
moo - > sem_gcfin_sigreq = 0 ;
2017-07-25 15:26:04 +00:00
}
2017-07-27 08:32:16 +00:00
else
2017-07-25 15:26:04 +00:00
{
2017-07-27 08:32:16 +00:00
/* the gcfin semaphore signalling is not requested and there are
* no runnable processes nor no waiting semaphores . if there is
* process waiting on the gcfin semaphore , i will just schedule
2017-08-20 14:43:34 +00:00
* it to run by calling signal_semaphore ( ) on moo - > sem_gcfin .
*/
2017-07-27 08:32:16 +00:00
/* TODO: check if this is the best implementation practice */
if ( moo - > processor - > active = = moo - > nil_process )
2017-07-25 15:26:04 +00:00
{
2017-08-20 14:43:34 +00:00
/* there is no active process. in most cases, the only process left
* should be the gc finalizer process started in the System > > startup .
* if there are other suspended processes at this point , the processes
* are not likely to run again .
*
* imagine the following single line program that creates a process
* but never start it .
*
* method ( # class ) main { | p | p : = [ ] newProcess . }
*
* the gc finalizer process and the process assigned to p exist .
* when the code reaches here , the ' p ' process still is alive
* despite no active process nor no process waiting on timers
* and semaphores . so when the entire program terminates , there
* might still be some suspended processes that are not possible
* to schedule .
*/
2017-11-21 09:15:22 +00:00
MOO_LOG4 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG ,
" Signaled GCFIN semaphore without gcfin signal request - total %zd runnable/running %zd suspended %zd - sem_io_wait_count %zu \n " ,
MOO_OOP_TO_SMOOI ( moo - > processor - > total_count ) ,
MOO_OOP_TO_SMOOI ( moo - > processor - > runnable . count ) ,
MOO_OOP_TO_SMOOI ( moo - > processor - > suspended . count ) ,
moo - > sem_io_wait_count ) ;
2017-07-27 08:32:16 +00:00
proc = signal_semaphore ( moo , moo - > sem_gcfin ) ;
if ( ( moo_oop_t ) proc ! = moo - > _nil )
{
MOO_ASSERT ( moo , proc - > state = = MOO_SMOOI_TO_OOP ( PROC_STATE_RUNNABLE ) ) ;
MOO_ASSERT ( moo , proc = = moo - > processor - > runnable . first ) ;
2017-08-20 14:43:34 +00:00
moo - > processor - > should_exit = moo - > _true ; /* prepare to inform the gc finalizer process */
switch_to_process_from_nil ( moo , proc ) ; /* sechedule the gc finalizer process */
2017-07-27 08:32:16 +00:00
}
2017-07-25 15:26:04 +00:00
}
}
}
2017-03-25 05:16:18 +00:00
#if 0
2017-02-11 05:48:30 +00:00
while ( moo - > sem_list_count > 0 )
{
/* handle async signals */
- - moo - > sem_list_count ;
signal_semaphore ( moo , moo - > sem_list [ moo - > sem_list_count ] ) ;
2017-07-27 08:32:16 +00:00
if ( moo - > processor - > active = = moo - > nil_process )
{
}
2017-02-11 05:48:30 +00:00
}
/*
if ( semaphore heap has pending request )
{
signal them . . .
} */
2017-07-27 08:32:16 +00:00
# endif
if ( moo - > processor - > active = = moo - > nil_process )
{
/* no more waiting semaphore and no more process */
MOO_ASSERT ( moo , moo - > processor - > runnable . count = MOO_SMOOI_TO_OOP ( 0 ) ) ;
MOO_LOG0 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG , " No more runnable process \n " ) ;
2017-11-04 03:57:38 +00:00
if ( MOO_OOP_TO_SMOOI ( moo - > processor - > suspended . count ) > 0 )
{
/* there exist suspended processes while no processes are runnable.
* most likely , the running program contains process / semaphore related bugs */
MOO_LOG1 ( moo , MOO_LOG_IC | MOO_LOG_DEBUG ,
" %zd suspended process(es) found - check your program \n " ,
MOO_OOP_TO_SMOOI ( moo - > processor - > suspended . count ) ) ;
}
2017-07-27 08:32:16 +00:00
return 0 ;
}
2017-02-11 05:48:30 +00:00
2017-07-27 08:32:16 +00:00
switch_to_next :
2017-02-11 05:48:30 +00:00
/* TODO: implement different process switching scheme - time-slice or clock based??? */
# if defined(MOO_EXTERNAL_PROCESS_SWITCH)
if ( ! moo - > proc_switched & & moo - > switch_proc ) { switch_to_next_runnable_process ( moo ) ; }
moo - > switch_proc = 0 ;
# else
if ( ! moo - > proc_switched ) { switch_to_next_runnable_process ( moo ) ; }
# endif
moo - > proc_switched = 0 ;
return 1 ;
}
2017-07-20 16:33:53 +00:00
2017-09-21 07:56:51 +00:00
static MOO_INLINE int do_return ( moo_t * moo , moo_oob_t bcode , moo_oop_t return_value )
2015-06-04 18:34:37 +00:00
{
2017-09-21 07:56:51 +00:00
#if 0
/* put the instruction pointer back to the return
* instruction ( RETURN_RECEIVER or RETURN_RECEIVER )
* if a context returns into this context again ,
* it ' ll be able to return as well again .
*
* Consider a program like this :
*
* # class MyObject ( Object )
* {
* # declare ( # classinst ) t1 t2 .
* # method ( # class ) xxxx
* {
* | g1 g2 |
* t1 dump .
* t2 : = [ g1 : = 50. g2 : = 100. ^ g1 + g2 ] .
* ( t1 < 100 ) ifFalse : [ ^ self ] .
* t1 : = t1 + 1.
* ^ self xxxx .
* }
* # method ( # class ) main
* {
* t1 : = 1.
* self xxxx .
* t2 : = t2 value .
* t2 dump .
* }
* }
*
* the ' xxxx ' method invoked by ' self xxxx ' has
* returned even before ' t2 value ' is executed .
* the ' ^ ' operator makes the active context to
* switch to its ' origin - > sender ' which is the
* method context of ' xxxx ' itself . placing its
* instruction pointer at the ' return ' instruction
* helps execute another return when the switching
* occurs .
*
* TODO : verify if this really works
*
*/
moo - > ip - - ;
# else
2016-06-20 15:42:51 +00:00
int unwind_protect ;
2017-01-09 09:54:49 +00:00
moo_oop_context_t unwind_start ;
moo_oop_context_t unwind_stop ;
2016-10-06 14:17:24 +00:00
2017-09-21 07:56:51 +00:00
if ( MOO_UNLIKELY ( moo - > active_context - > origin = = moo - > processor - > active - > initial_context - > origin ) )
{
/* method return from a processified block
*
* # method ( # class ) main
* {
* [ ^ 100 ] newProcess resume .
* ' 1111 ' dump .
* ' 1111 ' dump .
* ' 1111 ' dump .
* ^ 300.
* }
*
* ^ 100 doesn ' t terminate a main process as the block
* has been processified . on the other hand , ^ 100
* in the following program causes main to exit .
*
* # method ( # class ) main
* {
* [ ^ 100 ] value .
* ' 1111 ' dump .
* ' 1111 ' dump .
* ' 1111 ' dump .
* ^ 300.
* }
*/
2017-02-18 13:31:47 +00:00
2017-09-21 07:56:51 +00:00
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , moo - > active_context ) = = moo - > _block_context ) ;
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , moo - > processor - > active - > initial_context ) = = moo - > _block_context ) ;
2015-10-19 06:16:43 +00:00
2017-09-21 07:56:51 +00:00
/* decrement the instruction pointer back to the return instruction.
* even if the context is reentered , it will just return .
* moo - > ip - - ; */
2015-06-07 14:36:26 +00:00
2017-09-21 07:56:51 +00:00
terminate_process ( moo , moo - > processor - > active ) ;
}
else
{
unwind_protect = 0 ;
2015-06-06 07:24:35 +00:00
2017-09-21 07:56:51 +00:00
/* set the instruction pointer to an invalid value.
* this is stored into the current method context
* before context switching and marks a dead context */
if ( moo - > active_context - > origin = = moo - > active_context )
2015-06-29 13:52:40 +00:00
{
2017-09-21 07:56:51 +00:00
/* returning from a method */
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , moo - > active_context ) = = moo - > _method_context ) ;
2015-06-07 14:36:26 +00:00
2017-09-21 07:56:51 +00:00
/* mark that the context is dead. it will be
* save to the context object by SWITCH_ACTIVE_CONTEXT ( ) */
moo - > ip = - 1 ;
}
else
{
moo_oop_context_t ctx ;
2015-06-29 13:52:40 +00:00
2017-09-21 07:56:51 +00:00
/* method return from within a block(including a non-local return) */
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , moo - > active_context ) = = moo - > _block_context ) ;
2015-06-29 13:52:40 +00:00
2017-09-21 07:56:51 +00:00
ctx = moo - > active_context ;
while ( ( moo_oop_t ) ctx ! = moo - > _nil )
2015-06-30 14:47:39 +00:00
{
2017-09-21 07:56:51 +00:00
if ( MOO_CLASSOF ( moo , ctx ) = = moo - > _method_context )
2015-06-22 14:21:46 +00:00
{
2017-09-21 07:56:51 +00:00
moo_ooi_t preamble ;
preamble = MOO_OOP_TO_SMOOI ( ( ( moo_oop_method_t ) ctx - > method_or_nargs ) - > preamble ) ;
if ( MOO_METHOD_GET_PREAMBLE_CODE ( preamble ) = = MOO_METHOD_PREAMBLE_ENSURE )
2015-06-22 14:21:46 +00:00
{
2017-09-21 07:56:51 +00:00
if ( ! unwind_protect )
2015-06-22 14:21:46 +00:00
{
2017-09-21 07:56:51 +00:00
unwind_protect = 1 ;
unwind_start = ctx ;
2015-06-22 14:21:46 +00:00
}
2017-09-21 07:56:51 +00:00
unwind_stop = ctx ;
2015-06-22 14:21:46 +00:00
}
2015-06-29 13:52:40 +00:00
}
2017-09-21 07:56:51 +00:00
if ( ctx = = moo - > active_context - > origin ) goto non_local_return_ok ;
ctx = ctx - > sender ;
}
2015-06-22 14:21:46 +00:00
2017-09-21 07:56:51 +00:00
/* cannot return from a method that has returned already */
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , moo - > active_context - > origin ) = = moo - > _method_context ) ;
MOO_ASSERT ( moo , moo - > active_context - > origin - > ip = = MOO_SMOOI_TO_OOP ( - 1 ) ) ;
2015-06-22 14:21:46 +00:00
2017-09-21 07:56:51 +00:00
MOO_LOG0 ( moo , MOO_LOG_IC | MOO_LOG_ERROR , " Error - cannot return from dead context \n " ) ;
moo_seterrnum ( moo , MOO_EINTERN ) ; /* TODO: can i make this error catchable at the moo level? */
return - 1 ;
2016-06-05 18:01:35 +00:00
2017-09-21 07:56:51 +00:00
non_local_return_ok :
/*MOO_DEBUG2 (moo, "NON_LOCAL RETURN OK TO... %p %p\n", moo->active_context->origin, moo->active_context->origin->sender);*/
if ( bcode ! = BCODE_LOCAL_RETURN )
{
/* mark that the context is dead */
moo - > active_context - > origin - > ip = MOO_SMOOI_TO_OOP ( - 1 ) ;
2015-06-30 14:47:39 +00:00
}
2017-09-21 07:56:51 +00:00
}
2015-06-07 14:36:26 +00:00
2017-09-21 07:56:51 +00:00
/* the origin must always be a method context for both an active block context
* or an active method context */
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , moo - > active_context - > origin ) = = moo - > _method_context ) ;
2015-06-07 14:36:26 +00:00
2017-09-21 07:56:51 +00:00
/* restore the stack pointer */
moo - > sp = MOO_OOP_TO_SMOOI ( moo - > active_context - > origin - > sp ) ;
if ( bcode = = BCODE_LOCAL_RETURN & & moo - > active_context ! = moo - > active_context - > origin )
{
SWITCH_ACTIVE_CONTEXT ( moo , moo - > active_context - > origin ) ;
}
else
{
SWITCH_ACTIVE_CONTEXT ( moo , moo - > active_context - > origin - > sender ) ;
}
2015-06-30 14:47:39 +00:00
2017-09-21 07:56:51 +00:00
if ( unwind_protect )
{
static moo_ooch_t fbm [ ] = {
' u ' , ' n ' , ' w ' , ' i ' , ' n ' , ' d ' , ' T ' , ' o ' , ' : ' ,
' r ' , ' e ' , ' t ' , ' u ' , ' r ' , ' n ' , ' : '
} ;
2015-06-30 14:47:39 +00:00
2017-09-21 07:56:51 +00:00
MOO_STACK_PUSH ( moo , ( moo_oop_t ) unwind_start ) ;
MOO_STACK_PUSH ( moo , ( moo_oop_t ) unwind_stop ) ;
MOO_STACK_PUSH ( moo , ( moo_oop_t ) return_value ) ;
2015-06-30 14:47:39 +00:00
2017-09-21 07:56:51 +00:00
if ( send_message_with_str ( moo , fbm , 16 , 0 , 2 ) < = - 1 ) return - 1 ;
}
else
{
/* push the return value to the stack of the new active context */
MOO_STACK_PUSH ( moo , return_value ) ;
if ( moo - > active_context = = moo - > initial_context )
{
/* the new active context is the fake initial context.
* this context can ' t get executed further . */
MOO_ASSERT ( moo , ( moo_oop_t ) moo - > active_context - > sender = = moo - > _nil ) ;
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , moo - > active_context ) = = moo - > _method_context ) ;
MOO_ASSERT ( moo , moo - > active_context - > receiver_or_source = = moo - > _nil ) ;
MOO_ASSERT ( moo , moo - > active_context = = moo - > processor - > active - > initial_context ) ;
MOO_ASSERT ( moo , moo - > active_context - > origin = = moo - > processor - > active - > initial_context - > origin ) ;
MOO_ASSERT ( moo , moo - > active_context - > origin = = moo - > active_context ) ;
/* NOTE: this condition is true for the processified block context also.
* moo - > active_context - > origin = = moo - > processor - > active - > initial_context - > origin
* however , the check here is done after context switching and the
* processified block check has been done against the context before switching */
/* the stack contains the final return value so the stack pointer must be 0. */
MOO_ASSERT ( moo , moo - > sp = = 0 ) ;
if ( moo - > option . trait & MOO_AWAIT_PROCS )
{
terminate_process ( moo , moo - > processor - > active ) ;
2015-06-22 14:21:46 +00:00
}
else
{
2017-09-21 07:56:51 +00:00
/* graceful termination of the whole vm */
return 0 ;
2015-06-22 14:21:46 +00:00
}
2017-09-21 07:56:51 +00:00
/* TODO: store the return value to the VM register.
* the caller to moo_execute ( ) can fetch it to return it to the system */
2015-06-30 14:47:39 +00:00
}
2017-09-21 07:56:51 +00:00
}
}
# endif
2015-06-07 14:36:26 +00:00
2017-09-21 07:56:51 +00:00
return 1 ;
}
2015-06-16 04:31:28 +00:00
2015-06-07 14:36:26 +00:00
2017-09-21 07:56:51 +00:00
static MOO_INLINE void do_return_from_block ( moo_t * moo )
{
LOG_INST_0 ( moo , " return_from_block " ) ;
2015-06-09 12:14:18 +00:00
2017-09-21 07:56:51 +00:00
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , moo - > active_context ) = = moo - > _block_context ) ;
2015-06-29 13:52:40 +00:00
2017-09-21 07:56:51 +00:00
if ( moo - > active_context = = moo - > processor - > active - > initial_context )
{
/* the active context to return from is an initial context of
* the active process . this process must have been created
* over a block using the newProcess method . let ' s terminate
* the process . */
2015-06-29 13:52:40 +00:00
2017-09-21 07:56:51 +00:00
MOO_ASSERT ( moo , ( moo_oop_t ) moo - > active_context - > sender = = moo - > _nil ) ;
terminate_process ( moo , moo - > processor - > active ) ;
}
else
{
/* it is a normal block return as the active block context
* is not the initial context of a process */
2017-01-18 17:17:05 +00:00
2017-09-21 07:56:51 +00:00
/* the process stack is shared. the return value
* doesn ' t need to get moved . */
SWITCH_ACTIVE_CONTEXT ( moo , ( moo_oop_context_t ) moo - > active_context - > sender ) ;
}
}
2015-07-01 07:21:54 +00:00
2017-09-21 07:56:51 +00:00
static MOO_INLINE int make_block ( moo_t * moo )
{
moo_oop_context_t blkctx ;
moo_oob_t b1 , b2 ;
/* b1 - number of block arguments
* b2 - number of block temporaries */
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
FETCH_PARAM_CODE_TO ( moo , b2 ) ;
LOG_INST_2 ( moo , " make_block %zu %zu " , b1 , b2 ) ;
MOO_ASSERT ( moo , b1 > = 0 ) ;
MOO_ASSERT ( moo , b2 > = b1 ) ;
/* the block context object created here is used as a base
* object for block context activation . pf_block_value ( )
* clones a block context and activates the cloned context .
* this base block context is created with no stack for
* this reason */
blkctx = ( moo_oop_context_t ) moo_instantiate ( moo , moo - > _block_context , MOO_NULL , 0 ) ;
if ( ! blkctx ) return - 1 ;
/* the long forward jump instruction has the format of
* 11000100 KKKKKKKK or 11000100 KKKKKKKK KKKKKKKK
* depending on MOO_BCODE_LONG_PARAM_SIZE . change ' ip ' to point to
* the instruction after the jump . */
blkctx - > ip = MOO_SMOOI_TO_OOP ( moo - > ip + MOO_BCODE_LONG_PARAM_SIZE + 1 ) ;
/* stack pointer below the bottom. this base block context
* has an empty stack anyway . */
blkctx - > sp = MOO_SMOOI_TO_OOP ( - 1 ) ;
/* the number of arguments for a block context is local to the block */
blkctx - > method_or_nargs = MOO_SMOOI_TO_OOP ( b1 ) ;
/* the number of temporaries here is an accumulated count including
* the number of temporaries of a home context */
blkctx - > ntmprs = MOO_SMOOI_TO_OOP ( b2 ) ;
/* set the home context where it's defined */
blkctx - > home = ( moo_oop_t ) moo - > active_context ;
/* no source for a base block context. */
blkctx - > receiver_or_source = moo - > _nil ;
blkctx - > origin = moo - > active_context - > origin ;
/* push the new block context to the stack of the active context */
MOO_STACK_PUSH ( moo , ( moo_oop_t ) blkctx ) ;
return 0 ;
}
2017-01-30 16:48:42 +00:00
2017-09-21 07:56:51 +00:00
static int __execute ( moo_t * moo )
{
moo_oob_t bcode ;
moo_oow_t b1 , b2 ;
moo_oop_t return_value ;
2017-01-30 16:48:42 +00:00
2017-09-21 07:56:51 +00:00
# if defined(HAVE_LABELS_AS_VALUES)
static void * inst_table [ 256 ] =
{
# include "moo-bct.h"
} ;
2017-04-26 15:31:07 +00:00
2017-09-21 07:56:51 +00:00
# define BEGIN_DISPATCH_LOOP() __begin_inst_dispatch:
# define END_DISPATCH_LOOP() __end_inst_dispatch:
2017-09-26 23:48:33 +00:00
# define EXIT_DISPATCH_LOOP() goto __end_inst_dispatch
2017-09-21 07:56:51 +00:00
# define NEXT_INST() goto __begin_inst_dispatch
2017-04-26 15:31:07 +00:00
2017-09-21 07:56:51 +00:00
# define BEGIN_DISPATCH_TABLE() goto *inst_table[bcode];
# define END_DISPATCH_TABLE()
2015-07-01 07:21:54 +00:00
2017-09-21 07:56:51 +00:00
# define ON_INST(code) case_ ## code:
# define ON_UNKNOWN_INST() case_ ## DEFAULT:
2015-06-29 13:52:40 +00:00
2017-09-21 07:56:51 +00:00
# else
# define BEGIN_DISPATCH_LOOP() __begin_inst_dispatch:
# define END_DISPATCH_LOOP() __end_inst_dispatch:
2017-09-26 23:48:33 +00:00
# define EXIT_DISPATCH_LOOP() goto __end_inst_dispatch
2017-09-21 07:56:51 +00:00
# define NEXT_INST() goto __begin_inst_dispatch
2017-01-18 17:17:05 +00:00
2017-09-21 07:56:51 +00:00
# define BEGIN_DISPATCH_TABLE() switch (bcode) {
# define END_DISPATCH_TABLE() }
2017-04-26 15:31:07 +00:00
2017-09-21 07:56:51 +00:00
# define ON_INST(code) case code:
# define ON_UNKNOWN_INST() default:
2017-01-18 17:17:05 +00:00
2017-09-21 07:56:51 +00:00
# endif
2015-06-30 14:47:39 +00:00
2017-09-21 07:56:51 +00:00
MOO_ASSERT ( moo , moo - > active_context ! = MOO_NULL ) ;
2015-06-30 14:47:39 +00:00
2017-09-21 07:56:51 +00:00
/* TODO: initialize semaphore stuffs
* sem_heap
* sem_io .
* sem_list .
* these can get dirty if this function is called again esepcially after failure .
*/
BEGIN_DISPATCH_LOOP ( )
2017-09-26 23:48:33 +00:00
2017-09-21 07:56:51 +00:00
/* stop requested or no more runnable process */
if ( moo - > abort_req | | switch_process_if_needed ( moo ) = = 0 ) EXIT_DISPATCH_LOOP ( ) ;
# if defined(MOO_DEBUG_VM_EXEC)
moo - > last_inst_pointer = moo - > ip ;
# endif
bcode = FETCH_BYTE_CODE ( moo ) ;
# if defined(MOO_PROFILE_VM)
moo - > inst_counter + + ;
# endif
/* ==== DISPATCH TABLE ==== */
BEGIN_DISPATCH_TABLE ( )
ON_INST ( BCODE_PUSH_INSTVAR_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
goto push_instvar ;
ON_INST ( BCODE_PUSH_INSTVAR_0 )
ON_INST ( BCODE_PUSH_INSTVAR_1 )
ON_INST ( BCODE_PUSH_INSTVAR_2 )
ON_INST ( BCODE_PUSH_INSTVAR_3 )
ON_INST ( BCODE_PUSH_INSTVAR_4 )
ON_INST ( BCODE_PUSH_INSTVAR_5 )
ON_INST ( BCODE_PUSH_INSTVAR_6 )
ON_INST ( BCODE_PUSH_INSTVAR_7 )
b1 = bcode & 0x7 ; /* low 3 bits */
push_instvar :
LOG_INST_1 ( moo , " push_instvar %zu " , b1 ) ;
MOO_ASSERT ( moo , MOO_OBJ_GET_FLAGS_TYPE ( moo - > active_context - > origin - > receiver_or_source ) = = MOO_OBJ_TYPE_OOP ) ;
MOO_STACK_PUSH ( moo , ( ( moo_oop_oop_t ) moo - > active_context - > origin - > receiver_or_source ) - > slot [ b1 ] ) ;
NEXT_INST ( ) ;
/* ------------------------------------------------- */
ON_INST ( BCODE_STORE_INTO_INSTVAR_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
goto store_instvar ;
ON_INST ( BCODE_STORE_INTO_INSTVAR_0 )
ON_INST ( BCODE_STORE_INTO_INSTVAR_1 )
ON_INST ( BCODE_STORE_INTO_INSTVAR_2 )
ON_INST ( BCODE_STORE_INTO_INSTVAR_3 )
ON_INST ( BCODE_STORE_INTO_INSTVAR_4 )
ON_INST ( BCODE_STORE_INTO_INSTVAR_5 )
ON_INST ( BCODE_STORE_INTO_INSTVAR_6 )
ON_INST ( BCODE_STORE_INTO_INSTVAR_7 )
b1 = bcode & 0x7 ; /* low 3 bits */
store_instvar :
LOG_INST_1 ( moo , " store_into_instvar %zu " , b1 ) ;
MOO_ASSERT ( moo , MOO_OBJ_GET_FLAGS_TYPE ( moo - > active_context - > receiver_or_source ) = = MOO_OBJ_TYPE_OOP ) ;
( ( moo_oop_oop_t ) moo - > active_context - > origin - > receiver_or_source ) - > slot [ b1 ] = MOO_STACK_GETTOP ( moo ) ;
NEXT_INST ( ) ;
/* ------------------------------------------------- */
ON_INST ( BCODE_POP_INTO_INSTVAR_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
goto pop_into_instvar ;
ON_INST ( BCODE_POP_INTO_INSTVAR_0 )
ON_INST ( BCODE_POP_INTO_INSTVAR_1 )
ON_INST ( BCODE_POP_INTO_INSTVAR_2 )
ON_INST ( BCODE_POP_INTO_INSTVAR_3 )
ON_INST ( BCODE_POP_INTO_INSTVAR_4 )
ON_INST ( BCODE_POP_INTO_INSTVAR_5 )
ON_INST ( BCODE_POP_INTO_INSTVAR_6 )
ON_INST ( BCODE_POP_INTO_INSTVAR_7 )
b1 = bcode & 0x7 ; /* low 3 bits */
pop_into_instvar :
LOG_INST_1 ( moo , " pop_into_instvar %zu " , b1 ) ;
MOO_ASSERT ( moo , MOO_OBJ_GET_FLAGS_TYPE ( moo - > active_context - > receiver_or_source ) = = MOO_OBJ_TYPE_OOP ) ;
( ( moo_oop_oop_t ) moo - > active_context - > origin - > receiver_or_source ) - > slot [ b1 ] = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
/* ------------------------------------------------- */
ON_INST ( BCODE_PUSH_TEMPVAR_X )
ON_INST ( BCODE_STORE_INTO_TEMPVAR_X )
ON_INST ( BCODE_POP_INTO_TEMPVAR_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
goto handle_tempvar ;
ON_INST ( BCODE_PUSH_TEMPVAR_0 )
ON_INST ( BCODE_PUSH_TEMPVAR_1 )
ON_INST ( BCODE_PUSH_TEMPVAR_2 )
ON_INST ( BCODE_PUSH_TEMPVAR_3 )
ON_INST ( BCODE_PUSH_TEMPVAR_4 )
ON_INST ( BCODE_PUSH_TEMPVAR_5 )
ON_INST ( BCODE_PUSH_TEMPVAR_6 )
ON_INST ( BCODE_PUSH_TEMPVAR_7 )
ON_INST ( BCODE_STORE_INTO_TEMPVAR_0 )
ON_INST ( BCODE_STORE_INTO_TEMPVAR_1 )
ON_INST ( BCODE_STORE_INTO_TEMPVAR_2 )
ON_INST ( BCODE_STORE_INTO_TEMPVAR_3 )
ON_INST ( BCODE_STORE_INTO_TEMPVAR_4 )
ON_INST ( BCODE_STORE_INTO_TEMPVAR_5 )
ON_INST ( BCODE_STORE_INTO_TEMPVAR_6 )
ON_INST ( BCODE_STORE_INTO_TEMPVAR_7 )
ON_INST ( BCODE_POP_INTO_TEMPVAR_0 )
ON_INST ( BCODE_POP_INTO_TEMPVAR_1 )
ON_INST ( BCODE_POP_INTO_TEMPVAR_2 )
ON_INST ( BCODE_POP_INTO_TEMPVAR_3 )
ON_INST ( BCODE_POP_INTO_TEMPVAR_4 )
ON_INST ( BCODE_POP_INTO_TEMPVAR_5 )
ON_INST ( BCODE_POP_INTO_TEMPVAR_6 )
ON_INST ( BCODE_POP_INTO_TEMPVAR_7 )
{
moo_oop_context_t ctx ;
moo_ooi_t bx ;
b1 = bcode & 0x7 ; /* low 3 bits */
handle_tempvar :
# if defined(MOO_USE_CTXTEMPVAR)
/* when CTXTEMPVAR inststructions are used, the above
* instructions are used only for temporary access
* outside a block . i can assume that the temporary
* variable index is pointing to one of temporaries
* in the relevant method context */
ctx = moo - > active_context - > origin ;
bx = b1 ;
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , ctx ) = = moo - > _method_context ) ;
# else
/* otherwise, the index may point to a temporaries
* declared inside a block */
if ( moo - > active_context - > home ! = moo - > _nil )
{
/* this code assumes that the method context and
* the block context place some key fields in the
* same offset . such fields include ' home ' , ' ntmprs ' */
moo_oop_t home ;
moo_ooi_t home_ntmprs ;
2015-06-30 14:47:39 +00:00
2017-01-09 09:54:49 +00:00
ctx = moo - > active_context ;
2017-09-21 07:56:51 +00:00
home = ctx - > home ;
do
2015-06-30 14:47:39 +00:00
{
2017-09-21 07:56:51 +00:00
/* ntmprs contains the number of defined temporaries
* including those defined in the home context */
home_ntmprs = MOO_OOP_TO_SMOOI ( ( ( moo_oop_context_t ) home ) - > ntmprs ) ;
if ( b1 > = home_ntmprs ) break ;
ctx = ( moo_oop_context_t ) home ;
home = ( ( moo_oop_context_t ) home ) - > home ;
if ( home = = moo - > _nil )
{
home_ntmprs = 0 ;
break ;
}
2015-06-30 14:47:39 +00:00
}
2017-09-21 07:56:51 +00:00
while ( 1 ) ;
/* bx is the actual index within the actual context
* containing the temporary */
bx = b1 - home_ntmprs ;
}
else
{
ctx = moo - > active_context ;
bx = b1 ;
}
# endif
if ( ( bcode > > 4 ) & 1 )
{
/* push - bit 4 on */
LOG_INST_1 ( moo , " push_tempvar %zu " , b1 ) ;
MOO_STACK_PUSH ( moo , ctx - > slot [ bx ] ) ;
}
else
{
/* store or pop - bit 5 off */
ctx - > slot [ bx ] = MOO_STACK_GETTOP ( moo ) ;
2015-06-30 14:47:39 +00:00
2015-06-29 13:52:40 +00:00
if ( ( bcode > > 3 ) & 1 )
{
2017-09-21 07:56:51 +00:00
/* pop - bit 3 on */
LOG_INST_1 ( moo , " pop_into_tempvar %zu " , b1 ) ;
MOO_STACK_POP ( moo ) ;
2015-06-29 13:52:40 +00:00
}
else
{
2017-09-21 07:56:51 +00:00
LOG_INST_1 ( moo , " store_into_tempvar %zu " , b1 ) ;
2015-06-29 13:52:40 +00:00
}
2015-06-30 14:47:39 +00:00
}
2017-09-21 07:56:51 +00:00
NEXT_INST ( ) ;
}
2015-06-12 13:52:30 +00:00
2017-09-21 07:56:51 +00:00
/* ------------------------------------------------- */
ON_INST ( BCODE_PUSH_LITERAL_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
goto push_literal ;
ON_INST ( BCODE_PUSH_LITERAL_0 )
ON_INST ( BCODE_PUSH_LITERAL_1 )
ON_INST ( BCODE_PUSH_LITERAL_2 )
ON_INST ( BCODE_PUSH_LITERAL_3 )
ON_INST ( BCODE_PUSH_LITERAL_4 )
ON_INST ( BCODE_PUSH_LITERAL_5 )
ON_INST ( BCODE_PUSH_LITERAL_6 )
ON_INST ( BCODE_PUSH_LITERAL_7 )
b1 = bcode & 0x7 ; /* low 3 bits */
push_literal :
LOG_INST_1 ( moo , " push_literal @%zu " , b1 ) ;
MOO_STACK_PUSH ( moo , moo - > active_method - > slot [ b1 ] ) ;
NEXT_INST ( ) ;
/* ------------------------------------------------- */
ON_INST ( BCODE_PUSH_OBJECT_X )
ON_INST ( BCODE_STORE_INTO_OBJECT_X )
ON_INST ( BCODE_POP_INTO_OBJECT_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
goto handle_object ;
ON_INST ( BCODE_PUSH_OBJECT_0 )
ON_INST ( BCODE_PUSH_OBJECT_1 )
ON_INST ( BCODE_PUSH_OBJECT_2 )
ON_INST ( BCODE_PUSH_OBJECT_3 )
ON_INST ( BCODE_STORE_INTO_OBJECT_0 )
ON_INST ( BCODE_STORE_INTO_OBJECT_1 )
ON_INST ( BCODE_STORE_INTO_OBJECT_2 )
ON_INST ( BCODE_STORE_INTO_OBJECT_3 )
ON_INST ( BCODE_POP_INTO_OBJECT_0 )
ON_INST ( BCODE_POP_INTO_OBJECT_1 )
ON_INST ( BCODE_POP_INTO_OBJECT_2 )
ON_INST ( BCODE_POP_INTO_OBJECT_3 )
{
moo_oop_association_t ass ;
2015-06-23 14:00:26 +00:00
2017-09-21 07:56:51 +00:00
b1 = bcode & 0x3 ; /* low 2 bits */
handle_object :
ass = ( moo_oop_association_t ) moo - > active_method - > slot [ b1 ] ;
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , ass ) = = moo - > _association ) ;
2015-06-12 13:52:30 +00:00
2017-09-21 07:56:51 +00:00
if ( ( bcode > > 3 ) & 1 )
{
/* store or pop */
ass - > value = MOO_STACK_GETTOP ( moo ) ;
2015-06-23 14:00:26 +00:00
2017-09-21 07:56:51 +00:00
if ( ( bcode > > 2 ) & 1 )
{
/* pop */
LOG_INST_1 ( moo , " pop_into_object @%zu " , b1 ) ;
MOO_STACK_POP ( moo ) ;
2015-06-29 13:52:40 +00:00
}
else
{
2017-09-21 07:56:51 +00:00
LOG_INST_1 ( moo , " store_into_object @%zu " , b1 ) ;
2015-06-29 13:52:40 +00:00
}
2015-06-30 14:47:39 +00:00
}
2017-09-21 07:56:51 +00:00
else
2015-06-08 13:24:02 +00:00
{
2017-09-21 07:56:51 +00:00
/* push */
LOG_INST_1 ( moo , " push_object @%zu " , b1 ) ;
MOO_STACK_PUSH ( moo , ass - > value ) ;
}
NEXT_INST ( ) ;
}
2015-06-08 13:24:02 +00:00
2017-09-21 07:56:51 +00:00
/* -------------------------------------------------------- */
ON_INST ( BCODE_JUMP_FORWARD_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump_forward %zu " , b1 ) ;
moo - > ip + = b1 ;
NEXT_INST ( ) ;
ON_INST ( BCODE_JUMP_FORWARD_0 )
ON_INST ( BCODE_JUMP_FORWARD_1 )
ON_INST ( BCODE_JUMP_FORWARD_2 )
ON_INST ( BCODE_JUMP_FORWARD_3 )
LOG_INST_1 ( moo , " jump_forward %zu " , ( moo_oow_t ) ( bcode & 0x3 ) ) ;
moo - > ip + = ( bcode & 0x3 ) ; /* low 2 bits */
NEXT_INST ( ) ;
ON_INST ( BCODE_JUMP_BACKWARD_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump_backward %zu " , b1 ) ;
moo - > ip - = b1 ;
NEXT_INST ( ) ;
ON_INST ( BCODE_JUMP_BACKWARD_0 )
ON_INST ( BCODE_JUMP_BACKWARD_1 )
ON_INST ( BCODE_JUMP_BACKWARD_2 )
ON_INST ( BCODE_JUMP_BACKWARD_3 )
LOG_INST_1 ( moo , " jump_backward %zu " , ( moo_oow_t ) ( bcode & 0x3 ) ) ;
moo - > ip - = ( bcode & 0x3 ) ; /* low 2 bits */
NEXT_INST ( ) ;
ON_INST ( BCODE_JUMP_BACKWARD_IF_FALSE_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump_backward_if_false %zu " , b1 ) ;
if ( MOO_STACK_GETTOP ( moo ) = = moo - > _false ) moo - > ip - = b1 ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
ON_INST ( BCODE_JUMP_BACKWARD_IF_FALSE_0 )
ON_INST ( BCODE_JUMP_BACKWARD_IF_FALSE_1 )
ON_INST ( BCODE_JUMP_BACKWARD_IF_FALSE_2 )
ON_INST ( BCODE_JUMP_BACKWARD_IF_FALSE_3 )
LOG_INST_1 ( moo , " jump_backward_if_false %zu " , ( moo_oow_t ) ( bcode & 0x3 ) ) ;
if ( MOO_STACK_GETTOP ( moo ) = = moo - > _false ) moo - > ip - = ( bcode & 0x3 ) ; /* low 2 bits */
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
2015-06-08 13:24:02 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_JUMP_BACKWARD_IF_TRUE_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump_backward_if_true %zu " , b1 ) ;
/*if (MOO_STACK_GETTOP(moo) == moo->_true) moo->ip -= b1;*/
if ( MOO_STACK_GETTOP ( moo ) ! = moo - > _false ) moo - > ip - = b1 ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
ON_INST ( BCODE_JUMP_BACKWARD_IF_TRUE_0 )
ON_INST ( BCODE_JUMP_BACKWARD_IF_TRUE_1 )
ON_INST ( BCODE_JUMP_BACKWARD_IF_TRUE_2 )
ON_INST ( BCODE_JUMP_BACKWARD_IF_TRUE_3 )
LOG_INST_1 ( moo , " jump_backward_if_true %zu " , ( moo_oow_t ) ( bcode & 0x3 ) ) ;
/*if (MOO_STACK_GETTOP(moo) == moo->_true) moo->ip -= (bcode & 0x3);*/ /* low 2 bits */
if ( MOO_STACK_GETTOP ( moo ) ! = moo - > _false ) moo - > ip - = ( bcode & 0x3 ) ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
2015-06-09 12:14:18 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_JUMP_FORWARD_IF_FALSE )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump_forward_if_false %zu " , b1 ) ;
if ( MOO_STACK_GETTOP ( moo ) = = moo - > _false ) moo - > ip + = b1 ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
2015-06-07 14:36:26 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_JUMP_FORWARD_IF_TRUE )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump_forward_if_true %zu " , b1 ) ;
/*if (MOO_STACK_GETTOP(moo) == moo->_true) moo->ip += b1;*/
if ( MOO_STACK_GETTOP ( moo ) ! = moo - > _false ) moo - > ip + = b1 ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
ON_INST ( BCODE_JUMP2_FORWARD )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump2_forward %zu " , b1 ) ;
moo - > ip + = MAX_CODE_JUMP + b1 ;
NEXT_INST ( ) ;
ON_INST ( BCODE_JUMP2_BACKWARD )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump2_backward %zu " , b1 ) ;
moo - > ip - = MAX_CODE_JUMP + b1 ;
NEXT_INST ( ) ;
ON_INST ( BCODE_JUMP2_FORWARD_IF_FALSE )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump2_forward_if_false %zu " , b1 ) ;
if ( MOO_STACK_GETTOP ( moo ) = = moo - > _false ) moo - > ip + = MAX_CODE_JUMP + b1 ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
2015-06-07 14:36:26 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_JUMP2_FORWARD_IF_TRUE )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump2_forward_if_true %zu " , b1 ) ;
/*if (MOO_STACK_GETTOP(moo) == moo->_true) moo->ip += MAX_CODE_JUMP + b1;*/
if ( MOO_STACK_GETTOP ( moo ) ! = moo - > _false ) moo - > ip + = MAX_CODE_JUMP + b1 ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
2015-06-07 14:36:26 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_JUMP2_BACKWARD_IF_FALSE )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump2_backward_if_false %zu " , b1 ) ;
if ( MOO_STACK_GETTOP ( moo ) = = moo - > _false ) moo - > ip - = MAX_CODE_JUMP + b1 ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
2015-06-07 14:36:26 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_JUMP2_BACKWARD_IF_TRUE )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " jump2_backward_if_true %zu " , b1 ) ;
/* if (MOO_STACK_GETTOP(moo) == moo->_true) moo->ip -= MAX_CODE_JUMP + b1; */
if ( MOO_STACK_GETTOP ( moo ) ! = moo - > _false ) moo - > ip - = MAX_CODE_JUMP + b1 ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
/* -------------------------------------------------------- */
ON_INST ( BCODE_PUSH_CTXTEMPVAR_X )
ON_INST ( BCODE_STORE_INTO_CTXTEMPVAR_X )
ON_INST ( BCODE_POP_INTO_CTXTEMPVAR_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
FETCH_PARAM_CODE_TO ( moo , b2 ) ;
goto handle_ctxtempvar ;
ON_INST ( BCODE_PUSH_CTXTEMPVAR_0 )
ON_INST ( BCODE_PUSH_CTXTEMPVAR_1 )
ON_INST ( BCODE_PUSH_CTXTEMPVAR_2 )
ON_INST ( BCODE_PUSH_CTXTEMPVAR_3 )
ON_INST ( BCODE_STORE_INTO_CTXTEMPVAR_0 )
ON_INST ( BCODE_STORE_INTO_CTXTEMPVAR_1 )
ON_INST ( BCODE_STORE_INTO_CTXTEMPVAR_2 )
ON_INST ( BCODE_STORE_INTO_CTXTEMPVAR_3 )
ON_INST ( BCODE_POP_INTO_CTXTEMPVAR_0 )
ON_INST ( BCODE_POP_INTO_CTXTEMPVAR_1 )
ON_INST ( BCODE_POP_INTO_CTXTEMPVAR_2 )
ON_INST ( BCODE_POP_INTO_CTXTEMPVAR_3 )
{
moo_ooi_t i ;
moo_oop_context_t ctx ;
2015-06-09 12:14:18 +00:00
2017-09-21 07:56:51 +00:00
b1 = bcode & 0x3 ; /* low 2 bits */
b2 = FETCH_BYTE_CODE ( moo ) ;
2015-06-16 04:31:28 +00:00
2017-09-21 07:56:51 +00:00
handle_ctxtempvar :
2016-07-05 15:22:29 +00:00
2017-09-21 07:56:51 +00:00
ctx = moo - > active_context ;
MOO_ASSERT ( moo , ( moo_oop_t ) ctx ! = moo - > _nil ) ;
for ( i = 0 ; i < b1 ; i + + )
2017-06-01 15:42:05 +00:00
{
2017-09-21 07:56:51 +00:00
ctx = ( moo_oop_context_t ) ctx - > home ;
2017-06-01 15:42:05 +00:00
}
2017-09-21 07:56:51 +00:00
if ( ( bcode > > 3 ) & 1 )
{
/* store or pop */
ctx - > slot [ b2 ] = MOO_STACK_GETTOP ( moo ) ;
2015-07-02 15:45:48 +00:00
2017-09-21 07:56:51 +00:00
if ( ( bcode > > 2 ) & 1 )
{
/* pop */
MOO_STACK_POP ( moo ) ;
LOG_INST_2 ( moo , " pop_into_ctxtempvar %zu %zu " , b1 , b2 ) ;
}
else
{
LOG_INST_2 ( moo , " store_into_ctxtempvar %zu %zu " , b1 , b2 ) ;
}
}
else
2016-10-04 17:58:28 +00:00
{
2017-09-21 07:56:51 +00:00
/* push */
MOO_STACK_PUSH ( moo , ctx - > slot [ b2 ] ) ;
LOG_INST_2 ( moo , " push_ctxtempvar %zu %zu " , b1 , b2 ) ;
2016-10-04 17:58:28 +00:00
}
2015-07-02 15:45:48 +00:00
2017-09-21 07:56:51 +00:00
NEXT_INST ( ) ;
}
/* -------------------------------------------------------- */
ON_INST ( BCODE_PUSH_OBJVAR_X )
ON_INST ( BCODE_STORE_INTO_OBJVAR_X )
ON_INST ( BCODE_POP_INTO_OBJVAR_X )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
FETCH_PARAM_CODE_TO ( moo , b2 ) ;
goto handle_objvar ;
ON_INST ( BCODE_PUSH_OBJVAR_0 )
ON_INST ( BCODE_PUSH_OBJVAR_1 )
ON_INST ( BCODE_PUSH_OBJVAR_2 )
ON_INST ( BCODE_PUSH_OBJVAR_3 )
ON_INST ( BCODE_STORE_INTO_OBJVAR_0 )
ON_INST ( BCODE_STORE_INTO_OBJVAR_1 )
ON_INST ( BCODE_STORE_INTO_OBJVAR_2 )
ON_INST ( BCODE_STORE_INTO_OBJVAR_3 )
ON_INST ( BCODE_POP_INTO_OBJVAR_0 )
ON_INST ( BCODE_POP_INTO_OBJVAR_1 )
ON_INST ( BCODE_POP_INTO_OBJVAR_2 )
ON_INST ( BCODE_POP_INTO_OBJVAR_3 )
{
moo_oop_oop_t t ;
2017-02-07 17:40:34 +00:00
2017-09-21 07:56:51 +00:00
/* b1 -> variable index to the object indicated by b2.
* b2 - > object index stored in the literal frame . */
b1 = bcode & 0x3 ; /* low 2 bits */
b2 = FETCH_BYTE_CODE ( moo ) ;
2017-02-07 17:40:34 +00:00
2017-09-21 07:56:51 +00:00
handle_objvar :
t = ( moo_oop_oop_t ) moo - > active_method - > slot [ b2 ] ;
MOO_ASSERT ( moo , MOO_OBJ_GET_FLAGS_TYPE ( t ) = = MOO_OBJ_TYPE_OOP ) ;
MOO_ASSERT ( moo , b1 < MOO_OBJ_GET_SIZE ( t ) ) ;
2017-02-07 17:40:34 +00:00
2017-09-21 07:56:51 +00:00
if ( ( bcode > > 3 ) & 1 )
2017-02-05 13:54:52 +00:00
{
2017-09-21 07:56:51 +00:00
/* store or pop */
t - > slot [ b1 ] = MOO_STACK_GETTOP ( moo ) ;
2017-02-05 13:54:52 +00:00
2017-09-21 07:56:51 +00:00
if ( ( bcode > > 2 ) & 1 )
{
/* pop */
MOO_STACK_POP ( moo ) ;
LOG_INST_2 ( moo , " pop_into_objvar %zu %zu " , b1 , b2 ) ;
}
else
{
LOG_INST_2 ( moo , " store_into_objvar %zu %zu " , b1 , b2 ) ;
}
2017-02-05 13:54:52 +00:00
}
2017-09-21 07:56:51 +00:00
else
2017-02-05 13:54:52 +00:00
{
2017-09-21 07:56:51 +00:00
/* push */
LOG_INST_2 ( moo , " push_objvar %zu %zu " , b1 , b2 ) ;
MOO_STACK_PUSH ( moo , t - > slot [ b1 ] ) ;
}
NEXT_INST ( ) ;
}
2017-02-05 13:54:52 +00:00
2017-09-21 07:56:51 +00:00
/* -------------------------------------------------------- */
ON_INST ( BCODE_SEND_MESSAGE_X )
ON_INST ( BCODE_SEND_MESSAGE_TO_SUPER_X )
/* b1 -> number of arguments
* b2 - > selector index stored in the literal frame */
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
FETCH_PARAM_CODE_TO ( moo , b2 ) ;
goto handle_send_message ;
ON_INST ( BCODE_SEND_MESSAGE_0 )
ON_INST ( BCODE_SEND_MESSAGE_1 )
ON_INST ( BCODE_SEND_MESSAGE_2 )
ON_INST ( BCODE_SEND_MESSAGE_3 )
ON_INST ( BCODE_SEND_MESSAGE_TO_SUPER_0 )
ON_INST ( BCODE_SEND_MESSAGE_TO_SUPER_1 )
ON_INST ( BCODE_SEND_MESSAGE_TO_SUPER_2 )
ON_INST ( BCODE_SEND_MESSAGE_TO_SUPER_3 )
{
moo_oop_char_t selector ;
2017-02-05 13:54:52 +00:00
2017-09-21 07:56:51 +00:00
b1 = bcode & 0x3 ; /* low 2 bits */
b2 = FETCH_BYTE_CODE ( moo ) ;
2017-02-05 13:54:52 +00:00
2017-09-21 07:56:51 +00:00
handle_send_message :
/* get the selector from the literal frame */
selector = ( moo_oop_char_t ) moo - > active_method - > slot [ b2 ] ;
2015-06-22 14:21:46 +00:00
2017-09-21 07:56:51 +00:00
LOG_INST_3 ( moo , " send_message%hs %zu @%zu " , ( ( ( bcode > > 2 ) & 1 ) ? " _to_super " : " " ) , b1 , b2 ) ;
if ( send_message ( moo , selector , ( ( bcode > > 2 ) & 1 ) , b1 ) < = - 1 ) return - 1 ;
NEXT_INST ( ) ;
}
2015-06-08 13:24:02 +00:00
2017-09-21 07:56:51 +00:00
/* -------------------------------------------------------- */
2016-06-20 15:42:51 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_PUSH_RECEIVER )
LOG_INST_0 ( moo , " push_receiver " ) ;
MOO_STACK_PUSH ( moo , moo - > active_context - > origin - > receiver_or_source ) ;
NEXT_INST ( ) ;
2016-02-15 18:07:20 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_PUSH_NIL )
LOG_INST_0 ( moo , " push_nil " ) ;
MOO_STACK_PUSH ( moo , moo - > _nil ) ;
NEXT_INST ( ) ;
2017-01-22 18:01:26 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_PUSH_TRUE )
LOG_INST_0 ( moo , " push_true " ) ;
MOO_STACK_PUSH ( moo , moo - > _true ) ;
NEXT_INST ( ) ;
2016-06-20 15:42:51 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_PUSH_FALSE )
LOG_INST_0 ( moo , " push_false " ) ;
MOO_STACK_PUSH ( moo , moo - > _false ) ;
NEXT_INST ( ) ;
2016-06-20 15:42:51 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_PUSH_CONTEXT )
LOG_INST_0 ( moo , " push_context " ) ;
MOO_STACK_PUSH ( moo , ( moo_oop_t ) moo - > active_context ) ;
NEXT_INST ( ) ;
2016-06-20 15:42:51 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_PUSH_PROCESS )
LOG_INST_0 ( moo , " push_process " ) ;
MOO_STACK_PUSH ( moo , ( moo_oop_t ) moo - > processor - > active ) ;
NEXT_INST ( ) ;
2016-06-22 03:23:14 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_PUSH_RECEIVER_NS )
{
register moo_oop_t c ;
LOG_INST_0 ( moo , " push_receiver_ns " ) ;
c = ( moo_oop_t ) MOO_CLASSOF ( moo , moo - > active_context - > origin - > receiver_or_source ) ;
if ( c = = ( moo_oop_t ) moo - > _class ) c = moo - > active_context - > origin - > receiver_or_source ;
MOO_STACK_PUSH ( moo , ( moo_oop_t ) ( ( moo_oop_class_t ) c ) - > nsup ) ;
NEXT_INST ( ) ;
}
2016-06-20 15:42:51 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_PUSH_NEGONE )
LOG_INST_0 ( moo , " push_negone " ) ;
MOO_STACK_PUSH ( moo , MOO_SMOOI_TO_OOP ( - 1 ) ) ;
NEXT_INST ( ) ;
ON_INST ( BCODE_PUSH_ZERO )
LOG_INST_0 ( moo , " push_zero " ) ;
MOO_STACK_PUSH ( moo , MOO_SMOOI_TO_OOP ( 0 ) ) ;
NEXT_INST ( ) ;
ON_INST ( BCODE_PUSH_ONE )
LOG_INST_0 ( moo , " push_one " ) ;
MOO_STACK_PUSH ( moo , MOO_SMOOI_TO_OOP ( 1 ) ) ;
NEXT_INST ( ) ;
ON_INST ( BCODE_PUSH_TWO )
LOG_INST_0 ( moo , " push_two " ) ;
MOO_STACK_PUSH ( moo , MOO_SMOOI_TO_OOP ( 2 ) ) ;
NEXT_INST ( ) ;
ON_INST ( BCODE_PUSH_INTLIT )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " push_intlit %zu " , b1 ) ;
MOO_STACK_PUSH ( moo , MOO_SMOOI_TO_OOP ( b1 ) ) ;
NEXT_INST ( ) ;
ON_INST ( BCODE_PUSH_NEGINTLIT )
{
moo_ooi_t num ;
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
num = b1 ;
LOG_INST_1 ( moo , " push_negintlit %zu " , b1 ) ;
MOO_STACK_PUSH ( moo , MOO_SMOOI_TO_OOP ( - num ) ) ;
NEXT_INST ( ) ;
}
2016-02-15 18:07:20 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_PUSH_CHARLIT )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " push_charlit %zu " , b1 ) ;
MOO_STACK_PUSH ( moo , MOO_CHAR_TO_OOP ( b1 ) ) ;
NEXT_INST ( ) ;
/* -------------------------------------------------------- */
ON_INST ( BCODE_MAKE_DICTIONARY )
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " make_dictionary %zu " , b1 ) ;
/* Dictionary new: b1
* doing this allows users to redefine Dictionary whatever way they like .
* if i did the followings instead , the internal of Dictionary would get
* tied to the system dictionary implementation . the system dictionary
* implementation is flawed in that it accepts only a variable character
* object as a key . it ' s better to invoke ' Dictionary new : . . . ' .
t = ( moo_oop_t ) moo_makedic ( moo , moo - > _dictionary , b1 + 10 ) ;
MOO_STACK_PUSH ( moo , t ) ;
*/
MOO_STACK_PUSH ( moo , ( moo_oop_t ) moo - > _dictionary ) ;
MOO_STACK_PUSH ( moo , MOO_SMOOI_TO_OOP ( b1 ) ) ;
if ( send_message ( moo , moo - > dicnewsym , 0 , 1 ) < = - 1 ) return - 1 ;
NEXT_INST ( ) ;
ON_INST ( BCODE_POP_INTO_DICTIONARY )
LOG_INST_0 ( moo , " pop_into_dictionary " ) ;
/* dic __put_assoc: assoc
* whether the system dictinoary implementation is flawed or not ,
* the code would look like this if it were used .
t1 = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_POP ( moo ) ;
t2 = MOO_STACK_GETTOP ( moo ) ;
moo_putatdic ( moo , ( moo_oop_dic_t ) t2 , ( ( moo_oop_association_t ) t1 ) - > key , ( ( moo_oop_association_t ) t1 ) - > value ) ;
*/
if ( send_message ( moo , moo - > dicputassocsym , 0 , 1 ) < = - 1 ) return - 1 ;
NEXT_INST ( ) ;
2017-01-25 13:57:14 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_MAKE_ARRAY )
{
moo_oop_t t ;
2015-06-08 13:24:02 +00:00
2017-09-21 07:56:51 +00:00
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " make_array %zu " , b1 ) ;
2016-06-20 15:42:51 +00:00
2017-09-21 07:56:51 +00:00
/* create an empty array */
t = moo_instantiate ( moo , moo - > _array , MOO_NULL , b1 ) ;
if ( ! t ) return - 1 ;
2015-06-20 03:07:11 +00:00
2017-09-21 07:56:51 +00:00
MOO_STACK_PUSH ( moo , t ) ; /* push the array created */
NEXT_INST ( ) ;
}
2016-06-20 15:42:51 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_POP_INTO_ARRAY )
{
moo_oop_t t1 , t2 ;
FETCH_PARAM_CODE_TO ( moo , b1 ) ;
LOG_INST_1 ( moo , " pop_into_array %zu " , b1 ) ;
t1 = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_POP ( moo ) ;
t2 = MOO_STACK_GETTOP ( moo ) ;
( ( moo_oop_oop_t ) t2 ) - > slot [ b1 ] = t1 ;
NEXT_INST ( ) ;
}
2016-05-15 15:51:41 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_DUP_STACKTOP )
{
moo_oop_t t ;
LOG_INST_0 ( moo , " dup_stacktop " ) ;
MOO_ASSERT ( moo , ! MOO_STACK_ISEMPTY ( moo ) ) ;
t = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_PUSH ( moo , t ) ;
NEXT_INST ( ) ;
}
2015-06-25 13:37:50 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_POP_STACKTOP )
LOG_INST_0 ( moo , " pop_stacktop " ) ;
MOO_ASSERT ( moo , ! MOO_STACK_ISEMPTY ( moo ) ) ;
MOO_STACK_POP ( moo ) ;
NEXT_INST ( ) ;
2017-01-22 18:01:26 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_RETURN_STACKTOP )
LOG_INST_0 ( moo , " return_stacktop " ) ;
return_value = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_POP ( moo ) ;
goto handle_return ;
2015-10-08 14:26:04 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_RETURN_RECEIVER )
LOG_INST_0 ( moo , " return_receiver " ) ;
return_value = moo - > active_context - > origin - > receiver_or_source ;
handle_return :
{
int n ;
if ( ( n = do_return ( moo , bcode , return_value ) ) < = - 1 ) return - 1 ;
if ( n = = 0 ) EXIT_DISPATCH_LOOP ( ) ;
}
NEXT_INST ( ) ;
2015-06-25 13:37:50 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_LOCAL_RETURN )
LOG_INST_0 ( moo , " local_return " ) ;
return_value = MOO_STACK_GETTOP ( moo ) ;
MOO_STACK_POP ( moo ) ;
goto handle_return ;
2016-06-05 18:01:35 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_RETURN_FROM_BLOCK )
do_return_from_block ( moo ) ;
NEXT_INST ( ) ;
2016-06-05 18:01:35 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_MAKE_BLOCK )
if ( make_block ( moo ) < = - 1 ) return - 1 ;
NEXT_INST ( ) ;
2015-10-19 06:16:43 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_SEND_BLOCK_COPY )
{
moo_ooi_t nargs , ntmprs ;
moo_oop_context_t rctx ;
moo_oop_context_t blkctx ;
2015-06-17 13:46:16 +00:00
2017-09-21 07:56:51 +00:00
LOG_INST_0 ( moo , " send_block_copy " ) ;
2015-07-03 13:06:01 +00:00
2017-09-21 07:56:51 +00:00
/* it emulates thisContext blockCopy: nargs ofTmprCount: ntmprs */
MOO_ASSERT ( moo , moo - > sp > = 2 ) ;
2015-06-17 13:46:16 +00:00
2017-09-21 07:56:51 +00:00
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , MOO_STACK_GETTOP ( moo ) ) = = moo - > _small_integer ) ;
ntmprs = MOO_OOP_TO_SMOOI ( MOO_STACK_GETTOP ( moo ) ) ;
MOO_STACK_POP ( moo ) ;
2015-06-28 14:20:37 +00:00
2017-09-21 07:56:51 +00:00
MOO_ASSERT ( moo , MOO_CLASSOF ( moo , MOO_STACK_GETTOP ( moo ) ) = = moo - > _small_integer ) ;
nargs = MOO_OOP_TO_SMOOI ( MOO_STACK_GETTOP ( moo ) ) ;
MOO_STACK_POP ( moo ) ;
2015-06-28 14:20:37 +00:00
2017-09-21 07:56:51 +00:00
MOO_ASSERT ( moo , nargs > = 0 ) ;
MOO_ASSERT ( moo , ntmprs > = nargs ) ;
/* the block context object created here is used
* as a base object for block context activation .
* pf_block_value ( ) clones a block
* context and activates the cloned context .
* this base block context is created with no
* stack for this reason . */
blkctx = ( moo_oop_context_t ) moo_instantiate ( moo , moo - > _block_context , MOO_NULL , 0 ) ;
if ( ! blkctx ) return - 1 ;
/* get the receiver to the block copy message after block context instantiation
* not to get affected by potential GC */
rctx = ( moo_oop_context_t ) MOO_STACK_GETTOP ( moo ) ;
MOO_ASSERT ( moo , rctx = = moo - > active_context ) ;
/* [NOTE]
* blkctx - > sender is left to nil . it is set to the
* active context before it gets activated . see
* pf_block_value ( ) .
*
* blkctx - > home is set here to the active context .
* it ' s redundant to have them pushed to the stack
* though it is to emulate the message sending of
* blockCopy : withNtmprs : . BCODE_MAKE_BLOCK has been
* added to replace BCODE_SEND_BLOCK_COPY and pusing
* arguments to the stack .
*
* blkctx - > origin is set here by copying the origin
* of the active context .
*/
/* the extended jump instruction has the format of
* 0000 XXXX KKKKKKKK or 0000 XXXX KKKKKKKK KKKKKKKK
* depending on MOO_BCODE_LONG_PARAM_SIZE . change ' ip ' to point to
* the instruction after the jump . */
blkctx - > ip = MOO_SMOOI_TO_OOP ( moo - > ip + MOO_BCODE_LONG_PARAM_SIZE + 1 ) ;
blkctx - > sp = MOO_SMOOI_TO_OOP ( - 1 ) ;
/* the number of arguments for a block context is local to the block */
blkctx - > method_or_nargs = MOO_SMOOI_TO_OOP ( nargs ) ;
/* the number of temporaries here is an accumulated count including
* the number of temporaries of a home context */
blkctx - > ntmprs = MOO_SMOOI_TO_OOP ( ntmprs ) ;
blkctx - > home = ( moo_oop_t ) rctx ;
blkctx - > receiver_or_source = moo - > _nil ;
/* [NOTE]
* the origin of a method context is set to itself
* when it ' s created . so it ' s safe to simply copy
* the origin field this way .
*
* if the context that receives the blockCopy message
* is a method context , the following conditions are all true .
* rctx - > home = = moo - > _nil
* MOO_CLASSOF ( moo , rctx ) = = moo - > _method_context
* rctx = = ( moo_oop_t ) moo - > active_context
* rctx = = rctx - > origin
*
* if it is a block context , the following condition is true .
* MOO_CLASSOF ( moo , rctx ) = = moo - > _block_context
*/
blkctx - > origin = rctx - > origin ;
2015-06-25 13:37:50 +00:00
2017-09-21 07:56:51 +00:00
MOO_STACK_SETTOP ( moo , ( moo_oop_t ) blkctx ) ;
NEXT_INST ( ) ;
2015-06-07 14:36:26 +00:00
}
2015-06-04 18:34:37 +00:00
2017-09-21 07:56:51 +00:00
ON_INST ( BCODE_NOOP )
/* do nothing */
LOG_INST_0 ( moo , " noop " ) ;
NEXT_INST ( ) ;
ON_UNKNOWN_INST ( )
MOO_LOG1 ( moo , MOO_LOG_IC | MOO_LOG_FATAL , " Fatal error - unknown byte code 0x%zx \n " , bcode ) ;
moo_seterrnum ( moo , MOO_EINTERN ) ;
return - 1 ;
END_DISPATCH_TABLE ( )
/* ==== END OF DISPATCH TABLE ==== */
END_DISPATCH_LOOP ( )
2017-07-20 16:33:53 +00:00
return 0 ;
}
int moo_execute ( moo_t * moo )
{
int n ;
# if defined(MOO_PROFILE_VM)
moo - > inst_counter = 0 ;
# endif
if ( vm_startup ( moo ) < = - 1 ) return - 1 ;
moo - > proc_switched = 0 ;
moo - > abort_req = 0 ;
n = __execute ( moo ) ;
2017-01-09 09:54:49 +00:00
vm_cleanup ( moo ) ;
2017-07-20 16:33:53 +00:00
2017-01-09 09:54:49 +00:00
# if defined(MOO_PROFILE_VM)
2017-07-20 16:33:53 +00:00
MOO_LOG1 ( moo , MOO_LOG_IC | MOO_LOG_INFO , " TOTAL_INST_COUTNER = %zu \n " , moo - > inst_counter ) ;
2015-10-08 14:26:04 +00:00
# endif
2015-06-11 09:11:18 +00:00
2017-07-20 16:33:53 +00:00
return n ;
2015-06-04 18:34:37 +00:00
}
2017-02-14 14:00:14 +00:00
void moo_abort ( moo_t * moo )
2017-02-12 18:59:03 +00:00
{
moo - > abort_req = 1 ;
}
2017-01-09 09:54:49 +00:00
int moo_invoke ( moo_t * moo , const moo_oocs_t * objname , const moo_oocs_t * mthname )
2015-06-04 18:34:37 +00:00
{
2016-05-15 15:51:41 +00:00
int n ;
2017-01-09 09:54:49 +00:00
MOO_ASSERT ( moo , moo - > initial_context = = MOO_NULL ) ;
MOO_ASSERT ( moo , moo - > active_context = = MOO_NULL ) ;
MOO_ASSERT ( moo , moo - > active_method = = MOO_NULL ) ;
2016-05-15 15:51:41 +00:00
2017-01-09 09:54:49 +00:00
if ( start_initial_process_and_context ( moo , objname , mthname ) < = - 1 ) return - 1 ;
moo - > initial_context = moo - > processor - > active - > initial_context ;
2016-05-15 15:51:41 +00:00
2017-01-09 09:54:49 +00:00
n = moo_execute ( moo ) ;
2016-05-15 15:51:41 +00:00
/* TODO: reset processor fields. set processor->tally to zero. processor->active to nil_process... */
2017-01-09 09:54:49 +00:00
moo - > initial_context = MOO_NULL ;
moo - > active_context = MOO_NULL ;
moo - > active_method = MOO_NULL ;
2016-05-15 15:51:41 +00:00
return n ;
2015-06-04 18:34:37 +00:00
}
2015-10-08 14:26:04 +00:00
2017-02-11 05:48:30 +00:00
#if 0
int moo_invoke ( moo_t * moo , const moo_oocs_t * objname , const moo_oocs_t * mthname )
{
/* TODO: .... */
/* call
* System initializeClasses
* and invoke
* objname mthname
*/
}
# endif