Another inconclusive session: still nothing works, still making progress.
This commit is contained in:
parent
ef59563e25
commit
eed4711fee
35 changed files with 317 additions and 232 deletions
|
|
@ -1,5 +1,19 @@
|
||||||
# State of Play
|
# State of Play
|
||||||
|
|
||||||
|
## 20260421
|
||||||
|
|
||||||
|
### To have `c_` functions or not to have `c_` functions?
|
||||||
|
|
||||||
|
Up to now I've had a conscious design pattern of having C functions with names beginning with `c_` which were 'the simplest possible way of solving the problem in C', and C functions with names beginning `lisp_` which were (usually) wrappers around those functions designed to be callable from Lisp. The current current refactoring exercise — and the `0.1.0` design doctrine that I should only code in C things which are absolutely necessary to bootstrap the Lisp compiler — is calling into question the need for many of the `c_` functions. After all, the `lisp_` functions are callable from C, it's just a little more prolix.
|
||||||
|
|
||||||
|
However, there is an overhead to calling a `lisp_` function: you have to generate a new stack frame, and there is a overhead, and consequently a time penalty. It may be in the long term it will be worth reviving `c_` functions for performance optimisation; but I think the priority for `0.1.X` is functionality, not performance.
|
||||||
|
|
||||||
|
### Type checking stack frames
|
||||||
|
|
||||||
|
Passing everything around as `pso_pointers` bypasses even C's rather lax type safety. Of course this doesn't matter for code written in Lisp, because it is the compiler's responsibility to mechanically make sure that **only** stack frames are passed into functions as stack frames. But if something else was passed in as a stack frame, the results probable wouldn't be pretty, and at least while I'm mostly running boostrap functions written in C, there is a risk.
|
||||||
|
|
||||||
|
Type checking the stack frame every time a function is entered is an overhead that will grow big quickly. I'm inclined to not do it in production. But I think it's essential to do it during debugging. proposal [here]().
|
||||||
|
|
||||||
## 20260420
|
## 20260420
|
||||||
|
|
||||||
Still on side projects, but those side-projects are giving me thinking time;
|
Still on side projects, but those side-projects are giving me thinking time;
|
||||||
|
|
|
||||||
|
|
@ -41,10 +41,11 @@ bool environment_initialised = false;
|
||||||
|
|
||||||
struct pso_pointer initialise_environment( uint32_t node ) {
|
struct pso_pointer initialise_environment( uint32_t node ) {
|
||||||
struct pso_pointer result = initialise_memory( node );
|
struct pso_pointer result = initialise_memory( node );
|
||||||
|
struct pso_pointer frame = make_frame(0, nil);
|
||||||
|
|
||||||
if ( truep( result ) ) {
|
if ( c_truep( result ) ) {
|
||||||
debug_print( L"Initialising `nil`... ", DEBUG_BOOTSTRAP, 0 );
|
debug_print( L"Initialising `nil`... ", DEBUG_BOOTSTRAP, 0 );
|
||||||
struct pso_pointer n = allocate( NILTAG, 2 );
|
struct pso_pointer n = allocate( frame, NILTAG, 2 );
|
||||||
|
|
||||||
if ( ( n.page == 0 ) && ( n.offset == 0 ) ) {
|
if ( ( n.page == 0 ) && ( n.offset == 0 ) ) {
|
||||||
struct pso2 *object = pointer_to_object( n );
|
struct pso2 *object = pointer_to_object( n );
|
||||||
|
|
@ -55,16 +56,13 @@ struct pso_pointer initialise_environment( uint32_t node ) {
|
||||||
lock_object( nil );
|
lock_object( nil );
|
||||||
debug_print( L"success\n", DEBUG_BOOTSTRAP, 0 );
|
debug_print( L"success\n", DEBUG_BOOTSTRAP, 0 );
|
||||||
} else {
|
} else {
|
||||||
result =
|
result = nil;
|
||||||
make_exception( c_string_to_lisp_string
|
|
||||||
( L"Unexpected cell while allocating `nil`." ),
|
|
||||||
nil, nil, n );
|
|
||||||
debug_print( L"fail\n", DEBUG_BOOTSTRAP, 0 );
|
debug_print( L"fail\n", DEBUG_BOOTSTRAP, 0 );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ( !exceptionp( result ) ) {
|
if ( !c_nilp( result ) ) {
|
||||||
debug_print( L"Initialising `t`... ", DEBUG_BOOTSTRAP, 0 );
|
debug_print( L"Initialising `t`... ", DEBUG_BOOTSTRAP, 0 );
|
||||||
struct pso_pointer n = allocate( TRUETAG, 2 );
|
struct pso_pointer n = allocate( frame, TRUETAG, 2 );
|
||||||
|
|
||||||
// offset is in words, and size of a pso2 is four words
|
// offset is in words, and size of a pso2 is four words
|
||||||
if ( ( n.page == 0 ) && ( n.offset == 4 ) ) {
|
if ( ( n.page == 0 ) && ( n.offset == 4 ) ) {
|
||||||
|
|
@ -76,19 +74,16 @@ struct pso_pointer initialise_environment( uint32_t node ) {
|
||||||
lock_object( t );
|
lock_object( t );
|
||||||
debug_print( L"success\n", DEBUG_BOOTSTRAP, 0 );
|
debug_print( L"success\n", DEBUG_BOOTSTRAP, 0 );
|
||||||
} else {
|
} else {
|
||||||
result =
|
result = nil;
|
||||||
make_exception( c_string_to_lisp_string
|
|
||||||
( L"Unexpected cell while allocating `t`." ),
|
|
||||||
nil, nil, n );
|
|
||||||
debug_print( L"fail\n", DEBUG_BOOTSTRAP, 0 );
|
debug_print( L"fail\n", DEBUG_BOOTSTRAP, 0 );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ( !exceptionp( result ) ) {
|
if ( !exceptionp( result ) ) {
|
||||||
result = c_bind( c_string_to_lisp_symbol( L"nil" ), nil, nil );
|
result = c_bind( c_string_to_lisp_symbol( frame, L"nil" ), nil, nil );
|
||||||
debug_print( L"Environment after binding `nil`: ", DEBUG_BOOTSTRAP,
|
debug_print( L"Environment after binding `nil`: ", DEBUG_BOOTSTRAP,
|
||||||
0 );
|
0 );
|
||||||
debug_print_object( result, DEBUG_BOOTSTRAP, 0 );
|
debug_print_object( result, DEBUG_BOOTSTRAP, 0 );
|
||||||
result = c_bind( c_string_to_lisp_symbol( L"t" ), t, result );
|
result = c_bind( c_string_to_lisp_symbol( frame, L"t" ), t, result );
|
||||||
|
|
||||||
environment_initialised = true;
|
environment_initialised = true;
|
||||||
debug_print( L"Environment after binding `t`: ", DEBUG_BOOTSTRAP, 0 );
|
debug_print( L"Environment after binding `t`: ", DEBUG_BOOTSTRAP, 0 );
|
||||||
|
|
|
||||||
|
|
@ -436,7 +436,7 @@ struct pso_pointer add_meta_string( struct pso_pointer meta, char32_t *key,
|
||||||
|
|
||||||
return
|
return
|
||||||
make_cons( make_cons
|
make_cons( make_cons
|
||||||
( c_string_to_lisp_keyword( key ),
|
( c_string_to_lisp_keyword( frame_pointer, key ),
|
||||||
c_string_to_lisp_string( buffer ) ), meta );
|
c_string_to_lisp_string( buffer ) ), meta );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -681,7 +681,7 @@ struct pso_pointer lisp_read_char( struct pso_pointer frame_pointer,
|
||||||
struct pso_pointer stream_pointer = fetch_arg( frame, 0 );
|
struct pso_pointer stream_pointer = fetch_arg( frame, 0 );
|
||||||
if ( readp( stream_pointer ) ) {
|
if ( readp( stream_pointer ) ) {
|
||||||
result =
|
result =
|
||||||
make_string( url_fgetwc( stream_get_url_file( stream_pointer ) ),
|
make_string( frame_pointer, url_fgetwc( stream_get_url_file( stream_pointer ) ),
|
||||||
nil );
|
nil );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -709,7 +709,7 @@ struct pso_pointer lisp_slurp( struct pso_pointer frame_pointer,
|
||||||
|
|
||||||
if ( readp( fetch_arg( frame, 0 ) ) ) {
|
if ( readp( fetch_arg( frame, 0 ) ) ) {
|
||||||
URL_FILE *stream = stream_get_url_file( fetch_arg( frame, 0 ) );
|
URL_FILE *stream = stream_get_url_file( fetch_arg( frame, 0 ) );
|
||||||
struct pso_pointer cursor = make_string( url_fgetwc( stream ), nil );
|
struct pso_pointer cursor = make_string( frame_pointer, url_fgetwc( stream ), nil );
|
||||||
result = cursor;
|
result = cursor;
|
||||||
|
|
||||||
for ( wint_t c = url_fgetwc( stream ); !url_feof( stream ) && c != 0;
|
for ( wint_t c = url_fgetwc( stream ); !url_feof( stream ) && c != 0;
|
||||||
|
|
@ -721,7 +721,7 @@ struct pso_pointer lisp_slurp( struct pso_pointer frame_pointer,
|
||||||
debug_println( DEBUG_IO );
|
debug_println( DEBUG_IO );
|
||||||
|
|
||||||
struct pso2 *cell = pointer_to_object( cursor );
|
struct pso2 *cell = pointer_to_object( cursor );
|
||||||
cursor = make_string( ( char32_t ) c, nil );
|
cursor = make_string( frame_pointer, ( char32_t ) c, nil );
|
||||||
cell->payload.string.cdr = cursor;
|
cell->payload.string.cdr = cursor;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -38,7 +38,6 @@ bool node_initialised = false;
|
||||||
*/
|
*/
|
||||||
uint32_t node_index = 0;
|
uint32_t node_index = 0;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief The canonical `nil` pointer
|
* @brief The canonical `nil` pointer
|
||||||
*
|
*
|
||||||
|
|
@ -52,6 +51,16 @@ struct pso_pointer nil = ( struct pso_pointer ) { 0, 0, 0 };
|
||||||
*/
|
*/
|
||||||
struct pso_pointer t = ( struct pso_pointer ) { 0, 0, 4 };
|
struct pso_pointer t = ( struct pso_pointer ) { 0, 0, 4 };
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief whether this node is in debugging mode or not.
|
||||||
|
*/
|
||||||
|
struct pso_pointer in_debugging_mode =
|
||||||
|
#ifdef DEBUG
|
||||||
|
( struct pso_pointer ) { 0, 0, 4 };
|
||||||
|
#else
|
||||||
|
( struct pso_pointer ) { 0, 0, 0 };
|
||||||
|
#endif
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief The root of the data space.
|
* @brief The root of the data space.
|
||||||
*/
|
*/
|
||||||
|
|
@ -62,23 +71,24 @@ struct pso_pointer oblist = ( struct pso_pointer ) { 0, 0, 0 };
|
||||||
* @brief Set up the basic informetion about this node.
|
* @brief Set up the basic informetion about this node.
|
||||||
*
|
*
|
||||||
* @param index
|
* @param index
|
||||||
* @return struct pso_pointer
|
* @return struct pso_pointer the environment created during initialisation.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer initialise_node( uint32_t index ) {
|
struct pso_pointer initialise_node( uint32_t index ) {
|
||||||
node_index = index;
|
node_index = index;
|
||||||
|
|
||||||
struct pso_pointer result = initialise_environment( index );
|
struct pso_pointer result = initialise_environment( index );
|
||||||
|
|
||||||
if ( !nilp( result ) && !exceptionp( result ) ) {
|
if ( !c_nilp( result ) && !exceptionp( result ) ) {
|
||||||
|
node_initialised = true;
|
||||||
if ( initialise_io( ) == 0 ) {
|
if ( initialise_io( ) == 0 ) {
|
||||||
result = initialise_default_streams( result );
|
result = initialise_default_streams( result );
|
||||||
} else {
|
} else {
|
||||||
result =
|
result =
|
||||||
make_exception( c_string_to_lisp_string
|
make_exception( make_frame(1, nil,
|
||||||
( L"Failed to initialise default streams" ),
|
c_string_to_lisp_string( nil, L"Failed to initialise default streams" )));
|
||||||
nil, nil, nil );
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -11,6 +11,7 @@
|
||||||
#ifndef __psse_memory_node_h
|
#ifndef __psse_memory_node_h
|
||||||
#define __psse_memory_node_h
|
#define __psse_memory_node_h
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -19,6 +20,8 @@
|
||||||
*/
|
*/
|
||||||
extern uint32_t node_index;
|
extern uint32_t node_index;
|
||||||
|
|
||||||
|
extern bool node_initialised;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief The canonical `nil` pointer
|
* @brief The canonical `nil` pointer
|
||||||
*
|
*
|
||||||
|
|
@ -27,10 +30,11 @@ extern struct pso_pointer nil;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief the canonical `t` (true) pointer.
|
* @brief the canonical `t` (true) pointer.
|
||||||
*
|
|
||||||
*/
|
*/
|
||||||
extern struct pso_pointer t;
|
extern struct pso_pointer t;
|
||||||
|
|
||||||
|
extern struct pso_pointer in_debugging_mode;
|
||||||
|
|
||||||
extern struct pso_pointer oblist;
|
extern struct pso_pointer oblist;
|
||||||
|
|
||||||
struct pso_pointer initialise_node( int node_index );
|
struct pso_pointer initialise_node( int node_index );
|
||||||
|
|
|
||||||
|
|
@ -30,6 +30,7 @@
|
||||||
#include "memory/page.h"
|
#include "memory/page.h"
|
||||||
#include "memory/pointer.h"
|
#include "memory/pointer.h"
|
||||||
#include "memory/pso.h"
|
#include "memory/pso.h"
|
||||||
|
#include "memory/pso4.h"
|
||||||
#include "memory/tags.h"
|
#include "memory/tags.h"
|
||||||
|
|
||||||
#include "ops/truth.h"
|
#include "ops/truth.h"
|
||||||
|
|
@ -46,17 +47,13 @@
|
||||||
* possible due to infinite recursion, but those special cases need to be
|
* possible due to infinite recursion, but those special cases need to be
|
||||||
* audited carefully.
|
* audited carefully.
|
||||||
*
|
*
|
||||||
* The stack frame pointer is DELIBERATELY a C pointer, not a Lisp pointer,
|
* @param frame_pointer pointer to an active stack frame (or
|
||||||
* because you are definitely not supposed to be calling this function from
|
* nil, but only during initialisation).
|
||||||
* Lisp. Please do not!
|
|
||||||
*
|
|
||||||
* @param stack_pointer C (NOT Lisp!) pointer to an active stack frame (or
|
|
||||||
* NULL, but only during initialisation).
|
|
||||||
* @param tag The tag. Only the first three bytes will be used;
|
* @param tag The tag. Only the first three bytes will be used;
|
||||||
* @param size_class The size class for the object to be allocated;
|
* @param size_class The size class for the object to be allocated;
|
||||||
* @return struct pso_pointer a pointer to the newly allocated object
|
* @return struct pso_pointer a pointer to the newly allocated object
|
||||||
*/
|
*/
|
||||||
struct pso_pointer allocate( struct pso4 *stack_pointer, char *tag,
|
struct pso_pointer allocate( struct pso_pointer frame_pointer, char *tag,
|
||||||
uint8_t size_class ) {
|
uint8_t size_class ) {
|
||||||
// todo: issue #21: must have stack frame passed in.
|
// todo: issue #21: must have stack frame passed in.
|
||||||
|
|
||||||
|
|
@ -67,19 +64,19 @@ struct pso_pointer allocate( struct pso4 *stack_pointer, char *tag,
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
struct pso_pointer result = pop_freelist( size_class );
|
struct pso_pointer result = pop_freelist( size_class );
|
||||||
|
struct pso4* frame = pointer_to_pso4(frame_pointer);
|
||||||
|
|
||||||
if ( !nilp( result ) ) {
|
if ( !c_nilp( result ) ) {
|
||||||
strncpy( ( char * ) ( pointer_to_object( result )->header.tag.
|
strncpy( ( char * ) ( pointer_to_object( result )->header.tag.
|
||||||
bytes.mnemonic ), tag, TAGLENGTH );
|
bytes.mnemonic ), tag, TAGLENGTH );
|
||||||
|
|
||||||
debug_printf( DEBUG_ALLOC, 0, L"at page %d, offset %d... ",
|
debug_printf( DEBUG_ALLOC, 0, L"at page %d, offset %d... ",
|
||||||
result.page, result.offset );
|
result.page, result.offset );
|
||||||
if ( stack_pointer != NULL &&
|
if ( stackp(frame_pointer)) {
|
||||||
( stack_pointer->header.tag.value & 0xffffff ) == STACKTV ) {
|
|
||||||
struct pso_pointer locals = make_cons( result,
|
struct pso_pointer locals = make_cons( result,
|
||||||
stack_pointer->payload.
|
frame->payload.
|
||||||
stack_frame.locals );
|
stack_frame.locals );
|
||||||
stack_pointer->payload.stack_frame.locals = locals;
|
frame->payload.stack_frame.locals = locals;
|
||||||
|
|
||||||
} else if ( memory_initialised ) {
|
} else if ( memory_initialised ) {
|
||||||
fputws( L"WARNING: No stack frame passed to `allocate`.\n",
|
fputws( L"WARNING: No stack frame passed to `allocate`.\n",
|
||||||
|
|
@ -151,7 +148,7 @@ struct pso_pointer inc_ref( struct pso_pointer pointer ) {
|
||||||
struct pso_pointer dec_ref( struct pso_pointer pointer ) {
|
struct pso_pointer dec_ref( struct pso_pointer pointer ) {
|
||||||
struct pso2 *object = pointer_to_object( pointer );
|
struct pso2 *object = pointer_to_object( pointer );
|
||||||
|
|
||||||
if ( !nilp( pointer ) && object->header.count > 0
|
if ( !c_nilp( pointer ) && object->header.count > 0
|
||||||
&& object->header.count != MAXREFERENCE ) {
|
&& object->header.count != MAXREFERENCE ) {
|
||||||
object->header.count--;
|
object->header.count--;
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
|
|
||||||
|
|
@ -16,7 +16,7 @@
|
||||||
#include "memory/pointer.h"
|
#include "memory/pointer.h"
|
||||||
#include "memory/pso4.h"
|
#include "memory/pso4.h"
|
||||||
|
|
||||||
struct pso_pointer allocate( struct pso4 *stack_pointer, char *tag,
|
struct pso_pointer allocate( struct pso_pointer frame_pointer, char *tag,
|
||||||
uint8_t size_class );
|
uint8_t size_class );
|
||||||
|
|
||||||
struct pso_pointer dec_ref( struct pso_pointer pointer );
|
struct pso_pointer dec_ref( struct pso_pointer pointer );
|
||||||
|
|
|
||||||
|
|
@ -12,7 +12,4 @@
|
||||||
#include "memory/pso2.h"
|
#include "memory/pso2.h"
|
||||||
#include "memory/pso4.h"
|
#include "memory/pso4.h"
|
||||||
|
|
||||||
struct pso4 *pointer_to_pso4( struct pso_pointer p ) {
|
struct pso4 *pointer_to_pso4( struct pso_pointer p );
|
||||||
struct pso4 *result =
|
|
||||||
( struct pso4 * ) pointer_to_object_of_size_class( p, 4 );
|
|
||||||
}
|
|
||||||
|
|
|
||||||
|
|
@ -34,6 +34,7 @@
|
||||||
#define NAMESPACETAG "NSP"
|
#define NAMESPACETAG "NSP"
|
||||||
#define NILTAG "NIL"
|
#define NILTAG "NIL"
|
||||||
#define NLAMBDATAG "NLM"
|
#define NLAMBDATAG "NLM"
|
||||||
|
#define PACKSTRTAG "PST"
|
||||||
#define RATIOTAG "RAT"
|
#define RATIOTAG "RAT"
|
||||||
#define READTAG "RED"
|
#define READTAG "RED"
|
||||||
#define REALTAG "REA"
|
#define REALTAG "REA"
|
||||||
|
|
@ -61,6 +62,7 @@
|
||||||
#define NAMESPACETV 5264206
|
#define NAMESPACETV 5264206
|
||||||
#define NILTV 4999502
|
#define NILTV 4999502
|
||||||
#define NLAMBDATV 5065806
|
#define NLAMBDATV 5065806
|
||||||
|
#define PACKSTRTV 5526352
|
||||||
#define RATIOTV 5521746
|
#define RATIOTV 5521746
|
||||||
#define READTV 4474194
|
#define READTV 4474194
|
||||||
#define REALTV 4277586
|
#define REALTV 4277586
|
||||||
|
|
|
||||||
|
|
@ -89,7 +89,7 @@ struct pso_pointer c_interned( struct pso_pointer key,
|
||||||
* @return `true` if a pointer the key was found in the store..
|
* @return `true` if a pointer the key was found in the store..
|
||||||
*/
|
*/
|
||||||
bool c_internedp( struct pso_pointer key, struct pso_pointer store ) {
|
bool c_internedp( struct pso_pointer key, struct pso_pointer store ) {
|
||||||
return !nilp( search( key, store, true ) );
|
return !c_nilp( search( key, store, true ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -149,5 +149,5 @@ struct pso_pointer internedp(
|
||||||
struct pso_pointer store = or( make_frame( 2, frame_pointer,
|
struct pso_pointer store = or( make_frame( 2, frame_pointer,
|
||||||
fetch_arg( frame, 1 ), frame->payload.stack_frame.env));
|
fetch_arg( frame, 1 ), frame->payload.stack_frame.env));
|
||||||
|
|
||||||
return c_internedp( key, store );
|
return c_internedp( key, store ) ? t : nil;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -66,7 +66,7 @@ bool c_equal( struct pso_pointer a, struct pso_pointer b ) {
|
||||||
case KEYTV:
|
case KEYTV:
|
||||||
case STRINGTV:
|
case STRINGTV:
|
||||||
case SYMBOLTV:
|
case SYMBOLTV:
|
||||||
while ( result && !nilp( a ) && !nilp( b ) ) {
|
while ( result && !c_nilp( a ) && !c_nilp( b ) ) {
|
||||||
if ( pointer_to_object( a )->payload.string.character ==
|
if ( pointer_to_object( a )->payload.string.character ==
|
||||||
pointer_to_object( b )->payload.string.character ) {
|
pointer_to_object( b )->payload.string.character ) {
|
||||||
a = c_cdr( a );
|
a = c_cdr( a );
|
||||||
|
|
@ -75,7 +75,7 @@ bool c_equal( struct pso_pointer a, struct pso_pointer b ) {
|
||||||
result = false;
|
result = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
result = result && nilp( a ) && nilp( b );
|
result = result && c_nilp( a ) && c_nilp( b );
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
result = false;
|
result = false;
|
||||||
|
|
@ -109,7 +109,7 @@ struct pso_pointer eq(
|
||||||
|
|
||||||
if ( frame->payload.stack_frame.args > 1 ) {
|
if ( frame->payload.stack_frame.args > 1 ) {
|
||||||
for ( int b = 1;
|
for ( int b = 1;
|
||||||
( truep( result ) ) && ( b < frame->payload.stack_frame.args );
|
( c_truep( result ) ) && ( b < frame->payload.stack_frame.args );
|
||||||
b++ ) {
|
b++ ) {
|
||||||
result =
|
result =
|
||||||
c_eq( fetch_arg( frame, 0 ), fetch_arg( frame, b ) ) ? t : nil;
|
c_eq( fetch_arg( frame, 0 ), fetch_arg( frame, b ) ) ? t : nil;
|
||||||
|
|
|
||||||
|
|
@ -57,9 +57,10 @@ struct pso_pointer eval(
|
||||||
#ifdef MANAGED_POINTER_ONLY
|
#ifdef MANAGED_POINTER_ONLY
|
||||||
struct pso4 *frame = pointer_to_pso4( frame_pointer );
|
struct pso4 *frame = pointer_to_pso4( frame_pointer );
|
||||||
#endif
|
#endif
|
||||||
struct pso_pointer result = fetch_arg( frame, 0 );
|
struct pso_pointer arg = fetch_arg( frame, 0 );
|
||||||
|
struct pso_pointer result = nil;
|
||||||
|
|
||||||
switch ( get_tag_value( result ) ) {
|
switch ( get_tag_value( arg ) ) {
|
||||||
// case CONSTV:
|
// case CONSTV:
|
||||||
// result = eval_cons( frame, frame_pointer, env);
|
// result = eval_cons( frame, frame_pointer, env);
|
||||||
// break;
|
// break;
|
||||||
|
|
@ -67,9 +68,10 @@ struct pso_pointer eval(
|
||||||
case KEYTV:
|
case KEYTV:
|
||||||
case STRINGTV:
|
case STRINGTV:
|
||||||
// self evaluating
|
// self evaluating
|
||||||
|
result = nil;
|
||||||
break;
|
break;
|
||||||
case SYMBOLTV:
|
case SYMBOLTV:
|
||||||
result = c_assoc( result, fetch_env(frame_pointer) );
|
arg = c_assoc( arg, fetch_env(frame_pointer) );
|
||||||
break;
|
break;
|
||||||
// case LAMBDATV:
|
// case LAMBDATV:
|
||||||
// result = eval_lambda( frame, frame_pointer, env);
|
// result = eval_lambda( frame, frame_pointer, env);
|
||||||
|
|
@ -81,34 +83,34 @@ struct pso_pointer eval(
|
||||||
// result = eval_special( frame, frame_pointer, env);
|
// result = eval_special( frame, frame_pointer, env);
|
||||||
// break;
|
// break;
|
||||||
default:
|
default:
|
||||||
result =
|
arg =
|
||||||
make_exception( make_cons
|
make_exception(
|
||||||
( frame, c_string_to_lisp_string
|
make_frame(1, frame_pointer,
|
||||||
( frame,
|
make_cons( frame_pointer,
|
||||||
|
c_string_to_lisp_string( frame_pointer,
|
||||||
L"Can't yet evaluate things of this type: " ),
|
L"Can't yet evaluate things of this type: " ),
|
||||||
result ), frame_pointer, make_cons( frame,
|
arg ),
|
||||||
|
make_cons( frame_pointer,
|
||||||
make_cons
|
make_cons
|
||||||
( frame,
|
( frame_pointer,
|
||||||
c_string_to_lisp_keyword
|
c_string_to_lisp_keyword
|
||||||
( frame,
|
( frame_pointer,
|
||||||
L"tag" ),
|
L"tag" ),
|
||||||
get_tag_string
|
get_tag_string
|
||||||
( result ) ),
|
( arg ) ),
|
||||||
nil ),
|
nil ),
|
||||||
nil );
|
nil ));
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( exceptionp( result ) ) {
|
if ( exceptionp( arg ) ) {
|
||||||
struct pso3 *x =
|
struct pso3 *x =
|
||||||
( struct pso3 * ) pointer_to_object_with_tag_value( result,
|
( struct pso3 * ) pointer_to_object_with_tag_value( arg,
|
||||||
EXCEPTIONTV );
|
EXCEPTIONTV );
|
||||||
|
|
||||||
if ( nilp( x->payload.exception.stack ) ) {
|
if ( c_nilp( x->payload.exception.stack ) ) {
|
||||||
result =
|
|
||||||
make_exception( x->payload.exception.message, frame_pointer,
|
|
||||||
nil, result );
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return arg;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -12,18 +12,21 @@
|
||||||
#include "memory/pso2.h"
|
#include "memory/pso2.h"
|
||||||
#include "memory/pso4.h"
|
#include "memory/pso4.h"
|
||||||
|
|
||||||
|
#include "ops/stack_ops.h"
|
||||||
#include "payloads/stack.h"
|
#include "payloads/stack.h"
|
||||||
|
|
||||||
#include "ops/truth.h"
|
#include "ops/truth.h"
|
||||||
|
|
||||||
struct pso_pointer length( struct pso_pointer frame_pointer) {
|
struct pso_pointer length( struct pso_pointer frame_pointer) {
|
||||||
struct pso_pointer list = fetch_arg( frame_pointer, 0);
|
struct pso4* frame = pointer_to_pso4(frame_pointer);
|
||||||
|
|
||||||
|
struct pso_pointer list = fetch_arg( frame, 0);
|
||||||
int count = 0;
|
int count = 0;
|
||||||
|
|
||||||
for ( struct pso_pointer cursor = list; !nilp( cursor);
|
for ( struct pso_pointer cursor = list; !c_nilp( cursor);
|
||||||
cursor = cdr( make_frame( 1, frame_pointer, list))) {
|
cursor = cdr( make_frame( 1, frame_pointer, list))) {
|
||||||
count++;
|
count++;
|
||||||
}
|
}
|
||||||
|
|
||||||
return make_integer( pointer_to_pso4(frame_pointer), count);
|
return make_integer( frame_pointer, count);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -18,4 +18,5 @@
|
||||||
#include "payloads/function.h"
|
#include "payloads/function.h"
|
||||||
|
|
||||||
struct pso_pointer length( struct pso_pointer frame_pointer);
|
struct pso_pointer length( struct pso_pointer frame_pointer);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -54,7 +54,7 @@ void c_repl( bool show_prompt ) {
|
||||||
|
|
||||||
// TODO: NULL is not OK here, but will do until we have a REPL in Lisp.
|
// TODO: NULL is not OK here, but will do until we have a REPL in Lisp.
|
||||||
struct pso_pointer env =
|
struct pso_pointer env =
|
||||||
consp( oblist ) ? oblist : make_cons( NULL, oblist, nil );
|
consp( oblist ) ? oblist : make_cons( nil, oblist, nil );
|
||||||
struct pso_pointer input_stream = c_assoc( lisp_io_in, env );
|
struct pso_pointer input_stream = c_assoc( lisp_io_in, env );
|
||||||
struct pso_pointer output_stream = c_assoc( lisp_io_out, env );
|
struct pso_pointer output_stream = c_assoc( lisp_io_out, env );
|
||||||
|
|
||||||
|
|
@ -77,7 +77,7 @@ void c_repl( bool show_prompt ) {
|
||||||
/* bottom of stack */
|
/* bottom of stack */
|
||||||
struct pso_pointer frame_pointer = make_frame( 1, nil, input_stream );
|
struct pso_pointer frame_pointer = make_frame( 1, nil, input_stream );
|
||||||
|
|
||||||
if ( nilp( frame_pointer ) )
|
if ( c_nilp( frame_pointer ) )
|
||||||
break;
|
break;
|
||||||
struct pso_pointer input = read(
|
struct pso_pointer input = read(
|
||||||
#ifndef MANAGED_POINTER_ONLY
|
#ifndef MANAGED_POINTER_ONLY
|
||||||
|
|
@ -86,7 +86,7 @@ void c_repl( bool show_prompt ) {
|
||||||
frame_pointer, env );
|
frame_pointer, env );
|
||||||
|
|
||||||
frame_pointer = make_frame( 1, frame_pointer, input );
|
frame_pointer = make_frame( 1, frame_pointer, input );
|
||||||
if ( nilp( frame_pointer ) )
|
if ( c_nilp( frame_pointer ) )
|
||||||
break;
|
break;
|
||||||
|
|
||||||
struct pso_pointer result = eval(
|
struct pso_pointer result = eval(
|
||||||
|
|
|
||||||
|
|
@ -35,39 +35,41 @@
|
||||||
* @return a sequence like the `sequence` passed, but reversed; or `nil` if
|
* @return a sequence like the `sequence` passed, but reversed; or `nil` if
|
||||||
* the argument was not a sequence.
|
* the argument was not a sequence.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer c_reverse( struct pso_pointer sequence ) {
|
struct pso_pointer c_reverse( struct pso_pointer frame_pointer, struct pso_pointer sequence ) {
|
||||||
// todo: issue #21: must have stack frame passed in.
|
// todo: issue #21: must have stack frame passed in.
|
||||||
struct pso_pointer result = nil;
|
struct pso_pointer result = nil;
|
||||||
|
|
||||||
for ( struct pso_pointer cursor = sequence; !nilp( sequence );
|
for ( struct pso_pointer cursor = sequence; !c_nilp( sequence );
|
||||||
cursor = c_cdr( cursor ) ) {
|
cursor = c_cdr( cursor ) ) {
|
||||||
struct pso2 *object = pointer_to_object( cursor );
|
struct pso2 *object = pointer_to_object( cursor );
|
||||||
switch ( get_tag_value( cursor ) ) {
|
switch ( get_tag_value( cursor ) ) {
|
||||||
case CONSTV:
|
case CONSTV:
|
||||||
result = make_cons( c_car( cursor ), result );
|
result = make_cons( frame_pointer, c_car( cursor ), result );
|
||||||
break;
|
break;
|
||||||
case KEYTV:
|
case KEYTV:
|
||||||
// TODO: should you be able to reverse keywords and symbols?
|
// TODO: should you be able to reverse keywords and symbols?
|
||||||
result =
|
result =
|
||||||
make_string_like_thing( object->payload.string.character,
|
make_string_like_thing( frame_pointer, object->payload.string.character,
|
||||||
result, KEYTAG );
|
result, KEYTAG );
|
||||||
break;
|
break;
|
||||||
case STRINGTV:
|
case STRINGTV:
|
||||||
result =
|
result =
|
||||||
make_string_like_thing( object->payload.string.character,
|
make_string_like_thing( frame_pointer, object->payload.string.character,
|
||||||
result, STRINGTAG );
|
result, STRINGTAG );
|
||||||
break;
|
break;
|
||||||
case SYMBOLTV:
|
case SYMBOLTV:
|
||||||
// TODO: should you be able to reverse keywords and symbols?
|
// TODO: should you be able to reverse keywords and symbols?
|
||||||
result =
|
result =
|
||||||
make_string_like_thing( object->payload.string.character,
|
make_string_like_thing( frame_pointer, object->payload.string.character,
|
||||||
result, SYMBOLTAG );
|
result, SYMBOLTAG );
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
result =
|
result =
|
||||||
make_exception( make_cons( c_string_to_lisp_string
|
make_exception( make_frame( 1, frame_pointer,
|
||||||
( L"Invalid object in sequence" ),
|
make_cons( frame_pointer,
|
||||||
cursor ), nil, nil, nil );
|
c_string_to_lisp_string
|
||||||
|
( frame_pointer, L"Invalid object in sequence" ),
|
||||||
|
cursor ) ));
|
||||||
goto exit;
|
goto exit;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -46,7 +46,7 @@ uint32_t calculate_hash( wint_t c, struct pso_pointer ptr ) {
|
||||||
case KEYTV:
|
case KEYTV:
|
||||||
case STRINGTV:
|
case STRINGTV:
|
||||||
case SYMBOLTV:
|
case SYMBOLTV:
|
||||||
if ( nilp( cell->payload.string.cdr ) ) {
|
if ( c_nilp( cell->payload.string.cdr ) ) {
|
||||||
result = ( uint32_t ) c;
|
result = ( uint32_t ) c;
|
||||||
} else {
|
} else {
|
||||||
result =
|
result =
|
||||||
|
|
@ -69,12 +69,12 @@ uint32_t calculate_hash( wint_t c, struct pso_pointer ptr ) {
|
||||||
* char32_t in larger pso classes, so this function may be only for strings
|
* char32_t in larger pso classes, so this function may be only for strings
|
||||||
* (and thus simpler).
|
* (and thus simpler).
|
||||||
*/
|
*/
|
||||||
struct pso_pointer make_string_like_thing( struct pso4 *frame_pointer,
|
struct pso_pointer make_string_like_thing( struct pso_pointer frame_pointer,
|
||||||
wint_t c, struct pso_pointer tail,
|
wint_t c, struct pso_pointer tail,
|
||||||
char *tag ) {
|
char *tag ) {
|
||||||
struct pso_pointer pointer = tail;
|
struct pso_pointer pointer = tail;
|
||||||
|
|
||||||
if ( check_type( tail, tag ) || nilp( tail ) ) {
|
if ( check_type( tail, tag ) || c_nilp( tail ) ) {
|
||||||
pointer = allocate( frame_pointer, tag, CONS_SIZE_CLASS );
|
pointer = allocate( frame_pointer, tag, CONS_SIZE_CLASS );
|
||||||
struct pso2 *cell = pointer_to_object( pointer );
|
struct pso2 *cell = pointer_to_object( pointer );
|
||||||
|
|
||||||
|
|
@ -107,7 +107,7 @@ struct pso_pointer make_string_like_thing( struct pso4 *frame_pointer,
|
||||||
* @param c the character to add (prepend);
|
* @param c the character to add (prepend);
|
||||||
* @param tail the string which is being built.
|
* @param tail the string which is being built.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer make_string( struct pso4 *frame_pointer, wint_t c,
|
struct pso_pointer make_string( struct pso_pointer frame_pointer, wint_t c,
|
||||||
struct pso_pointer tail ) {
|
struct pso_pointer tail ) {
|
||||||
return make_string_like_thing( frame_pointer, c, tail, STRINGTAG );
|
return make_string_like_thing( frame_pointer, c, tail, STRINGTAG );
|
||||||
}
|
}
|
||||||
|
|
@ -120,7 +120,7 @@ struct pso_pointer make_string( struct pso4 *frame_pointer, wint_t c,
|
||||||
* @param c the character to add (prepend);
|
* @param c the character to add (prepend);
|
||||||
* @param tail the keyword which is being built.
|
* @param tail the keyword which is being built.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer make_keyword( struct pso4 *frame_pointer, wint_t c,
|
struct pso_pointer make_keyword( struct pso_pointer frame_pointer, wint_t c,
|
||||||
struct pso_pointer tail ) {
|
struct pso_pointer tail ) {
|
||||||
return make_string_like_thing( frame_pointer, c, tail, KEYTAG );
|
return make_string_like_thing( frame_pointer, c, tail, KEYTAG );
|
||||||
}
|
}
|
||||||
|
|
@ -133,7 +133,7 @@ struct pso_pointer make_keyword( struct pso4 *frame_pointer, wint_t c,
|
||||||
* @param c the character to add (prepend);
|
* @param c the character to add (prepend);
|
||||||
* @param tail the symbol which is being built.
|
* @param tail the symbol which is being built.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer make_symbol( struct pso4 *frame_pointer, wint_t c,
|
struct pso_pointer make_symbol( struct pso_pointer frame_pointer, wint_t c,
|
||||||
struct pso_pointer tail ) {
|
struct pso_pointer tail ) {
|
||||||
return make_string_like_thing( frame_pointer, c, tail, SYMBOLTAG );
|
return make_string_like_thing( frame_pointer, c, tail, SYMBOLTAG );
|
||||||
}
|
}
|
||||||
|
|
@ -142,7 +142,7 @@ struct pso_pointer make_symbol( struct pso4 *frame_pointer, wint_t c,
|
||||||
/**
|
/**
|
||||||
* Return a lisp string representation of this wide character string.
|
* Return a lisp string representation of this wide character string.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer c_string_to_lisp_string( struct pso4 *frame_pointer,
|
struct pso_pointer c_string_to_lisp_string( struct pso_pointer frame_pointer,
|
||||||
char32_t *string ) {
|
char32_t *string ) {
|
||||||
struct pso_pointer result = nil;
|
struct pso_pointer result = nil;
|
||||||
|
|
||||||
|
|
@ -164,7 +164,7 @@ struct pso_pointer c_string_to_lisp_string( struct pso4 *frame_pointer,
|
||||||
* Return a lisp symbol representation of this wide character string. In
|
* Return a lisp symbol representation of this wide character string. In
|
||||||
* symbols, I am accepting only lower case characters.
|
* symbols, I am accepting only lower case characters.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer c_string_to_lisp_symbol( struct pso4 *frame_pointer,
|
struct pso_pointer c_string_to_lisp_symbol( struct pso_pointer frame_pointer,
|
||||||
char32_t *symbol ) {
|
char32_t *symbol ) {
|
||||||
struct pso_pointer result = nil;
|
struct pso_pointer result = nil;
|
||||||
|
|
||||||
|
|
@ -183,7 +183,7 @@ struct pso_pointer c_string_to_lisp_symbol( struct pso4 *frame_pointer,
|
||||||
* Return a lisp keyword representation of this wide character string. In
|
* Return a lisp keyword representation of this wide character string. In
|
||||||
* keywords, I am accepting only lower case characters and numbers.
|
* keywords, I am accepting only lower case characters and numbers.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer c_string_to_lisp_keyword( struct pso4 *frame_pointer,
|
struct pso_pointer c_string_to_lisp_keyword( struct pso_pointer frame_pointer,
|
||||||
char32_t *symbol ) {
|
char32_t *symbol ) {
|
||||||
struct pso_pointer result = nil;
|
struct pso_pointer result = nil;
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -17,26 +17,26 @@
|
||||||
#include <wchar.h>
|
#include <wchar.h>
|
||||||
#include <wctype.h>
|
#include <wctype.h>
|
||||||
|
|
||||||
struct pso_pointer make_string_like_thing( struct pso4 *frame_pointer,
|
struct pso_pointer make_string_like_thing( struct pso_pointer frame_pointer,
|
||||||
wint_t c, struct pso_pointer tail,
|
wint_t c, struct pso_pointer tail,
|
||||||
char *tag );
|
char *tag );
|
||||||
|
|
||||||
struct pso_pointer make_string( struct pso4 *frame_pointer, wint_t c,
|
struct pso_pointer make_string( struct pso_pointer frame_pointer, wint_t c,
|
||||||
struct pso_pointer tail );
|
struct pso_pointer tail );
|
||||||
|
|
||||||
struct pso_pointer make_keyword( struct pso4 *frame_pointer, wint_t c,
|
struct pso_pointer make_keyword( struct pso_pointer frame_pointer, wint_t c,
|
||||||
struct pso_pointer tail );
|
struct pso_pointer tail );
|
||||||
|
|
||||||
struct pso_pointer make_symbol( struct pso4 *frame_pointer, wint_t c,
|
struct pso_pointer make_symbol( struct pso_pointer frame_pointer, wint_t c,
|
||||||
struct pso_pointer tail );
|
struct pso_pointer tail );
|
||||||
|
|
||||||
struct pso_pointer c_string_to_lisp_string( struct pso4 *frame_pointer,
|
struct pso_pointer c_string_to_lisp_string( struct pso_pointer frame_pointer,
|
||||||
char32_t * string );
|
char32_t * string );
|
||||||
|
|
||||||
struct pso_pointer c_string_to_lisp_keyword( struct pso4 *frame_pointer,
|
struct pso_pointer c_string_to_lisp_keyword( struct pso_pointer frame_pointer,
|
||||||
char32_t * symbol );
|
char32_t * symbol );
|
||||||
|
|
||||||
struct pso_pointer c_string_to_lisp_symbol( struct pso4 *frame_pointer,
|
struct pso_pointer c_string_to_lisp_symbol( struct pso_pointer frame_pointer,
|
||||||
char32_t * symbol );
|
char32_t * symbol );
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -27,21 +27,10 @@
|
||||||
* @return true if `p` points to `nil`.
|
* @return true if `p` points to `nil`.
|
||||||
* @return false otherwise.
|
* @return false otherwise.
|
||||||
*/
|
*/
|
||||||
bool nilp( struct pso_pointer p ) {
|
bool c_nilp(struct pso_pointer p) {
|
||||||
return ( p.page == 0 && p.offset == 0 );
|
return ( p.page == 0 && p.offset == 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Return `true` if `p` points to `nil`, else `false`.
|
|
||||||
*
|
|
||||||
* @param p a pointer
|
|
||||||
* @return true if `p` points to `nil`;
|
|
||||||
* @return false otherwise.
|
|
||||||
*/
|
|
||||||
bool not( struct pso_pointer p ) {
|
|
||||||
return !nilp( p );
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief `true` if `p` points to `t`, else `false`.
|
* @brief `true` if `p` points to `t`, else `false`.
|
||||||
*
|
*
|
||||||
|
|
@ -56,52 +45,82 @@ bool not( struct pso_pointer p ) {
|
||||||
* @return true if `p` points to `t`.
|
* @return true if `p` points to `t`.
|
||||||
* @return false otherwise.
|
* @return false otherwise.
|
||||||
*/
|
*/
|
||||||
bool truep( struct pso_pointer p ) {
|
bool c_truep( struct pso_pointer p ) {
|
||||||
return ( p.page == 0 && p.offset == 4 );
|
return ( p.page == 0 && p.offset == 4 );
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief return `t` if the first argument in this frame is `nil`, else `t`.
|
* @brief return `t` if the first argument in this frame is `nil`, else `t`.
|
||||||
*
|
*
|
||||||
* @param frame The current stack frame;
|
|
||||||
* @param frame_pointer A pointer to the current stack frame;
|
* @param frame_pointer A pointer to the current stack frame;
|
||||||
* @param env the evaluation environment.
|
|
||||||
* @return `t` if the first argument in this frame is `nil`, else `t`
|
* @return `t` if the first argument in this frame is `nil`, else `t`
|
||||||
*/
|
*/
|
||||||
struct pso_pointer lisp_nilp( struct pso_pointer frame_pointer,
|
struct pso_pointer nilp( struct pso_pointer frame_pointer ) {
|
||||||
struct pso_pointer env ) {
|
|
||||||
struct pso4 *frame = pointer_to_pso4( frame_pointer );
|
struct pso4 *frame = pointer_to_pso4( frame_pointer );
|
||||||
|
|
||||||
return ( nilp( fetch_arg( frame, 0 ) ) ? t : nil );
|
return ( c_nilp( fetch_arg( frame, 0 ) ) ? t : nil );
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief return `t` if the first argument in this frame is `t`, else `nil`.
|
* @brief return `t` if the first argument in this frame is `t`, else `nil`.
|
||||||
*
|
*
|
||||||
* @param frame The current stack frame;
|
|
||||||
* @param frame_pointer A pointer to the current stack frame;
|
* @param frame_pointer A pointer to the current stack frame;
|
||||||
* @param env the evaluation environment.
|
|
||||||
* @return `t` if the first argument in this frame is `t`, else `nil`.
|
* @return `t` if the first argument in this frame is `t`, else `nil`.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer lisp_truep( struct pso_pointer frame_pointer,
|
struct pso_pointer truep( struct pso_pointer frame_pointer ) {
|
||||||
struct pso_pointer env ) {
|
|
||||||
struct pso4 *frame = pointer_to_pso4( frame_pointer );
|
struct pso4 *frame = pointer_to_pso4( frame_pointer );
|
||||||
|
|
||||||
return ( truep( fetch_arg( frame, 0 ) ) ? t : nil );
|
return ( c_truep( fetch_arg( frame, 0 ) ) ? t : nil );
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief return `t` if the first argument in this frame is not `nil`, else
|
* @brief return `t` if the first argument in this frame is not `nil`, else
|
||||||
* `t`.
|
* `t`.
|
||||||
*
|
*
|
||||||
* @param frame The current stack frame;
|
|
||||||
* @param frame_pointer A pointer to the current stack frame;
|
* @param frame_pointer A pointer to the current stack frame;
|
||||||
* @param env the evaluation environment.
|
|
||||||
* @return `t` if the first argument in this frame is not `nil`, else `t`.
|
* @return `t` if the first argument in this frame is not `nil`, else `t`.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer lisp_not( struct pso_pointer frame_pointer,
|
struct pso_pointer not( struct pso_pointer frame_pointer) {
|
||||||
struct pso_pointer env ) {
|
|
||||||
struct pso4 *frame = pointer_to_pso4( frame_pointer );
|
struct pso4 *frame = pointer_to_pso4( frame_pointer );
|
||||||
|
|
||||||
return ( not( fetch_arg( frame, 0 ) ) ? t : nil );
|
return ( !c_nilp( fetch_arg( frame, 0 ) ) ? t : nil );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief (and args...)
|
||||||
|
*
|
||||||
|
* @return `nil` if any `arg` is `nil`, else `t`.
|
||||||
|
*/
|
||||||
|
struct pso_pointer and( struct pso_pointer frame_pointer) {
|
||||||
|
struct pso4* frame = pointer_to_pso4( frame_pointer);
|
||||||
|
struct pso_pointer result = t;
|
||||||
|
|
||||||
|
for (int arg = 0; c_truep(result) && arg < frame->payload.stack_frame.args; arg++) {
|
||||||
|
if (c_nilp(fetch_arg(frame, arg))) {
|
||||||
|
result = nil;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief (or args...)
|
||||||
|
*
|
||||||
|
* @return `t` if any `arg` is non-nil, else `nil`.
|
||||||
|
*/
|
||||||
|
struct pso_pointer or( struct pso_pointer frame_pointer) {
|
||||||
|
struct pso4* frame = pointer_to_pso4( frame_pointer);
|
||||||
|
struct pso_pointer result = nil;
|
||||||
|
|
||||||
|
for (int arg = 0; c_truep(result) && arg < frame->payload.stack_frame.args; arg++) {
|
||||||
|
if (!c_nilp(fetch_arg(frame, arg))) {
|
||||||
|
result = t;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -14,21 +14,18 @@
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
|
|
||||||
#include "memory/pointer.h"
|
#include "memory/pointer.h"
|
||||||
#include "memory/pso4.h"
|
|
||||||
|
|
||||||
bool nilp( struct pso_pointer p );
|
struct pso_pointer nilp( struct pso_pointer frame_pointer );
|
||||||
|
|
||||||
struct pso_pointer lisp_nilp( struct pso_pointer frame_pointer,
|
struct pso_pointer not( struct pso_pointer frame_pointer );
|
||||||
struct pso_pointer env );
|
|
||||||
|
|
||||||
bool not( struct pso_pointer p );
|
struct pso_pointer truep( struct pso_pointer frame_pointer );
|
||||||
|
|
||||||
struct pso_pointer lisp_not( struct pso_pointer frame_pointer,
|
struct pso_pointer and( struct pso_pointer frame_pointer );
|
||||||
struct pso_pointer env );
|
|
||||||
|
|
||||||
bool truep( struct pso_pointer p );
|
struct pso_pointer or( struct pso_pointer frame_pointer );
|
||||||
|
|
||||||
struct pso_pointer lisp_truep( struct pso_pointer frame_pointer,
|
bool c_nilp(struct pso_pointer p);
|
||||||
struct pso_pointer env );
|
bool c_truep(struct pso_pointer p);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -11,7 +11,6 @@
|
||||||
* wide characters
|
* wide characters
|
||||||
*/
|
*/
|
||||||
#include <uchar.h>
|
#include <uchar.h>
|
||||||
#include <wchar.h>
|
|
||||||
#include <wctype.h>
|
#include <wctype.h>
|
||||||
|
|
||||||
#include "memory/node.h"
|
#include "memory/node.h"
|
||||||
|
|
@ -22,12 +21,12 @@
|
||||||
|
|
||||||
#include "ops/truth.h"
|
#include "ops/truth.h"
|
||||||
|
|
||||||
#include "payloads/character.h"
|
// #include "payloads/character.h"
|
||||||
|
|
||||||
struct pso_pointer make_character( struct pso4 *frame_pointer, wint_t c ) {
|
struct pso_pointer make_character( struct pso_pointer frame_pointer, wint_t c ) {
|
||||||
struct pso_pointer result = allocate( frame_pointer, CHARACTERTAG, 2 );
|
struct pso_pointer result = allocate( frame_pointer, CHARACTERTAG, 2 );
|
||||||
|
|
||||||
if ( !nilp( result ) ) {
|
if ( !c_nilp( result ) ) {
|
||||||
pointer_to_object( result )->payload.character.character =
|
pointer_to_object( result )->payload.character.character =
|
||||||
( char32_t ) c;
|
( char32_t ) c;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -38,5 +38,5 @@ struct character_payload {
|
||||||
char32_t character;
|
char32_t character;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct pso_pointer make_character( struct pso4 *frame_pointer, wint_t c );
|
struct pso_pointer make_character( struct pso_pointer frame_pointer, wint_t c );
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -33,19 +33,18 @@
|
||||||
*/
|
*/
|
||||||
struct pso_pointer cons(struct pso_pointer frame_pointer) {
|
struct pso_pointer cons(struct pso_pointer frame_pointer) {
|
||||||
struct pso4 *frame = pointer_to_pso4(frame_pointer);
|
struct pso4 *frame = pointer_to_pso4(frame_pointer);
|
||||||
struct pso_pointer result = allocate( frame, CONSTAG, 2 );
|
struct pso_pointer result = allocate(frame_pointer, CONSTAG, 2);
|
||||||
|
|
||||||
if ( stackp( frame ) ) {
|
|
||||||
struct pso2 *object = pointer_to_object(result);
|
struct pso2 *object = pointer_to_object(result);
|
||||||
object->payload.cons.car =
|
object->payload.cons.car = inc_ref(fetch_arg(frame, 0));
|
||||||
inc_ref( frame->payload.stack_frame.args[0] );
|
object->payload.cons.cdr = inc_ref(fetch_arg(frame, 1));
|
||||||
object->payload.cons.cdr =
|
|
||||||
inc_ref( frame->payload.stack_frame.args[0] );
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct pso_pointer make_cons(struct pso_pointer frame_pointer, struct pso_pointer car, struct pso_pointer cdr){
|
||||||
|
return cons( make_frame(2, frame_pointer, car, cdr));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief return the car of this cons cell.
|
* @brief return the car of this cons cell.
|
||||||
|
|
@ -58,19 +57,21 @@ struct pso_pointer cons( struct pso_pointer frame_pointer ) {
|
||||||
*/
|
*/
|
||||||
struct pso_pointer car(struct pso_pointer frame_pointer) {
|
struct pso_pointer car(struct pso_pointer frame_pointer) {
|
||||||
struct pso_pointer result = nil;
|
struct pso_pointer result = nil;
|
||||||
struct pso_pointer cons = fetch_arg( pointer_to_pso4( frame_pointer), 0);
|
struct pso4 *frame = pointer_to_pso4(frame_pointer);
|
||||||
|
struct pso_pointer cons = fetch_arg(frame, 0);
|
||||||
struct pso2 *object = pointer_to_object(cons);
|
struct pso2 *object = pointer_to_object(cons);
|
||||||
|
|
||||||
if (consp(cons)) {
|
if (consp(cons)) {
|
||||||
result = object->payload.cons.car;
|
result = object->payload.cons.car;
|
||||||
} else {
|
} else {
|
||||||
result =
|
result = make_exception(make_frame(
|
||||||
make_exception( make_frame( 2, frame_pointer,
|
2, frame_pointer,
|
||||||
c_string_to_lisp_string(frame_pointer, L"Invalid type for car"),
|
c_string_to_lisp_string(frame_pointer, L"Invalid type for car"),
|
||||||
make_cons(
|
make_cons(frame_pointer, make_cons(
|
||||||
make_cons( frame_pointer,
|
frame_pointer,
|
||||||
c_string_to_lisp_keyword(frame_pointer, L"type"),
|
c_string_to_lisp_keyword(frame_pointer, L"type"),
|
||||||
get_tag_string( cons )), nil)));
|
get_tag_string(cons)),
|
||||||
|
nil)));
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
|
|
@ -87,11 +88,11 @@ struct pso_pointer car( struct pso_pointer frame_pointer ) {
|
||||||
*/
|
*/
|
||||||
struct pso_pointer cdr(struct pso_pointer frame_pointer) {
|
struct pso_pointer cdr(struct pso_pointer frame_pointer) {
|
||||||
struct pso_pointer result = nil;
|
struct pso_pointer result = nil;
|
||||||
struct pso4 *sp = pointer_to_pso4(frame_pointer);
|
struct pso4 *frame = pointer_to_pso4(frame_pointer);
|
||||||
struct pso_pointer cons = fetch_arg(sp, 0);
|
struct pso_pointer cons = fetch_arg(frame, 0);
|
||||||
struct pso2 *object = pointer_to_object(cons);
|
struct pso2 *object = pointer_to_object(cons);
|
||||||
|
|
||||||
switch ( get_tag_value( p ) ) {
|
switch (get_tag_value(cons)) {
|
||||||
case CONSTV:
|
case CONSTV:
|
||||||
result = object->payload.cons.cdr;
|
result = object->payload.cons.cdr;
|
||||||
break;
|
break;
|
||||||
|
|
@ -101,13 +102,16 @@ struct pso_pointer cdr( struct pso_pointer frame_pointer ) {
|
||||||
result = object->payload.string.cdr;
|
result = object->payload.string.cdr;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
result =
|
struct pso_pointer type_binding =
|
||||||
make_exception( make_frame( 2, frame_pointer,
|
|
||||||
c_string_to_lisp_string( frame_pointer, L"Invalid type for cdr" ),
|
|
||||||
make_cons(
|
|
||||||
make_cons(frame_pointer,
|
make_cons(frame_pointer,
|
||||||
c_string_to_lisp_keyword(frame_pointer, L"type"),
|
c_string_to_lisp_keyword(frame_pointer, L"type"),
|
||||||
get_tag_string( cons )), nil)));
|
get_tag_string(cons));
|
||||||
|
result = make_exception(make_frame(
|
||||||
|
2, frame_pointer,
|
||||||
|
c_string_to_lisp_string(frame_pointer, L"Invalid type for cdr"),
|
||||||
|
make_cons(frame_pointer,
|
||||||
|
type_binding,
|
||||||
|
nil)));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -121,12 +125,15 @@ struct pso_pointer cdr( struct pso_pointer frame_pointer ) {
|
||||||
* Lisp calling conventions; one expected arg, the pointer to the cell to
|
* Lisp calling conventions; one expected arg, the pointer to the cell to
|
||||||
* be destroyed.
|
* be destroyed.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer destroy_cons( struct pso_pointer fp,
|
struct pso_pointer destroy_cons(struct pso_pointer fp) {
|
||||||
struct pso_pointer env ) {
|
|
||||||
if (stackp(fp)) {
|
if (stackp(fp)) {
|
||||||
struct pso4 *frame = pointer_to_pso4(fp);
|
struct pso4 *frame = pointer_to_pso4(fp);
|
||||||
struct pso_pointer p = frame->payload.stack_frame.arg[0];
|
struct pso_pointer p = frame->payload.stack_frame.arg[0];
|
||||||
dec_ref( c_car( p ) );
|
|
||||||
dec_ref( c_cdr( frame, p ) );
|
if (check_tag(p, CONSTV)) {
|
||||||
|
struct pso2 *cons = pointer_to_object(p);
|
||||||
|
dec_ref(cons->payload.cons.car);
|
||||||
|
dec_ref(cons->payload.cons.cdr);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -33,7 +33,24 @@ struct pso_pointer cdr( struct pso_pointer frame_pointer );
|
||||||
|
|
||||||
struct pso_pointer cons( struct pso_pointer frame_pointer );
|
struct pso_pointer cons( struct pso_pointer frame_pointer );
|
||||||
|
|
||||||
struct pso_pointer destroy_cons( struct pso_pointer fp,
|
struct pso_pointer destroy_cons( struct pso_pointer frame_pointer);
|
||||||
struct pso_pointer env );
|
|
||||||
|
struct pso_pointer make_cons(struct pso_pointer frame_pointer,
|
||||||
|
struct pso_pointer car,
|
||||||
|
struct pso_pointer cdr);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* macro short-cuts for make_cons.
|
||||||
|
*/
|
||||||
|
// #define make_cons(frame_pointer,car,cdr) (cons(make_frame(2, frame_pointer, car, cdr)))
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Variant which assumes a convention that the frame pointer will always be
|
||||||
|
* called `frame_pointer`
|
||||||
|
*/
|
||||||
|
#define make_cons2(car,cdr) (cons(make_frame(2, frame_pointer, car, cdr)))
|
||||||
|
|
||||||
|
#define c_car(p)(consp(p) ? pointer_to_object(p)->payload.cons.car : nil)
|
||||||
|
#define c_cdr(p)(consp(p) ? pointer_to_object(p)->payload.cons.cdr : nil)
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -6,6 +6,13 @@
|
||||||
* (c) 2026 Simon Brooke <simon@journeyman.cc>
|
* (c) 2026 Simon Brooke <simon@journeyman.cc>
|
||||||
* Licensed under GPL version 2.0, or, at your option, any later version.
|
* Licensed under GPL version 2.0, or, at your option, any later version.
|
||||||
*/
|
*/
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
* wide characters
|
||||||
|
*/
|
||||||
|
#include <wchar.h>
|
||||||
|
#include <wctype.h>
|
||||||
|
|
||||||
|
|
||||||
#include "memory/node.h"
|
#include "memory/node.h"
|
||||||
|
|
@ -17,7 +24,12 @@
|
||||||
|
|
||||||
#include "payloads/exception.h"
|
#include "payloads/exception.h"
|
||||||
|
|
||||||
|
#include "ops/stack_ops.h"
|
||||||
#include "ops/truth.h"
|
#include "ops/truth.h"
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief allocate an exception object, and, if successful, return a pointer
|
* @brief allocate an exception object, and, if successful, return a pointer
|
||||||
|
|
@ -31,8 +43,7 @@
|
||||||
* otherwise it will return a pointer to a new exception.
|
* otherwise it will return a pointer to a new exception.
|
||||||
*
|
*
|
||||||
* @param message expected to be a string, but anything printable is accepted.
|
* @param message expected to be a string, but anything printable is accepted.
|
||||||
* @param frame the stack frame in which the exception was `thrown`, if any.
|
b * @param meta metadata for this exception. Must be an assoc list, hashtable,
|
||||||
* @param meta metadata for this exception. Must be an assoc list, hashtable,
|
|
||||||
* or `nil`
|
* or `nil`
|
||||||
* @param cause the exception that caused this exception to be `thrown`.
|
* @param cause the exception that caused this exception to be `thrown`.
|
||||||
*/
|
*/
|
||||||
|
|
@ -44,13 +55,13 @@ struct pso_pointer make_exception( struct pso_pointer frame_pointer) {
|
||||||
struct pso_pointer cause = fetch_arg( frame, 2);
|
struct pso_pointer cause = fetch_arg( frame, 2);
|
||||||
|
|
||||||
struct pso_pointer result =
|
struct pso_pointer result =
|
||||||
allocate( pointer_to_pso4( frame ), EXCEPTIONTAG, 3 );
|
allocate( frame_pointer, EXCEPTIONTAG, 3 );
|
||||||
|
|
||||||
if ( !nilp( result ) && !exceptionp( result ) ) {
|
if ( !c_nilp( result ) && !exceptionp( result ) ) {
|
||||||
struct pso3 *object = ( struct pso3 * ) pointer_to_object( result );
|
struct pso3 *object = ( struct pso3 * ) pointer_to_object( result );
|
||||||
|
|
||||||
object->payload.exception.message = message;
|
object->payload.exception.message = message;
|
||||||
object->payload.exception.stack = stackp( frame ) ? frame : nil;
|
object->payload.exception.stack = stackp( frame_pointer ) ? frame_pointer : nil;
|
||||||
object->payload.exception.meta = ( consp( meta )
|
object->payload.exception.meta = ( consp( meta )
|
||||||
|| hashtabp( meta ) ) ? meta : nil;
|
|| hashtabp( meta ) ) ? meta : nil;
|
||||||
object->payload.exception.cause = exceptionp( cause ) ? cause : nil;
|
object->payload.exception.cause = exceptionp( cause ) ? cause : nil;
|
||||||
|
|
|
||||||
|
|
@ -26,10 +26,7 @@ struct exception_payload {
|
||||||
struct pso_pointer cause;
|
struct pso_pointer cause;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct pso_pointer make_exception( struct pso_pointer message,
|
struct pso_pointer make_exception( struct pso_pointer frame_pointer );
|
||||||
struct pso_pointer frame_pointer,
|
|
||||||
struct pso_pointer meta,
|
|
||||||
struct pso_pointer cause );
|
|
||||||
|
|
||||||
struct pso_pointer destroy_exception( struct pso_pointer fp,
|
struct pso_pointer destroy_exception( struct pso_pointer fp,
|
||||||
struct pso_pointer env );
|
struct pso_pointer env );
|
||||||
|
|
|
||||||
|
|
@ -25,7 +25,7 @@
|
||||||
* @param more `nil`, or a pointer to the more significant cell(s) of this number.
|
* @param more `nil`, or a pointer to the more significant cell(s) of this number.
|
||||||
* *NOTE* that if `more` is not `nil`, `value` *must not* exceed `MAX_INTEGER`.
|
* *NOTE* that if `more` is not `nil`, `value` *must not* exceed `MAX_INTEGER`.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer make_integer( struct pso4 *frame_pointer, int64_t value ) {
|
struct pso_pointer make_integer( struct pso_pointer frame_pointer, int64_t value ) {
|
||||||
struct pso_pointer result = nil;
|
struct pso_pointer result = nil;
|
||||||
debug_print( L"Entering make_integer\n", DEBUG_ALLOC, 0 );
|
debug_print( L"Entering make_integer\n", DEBUG_ALLOC, 0 );
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -25,6 +25,6 @@ struct integer_payload {
|
||||||
__int128_t value;
|
__int128_t value;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct pso_pointer make_integer( struct pso4 *frame_pointer, int64_t value );
|
struct pso_pointer make_integer( struct pso_pointer frame_pointer, int64_t value );
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -16,6 +16,7 @@
|
||||||
#include "memory/node.h"
|
#include "memory/node.h"
|
||||||
#include "memory/pointer.h"
|
#include "memory/pointer.h"
|
||||||
#include "memory/pso.h"
|
#include "memory/pso.h"
|
||||||
|
#include "memory/pso2.h"
|
||||||
#include "memory/pso4.h"
|
#include "memory/pso4.h"
|
||||||
#include "memory/tags.h"
|
#include "memory/tags.h"
|
||||||
|
|
||||||
|
|
@ -34,7 +35,7 @@ struct pso_pointer destroy_string( struct pso_pointer fp,
|
||||||
struct pso4 *frame = pointer_to_pso4( fp );
|
struct pso4 *frame = pointer_to_pso4( fp );
|
||||||
struct pso_pointer p = frame->payload.stack_frame.arg[0];
|
struct pso_pointer p = frame->payload.stack_frame.arg[0];
|
||||||
|
|
||||||
dec_ref( c_cdr( frame, p ) );
|
dec_ref( c_cdr( p ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil;
|
return nil;
|
||||||
|
|
|
||||||
|
|
@ -25,7 +25,7 @@
|
||||||
* @param metadata a pointer to an associaton containing metadata on the stream.
|
* @param metadata a pointer to an associaton containing metadata on the stream.
|
||||||
* @return a pointer to the new read stream.
|
* @return a pointer to the new read stream.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer make_read_stream( struct pso4 *frame_pointer,
|
struct pso_pointer make_read_stream( struct pso_pointer frame_pointer,
|
||||||
URL_FILE *input,
|
URL_FILE *input,
|
||||||
struct pso_pointer metadata ) {
|
struct pso_pointer metadata ) {
|
||||||
struct pso_pointer pointer = allocate( frame_pointer, READTAG, 2 );
|
struct pso_pointer pointer = allocate( frame_pointer, READTAG, 2 );
|
||||||
|
|
|
||||||
|
|
@ -29,7 +29,7 @@ struct stream_payload {
|
||||||
struct pso_pointer meta;
|
struct pso_pointer meta;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct pso_pointer make_read_stream( struct pso4 *frame_pointer,
|
struct pso_pointer make_read_stream( struct pso_pointer frame_pointer,
|
||||||
URL_FILE * input,
|
URL_FILE * input,
|
||||||
struct pso_pointer metadata );
|
struct pso_pointer metadata );
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
/**
|
/**
|
||||||
* payloads/stack.h
|
* payloads/stack.c
|
||||||
*
|
*
|
||||||
* a Lisp stack frame.
|
* a Lisp stack frame.
|
||||||
*
|
*
|
||||||
|
|
@ -23,6 +23,7 @@
|
||||||
#include "payloads/cons.h"
|
#include "payloads/cons.h"
|
||||||
|
|
||||||
#include "ops/reverse.h"
|
#include "ops/reverse.h"
|
||||||
|
#include "ops/list_ops.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Construct a stack frame with this `previous` pointer, and arguments
|
* @brief Construct a stack frame with this `previous` pointer, and arguments
|
||||||
|
|
@ -43,7 +44,7 @@ struct pso_pointer make_frame( int arg_count, struct pso_pointer previous,
|
||||||
|
|
||||||
struct pso4 *prev_frame = pointer_to_pso4( previous );
|
struct pso4 *prev_frame = pointer_to_pso4( previous );
|
||||||
struct pso_pointer new_pointer =
|
struct pso_pointer new_pointer =
|
||||||
allocate( pointer_to_pso4( previous ), STACKTAG, 4 );
|
allocate( previous, STACKTAG, 4 );
|
||||||
struct pso4* new_frame = pointer_to_pso4(new_pointer);
|
struct pso4* new_frame = pointer_to_pso4(new_pointer);
|
||||||
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
|
@ -78,7 +79,7 @@ struct pso_pointer make_frame( int arg_count, struct pso_pointer previous,
|
||||||
|
|
||||||
for ( ; cursor < arg_count; cursor++ ) {
|
for ( ; cursor < arg_count; cursor++ ) {
|
||||||
more_args =
|
more_args =
|
||||||
make_cons( prev_frame, va_arg( args, struct pso_pointer ),
|
make_cons( previous, va_arg( args, struct pso_pointer ),
|
||||||
more_args );
|
more_args );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -117,7 +118,7 @@ struct pso_pointer make_frame_with_env( int arg_count,
|
||||||
|
|
||||||
struct pso4 *prev_frame = pointer_to_pso4( previous );
|
struct pso4 *prev_frame = pointer_to_pso4( previous );
|
||||||
struct pso_pointer new_pointer =
|
struct pso_pointer new_pointer =
|
||||||
allocate( pointer_to_pso4( previous ), STACKTAG, 4 );
|
allocate( previous, STACKTAG, 4 );
|
||||||
struct pso4* new_frame = pointer_to_pso4(new_pointer);
|
struct pso4* new_frame = pointer_to_pso4(new_pointer);
|
||||||
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
|
@ -152,7 +153,7 @@ struct pso_pointer make_frame_with_env( int arg_count,
|
||||||
|
|
||||||
for ( ; cursor < arg_count; cursor++ ) {
|
for ( ; cursor < arg_count; cursor++ ) {
|
||||||
more_args =
|
more_args =
|
||||||
make_cons( prev_frame, va_arg( args, struct pso_pointer ),
|
make_cons( previous, va_arg( args, struct pso_pointer ),
|
||||||
more_args );
|
more_args );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -184,10 +185,10 @@ struct pso_pointer make_frame_with_arglist_and_env( struct pso_pointer previous,
|
||||||
struct pso_pointer env) {
|
struct pso_pointer env) {
|
||||||
struct pso4 *prev_frame = pointer_to_pso4( previous );
|
struct pso4 *prev_frame = pointer_to_pso4( previous );
|
||||||
struct pso_pointer new_pointer =
|
struct pso_pointer new_pointer =
|
||||||
allocate( pointer_to_pso4( previous ), STACKTAG, 4 );
|
allocate( previous, STACKTAG, 4 );
|
||||||
struct pso4* new_frame = pointer_to_pso4(new_pointer);
|
struct pso4* new_frame = pointer_to_pso4(new_pointer);
|
||||||
int arg_count = c_length(argvalues);
|
struct pso_pointer arg_length = length(make_frame(1, previous, argvalues));
|
||||||
|
int arg_count = integerp(arg_length) ? pointer_to_object(arg_length)->payload.integer.value : 0;
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
debug_printf( DEBUG_ALLOC, 0,
|
debug_printf( DEBUG_ALLOC, 0,
|
||||||
L"\nAllocating stack frame with %d arguments at page %d, "
|
L"\nAllocating stack frame with %d arguments at page %d, "
|
||||||
|
|
@ -216,7 +217,7 @@ struct pso_pointer make_frame_with_arglist_and_env( struct pso_pointer previous,
|
||||||
argvalues = cdr( make_frame( 1, previous, argvalues));
|
argvalues = cdr( make_frame( 1, previous, argvalues));
|
||||||
}
|
}
|
||||||
if ( cursor < arg_count ) {
|
if ( cursor < arg_count ) {
|
||||||
new_frame->payload.stack_frame.more = inc_ref( cursor);
|
new_frame->payload.stack_frame.more = inc_ref( argvalues);
|
||||||
} else {
|
} else {
|
||||||
for ( ; cursor < args_in_frame; cursor++ ) {
|
for ( ; cursor < args_in_frame; cursor++ ) {
|
||||||
new_frame->payload.stack_frame.arg[cursor] = nil;
|
new_frame->payload.stack_frame.arg[cursor] = nil;
|
||||||
|
|
|
||||||
|
|
@ -43,9 +43,18 @@ struct stack_frame_payload {
|
||||||
|
|
||||||
struct pso_pointer make_frame( int arg_count, struct pso_pointer previous,
|
struct pso_pointer make_frame( int arg_count, struct pso_pointer previous,
|
||||||
... );
|
... );
|
||||||
|
|
||||||
struct pso_pointer make_frame_with_env( int arg_count,
|
struct pso_pointer make_frame_with_env( int arg_count,
|
||||||
struct pso_pointer previous,
|
struct pso_pointer previous,
|
||||||
struct pso_pointer env, ... )
|
struct pso_pointer env, ... );
|
||||||
|
|
||||||
|
struct pso_pointer make_frame_with_arglist_and_env( struct pso_pointer previous,
|
||||||
|
struct pso_pointer argvalues,
|
||||||
|
struct pso_pointer env);
|
||||||
|
|
||||||
|
struct pso_pointer make_frame_with_arglist( struct pso_pointer previous,
|
||||||
|
struct pso_pointer argvalues);
|
||||||
|
|
||||||
struct pso_pointer destroy_stack_frame( struct pso_pointer fp,
|
struct pso_pointer destroy_stack_frame( struct pso_pointer fp,
|
||||||
struct pso_pointer env );
|
struct pso_pointer env );
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -25,7 +25,7 @@
|
||||||
* @param metadata a pointer to an associaton containing metadata on the stream.
|
* @param metadata a pointer to an associaton containing metadata on the stream.
|
||||||
* @return a pointer to the new read stream.
|
* @return a pointer to the new read stream.
|
||||||
*/
|
*/
|
||||||
struct pso_pointer make_write_stream( struct pso4 *frame_pointer,
|
struct pso_pointer make_write_stream( struct pso_pointer frame_pointer,
|
||||||
URL_FILE *output,
|
URL_FILE *output,
|
||||||
struct pso_pointer metadata ) {
|
struct pso_pointer metadata ) {
|
||||||
struct pso_pointer pointer = allocate( frame_pointer, WRITETAG, 2 );
|
struct pso_pointer pointer = allocate( frame_pointer, WRITETAG, 2 );
|
||||||
|
|
|
||||||
|
|
@ -13,7 +13,7 @@
|
||||||
/* write stream shares a payload with /see read_streem.h */
|
/* write stream shares a payload with /see read_streem.h */
|
||||||
|
|
||||||
#include "io/fopen.h"
|
#include "io/fopen.h"
|
||||||
struct pso_pointer make_write_stream( struct pso4 *frame_pointer,
|
struct pso_pointer make_write_stream( struct pso_pointer frame_pointer,
|
||||||
URL_FILE * output,
|
URL_FILE * output,
|
||||||
struct pso_pointer metadata );
|
struct pso_pointer metadata );
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue