2024-03-18 09:47:07 -07:00
// Compilation logic
2024-02-04 12:23:59 -08:00
# include <ctype.h>
2024-09-15 18:18:42 -07:00
# include <glob.h>
2024-02-04 12:23:59 -08:00
# include <gc.h>
2024-08-12 22:30:25 -07:00
# include <gc/cord.h>
# include <gmp.h>
2024-02-04 12:23:59 -08:00
# include <stdio.h>
2024-03-03 16:12:53 -08:00
# include <uninorm.h>
2024-02-04 12:23:59 -08:00
# include "ast.h"
2024-02-04 15:04:41 -08:00
# include "compile.h"
2024-09-13 11:23:24 -07:00
# include "cordhelpers.h"
2024-02-24 12:24:44 -08:00
# include "enums.h"
2024-02-17 13:56:19 -08:00
# include "environment.h"
2024-09-15 18:18:42 -07:00
# include "stdlib/integers.h"
# include "stdlib/patterns.h"
# include "stdlib/text.h"
2024-09-13 17:18:08 -07:00
# include "stdlib/util.h"
2024-09-15 18:18:42 -07:00
# include "structs.h"
# include "typecheck.h"
2024-02-04 12:23:59 -08:00
2024-09-12 11:27:13 -07:00
typedef ast_t * ( * comprehension_body_t ) ( ast_t * , ast_t * ) ;
2024-08-20 13:20:41 -07:00
static CORD compile_to_pointer_depth ( env_t * env , ast_t * ast , int64_t target_depth , bool needs_incref ) ;
2024-05-15 10:40:27 -07:00
static env_t * with_enum_scope ( env_t * env , type_t * t ) ;
2024-08-10 17:50:15 -07:00
static CORD compile_math_method ( env_t * env , binop_e op , ast_t * lhs , ast_t * rhs , type_t * required_type ) ;
2024-07-04 13:23:05 -07:00
static CORD compile_string ( env_t * env , ast_t * ast , CORD color ) ;
2024-07-13 14:17:58 -07:00
static CORD compile_arguments ( env_t * env , ast_t * call_ast , arg_t * spec_args , arg_ast_t * call_args ) ;
2024-08-12 14:44:05 -07:00
static CORD compile_maybe_incref ( env_t * env , ast_t * ast ) ;
2024-08-17 20:43:30 -07:00
static CORD compile_int_to_type ( env_t * env , ast_t * ast , type_t * target ) ;
2024-11-03 12:48:13 -08:00
static CORD compile_unsigned_type ( type_t * t ) ;
2024-09-11 09:01:17 -07:00
static CORD promote_to_optional ( type_t * t , CORD code ) ;
2024-09-11 11:53:48 -07:00
static CORD compile_null ( type_t * t ) ;
2024-11-21 10:00:53 -08:00
static CORD compile_to_type ( env_t * env , ast_t * ast , type_t * t ) ;
2024-03-22 10:53:23 -07:00
2024-09-11 09:01:17 -07:00
CORD promote_to_optional ( type_t * t , CORD code )
{
2024-09-11 09:56:16 -07:00
if ( t = = THREAD_TYPE ) {
return code ;
} else if ( t - > tag = = IntType ) {
2024-09-11 09:01:17 -07:00
switch ( Match ( t , IntType ) - > bits ) {
2024-09-11 21:55:43 -07:00
case TYPE_IBITS8 : return CORD_all ( " ((OptionalInt8_t){ " , code , " }) " ) ;
case TYPE_IBITS16 : return CORD_all ( " ((OptionalInt16_t){ " , code , " }) " ) ;
case TYPE_IBITS32 : return CORD_all ( " ((OptionalInt32_t){ " , code , " }) " ) ;
case TYPE_IBITS64 : return CORD_all ( " ((OptionalInt64_t){ " , code , " }) " ) ;
2024-09-11 09:01:17 -07:00
default : errx ( 1 , " Unsupported in type: %T " , t ) ;
}
2024-09-15 12:33:47 -07:00
} else if ( t - > tag = = ByteType ) {
return CORD_all ( " ((OptionalByte_t) { " , code, " } ) " );
2024-09-11 09:01:17 -07:00
} else if ( t - > tag = = StructType ) {
2024-10-08 18:10:36 -07:00
return CORD_all ( " ({ " , compile_type ( Type ( OptionalType , . type = t ) ) , " nonnull = { . value = " , code, " } ; nonnull . is_null = false ; nonnull ; } ) " );
2024-09-11 09:01:17 -07:00
} else {
return code ;
}
}
2024-09-11 20:13:41 -07:00
2024-03-09 13:03:38 -08:00
static bool promote ( env_t * env , CORD * code , type_t * actual , type_t * needed )
{
if ( type_eq ( actual , needed ) )
return true ;
if ( ! can_promote ( actual , needed ) )
return false ;
2024-09-11 20:13:41 -07:00
// Optional promotion:
if ( needed - > tag = = OptionalType & & type_eq ( actual , Match ( needed , OptionalType ) - > type ) ) {
* code = promote_to_optional ( actual , * code ) ;
return true ;
}
2024-08-18 08:49:51 -07:00
if ( actual - > tag = = IntType & & needed - > tag = = BigIntType ) {
2024-08-12 23:45:57 -07:00
* code = CORD_all ( " I( " , * code , " ) " ) ;
return true ;
}
2024-08-18 09:47:29 -07:00
if ( ( actual - > tag = = IntType | | actual - > tag = = BigIntType ) & & needed - > tag = = NumType ) {
2024-08-14 11:57:01 -07:00
* code = CORD_all ( type_to_cord ( actual ) , " _to_ " , type_to_cord ( needed ) , " ( " , * code , " ) " ) ;
2024-08-12 23:51:10 -07:00
return true ;
}
2024-08-17 12:14:45 -07:00
if ( actual - > tag = = NumType & & needed - > tag = = IntType )
return false ;
2024-03-09 13:03:38 -08:00
if ( actual - > tag = = IntType | | actual - > tag = = NumType )
return true ;
2024-09-12 10:05:08 -07:00
if ( needed - > tag = = EnumType ) {
const char * tag = enum_single_value_tag ( needed , actual ) ;
binding_t * b = get_binding ( Match ( needed , EnumType ) - > env , tag ) ;
assert ( b & & b - > type - > tag = = FunctionType ) ;
// Single-value enum constructor:
if ( ! promote ( env , code , actual , Match ( b - > type , FunctionType ) - > args - > type ) )
return false ;
* code = CORD_all ( b - > code , " ( " , * code , " ) " ) ;
return true ;
}
2024-09-02 16:53:09 -07:00
// Text to C String
if ( actual - > tag = = TextType & & ! Match ( actual , TextType ) - > lang & & needed - > tag = = CStringType ) {
* code = CORD_all ( " Text$as_c_string( " , * code , " ) " ) ;
return true ;
}
2024-03-09 13:03:38 -08:00
// Automatic dereferencing:
2024-09-10 22:31:31 -07:00
if ( actual - > tag = = PointerType
2024-03-09 13:03:38 -08:00
& & can_promote ( Match ( actual , PointerType ) - > pointed , needed ) ) {
* code = CORD_all ( " *( " , * code , " ) " ) ;
return promote ( env , code , Match ( actual , PointerType ) - > pointed , needed ) ;
}
2024-09-10 22:31:31 -07:00
// Stack ref promotion:
2024-03-09 13:03:38 -08:00
if ( actual - > tag = = PointerType & & needed - > tag = = PointerType )
return true ;
2024-05-12 16:21:44 -07:00
if ( needed - > tag = = ClosureType & & actual - > tag = = FunctionType ) {
2024-09-11 10:56:39 -07:00
* code = CORD_all ( " ((Closure_t){ " , * code , " , NULL}) " ) ;
2024-03-09 13:03:38 -08:00
return true ;
}
2024-05-12 16:21:44 -07:00
if ( needed - > tag = = ClosureType & & actual - > tag = = ClosureType )
return true ;
2024-05-12 13:09:24 -07:00
if ( needed - > tag = = FunctionType & & actual - > tag = = FunctionType ) {
2024-06-16 15:09:54 -07:00
* code = CORD_all ( " ( " , compile_type ( needed ) , " ) " , * code ) ;
2024-05-12 13:09:24 -07:00
return true ;
}
2024-08-10 12:15:38 -07:00
// Set -> Array promotion:
if ( needed - > tag = = ArrayType & & actual - > tag = = SetType
& & type_eq ( Match ( needed , ArrayType ) - > item_type , Match ( actual , SetType ) - > item_type ) ) {
* code = CORD_all ( " ( " , * code , " ).entries " ) ;
return true ;
}
2024-03-09 13:03:38 -08:00
return false ;
}
2024-08-12 14:44:05 -07:00
CORD compile_maybe_incref ( env_t * env , ast_t * ast )
{
type_t * t = get_type ( env , ast ) ;
2024-09-06 01:14:50 -07:00
if ( is_idempotent ( ast ) & & can_be_mutated ( env , ast ) ) {
2024-08-12 14:44:05 -07:00
if ( t - > tag = = ArrayType )
return CORD_all ( " ARRAY_COPY( " , compile ( env , ast ) , " ) " ) ;
else if ( t - > tag = = TableType | | t - > tag = = SetType )
return CORD_all ( " TABLE_COPY( " , compile ( env , ast ) , " ) " ) ;
}
return compile ( env , ast ) ;
}
2024-09-05 11:57:31 -07:00
static Table_t * get_closed_vars ( env_t * env , ast_t * lambda_ast )
2024-06-06 13:28:53 -07:00
{
auto lambda = Match ( lambda_ast , Lambda ) ;
env_t * body_scope = fresh_scope ( env ) ;
2024-07-14 11:13:23 -07:00
body_scope - > code = new ( compilation_unit_t ) ; // Don't put any code in the headers or anything
2024-06-06 13:28:53 -07:00
for ( arg_ast_t * arg = lambda - > args ; arg ; arg = arg - > next ) {
type_t * arg_type = get_arg_ast_type ( env , arg ) ;
set_binding ( body_scope , arg - > name , new ( binding_t , . type = arg_type , . code = CORD_cat ( " $ " , arg - > name ) ) ) ;
}
fn_ctx_t fn_ctx = ( fn_ctx_t ) {
2024-07-14 11:13:23 -07:00
. parent = env - > fn_ctx ,
. closure_scope = env - > locals ,
2024-09-05 11:57:31 -07:00
. closed_vars = new ( Table_t ) ,
2024-06-06 13:28:53 -07:00
} ;
body_scope - > fn_ctx = & fn_ctx ;
body_scope - > locals - > fallback = env - > globals ;
2024-07-13 15:43:50 -07:00
type_t * ret_t = get_type ( body_scope , lambda - > body ) ;
if ( ret_t - > tag = = ReturnType )
ret_t = Match ( ret_t , ReturnType ) - > ret ;
2024-06-06 13:28:53 -07:00
fn_ctx . return_type = ret_t ;
// Find which variables are captured in the closure:
env_t * tmp_scope = fresh_scope ( body_scope ) ;
for ( ast_list_t * stmt = Match ( lambda - > body , Block ) - > statements ; stmt ; stmt = stmt - > next ) {
2024-07-14 10:22:42 -07:00
bind_statement ( tmp_scope , stmt - > ast ) ;
2024-06-06 13:28:53 -07:00
type_t * stmt_type = get_type ( tmp_scope , stmt - > ast ) ;
2024-07-14 10:22:42 -07:00
if ( stmt - > next | | ( stmt_type - > tag = = VoidType | | stmt_type - > tag = = AbortType | | get_type ( tmp_scope , stmt - > ast ) - > tag = = ReturnType ) )
2024-06-06 13:28:53 -07:00
( void ) compile_statement ( tmp_scope , stmt - > ast ) ;
else
( void ) compile ( tmp_scope , stmt - > ast ) ;
}
return fn_ctx . closed_vars ;
}
2024-06-16 15:09:54 -07:00
CORD compile_declaration ( type_t * t , CORD name )
2024-03-09 11:02:19 -08:00
{
if ( t - > tag = = FunctionType ) {
auto fn = Match ( t , FunctionType ) ;
2024-06-16 15:09:54 -07:00
CORD code = CORD_all ( compile_type ( fn - > ret ) , " (* " , name , " )( " ) ;
2024-03-09 11:02:19 -08:00
for ( arg_t * arg = fn - > args ; arg ; arg = arg - > next ) {
2024-06-16 15:09:54 -07:00
code = CORD_all ( code , compile_type ( arg - > type ) ) ;
2024-03-09 11:02:19 -08:00
if ( arg - > next ) code = CORD_cat ( code , " , " ) ;
}
return CORD_all ( code , " ) " ) ;
2024-06-06 13:28:53 -07:00
} else if ( t - > tag ! = ModuleType ) {
2024-06-16 15:09:54 -07:00
return CORD_all ( compile_type ( t ) , " " , name ) ;
2024-06-06 13:28:53 -07:00
} else {
return CORD_EMPTY ;
2024-03-09 11:02:19 -08:00
}
}
2024-11-03 12:48:13 -08:00
PUREFUNC CORD compile_unsigned_type ( type_t * t )
{
if ( t - > tag ! = IntType )
errx ( 1 , " Not an int type, so unsigned doesn't make sense! " ) ;
switch ( Match ( t , IntType ) - > bits ) {
case TYPE_IBITS8 : return " uint8_t " ;
case TYPE_IBITS16 : return " uint16_t " ;
case TYPE_IBITS32 : return " uint32_t " ;
case TYPE_IBITS64 : return " uint64_t " ;
default : errx ( 1 , " Invalid integer bit size " ) ;
}
}
2024-06-16 15:09:54 -07:00
CORD compile_type ( type_t * t )
2024-02-17 23:34:39 -08:00
{
2024-09-24 11:54:22 -07:00
if ( t = = THREAD_TYPE ) return " Thread_t " ;
else if ( t = = RANGE_TYPE ) return " Range_t " ;
2024-11-03 19:37:48 -08:00
else if ( t = = RNG_TYPE ) return " RNG_t " ;
2024-11-09 13:27:54 -08:00
else if ( t = = MATCH_TYPE ) return " Match_t " ;
2024-09-24 11:54:22 -07:00
2024-02-17 23:34:39 -08:00
switch ( t - > tag ) {
2024-07-13 15:43:50 -07:00
case ReturnType : errx ( 1 , " Shouldn't be compiling ReturnType to a type " ) ;
2024-02-17 23:34:39 -08:00
case AbortType : return " void " ;
case VoidType : return " void " ;
case MemoryType : return " void " ;
case BoolType : return " Bool_t " ;
2024-09-15 12:33:47 -07:00
case ByteType : return " Byte_t " ;
2024-05-18 11:38:41 -07:00
case CStringType : return " char* " ;
2024-11-17 11:49:03 -08:00
case MomentType : return " Moment_t " ;
2024-08-18 08:49:51 -07:00
case BigIntType : return " Int_t " ;
case IntType : return CORD_asprintf ( " Int%ld_t " , Match ( t , IntType ) - > bits ) ;
case NumType : return Match ( t , NumType ) - > bits = = TYPE_NBITS64 ? " Num_t " : CORD_asprintf ( " Num%ld_t " , Match ( t , NumType ) - > bits ) ;
2024-03-03 15:15:45 -08:00
case TextType : {
2024-03-21 22:52:00 -07:00
auto text = Match ( t , TextType ) ;
2024-09-24 11:54:22 -07:00
if ( ! text - > lang | | streq ( text - > lang , " Text " ) )
return " Text_t " ;
else if ( streq ( text - > lang , " Pattern " ) )
return " Pattern_t " ;
else if ( streq ( text - > lang , " Path " ) )
return " Path_t " ;
else if ( streq ( text - > lang , " Shell " ) )
return " Shell_t " ;
else
return CORD_all ( namespace_prefix ( text - > env , text - > env - > namespace - > parent ) , text - > lang , " _t " ) ;
2024-02-17 23:34:39 -08:00
}
2024-09-05 11:56:37 -07:00
case ArrayType : return " Array_t " ;
2024-09-05 11:57:31 -07:00
case SetType : return " Table_t " ;
2024-09-11 10:57:21 -07:00
case ChannelType : return " Channel_t* " ;
2024-09-05 11:57:31 -07:00
case TableType : return " Table_t " ;
2024-03-09 11:02:19 -08:00
case FunctionType : {
auto fn = Match ( t , FunctionType ) ;
2024-06-16 15:09:54 -07:00
CORD code = CORD_all ( compile_type ( fn - > ret ) , " (*)( " ) ;
2024-03-09 11:02:19 -08:00
for ( arg_t * arg = fn - > args ; arg ; arg = arg - > next ) {
2024-06-16 15:09:54 -07:00
code = CORD_all ( code , compile_type ( arg - > type ) ) ;
2024-03-09 11:02:19 -08:00
if ( arg - > next ) code = CORD_cat ( code , " , " ) ;
}
return CORD_all ( code , " ) " ) ;
}
2024-09-11 10:56:39 -07:00
case ClosureType : return " Closure_t " ;
2024-06-16 15:09:54 -07:00
case PointerType : return CORD_cat ( compile_type ( Match ( t , PointerType ) - > pointed ) , " * " ) ;
2024-03-21 22:52:00 -07:00
case StructType : {
2024-09-11 09:01:17 -07:00
auto s = Match ( t , StructType ) ;
2024-09-24 11:54:22 -07:00
return CORD_all ( " struct " , namespace_prefix ( s - > env , s - > env - > namespace - > parent ) , s - > name , " _s " ) ;
2024-03-21 22:52:00 -07:00
}
case EnumType : {
auto e = Match ( t , EnumType ) ;
2024-09-24 11:54:22 -07:00
return CORD_all ( namespace_prefix ( e - > env , e - > env - > namespace - > parent ) , e - > name , " _t " ) ;
2024-03-21 22:52:00 -07:00
}
2024-09-10 22:31:31 -07:00
case OptionalType : {
type_t * nonnull = Match ( t , OptionalType ) - > type ;
2024-09-11 09:01:17 -07:00
switch ( nonnull - > tag ) {
2024-09-27 11:45:06 -07:00
case CStringType : case FunctionType : case ClosureType :
2024-09-11 09:50:46 -07:00
case PointerType : case EnumType : case ChannelType :
2024-09-11 09:01:17 -07:00
return compile_type ( nonnull ) ;
2024-11-03 22:24:28 -08:00
case TextType :
return Match ( nonnull , TextType ) - > lang ? compile_type ( nonnull ) : " OptionalText_t " ;
case IntType : case BigIntType : case NumType : case BoolType : case ByteType :
2024-11-17 11:49:03 -08:00
case ArrayType : case TableType : case SetType : case MomentType :
2024-09-11 08:08:15 -07:00
return CORD_all ( " Optional " , compile_type ( nonnull ) ) ;
2024-09-11 09:01:17 -07:00
case StructType : {
if ( nonnull = = THREAD_TYPE )
2024-09-24 11:54:22 -07:00
return " Thread_t " ;
2024-09-11 09:01:17 -07:00
auto s = Match ( nonnull , StructType ) ;
2024-09-24 11:54:22 -07:00
return CORD_all ( namespace_prefix ( s - > env , s - > env - > namespace - > parent ) , " $Optional " , s - > name , " _t " ) ;
2024-09-11 09:01:17 -07:00
}
default :
2024-09-10 22:31:31 -07:00
compiler_err ( NULL , NULL , NULL , " Optional types are not supported for: %T " , t ) ;
2024-09-11 09:01:17 -07:00
}
2024-09-10 22:31:31 -07:00
}
2024-09-30 11:39:30 -07:00
case TypeInfoType : return " TypeInfo_t " ;
2024-06-06 13:28:53 -07:00
default : compiler_err ( NULL , NULL , NULL , " Compiling type is not implemented for type with tag %d " , t - > tag ) ;
2024-02-17 23:34:39 -08:00
}
}
2024-03-24 11:28:20 -07:00
static CORD compile_lvalue ( env_t * env , ast_t * ast )
2024-03-14 10:28:30 -07:00
{
if ( ! can_be_mutated ( env , ast ) ) {
2024-10-27 17:25:16 -07:00
if ( ast - > tag = = Index ) {
ast_t * subject = Match ( ast , Index ) - > indexed ;
type_t * t = get_type ( env , subject ) ;
if ( t - > tag = = PointerType & & Match ( t , PointerType ) - > is_view )
code_err ( ast , " This is a read-only view and you can't mutate its contents " ) ;
else
code_err ( subject , " This is an immutable value, you can't mutate its contents " ) ;
} else if ( ast - > tag = = FieldAccess ) {
ast_t * subject = Match ( ast , FieldAccess ) - > fielded ;
type_t * t = get_type ( env , subject ) ;
if ( t - > tag = = PointerType & & Match ( t , PointerType ) - > is_view )
code_err ( subject , " This is a read-only view and you can't mutate its fields " ) ;
else
code_err ( subject , " This is an immutable value, you can't assign to its fields " ) ;
2024-03-14 10:28:30 -07:00
} else {
2024-08-18 12:22:51 -07:00
code_err ( ast , " This is a value of type %T and can't be used as an assignment target " , get_type ( env , ast ) ) ;
2024-03-14 10:28:30 -07:00
}
}
2024-03-24 11:28:20 -07:00
if ( ast - > tag = = Index ) {
auto index = Match ( ast , Index ) ;
2024-03-22 10:53:23 -07:00
type_t * container_t = get_type ( env , index - > indexed ) ;
2024-09-10 22:31:31 -07:00
if ( container_t - > tag = = OptionalType )
code_err ( index - > indexed , " This value might be null, so it can't be safely used as an assignment target " ) ;
if ( ! index - > index & & container_t - > tag = = PointerType )
2024-03-24 11:28:20 -07:00
return compile ( env , ast ) ;
2024-09-10 22:31:31 -07:00
2024-03-22 10:53:23 -07:00
container_t = value_type ( container_t ) ;
2024-03-24 11:28:20 -07:00
if ( container_t - > tag = = ArrayType ) {
2024-08-20 13:20:41 -07:00
CORD target_code = compile_to_pointer_depth ( env , index - > indexed , 1 , false ) ;
2024-03-22 10:53:23 -07:00
type_t * item_type = Match ( container_t , ArrayType ) - > item_type ;
2024-08-18 12:51:19 -07:00
if ( index - > unchecked ) {
return CORD_all ( " Array_lvalue_unchecked( " , compile_type ( item_type ) , " , " , target_code , " , " ,
compile_int_to_type ( env , index - > index , Type ( IntType , . bits = TYPE_IBITS64 ) ) ,
2024-10-08 18:10:36 -07:00
" , " , CORD_asprintf ( " %ld " , type_size ( item_type ) ) , " ) " ) ;
2024-08-18 12:51:19 -07:00
} else {
return CORD_all ( " Array_lvalue( " , compile_type ( item_type ) , " , " , target_code , " , " ,
compile_int_to_type ( env , index - > index , Type ( IntType , . bits = TYPE_IBITS64 ) ) ,
2024-10-08 18:10:36 -07:00
" , " , CORD_asprintf ( " %ld " , type_size ( item_type ) ) ,
2024-09-04 02:04:45 -07:00
" , " , heap_strf ( " %ld " , ast - > start - ast - > file - > text ) ,
2024-08-18 12:51:19 -07:00
" , " , heap_strf ( " %ld " , ast - > end - ast - > file - > text ) , " ) " ) ;
}
2024-03-24 11:28:20 -07:00
} else {
code_err ( ast , " I don't know how to assign to this target " ) ;
2024-03-22 10:53:23 -07:00
}
2024-03-24 11:28:20 -07:00
} else if ( ast - > tag = = Var | | ast - > tag = = FieldAccess ) {
return compile ( env , ast ) ;
} else {
code_err ( ast , " I don't know how to assign to this " ) ;
2024-03-22 10:53:23 -07:00
}
2024-03-24 11:28:20 -07:00
}
static CORD compile_assignment ( env_t * env , ast_t * target , CORD value )
{
2024-09-03 20:32:02 -07:00
return CORD_all ( compile_lvalue ( env , target ) , " = " , value ) ;
2024-03-22 10:53:23 -07:00
}
2024-09-05 09:28:28 -07:00
static CORD compile_inline_block ( env_t * env , ast_t * ast )
{
if ( ast - > tag ! = Block )
return compile_statement ( env , ast ) ;
CORD code = CORD_EMPTY ;
ast_list_t * stmts = Match ( ast , Block ) - > statements ;
deferral_t * prev_deferred = env - > deferred ;
env = fresh_scope ( env ) ;
for ( ast_list_t * stmt = stmts ; stmt ; stmt = stmt - > next )
prebind_statement ( env , stmt - > ast ) ;
for ( ast_list_t * stmt = stmts ; stmt ; stmt = stmt - > next ) {
bind_statement ( env , stmt - > ast ) ;
code = CORD_all ( code , compile_statement ( env , stmt - > ast ) , " \n " ) ;
}
for ( deferral_t * deferred = env - > deferred ; deferred & & deferred ! = prev_deferred ; deferred = deferred - > next ) {
code = CORD_all ( code , compile_statement ( deferred - > defer_env , deferred - > block ) ) ;
}
return code ;
}
2024-09-11 21:13:53 -07:00
static CORD optional_into_nonnull ( type_t * t , CORD value )
2024-09-10 22:31:31 -07:00
{
2024-09-11 21:13:53 -07:00
if ( t - > tag = = OptionalType ) t = Match ( t , OptionalType ) - > type ;
2024-09-11 19:28:43 -07:00
switch ( t - > tag ) {
2024-09-11 08:08:15 -07:00
case IntType :
2024-09-11 21:13:53 -07:00
return CORD_all ( value , " .i " ) ;
2024-09-11 09:01:17 -07:00
case StructType :
2024-09-11 19:50:03 -07:00
if ( t = = THREAD_TYPE )
2024-09-11 21:13:53 -07:00
return value ;
return CORD_all ( value , " .value " ) ;
2024-09-11 08:08:15 -07:00
default :
2024-09-11 21:13:53 -07:00
return value ;
2024-09-11 08:08:15 -07:00
}
2024-09-10 22:31:31 -07:00
}
2024-09-11 20:55:49 -07:00
static CORD check_null ( type_t * t , CORD value )
2024-09-10 22:31:31 -07:00
{
t = Match ( t , OptionalType ) - > type ;
2024-09-11 09:50:46 -07:00
if ( t - > tag = = PointerType | | t - > tag = = FunctionType | | t - > tag = = CStringType
| | t - > tag = = ChannelType | | t = = THREAD_TYPE )
2024-09-11 20:55:49 -07:00
return CORD_all ( " ( " , value , " == NULL) " ) ;
2024-09-10 22:31:31 -07:00
else if ( t - > tag = = BigIntType )
2024-09-11 20:55:49 -07:00
return CORD_all ( " (( " , value , " ) . small = = 0 ) " ) ;
2024-09-10 22:31:31 -07:00
else if ( t - > tag = = ClosureType )
2024-09-11 20:55:49 -07:00
return CORD_all ( " (( " , value , " ) . fn = = NULL ) " ) ;
2024-09-10 22:31:31 -07:00
else if ( t - > tag = = NumType )
2024-09-11 20:55:49 -07:00
return CORD_all ( " isnan( " , value , " ) " ) ;
2024-09-10 22:31:31 -07:00
else if ( t - > tag = = ArrayType )
2024-09-11 20:55:49 -07:00
return CORD_all ( " (( " , value , " ) . length < 0 ) " ) ;
2024-09-10 22:31:31 -07:00
else if ( t - > tag = = TableType | | t - > tag = = SetType )
2024-09-11 20:55:49 -07:00
return CORD_all ( " (( " , value , " ) . entries . length < 0 ) " ) ;
2024-09-10 22:31:31 -07:00
else if ( t - > tag = = BoolType )
2024-09-11 20:55:49 -07:00
return CORD_all ( " (( " , value , " ) = = NULL_BOOL ) " ) ;
2024-09-10 22:31:31 -07:00
else if ( t - > tag = = TextType )
2024-09-11 20:55:49 -07:00
return CORD_all ( " (( " , value , " ) . length < 0 ) " ) ;
2024-09-15 12:33:47 -07:00
else if ( t - > tag = = IntType | | t - > tag = = ByteType | | t - > tag = = StructType )
2024-09-11 20:55:49 -07:00
return CORD_all ( " ( " , value , " ) . is_null " ) ;
2024-09-11 09:29:48 -07:00
else if ( t - > tag = = EnumType )
2024-09-11 20:55:49 -07:00
return CORD_all ( " (( " , value , " ) . tag = = 0 ) " ) ;
2024-11-17 11:49:03 -08:00
else if ( t - > tag = = MomentType )
2024-09-29 17:06:09 -07:00
return CORD_all ( " (( " , value , " ) . tv_usec < 0 ) " ) ;
2024-09-10 22:31:31 -07:00
errx ( 1 , " Optional check not implemented for: %T " , t ) ;
}
2024-11-11 21:04:56 -08:00
static CORD compile_condition ( env_t * env , ast_t * ast )
{
type_t * t = get_type ( env , ast ) ;
if ( t - > tag = = BoolType ) {
return compile ( env , ast ) ;
} else if ( t - > tag = = TextType ) {
return CORD_all ( " ( " , compile ( env , ast ) , " ).length " ) ;
} else if ( t - > tag = = ArrayType ) {
return CORD_all ( " ( " , compile ( env , ast ) , " ).length " ) ;
} else if ( t - > tag = = TableType | | t - > tag = = SetType ) {
return CORD_all ( " ( " , compile ( env , ast ) , " ).entries.length " ) ;
} else if ( t - > tag = = OptionalType ) {
return CORD_all ( " ! " , check_null ( t , compile ( env , ast ) ) ) ;
} else if ( t - > tag = = PointerType ) {
code_err ( ast , " This pointer will always be non-null, so it should not be used in a conditional. " ) ;
} else {
code_err ( ast , " %T values cannot be used for conditionals " , t ) ;
}
}
2024-02-13 11:42:33 -08:00
CORD compile_statement ( env_t * env , ast_t * ast )
2024-02-04 15:04:41 -08:00
{
switch ( ast - > tag ) {
2024-03-14 10:28:30 -07:00
case When : {
2024-05-23 21:03:46 -07:00
// Typecheck to verify exhaustiveness:
type_t * result_t = get_type ( env , ast ) ;
( void ) result_t ;
2024-03-14 10:28:30 -07:00
auto when = Match ( ast , When ) ;
type_t * subject_t = get_type ( env , when - > subject ) ;
2024-05-23 21:03:46 -07:00
2024-03-14 10:28:30 -07:00
auto enum_t = Match ( subject_t , EnumType ) ;
2024-06-16 15:09:54 -07:00
CORD code = CORD_all ( " { " , compile_type ( subject_t ) , " subject = " , compile ( env , when - > subject ) , " ; \n "
2024-08-18 17:58:36 -07:00
" switch (subject.tag) { " ) ;
2024-03-14 10:28:30 -07:00
for ( when_clause_t * clause = when - > clauses ; clause ; clause = clause - > next ) {
const char * clause_tag_name = Match ( clause - > tag_name , Var ) - > name ;
2024-09-24 11:54:22 -07:00
code = CORD_all ( code , " case " , namespace_prefix ( enum_t - > env , enum_t - > env - > namespace ) , " tag$ " , clause_tag_name , " : { \n " ) ;
2024-03-14 10:28:30 -07:00
type_t * tag_type = NULL ;
for ( tag_t * tag = enum_t - > tags ; tag ; tag = tag - > next ) {
if ( streq ( tag - > name , clause_tag_name ) ) {
tag_type = tag - > type ;
break ;
}
}
assert ( tag_type ) ;
env_t * scope = env ;
2024-05-01 10:53:51 -07:00
auto tag_struct = Match ( tag_type , StructType ) ;
if ( clause - > args & & ! clause - > args - > next & & tag_struct - > fields & & tag_struct - > fields - > next ) {
2024-11-03 08:25:18 -08:00
code = CORD_all ( code , compile_declaration ( tag_type , compile ( env , clause - > args - > ast ) ) , " = subject.$ " , clause_tag_name , " ; \n " ) ;
2024-05-01 10:53:51 -07:00
scope = fresh_scope ( scope ) ;
set_binding ( scope , Match ( clause - > args - > ast , Var ) - > name , new ( binding_t , . type = tag_type ) ) ;
} else if ( clause - > args ) {
scope = fresh_scope ( scope ) ;
ast_list_t * var = clause - > args ;
arg_t * field = tag_struct - > fields ;
while ( var | | field ) {
if ( ! var )
code_err ( clause - > tag_name , " The field %T.%s.%s wasn't accounted for " , subject_t , clause_tag_name , field - > name ) ;
if ( ! field )
code_err ( var - > ast , " This is one more field than %T has " , subject_t ) ;
2024-11-03 08:25:18 -08:00
code = CORD_all ( code , compile_declaration ( field - > type , compile ( env , var - > ast ) ) , " = subject.$ " , clause_tag_name , " .$ " , field - > name , " ; \n " ) ;
2024-05-01 10:53:51 -07:00
set_binding ( scope , Match ( var - > ast , Var ) - > name , new ( binding_t , . type = field - > type ) ) ;
var = var - > next ;
field = field - > next ;
}
2024-03-14 10:28:30 -07:00
}
2024-10-09 18:01:43 -07:00
if ( clause - > body - > tag = = Block ) {
ast_list_t * statements = Match ( clause - > body , Block ) - > statements ;
if ( ! statements | | ( statements - > ast - > tag = = Pass & & ! statements - > next ) )
code = CORD_all ( code , " break; \n } \n " ) ;
else
code = CORD_all ( code , compile_inline_block ( scope , clause - > body ) , " \n break; \n } \n " ) ;
} else {
code = CORD_all ( code , compile_statement ( scope , clause - > body ) , " \n break; \n } \n " ) ;
}
2024-03-14 10:28:30 -07:00
}
if ( when - > else_body ) {
2024-10-09 18:01:43 -07:00
if ( when - > else_body - > tag = = Block ) {
ast_list_t * statements = Match ( when - > else_body , Block ) - > statements ;
if ( ! statements | | ( statements - > ast - > tag = = Pass & & ! statements - > next ) )
code = CORD_all ( code , " default: break; " ) ;
else
code = CORD_all ( code , " default: { \n " , compile_inline_block ( env , when - > else_body ) , " \n break; \n } \n " ) ;
} else {
code = CORD_all ( code , " default: { \n " , compile_statement ( env , when - > else_body ) , " \n break; \n } \n " ) ;
}
2024-07-23 16:46:42 -07:00
} else {
code = CORD_all ( code , " default: errx(1, \" Invalid tag! \" ); \n " ) ;
2024-03-14 10:28:30 -07:00
}
code = CORD_all ( code , " \n } \n } " ) ;
return code ;
}
case DocTest : {
auto test = Match ( ast , DocTest ) ;
type_t * expr_t = get_type ( env , test - > expr ) ;
if ( ! expr_t )
code_err ( test - > expr , " I couldn't figure out the type of this expression " ) ;
2024-09-02 15:47:39 -07:00
CORD output = CORD_EMPTY ;
2024-03-14 10:28:30 -07:00
if ( test - > output ) {
const uint8_t * raw = ( const uint8_t * ) CORD_to_const_char_star ( test - > output ) ;
uint8_t buf [ 128 ] = { 0 } ;
size_t norm_len = sizeof ( buf ) ;
uint8_t * norm = u8_normalize ( UNINORM_NFD , ( uint8_t * ) raw , strlen ( ( char * ) raw ) + 1 , buf , & norm_len ) ;
assert ( norm [ norm_len - 1 ] = = 0 ) ;
output = CORD_from_char_star ( ( char * ) norm ) ;
if ( norm & & norm ! = buf ) free ( norm ) ;
}
if ( test - > expr - > tag = = Declare ) {
auto decl = Match ( test - > expr , Declare ) ;
2024-09-17 21:30:05 -07:00
const char * varname = Match ( decl - > var , Var ) - > name ;
if ( streq ( varname , " _ " ) )
return compile_statement ( env , WrapAST ( ast , DocTest , . expr = decl - > value , . output = test - > output , . skip_source = test - > skip_source ) ) ;
2024-09-04 10:49:41 -07:00
CORD var = CORD_all ( " $ " , Match ( decl - > var , Var ) - > name ) ;
type_t * t = get_type ( env , decl - > value ) ;
CORD val_code = compile_maybe_incref ( env , decl - > value ) ;
if ( t - > tag = = FunctionType ) {
assert ( promote ( env , & val_code , t , Type ( ClosureType , t ) ) ) ;
t = Type ( ClosureType , t ) ;
2024-04-21 08:22:11 -07:00
}
2024-09-04 10:49:41 -07:00
return CORD_asprintf (
" %r; \n "
" test((%r = %r), %r, %r, %ld, %ld); \n " ,
compile_declaration ( t , var ) ,
var , val_code ,
compile_type_info ( env , get_type ( env , decl - > value ) ) ,
CORD_quoted ( output ) ,
( int64_t ) ( test - > expr - > start - test - > expr - > file - > text ) ,
( int64_t ) ( test - > expr - > end - test - > expr - > file - > text ) ) ;
2024-03-14 10:28:30 -07:00
} else if ( test - > expr - > tag = = Assign ) {
auto assign = Match ( test - > expr , Assign ) ;
2024-09-03 20:32:02 -07:00
if ( ! assign - > targets - > next & & assign - > targets - > ast - > tag = = Var & & is_idempotent ( assign - > targets - > ast ) ) {
2024-03-14 10:28:30 -07:00
// Common case: assigning to one variable:
2024-05-23 21:09:39 -07:00
type_t * lhs_t = get_type ( env , assign - > targets - > ast ) ;
env_t * val_scope = with_enum_scope ( env , lhs_t ) ;
2024-11-21 10:00:53 -08:00
CORD value = compile_to_type ( val_scope , assign - > values - > ast , lhs_t ) ;
2024-05-13 21:30:57 -07:00
return CORD_asprintf (
2024-09-04 02:01:21 -07:00
" test((%r), %r, %r, %ld, %ld); " ,
2024-05-23 21:09:39 -07:00
compile_assignment ( env , assign - > targets - > ast , value ) ,
compile_type_info ( env , lhs_t ) ,
2024-09-02 15:47:39 -07:00
CORD_quoted ( test - > output ) ,
2024-03-14 10:28:30 -07:00
( int64_t ) ( test - > expr - > start - test - > expr - > file - > text ) ,
( int64_t ) ( test - > expr - > end - test - > expr - > file - > text ) ) ;
} else {
// Multi-assign or assignment to potentially non-idempotent targets
if ( test - > output & & assign - > targets - > next )
code_err ( ast , " Sorry, but doctesting with '=' is not supported for multi-assignments " ) ;
2024-05-13 21:30:57 -07:00
CORD code = " test(({ // Assignment \n " ;
2024-09-04 02:01:21 -07:00
2024-03-14 10:28:30 -07:00
int64_t i = 1 ;
2024-04-23 09:54:56 -07:00
for ( ast_list_t * target = assign - > targets , * value = assign - > values ; target & & value ; target = target - > next , value = value - > next ) {
type_t * target_type = get_type ( env , target - > ast ) ;
2024-05-15 10:42:45 -07:00
env_t * val_scope = with_enum_scope ( env , target_type ) ;
2024-11-21 10:00:53 -08:00
CORD val_code = compile_to_type ( val_scope , value - > ast , target_type ) ;
2024-06-16 15:09:54 -07:00
CORD_appendf ( & code , " %r $%ld = %r; \n " , compile_type ( target_type ) , i + + , val_code ) ;
2024-04-23 09:54:56 -07:00
}
2024-03-14 10:28:30 -07:00
i = 1 ;
2024-03-22 10:53:23 -07:00
for ( ast_list_t * target = assign - > targets ; target ; target = target - > next )
2024-09-03 20:32:02 -07:00
code = CORD_all ( code , compile_assignment ( env , target - > ast , CORD_asprintf ( " $%ld " , i + + ) ) , " ; \n " ) ;
2024-03-14 10:28:30 -07:00
2024-09-04 02:01:21 -07:00
CORD_appendf ( & code , " $1; }), %r, %r, %ld, %ld); " ,
2024-03-14 10:28:30 -07:00
compile_type_info ( env , get_type ( env , assign - > targets - > ast ) ) ,
2024-09-02 15:47:39 -07:00
CORD_quoted ( test - > output ) ,
2024-03-14 10:28:30 -07:00
( int64_t ) ( test - > expr - > start - test - > expr - > file - > text ) ,
( int64_t ) ( test - > expr - > end - test - > expr - > file - > text ) ) ;
2024-05-13 21:30:57 -07:00
return code ;
2024-03-14 10:28:30 -07:00
}
2024-07-01 08:31:50 -07:00
} else if ( test - > expr - > tag = = UpdateAssign ) {
2024-09-04 02:01:21 -07:00
type_t * lhs_t = get_type ( env , Match ( test - > expr , UpdateAssign ) - > lhs ) ;
2024-07-01 08:31:50 -07:00
return CORD_asprintf (
2024-09-04 02:01:21 -07:00
" test(({%r %r;}), %r, %r, %ld, %ld); " ,
2024-07-01 08:31:50 -07:00
compile_statement ( env , test - > expr ) ,
compile_lvalue ( env , Match ( test - > expr , UpdateAssign ) - > lhs ) ,
2024-09-04 02:01:21 -07:00
compile_type_info ( env , lhs_t ) ,
2024-09-02 15:47:39 -07:00
CORD_quoted ( test - > output ) ,
2024-07-01 08:31:50 -07:00
( int64_t ) ( test - > expr - > start - test - > expr - > file - > text ) ,
( int64_t ) ( test - > expr - > end - test - > expr - > file - > text ) ) ;
2024-07-13 15:43:50 -07:00
} else if ( expr_t - > tag = = VoidType | | expr_t - > tag = = AbortType | | expr_t - > tag = = ReturnType ) {
2024-03-14 10:28:30 -07:00
return CORD_asprintf (
2024-09-04 02:01:21 -07:00
" test(({%r NULL;}), NULL, NULL, %ld, %ld); " ,
2024-03-14 10:28:30 -07:00
compile_statement ( env , test - > expr ) ,
( int64_t ) ( test - > expr - > start - test - > expr - > file - > text ) ,
( int64_t ) ( test - > expr - > end - test - > expr - > file - > text ) ) ;
} else {
return CORD_asprintf (
2024-09-03 20:32:02 -07:00
" test(%r, %r, %r, %ld, %ld); " ,
2024-09-04 02:01:21 -07:00
compile ( env , test - > expr ) ,
2024-03-14 10:28:30 -07:00
compile_type_info ( env , expr_t ) ,
2024-09-02 15:47:39 -07:00
CORD_quoted ( output ) ,
2024-03-14 10:28:30 -07:00
( int64_t ) ( test - > expr - > start - test - > expr - > file - > text ) ,
( int64_t ) ( test - > expr - > end - test - > expr - > file - > text ) ) ;
}
}
case Declare : {
auto decl = Match ( ast , Declare ) ;
2024-09-27 11:56:24 -07:00
if ( streq ( Match ( decl - > var , Var ) - > name , " _ " ) ) { // Explicit discard
2024-09-17 21:30:05 -07:00
return CORD_all ( " (void) " , compile(env, decl->value), " ; " );
2024-03-19 11:22:03 -07:00
} else {
type_t * t = get_type ( env , decl - > value ) ;
2024-07-13 15:43:50 -07:00
if ( t - > tag = = AbortType | | t - > tag = = VoidType | | t - > tag = = ReturnType )
2024-03-19 11:22:03 -07:00
code_err ( ast , " You can't declare a variable with a %T value " , t ) ;
2024-08-20 21:48:08 -07:00
CORD val_code = compile_maybe_incref ( env , decl - > value ) ;
if ( t - > tag = = FunctionType ) {
assert ( promote ( env , & val_code , t , Type ( ClosureType , t ) ) ) ;
t = Type ( ClosureType , t ) ;
}
return CORD_all ( compile_declaration ( t , CORD_cat ( " $ " , Match ( decl - > var , Var ) - > name ) ) , " = " , val_code , " ; " );
2024-03-19 11:22:03 -07:00
}
2024-03-14 10:28:30 -07:00
}
case Assign : {
auto assign = Match ( ast , Assign ) ;
2024-05-15 10:39:35 -07:00
// Single assignment, no temp vars needed:
2024-09-03 20:32:02 -07:00
if ( assign - > targets & & ! assign - > targets - > next & & is_idempotent ( assign - > targets - > ast ) ) {
2024-05-15 10:39:35 -07:00
type_t * lhs_t = get_type ( env , assign - > targets - > ast ) ;
2024-05-15 10:40:27 -07:00
env_t * val_env = with_enum_scope ( env , lhs_t ) ;
2024-11-21 10:00:53 -08:00
CORD val = compile_to_type ( val_env , assign - > values - > ast , lhs_t ) ;
2024-09-03 20:32:02 -07:00
return CORD_all ( compile_assignment ( env , assign - > targets - > ast , val ) , " ; \ n " );
2024-05-15 10:39:35 -07:00
}
2024-03-14 10:28:30 -07:00
CORD code = " { // Assignment \n " ;
int64_t i = 1 ;
2024-05-15 10:39:35 -07:00
for ( ast_list_t * value = assign - > values , * target = assign - > targets ; value & & target ; value = value - > next , target = target - > next ) {
type_t * lhs_t = get_type ( env , target - > ast ) ;
2024-05-15 10:40:27 -07:00
env_t * val_env = with_enum_scope ( env , lhs_t ) ;
2024-11-21 10:00:53 -08:00
CORD val = compile_to_type ( val_env , value - > ast , lhs_t ) ;
2024-06-16 15:09:54 -07:00
CORD_appendf ( & code , " %r $%ld = %r; \n " , compile_type ( lhs_t ) , i + + , val ) ;
2024-05-15 10:39:35 -07:00
}
2024-03-14 10:28:30 -07:00
i = 1 ;
for ( ast_list_t * target = assign - > targets ; target ; target = target - > next ) {
2024-09-03 20:32:02 -07:00
code = CORD_all ( code , compile_assignment ( env , target - > ast , CORD_asprintf ( " $%ld " , i + + ) ) , " ; \n " ) ;
2024-03-14 10:28:30 -07:00
}
return CORD_cat ( code , " \n } " ) ;
}
case UpdateAssign : {
auto update = Match ( ast , UpdateAssign ) ;
2024-03-24 11:28:20 -07:00
CORD lhs = compile_lvalue ( env , update - > lhs ) ;
2024-07-01 08:20:41 -07:00
2024-08-10 17:50:15 -07:00
CORD method_call = compile_math_method ( env , update - > op , update - > lhs , update - > rhs , get_type ( env , update - > lhs ) ) ;
2024-07-01 08:20:41 -07:00
if ( method_call )
return CORD_all ( lhs , " = " , method_call , " ; " ) ;
2024-03-14 10:28:30 -07:00
CORD rhs = compile ( env , update - > rhs ) ;
type_t * lhs_t = get_type ( env , update - > lhs ) ;
type_t * rhs_t = get_type ( env , update - > rhs ) ;
2024-08-17 20:43:30 -07:00
if ( ! promote ( env , & rhs , rhs_t , lhs_t ) ) {
2024-11-03 22:12:37 -08:00
if ( update - > rhs - > tag = = Int & & ( lhs_t - > tag = = IntType | | lhs_t - > tag = = ByteType ) )
2024-08-17 20:43:30 -07:00
rhs = compile_int_to_type ( env , update - > rhs , lhs_t ) ;
2024-08-17 23:58:44 -07:00
else if ( ! ( lhs_t - > tag = = ArrayType & & promote ( env , & rhs , rhs_t , Match ( lhs_t , ArrayType ) - > item_type ) ) )
2024-08-17 20:43:30 -07:00
code_err ( ast , " I can't do operations between %T and %T " , lhs_t , rhs_t ) ;
}
2024-07-01 08:29:28 -07:00
2024-03-14 10:28:30 -07:00
switch ( update - > op ) {
2024-07-01 08:43:18 -07:00
case BINOP_MULT :
2024-09-15 12:33:47 -07:00
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = NumType & & lhs_t - > tag ! = ByteType )
2024-07-01 08:43:18 -07:00
code_err ( ast , " I can't do a multiply assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
return CORD_all ( lhs , " *= " , rhs , " ; " ) ;
case BINOP_DIVIDE :
2024-09-15 12:33:47 -07:00
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = NumType & & lhs_t - > tag ! = ByteType )
2024-07-01 08:43:18 -07:00
code_err ( ast , " I can't do a divide assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
return CORD_all ( lhs , " /= " , rhs , " ; " ) ;
case BINOP_MOD :
2024-09-15 12:33:47 -07:00
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = NumType & & lhs_t - > tag ! = ByteType )
2024-07-01 08:43:18 -07:00
code_err ( ast , " I can't do a mod assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
return CORD_all ( lhs , " = " , lhs , " % " , rhs ) ;
case BINOP_MOD1 :
2024-09-15 12:33:47 -07:00
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = NumType & & lhs_t - > tag ! = ByteType )
2024-07-01 08:43:18 -07:00
code_err ( ast , " I can't do a mod assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
2024-08-10 17:40:41 -07:00
return CORD_all ( lhs , " = ((( " , lhs , " ) - 1 ) % " , rhs, " ) + 1 ; " );
2024-07-01 08:43:18 -07:00
case BINOP_PLUS :
2024-09-15 12:33:47 -07:00
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = NumType & & lhs_t - > tag ! = ByteType )
2024-07-01 08:43:18 -07:00
code_err ( ast , " I can't do an addition assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
return CORD_all ( lhs , " += " , rhs , " ; " ) ;
case BINOP_MINUS :
2024-09-15 12:33:47 -07:00
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = NumType & & lhs_t - > tag ! = ByteType )
2024-07-01 08:43:18 -07:00
code_err ( ast , " I can't do a subtraction assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
return CORD_all ( lhs , " -= " , rhs , " ; " ) ;
2024-03-14 10:28:30 -07:00
case BINOP_POWER : {
if ( lhs_t - > tag ! = NumType )
code_err ( ast , " '^=' is only supported for Num types " ) ;
2024-08-18 08:49:51 -07:00
if ( lhs_t - > tag = = NumType & & Match ( lhs_t , NumType ) - > bits = = TYPE_NBITS32 )
2024-03-18 15:06:20 -07:00
return CORD_all ( lhs , " = powf( " , lhs , " , " , rhs , " ) ; " );
2024-03-14 10:28:30 -07:00
else
2024-03-18 15:06:20 -07:00
return CORD_all ( lhs , " = pow( " , lhs , " , " , rhs , " ) ; " );
2024-03-14 10:28:30 -07:00
}
2024-07-01 08:43:18 -07:00
case BINOP_LSHIFT :
2024-09-15 12:33:47 -07:00
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = ByteType )
2024-07-01 08:43:18 -07:00
code_err ( ast , " I can't do a shift assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
return CORD_all ( lhs , " <<= " , rhs , " ; " ) ;
case BINOP_RSHIFT :
2024-09-15 12:33:47 -07:00
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = ByteType )
2024-07-01 08:43:18 -07:00
code_err ( ast , " I can't do a shift assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
return CORD_all ( lhs , " >>= " , rhs , " ; " ) ;
2024-11-03 13:06:26 -08:00
case BINOP_ULSHIFT :
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = ByteType )
code_err ( ast , " I can't do a shift assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
return CORD_all ( " { " , compile_unsigned_type ( lhs_t ) , " *dest = (void*)&( " , lhs , " ) ; * dest < < = " , rhs, " ; } " );
case BINOP_URSHIFT :
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = ByteType )
code_err ( ast , " I can't do a shift assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
return CORD_all ( " { " , compile_unsigned_type ( lhs_t ) , " *dest = (void*)&( " , lhs , " ) ; * dest > > = " , rhs, " ; } " );
2024-03-14 10:28:30 -07:00
case BINOP_AND : {
2024-08-17 20:43:30 -07:00
if ( lhs_t - > tag = = BoolType )
2024-07-01 08:20:41 -07:00
return CORD_all ( " if ( " , lhs , " ) " , lhs, " = " , rhs, " ; " );
2024-09-15 12:33:47 -07:00
else if ( lhs_t - > tag = = IntType | | lhs_t - > tag = = ByteType )
2024-07-01 08:20:41 -07:00
return CORD_all ( lhs , " &= " , rhs , " ; " ) ;
2024-09-27 10:13:30 -07:00
else if ( lhs_t - > tag = = OptionalType )
return CORD_all ( " if (!( " , check_null ( lhs_t , lhs ) , " )) " , lhs , " = " , promote_to_optional ( rhs_t , rhs ) , " ; " );
2024-03-14 10:28:30 -07:00
else
2024-08-17 20:43:30 -07:00
code_err ( ast , " 'or=' is not implemented for %T types " , lhs_t ) ;
2024-03-14 10:28:30 -07:00
}
case BINOP_OR : {
2024-08-17 20:43:30 -07:00
if ( lhs_t - > tag = = BoolType )
2024-07-01 08:20:41 -07:00
return CORD_all ( " if (!( " , lhs , " ) ) " , lhs, " = " , rhs, " ; " );
2024-09-15 12:33:47 -07:00
else if ( lhs_t - > tag = = IntType | | lhs_t - > tag = = ByteType )
2024-07-01 08:20:41 -07:00
return CORD_all ( lhs , " |= " , rhs , " ; " ) ;
2024-09-27 10:13:30 -07:00
else if ( lhs_t - > tag = = OptionalType )
return CORD_all ( " if ( " , check_null ( lhs_t , lhs ) , " ) " , lhs , " = " , promote_to_optional ( rhs_t , rhs ) , " ; " );
2024-03-14 10:28:30 -07:00
else
2024-08-17 20:43:30 -07:00
code_err ( ast , " 'or=' is not implemented for %T types " , lhs_t ) ;
2024-03-14 10:28:30 -07:00
}
2024-07-01 08:43:18 -07:00
case BINOP_XOR :
2024-09-15 12:33:47 -07:00
if ( lhs_t - > tag ! = IntType & & lhs_t - > tag ! = BoolType & & lhs_t - > tag ! = ByteType )
2024-07-01 08:43:18 -07:00
code_err ( ast , " I can't do an xor assignment with this operator between %T and %T " , lhs_t , rhs_t ) ;
return CORD_all ( lhs , " ^= " , rhs , " ; " ) ;
2024-03-14 10:28:30 -07:00
case BINOP_CONCAT : {
2024-08-17 20:43:30 -07:00
if ( lhs_t - > tag = = TextType ) {
2024-09-03 20:35:42 -07:00
return CORD_all ( lhs , " = Texts( " , lhs , " , " , rhs , " ) ; " );
2024-08-17 20:43:30 -07:00
} else if ( lhs_t - > tag = = ArrayType ) {
2024-10-08 18:10:36 -07:00
CORD padded_item_size = CORD_asprintf ( " %ld " , type_size ( Match ( lhs_t , ArrayType ) - > item_type ) ) ;
2024-03-14 10:28:30 -07:00
if ( promote ( env , & rhs , rhs_t , Match ( lhs_t , ArrayType ) - > item_type ) ) {
// arr ++= item
if ( update - > lhs - > tag = = Var )
2024-08-12 22:30:25 -07:00
return CORD_all ( " Array$insert(& " , lhs , " , stack( " , rhs , " ) , I ( 0 ) , " , padded_item_size, " ) ; " );
2024-03-14 10:28:30 -07:00
else
2024-08-03 12:33:50 -07:00
return CORD_all ( lhs , " Array$concat( " , lhs , " , Array( " , rhs , " ) , " , padded_item_size, " ) ; " );
2024-03-14 10:28:30 -07:00
} else {
// arr ++= [...]
if ( update - > lhs - > tag = = Var )
2024-08-12 22:30:25 -07:00
return CORD_all ( " Array$insert_all(& " , lhs , " , " , rhs , " , I(0) , " , padded_item_size, " ) ; " );
2024-03-14 10:28:30 -07:00
else
2024-08-03 12:33:50 -07:00
return CORD_all ( lhs , " Array$concat( " , lhs , " , " , rhs , " , " , padded_item_size , " ) ; " );
2024-03-14 10:28:30 -07:00
}
} else {
2024-08-17 20:43:30 -07:00
code_err ( ast , " '++=' is not implemented for %T types " , lhs_t ) ;
2024-03-14 10:28:30 -07:00
}
}
default : code_err ( ast , " Update assignments are not implemented for this operation " ) ;
}
}
case StructDef : {
compile_struct_def ( env , ast ) ;
return CORD_EMPTY ;
}
case EnumDef : {
compile_enum_def ( env , ast ) ;
return CORD_EMPTY ;
}
case LangDef : {
auto def = Match ( ast , LangDef ) ;
2024-09-30 11:39:30 -07:00
CORD_appendf ( & env - > code - > typeinfos , " public const TypeInfo_t %r%s = {%zu, %zu, {.tag=TextInfo, .TextInfo={%r}}}; \n " ,
2024-09-24 11:54:22 -07:00
namespace_prefix ( env , env - > namespace ) , def - > name , sizeof ( Text_t ) , __alignof__ ( Text_t ) ,
2024-09-02 15:47:39 -07:00
CORD_quoted ( def - > name ) ) ;
2024-03-14 10:28:30 -07:00
compile_namespace ( env , def - > name , def - > namespace ) ;
return CORD_EMPTY ;
}
case FunctionDef : {
auto fndef = Match ( ast , FunctionDef ) ;
2024-04-17 10:44:01 -07:00
bool is_private = Match ( fndef - > name , Var ) - > name [ 0 ] = = ' _ ' ;
2024-06-16 13:08:35 -07:00
CORD name = compile ( env , fndef - > name ) ;
2024-03-14 10:28:30 -07:00
type_t * ret_t = fndef - > ret_type ? parse_type_ast ( env , fndef - > ret_type ) : Type ( VoidType ) ;
CORD arg_signature = " ( " ;
2024-09-27 11:22:36 -07:00
Table_t used_names = { } ;
2024-03-14 10:28:30 -07:00
for ( arg_ast_t * arg = fndef - > args ; arg ; arg = arg - > next ) {
type_t * arg_type = get_arg_ast_type ( env , arg ) ;
2024-06-16 15:09:54 -07:00
arg_signature = CORD_cat ( arg_signature , compile_declaration ( arg_type , CORD_cat ( " $ " , arg - > name ) ) ) ;
2024-03-14 10:28:30 -07:00
if ( arg - > next ) arg_signature = CORD_cat ( arg_signature , " , " ) ;
2024-09-27 11:22:36 -07:00
if ( Table $ str_get ( used_names , arg - > name ) )
code_err ( ast , " The argument name '%s' is used more than once " , arg - > name ) ;
Table $ str_set ( & used_names , arg - > name , arg - > name ) ;
2024-03-14 10:28:30 -07:00
}
arg_signature = CORD_cat ( arg_signature , " ) " ) ;
2024-06-16 15:09:54 -07:00
CORD ret_type_code = compile_type ( ret_t ) ;
2024-03-14 10:28:30 -07:00
2024-04-17 10:44:01 -07:00
if ( is_private )
2024-03-14 10:28:30 -07:00
env - > code - > staticdefs = CORD_all ( env - > code - > staticdefs , " static " , ret_type_code , " " , name , arg_signature , " ; \n " ) ;
CORD code ;
if ( fndef - > cache ) {
code = CORD_all ( " static " , ret_type_code , " " , name , " $uncached " , arg_signature ) ;
} else {
code = CORD_all ( ret_type_code , " " , name , arg_signature ) ;
if ( fndef - > is_inline )
2024-10-30 11:02:34 -07:00
code = CORD_cat ( " INLINE " , code ) ;
2024-04-17 10:44:01 -07:00
if ( ! is_private )
2024-03-14 10:28:30 -07:00
code = CORD_cat ( " public " , code ) ;
}
2024-04-10 08:49:43 -07:00
env_t * body_scope = fresh_scope ( env ) ;
2024-07-04 15:09:33 -07:00
body_scope - > deferred = NULL ;
2024-06-16 12:12:00 -07:00
body_scope - > namespace = NULL ;
2024-03-14 10:28:30 -07:00
for ( arg_ast_t * arg = fndef - > args ; arg ; arg = arg - > next ) {
type_t * arg_type = get_arg_ast_type ( env , arg ) ;
2024-04-16 10:50:07 -07:00
set_binding ( body_scope , arg - > name , new ( binding_t , . type = arg_type , . code = CORD_cat ( " $ " , arg - > name ) ) ) ;
2024-03-14 10:28:30 -07:00
}
2024-03-15 10:35:30 -07:00
fn_ctx_t fn_ctx = ( fn_ctx_t ) {
2024-07-14 11:13:23 -07:00
. parent = NULL ,
2024-03-14 10:28:30 -07:00
. return_type = ret_t ,
. closure_scope = NULL ,
. closed_vars = NULL ,
} ;
body_scope - > fn_ctx = & fn_ctx ;
2024-07-13 15:43:50 -07:00
type_t * body_type = get_type ( body_scope , fndef - > body ) ;
if ( ret_t - > tag ! = VoidType & & ret_t - > tag ! = AbortType & & body_type - > tag ! = AbortType & & body_type - > tag ! = ReturnType )
2024-07-04 15:27:08 -07:00
code_err ( ast , " This function can reach the end without returning a %T value! " , ret_t ) ;
2024-03-14 10:28:30 -07:00
CORD body = compile_statement ( body_scope , fndef - > body ) ;
2024-08-17 11:41:31 -07:00
if ( streq ( Match ( fndef - > name , Var ) - > name , " main " ) )
2024-09-24 11:54:22 -07:00
body = CORD_all ( " $ " , env - > namespace - > name , " $$initialize(); \n " , body ) ;
2024-03-14 10:28:30 -07:00
if ( CORD_fetch ( body , 0 ) ! = ' { ' )
body = CORD_asprintf ( " { \n %r \n } " , body ) ;
env - > code - > funcs = CORD_all ( env - > code - > funcs , code , " " , body , " \n " ) ;
2024-08-15 22:46:05 -07:00
if ( fndef - > cache & & fndef - > args = = NULL ) { // no-args cache just uses a static var
CORD wrapper = CORD_all (
is_private ? CORD_EMPTY : " public " , ret_type_code , " " , name , " (void) { \n "
" static " , compile_declaration ( ret_t , " cached_result " ) , " ; \n " ,
" static bool initialized = false; \n " ,
" if (!initialized) { \n "
" \t cached_result = " , name , " $uncached(); \n " ,
" \t initialized = true; \n " ,
" } \n " ,
" return cached_result; \n "
" } \n " ) ;
env - > code - > funcs = CORD_cat ( env - > code - > funcs , wrapper ) ;
} else if ( fndef - > cache & & fndef - > cache - > tag = = Int ) {
2024-11-09 12:11:11 -08:00
OptionalInt64_t cache_size = Int64 $ parse ( Text $ from_str ( Match ( fndef - > cache , Int ) - > str ) ) ;
2024-07-23 16:46:42 -07:00
const char * arg_type_name = heap_strf ( " %s$args " , Match ( fndef - > name , Var ) - > name ) ;
2024-03-14 10:28:30 -07:00
ast_t * args_def = FakeAST ( StructDef , . name = arg_type_name , . fields = fndef - > args ) ;
2024-05-18 22:46:30 -07:00
prebind_statement ( env , args_def ) ;
2024-03-14 10:28:30 -07:00
bind_statement ( env , args_def ) ;
( void ) compile_statement ( env , args_def ) ;
2024-03-29 09:54:31 -07:00
type_t * args_t = Table $ str_get ( * env - > types , arg_type_name ) ;
2024-03-14 10:28:30 -07:00
assert ( args_t ) ;
CORD all_args = CORD_EMPTY ;
for ( arg_ast_t * arg = fndef - > args ; arg ; arg = arg - > next )
2024-04-16 10:50:07 -07:00
all_args = CORD_all ( all_args , " $ " , arg - > name , arg - > next ? " , " : CORD_EMPTY ) ;
2024-03-14 10:28:30 -07:00
CORD pop_code = CORD_EMPTY ;
2024-09-11 11:53:48 -07:00
if ( fndef - > cache - > tag = = Int & & ! cache_size . is_null & & cache_size . i > 0 ) {
pop_code = CORD_all ( " if (cache.entries.length > " , CORD_asprintf ( " %ld " , cache_size . i ) ,
2024-11-03 19:37:48 -08:00
" ) Table$remove(&cache, cache.entries.data + cache.entries.stride*RNG$int64(default_rng, I(0), I(cache.entries.length-1)), table_type); \n " ) ;
2024-03-14 10:28:30 -07:00
}
2024-09-17 22:01:15 -07:00
CORD arg_typedef = compile_struct_header ( env , args_def ) ;
2024-06-06 13:28:53 -07:00
env - > code - > local_typedefs = CORD_all ( env - > code - > local_typedefs , arg_typedef ) ;
2024-03-14 10:28:30 -07:00
CORD wrapper = CORD_all (
2024-04-17 10:44:01 -07:00
is_private ? CORD_EMPTY : " public " , ret_type_code , " " , name , arg_signature , " { \n "
2024-09-05 11:57:31 -07:00
" static Table_t cache = {}; \n " ,
2024-06-16 15:09:54 -07:00
compile_type ( args_t ) , " args = { " , all_args , " }; \n "
2024-09-30 11:39:30 -07:00
" const TypeInfo_t *table_type = Table$info( " , compile_type_info ( env , args_t ) , " , " , compile_type_info ( env , ret_t ) , " ); \n " ,
2024-09-10 22:31:31 -07:00
compile_declaration ( Type ( PointerType , . pointed = ret_t ) , " cached " ) , " = Table$get_raw(cache, &args, table_type); \n "
2024-04-16 10:50:07 -07:00
" if (cached) return *cached; \n " ,
2024-08-15 22:46:05 -07:00
compile_declaration ( ret_t , " ret " ) , " = " , name , " $uncached( " , all_args , " ); \n " ,
2024-03-14 10:28:30 -07:00
pop_code ,
2024-04-16 10:50:07 -07:00
" Table$set(&cache, &args, &ret, table_type); \n "
" return ret; \n "
2024-03-14 10:28:30 -07:00
" } \n " ) ;
env - > code - > funcs = CORD_cat ( env - > code - > funcs , wrapper ) ;
}
2024-09-12 20:41:32 -07:00
CORD text = CORD_all ( " func " , Match ( fndef - > name , Var ) - > name , " ( " ) ;
for ( arg_ast_t * arg = fndef - > args ; arg ; arg = arg - > next ) {
text = CORD_cat ( text , type_to_cord ( get_arg_ast_type ( env , arg ) ) ) ;
if ( arg - > next ) text = CORD_cat ( text , " , " ) ;
}
if ( ret_t & & ret_t - > tag ! = VoidType )
text = CORD_all ( text , " )-> " , type_to_cord ( ret_t ) ) ;
else
text = CORD_all ( text , " ) " ) ;
2024-10-30 11:02:34 -07:00
if ( ! fndef - > is_inline ) {
env - > code - > function_naming = CORD_all (
env - > code - > function_naming ,
CORD_asprintf ( " register_function(%r, Text( \" %r [%s.tm:%ld] \" )); \n " ,
name , text , file_base_name ( ast - > file - > filename ) , get_line_number ( ast - > file , ast - > start ) ) ) ;
}
2024-03-14 10:28:30 -07:00
return CORD_EMPTY ;
}
case Skip : {
2024-03-15 10:35:30 -07:00
const char * target = Match ( ast , Skip ) - > target ;
for ( loop_ctx_t * ctx = env - > loop_ctx ; ctx ; ctx = ctx - > next ) {
2024-07-13 14:17:58 -07:00
bool matched = ! target | | CORD_cmp ( target , ctx - > loop_name ) = = 0 ;
for ( ast_list_t * var = ctx - > loop_vars ; var & & ! matched ; var = var - > next )
matched = ( CORD_cmp ( target , Match ( var - > ast , Var ) - > name ) = = 0 ) ;
if ( matched ) {
2024-03-15 10:35:30 -07:00
if ( ! ctx - > skip_label ) {
static int64_t skip_label_count = 1 ;
CORD_sprintf ( & ctx - > skip_label , " skip_%ld " , skip_label_count ) ;
+ + skip_label_count ;
}
2024-07-04 15:00:01 -07:00
CORD code = CORD_EMPTY ;
for ( deferral_t * deferred = env - > deferred ; deferred & & deferred ! = ctx - > deferred ; deferred = deferred - > next )
code = CORD_all ( code , compile_statement ( deferred - > defer_env , deferred - > block ) ) ;
return CORD_all ( code , " goto " , ctx - > skip_label , " ; " ) ;
2024-03-15 10:35:30 -07:00
}
}
if ( env - > loop_ctx )
code_err ( ast , " This 'skip' is not inside any loop " ) ;
else if ( target )
code_err ( ast , " No loop target named '%s' was found " , target ) ;
else
2024-09-05 08:51:07 -07:00
return " continue; " ;
2024-03-14 10:28:30 -07:00
}
case Stop : {
2024-03-15 10:35:30 -07:00
const char * target = Match ( ast , Stop ) - > target ;
for ( loop_ctx_t * ctx = env - > loop_ctx ; ctx ; ctx = ctx - > next ) {
2024-07-13 14:17:58 -07:00
bool matched = ! target | | CORD_cmp ( target , ctx - > loop_name ) = = 0 ;
for ( ast_list_t * var = ctx - > loop_vars ; var & & ! matched ; var = var - > next )
matched = ( CORD_cmp ( target , Match ( var - > ast , Var ) - > name ) = = 0 ) ;
if ( matched ) {
2024-03-15 10:35:30 -07:00
if ( ! ctx - > stop_label ) {
static int64_t stop_label_count = 1 ;
CORD_sprintf ( & ctx - > stop_label , " stop_%ld " , stop_label_count ) ;
+ + stop_label_count ;
}
2024-07-04 15:00:01 -07:00
CORD code = CORD_EMPTY ;
for ( deferral_t * deferred = env - > deferred ; deferred & & deferred ! = ctx - > deferred ; deferred = deferred - > next )
code = CORD_all ( code , compile_statement ( deferred - > defer_env , deferred - > block ) ) ;
return CORD_all ( code , " goto " , ctx - > stop_label , " ; " ) ;
2024-03-15 10:35:30 -07:00
}
}
if ( env - > loop_ctx )
code_err ( ast , " This 'stop' is not inside any loop " ) ;
else if ( target )
code_err ( ast , " No loop target named '%s' was found " , target ) ;
else
2024-09-05 08:51:07 -07:00
return " break; " ;
2024-03-14 10:28:30 -07:00
}
case Pass : return " ; " ;
2024-07-04 15:00:01 -07:00
case Defer : {
ast_t * body = Match ( ast , Defer ) - > body ;
2024-09-05 11:57:31 -07:00
Table_t * closed_vars = get_closed_vars ( env , FakeAST ( Lambda , . args = NULL , . body = body ) ) ;
2024-07-04 15:00:01 -07:00
static int defer_id = 0 ;
env_t * defer_env = fresh_scope ( env ) ;
CORD code = CORD_EMPTY ;
for ( int64_t i = 1 ; i < = Table $ length ( * closed_vars ) ; i + + ) {
struct { const char * name ; binding_t * b ; } * entry = Table $ entry ( * closed_vars , i ) ;
if ( entry - > b - > type - > tag = = ModuleType )
continue ;
2024-10-08 21:45:52 -07:00
if ( CORD_ncmp ( entry - > b - > code , 0 , " userdata-> " , 0 , strlen ( " userdata-> " ) ) = = 0 ) {
set_binding ( defer_env , entry - > name , entry - > b ) ;
} else {
CORD defer_name = CORD_asprintf ( " defer$%d$%s " , + + defer_id , entry - > name ) ;
code = CORD_all (
code , compile_declaration ( entry - > b - > type , defer_name ) , " = " , entry - > b - > code , " ; \n " ) ;
set_binding ( defer_env , entry - > name , new ( binding_t , . type = entry - > b - > type , . code = defer_name ) ) ;
}
2024-07-14 11:13:23 -07:00
if ( env - > fn_ctx - > closed_vars )
Table $ str_set ( env - > fn_ctx - > closed_vars , entry - > name , entry - > b ) ;
2024-07-04 15:00:01 -07:00
}
env - > deferred = new ( deferral_t , . defer_env = defer_env , . block = body , . next = env - > deferred ) ;
return code ;
}
2024-07-04 13:23:05 -07:00
case PrintStatement : {
ast_list_t * to_print = Match ( ast , PrintStatement ) - > to_print ;
if ( ! to_print )
return CORD_EMPTY ;
2024-09-04 10:37:00 -07:00
CORD code = " say( " ;
if ( to_print - > next ) code = CORD_all ( code , " Texts( " ) ;
2024-07-04 13:23:05 -07:00
for ( ast_list_t * chunk = to_print ; chunk ; chunk = chunk - > next ) {
2024-07-04 13:25:06 -07:00
if ( chunk - > ast - > tag = = TextLiteral ) {
2024-07-04 13:23:05 -07:00
code = CORD_cat ( code , compile ( env , chunk - > ast ) ) ;
} else {
code = CORD_cat ( code , compile_string ( env , chunk - > ast , " USE_COLOR " ) ) ;
}
if ( chunk - > next ) code = CORD_cat ( code , " , " ) ;
}
2024-09-04 10:37:00 -07:00
if ( to_print - > next ) code = CORD_all ( code , " ) " ) ;
return CORD_cat ( code , " , yes) ; " );
2024-07-04 13:23:05 -07:00
}
2024-03-14 10:28:30 -07:00
case Return : {
if ( ! env - > fn_ctx ) code_err ( ast , " This return statement is not inside any function " ) ;
auto ret = Match ( ast , Return ) - > value ;
assert ( env - > fn_ctx - > return_type ) ;
2024-07-04 15:00:01 -07:00
CORD code = CORD_EMPTY ;
for ( deferral_t * deferred = env - > deferred ; deferred ; deferred = deferred - > next ) {
code = CORD_all ( code , compile_statement ( deferred - > defer_env , deferred - > block ) ) ;
}
2024-03-14 10:28:30 -07:00
if ( ret ) {
2024-07-04 15:27:08 -07:00
if ( env - > fn_ctx - > return_type - > tag = = VoidType | | env - > fn_ctx - > return_type - > tag = = AbortType )
code_err ( ast , " This function is not supposed to return any values, according to its type signature " ) ;
2024-05-15 10:40:27 -07:00
env = with_enum_scope ( env , env - > fn_ctx - > return_type ) ;
2024-11-21 10:00:53 -08:00
CORD value = compile_to_type ( env , ret , env - > fn_ctx - > return_type ) ;
2024-10-08 21:45:52 -07:00
if ( env - > deferred ) {
code = CORD_all ( compile_declaration ( env - > fn_ctx - > return_type , " ret " ) , " = " , value , " ; \n " , code ) ;
value = " ret " ;
}
2024-07-04 15:00:01 -07:00
return CORD_all ( code , " return " , value , " ; " ) ;
2024-03-14 10:28:30 -07:00
} else {
if ( env - > fn_ctx - > return_type - > tag ! = VoidType )
2024-07-04 15:27:08 -07:00
code_err ( ast , " This function expects you to return a %T value " , env - > fn_ctx - > return_type ) ;
2024-07-04 15:00:01 -07:00
return CORD_all ( code , " return; " ) ;
2024-03-14 10:28:30 -07:00
}
}
case While : {
auto while_ = Match ( ast , While ) ;
2024-03-15 10:45:25 -07:00
env_t * scope = fresh_scope ( env ) ;
loop_ctx_t loop_ctx = ( loop_ctx_t ) {
. loop_name = " while " ,
2024-07-04 15:00:01 -07:00
. deferred = scope - > deferred ,
2024-03-15 10:45:25 -07:00
. next = env - > loop_ctx ,
} ;
scope - > loop_ctx = & loop_ctx ;
CORD body = compile_statement ( scope , while_ - > body ) ;
if ( loop_ctx . skip_label )
body = CORD_all ( body , " \n " , loop_ctx . skip_label , " : continue; " ) ;
2024-05-23 09:40:21 -07:00
CORD loop = CORD_all ( " while ( " , while_ - > condition ? compile ( scope , while_ - > condition ) : " yes " , " ) { \n \t " , body , " \n } " ) ;
2024-03-15 10:45:25 -07:00
if ( loop_ctx . stop_label )
loop = CORD_all ( loop , " \n " , loop_ctx . stop_label , " :; " ) ;
return loop ;
2024-03-14 10:28:30 -07:00
}
2024-11-03 09:20:53 -08:00
case Repeat : {
ast_t * body = Match ( ast , Repeat ) - > body ;
env_t * scope = fresh_scope ( env ) ;
loop_ctx_t loop_ctx = ( loop_ctx_t ) {
. loop_name = " repeat " ,
. deferred = scope - > deferred ,
. next = env - > loop_ctx ,
} ;
scope - > loop_ctx = & loop_ctx ;
CORD body_code = compile_statement ( scope , body ) ;
if ( loop_ctx . skip_label )
body_code = CORD_all ( body_code , " \n " , loop_ctx . skip_label , " : continue; " ) ;
CORD loop = CORD_all ( " for (;;) { \n \t " , body_code , " \n } " ) ;
if ( loop_ctx . stop_label )
loop = CORD_all ( loop , " \n " , loop_ctx . stop_label , " :; " ) ;
return loop ;
}
2024-03-14 10:28:30 -07:00
case For : {
auto for_ = Match ( ast , For ) ;
2024-09-05 08:23:00 -07:00
// If we're iterating over a comprehension, that's actually just doing
// one loop, we don't need to compile the comprehension as an array
2024-11-02 17:22:19 -07:00
// comprehension. This is a common case for reducers like `(+: i*2 for i in 5)`
2024-09-05 08:23:00 -07:00
// or `(and) x:is_good() for x in xs`
if ( for_ - > iter - > tag = = Comprehension ) {
auto comp = Match ( for_ - > iter , Comprehension ) ;
ast_t * body = for_ - > body ;
if ( for_ - > vars ) {
if ( for_ - > vars - > next )
code_err ( for_ - > vars - > next - > ast , " This is too many variables for iteration " ) ;
body = WrapAST (
ast , Block ,
. statements = new ( ast_list_t , . ast = WrapAST ( ast , Declare , . var = for_ - > vars - > ast , . value = comp - > expr ) ,
. next = body - > tag = = Block ? Match ( body , Block ) - > statements : new ( ast_list_t , . ast = body ) ) ) ;
}
if ( comp - > filter )
body = WrapAST ( for_ - > body , If , . condition = comp - > filter , . body = body ) ;
ast_t * loop = WrapAST ( ast , For , . vars = comp - > vars , . iter = comp - > iter , . body = body ) ;
return compile_statement ( env , loop ) ;
}
2024-03-15 10:35:30 -07:00
env_t * body_scope = for_scope ( env , ast ) ;
loop_ctx_t loop_ctx = ( loop_ctx_t ) {
2024-03-15 10:45:25 -07:00
. loop_name = " for " ,
2024-07-13 14:17:58 -07:00
. loop_vars = for_ - > vars ,
2024-07-04 15:00:01 -07:00
. deferred = body_scope - > deferred ,
2024-03-15 10:35:30 -07:00
. next = body_scope - > loop_ctx ,
} ;
body_scope - > loop_ctx = & loop_ctx ;
2024-09-05 09:28:28 -07:00
// Naked means no enclosing braces:
CORD naked_body = compile_inline_block ( body_scope , for_ - > body ) ;
2024-03-15 10:38:25 -07:00
if ( loop_ctx . skip_label )
2024-09-05 09:28:28 -07:00
naked_body = CORD_all ( naked_body , " \n " , loop_ctx . skip_label , " : continue; " ) ;
2024-03-15 10:38:25 -07:00
CORD stop = loop_ctx . stop_label ? CORD_all ( " \n " , loop_ctx . stop_label , " :; " ) : CORD_EMPTY ;
2024-03-15 10:35:30 -07:00
2024-11-09 14:54:32 -08:00
type_t * iter_t = value_type ( get_type ( env , for_ - > iter ) ) ;
type_t * iter_value_t = value_type ( iter_t ) ;
if ( iter_value_t = = RANGE_TYPE ) {
CORD range = compile_to_pointer_depth ( env , for_ - > iter , 0 , false ) ;
2024-08-17 11:41:31 -07:00
CORD value = for_ - > vars ? compile ( body_scope , for_ - > vars - > ast ) : " i " ;
2024-08-05 11:40:28 -07:00
if ( for_ - > empty )
code_err ( ast , " Ranges are never empty, they always contain at least their starting element " ) ;
return CORD_all (
" { \n "
" const Range_t range = " , range , " ; \n "
2024-08-13 00:20:01 -07:00
" if (range.step.small == 0) fail( \" This range has a 'step' of zero and will loop infinitely! \" ); \n "
" bool negative = (Int$compare_value(range.step, I(0)) < 0); \n "
2024-08-13 08:35:22 -07:00
" for (Int_t " , value , " = range.first; ({ int32_t cmp = Int$compare_value( " , value , " , range.last); negative ? cmp >= 0 : cmp <= 0;}) ; " , value , " = Int$plus( " , value , " , range.step)) { \n "
2024-09-05 09:28:28 -07:00
" \t " , naked_body ,
2024-08-05 11:40:28 -07:00
" \n } " ,
stop ,
" \n } " ) ;
}
2024-11-09 14:54:32 -08:00
switch ( iter_value_t - > tag ) {
2024-03-14 10:28:30 -07:00
case ArrayType : {
2024-11-09 14:54:32 -08:00
type_t * item_t = Match ( iter_value_t , ArrayType ) - > item_type ;
2024-08-12 22:30:25 -07:00
CORD index = CORD_EMPTY ;
2024-08-12 14:44:05 -07:00
CORD value = CORD_EMPTY ;
2024-07-13 14:17:58 -07:00
if ( for_ - > vars ) {
if ( for_ - > vars - > next ) {
if ( for_ - > vars - > next - > next )
code_err ( for_ - > vars - > next - > next - > ast , " This is too many variables for this loop " ) ;
2024-08-17 11:41:31 -07:00
index = compile ( body_scope , for_ - > vars - > ast ) ;
value = compile ( body_scope , for_ - > vars - > next - > ast ) ;
2024-07-13 14:17:58 -07:00
} else {
2024-08-17 11:41:31 -07:00
value = compile ( body_scope , for_ - > vars - > ast ) ;
2024-07-13 14:17:58 -07:00
}
}
2024-08-12 14:44:05 -07:00
CORD loop = CORD_EMPTY ;
2024-11-09 14:54:32 -08:00
loop = CORD_all ( loop , " for (int64_t i = 1; i <= iterating.length; ++i) " ) ;
2024-08-12 22:30:25 -07:00
if ( index ! = CORD_EMPTY )
2024-09-05 09:28:28 -07:00
naked_body = CORD_all ( " Int_t " , index , " = I(i); \n " , naked_body ) ;
2024-08-12 22:30:25 -07:00
2024-08-12 14:44:05 -07:00
if ( value ! = CORD_EMPTY ) {
loop = CORD_all ( loop , " { \n " ,
compile_declaration ( item_t , value ) ,
2024-08-12 22:30:25 -07:00
" = *( " , compile_type ( item_t ) , " *)(iterating.data + (i-1)*iterating.stride); \n " ,
2024-09-05 09:28:28 -07:00
naked_body , " \n } " ) ;
2024-08-12 14:44:05 -07:00
} else {
2024-09-05 09:28:28 -07:00
loop = CORD_all ( loop , " { \n " , naked_body , " \n } " ) ;
2024-07-20 14:13:15 -07:00
}
2024-08-10 12:15:38 -07:00
2024-11-09 14:54:32 -08:00
if ( for_ - > empty )
loop = CORD_all ( " if (iterating.length > 0) { \n " , loop , " \n } else " , compile_statement ( env , for_ - > empty ) ) ;
if ( iter_t - > tag = = PointerType ) {
2024-08-12 14:44:05 -07:00
loop = CORD_all ( " { \n "
2024-11-09 14:54:32 -08:00
" Array_t *ptr = " , compile_to_pointer_depth ( env , for_ - > iter , 1 , false ) , " ; \n "
" \n ARRAY_INCREF(*ptr); \n "
" Array_t iterating = *ptr; \n " ,
2024-08-12 14:44:05 -07:00
loop ,
stop ,
2024-11-09 14:54:32 -08:00
" \n ARRAY_DECREF(*ptr); \n "
2024-08-12 14:44:05 -07:00
" } \n " ) ;
} else {
loop = CORD_all ( " { \n "
2024-11-09 14:54:32 -08:00
" Array_t iterating = " , compile_to_pointer_depth ( env , for_ - > iter , 0 , false ) , " ; \n " ,
2024-08-12 14:44:05 -07:00
loop ,
stop ,
" } \n " ) ;
2024-08-10 12:15:38 -07:00
}
return loop ;
}
2024-08-12 14:44:05 -07:00
case SetType : case TableType : {
CORD loop = " for (int64_t i = 0; i < iterating.length; ++i) { \n " ;
2024-07-13 14:17:58 -07:00
if ( for_ - > vars ) {
2024-11-09 14:54:32 -08:00
if ( iter_value_t - > tag = = SetType ) {
2024-08-12 14:44:05 -07:00
if ( for_ - > vars - > next )
code_err ( for_ - > vars - > next - > ast , " This is too many variables for this loop " ) ;
2024-08-17 11:41:31 -07:00
CORD item = compile ( body_scope , for_ - > vars - > ast ) ;
2024-11-09 14:54:32 -08:00
type_t * item_type = Match ( iter_value_t , SetType ) - > item_type ;
2024-08-12 14:44:05 -07:00
loop = CORD_all ( loop , compile_declaration ( item_type , item ) , " = *( " , compile_type ( item_type ) , " *)( " ,
" iterating.data + i*iterating.stride); \n " ) ;
2024-07-13 14:17:58 -07:00
} else {
2024-08-17 11:41:31 -07:00
CORD key = compile ( body_scope , for_ - > vars - > ast ) ;
2024-11-09 14:54:32 -08:00
type_t * key_t = Match ( iter_value_t , TableType ) - > key_type ;
2024-08-12 14:44:05 -07:00
loop = CORD_all ( loop , compile_declaration ( key_t , key ) , " = *( " , compile_type ( key_t ) , " *)( " ,
" iterating.data + i*iterating.stride); \n " ) ;
if ( for_ - > vars - > next ) {
if ( for_ - > vars - > next - > next )
code_err ( for_ - > vars - > next - > next - > ast , " This is too many variables for this loop " ) ;
2024-11-09 14:54:32 -08:00
type_t * value_t = Match ( iter_value_t , TableType ) - > value_type ;
2024-08-17 11:41:31 -07:00
CORD value = compile ( body_scope , for_ - > vars - > next - > ast ) ;
2024-08-12 14:44:05 -07:00
size_t value_offset = type_size ( key_t ) ;
if ( type_align ( value_t ) > 1 & & value_offset % type_align ( value_t ) )
value_offset + = type_align ( value_t ) - ( value_offset % type_align ( value_t ) ) ; // padding
loop = CORD_all ( loop , compile_declaration ( value_t , value ) , " = *( " , compile_type ( value_t ) , " *)( " ,
" iterating.data + i*iterating.stride + " , heap_strf ( " %zu " , value_offset ) , " ); \n " ) ;
}
2024-07-13 14:17:58 -07:00
}
}
2024-09-05 09:28:28 -07:00
loop = CORD_all ( loop , naked_body , " \n } " ) ;
2024-08-12 14:44:05 -07:00
if ( for_ - > empty ) {
loop = CORD_all ( " if (iterating.length > 0) { \n " , loop , " \n } else " , compile_statement ( env , for_ - > empty ) ) ;
2024-07-13 14:17:58 -07:00
}
2024-08-12 14:44:05 -07:00
2024-11-09 14:54:32 -08:00
if ( iter_t - > tag = = PointerType ) {
loop = CORD_all (
" { \n " ,
" Table_t *t = " , compile_to_pointer_depth ( env , for_ - > iter , 1 , false ) , " ; \n "
" ARRAY_INCREF(t->entries); \n "
" Array_t iterating = t->entries; \n " ,
loop ,
" ARRAY_DECREF(t->entries); \n "
" } \n " ) ;
2024-08-12 14:44:05 -07:00
} else {
loop = CORD_all (
" { \n " ,
2024-11-09 14:57:02 -08:00
" Array_t iterating = ( " , compile_to_pointer_depth ( env , for_ - > iter , 0 , false ) , " ).entries; \n " ,
2024-08-12 14:44:05 -07:00
loop ,
" } \n " ) ;
2024-03-14 10:28:30 -07:00
}
2024-03-26 11:02:48 -07:00
return loop ;
2024-03-14 10:28:30 -07:00
}
2024-08-18 09:47:29 -07:00
case BigIntType : {
2024-09-05 09:28:28 -07:00
CORD n ;
if ( for_ - > iter - > tag = = Int ) {
const char * str = Match ( for_ - > iter , Int ) - > str ;
2024-09-11 11:53:48 -07:00
Int_t int_val = Int $ from_str ( str ) ;
if ( int_val . small = = 0 )
code_err ( for_ - > iter , " Failed to parse this integer " ) ;
2024-09-05 09:28:28 -07:00
mpz_t i ;
mpz_init_set_int ( i , int_val ) ;
if ( mpz_cmpabs_ui ( i , BIGGEST_SMALL_INT ) < = 0 )
n = mpz_get_str ( NULL , 10 , i ) ;
else
goto big_n ;
2024-09-05 09:41:13 -07:00
2024-09-05 09:44:01 -07:00
if ( for_ - > empty & & mpz_cmp_si ( i , 0 ) < = 0 ) {
2024-09-05 09:41:13 -07:00
return compile_statement ( env , for_ - > empty ) ;
} else {
return CORD_all (
" for (int64_t i = 1; i <= " , n , " ; ++i) { \n " ,
for_ - > vars ? CORD_all ( " \t Int_t " , compile ( body_scope , for_ - > vars - > ast ) , " = I_small(i); \n " ) : CORD_EMPTY ,
" \t " , naked_body ,
" } \n " ,
stop , " \n " ) ;
}
2024-09-05 09:28:28 -07:00
}
2024-09-05 09:41:13 -07:00
big_n :
2024-11-09 14:54:32 -08:00
n = compile_to_pointer_depth ( env , for_ - > iter , 0 , false ) ;
2024-09-05 09:41:13 -07:00
CORD i = for_ - > vars ? compile ( body_scope , for_ - > vars - > ast ) : " i " ;
CORD n_var = for_ - > vars ? CORD_all ( " max " , i ) : " n " ;
2024-07-13 14:17:58 -07:00
if ( for_ - > empty ) {
2024-03-14 10:28:30 -07:00
return CORD_all (
" { \n "
2024-11-09 14:54:32 -08:00
" Int_t " , n_var , " = " , n , " ; \n "
2024-09-05 09:44:01 -07:00
" if (Int$compare_value( " , n_var , " , I(0)) > 0) { \n "
2024-09-05 09:41:13 -07:00
" for (Int_t " , i , " = I(1); Int$compare_value( " , i , " , " , n_var , " ) <= 0; " , i , " = Int$plus( " , i , " , I(1))) { \n " ,
2024-09-05 09:28:28 -07:00
" \t " , naked_body ,
" } \n "
" } else " , compile_statement ( env , for_ - > empty ) ,
stop , " \n "
" } \n " ) ;
2024-03-14 10:28:30 -07:00
} else {
return CORD_all (
2024-09-05 09:41:13 -07:00
" for (Int_t " , i , " = I(1), " , n_var , " = " , n , " ; Int$compare_value( " , i , " , " , n_var , " ) <= 0; " , i , " = Int$plus( " , i , " , I(1))) { \n " ,
2024-09-05 09:28:28 -07:00
" \t " , naked_body ,
" } \n " ,
stop , " \n " ) ;
2024-03-14 10:28:30 -07:00
}
}
2024-07-13 14:17:58 -07:00
case FunctionType : case ClosureType : {
2024-07-20 13:45:13 -07:00
// Iterator function:
2024-07-13 14:17:58 -07:00
CORD code = " { \n " ;
2024-09-27 11:45:06 -07:00
CORD next_fn ;
if ( is_idempotent ( for_ - > iter ) ) {
2024-11-09 14:54:32 -08:00
next_fn = compile_to_pointer_depth ( env , for_ - > iter , 0 , false ) ;
2024-09-27 11:45:06 -07:00
} else {
2024-11-09 14:54:32 -08:00
code = CORD_all ( code , compile_declaration ( iter_value_t , " next " ) , " = " , compile_to_pointer_depth ( env , for_ - > iter , 0 , false ) , " ; \n " ) ;
2024-09-27 11:45:06 -07:00
next_fn = " next " ;
}
2024-07-13 14:17:58 -07:00
2024-11-09 14:54:32 -08:00
auto fn = iter_value_t - > tag = = ClosureType ? Match ( Match ( iter_value_t , ClosureType ) - > fn , FunctionType ) : Match ( iter_value_t , FunctionType ) ;
2024-07-20 13:45:13 -07:00
2024-09-27 11:45:06 -07:00
CORD get_next ;
2024-11-09 14:54:32 -08:00
if ( iter_value_t - > tag = = ClosureType ) {
type_t * fn_t = Match ( iter_value_t , ClosureType ) - > fn ;
2024-07-13 14:17:58 -07:00
arg_t * closure_fn_args = NULL ;
for ( arg_t * arg = Match ( fn_t , FunctionType ) - > args ; arg ; arg = arg - > next )
closure_fn_args = new ( arg_t , . name = arg - > name , . type = arg - > type , . default_val = arg - > default_val , . next = closure_fn_args ) ;
closure_fn_args = new ( arg_t , . name = " userdata " , . type = Type ( PointerType , . pointed = Type ( MemoryType ) ) , . next = closure_fn_args ) ;
REVERSE_LIST ( closure_fn_args ) ;
CORD fn_type_code = compile_type ( Type ( FunctionType , . args = closure_fn_args , . ret = Match ( fn_t , FunctionType ) - > ret ) ) ;
2024-09-27 11:45:06 -07:00
get_next = CORD_all ( " (( " , fn_type_code , " ) " , next_fn , " .fn)( " , next_fn , " .userdata) " ) ;
2024-07-13 14:17:58 -07:00
} else {
2024-09-27 11:45:06 -07:00
get_next = CORD_all ( next_fn , " () " ) ;
2024-09-11 19:28:43 -07:00
}
2024-07-13 15:05:14 -07:00
2024-09-27 11:45:06 -07:00
if ( fn - > ret - > tag = = OptionalType ) {
// Use an optional variable `cur` for each iteration step, which will be checked for null
code = CORD_all ( code , compile_declaration ( fn - > ret , " cur " ) , " ; \n " ) ;
get_next = CORD_all ( " (cur= " , get_next , " , ! " , check_null ( fn - > ret , " cur " ) , " ) " ) ;
if ( for_ - > vars ) {
naked_body = CORD_all (
compile_declaration ( Match ( fn - > ret , OptionalType ) - > type , CORD_all ( " $ " , Match ( for_ - > vars - > ast , Var ) - > name ) ) ,
" = " , optional_into_nonnull ( fn - > ret , " cur " ) , " ; \n " ,
naked_body ) ;
}
if ( for_ - > empty ) {
code = CORD_all ( code , " if ( " , get_next , " ) { \n "
" \t do{ \n \t \t " , naked_body , " \t } while( " , get_next , " ); \n "
" } else { \n \t " , compile_statement ( env , for_ - > empty ) , " } " , stop , " \n } \n " ) ;
} else {
code = CORD_all ( code , " while( " , get_next , " ) { \n \t " , naked_body , " } \n " , stop , " \n } \n " ) ;
}
2024-07-13 15:05:14 -07:00
} else {
2024-09-27 11:45:06 -07:00
if ( for_ - > vars ) {
naked_body = CORD_all (
compile_declaration ( fn - > ret , CORD_all ( " $ " , Match ( for_ - > vars - > ast , Var ) - > name ) ) ,
" = " , get_next , " ; \n " , naked_body ) ;
} else {
naked_body = CORD_all ( get_next , " ; \n " , naked_body ) ;
}
if ( for_ - > empty )
code_err ( for_ - > empty , " This iteration loop will always have values, so this block will never run " ) ;
code = CORD_all ( code , " for (;;) { \n \t " , naked_body , " } \n " , stop , " \n } \n " ) ;
2024-07-13 15:05:14 -07:00
}
2024-09-27 11:45:06 -07:00
2024-07-13 14:17:58 -07:00
return code ;
}
2024-03-14 10:28:30 -07:00
default : code_err ( for_ - > iter , " Iteration is not implemented for type: %T " , iter_t ) ;
}
}
case If : {
auto if_ = Match ( ast , If ) ;
2024-09-10 22:48:15 -07:00
ast_t * condition = if_ - > condition ;
if ( condition - > tag = = Declare ) {
2024-11-11 21:04:56 -08:00
env_t * truthy_scope = fresh_scope ( env ) ;
CORD code = CORD_all ( " IF_DECLARE( " , compile_statement ( truthy_scope , condition ) , " , " ) ;
bind_statement ( truthy_scope , condition ) ;
ast_t * var = Match ( condition , Declare ) - > var ;
code = CORD_all ( code , compile_condition ( truthy_scope , var ) , " , " ) ;
type_t * cond_t = get_type ( truthy_scope , var ) ;
if ( cond_t - > tag = = OptionalType ) {
set_binding ( truthy_scope , Match ( var , Var ) - > name ,
2024-09-11 21:13:53 -07:00
new ( binding_t , . type = Match ( cond_t , OptionalType ) - > type ,
2024-11-11 21:04:56 -08:00
. code = optional_into_nonnull ( cond_t , compile ( truthy_scope , var ) ) ) ) ;
2024-09-10 22:31:31 -07:00
}
2024-11-11 21:04:56 -08:00
code = CORD_all ( code , compile_statement ( truthy_scope , if_ - > body ) , " ) " ) ;
if ( if_ - > else_body )
code = CORD_all ( code , " \n else " , compile_statement ( env , if_ - > else_body ) ) ;
return code ;
2024-09-10 22:51:16 -07:00
} else {
2024-11-11 21:04:56 -08:00
CORD code = CORD_all ( " if ( " , compile_condition ( env , condition ) , " ) " ) ;
env_t * truthy_scope = env ;
type_t * cond_t = get_type ( env , condition ) ;
if ( condition - > tag = = Var & & cond_t - > tag = = OptionalType ) {
truthy_scope = fresh_scope ( env ) ;
set_binding ( truthy_scope , Match ( condition , Var ) - > name ,
new ( binding_t , . type = Match ( cond_t , OptionalType ) - > type ,
. code = optional_into_nonnull ( cond_t , compile ( truthy_scope , condition ) ) ) ) ;
}
code = CORD_all ( code , compile_statement ( truthy_scope , if_ - > body ) ) ;
if ( if_ - > else_body )
code = CORD_all ( code , " \n else " , compile_statement ( env , if_ - > else_body ) ) ;
return code ;
2024-09-10 22:31:31 -07:00
}
2024-03-14 10:28:30 -07:00
}
case Block : {
2024-09-05 09:28:28 -07:00
return CORD_all ( " { \n " , compile_inline_block ( env , ast ) , " } \n " ) ;
2024-03-14 10:28:30 -07:00
}
2024-03-17 12:26:25 -07:00
case Comprehension : {
2024-09-12 11:27:13 -07:00
if ( ! env - > comprehension_action )
code_err ( ast , " I don't know what to do with this comprehension! " ) ;
2024-03-17 12:26:25 -07:00
auto comp = Match ( ast , Comprehension ) ;
if ( comp - > expr - > tag = = Comprehension ) { // Nested comprehension
ast_t * body = comp - > filter ? WrapAST ( ast , If , . condition = comp - > filter , . body = comp - > expr ) : comp - > expr ;
2024-07-13 14:17:58 -07:00
ast_t * loop = WrapAST ( ast , For , . vars = comp - > vars , . iter = comp - > iter , . body = body ) ;
2024-03-17 12:26:25 -07:00
return compile_statement ( env , loop ) ;
}
2024-09-12 11:27:13 -07:00
// Array/Set/Table comprehension:
comprehension_body_t get_body = ( void * ) env - > comprehension_action - > fn ;
ast_t * body = get_body ( comp - > expr , env - > comprehension_action - > userdata ) ;
if ( comp - > filter )
body = WrapAST ( comp - > expr , If , . condition = comp - > filter , . body = body ) ;
ast_t * loop = WrapAST ( ast , For , . vars = comp - > vars , . iter = comp - > iter , . body = body ) ;
return compile_statement ( env , loop ) ;
2024-03-17 12:26:25 -07:00
}
2024-06-06 13:28:53 -07:00
case Extern : return CORD_EMPTY ;
2024-09-15 14:34:34 -07:00
case InlineCCode : {
auto inline_code = Match ( ast , InlineCCode ) ;
if ( inline_code - > type )
return CORD_all ( " ({ " , inline_code - > code , " ; }) " ) ;
else
return inline_code - > code ;
}
2024-08-18 21:52:35 -07:00
case Use : {
2024-08-22 11:02:48 -07:00
auto use = Match ( ast , Use ) ;
if ( use - > what = = USE_LOCAL ) {
2024-10-09 22:09:17 -07:00
CORD name = file_base_id ( Match ( ast , Use ) - > path ) ;
2024-09-24 11:54:22 -07:00
env - > code - > variable_initializers = CORD_all ( env - > code - > variable_initializers , " $ " , name , " $$initialize(); \n " ) ;
2024-09-18 12:39:22 -07:00
} else if ( use - > what = = USE_C_CODE ) {
return CORD_all ( " #include \" " , use - > path , " \" \n " ) ;
2024-08-22 11:02:48 -07:00
} else if ( use - > what = = USE_MODULE ) {
2024-09-15 18:18:42 -07:00
glob_t tm_files ;
2024-09-22 12:07:44 -07:00
if ( glob ( heap_strf ( " ~/.local/share/tomo/installed/%s/[!._0-9]*.tm " , use - > path ) , GLOB_TILDE , NULL , & tm_files ) ! = 0 )
2024-09-15 18:18:42 -07:00
code_err ( ast , " Could not find library " ) ;
2024-09-22 12:07:44 -07:00
const char * lib_id = Text $ as_c_string (
Text $ replace ( Text $ from_str ( use - > path ) , Pattern ( " {1+ !alphanumeric} " ) , Text ( " _ " ) , Pattern ( " " ) , false ) ) ;
2024-09-15 18:18:42 -07:00
for ( size_t i = 0 ; i < tm_files . gl_pathc ; i + + ) {
const char * filename = tm_files . gl_pathv [ i ] ;
2024-08-22 11:02:48 -07:00
env - > code - > variable_initializers = CORD_all (
2024-10-09 22:09:17 -07:00
env - > code - > variable_initializers , " $ " , lib_id , " $ " , file_base_id ( filename ) , " $$initialize(); \n " ) ;
2024-08-22 11:02:48 -07:00
}
2024-09-15 18:18:42 -07:00
globfree ( & tm_files ) ;
2024-08-22 11:02:48 -07:00
}
2024-08-18 21:52:35 -07:00
return CORD_EMPTY ;
}
2024-02-04 15:04:41 -08:00
default :
2024-05-12 10:50:06 -07:00
if ( ! is_discardable ( env , ast ) )
2024-09-16 11:27:18 -07:00
code_err ( ast , " The %T result of this statement cannot be discarded " , get_type ( env , ast ) ) ;
2024-03-14 10:28:30 -07:00
return CORD_asprintf ( " (void) % r ; " , compile(env, ast));
2024-02-04 15:04:41 -08:00
}
}
2024-04-13 12:20:42 -07:00
// CORD compile_statement(env_t *env, ast_t *ast) {
// CORD stmt = _compile_statement(env, ast);
// if (!stmt)
// return stmt;
// int64_t line = get_line_number(ast->file, ast->start);
// return CORD_asprintf("#line %ld\n%r", line, stmt);
// }
2024-03-09 15:32:36 -08:00
CORD expr_as_text ( env_t * env , CORD expr , type_t * t , CORD color )
2024-02-17 17:47:43 -08:00
{
switch ( t - > tag ) {
2024-09-05 12:31:54 -07:00
case MemoryType : return CORD_asprintf ( " Memory$as_text(stack(%r), %r, &Memory$info) " , expr , color ) ;
2024-09-05 12:44:49 -07:00
case BoolType :
// NOTE: this cannot use stack(), since bools may actually be bit fields:
return CORD_asprintf ( " Bool$as_text((Bool_t[1]) { % r } , % r , & Bool $ info ) " , expr, color);
2024-09-05 12:31:54 -07:00
case CStringType : return CORD_asprintf ( " CString$as_text(stack(%r), %r, &CString$info) " , expr , color ) ;
2024-11-17 11:49:03 -08:00
case MomentType : return CORD_asprintf ( " Moment$as_text(stack(%r), %r, &Moment$info) " , expr , color ) ;
2024-09-15 12:33:47 -07:00
case BigIntType : case IntType : case ByteType : case NumType : {
2024-03-03 10:37:05 -08:00
CORD name = type_to_cord ( t ) ;
2024-09-05 12:31:54 -07:00
return CORD_asprintf ( " %r$as_text(stack(%r) , % r , & % r $ info ) " , name, expr, color, name) ;
2024-03-03 15:16:33 -08:00
}
2024-03-09 15:32:36 -08:00
case TextType : {
2024-07-04 13:22:52 -07:00
return CORD_asprintf ( " Text$as_text(stack(%r) , % r , % r ) " , expr, color, compile_type_info(env, t)) ;
2024-03-09 15:32:36 -08:00
}
2024-05-12 17:12:00 -07:00
case ArrayType : return CORD_asprintf ( " Array$as_text(stack(%r), %r, %r) " , expr , color , compile_type_info ( env , t ) ) ;
2024-08-10 12:15:38 -07:00
case SetType : return CORD_asprintf ( " Table$as_text(stack(%r), %r, %r) " , expr , color , compile_type_info ( env , t ) ) ;
2024-08-11 11:47:34 -07:00
case ChannelType : return CORD_asprintf ( " Channel$as_text(stack(%r), %r, %r) " , expr , color , compile_type_info ( env , t ) ) ;
2024-05-12 17:12:00 -07:00
case TableType : return CORD_asprintf ( " Table$as_text(stack(%r), %r, %r) " , expr , color , compile_type_info ( env , t ) ) ;
case FunctionType : case ClosureType : return CORD_asprintf ( " Func$as_text(stack(%r), %r, %r) " , expr , color , compile_type_info ( env , t ) ) ;
case PointerType : return CORD_asprintf ( " Pointer$as_text(stack(%r), %r, %r) " , expr , color , compile_type_info ( env , t ) ) ;
2024-09-10 22:31:31 -07:00
case OptionalType : return CORD_asprintf ( " Optional$as_text(stack(%r), %r, %r) " , expr , color , compile_type_info ( env , t ) ) ;
2024-10-08 10:35:18 -07:00
case StructType : case EnumType : return CORD_asprintf ( " generic_as_text(stack(%r), %r, %r) " , expr , color , compile_type_info ( env , t ) ) ;
2024-02-17 17:47:43 -08:00
default : compiler_err ( NULL , NULL , NULL , " Stringifying is not supported for %T " , t ) ;
}
}
CORD compile_string ( env_t * env , ast_t * ast , CORD color )
{
type_t * t = get_type ( env , ast ) ;
CORD expr = compile ( env , ast ) ;
2024-03-09 15:32:36 -08:00
return expr_as_text ( env , expr , t , color ) ;
2024-02-17 17:47:43 -08:00
}
2024-08-20 13:20:41 -07:00
CORD compile_to_pointer_depth ( env_t * env , ast_t * ast , int64_t target_depth , bool needs_incref )
2024-02-18 11:28:35 -08:00
{
CORD val = compile ( env , ast ) ;
type_t * t = get_type ( env , ast ) ;
int64_t depth = 0 ;
for ( type_t * tt = t ; tt - > tag = = PointerType ; tt = Match ( tt , PointerType ) - > pointed )
+ + depth ;
2024-08-20 13:20:41 -07:00
// Passing a literal value won't trigger an incref, because it's ephemeral,
// e.g. [10, 20]:reversed()
if ( t - > tag ! = PointerType & & needs_incref & & ! can_be_mutated ( env , ast ) )
needs_incref = false ;
2024-02-18 11:28:35 -08:00
while ( depth ! = target_depth ) {
if ( depth < target_depth ) {
if ( ast - > tag = = Var & & target_depth = = 1 )
2024-03-03 11:34:13 -08:00
val = CORD_all ( " (& " , val , " ) " ) ;
2024-02-18 11:28:35 -08:00
else
2024-07-22 10:54:03 -07:00
code_err ( ast , " This should be a pointer, not %T " , get_type ( env , ast ) ) ;
2024-10-27 16:54:48 -07:00
t = Type ( PointerType , . pointed = t , . is_view = true ) ;
2024-02-18 11:28:35 -08:00
+ + depth ;
} else {
auto ptr = Match ( t , PointerType ) ;
val = CORD_all ( " *( " , val , " ) " ) ;
t = ptr - > pointed ;
- - depth ;
}
}
2024-08-20 13:00:26 -07:00
while ( t - > tag = = PointerType ) {
auto ptr = Match ( t , PointerType ) ;
t = ptr - > pointed ;
2024-02-20 10:06:03 -08:00
}
2024-08-20 13:20:41 -07:00
if ( needs_incref & & t - > tag = = ArrayType )
val = CORD_all ( " ARRAY_COPY( " , val , " ) " ) ;
else if ( needs_incref & & ( t - > tag = = TableType | | t - > tag = = SetType ) )
val = CORD_all ( " TABLE_COPY( " , val , " ) " ) ;
2024-02-18 11:28:35 -08:00
return val ;
}
2024-11-21 10:00:53 -08:00
CORD compile_to_type ( env_t * env , ast_t * ast , type_t * t )
2024-09-12 10:05:08 -07:00
{
2024-11-04 12:24:10 -08:00
if ( ast - > tag = = Int & & is_numeric_type ( t ) )
return compile_int_to_type ( env , ast , t ) ;
2024-11-21 10:00:53 -08:00
if ( ast - > tag = = Null & & Match ( ast , Null ) - > type = = NULL )
return compile_null ( t ) ;
2024-09-12 10:05:08 -07:00
CORD code = compile ( env , ast ) ;
type_t * actual = get_type ( env , ast ) ;
if ( ! promote ( env , & code , actual , t ) )
code_err ( ast , " I expected a %T here, but this is a %T " , t , actual ) ;
return code ;
}
2024-05-15 10:40:27 -07:00
env_t * with_enum_scope ( env_t * env , type_t * t )
2024-05-15 10:24:43 -07:00
{
if ( t - > tag ! = EnumType ) return env ;
env = fresh_scope ( env ) ;
env_t * ns_env = Match ( t , EnumType ) - > env ;
for ( tag_t * tag = Match ( t , EnumType ) - > tags ; tag ; tag = tag - > next ) {
if ( get_binding ( env , tag - > name ) )
continue ;
2024-05-18 22:46:30 -07:00
binding_t * b = get_binding ( ns_env , tag - > name ) ;
assert ( b ) ;
set_binding ( env , tag - > name , b ) ;
2024-05-15 10:24:43 -07:00
}
return env ;
}
2024-08-17 20:43:30 -07:00
CORD compile_int_to_type ( env_t * env , ast_t * ast , type_t * target )
{
2024-08-18 17:39:57 -07:00
if ( ast - > tag ! = Int ) {
CORD code = compile ( env , ast ) ;
type_t * actual_type = get_type ( env , ast ) ;
if ( ! promote ( env , & code , actual_type , target ) )
code = CORD_all ( type_to_cord ( actual_type ) , " _to_ " , type_to_cord ( target ) , " ( " , code , " , no) " ) ;
return code ;
}
2024-11-05 12:18:32 -08:00
if ( target - > tag = = BigIntType )
return compile ( env , ast ) ;
const char * literal = Match ( ast , Int ) - > str ;
OptionalInt_t int_val = Int $ from_str ( literal ) ;
2024-09-11 11:53:48 -07:00
if ( int_val . small = = 0 )
code_err ( ast , " Failed to parse this integer " ) ;
2024-08-17 20:43:30 -07:00
mpz_t i ;
mpz_init_set_int ( i , int_val ) ;
2024-11-05 12:18:32 -08:00
char * c_literal ;
if ( strncmp ( literal , " 0x " , 2 ) = = 0 | | strncmp ( literal , " 0X " , 2 ) = = 0 | | strncmp ( literal , " 0b " , 2 ) = = 0 ) {
gmp_asprintf ( & c_literal , " 0x%ZX " , i ) ;
} else if ( strncmp ( literal , " 0o " , 2 ) = = 0 ) {
gmp_asprintf ( & c_literal , " %#Zo " , i ) ;
} else {
gmp_asprintf ( & c_literal , " %#Zd " , i ) ;
}
2024-11-03 22:12:37 -08:00
if ( target - > tag = = ByteType ) {
if ( mpz_cmp_si ( i , UINT8_MAX ) < = 0 & & mpz_cmp_si ( i , 0 ) > = 0 )
2024-11-05 12:18:32 -08:00
return CORD_asprintf ( " (Byte_t) ( % s ) " , c_literal) ;
2024-11-03 22:12:37 -08:00
code_err ( ast , " This integer cannot fit in a byte " ) ;
2024-11-04 12:24:10 -08:00
} else if ( target - > tag = = NumType ) {
if ( Match ( target , NumType ) - > bits = = TYPE_NBITS64 ) {
2024-11-05 12:18:32 -08:00
return CORD_asprintf ( " N64(%s) " , c_literal) ;
2024-11-04 12:24:10 -08:00
} else {
2024-11-05 12:18:32 -08:00
return CORD_asprintf ( " N32(%s) " , c_literal) ;
2024-11-04 12:24:10 -08:00
}
} else if ( target - > tag = = IntType ) {
int64_t target_bits = ( int64_t ) Match ( target , IntType ) - > bits ;
switch ( target_bits ) {
case TYPE_IBITS64 :
if ( mpz_cmp_si ( i , INT64_MAX ) < = 0 & & mpz_cmp_si ( i , INT64_MIN ) > = 0 )
2024-11-05 12:18:32 -08:00
return CORD_asprintf ( " I64(%s) " , c_literal) ;
2024-11-04 12:24:10 -08:00
break ;
case TYPE_IBITS32 :
if ( mpz_cmp_si ( i , INT32_MAX ) < = 0 & & mpz_cmp_si ( i , INT32_MIN ) > = 0 )
2024-11-05 12:18:32 -08:00
return CORD_asprintf ( " I32(%s) " , c_literal) ;
2024-11-04 12:24:10 -08:00
break ;
case TYPE_IBITS16 :
if ( mpz_cmp_si ( i , INT16_MAX ) < = 0 & & mpz_cmp_si ( i , INT16_MIN ) > = 0 )
2024-11-05 12:18:32 -08:00
return CORD_asprintf ( " I16(%s) " , c_literal) ;
2024-11-04 12:24:10 -08:00
break ;
case TYPE_IBITS8 :
if ( mpz_cmp_si ( i , INT8_MAX ) < = 0 & & mpz_cmp_si ( i , INT8_MIN ) > = 0 )
2024-11-05 12:18:32 -08:00
return CORD_asprintf ( " I8(%s) " , c_literal) ;
2024-11-04 12:24:10 -08:00
break ;
default : break ;
}
code_err ( ast , " This integer cannot fit in a %d-bit value " , target_bits ) ;
} else {
code_err ( ast , " I don't know how to compile this to a %T " , target ) ;
2024-11-03 22:12:37 -08:00
}
2024-08-17 20:43:30 -07:00
}
2024-07-13 14:17:58 -07:00
CORD compile_arguments ( env_t * env , ast_t * call_ast , arg_t * spec_args , arg_ast_t * call_args )
2024-03-08 11:23:16 -08:00
{
2024-09-05 11:57:31 -07:00
Table_t used_args = { } ;
2024-03-08 11:23:16 -08:00
CORD code = CORD_EMPTY ;
2024-09-08 18:55:15 -07:00
env_t * default_scope = namespace_scope ( env ) ;
2024-03-08 11:23:16 -08:00
for ( arg_t * spec_arg = spec_args ; spec_arg ; spec_arg = spec_arg - > next ) {
2024-09-08 14:17:15 -07:00
int64_t i = 1 ;
2024-03-08 11:23:16 -08:00
// Find keyword:
if ( spec_arg - > name ) {
for ( arg_ast_t * call_arg = call_args ; call_arg ; call_arg = call_arg - > next ) {
if ( call_arg - > name & & streq ( call_arg - > name , spec_arg - > name ) ) {
2024-08-13 01:24:36 -07:00
CORD value ;
2024-08-17 20:43:30 -07:00
if ( spec_arg - > type - > tag = = IntType & & call_arg - > value - > tag = = Int ) {
value = compile_int_to_type ( env , call_arg - > value , spec_arg - > type ) ;
} else if ( spec_arg - > type - > tag = = NumType & & call_arg - > value - > tag = = Int ) {
2024-09-11 11:53:48 -07:00
OptionalInt_t int_val = Int $ from_str ( Match ( call_arg - > value , Int ) - > str ) ;
if ( int_val . small = = 0 )
code_err ( call_arg - > value , " Failed to parse this integer " ) ;
2024-08-14 11:57:01 -07:00
double n = Int_to_Num ( int_val ) ;
2024-08-18 08:49:51 -07:00
value = CORD_asprintf ( Match ( spec_arg - > type , NumType ) - > bits = = TYPE_NBITS64
2024-08-18 22:46:37 -07:00
? " N64(%.20g) " : " N32(%.10g) " , n ) ;
2024-08-13 01:24:36 -07:00
} else {
env_t * arg_env = with_enum_scope ( env , spec_arg - > type ) ;
type_t * actual_t = get_type ( arg_env , call_arg - > value ) ;
2024-09-06 01:11:00 -07:00
value = compile_maybe_incref ( arg_env , call_arg - > value ) ;
2024-08-13 01:24:36 -07:00
if ( ! promote ( arg_env , & value , actual_t , spec_arg - > type ) )
code_err ( call_arg - > value , " This argument is supposed to be a %T, but this value is a %T " , spec_arg - > type , actual_t ) ;
}
2024-03-29 09:54:31 -07:00
Table $ str_set ( & used_args , call_arg - > name , call_arg ) ;
2024-03-08 11:23:16 -08:00
if ( code ) code = CORD_cat ( code , " , " ) ;
2024-03-09 13:03:38 -08:00
code = CORD_cat ( code , value ) ;
2024-03-08 11:23:16 -08:00
goto found_it ;
}
}
}
// Find positional:
for ( arg_ast_t * call_arg = call_args ; call_arg ; call_arg = call_arg - > next ) {
if ( call_arg - > name ) continue ;
const char * pseudoname = heap_strf ( " %ld " , i + + ) ;
2024-03-29 09:54:31 -07:00
if ( ! Table $ str_get ( used_args , pseudoname ) ) {
2024-08-13 01:24:36 -07:00
CORD value ;
2024-08-17 20:43:30 -07:00
if ( spec_arg - > type - > tag = = IntType & & call_arg - > value - > tag = = Int ) {
value = compile_int_to_type ( env , call_arg - > value , spec_arg - > type ) ;
} else if ( spec_arg - > type - > tag = = NumType & & call_arg - > value - > tag = = Int ) {
2024-09-11 11:53:48 -07:00
OptionalInt_t int_val = Int $ from_str ( Match ( call_arg - > value , Int ) - > str ) ;
if ( int_val . small = = 0 )
code_err ( call_arg - > value , " Failed to parse this integer " ) ;
2024-08-14 11:57:01 -07:00
double n = Int_to_Num ( int_val ) ;
2024-08-18 08:49:51 -07:00
value = CORD_asprintf ( Match ( spec_arg - > type , NumType ) - > bits = = TYPE_NBITS64
2024-08-18 22:46:37 -07:00
? " N64(%.20g) " : " N32(%.10g) " , n ) ;
2024-08-13 01:24:36 -07:00
} else {
env_t * arg_env = with_enum_scope ( env , spec_arg - > type ) ;
type_t * actual_t = get_type ( arg_env , call_arg - > value ) ;
2024-09-06 01:11:00 -07:00
value = compile_maybe_incref ( arg_env , call_arg - > value ) ;
2024-08-13 01:24:36 -07:00
if ( ! promote ( arg_env , & value , actual_t , spec_arg - > type ) )
code_err ( call_arg - > value , " This argument is supposed to be a %T, but this value is a %T " , spec_arg - > type , actual_t ) ;
}
2024-03-29 09:54:31 -07:00
Table $ str_set ( & used_args , pseudoname , call_arg ) ;
2024-03-08 11:23:16 -08:00
if ( code ) code = CORD_cat ( code , " , " ) ;
2024-03-09 13:03:38 -08:00
code = CORD_cat ( code , value ) ;
2024-03-08 11:23:16 -08:00
goto found_it ;
}
}
if ( spec_arg - > default_val ) {
if ( code ) code = CORD_cat ( code , " , " ) ;
2024-09-06 01:14:50 -07:00
code = CORD_cat ( code , compile_maybe_incref ( default_scope , spec_arg - > default_val ) ) ;
2024-03-08 11:23:16 -08:00
goto found_it ;
}
assert ( spec_arg - > name ) ;
code_err ( call_ast , " The required argument '%s' was not provided " , spec_arg - > name ) ;
found_it : continue ;
}
int64_t i = 1 ;
for ( arg_ast_t * call_arg = call_args ; call_arg ; call_arg = call_arg - > next ) {
if ( call_arg - > name ) {
2024-03-29 09:54:31 -07:00
if ( ! Table $ str_get ( used_args , call_arg - > name ) )
2024-03-08 11:23:16 -08:00
code_err ( call_arg - > value , " There is no argument with the name '%s' " , call_arg - > name ) ;
} else {
const char * pseudoname = heap_strf ( " %ld " , i + + ) ;
2024-03-29 09:54:31 -07:00
if ( ! Table $ str_get ( used_args , pseudoname ) )
2024-03-08 11:23:16 -08:00
code_err ( call_arg - > value , " This is one argument too many! " ) ;
}
}
return code ;
}
2024-08-10 17:50:15 -07:00
CORD compile_math_method ( env_t * env , binop_e op , ast_t * lhs , ast_t * rhs , type_t * required_type )
2024-07-01 08:12:00 -07:00
{
2024-08-10 17:50:15 -07:00
// Math methods are things like plus(), minus(), etc. If we don't find a
2024-07-01 08:12:00 -07:00
// matching method, return CORD_EMPTY.
const char * method_name = binop_method_names [ op ] ;
if ( ! method_name )
return CORD_EMPTY ;
type_t * lhs_t = get_type ( env , lhs ) ;
type_t * rhs_t = get_type ( env , rhs ) ;
2024-08-10 17:50:15 -07:00
# define binding_works(b, lhs_t, rhs_t, ret_t) \
( b & & b - > type - > tag = = FunctionType & & ( { auto fn = Match ( b - > type , FunctionType ) ; \
( type_eq ( fn - > ret , ret_t ) \
& & ( fn - > args & & type_eq ( fn - > args - > type , lhs_t ) ) \
2024-09-24 10:17:44 -07:00
& & ( fn - > args - > next & & can_promote ( rhs_t , fn - > args - > next - > type ) ) \
2024-08-10 17:50:15 -07:00
& & ( ! required_type | | type_eq ( required_type , fn - > ret ) ) ) ; } ) )
2024-09-24 10:17:44 -07:00
arg_ast_t * args = new ( arg_ast_t , . value = lhs , . next = new ( arg_ast_t , . value = rhs ) ) ;
2024-08-10 17:50:15 -07:00
switch ( op ) {
case BINOP_MULT : {
2024-08-12 22:30:25 -07:00
if ( type_eq ( lhs_t , rhs_t ) ) {
binding_t * b = get_namespace_binding ( env , lhs , binop_method_names [ op ] ) ;
if ( binding_works ( b , lhs_t , rhs_t , lhs_t ) )
2024-09-24 10:17:44 -07:00
return CORD_all ( b - > code , " ( " , compile_arguments ( env , lhs , Match ( b - > type , FunctionType ) - > args , args ) , " ) " ) ;
2024-08-18 09:47:29 -07:00
} else if ( lhs_t - > tag = = NumType | | lhs_t - > tag = = IntType | | lhs_t - > tag = = BigIntType ) {
2024-08-10 17:50:15 -07:00
binding_t * b = get_namespace_binding ( env , rhs , " scaled_by " ) ;
2024-09-24 10:17:44 -07:00
if ( binding_works ( b , rhs_t , lhs_t , rhs_t ) ) {
REVERSE_LIST ( args ) ;
return CORD_all ( b - > code , " ( " , compile_arguments ( env , lhs , Match ( b - > type , FunctionType ) - > args , args ) , " ) " ) ;
}
2024-08-18 09:47:29 -07:00
} else if ( rhs_t - > tag = = NumType | | rhs_t - > tag = = IntType | | rhs_t - > tag = = BigIntType ) {
2024-08-10 17:50:15 -07:00
binding_t * b = get_namespace_binding ( env , lhs , " scaled_by " ) ;
if ( binding_works ( b , lhs_t , rhs_t , lhs_t ) )
2024-09-24 10:17:44 -07:00
return CORD_all ( b - > code , " ( " , compile_arguments ( env , lhs , Match ( b - > type , FunctionType ) - > args , args ) , " ) " ) ;
2024-07-01 08:12:00 -07:00
}
2024-08-10 17:50:15 -07:00
break ;
}
case BINOP_PLUS : case BINOP_MINUS : case BINOP_AND : case BINOP_OR : case BINOP_XOR : {
if ( type_eq ( lhs_t , rhs_t ) ) {
binding_t * b = get_namespace_binding ( env , lhs , binop_method_names [ op ] ) ;
if ( binding_works ( b , lhs_t , rhs_t , lhs_t ) )
return CORD_all ( b - > code , " ( " , compile ( env , lhs ) , " , " , compile ( env , rhs ) , " ) " ) ;
2024-07-01 08:12:00 -07:00
}
2024-08-10 17:50:15 -07:00
break ;
}
case BINOP_DIVIDE : case BINOP_MOD : case BINOP_MOD1 : {
2024-09-24 10:17:44 -07:00
if ( is_numeric_type ( rhs_t ) ) {
2024-08-10 17:50:15 -07:00
binding_t * b = get_namespace_binding ( env , lhs , binop_method_names [ op ] ) ;
if ( binding_works ( b , lhs_t , rhs_t , lhs_t ) )
2024-09-24 10:17:44 -07:00
return CORD_all ( b - > code , " ( " , compile_arguments ( env , lhs , Match ( b - > type , FunctionType ) - > args , args ) , " ) " ) ;
2024-08-10 17:50:15 -07:00
}
break ;
}
2024-11-03 12:48:13 -08:00
case BINOP_LSHIFT : case BINOP_RSHIFT : case BINOP_ULSHIFT : case BINOP_URSHIFT : {
2024-08-18 09:47:29 -07:00
if ( rhs_t - > tag = = IntType | | rhs_t - > tag = = BigIntType ) {
2024-08-10 17:50:15 -07:00
binding_t * b = get_namespace_binding ( env , lhs , binop_method_names [ op ] ) ;
if ( binding_works ( b , lhs_t , rhs_t , lhs_t ) )
2024-09-24 10:17:44 -07:00
return CORD_all ( b - > code , " ( " , compile_arguments ( env , lhs , Match ( b - > type , FunctionType ) - > args , args ) , " ) " ) ;
2024-08-10 17:50:15 -07:00
}
break ;
}
case BINOP_POWER : {
2024-08-18 09:47:29 -07:00
if ( rhs_t - > tag = = NumType | | rhs_t - > tag = = IntType | | rhs_t - > tag = = BigIntType ) {
2024-08-10 17:50:15 -07:00
binding_t * b = get_namespace_binding ( env , lhs , binop_method_names [ op ] ) ;
if ( binding_works ( b , lhs_t , rhs_t , lhs_t ) )
2024-09-24 10:17:44 -07:00
return CORD_all ( b - > code , " ( " , compile_arguments ( env , lhs , Match ( b - > type , FunctionType ) - > args , args ) , " ) " ) ;
2024-08-10 17:50:15 -07:00
}
break ;
}
default : break ;
2024-07-01 08:12:00 -07:00
}
return CORD_EMPTY ;
}
2024-09-03 21:18:03 -07:00
static CORD compile_string_literal ( CORD literal )
{
CORD code = " \" " ;
CORD_pos i ;
2024-10-29 20:14:31 -07:00
# pragma GCC diagnostic push
2024-09-08 14:17:15 -07:00
# pragma GCC diagnostic ignored "-Wsign-conversion"
2024-09-03 21:18:03 -07:00
CORD_FOR ( i , literal ) {
2024-10-29 20:14:31 -07:00
# pragma GCC diagnostic pop
2024-09-03 21:18:03 -07:00
char c = CORD_pos_fetch ( i ) ;
switch ( c ) {
case ' \\ ' : code = CORD_cat ( code , " \\ \\ " ) ; break ;
case ' " ' : code = CORD_cat ( code , " \\ \" " ) ; break ;
case ' \a ' : code = CORD_cat ( code , " \\ a " ) ; break ;
case ' \b ' : code = CORD_cat ( code , " \\ b " ) ; break ;
case ' \n ' : code = CORD_cat ( code , " \\ n " ) ; break ;
case ' \r ' : code = CORD_cat ( code , " \\ r " ) ; break ;
case ' \t ' : code = CORD_cat ( code , " \\ t " ) ; break ;
case ' \v ' : code = CORD_cat ( code , " \\ v " ) ; break ;
default : {
if ( isprint ( c ) )
code = CORD_cat_char ( code , c ) ;
else
CORD_sprintf ( & code , " %r \\ x%02X \" \" " , code , ( uint8_t ) c ) ;
break ;
}
}
}
return CORD_cat ( code , " \" " ) ;
}
static bool string_literal_is_all_ascii ( CORD literal )
{
CORD_pos i ;
2024-10-29 20:14:31 -07:00
# pragma GCC diagnostic push
# pragma GCC diagnostic ignored "-Wsign-conversion"
2024-09-03 21:18:03 -07:00
CORD_FOR ( i , literal ) {
2024-10-29 20:14:31 -07:00
# pragma GCC diagnostic pop
2024-09-03 21:18:03 -07:00
if ( ! isascii ( CORD_pos_fetch ( i ) ) )
return false ;
}
return true ;
}
2024-09-11 11:53:48 -07:00
CORD compile_null ( type_t * t )
{
2024-09-11 21:55:43 -07:00
if ( t - > tag = = OptionalType )
t = Match ( t , OptionalType ) - > type ;
2024-09-11 11:53:48 -07:00
if ( t = = THREAD_TYPE ) return " NULL " ;
switch ( t - > tag ) {
case BigIntType : return " NULL_INT " ;
case IntType : {
switch ( Match ( t , IntType ) - > bits ) {
case TYPE_IBITS8 : return " NULL_INT8 " ;
case TYPE_IBITS16 : return " NULL_INT16 " ;
case TYPE_IBITS32 : return " NULL_INT32 " ;
case TYPE_IBITS64 : return " NULL_INT64 " ;
default : errx ( 1 , " Invalid integer bit size " ) ;
}
break ;
}
case BoolType : return " NULL_BOOL " ;
2024-09-15 12:33:47 -07:00
case ByteType : return " NULL_BYTE " ;
2024-09-11 11:53:48 -07:00
case ArrayType : return " NULL_ARRAY " ;
case TableType : return " NULL_TABLE " ;
case SetType : return " NULL_TABLE " ;
case ChannelType : return " NULL " ;
case TextType : return " NULL_TEXT " ;
case CStringType : return " NULL " ;
2024-11-17 11:49:03 -08:00
case MomentType : return " NULL_MOMENT " ;
2024-09-11 11:53:48 -07:00
case PointerType : return CORD_all ( " (( " , compile_type ( t ) , " )NULL) " ) ;
case ClosureType : return " NULL_CLOSURE " ;
case NumType : return " nan( \" null \" ) " ;
2024-10-08 18:10:36 -07:00
case StructType : return CORD_all ( " (( " , compile_type ( Type ( OptionalType , . type = t ) ) , " ){.is_null=yes}) " ) ;
2024-09-11 11:53:48 -07:00
case EnumType : {
env_t * enum_env = Match ( t , EnumType ) - > env ;
2024-09-24 11:54:22 -07:00
return CORD_all ( " (( " , compile_type ( t ) , " ) { " , namespace_prefix(enum_env, enum_env->namespace), " null } ) " );
2024-09-11 11:53:48 -07:00
}
default : compiler_err ( NULL , NULL , NULL , " Null isn't implemented for this type: %T " , t ) ;
}
}
2024-09-12 11:27:13 -07:00
static ast_t * add_to_table_comprehension ( ast_t * entry , ast_t * subject )
{
auto e = Match ( entry , TableEntry ) ;
return WrapAST ( entry , MethodCall , . name = " set " , . self = subject ,
. args = new ( arg_ast_t , . value = e - > key , . next = new ( arg_ast_t , . value = e - > value ) ) ) ;
}
static ast_t * add_to_array_comprehension ( ast_t * item , ast_t * subject )
{
return WrapAST ( item , MethodCall , . name = " insert " , . self = subject , . args = new ( arg_ast_t , . value = item ) ) ;
}
static ast_t * add_to_set_comprehension ( ast_t * item , ast_t * subject )
{
return WrapAST ( item , MethodCall , . name = " add " , . self = subject , . args = new ( arg_ast_t , . value = item ) ) ;
}
2024-11-05 12:18:32 -08:00
static CORD compile_num_to_type ( ast_t * ast , type_t * type )
{
double n = Match ( ast , Num ) - > n ;
if ( type - > tag ! = NumType )
code_err ( ast , " I can't compile a number literal to a %T " , type ) ;
switch ( Match ( type , NumType ) - > bits ) {
case TYPE_NBITS64 :
return CORD_asprintf ( " N64(%.20g) " , n) ;
case TYPE_NBITS32 :
return CORD_asprintf ( " N32(%.10g) " , n) ;
default : code_err ( ast , " This is not a valid number bit width " ) ;
}
}
2024-02-13 11:42:33 -08:00
CORD compile ( env_t * env , ast_t * ast )
2024-02-04 12:23:59 -08:00
{
switch ( ast - > tag ) {
2024-09-11 10:55:41 -07:00
case Null : {
2024-11-21 10:00:53 -08:00
if ( ! Match ( ast , Null ) - > type )
code_err ( ast , " This 'NULL' needs to specify what type it is using `!Type` syntax " ) ;
2024-09-11 10:55:41 -07:00
type_t * t = parse_type_ast ( env , Match ( ast , Null ) - > type ) ;
2024-09-11 11:53:48 -07:00
return compile_null ( t ) ;
2024-04-12 10:09:31 -07:00
}
2024-02-12 00:00:31 -08:00
case Bool : return Match ( ast , Bool ) - > b ? " yes " : " no " ;
2024-11-17 11:49:03 -08:00
case Moment : {
auto moment = Match ( ast , Moment ) - > moment ;
return CORD_asprintf ( " ((Moment_t) { . tv_sec = % ld , . tv_usec = % ld } ) " , moment.tv_sec, moment.tv_usec);
2024-09-30 10:55:55 -07:00
}
2024-02-29 10:28:39 -08:00
case Var : {
binding_t * b = get_binding ( env , Match ( ast , Var ) - > name ) ;
if ( b )
2024-04-16 10:50:07 -07:00
return b - > code ? b - > code : CORD_cat ( " $ " , Match ( ast , Var ) - > name ) ;
return CORD_cat ( " $ " , Match ( ast , Var ) - > name ) ;
2024-02-29 10:49:24 -08:00
// code_err(ast, "I don't know of any variable by this name");
2024-02-29 10:28:39 -08:00
}
2024-08-12 22:30:25 -07:00
case Int : {
const char * str = Match ( ast , Int ) - > str ;
2024-09-11 11:53:48 -07:00
OptionalInt_t int_val = Int $ from_str ( str ) ;
if ( int_val . small = = 0 )
code_err ( ast , " Failed to parse this integer " ) ;
2024-08-12 22:30:25 -07:00
mpz_t i ;
mpz_init_set_int ( i , int_val ) ;
2024-08-13 11:13:02 -07:00
if ( mpz_cmpabs_ui ( i , BIGGEST_SMALL_INT ) < = 0 ) {
2024-08-17 12:17:41 -07:00
return CORD_asprintf ( " I_small(%s) " , str) ;
2024-08-13 11:13:02 -07:00
} else if ( mpz_cmp_si ( i , INT64_MAX ) < = 0 & & mpz_cmp_si ( i , INT64_MIN ) > = 0 ) {
2024-08-14 11:57:01 -07:00
return CORD_asprintf ( " Int64_to_Int(%s) " , str) ;
2024-08-12 22:30:25 -07:00
} else {
2024-09-11 12:03:39 -07:00
return CORD_asprintf ( " Int$from_str( \" %s \" ) " , str) ;
2024-08-12 22:30:25 -07:00
}
}
2024-02-13 20:37:24 -08:00
case Num : {
2024-11-05 12:18:32 -08:00
return CORD_asprintf ( " N64(%.20g) " , Match(ast, Num)->n) ;
2024-02-13 20:37:24 -08:00
}
2024-03-14 10:47:40 -07:00
case Not : {
ast_t * value = Match ( ast , Not ) - > value ;
2024-09-16 13:10:05 -07:00
type_t * t = get_type ( env , value ) ;
2024-08-12 22:30:25 -07:00
binding_t * b = get_namespace_binding ( env , value , " negated " ) ;
if ( b & & b - > type - > tag = = FunctionType ) {
auto fn = Match ( b - > type , FunctionType ) ;
if ( fn - > args & & can_promote ( t , get_arg_type ( env , fn - > args ) ) )
return CORD_all ( b - > code , " ( " , compile_arguments ( env , ast , fn - > args , new ( arg_ast_t , . value = value ) ) , " ) " ) ;
}
2024-03-14 10:47:40 -07:00
if ( t - > tag = = BoolType )
return CORD_all ( " !( " , compile ( env , value ) , " ) " ) ;
2024-09-15 12:33:47 -07:00
else if ( t - > tag = = IntType | | t - > tag = = ByteType )
2024-03-14 10:47:40 -07:00
return CORD_all ( " ~( " , compile ( env , value ) , " ) " ) ;
2024-08-18 17:39:57 -07:00
else if ( t - > tag = = ArrayType )
return CORD_all ( " (( " , compile ( env , value ) , " ).length == 0) " ) ;
else if ( t - > tag = = SetType | | t - > tag = = TableType )
return CORD_all ( " (( " , compile ( env , value ) , " ).entries.length == 0) " ) ;
2024-03-14 10:47:40 -07:00
else if ( t - > tag = = TextType )
2024-08-18 17:39:57 -07:00
return CORD_all ( " ( " , compile ( env , value ) , " == CORD_EMPTY) " ) ;
2024-09-10 22:31:31 -07:00
else if ( t - > tag = = OptionalType )
2024-09-11 20:55:49 -07:00
return check_null ( t , compile ( env , value ) ) ;
2024-08-10 17:50:15 -07:00
code_err ( ast , " I don't know how to negate values of type %T " , t ) ;
2024-03-14 10:47:40 -07:00
}
2024-07-04 10:37:23 -07:00
case Negative : {
ast_t * value = Match ( ast , Negative ) - > value ;
type_t * t = get_type ( env , value ) ;
2024-08-10 17:50:15 -07:00
binding_t * b = get_namespace_binding ( env , value , " negative " ) ;
2024-07-04 10:37:23 -07:00
if ( b & & b - > type - > tag = = FunctionType ) {
auto fn = Match ( b - > type , FunctionType ) ;
if ( fn - > args & & can_promote ( t , get_arg_type ( env , fn - > args ) ) )
return CORD_all ( b - > code , " ( " , compile_arguments ( env , ast , fn - > args , new ( arg_ast_t , . value = value ) ) , " ) " ) ;
}
2024-08-12 22:30:25 -07:00
if ( t - > tag = = IntType | | t - > tag = = NumType )
return CORD_all ( " -( " , compile ( env , value ) , " ) " ) ;
2024-07-04 10:37:23 -07:00
code_err ( ast , " I don't know how to get the negative value of type %T " , t ) ;
}
2024-10-08 10:35:18 -07:00
// TODO: for constructors, do new(T, ...) instead of heap((T){...})
2024-04-16 10:50:07 -07:00
case HeapAllocate : return CORD_asprintf ( " heap(%r) " , compile ( env , Match ( ast , HeapAllocate ) - > value ) ) ;
2024-04-30 10:18:47 -07:00
case Optional : {
2024-09-11 08:08:15 -07:00
ast_t * value = Match ( ast , Optional ) - > value ;
CORD value_code = compile ( env , value ) ;
2024-09-11 09:01:17 -07:00
return promote_to_optional ( get_type ( env , value ) , value_code ) ;
2024-04-30 10:18:47 -07:00
}
2024-09-11 21:13:53 -07:00
case NonOptional : {
ast_t * value = Match ( ast , NonOptional ) - > value ;
type_t * t = get_type ( env , value ) ;
CORD value_code = compile ( env , value ) ;
return CORD_all ( " ({ " , compile_declaration ( t , " opt " ) , " = " , value_code , " ; " ,
" if ( " , check_null ( t , " opt " ) , " ) \n " ,
CORD_asprintf ( " fail_source(%r, %ld, %ld, \" This value was expected to be non-null, but it's null! \" ); \n " ,
CORD_quoted ( ast - > file - > filename ) ,
( long ) ( value - > start - value - > file - > text ) ,
( long ) ( value - > end - value - > file - > text ) ) ,
optional_into_nonnull ( t , " opt " ) , " ; }) " ) ;
}
2024-02-04 12:23:59 -08:00
case BinaryOp : {
auto binop = Match ( ast , BinaryOp ) ;
2024-08-10 17:50:15 -07:00
CORD method_call = compile_math_method ( env , binop - > op , binop - > lhs , binop - > rhs , NULL ) ;
2024-07-01 08:12:00 -07:00
if ( method_call ! = CORD_EMPTY )
return method_call ;
2024-02-22 10:18:47 -08:00
type_t * lhs_t = get_type ( env , binop - > lhs ) ;
type_t * rhs_t = get_type ( env , binop - > rhs ) ;
2024-09-16 12:48:06 -07:00
2024-09-16 13:06:19 -07:00
if ( binop - > op = = BINOP_OR & & lhs_t - > tag = = OptionalType ) {
if ( rhs_t - > tag = = AbortType | | rhs_t - > tag = = ReturnType ) {
return CORD_all ( " ({ " , compile_declaration ( lhs_t , " lhs " ) , " = " , compile ( env , binop - > lhs ) , " ; " ,
" if ( " , check_null ( lhs_t , " lhs " ) , " ) " , compile_statement ( env , binop - > rhs ) , " " ,
optional_into_nonnull ( lhs_t , " lhs " ) , " ; }) " ) ;
} else if ( rhs_t - > tag = = OptionalType & & type_eq ( lhs_t , rhs_t ) ) {
return CORD_all ( " ({ " , compile_declaration ( lhs_t , " lhs " ) , " = " , compile ( env , binop - > lhs ) , " ; " ,
check_null ( lhs_t , " lhs " ) , " ? " , compile ( env , binop - > rhs ) , " : lhs; }) " ) ;
} else if ( rhs_t - > tag ! = OptionalType & & type_eq ( Match ( lhs_t , OptionalType ) - > type , rhs_t ) ) {
return CORD_all ( " ({ " , compile_declaration ( lhs_t , " lhs " ) , " = " , compile ( env , binop - > lhs ) , " ; " ,
check_null ( lhs_t , " lhs " ) , " ? " , compile ( env , binop - > rhs ) , " : " ,
optional_into_nonnull ( lhs_t , " lhs " ) , " ; }) " ) ;
} else {
code_err ( ast , " I don't know how to do an 'or' operation between %T and %T " , lhs_t , rhs_t ) ;
}
2024-11-07 10:27:09 -08:00
} else if ( binop - > op = = BINOP_AND & & lhs_t - > tag = = OptionalType ) {
if ( rhs_t - > tag = = AbortType | | rhs_t - > tag = = ReturnType ) {
return CORD_all ( " ({ " , compile_declaration ( lhs_t , " lhs " ) , " = " , compile ( env , binop - > lhs ) , " ; " ,
" if (! " , check_null ( lhs_t , " lhs " ) , " ) " , compile_statement ( env , binop - > rhs ) , " " ,
optional_into_nonnull ( lhs_t , " lhs " ) , " ; }) " ) ;
} else if ( rhs_t - > tag = = OptionalType & & type_eq ( lhs_t , rhs_t ) ) {
return CORD_all ( " ({ " , compile_declaration ( lhs_t , " lhs " ) , " = " , compile ( env , binop - > lhs ) , " ; " ,
check_null ( lhs_t , " lhs " ) , " ? lhs : " , compile ( env , binop - > rhs ) , " ; }) " ) ;
} else {
code_err ( ast , " I don't know how to do an 'or' operation between %T and %T " , lhs_t , rhs_t ) ;
}
2024-09-16 12:48:06 -07:00
}
2024-11-04 12:24:10 -08:00
CORD lhs , rhs ;
if ( lhs_t - > tag = = BigIntType & & rhs_t - > tag ! = BigIntType & & is_numeric_type ( rhs_t ) & & binop - > lhs - > tag = = Int ) {
lhs = compile_int_to_type ( env , binop - > lhs , rhs_t ) ;
lhs_t = rhs_t ;
rhs = compile ( env , binop - > rhs ) ;
} else if ( rhs_t - > tag = = BigIntType & & lhs_t - > tag ! = BigIntType & & is_numeric_type ( lhs_t ) & & binop - > rhs - > tag = = Int ) {
lhs = compile ( env , binop - > lhs ) ;
rhs = compile_int_to_type ( env , binop - > rhs , lhs_t ) ;
rhs_t = lhs_t ;
} else {
lhs = compile ( env , binop - > lhs ) ;
rhs = compile ( env , binop - > rhs ) ;
2024-11-03 12:15:03 -08:00
}
2024-02-22 10:18:47 -08:00
type_t * operand_t ;
2024-03-09 13:03:38 -08:00
if ( promote ( env , & rhs , rhs_t , lhs_t ) )
2024-02-22 10:18:47 -08:00
operand_t = lhs_t ;
2024-03-09 13:03:38 -08:00
else if ( promote ( env , & lhs , lhs_t , rhs_t ) )
2024-02-22 10:18:47 -08:00
operand_t = rhs_t ;
else
2024-02-22 10:26:43 -08:00
code_err ( ast , " I can't do operations between %T and %T " , lhs_t , rhs_t ) ;
2024-02-17 17:21:01 -08:00
2024-02-04 12:23:59 -08:00
switch ( binop - > op ) {
2024-02-22 10:09:46 -08:00
case BINOP_POWER : {
2024-08-12 23:09:18 -07:00
if ( operand_t - > tag ! = NumType )
2024-11-03 12:15:03 -08:00
code_err ( ast , " Exponentiation is only supported for Num types, not %T " , operand_t ) ;
2024-08-18 08:49:51 -07:00
if ( operand_t - > tag = = NumType & & Match ( operand_t , NumType ) - > bits = = TYPE_NBITS32 )
2024-02-22 10:09:46 -08:00
return CORD_all ( " powf( " , lhs , " , " , rhs , " ) " ) ;
else
return CORD_all ( " pow( " , lhs , " , " , rhs , " ) " ) ;
}
2024-02-17 17:25:31 -08:00
case BINOP_MULT : {
2024-09-15 12:33:47 -07:00
if ( operand_t - > tag ! = IntType & & operand_t - > tag ! = NumType & & operand_t - > tag ! = ByteType )
2024-11-03 12:15:03 -08:00
code_err ( ast , " Math operations are only supported for values of the same numeric type, not %T vs %T " , lhs_t , rhs_t ) ;
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " * " , rhs , " ) " ) ;
2024-02-17 17:25:31 -08:00
}
case BINOP_DIVIDE : {
2024-09-15 12:33:47 -07:00
if ( operand_t - > tag ! = IntType & & operand_t - > tag ! = NumType & & operand_t - > tag ! = ByteType )
2024-11-03 12:15:03 -08:00
code_err ( ast , " Math operations are only supported for values of the same numeric type, not %T vs %T " , lhs_t , rhs_t ) ;
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " / " , rhs , " ) " ) ;
2024-02-17 17:25:31 -08:00
}
case BINOP_MOD : {
2024-09-15 12:33:47 -07:00
if ( operand_t - > tag ! = IntType & & operand_t - > tag ! = NumType & & operand_t - > tag ! = ByteType )
2024-11-03 12:15:03 -08:00
code_err ( ast , " Math operations are only supported for values of the same numeric type, not %T vs %T " , lhs_t , rhs_t ) ;
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " % " , rhs , " ) " ) ;
2024-02-17 17:25:31 -08:00
}
case BINOP_MOD1 : {
2024-09-15 12:33:47 -07:00
if ( operand_t - > tag ! = IntType & & operand_t - > tag ! = NumType & & operand_t - > tag ! = ByteType )
2024-11-03 12:15:03 -08:00
code_err ( ast , " Math operations are only supported for values of the same numeric type, not %T vs %T " , lhs_t , rhs_t ) ;
2024-08-10 17:40:41 -07:00
return CORD_all ( " (((( " , lhs , " ) - 1 ) % ( " , rhs, " ) ) + 1 ) " ) ;
2024-02-17 17:25:31 -08:00
}
case BINOP_PLUS : {
2024-09-15 12:33:47 -07:00
if ( operand_t - > tag ! = IntType & & operand_t - > tag ! = NumType & & operand_t - > tag ! = ByteType )
2024-11-03 12:15:03 -08:00
code_err ( ast , " Math operations are only supported for values of the same numeric type, not %T vs %T " , lhs_t , rhs_t ) ;
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " + " , rhs , " ) " ) ;
2024-02-17 17:25:31 -08:00
}
case BINOP_MINUS : {
2024-09-15 12:33:47 -07:00
if ( operand_t - > tag ! = IntType & & operand_t - > tag ! = NumType & & operand_t - > tag ! = ByteType )
2024-11-03 12:15:03 -08:00
code_err ( ast , " Math operations are only supported for values of the same numeric type, not %T vs %T " , lhs_t , rhs_t ) ;
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " - " , rhs , " ) " ) ;
2024-02-17 17:25:31 -08:00
}
case BINOP_LSHIFT : {
2024-09-15 12:33:47 -07:00
if ( operand_t - > tag ! = IntType & & operand_t - > tag ! = NumType & & operand_t - > tag ! = ByteType )
2024-11-03 12:15:03 -08:00
code_err ( ast , " Math operations are only supported for values of the same numeric type, not %T vs %T " , lhs_t , rhs_t ) ;
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " << " , rhs , " ) " ) ;
2024-02-17 17:25:31 -08:00
}
case BINOP_RSHIFT : {
2024-09-15 12:33:47 -07:00
if ( operand_t - > tag ! = IntType & & operand_t - > tag ! = NumType & & operand_t - > tag ! = ByteType )
2024-11-03 12:15:03 -08:00
code_err ( ast , " Math operations are only supported for values of the same numeric type, not %T vs %T " , lhs_t , rhs_t ) ;
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " >> " , rhs , " ) " ) ;
2024-02-17 17:25:31 -08:00
}
2024-11-03 12:48:13 -08:00
case BINOP_ULSHIFT : {
if ( operand_t - > tag ! = IntType & & operand_t - > tag ! = NumType & & operand_t - > tag ! = ByteType )
code_err ( ast , " Math operations are only supported for values of the same numeric type, not %T vs %T " , lhs_t , rhs_t ) ;
return CORD_all ( " ( " , compile_type ( operand_t ) , " )(( " , compile_unsigned_type ( lhs_t ) , " ) " , lhs , " << " , rhs , " ) " ) ;
}
case BINOP_URSHIFT : {
if ( operand_t - > tag ! = IntType & & operand_t - > tag ! = NumType & & operand_t - > tag ! = ByteType )
code_err ( ast , " Math operations are only supported for values of the same numeric type, not %T vs %T " , lhs_t , rhs_t ) ;
return CORD_all ( " ( " , compile_type ( operand_t ) , " )(( " , compile_unsigned_type ( lhs_t ) , " ) " , lhs , " >> " , rhs , " ) " ) ;
}
2024-02-17 17:21:01 -08:00
case BINOP_EQ : {
2024-02-22 10:18:47 -08:00
switch ( operand_t - > tag ) {
2024-08-18 08:49:51 -07:00
case BigIntType :
return CORD_all ( " Int$equal_value( " , lhs , " , " , rhs , " ) " ) ;
2024-09-15 12:33:47 -07:00
case BoolType : case ByteType : case IntType : case NumType : case PointerType : case FunctionType :
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " == " , rhs , " ) " ) ;
2024-02-17 17:21:01 -08:00
default :
2024-05-12 17:12:00 -07:00
return CORD_asprintf ( " generic_equal(stack(%r) , stack ( % r ) , % r ) " , lhs, rhs, compile_type_info(env, operand_t)) ;
2024-02-17 17:21:01 -08:00
}
}
case BINOP_NE : {
2024-02-22 10:18:47 -08:00
switch ( operand_t - > tag ) {
2024-08-18 08:49:51 -07:00
case BigIntType :
return CORD_all ( " !Int$equal_value( " , lhs , " , " , rhs , " ) " ) ;
2024-09-15 12:33:47 -07:00
case BoolType : case ByteType : case IntType : case NumType : case PointerType : case FunctionType :
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " != " , rhs , " ) " ) ;
2024-02-17 17:21:01 -08:00
default :
2024-05-12 17:12:00 -07:00
return CORD_asprintf ( " !generic_equal(stack(%r) , stack ( % r ) , % r ) " , lhs, rhs, compile_type_info(env, operand_t)) ;
2024-02-17 17:21:01 -08:00
}
}
case BINOP_LT : {
2024-02-22 10:18:47 -08:00
switch ( operand_t - > tag ) {
2024-08-18 08:49:51 -07:00
case BigIntType :
return CORD_all ( " (Int$compare_value( " , lhs , " , " , rhs , " ) < 0 ) " ) ;
2024-09-15 12:33:47 -07:00
case BoolType : case ByteType : case IntType : case NumType : case PointerType : case FunctionType :
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " < " , rhs , " ) " ) ;
2024-02-17 17:21:01 -08:00
default :
2024-05-12 17:12:00 -07:00
return CORD_asprintf ( " (generic_compare(stack(%r) , stack ( % r ) , % r ) < 0 ) " , lhs, rhs, compile_type_info(env, operand_t)) ;
2024-02-17 17:21:01 -08:00
}
}
case BINOP_LE : {
2024-02-22 10:18:47 -08:00
switch ( operand_t - > tag ) {
2024-08-18 08:49:51 -07:00
case BigIntType :
return CORD_all ( " (Int$compare_value( " , lhs , " , " , rhs , " ) < = 0 ) " ) ;
2024-09-15 12:33:47 -07:00
case BoolType : case ByteType : case IntType : case NumType : case PointerType : case FunctionType :
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " <= " , rhs , " ) " ) ;
2024-02-17 17:21:01 -08:00
default :
2024-05-12 17:12:00 -07:00
return CORD_asprintf ( " (generic_compare(stack(%r) , stack ( % r ) , % r ) < = 0 ) " , lhs, rhs, compile_type_info(env, operand_t)) ;
2024-02-17 17:21:01 -08:00
}
}
case BINOP_GT : {
2024-02-22 10:18:47 -08:00
switch ( operand_t - > tag ) {
2024-08-18 08:49:51 -07:00
case BigIntType :
return CORD_all ( " (Int$compare_value( " , lhs , " , " , rhs , " ) > 0 ) " ) ;
2024-09-15 12:33:47 -07:00
case BoolType : case ByteType : case IntType : case NumType : case PointerType : case FunctionType :
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " > " , rhs , " ) " ) ;
2024-02-17 17:21:01 -08:00
default :
2024-05-12 17:12:00 -07:00
return CORD_asprintf ( " (generic_compare(stack(%r) , stack ( % r ) , % r ) > 0 ) " , lhs, rhs, compile_type_info(env, operand_t)) ;
2024-02-17 17:21:01 -08:00
}
}
case BINOP_GE : {
2024-02-22 10:18:47 -08:00
switch ( operand_t - > tag ) {
2024-08-18 08:49:51 -07:00
case BigIntType :
return CORD_all ( " (Int$compare_value( " , lhs , " , " , rhs , " ) > = 0 ) " ) ;
2024-09-15 12:33:47 -07:00
case BoolType : case ByteType : case IntType : case NumType : case PointerType : case FunctionType :
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " >= " , rhs , " ) " ) ;
2024-02-17 17:21:01 -08:00
default :
2024-05-12 17:12:00 -07:00
return CORD_asprintf ( " (generic_compare(stack(%r) , stack ( % r ) , % r ) > = 0 ) " , lhs, rhs, compile_type_info(env, operand_t)) ;
2024-02-17 17:21:01 -08:00
}
}
2024-02-17 17:25:31 -08:00
case BINOP_AND : {
2024-02-22 10:18:47 -08:00
if ( operand_t - > tag = = BoolType )
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " && " , rhs , " ) " ) ;
2024-09-15 12:33:47 -07:00
else if ( operand_t - > tag = = IntType | | operand_t - > tag = = ByteType )
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " & " , rhs , " ) " ) ;
2024-02-17 17:25:31 -08:00
else
2024-11-07 10:27:09 -08:00
code_err ( ast , " The 'and' operator isn't supported for %T and %T " , lhs_t , rhs_t ) ;
2024-02-17 17:25:31 -08:00
}
2024-04-02 10:08:06 -07:00
case BINOP_CMP : {
2024-05-12 17:12:00 -07:00
return CORD_all ( " generic_compare(stack( " , lhs , " ) , stack ( " , rhs, " ) , " , compile_type_info(env, operand_t), " ) " ) ;
2024-04-02 10:08:06 -07:00
}
2024-02-17 17:25:31 -08:00
case BINOP_OR : {
2024-02-22 10:18:47 -08:00
if ( operand_t - > tag = = BoolType )
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " || " , rhs , " ) " ) ;
2024-09-15 12:33:47 -07:00
else if ( operand_t - > tag = = IntType | | operand_t - > tag = = ByteType )
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " | " , rhs , " ) " ) ;
2024-02-17 17:25:31 -08:00
else
2024-11-07 10:27:09 -08:00
code_err ( ast , " The 'or' operator isn't supported for %T and %T " , lhs_t , rhs_t ) ;
2024-02-17 17:25:31 -08:00
}
2024-03-14 10:28:30 -07:00
case BINOP_XOR : {
2024-09-16 12:48:06 -07:00
// TODO: support optional values in `xor` expressions
2024-09-15 12:33:47 -07:00
if ( operand_t - > tag = = BoolType | | operand_t - > tag = = IntType | | operand_t - > tag = = ByteType )
2024-08-10 17:40:41 -07:00
return CORD_all ( " ( " , lhs , " ^ " , rhs , " ) " ) ;
2024-03-14 10:28:30 -07:00
else
2024-11-07 10:27:09 -08:00
code_err ( ast , " The 'xor' operator isn't supported for %T and %T " , lhs_t , rhs_t ) ;
2024-03-14 10:28:30 -07:00
}
case BINOP_CONCAT : {
switch ( operand_t - > tag ) {
case TextType : {
2024-09-08 23:43:15 -07:00
const char * lang = Match ( operand_t , TextType ) - > lang ;
if ( streq ( lang , " Path " ) )
return CORD_all ( " Path$concat( " , lhs , " , " , rhs , " ) " ) ;
2024-09-02 16:18:21 -07:00
return CORD_all ( " Text$concat( " , lhs , " , " , rhs , " ) " ) ;
2024-03-14 10:28:30 -07:00
}
case ArrayType : {
2024-10-08 18:10:36 -07:00
CORD padded_item_size = CORD_asprintf ( " %ld " , type_size ( Match ( operand_t , ArrayType ) - > item_type ) ) ;
2024-08-03 12:33:50 -07:00
return CORD_all ( " Array$concat( " , lhs , " , " , rhs , " , " , padded_item_size , " ) " ) ;
2024-03-14 10:28:30 -07:00
}
default :
code_err ( ast , " Concatenation isn't supported for %T types " , operand_t ) ;
2024-02-23 10:29:20 -08:00
}
2024-02-23 10:24:06 -08:00
}
2024-03-14 10:28:30 -07:00
default : break ;
2024-02-04 12:23:59 -08:00
}
2024-03-14 10:28:30 -07:00
code_err ( ast , " unimplemented binop " ) ;
2024-02-04 12:23:59 -08:00
}
2024-03-03 15:15:45 -08:00
case TextLiteral : {
CORD literal = Match ( ast , TextLiteral ) - > cord ;
2024-02-11 12:31:30 -08:00
if ( literal = = CORD_EMPTY )
2024-09-03 11:48:54 -07:00
return " Text( \" \" ) " ;
2024-09-03 21:18:03 -07:00
if ( string_literal_is_all_ascii ( literal ) )
return CORD_all ( " Text( " , compile_string_literal ( literal ) , " ) " ) ;
else
return CORD_all ( " Text$from_str( " , compile_string_literal ( literal ) , " ) " ) ;
2024-02-04 12:23:59 -08:00
}
2024-03-03 15:15:45 -08:00
case TextJoin : {
2024-03-09 15:22:12 -08:00
const char * lang = Match ( ast , TextJoin ) - > lang ;
2024-09-03 21:18:03 -07:00
type_t * text_t = lang ? Table $ str_get ( * env - > types , lang ) : TEXT_TYPE ;
2024-03-09 15:47:56 -08:00
if ( ! text_t | | text_t - > tag ! = TextType )
code_err ( ast , " %s is not a valid text language name " , lang ) ;
2024-09-03 21:18:03 -07:00
2024-09-24 11:54:22 -07:00
CORD lang_constructor ;
if ( ! lang | | streq ( lang , " Text " ) )
lang_constructor = " Text " ;
else if ( streq ( lang , " Pattern " ) | | streq ( lang , " Path " ) | | streq ( lang , " Shell " ) )
lang_constructor = lang ;
else
lang_constructor = CORD_all ( namespace_prefix ( Match ( text_t , TextType ) - > env , Match ( text_t , TextType ) - > env - > namespace - > parent ) , lang ) ;
2024-09-03 21:18:03 -07:00
2024-03-03 15:15:45 -08:00
ast_list_t * chunks = Match ( ast , TextJoin ) - > children ;
2024-02-10 12:33:35 -08:00
if ( ! chunks ) {
2024-09-03 21:18:03 -07:00
return CORD_all ( lang_constructor , " ( \" \" ) " ) ;
2024-03-09 15:22:12 -08:00
} else if ( ! chunks - > next & & chunks - > ast - > tag = = TextLiteral ) {
2024-09-03 21:18:03 -07:00
CORD literal = Match ( chunks - > ast , TextLiteral ) - > cord ;
if ( string_literal_is_all_ascii ( literal ) )
return CORD_all ( lang_constructor , " ( " , compile_string_literal ( literal ) , " ) " ) ;
return CORD_all ( " (( " , compile_type ( text_t ) , " ) " , compile ( env , chunks - > ast ) , " ) " ) ;
2024-02-10 12:33:35 -08:00
} else {
2024-08-19 10:14:36 -07:00
CORD code = CORD_EMPTY ;
2024-02-10 12:33:35 -08:00
for ( ast_list_t * chunk = chunks ; chunk ; chunk = chunk - > next ) {
2024-03-09 15:22:12 -08:00
CORD chunk_code ;
2024-02-18 01:26:26 -08:00
type_t * chunk_t = get_type ( env , chunk - > ast ) ;
2024-03-09 15:22:12 -08:00
if ( chunk - > ast - > tag = = TextLiteral ) {
chunk_code = compile ( env , chunk - > ast ) ;
} else if ( chunk_t - > tag = = TextType & & streq ( Match ( chunk_t , TextType ) - > lang , lang ) ) {
2024-09-09 13:54:08 -07:00
binding_t * esc = get_lang_escape_function ( env , lang , chunk_t ) ;
if ( esc ) {
arg_t * arg_spec = Match ( esc - > type , FunctionType ) - > args ;
arg_ast_t * args = new ( arg_ast_t , . value = chunk - > ast ) ;
chunk_code = CORD_all ( esc - > code , " ( " , compile_arguments ( env , ast , arg_spec , args ) , " ) " ) ;
} else {
chunk_code = compile ( env , chunk - > ast ) ;
}
2024-08-19 10:23:02 -07:00
} else if ( lang ) {
binding_t * esc = get_lang_escape_function ( env , lang , chunk_t ) ;
if ( ! esc )
code_err ( chunk - > ast , " I don't know how to convert %T to %T " , chunk_t , text_t ) ;
arg_t * arg_spec = Match ( esc - > type , FunctionType ) - > args ;
arg_ast_t * args = new ( arg_ast_t , . value = chunk - > ast ) ;
chunk_code = CORD_all ( esc - > code , " ( " , compile_arguments ( env , ast , arg_spec , args ) , " ) " ) ;
2024-03-09 15:22:12 -08:00
} else {
chunk_code = compile_string ( env , chunk - > ast , " no " ) ;
}
code = CORD_cat ( code , chunk_code ) ;
2024-02-18 01:26:26 -08:00
if ( chunk - > next ) code = CORD_cat ( code , " , " ) ;
2024-02-10 12:33:35 -08:00
}
2024-08-19 10:14:36 -07:00
if ( chunks - > next )
2024-09-03 21:18:03 -07:00
return CORD_all ( lang_constructor , " s( " , code , " ) " ) ;
2024-08-19 10:14:36 -07:00
else
return code ;
2024-02-04 12:23:59 -08:00
}
}
case Block : {
2024-02-04 15:04:41 -08:00
ast_list_t * stmts = Match ( ast , Block ) - > statements ;
if ( stmts & & ! stmts - > next )
2024-03-14 10:28:30 -07:00
return compile ( env , stmts - > ast ) ;
2024-02-04 15:04:41 -08:00
2024-03-14 10:28:30 -07:00
CORD code = " ({ \n " ;
2024-07-04 15:00:01 -07:00
deferral_t * prev_deferred = env - > deferred ;
2024-02-17 14:00:21 -08:00
env = fresh_scope ( env ) ;
2024-05-18 22:46:30 -07:00
for ( ast_list_t * stmt = stmts ; stmt ; stmt = stmt - > next )
prebind_statement ( env , stmt - > ast ) ;
2024-02-04 15:04:41 -08:00
for ( ast_list_t * stmt = stmts ; stmt ; stmt = stmt - > next ) {
2024-02-24 11:29:40 -08:00
bind_statement ( env , stmt - > ast ) ;
2024-03-14 10:28:30 -07:00
if ( stmt - > next ) {
code = CORD_all ( code , compile_statement ( env , stmt - > ast ) , " \n " ) ;
} else {
2024-07-04 15:00:01 -07:00
// TODO: put defer after evaluating block expression
for ( deferral_t * deferred = env - > deferred ; deferred & & deferred ! = prev_deferred ; deferred = deferred - > next ) {
code = CORD_all ( code , compile_statement ( deferred - > defer_env , deferred - > block ) ) ;
}
2024-03-14 10:28:30 -07:00
code = CORD_all ( code , compile ( env , stmt - > ast ) , " ; \n " ) ;
}
2024-02-25 10:13:36 -08:00
}
2024-07-04 15:00:01 -07:00
2024-03-14 10:28:30 -07:00
return CORD_cat ( code , " }) " ) ;
2024-02-04 12:23:59 -08:00
}
2024-03-05 11:46:01 -08:00
case Min : case Max : {
type_t * t = get_type ( env , ast ) ;
ast_t * key = ast - > tag = = Min ? Match ( ast , Min ) - > key : Match ( ast , Max ) - > key ;
ast_t * lhs = ast - > tag = = Min ? Match ( ast , Min ) - > lhs : Match ( ast , Max ) - > lhs ;
ast_t * rhs = ast - > tag = = Min ? Match ( ast , Min ) - > rhs : Match ( ast , Max ) - > rhs ;
2024-03-07 09:15:38 -08:00
const char * key_name = " $ " ;
2024-03-05 11:46:01 -08:00
if ( key = = NULL ) key = FakeAST ( Var , key_name ) ;
env_t * expr_env = fresh_scope ( env ) ;
2024-04-16 10:50:07 -07:00
set_binding ( expr_env , key_name , new ( binding_t , . type = t , . code = " ternary$lhs " ) ) ;
2024-03-05 11:46:01 -08:00
CORD lhs_key = compile ( expr_env , key ) ;
2024-04-16 10:50:07 -07:00
set_binding ( expr_env , key_name , new ( binding_t , . type = t , . code = " ternary$rhs " ) ) ;
2024-03-05 11:46:01 -08:00
CORD rhs_key = compile ( expr_env , key ) ;
type_t * key_t = get_type ( expr_env , key ) ;
CORD comparison ;
2024-08-18 08:49:51 -07:00
if ( key_t - > tag = = BigIntType )
2024-08-13 00:05:56 -07:00
comparison = CORD_all ( " (Int$compare_value( " , lhs_key , " , " , rhs_key , " ) " , ( ast - > tag = = Min ? " <= " : " >= " ) , " 0) " ) ;
2024-09-15 12:33:47 -07:00
else if ( key_t - > tag = = IntType | | key_t - > tag = = NumType | | key_t - > tag = = BoolType | | key_t - > tag = = PointerType | | key_t - > tag = = ByteType )
2024-03-05 11:46:01 -08:00
comparison = CORD_all ( " (( " , lhs_key , " ) " , ( ast - > tag = = Min ? " <= " : " >= " ) , " ( " , rhs_key , " )) " ) ;
else
2024-05-12 17:12:00 -07:00
comparison = CORD_all ( " generic_compare(stack( " , lhs_key , " ), stack( " , rhs_key , " ), " , compile_type_info ( env , key_t ) , " ) " ,
2024-03-05 11:46:01 -08:00
( ast - > tag = = Min ? " <= " : " >= " ) , " 0 " ) ;
return CORD_all (
" ({ \n " ,
2024-06-16 15:09:54 -07:00
compile_type ( t ) , " ternary$lhs = " , compile ( env , lhs ) , " , ternary$rhs = " , compile ( env , rhs ) , " ; \n " ,
2024-04-16 10:50:07 -07:00
comparison , " ? ternary$lhs : ternary$rhs; \n "
2024-03-05 11:46:01 -08:00
" }) " ) ;
2024-02-10 14:36:04 -08:00
}
2024-02-04 18:13:50 -08:00
case Array : {
2024-08-04 11:22:58 -07:00
type_t * array_type = get_type ( env , ast ) ;
2024-09-12 10:11:03 -07:00
type_t * item_type = Match ( array_type , ArrayType ) - > item_type ;
2024-10-08 18:10:36 -07:00
if ( type_size ( Match ( array_type , ArrayType ) - > item_type ) > ARRAY_MAX_STRIDE )
2024-08-04 11:22:58 -07:00
code_err ( ast , " This array holds items that take up %ld bytes, but the maximum supported size is %ld bytes. Consider using an array of pointers instead. " ,
2024-10-08 18:10:36 -07:00
type_size ( item_type ) , ARRAY_MAX_STRIDE ) ;
2024-08-04 11:22:58 -07:00
2024-02-04 18:13:50 -08:00
auto array = Match ( ast , Array ) ;
if ( ! array - > items )
2024-09-05 11:56:37 -07:00
return " (Array_t){.length=0} " ;
2024-03-05 23:27:01 -08:00
int64_t n = 0 ;
2024-03-13 23:37:56 -07:00
for ( ast_list_t * item = array - > items ; item ; item = item - > next ) {
2024-03-05 23:27:01 -08:00
+ + n ;
2024-03-17 11:46:36 -07:00
if ( item - > ast - > tag = = Comprehension )
2024-03-13 23:37:56 -07:00
goto array_comprehension ;
}
2024-03-05 23:27:01 -08:00
2024-03-13 23:48:07 -07:00
{
2024-09-12 10:11:03 -07:00
env_t * scope = item_type - > tag = = EnumType ? with_enum_scope ( env , item_type ) : env ;
2024-06-16 15:09:54 -07:00
CORD code = CORD_all ( " TypedArrayN( " , compile_type ( item_type ) , CORD_asprintf ( " , %ld " , n ) ) ;
2024-09-12 10:05:08 -07:00
for ( ast_list_t * item = array - > items ; item ; item = item - > next ) {
2024-09-12 10:11:03 -07:00
code = CORD_all ( code , " , " , compile_to_type ( scope , item - > ast , item_type ) ) ;
2024-09-12 10:05:08 -07:00
}
2024-03-13 23:48:07 -07:00
return CORD_cat ( code , " ) " ) ;
}
2024-03-13 23:37:56 -07:00
array_comprehension :
{
2024-09-12 10:11:03 -07:00
env_t * scope = item_type - > tag = = EnumType ? with_enum_scope ( env , item_type ) : fresh_scope ( env ) ;
2024-03-17 12:26:25 -07:00
static int64_t comp_num = 1 ;
2024-09-12 11:27:13 -07:00
const char * comprehension_name = heap_strf ( " arr$%ld " , comp_num + + ) ;
ast_t * comprehension_var = FakeAST ( InlineCCode , . code = CORD_all ( " & " , comprehension_name ) ,
2024-10-27 16:54:48 -07:00
. type = Type ( PointerType , . pointed = array_type , . is_view = true ) ) ;
2024-09-12 11:27:13 -07:00
Closure_t comp_action = { . fn = add_to_array_comprehension , . userdata = comprehension_var } ;
scope - > comprehension_action = & comp_action ;
CORD code = CORD_all ( " ({ Array_t " , comprehension_name , " = {}; " ) ;
// set_binding(scope, comprehension_name, new(binding_t, .type=array_type, .code=comprehension_name));
2024-03-13 23:37:56 -07:00
for ( ast_list_t * item = array - > items ; item ; item = item - > next ) {
2024-09-12 11:27:13 -07:00
if ( item - > ast - > tag = = Comprehension )
2024-03-17 12:26:25 -07:00
code = CORD_all ( code , " \n " , compile_statement ( scope , item - > ast ) ) ;
2024-09-12 11:27:13 -07:00
else
code = CORD_all ( code , compile_statement ( env , add_to_array_comprehension ( item - > ast , comprehension_var ) ) ) ;
2024-03-13 23:37:56 -07:00
}
2024-09-12 11:27:13 -07:00
code = CORD_all ( code , " " , comprehension_name , " ; }) " ) ;
2024-03-13 23:37:56 -07:00
return code ;
}
2024-02-04 18:13:50 -08:00
}
2024-08-11 11:47:34 -07:00
case Channel : {
2024-08-11 12:04:22 -07:00
auto chan = Match ( ast , Channel ) ;
type_t * item_t = parse_type_ast ( env , chan - > item_type ) ;
2024-08-11 11:47:34 -07:00
if ( ! can_send_over_channel ( item_t ) )
code_err ( ast , " This item type can't be sent over a channel because it contains reference to memory that may not be thread-safe. " ) ;
2024-08-11 12:04:22 -07:00
if ( chan - > max_size ) {
CORD max_size = compile ( env , chan - > max_size ) ;
if ( ! promote ( env , & max_size , get_type ( env , chan - > max_size ) , INT_TYPE ) )
code_err ( chan - > max_size , " This value must be an integer, not %T " , get_type ( env , chan - > max_size ) ) ;
return CORD_all ( " Channel$new( " , max_size , " ) " ) ;
} else {
2024-08-13 00:27:32 -07:00
return " Channel$new(I(INT32_MAX)) " ;
2024-08-11 12:04:22 -07:00
}
2024-08-11 11:47:34 -07:00
}
2024-02-17 21:12:23 -08:00
case Table : {
auto table = Match ( ast , Table ) ;
if ( ! table - > entries ) {
2024-09-05 11:57:31 -07:00
CORD code = " ((Table_t){ " ;
2024-02-17 21:12:23 -08:00
if ( table - > fallback )
2024-09-10 22:31:31 -07:00
code = CORD_all ( code , " .fallback=heap( " , compile ( env , table - > fallback ) , " ) " ) ;
2024-08-03 13:23:28 -07:00
return CORD_cat ( code , " }) " ) ;
2024-02-17 21:12:23 -08:00
}
2024-03-17 11:46:36 -07:00
2024-03-13 23:48:07 -07:00
type_t * table_type = get_type ( env , ast ) ;
type_t * key_t = Match ( table_type , TableType ) - > key_type ;
type_t * value_t = Match ( table_type , TableType ) - > value_type ;
2024-02-18 01:26:26 -08:00
2024-03-17 11:46:36 -07:00
for ( ast_list_t * entry = table - > entries ; entry ; entry = entry - > next ) {
if ( entry - > ast - > tag = = Comprehension )
goto table_comprehension ;
}
{ // No comprehension:
2024-09-12 10:11:03 -07:00
env_t * key_scope = key_t - > tag = = EnumType ? with_enum_scope ( env , key_t ) : env ;
env_t * value_scope = value_t - > tag = = EnumType ? with_enum_scope ( env , value_t ) : env ;
2024-04-16 10:50:07 -07:00
CORD code = CORD_all ( " Table( " ,
2024-06-16 15:09:54 -07:00
compile_type ( key_t ) , " , " ,
compile_type ( value_t ) , " , " ,
2024-03-17 11:46:36 -07:00
compile_type_info ( env , key_t ) , " , " ,
compile_type_info ( env , value_t ) ) ;
if ( table - > fallback )
2024-04-16 10:50:07 -07:00
code = CORD_all ( code , " , /*fallback:*/ heap( " , compile ( env , table - > fallback ) , " ) " ) ;
2024-03-17 11:46:36 -07:00
else
code = CORD_all ( code , " , /*fallback:*/ NULL " ) ;
2024-02-18 01:26:26 -08:00
2024-03-17 11:46:36 -07:00
size_t n = 0 ;
for ( ast_list_t * entry = table - > entries ; entry ; entry = entry - > next )
+ + n ;
CORD_appendf ( & code , " , %zu " , n ) ;
for ( ast_list_t * entry = table - > entries ; entry ; entry = entry - > next ) {
auto e = Match ( entry - > ast , TableEntry ) ;
2024-09-12 10:11:03 -07:00
code = CORD_all ( code , " , \n \t { " , compile_to_type ( key_scope , e - > key , key_t ) , " , " ,
compile_to_type ( value_scope , e - > value , value_t ) , " } " ) ;
2024-03-17 11:46:36 -07:00
}
return CORD_cat ( code , " ) " ) ;
}
table_comprehension :
{
2024-03-17 12:26:25 -07:00
static int64_t comp_num = 1 ;
env_t * scope = fresh_scope ( env ) ;
2024-09-12 11:27:13 -07:00
const char * comprehension_name = heap_strf ( " table$%ld " , comp_num + + ) ;
ast_t * comprehension_var = FakeAST ( InlineCCode , . code = CORD_all ( " & " , comprehension_name ) ,
2024-10-27 16:54:48 -07:00
. type = Type ( PointerType , . pointed = table_type , . is_view = true ) ) ;
2024-03-17 12:26:25 -07:00
2024-09-12 11:27:13 -07:00
CORD code = CORD_all ( " ({ Table_t " , comprehension_name , " = { " ) ;
2024-03-17 11:46:36 -07:00
if ( table - > fallback )
2024-04-16 10:50:07 -07:00
code = CORD_all ( code , " .fallback=heap( " , compile ( env , table - > fallback ) , " ), " ) ;
2024-03-17 11:46:36 -07:00
code = CORD_cat ( code , " }; " ) ;
2024-09-12 11:27:13 -07:00
Closure_t comp_action = { . fn = add_to_table_comprehension , . userdata = comprehension_var } ;
scope - > comprehension_action = & comp_action ;
2024-03-17 11:46:36 -07:00
for ( ast_list_t * entry = table - > entries ; entry ; entry = entry - > next ) {
2024-09-12 11:27:13 -07:00
if ( entry - > ast - > tag = = Comprehension )
2024-03-17 12:26:25 -07:00
code = CORD_all ( code , " \n " , compile_statement ( scope , entry - > ast ) ) ;
2024-09-12 11:27:13 -07:00
else
code = CORD_all ( code , compile_statement ( env , add_to_table_comprehension ( entry - > ast , comprehension_var ) ) ) ;
2024-03-17 11:46:36 -07:00
}
2024-09-12 11:27:13 -07:00
code = CORD_all ( code , " " , comprehension_name , " ; }) " ) ;
2024-03-17 11:46:36 -07:00
return code ;
2024-02-17 21:12:23 -08:00
}
2024-08-10 12:15:38 -07:00
}
case Set : {
auto set = Match ( ast , Set ) ;
if ( ! set - > items )
2024-09-05 11:57:31 -07:00
return " ((Table_t){}) " ;
2024-08-10 12:15:38 -07:00
type_t * set_type = get_type ( env , ast ) ;
type_t * item_type = Match ( set_type , SetType ) - > item_type ;
2024-09-12 11:27:13 -07:00
size_t n = 0 ;
2024-08-10 12:15:38 -07:00
for ( ast_list_t * item = set - > items ; item ; item = item - > next ) {
2024-09-12 11:27:13 -07:00
+ + n ;
2024-08-10 12:15:38 -07:00
if ( item - > ast - > tag = = Comprehension )
goto set_comprehension ;
}
{ // No comprehension:
CORD code = CORD_all ( " Set( " ,
compile_type ( item_type ) , " , " ,
compile_type_info ( env , item_type ) ) ;
CORD_appendf ( & code , " , %zu " , n ) ;
2024-09-12 10:11:03 -07:00
env_t * scope = item_type - > tag = = EnumType ? with_enum_scope ( env , item_type ) : env ;
2024-08-10 12:15:38 -07:00
for ( ast_list_t * item = set - > items ; item ; item = item - > next ) {
2024-09-12 10:11:03 -07:00
code = CORD_all ( code , " , " , compile_to_type ( scope , item - > ast , item_type ) ) ;
2024-08-10 12:15:38 -07:00
}
return CORD_cat ( code , " ) " ) ;
}
set_comprehension :
{
static int64_t comp_num = 1 ;
2024-09-12 10:11:03 -07:00
env_t * scope = item_type - > tag = = EnumType ? with_enum_scope ( env , item_type ) : fresh_scope ( env ) ;
2024-09-12 11:27:13 -07:00
const char * comprehension_name = heap_strf ( " set$%ld " , comp_num + + ) ;
ast_t * comprehension_var = FakeAST ( InlineCCode , . code = CORD_all ( " & " , comprehension_name ) ,
2024-10-27 16:54:48 -07:00
. type = Type ( PointerType , . pointed = set_type , . is_view = true ) ) ;
2024-09-12 11:27:13 -07:00
CORD code = CORD_all ( " ({ Table_t " , comprehension_name , " = {}; " ) ;
Closure_t comp_action = { . fn = add_to_set_comprehension , . userdata = comprehension_var } ;
scope - > comprehension_action = & comp_action ;
2024-08-10 12:15:38 -07:00
for ( ast_list_t * item = set - > items ; item ; item = item - > next ) {
2024-09-12 11:27:13 -07:00
if ( item - > ast - > tag = = Comprehension )
2024-08-10 12:15:38 -07:00
code = CORD_all ( code , " \n " , compile_statement ( scope , item - > ast ) ) ;
2024-09-12 11:27:13 -07:00
else
code = CORD_all ( code , compile_statement ( env , add_to_set_comprehension ( item - > ast , comprehension_var ) ) ) ;
2024-08-10 12:15:38 -07:00
}
2024-09-12 11:27:13 -07:00
code = CORD_all ( code , " " , comprehension_name , " ; }) " ) ;
2024-08-10 12:15:38 -07:00
return code ;
}
2024-02-17 21:12:23 -08:00
}
2024-03-17 11:46:36 -07:00
case Comprehension : {
2024-03-17 12:59:06 -07:00
ast_t * base = Match ( ast , Comprehension ) - > expr ;
while ( base - > tag = = Comprehension )
base = Match ( ast , Comprehension ) - > expr ;
if ( base - > tag = = TableEntry )
return compile ( env , WrapAST ( ast , Table , . entries = new ( ast_list_t , . ast = ast ) ) ) ;
else
return compile ( env , WrapAST ( ast , Array , . items = new ( ast_list_t , . ast = ast ) ) ) ;
2024-03-17 11:46:36 -07:00
}
2024-03-09 11:02:19 -08:00
case Lambda : {
auto lambda = Match ( ast , Lambda ) ;
2024-09-24 11:54:22 -07:00
CORD name = CORD_asprintf ( " %rlambda$%ld " , namespace_prefix ( env , env - > namespace ) , lambda - > id ) ;
2024-03-09 11:02:19 -08:00
2024-09-12 20:41:32 -07:00
env - > code - > function_naming = CORD_all (
env - > code - > function_naming ,
CORD_asprintf ( " register_function(%r, Text( \" %r [%s.tm:%ld] \" )); \n " ,
name , type_to_cord ( get_type ( env , ast ) ) , file_base_name ( ast - > file - > filename ) , get_line_number ( ast - > file , ast - > start ) ) ) ;
2024-03-09 11:02:19 -08:00
env_t * body_scope = fresh_scope ( env ) ;
for ( arg_ast_t * arg = lambda - > args ; arg ; arg = arg - > next ) {
type_t * arg_type = get_arg_ast_type ( env , arg ) ;
2024-04-16 10:50:07 -07:00
set_binding ( body_scope , arg - > name , new ( binding_t , . type = arg_type , . code = CORD_cat ( " $ " , arg - > name ) ) ) ;
2024-03-09 11:02:19 -08:00
}
2024-03-09 13:03:38 -08:00
2024-03-15 10:35:30 -07:00
fn_ctx_t fn_ctx = ( fn_ctx_t ) {
2024-07-14 11:13:23 -07:00
. parent = env - > fn_ctx ,
. closure_scope = env - > locals ,
2024-09-05 11:57:31 -07:00
. closed_vars = new ( Table_t ) ,
2024-03-09 13:03:38 -08:00
} ;
body_scope - > fn_ctx = & fn_ctx ;
body_scope - > locals - > fallback = env - > globals ;
2024-07-04 15:09:33 -07:00
body_scope - > deferred = NULL ;
2024-07-13 15:43:50 -07:00
type_t * ret_t = get_type ( body_scope , lambda - > body ) ;
if ( ret_t - > tag = = ReturnType )
ret_t = Match ( ret_t , ReturnType ) - > ret ;
2024-10-09 10:26:28 -07:00
if ( lambda - > ret_type ) {
type_t * declared = parse_type_ast ( env , lambda - > ret_type ) ;
if ( can_promote ( ret_t , declared ) )
ret_t = declared ;
else
code_err ( ast , " This function was declared to return a value of type %T, but actually returns a value of type %T " ,
declared , ret_t ) ;
}
2024-06-06 13:28:53 -07:00
fn_ctx . return_type = ret_t ;
2024-03-09 13:03:38 -08:00
2024-07-14 11:13:23 -07:00
if ( env - > fn_ctx - > closed_vars ) {
for ( int64_t i = 1 ; i < = Table $ length ( * env - > fn_ctx - > closed_vars ) ; i + + ) {
struct { const char * name ; binding_t * b ; } * entry = Table $ entry ( * env - > fn_ctx - > closed_vars , i ) ;
set_binding ( body_scope , entry - > name , new ( binding_t , . type = entry - > b - > type , . code = CORD_cat ( " userdata-> " , entry - > name ) ) ) ;
Table $ str_set ( fn_ctx . closed_vars , entry - > name , entry - > b ) ;
}
}
2024-09-17 13:20:30 -07:00
Table_t * closed_vars = get_closed_vars ( env , ast ) ;
if ( Table $ length ( * closed_vars ) > 0 ) { // Create a typedef for the lambda's closure userdata
CORD def = " typedef struct { " ;
for ( int64_t i = 1 ; i < = Table $ length ( * closed_vars ) ; i + + ) {
struct { const char * name ; binding_t * b ; } * entry = Table $ entry ( * closed_vars , i ) ;
if ( entry - > b - > type - > tag = = ModuleType )
continue ;
2024-10-08 21:36:00 -07:00
set_binding ( body_scope , entry - > name , new ( binding_t , . type = entry - > b - > type , . code = CORD_cat ( " userdata-> " , entry - > name ) ) ) ;
2024-09-17 13:20:30 -07:00
def = CORD_all ( def , compile_declaration ( entry - > b - > type , entry - > name ) , " ; " ) ;
}
def = CORD_all ( def , " } " , name , " $userdata_t; " ) ;
env - > code - > local_typedefs = CORD_all ( env - > code - > local_typedefs , def ) ;
}
2024-06-16 15:09:54 -07:00
CORD code = CORD_all ( " static " , compile_type ( ret_t ) , " " , name , " ( " ) ;
2024-03-09 11:02:19 -08:00
for ( arg_ast_t * arg = lambda - > args ; arg ; arg = arg - > next ) {
type_t * arg_type = get_arg_ast_type ( env , arg ) ;
2024-06-16 15:09:54 -07:00
code = CORD_all ( code , compile_type ( arg_type ) , " $ " , arg - > name , " , " ) ;
2024-03-09 11:02:19 -08:00
}
2024-03-09 13:03:38 -08:00
CORD userdata ;
2024-06-06 13:28:53 -07:00
if ( Table $ length ( * closed_vars ) = = 0 ) {
2024-04-16 10:50:07 -07:00
code = CORD_cat ( code , " void *userdata) " ) ;
2024-03-09 13:03:38 -08:00
userdata = " NULL " ;
} else {
userdata = CORD_all ( " new( " , name , " $userdata_t " ) ;
2024-06-06 13:28:53 -07:00
for ( int64_t i = 1 ; i < = Table $ length ( * closed_vars ) ; i + + ) {
struct { const char * name ; binding_t * b ; } * entry = Table $ entry ( * closed_vars , i ) ;
2024-05-27 13:33:51 -07:00
if ( entry - > b - > type - > tag = = ModuleType )
continue ;
2024-08-12 14:44:05 -07:00
CORD binding_code = get_binding ( env , entry - > name ) - > code ;
if ( entry - > b - > type - > tag = = ArrayType )
userdata = CORD_all ( userdata , " , ARRAY_COPY( " , binding_code , " ) " ) ;
else if ( entry - > b - > type - > tag = = TableType | | entry - > b - > type - > tag = = SetType )
userdata = CORD_all ( userdata , " , TABLE_COPY( " , binding_code , " ) " ) ;
else
userdata = CORD_all ( userdata , " , " , binding_code ) ;
2024-03-09 13:03:38 -08:00
}
userdata = CORD_all ( userdata , " ) " ) ;
2024-04-16 10:50:07 -07:00
code = CORD_all ( code , name , " $userdata_t *userdata) " ) ;
2024-03-09 13:03:38 -08:00
}
2024-03-09 11:02:19 -08:00
2024-03-09 11:09:18 -08:00
CORD body = CORD_EMPTY ;
for ( ast_list_t * stmt = Match ( lambda - > body , Block ) - > statements ; stmt ; stmt = stmt - > next ) {
2024-07-14 10:22:42 -07:00
bind_statement ( body_scope , stmt - > ast ) ;
2024-07-13 15:43:50 -07:00
if ( stmt - > next | | ret_t - > tag = = VoidType | | ret_t - > tag = = AbortType | | get_type ( body_scope , stmt - > ast ) - > tag = = ReturnType )
2024-03-09 11:09:18 -08:00
body = CORD_all ( body , compile_statement ( body_scope , stmt - > ast ) , " \n " ) ;
else
body = CORD_all ( body , compile_statement ( body_scope , FakeAST ( Return , stmt - > ast ) ) , " \n " ) ;
}
2024-07-04 15:09:33 -07:00
if ( ( ret_t - > tag = = VoidType | | ret_t - > tag = = AbortType ) & & body_scope - > deferred )
body = CORD_all ( body , compile_statement ( body_scope , FakeAST ( Return ) ) , " \n " ) ;
env - > code - > funcs = CORD_all ( env - > code - > funcs , code , " { \n " , body , " \n } \n " ) ;
2024-09-11 10:56:39 -07:00
return CORD_all ( " ((Closure_t) { " , name, " , " , userdata, " } ) " );
2024-03-09 11:02:19 -08:00
}
2024-03-08 11:23:16 -08:00
case MethodCall : {
auto call = Match ( ast , MethodCall ) ;
type_t * self_t = get_type ( env , call - > self ) ;
2024-11-09 14:26:01 -08:00
int64_t pointer_depth = 0 ;
type_t * self_value_t = self_t ;
for ( ; self_value_t - > tag = = PointerType ; self_value_t = Match ( self_value_t , PointerType ) - > pointed )
pointer_depth + = 1 ;
CORD self = compile ( env , call - > self ) ;
# define EXPECT_POINTER(article, name) do { \
if ( pointer_depth < 1 ) code_err ( call - > self , " I expected " article " " name " @pointer here, not " article " " name " value " ) ; \
else if ( pointer_depth > 1 ) code_err ( call - > self , " I expected " article " " name " @pointer here, not a nested " name " pointer " ) ; \
} while ( 0 )
2024-03-08 11:23:16 -08:00
switch ( self_value_t - > tag ) {
case ArrayType : {
2024-04-02 10:08:06 -07:00
type_t * item_t = Match ( self_value_t , ArrayType ) - > item_type ;
2024-10-08 18:10:36 -07:00
CORD padded_item_size = CORD_asprintf ( " %ld " , type_size ( item_t ) ) ;
2024-11-03 13:10:03 -08:00
2024-11-03 19:37:48 -08:00
ast_t * default_rng = FakeAST ( InlineCCode , . code = " default_rng " , . type = RNG_TYPE ) ;
2024-03-08 11:23:16 -08:00
if ( streq ( call - > name , " insert " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " an " , " array " ) ;
2024-03-13 23:37:56 -07:00
arg_t * arg_spec = new ( arg_t , . name = " item " , . type = item_t ,
2024-08-18 08:49:51 -07:00
. next = new ( arg_t , . name = " at " , . type = INT_TYPE , . default_val = FakeAST ( Int , . str = " 0 " ) ) ) ;
2024-03-29 09:54:31 -07:00
return CORD_all ( " Array$insert_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
2024-08-03 12:33:50 -07:00
padded_item_size , " ) " ) ;
2024-03-08 11:25:17 -08:00
} else if ( streq ( call - > name , " insert_all " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " an " , " array " ) ;
2024-03-08 11:25:17 -08:00
arg_t * arg_spec = new ( arg_t , . name = " items " , . type = self_value_t ,
2024-08-18 08:49:51 -07:00
. next = new ( arg_t , . name = " at " , . type = INT_TYPE , . default_val = FakeAST ( Int , . str = " 0 " ) ) ) ;
2024-03-29 09:54:31 -07:00
return CORD_all ( " Array$insert_all( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
2024-08-03 12:33:50 -07:00
padded_item_size , " ) " ) ;
2024-08-14 23:39:35 -07:00
} else if ( streq ( call - > name , " remove_at " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " an " , " array " ) ;
2024-08-18 08:49:51 -07:00
arg_t * arg_spec = new ( arg_t , . name = " index " , . type = INT_TYPE , . default_val = FakeAST ( Int , . str = " -1 " ) ,
. next = new ( arg_t , . name = " count " , . type = INT_TYPE , . default_val = FakeAST ( Int , . str = " 1 " ) ) ) ;
2024-08-14 23:39:35 -07:00
return CORD_all ( " Array$remove_at( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
2024-08-03 12:33:50 -07:00
padded_item_size , " ) " ) ;
2024-08-14 23:39:35 -07:00
} else if ( streq ( call - > name , " remove_item " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " an " , " array " ) ;
2024-08-14 23:39:35 -07:00
arg_t * arg_spec = new ( arg_t , . name = " item " , . type = item_t ,
2024-08-18 08:49:51 -07:00
. next = new ( arg_t , . name = " max_count " , . type = INT_TYPE , . default_val = FakeAST ( Int , . str = " -1 " ) ) ) ;
2024-08-14 23:39:35 -07:00
return CORD_all ( " Array$remove_item_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
compile_type_info ( env , self_value_t ) , " ) " ) ;
2024-03-08 11:23:16 -08:00
} else if ( streq ( call - > name , " random " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-11-03 19:37:48 -08:00
arg_t * arg_spec = new ( arg_t , . name = " rng " , . type = RNG_TYPE , . default_val = default_rng ) ;
2024-11-03 13:10:03 -08:00
return CORD_all ( " Array$random_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " , compile_type ( item_t ) , " ) " ) ;
2024-08-14 23:39:35 -07:00
} else if ( streq ( call - > name , " has " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-14 23:39:35 -07:00
arg_t * arg_spec = new ( arg_t , . name = " item " , . type = item_t ) ;
return CORD_all ( " Array$has_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
compile_type_info ( env , self_value_t ) , " ) " ) ;
2024-04-02 20:28:59 -07:00
} else if ( streq ( call - > name , " sample " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-12 22:30:25 -07:00
arg_t * arg_spec = new ( arg_t , . name = " count " , . type = INT_TYPE ,
2024-10-02 11:42:51 -07:00
. next = new ( arg_t , . name = " weights " , . type = Type ( ArrayType , . item_type = Type ( NumType ) ) ,
. default_val = FakeAST ( Null , . type = new ( type_ast_t , . tag = ArrayTypeAST ,
2024-11-03 19:37:48 -08:00
. __data . ArrayTypeAST . item = new ( type_ast_t , . tag = VarTypeAST , . __data . VarTypeAST . name = " Num " ) ) ) ,
. next = new ( arg_t , . name = " rng " , . type = RNG_TYPE , . default_val = default_rng ) ) ) ;
2024-04-02 20:28:59 -07:00
return CORD_all ( " Array$sample( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
2024-08-03 12:33:50 -07:00
padded_item_size , " ) " ) ;
2024-03-08 11:23:16 -08:00
} else if ( streq ( call - > name , " shuffle " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " an " , " array " ) ;
2024-11-03 19:37:48 -08:00
arg_t * arg_spec = new ( arg_t , . name = " rng " , . type = RNG_TYPE , . default_val = default_rng ) ;
2024-11-03 13:10:03 -08:00
return CORD_all ( " Array$shuffle( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " , padded_item_size , " ) " ) ;
2024-08-14 23:17:53 -07:00
} else if ( streq ( call - > name , " shuffled " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-11-03 19:37:48 -08:00
arg_t * arg_spec = new ( arg_t , . name = " rng " , . type = RNG_TYPE , . default_val = default_rng ) ;
2024-11-03 13:10:03 -08:00
return CORD_all ( " Array$shuffled( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " , padded_item_size , " ) " ) ;
2024-04-02 10:13:33 -07:00
} else if ( streq ( call - > name , " sort " ) | | streq ( call - > name , " sorted " ) ) {
2024-11-09 14:26:01 -08:00
if ( streq ( call - > name , " sort " ) )
EXPECT_POINTER ( " an " , " array " ) ;
else
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-04-02 10:08:06 -07:00
CORD comparison ;
if ( call - > args ) {
2024-10-27 16:54:48 -07:00
type_t * item_ptr = Type ( PointerType , . pointed = item_t , . is_view = true ) ;
2024-04-02 10:08:06 -07:00
type_t * fn_t = Type ( FunctionType , . args = new ( arg_t , . name = " x " , . type = item_ptr , . next = new ( arg_t , . name = " y " , . type = item_ptr ) ) ,
2024-08-18 08:49:51 -07:00
. ret = Type ( IntType , . bits = TYPE_IBITS32 ) ) ;
2024-04-02 10:08:06 -07:00
arg_t * arg_spec = new ( arg_t , . name = " by " , . type = Type ( ClosureType , . fn = fn_t ) ) ;
comparison = compile_arguments ( env , ast , arg_spec , call - > args ) ;
} else {
2024-09-11 10:56:39 -07:00
comparison = CORD_all ( " ((Closure_t){.fn=generic_compare, .userdata=(void*) " , compile_type_info ( env , item_t ) , " }) " ) ;
2024-04-02 10:08:06 -07:00
}
2024-08-03 12:33:50 -07:00
return CORD_all ( " Array$ " , call - > name , " ( " , self , " , " , comparison , " , " , padded_item_size , " ) " ) ;
2024-04-19 10:29:04 -07:00
} else if ( streq ( call - > name , " heapify " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " an " , " array " ) ;
2024-04-19 10:29:04 -07:00
CORD comparison ;
if ( call - > args ) {
2024-10-27 16:54:48 -07:00
type_t * item_ptr = Type ( PointerType , . pointed = item_t , . is_view = true ) ;
2024-04-19 10:29:04 -07:00
type_t * fn_t = Type ( FunctionType , . args = new ( arg_t , . name = " x " , . type = item_ptr , . next = new ( arg_t , . name = " y " , . type = item_ptr ) ) ,
2024-08-18 08:49:51 -07:00
. ret = Type ( IntType , . bits = TYPE_IBITS32 ) ) ;
2024-04-19 10:29:04 -07:00
arg_t * arg_spec = new ( arg_t , . name = " by " , . type = Type ( ClosureType , . fn = fn_t ) ) ;
comparison = compile_arguments ( env , ast , arg_spec , call - > args ) ;
} else {
2024-09-11 10:56:39 -07:00
comparison = CORD_all ( " ((Closure_t){.fn=generic_compare, .userdata=(void*) " , compile_type_info ( env , item_t ) , " }) " ) ;
2024-04-19 10:29:04 -07:00
}
2024-08-03 12:33:50 -07:00
return CORD_all ( " Array$heapify( " , self , " , " , comparison , " , " , padded_item_size , " ) " ) ;
2024-04-19 10:29:04 -07:00
} else if ( streq ( call - > name , " heap_push " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " an " , " array " ) ;
2024-10-27 16:54:48 -07:00
type_t * item_ptr = Type ( PointerType , . pointed = item_t , . is_view = true ) ;
2024-04-19 10:29:04 -07:00
type_t * fn_t = Type ( FunctionType , . args = new ( arg_t , . name = " x " , . type = item_ptr , . next = new ( arg_t , . name = " y " , . type = item_ptr ) ) ,
2024-08-18 08:49:51 -07:00
. ret = Type ( IntType , . bits = TYPE_IBITS32 ) ) ;
2024-08-15 10:59:01 -07:00
ast_t * default_cmp = FakeAST ( InlineCCode ,
2024-09-11 10:56:39 -07:00
. code = CORD_all ( " ((Closure_t){.fn=generic_compare, .userdata=(void*) " ,
2024-08-15 10:59:01 -07:00
compile_type_info ( env , item_t ) , " }) " ) ,
2024-09-12 11:27:13 -07:00
. type = Type ( ClosureType , . fn = fn_t ) ) ;
2024-08-15 10:59:01 -07:00
arg_t * arg_spec = new ( arg_t , . name = " item " , . type = item_t ,
. next = new ( arg_t , . name = " by " , . type = Type ( ClosureType , . fn = fn_t ) , . default_val = default_cmp ) ) ;
2024-04-19 10:29:04 -07:00
CORD arg_code = compile_arguments ( env , ast , arg_spec , call - > args ) ;
2024-08-03 12:33:50 -07:00
return CORD_all ( " Array$heap_push_value( " , self , " , " , arg_code , " , " , padded_item_size , " ) " ) ;
2024-04-19 10:29:04 -07:00
} else if ( streq ( call - > name , " heap_pop " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " an " , " array " ) ;
2024-10-27 16:54:48 -07:00
type_t * item_ptr = Type ( PointerType , . pointed = item_t , . is_view = true ) ;
2024-04-19 10:29:04 -07:00
type_t * fn_t = Type ( FunctionType , . args = new ( arg_t , . name = " x " , . type = item_ptr , . next = new ( arg_t , . name = " y " , . type = item_ptr ) ) ,
2024-08-18 08:49:51 -07:00
. ret = Type ( IntType , . bits = TYPE_IBITS32 ) ) ;
2024-08-15 10:59:01 -07:00
ast_t * default_cmp = FakeAST ( InlineCCode ,
2024-09-11 10:56:39 -07:00
. code = CORD_all ( " ((Closure_t){.fn=generic_compare, .userdata=(void*) " ,
2024-08-15 10:59:01 -07:00
compile_type_info ( env , item_t ) , " }) " ) ,
2024-09-12 11:27:13 -07:00
. type = Type ( ClosureType , . fn = fn_t ) ) ;
2024-04-19 10:29:04 -07:00
arg_t * arg_spec = new ( arg_t , . name = " by " , . type = Type ( ClosureType , . fn = fn_t ) , . default_val = default_cmp ) ;
CORD arg_code = compile_arguments ( env , ast , arg_spec , call - > args ) ;
2024-08-03 12:33:50 -07:00
return CORD_all ( " Array$heap_pop_value( " , self , " , " , arg_code , " , " , padded_item_size , " , " , compile_type ( item_t ) , " ) " ) ;
2024-08-14 22:59:42 -07:00
} else if ( streq ( call - > name , " binary_search " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , call - > args ! = NULL ) ;
2024-10-27 16:54:48 -07:00
type_t * item_ptr = Type ( PointerType , . pointed = item_t , . is_view = true ) ;
2024-08-14 22:59:42 -07:00
type_t * fn_t = Type ( FunctionType , . args = new ( arg_t , . name = " x " , . type = item_ptr , . next = new ( arg_t , . name = " y " , . type = item_ptr ) ) ,
2024-08-18 08:49:51 -07:00
. ret = Type ( IntType , . bits = TYPE_IBITS32 ) ) ;
2024-08-15 10:59:01 -07:00
ast_t * default_cmp = FakeAST ( InlineCCode ,
2024-09-11 10:56:39 -07:00
. code = CORD_all ( " ((Closure_t){.fn=generic_compare, .userdata=(void*) " ,
2024-08-15 10:59:01 -07:00
compile_type_info ( env , item_t ) , " }) " ) ,
2024-09-12 11:27:13 -07:00
. type = Type ( ClosureType , . fn = fn_t ) ) ;
2024-08-15 10:59:01 -07:00
arg_t * arg_spec = new ( arg_t , . name = " target " , . type = item_t ,
. next = new ( arg_t , . name = " by " , . type = Type ( ClosureType , . fn = fn_t ) , . default_val = default_cmp ) ) ;
2024-08-14 22:59:42 -07:00
CORD arg_code = compile_arguments ( env , ast , arg_spec , call - > args ) ;
return CORD_all ( " Array$binary_search_value( " , self , " , " , arg_code , " ) " ) ;
2024-03-08 11:23:16 -08:00
} else if ( streq ( call - > name , " clear " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " an " , " array " ) ;
2024-04-02 10:08:06 -07:00
( void ) compile_arguments ( env , ast , NULL , call - > args ) ;
2024-03-29 09:54:31 -07:00
return CORD_all ( " Array$clear( " , self , " ) " ) ;
2024-08-15 00:01:04 -07:00
} else if ( streq ( call - > name , " find " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-15 00:01:04 -07:00
arg_t * arg_spec = new ( arg_t , . name = " item " , . type = item_t ) ;
return CORD_all ( " Array$find_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) ,
" , " , compile_type_info ( env , self_value_t ) , " ) " ) ;
2024-08-18 19:28:04 -07:00
} else if ( streq ( call - > name , " first " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , call - > args ! = NULL ) ;
2024-10-27 16:54:48 -07:00
type_t * item_ptr = Type ( PointerType , . pointed = item_t , . is_view = true ) ;
2024-08-18 19:28:04 -07:00
type_t * predicate_type = Type (
ClosureType , . fn = Type ( FunctionType , . args = new ( arg_t , . name = " item " , . type = item_ptr ) , . ret = Type ( BoolType ) ) ) ;
arg_t * arg_spec = new ( arg_t , . name = " predicate " , . type = predicate_type ) ;
return CORD_all ( " Array$first( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " ) " ) ;
2024-07-10 10:34:45 -07:00
} else if ( streq ( call - > name , " from " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , true ) ;
2024-08-12 22:30:25 -07:00
arg_t * arg_spec = new ( arg_t , . name = " first " , . type = INT_TYPE ) ;
2024-07-10 10:34:45 -07:00
return CORD_all ( " Array$from( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " ) " ) ;
2024-07-10 10:42:58 -07:00
} else if ( streq ( call - > name , " to " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , true ) ;
2024-08-12 22:30:25 -07:00
arg_t * arg_spec = new ( arg_t , . name = " last " , . type = INT_TYPE ) ;
2024-07-10 10:42:58 -07:00
return CORD_all ( " Array$to( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " ) " ) ;
2024-07-10 10:34:45 -07:00
} else if ( streq ( call - > name , " by " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , true ) ;
2024-08-12 22:30:25 -07:00
arg_t * arg_spec = new ( arg_t , . name = " stride " , . type = INT_TYPE ) ;
2024-08-03 12:33:50 -07:00
return CORD_all ( " Array$by( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " , padded_item_size , " ) " ) ;
2024-03-26 11:59:52 -07:00
} else if ( streq ( call - > name , " reversed " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , true ) ;
2024-04-02 10:08:06 -07:00
( void ) compile_arguments ( env , ast , NULL , call - > args ) ;
2024-08-03 12:33:50 -07:00
return CORD_all ( " Array$reversed( " , self , " , " , padded_item_size , " ) " ) ;
2024-08-10 13:19:36 -07:00
} else if ( streq ( call - > name , " unique " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-10 13:19:36 -07:00
( void ) compile_arguments ( env , ast , NULL , call - > args ) ;
2024-09-05 12:31:54 -07:00
return CORD_all ( " Table$from_entries( " , self , " , Set$info( " , compile_type_info ( env , item_t ) , " )) " ) ;
2024-08-10 13:36:50 -07:00
} else if ( streq ( call - > name , " counts " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-10 13:36:50 -07:00
( void ) compile_arguments ( env , ast , NULL , call - > args ) ;
return CORD_all ( " Array$counts( " , self , " , " , compile_type_info ( env , self_value_t ) , " ) " ) ;
2024-03-08 11:23:16 -08:00
} else code_err ( ast , " There is no '%s' method for arrays " , call - > name ) ;
}
2024-08-10 12:15:38 -07:00
case SetType : {
auto set = Match ( self_value_t , SetType ) ;
if ( streq ( call - > name , " has " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " key " , . type = set - > item_type ) ;
return CORD_all ( " Table$has_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
compile_type_info ( env , self_value_t ) , " ) " ) ;
} else if ( streq ( call - > name , " add " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " a " , " set " ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " item " , . type = set - > item_type ) ;
return CORD_all ( " Table$set_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , NULL, " ,
compile_type_info ( env , self_value_t ) , " ) " ) ;
} else if ( streq ( call - > name , " add_all " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " a " , " set " ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " items " , . type = Type ( ArrayType , . item_type = Match ( self_value_t , SetType ) - > item_type ) ) ;
2024-11-09 14:26:01 -08:00
return CORD_all ( " ({ Table_t *set = " , self , " ; " ,
2024-09-05 11:56:37 -07:00
" Array_t to_add = " , compile_arguments ( env , ast , arg_spec , call - > args ) , " ; " ,
2024-08-10 12:15:38 -07:00
" for (int64_t i = 0; i < to_add.length; i++) \n "
" Table$set(set, to_add.data + i*to_add.stride, NULL, " , compile_type_info ( env , self_value_t ) , " ); \n " ,
" (void)0; }) " ) ;
} else if ( streq ( call - > name , " remove " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " a " , " set " ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " item " , . type = set - > item_type ) ;
return CORD_all ( " Table$remove_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
compile_type_info ( env , self_value_t ) , " ) " ) ;
} else if ( streq ( call - > name , " remove_all " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " a " , " set " ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " items " , . type = Type ( ArrayType , . item_type = Match ( self_value_t , SetType ) - > item_type ) ) ;
2024-11-09 14:26:01 -08:00
return CORD_all ( " ({ Table_t *set = " , self , " ; " ,
2024-09-05 11:56:37 -07:00
" Array_t to_add = " , compile_arguments ( env , ast , arg_spec , call - > args ) , " ; " ,
2024-08-10 12:15:38 -07:00
" for (int64_t i = 0; i < to_add.length; i++) \n "
" Table$remove(set, to_add.data + i*to_add.stride, " , compile_type_info ( env , self_value_t ) , " ); \n " ,
" (void)0; }) " ) ;
} else if ( streq ( call - > name , " clear " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " a " , " set " ) ;
2024-08-10 12:15:38 -07:00
( void ) compile_arguments ( env , ast , NULL , call - > args ) ;
return CORD_all ( " Table$clear( " , self , " ) " ) ;
} else if ( streq ( call - > name , " with " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " other " , . type = self_value_t ) ;
return CORD_all ( " Table$with( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) ,
" , " , compile_type_info ( env , self_value_t ) , " ) " ) ;
} else if ( streq ( call - > name , " overlap " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " other " , . type = self_value_t ) ;
return CORD_all ( " Table$overlap( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) ,
" , " , compile_type_info ( env , self_value_t ) , " ) " ) ;
} else if ( streq ( call - > name , " without " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " other " , . type = self_value_t ) ;
return CORD_all ( " Table$without( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) ,
" , " , compile_type_info ( env , self_value_t ) , " ) " ) ;
} else if ( streq ( call - > name , " is_subset_of " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " other " , . type = self_value_t ,
. next = new ( arg_t , . name = " strict " , . type = Type ( BoolType ) , . default_val = FakeAST ( Bool , false ) ) ) ;
return CORD_all ( " Table$is_subset_of( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) ,
" , " , compile_type_info ( env , self_value_t ) , " ) " ) ;
} else if ( streq ( call - > name , " is_superset_of " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " other " , . type = self_value_t ,
. next = new ( arg_t , . name = " strict " , . type = Type ( BoolType ) , . default_val = FakeAST ( Bool , false ) ) ) ;
return CORD_all ( " Table$is_superset_of( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) ,
" , " , compile_type_info ( env , self_value_t ) , " ) " ) ;
} else code_err ( ast , " There is no '%s' method for tables " , call - > name ) ;
}
2024-08-11 11:47:34 -07:00
case ChannelType : {
type_t * item_t = Match ( self_value_t , ChannelType ) - > item_type ;
2024-10-08 18:10:36 -07:00
CORD padded_item_size = CORD_asprintf ( " %ld " , type_size ( item_t ) ) ;
2024-09-03 00:53:36 -07:00
arg_t * front_default_end = new ( arg_t , . name = " front " , . type = Type ( BoolType ) , . default_val = FakeAST ( Bool , false ) ) ;
arg_t * front_default_start = new ( arg_t , . name = " front " , . type = Type ( BoolType ) , . default_val = FakeAST ( Bool , true ) ) ;
2024-08-18 20:24:23 -07:00
if ( streq ( call - > name , " give " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-09-03 00:53:36 -07:00
arg_t * arg_spec = new ( arg_t , . name = " item " , . type = item_t , . next = front_default_end ) ;
2024-08-18 20:24:23 -07:00
return CORD_all ( " Channel$give_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
2024-08-11 11:47:34 -07:00
padded_item_size , " ) " ) ;
2024-08-18 20:24:23 -07:00
} else if ( streq ( call - > name , " give_all " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-09-03 00:53:36 -07:00
arg_t * arg_spec = new ( arg_t , . name = " to_give " , . type = Type ( ArrayType , . item_type = item_t ) , . next = front_default_end ) ;
2024-08-18 20:24:23 -07:00
return CORD_all ( " Channel$give_all( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
2024-08-11 11:47:34 -07:00
padded_item_size , " ) " ) ;
2024-08-18 20:24:23 -07:00
} else if ( streq ( call - > name , " get " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-09-03 00:53:36 -07:00
arg_t * arg_spec = front_default_start ;
2024-08-18 20:59:13 -07:00
return CORD_all ( " Channel$get_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) ,
" , " , compile_type ( item_t ) , " , " , padded_item_size , " ) " ) ;
2024-08-18 20:31:36 -07:00
} else if ( streq ( call - > name , " peek " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-09-03 00:53:36 -07:00
arg_t * arg_spec = front_default_start ;
2024-08-18 20:59:13 -07:00
return CORD_all ( " Channel$peek_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) ,
" , " , compile_type ( item_t ) , " ) " ) ;
2024-08-11 11:47:34 -07:00
} else if ( streq ( call - > name , " clear " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-11 11:47:34 -07:00
( void ) compile_arguments ( env , ast , NULL , call - > args ) ;
return CORD_all ( " Channel$clear( " , self , " ) " ) ;
} else if ( streq ( call - > name , " view " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-11 11:47:34 -07:00
( void ) compile_arguments ( env , ast , NULL , call - > args ) ;
return CORD_all ( " Channel$view( " , self , " ) " ) ;
} else code_err ( ast , " There is no '%s' method for channels " , call - > name ) ;
}
2024-03-08 23:48:09 -08:00
case TableType : {
auto table = Match ( self_value_t , TableType ) ;
if ( streq ( call - > name , " get " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-17 12:01:01 -07:00
arg_t * arg_spec = new ( arg_t , . name = " key " , . type = table - > key_type ) ;
2024-09-11 21:55:43 -07:00
return CORD_all (
" Table$get_optional( " , self , " , " , compile_type ( table - > key_type ) , " , " ,
compile_type ( table - > value_type ) , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
" _, " , optional_into_nonnull ( table - > value_type , " (*_) " ) , " , " , compile_null ( table - > value_type ) , " , " ,
compile_type_info ( env , self_value_t ) , " ) " ) ;
2024-08-10 12:15:38 -07:00
} else if ( streq ( call - > name , " has " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-10 12:15:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " key " , . type = table - > key_type ) ;
return CORD_all ( " Table$has_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
2024-03-08 23:48:09 -08:00
compile_type_info ( env , self_value_t ) , " ) " ) ;
2024-03-14 10:28:30 -07:00
} else if ( streq ( call - > name , " set " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " a " , " table " ) ;
2024-03-17 11:46:36 -07:00
arg_t * arg_spec = new ( arg_t , . name = " key " , . type = table - > key_type ,
. next = new ( arg_t , . name = " value " , . type = table - > value_type ) ) ;
2024-03-29 09:54:31 -07:00
return CORD_all ( " Table$set_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
2024-03-14 10:28:30 -07:00
compile_type_info ( env , self_value_t ) , " ) " ) ;
2024-08-10 13:03:41 -07:00
} else if ( streq ( call - > name , " bump " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " a " , " table " ) ;
2024-08-10 13:03:41 -07:00
if ( ! ( table - > value_type - > tag = = IntType | | table - > value_type - > tag = = NumType ) )
code_err ( ast , " bump() is only supported for tables with numeric value types, not %T " , self_value_t ) ;
ast_t * one = table - > value_type - > tag = = IntType
2024-08-18 08:49:51 -07:00
? FakeAST ( Int , . str = " 1 " )
: FakeAST ( Num , . n = 1 ) ;
2024-08-10 13:03:41 -07:00
arg_t * arg_spec = new ( arg_t , . name = " key " , . type = table - > key_type ,
. next = new ( arg_t , . name = " amount " , . type = table - > value_type , . default_val = one ) ) ;
return CORD_all ( " Table$bump( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
compile_type_info ( env , self_value_t ) , " ) " ) ;
2024-03-14 10:28:30 -07:00
} else if ( streq ( call - > name , " remove " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " a " , " table " ) ;
2024-08-08 10:20:38 -07:00
arg_t * arg_spec = new ( arg_t , . name = " key " , . type = table - > key_type ) ;
return CORD_all ( " Table$remove_value( " , self , " , " , compile_arguments ( env , ast , arg_spec , call - > args ) , " , " ,
2024-03-14 10:28:30 -07:00
compile_type_info ( env , self_value_t ) , " ) " ) ;
} else if ( streq ( call - > name , " clear " ) ) {
2024-11-09 14:26:01 -08:00
EXPECT_POINTER ( " a " , " table " ) ;
2024-04-02 10:08:06 -07:00
( void ) compile_arguments ( env , ast , NULL , call - > args ) ;
2024-03-29 09:54:31 -07:00
return CORD_all ( " Table$clear( " , self , " ) " ) ;
2024-08-03 14:42:45 -07:00
} else if ( streq ( call - > name , " sorted " ) ) {
2024-11-09 14:26:01 -08:00
self = compile_to_pointer_depth ( env , call - > self , 0 , false ) ;
2024-08-03 14:42:45 -07:00
( void ) compile_arguments ( env , ast , NULL , call - > args ) ;
return CORD_all ( " Table$sorted( " , self , " , " , compile_type_info ( env , self_value_t ) , " ) " ) ;
2024-03-14 10:28:30 -07:00
} else code_err ( ast , " There is no '%s' method for tables " , call - > name ) ;
2024-02-22 09:45:12 -08:00
}
2024-03-14 10:28:30 -07:00
default : {
auto methodcall = Match ( ast , MethodCall ) ;
type_t * fn_t = get_method_type ( env , methodcall - > self , methodcall - > name ) ;
arg_ast_t * args = new ( arg_ast_t , . value = methodcall - > self , . next = methodcall - > args ) ;
binding_t * b = get_namespace_binding ( env , methodcall - > self , methodcall - > name ) ;
if ( ! b ) code_err ( ast , " No such method " ) ;
return CORD_all ( b - > code , " ( " , compile_arguments ( env , ast , Match ( fn_t , FunctionType ) - > args , args ) , " ) " ) ;
2024-02-22 09:45:12 -08:00
}
2024-02-18 01:26:26 -08:00
}
2024-11-09 14:26:01 -08:00
# undef EXPECT_POINTER
2024-03-14 10:28:30 -07:00
}
case FunctionCall : {
auto call = Match ( ast , FunctionCall ) ;
type_t * fn_t = get_type ( env , call - > fn ) ;
if ( fn_t - > tag = = FunctionType ) {
CORD fn = compile ( env , call - > fn ) ;
return CORD_all ( fn , " ( " , compile_arguments ( env , ast , Match ( fn_t , FunctionType ) - > args , call - > args ) , " ) " ) ;
} else if ( fn_t - > tag = = TypeInfoType ) {
type_t * t = Match ( fn_t , TypeInfoType ) - > type ;
2024-04-23 10:12:49 -07:00
if ( t - > tag = = StructType ) {
2024-05-12 10:50:06 -07:00
// Struct constructor:
2024-04-23 10:12:49 -07:00
fn_t = Type ( FunctionType , . args = Match ( t , StructType ) - > fields , . ret = t ) ;
2024-06-16 15:09:54 -07:00
return CORD_all ( " (( " , compile_type ( t ) , " ) { " , compile_arguments(env, ast, Match(fn_t, FunctionType)->args, call->args), " } ) " );
2024-11-05 12:18:32 -08:00
} else if ( is_numeric_type ( t ) & & call - > args & & call - > args - > value - > tag = = Int ) {
if ( call - > args - > next )
code_err ( call - > args - > next - > value , " This is too many arguments to an integer literal constructor " ) ;
return compile_int_to_type ( env , call - > args - > value , t ) ;
} else if ( t - > tag = = NumType & & call - > args & & call - > args - > value - > tag = = Num ) {
if ( call - > args - > next )
code_err ( call - > args - > next - > value , " This is too many arguments to a number literal constructor " ) ;
return compile_num_to_type ( call - > args - > value , t ) ;
2024-08-18 08:49:51 -07:00
} else if ( t - > tag = = NumType | | t - > tag = = BigIntType ) {
2024-08-19 10:23:02 -07:00
if ( ! call - > args ) code_err ( ast , " This constructor needs a value " ) ;
2024-08-12 23:20:48 -07:00
type_t * actual = get_type ( env , call - > args - > value ) ;
2024-08-14 11:57:01 -07:00
arg_t * args = new ( arg_t , . name = " i " , . type = actual ) ; // No truncation argument
CORD arg_code = compile_arguments ( env , ast , args , call - > args ) ;
2024-11-04 12:24:10 -08:00
if ( is_numeric_type ( actual ) ) {
2024-09-24 09:40:56 -07:00
return CORD_all ( type_to_cord ( actual ) , " _to_ " , type_to_cord ( t ) , " ( " , arg_code , " ) " ) ;
} else if ( actual - > tag = = BoolType ) {
if ( t - > tag = = NumType ) {
return CORD_all ( " (( " , compile_type ( t ) , " )( " , arg_code , " )) " ) ;
} else {
return CORD_all ( " I((int) ( " , arg_code, " ) ) " ) ;
}
} else {
code_err ( ast , " You cannot convert a %T to a %T this way. " , actual , t ) ;
}
2024-11-03 22:12:37 -08:00
} else if ( t - > tag = = IntType | | t - > tag = = ByteType ) {
2024-04-23 10:12:49 -07:00
type_t * actual = get_type ( env , call - > args - > value ) ;
2024-11-04 12:24:10 -08:00
if ( is_numeric_type ( actual ) ) {
2024-09-24 09:40:56 -07:00
arg_t * args = new ( arg_t , . name = " i " , . type = actual , . next = new ( arg_t , . name = " truncate " , . type = Type ( BoolType ) ,
. default_val = FakeAST ( Bool , false ) ) ) ;
CORD arg_code = compile_arguments ( env , ast , args , call - > args ) ;
return CORD_all ( type_to_cord ( actual ) , " _to_ " , type_to_cord ( t ) , " ( " , arg_code , " ) " ) ;
} else if ( actual - > tag = = BoolType ) {
arg_t * args = new ( arg_t , . name = " i " , . type = actual ) ;
CORD arg_code = compile_arguments ( env , ast , args , call - > args ) ;
return CORD_all ( " (( " , compile_type ( t ) , " )( " , arg_code , " )) " ) ;
} else {
code_err ( ast , " You cannot convert a %T to a %T this way. " , actual , t ) ;
}
2024-05-18 11:38:41 -07:00
} else if ( t - > tag = = TextType ) {
2024-08-19 10:23:02 -07:00
if ( ! call - > args ) code_err ( ast , " This constructor needs a value " ) ;
const char * lang = Match ( t , TextType ) - > lang ;
if ( lang ) { // Escape for DSL
type_t * first_type = get_type ( env , call - > args - > value ) ;
2024-08-19 10:23:59 -07:00
if ( type_eq ( first_type , t ) )
return compile ( env , call - > args - > value ) ;
2024-08-19 10:23:02 -07:00
binding_t * esc = get_lang_escape_function ( env , lang , first_type ) ;
if ( ! esc )
code_err ( ast , " I don't know how to convert %T to %T " , first_type , t ) ;
arg_t * arg_spec = Match ( esc - > type , FunctionType ) - > args ;
return CORD_all ( esc - > code , " ( " , compile_arguments ( env , ast , arg_spec , call - > args ) , " ) " ) ;
} else {
// Text constructor:
if ( ! call - > args | | call - > args - > next )
code_err ( call - > fn , " This constructor takes exactly 1 argument " ) ;
type_t * actual = get_type ( env , call - > args - > value ) ;
if ( type_eq ( actual , t ) )
return compile ( env , call - > args - > value ) ;
return expr_as_text ( env , compile ( env , call - > args - > value ) , actual , " no " ) ;
}
2024-05-18 11:38:41 -07:00
} else if ( t - > tag = = CStringType ) {
// C String constructor:
if ( ! call - > args | | call - > args - > next )
code_err ( call - > fn , " This constructor takes exactly 1 argument " ) ;
2024-09-11 23:18:05 -07:00
if ( call - > args - > value - > tag = = TextLiteral )
return compile_string_literal ( Match ( call - > args - > value , TextLiteral ) - > cord ) ;
else if ( call - > args - > value - > tag = = TextJoin & & Match ( call - > args - > value , TextJoin ) - > children = = NULL )
return " \" \" " ;
else if ( call - > args - > value - > tag = = TextJoin & & Match ( call - > args - > value , TextJoin ) - > children - > next = = NULL )
return compile_string_literal ( Match ( Match ( call - > args - > value , TextJoin ) - > children - > ast , TextLiteral ) - > cord ) ;
2024-05-18 11:38:41 -07:00
type_t * actual = get_type ( env , call - > args - > value ) ;
2024-09-02 16:18:21 -07:00
return CORD_all ( " Text$as_c_string( " , expr_as_text ( env , compile ( env , call - > args - > value ) , actual , " no " ) , " ) " ) ;
2024-11-17 11:49:03 -08:00
} else if ( t - > tag = = MomentType ) {
// Moment constructor:
2024-09-29 17:19:46 -07:00
binding_t * new_binding = get_binding ( Match ( fn_t , TypeInfoType ) - > env , " new " ) ;
CORD arg_code = compile_arguments ( env , ast , Match ( new_binding - > type , FunctionType ) - > args , call - > args ) ;
return CORD_all ( new_binding - > code , " ( " , arg_code , " ) " ) ;
2024-04-23 10:12:49 -07:00
} else {
2024-03-14 10:28:30 -07:00
code_err ( call - > fn , " This is not a type that has a constructor " ) ;
2024-04-23 10:12:49 -07:00
}
2024-03-14 10:28:30 -07:00
} else if ( fn_t - > tag = = ClosureType ) {
fn_t = Match ( fn_t , ClosureType ) - > fn ;
arg_t * type_args = Match ( fn_t , FunctionType ) - > args ;
2024-02-18 01:26:26 -08:00
2024-03-14 10:28:30 -07:00
arg_t * closure_fn_args = NULL ;
for ( arg_t * arg = Match ( fn_t , FunctionType ) - > args ; arg ; arg = arg - > next )
closure_fn_args = new ( arg_t , . name = arg - > name , . type = arg - > type , . default_val = arg - > default_val , . next = closure_fn_args ) ;
2024-04-16 10:50:07 -07:00
closure_fn_args = new ( arg_t , . name = " userdata " , . type = Type ( PointerType , . pointed = Type ( MemoryType ) ) , . next = closure_fn_args ) ;
2024-03-14 10:28:30 -07:00
REVERSE_LIST ( closure_fn_args ) ;
2024-06-16 15:09:54 -07:00
CORD fn_type_code = compile_type ( Type ( FunctionType , . args = closure_fn_args , . ret = Match ( fn_t , FunctionType ) - > ret ) ) ;
2024-03-14 10:28:30 -07:00
CORD closure = compile ( env , call - > fn ) ;
CORD arg_code = compile_arguments ( env , ast , type_args , call - > args ) ;
if ( arg_code ) arg_code = CORD_cat ( arg_code , " , " ) ;
if ( call - > fn - > tag = = Var ) {
return CORD_all ( " (( " , fn_type_code , " ) " , closure, " . fn ) ( " , arg_code, closure, " . userdata ) " ) ;
2024-03-06 09:56:30 -08:00
} else {
2024-09-11 10:56:39 -07:00
return CORD_all ( " ({ Closure_t closure = " , closure , " ; (( " , fn_type_code , " )closure.fn)( " ,
2024-04-16 10:50:07 -07:00
arg_code , " closure.userdata); }) " ) ;
2024-03-06 09:56:30 -08:00
}
2024-03-14 10:28:30 -07:00
} else {
code_err ( call - > fn , " This is not a function, it's a %T " , fn_t ) ;
2024-02-18 01:26:26 -08:00
}
2024-03-14 10:28:30 -07:00
}
2024-09-06 11:15:55 -07:00
case When : {
auto original = Match ( ast , When ) ;
ast_t * when_var = WrapAST ( ast , Var , . name = " when " ) ;
when_clause_t * new_clauses = NULL ;
type_t * subject_t = get_type ( env , original - > subject ) ;
for ( when_clause_t * clause = original - > clauses ; clause ; clause = clause - > next ) {
type_t * clause_type = get_clause_type ( env , subject_t , clause ) ;
if ( clause_type - > tag = = AbortType | | clause_type - > tag = = ReturnType ) {
new_clauses = new ( when_clause_t , . tag_name = clause - > tag_name , . args = clause - > args , . body = clause - > body , . next = new_clauses ) ;
} else {
ast_t * assign = WrapAST ( clause - > body , Assign ,
. targets = new ( ast_list_t , . ast = when_var ) ,
. values = new ( ast_list_t , . ast = clause - > body ) ) ;
new_clauses = new ( when_clause_t , . tag_name = clause - > tag_name , . args = clause - > args , . body = assign , . next = new_clauses ) ;
}
}
REVERSE_LIST ( new_clauses ) ;
ast_t * else_body = original - > else_body ;
if ( else_body ) {
type_t * clause_type = get_type ( env , else_body ) ;
if ( clause_type - > tag ! = AbortType & & clause_type - > tag ! = ReturnType ) {
else_body = WrapAST ( else_body , Assign ,
. targets = new ( ast_list_t , . ast = when_var ) ,
. values = new ( ast_list_t , . ast = else_body ) ) ;
}
}
type_t * t = get_type ( env , ast ) ;
env_t * when_env = fresh_scope ( env ) ;
set_binding ( when_env , " when " , new ( binding_t , . type = t , . code = " when " ) ) ;
return CORD_all (
" ({ " , compile_declaration ( t , " when " ) , " ; \n " ,
compile_statement ( when_env , WrapAST ( ast , When , . subject = original - > subject , . clauses = new_clauses , . else_body = else_body ) ) ,
" when; }) " ) ;
}
2024-03-14 10:28:30 -07:00
case If : {
auto if_ = Match ( ast , If ) ;
2024-09-15 13:29:45 -07:00
ast_t * condition = if_ - > condition ;
CORD decl_code = CORD_EMPTY ;
env_t * truthy_scope = env , * falsey_scope = env ;
2024-03-14 10:28:30 -07:00
2024-11-11 21:04:56 -08:00
CORD condition_code ;
2024-09-15 13:29:45 -07:00
if ( condition - > tag = = Declare ) {
2024-11-11 21:04:56 -08:00
type_t * condition_type = get_type ( env , Match ( condition , Declare ) - > value ) ;
if ( condition_type - > tag ! = OptionalType )
code_err ( condition , " This `if var := ...:` declaration should be an optional type, not %T " , condition_type ) ;
2024-09-15 13:29:45 -07:00
decl_code = compile_statement ( env , condition ) ;
2024-11-11 21:04:56 -08:00
ast_t * var = Match ( condition , Declare ) - > var ;
truthy_scope = fresh_scope ( env ) ;
bind_statement ( truthy_scope , condition ) ;
condition_code = compile_condition ( truthy_scope , var ) ;
set_binding ( truthy_scope , Match ( var , Var ) - > name ,
new ( binding_t , . type = Match ( condition_type , OptionalType ) - > type ,
. code = optional_into_nonnull ( condition_type , compile ( truthy_scope , var ) ) ) ) ;
2024-09-15 13:29:45 -07:00
} else if ( condition - > tag = = Var ) {
2024-11-11 21:04:56 -08:00
type_t * condition_type = get_type ( env , condition ) ;
condition_code = compile ( env , condition ) ;
2024-09-15 13:29:45 -07:00
if ( condition_type - > tag = = OptionalType ) {
truthy_scope = fresh_scope ( env ) ;
2024-11-11 21:04:56 -08:00
set_binding ( truthy_scope , Match ( condition , Var ) - > name ,
new ( binding_t , . type = Match ( condition_type , OptionalType ) - > type ,
. code = optional_into_nonnull ( condition_type , compile ( truthy_scope , condition ) ) ) ) ;
2024-09-15 13:29:45 -07:00
}
} else {
condition_code = compile ( env , condition ) ;
}
2024-09-06 20:01:31 -07:00
2024-09-15 13:29:45 -07:00
type_t * true_type = get_type ( truthy_scope , if_ - > body ) ;
type_t * false_type = get_type ( falsey_scope , if_ - > else_body ) ;
2024-07-13 15:43:50 -07:00
if ( true_type - > tag = = AbortType | | true_type - > tag = = ReturnType )
2024-09-15 13:29:45 -07:00
return CORD_all ( " ({ " , decl_code , " if ( " , condition_code , " ) " , compile_statement ( truthy_scope , if_ - > body ) ,
" \n " , compile ( falsey_scope , if_ - > else_body ) , " ; }) " ) ;
2024-07-13 15:43:50 -07:00
else if ( false_type - > tag = = AbortType | | false_type - > tag = = ReturnType )
2024-09-15 13:29:45 -07:00
return CORD_all ( " ({ " , decl_code , " if (!( " , condition_code , " )) " , compile_statement ( falsey_scope , if_ - > else_body ) ,
" \n " , compile ( truthy_scope , if_ - > body ) , " ; }) " ) ;
else if ( decl_code ! = CORD_EMPTY )
return CORD_all ( " ({ " , decl_code , " ( " , condition_code , " ) ? " , compile ( truthy_scope , if_ - > body ) , " : " ,
compile ( falsey_scope , if_ - > else_body ) , " ;}) " ) ;
2024-05-23 21:03:46 -07:00
else
2024-09-15 13:29:45 -07:00
return CORD_all ( " (( " , condition_code , " ) ? " ,
compile ( truthy_scope , if_ - > body ) , " : " , compile ( falsey_scope , if_ - > else_body ) , " ) " ) ;
2024-02-04 18:13:50 -08:00
}
2024-02-25 12:28:46 -08:00
case Reduction : {
auto reduction = Match ( ast , Reduction ) ;
2024-11-08 11:10:19 -08:00
binop_e op = reduction - > op ;
2024-11-02 19:34:35 -07:00
2024-11-07 10:27:09 -08:00
type_t * iter_t = get_type ( env , reduction - > iter ) ;
type_t * item_t = get_iterated_type ( iter_t ) ;
if ( ! item_t ) code_err ( reduction - > iter , " I couldn't figure out how to iterate over this type: %T " , iter_t ) ;
2024-11-08 11:10:19 -08:00
static int64_t next_id = 1 ;
ast_t * item = FakeAST ( Var , heap_strf ( " $it%ld " , next_id + + ) ) ;
ast_t * body = FakeAST ( InlineCCode , . code = " {} " ) ; // placeholder
ast_t * loop = FakeAST ( For , . vars = new ( ast_list_t , . ast = item ) , . iter = reduction - > iter , . body = body ) ;
env_t * body_scope = for_scope ( env , loop ) ;
2024-11-02 19:34:35 -07:00
if ( op = = BINOP_EQ | | op = = BINOP_NE | | op = = BINOP_LT | | op = = BINOP_LE | | op = = BINOP_GT | | op = = BINOP_GE ) {
// Chained comparisons like ==, <, etc.
CORD code = CORD_all (
" ({ // Reduction: \n " ,
compile_declaration ( item_t , " prev " ) , " ; \n "
" OptionalBool_t result = NULL_BOOL; \n "
) ;
2024-11-08 11:10:19 -08:00
ast_t * comparison = WrapAST ( ast , BinaryOp , . op = op , . lhs = FakeAST ( InlineCCode , . code = " prev " , . type = item_t ) , . rhs = item ) ;
2024-11-02 19:34:35 -07:00
body - > __data . InlineCCode . code = CORD_all (
" if (result == NULL_BOOL) { \n "
" prev = " , compile ( body_scope , item ) , " ; \n "
" result = yes; \n "
" } else { \n "
2024-11-08 11:10:19 -08:00
" if ( " , compile ( body_scope , comparison ) , " ) { \n " ,
2024-11-02 19:34:35 -07:00
" prev = " , compile ( body_scope , item ) , " ; \n " ,
" } else { \n "
" result = no; \n " ,
" break; \n " ,
" } \n " ,
" } \n " ) ;
2024-11-08 11:10:19 -08:00
code = CORD_all ( code , compile_statement ( env , loop ) , " \n result;}) " ) ;
return code ;
} else if ( op = = BINOP_MIN | | op = = BINOP_MAX ) {
// Min/max:
const char * superlative = op = = BINOP_MIN ? " min " : " max " ;
CORD code = CORD_all (
" ({ // Reduction: \n " ,
compile_declaration ( item_t , superlative ) , " ; \n "
" Bool_t has_value = no; \n "
) ;
CORD item_code = compile ( body_scope , item ) ;
binop_e cmp_op = op = = BINOP_MIN ? BINOP_LT : BINOP_GT ;
if ( reduction - > key ) {
env_t * key_scope = fresh_scope ( env ) ;
set_binding ( key_scope , " $ " , new ( binding_t , . type = item_t , . code = item_code ) ) ;
type_t * key_type = get_type ( key_scope , reduction - > key ) ;
const char * superlative_key = op = = BINOP_MIN ? " min_key " : " max_key " ;
code = CORD_all ( code , compile_declaration ( key_type , superlative_key ) , " ; \n " ) ;
ast_t * comparison = WrapAST ( ast , BinaryOp , . op = cmp_op ,
. lhs = FakeAST ( InlineCCode , . code = " key " , . type = key_type ) ,
. rhs = FakeAST ( InlineCCode , . code = superlative_key , . type = key_type ) ) ;
body - > __data . InlineCCode . code = CORD_all (
compile_declaration ( key_type , " key " ) , " = " , compile ( key_scope , reduction - > key ) , " ; \n " ,
" if (!has_value || " , compile ( body_scope , comparison ) , " ) { \n "
" " , superlative , " = " , compile ( body_scope , item ) , " ; \n "
" " , superlative_key , " = key; \n "
" has_value = yes; \n "
" } \n " ) ;
} else {
ast_t * comparison = WrapAST ( ast , BinaryOp , . op = cmp_op , . lhs = item , . rhs = FakeAST ( InlineCCode , . code = superlative , . type = item_t ) ) ;
body - > __data . InlineCCode . code = CORD_all (
" if (!has_value || " , compile ( body_scope , comparison ) , " ) { \n "
" " , superlative , " = " , compile ( body_scope , item ) , " ; \n "
" has_value = yes; \n "
" } \n " ) ;
}
code = CORD_all ( code , compile_statement ( env , loop ) , " \n has_value ? " , promote_to_optional ( item_t , superlative ) ,
" : " , compile_null ( item_t ) , " ;}) " ) ;
2024-11-02 19:34:35 -07:00
return code ;
} else {
// Accumulator-style reductions like +, ++, *, etc.
CORD code = CORD_all (
" ({ // Reduction: \n " ,
2024-11-07 10:27:09 -08:00
compile_declaration ( item_t , " reduction " ) , " ; \n "
2024-11-02 19:34:35 -07:00
" Bool_t has_value = no; \n "
) ;
// For the special case of (or)/(and), we need to early out if we can:
CORD early_out = CORD_EMPTY ;
if ( op = = BINOP_CMP ) {
2024-11-07 10:27:09 -08:00
if ( item_t - > tag ! = IntType | | Match ( item_t , IntType ) - > bits ! = TYPE_IBITS32 )
2024-11-02 19:34:35 -07:00
code_err ( ast , " <> reductions are only supported for Int32 values " ) ;
} else if ( op = = BINOP_AND ) {
2024-11-07 10:27:09 -08:00
if ( item_t - > tag = = BoolType )
2024-11-02 19:34:35 -07:00
early_out = " if (!reduction) break; " ;
2024-11-07 10:27:09 -08:00
else if ( item_t - > tag = = OptionalType )
early_out = CORD_all ( " if ( " , check_null ( item_t , " reduction " ) , " ) break; " ) ;
2024-11-02 19:34:35 -07:00
} else if ( op = = BINOP_OR ) {
2024-11-07 10:27:09 -08:00
if ( item_t - > tag = = BoolType )
2024-11-02 19:34:35 -07:00
early_out = " if (reduction) break; " ;
2024-11-07 10:27:09 -08:00
else if ( item_t - > tag = = OptionalType )
early_out = CORD_all ( " if (! " , check_null ( item_t , " reduction " ) , " ) break; " ) ;
2024-11-02 19:34:35 -07:00
}
2024-11-08 11:10:19 -08:00
ast_t * combination = WrapAST ( ast , BinaryOp , . op = op , . lhs = FakeAST ( InlineCCode , . code = " reduction " , . type = item_t ) , . rhs = item ) ;
2024-11-02 19:34:35 -07:00
body - > __data . InlineCCode . code = CORD_all (
" if (!has_value) { \n "
" reduction = " , compile ( body_scope , item ) , " ; \n "
" has_value = yes; \n "
" } else { \n "
2024-11-08 11:10:19 -08:00
" reduction = " , compile ( body_scope , combination ) , " ; \n " ,
2024-11-02 19:34:35 -07:00
early_out ,
" } \n " ) ;
2024-11-08 11:10:19 -08:00
code = CORD_all ( code , compile_statement ( env , loop ) , " \n has_value ? " , promote_to_optional ( item_t , " reduction " ) ,
2024-11-07 10:27:09 -08:00
" : " , compile_null ( item_t ) , " ;}) " ) ;
2024-11-02 19:34:35 -07:00
return code ;
}
2024-02-25 12:28:46 -08:00
}
2024-02-11 22:46:32 -08:00
case FieldAccess : {
auto f = Match ( ast , FieldAccess ) ;
2024-02-17 17:47:43 -08:00
type_t * fielded_t = get_type ( env , f - > fielded ) ;
2024-02-20 10:06:03 -08:00
type_t * value_t = value_type ( fielded_t ) ;
switch ( value_t - > tag ) {
2024-03-03 10:04:50 -08:00
case TypeInfoType : {
auto info = Match ( value_t , TypeInfoType ) ;
2024-08-17 18:03:04 -07:00
if ( f - > field [ 0 ] = = ' _ ' ) {
2024-09-05 11:57:31 -07:00
for ( Table_t * locals = env - > locals ; locals ; locals = locals - > fallback ) {
2024-08-17 18:03:04 -07:00
if ( locals = = info - > env - > locals )
goto is_inside_type ;
}
code_err ( ast , " Fields that start with underscores are not accessible on types outside of the type definition. " , f - > field ) ;
is_inside_type : ;
}
2024-03-21 10:33:10 -07:00
binding_t * b = get_binding ( info - > env , f - > field ) ;
2024-03-03 10:04:50 -08:00
if ( ! b ) code_err ( ast , " I couldn't find the field '%s' on this type " , f - > field ) ;
if ( ! b - > code ) code_err ( ast , " I couldn't figure out how to compile this field " ) ;
return b - > code ;
}
2024-08-19 11:29:58 -07:00
case TextType : {
const char * lang = Match ( value_t , TextType ) - > lang ;
if ( lang & & streq ( f - > field , " text_content " ) ) {
2024-08-20 13:20:41 -07:00
CORD text = compile_to_pointer_depth ( env , f - > fielded , 0 , false ) ;
2024-08-19 11:29:58 -07:00
return CORD_all ( " ((Text_t) " , text, " ) " ) ;
2024-09-02 17:22:13 -07:00
} else if ( streq ( f - > field , " length " ) ) {
return CORD_all ( " Int64_to_Int(( " , compile_to_pointer_depth ( env , f - > fielded , 0 , false ) , " ).length) " ) ;
2024-08-19 11:29:58 -07:00
}
code_err ( ast , " There is no '%s' field on %T values " , f - > field , value_t ) ;
}
2024-02-20 10:06:03 -08:00
case StructType : {
for ( arg_t * field = Match ( value_t , StructType ) - > fields ; field ; field = field - > next ) {
if ( streq ( field - > name , f - > field ) ) {
2024-11-09 13:27:54 -08:00
const char * prefix = ( value_t = = RANGE_TYPE | | value_t = = MATCH_TYPE ) ? " " : " $ " ;
2024-02-20 10:06:03 -08:00
if ( fielded_t - > tag = = PointerType ) {
2024-08-20 13:20:41 -07:00
CORD fielded = compile_to_pointer_depth ( env , f - > fielded , 1 , false ) ;
2024-11-09 13:27:54 -08:00
return CORD_asprintf ( " (%r) - > % s % s " , fielded, prefix, f->field) ;
2024-02-20 10:06:03 -08:00
} else {
CORD fielded = compile ( env , f - > fielded ) ;
2024-11-09 13:27:54 -08:00
return CORD_asprintf ( " (%r) . % s % s " , fielded, prefix, f->field) ;
2024-02-20 10:06:03 -08:00
}
}
}
code_err ( ast , " The field '%s' is not a valid field name of %T " , f - > field , value_t ) ;
}
2024-08-18 17:58:36 -07:00
case EnumType : {
auto e = Match ( value_t , EnumType ) ;
for ( tag_t * tag = e - > tags ; tag ; tag = tag - > next ) {
if ( streq ( f - > field , tag - > name ) ) {
2024-09-24 11:54:22 -07:00
CORD prefix = namespace_prefix ( e - > env , e - > env - > namespace ) ;
2024-08-18 17:58:36 -07:00
if ( fielded_t - > tag = = PointerType ) {
2024-08-20 13:20:41 -07:00
CORD fielded = compile_to_pointer_depth ( env , f - > fielded , 1 , false ) ;
2024-08-18 17:58:36 -07:00
return CORD_all ( " (( " , fielded , " ) - > tag = = " , prefix, " tag $ " , tag->name, " ) " ) ;
} else {
CORD fielded = compile ( env , f - > fielded ) ;
return CORD_all ( " (( " , fielded , " ) . tag = = " , prefix, " tag $ " , tag->name, " ) " ) ;
}
}
}
code_err ( ast , " The field '%s' is not a valid tag name of %T " , f - > field , value_t ) ;
}
2024-08-18 17:28:16 -07:00
case ArrayType : {
if ( streq ( f - > field , " length " ) )
2024-08-20 13:20:41 -07:00
return CORD_all ( " Int64_to_Int(( " , compile_to_pointer_depth ( env , f - > fielded , 0 , false ) , " ).length) " ) ;
2024-08-18 17:28:16 -07:00
code_err ( ast , " There is no %s field on arrays " , f - > field ) ;
}
case ChannelType : {
if ( streq ( f - > field , " max_size " ) )
2024-08-20 13:20:41 -07:00
return CORD_all ( " Int64_to_Int(( " , compile_to_pointer_depth ( env , f - > fielded , 0 , false ) , " )->max_size) " ) ;
2024-08-18 17:28:16 -07:00
code_err ( ast , " There is no %s field on arrays " , f - > field ) ;
}
2024-08-10 12:15:38 -07:00
case SetType : {
2024-08-12 15:00:54 -07:00
if ( streq ( f - > field , " items " ) )
2024-08-20 13:20:41 -07:00
return CORD_all ( " ARRAY_COPY(( " , compile_to_pointer_depth ( env , f - > fielded , 0 , false ) , " ).entries) " ) ;
2024-08-18 17:28:16 -07:00
else if ( streq ( f - > field , " length " ) )
2024-08-20 13:20:41 -07:00
return CORD_all ( " Int64_to_Int(( " , compile_to_pointer_depth ( env , f - > fielded , 0 , false ) , " ).entries.length) " ) ;
2024-08-10 12:15:38 -07:00
code_err ( ast , " There is no '%s' field on sets " , f - > field ) ;
}
2024-02-25 11:35:25 -08:00
case TableType : {
2024-08-18 17:28:16 -07:00
if ( streq ( f - > field , " length " ) ) {
2024-08-20 13:20:41 -07:00
return CORD_all ( " Int64_to_Int(( " , compile_to_pointer_depth ( env , f - > fielded , 0 , false ) , " ).entries.length) " ) ;
2024-08-18 17:28:16 -07:00
} else if ( streq ( f - > field , " keys " ) ) {
2024-08-20 13:20:41 -07:00
return CORD_all ( " ARRAY_COPY(( " , compile_to_pointer_depth ( env , f - > fielded , 0 , false ) , " ).entries) " ) ;
2024-02-25 11:35:25 -08:00
} else if ( streq ( f - > field , " values " ) ) {
auto table = Match ( value_t , TableType ) ;
size_t offset = type_size ( table - > key_type ) ;
size_t align = type_align ( table - > value_type ) ;
if ( align > 1 & & offset % align > 0 )
offset + = align - ( offset % align ) ;
2024-09-05 11:56:37 -07:00
return CORD_all ( " ({ Array_t *entries = &( " , compile_to_pointer_depth ( env , f - > fielded , 0 , false ) , " ).entries ; \ n "
2024-08-12 14:57:14 -07:00
" ARRAY_INCREF(*entries); \n "
2024-09-05 11:56:37 -07:00
" Array_t values = *entries; \n "
2024-08-12 14:57:14 -07:00
" values.data += " , CORD_asprintf ( " %zu " , offset ) , " ; \n "
" values; }) " ) ;
2024-02-25 11:35:25 -08:00
} else if ( streq ( f - > field , " fallback " ) ) {
2024-09-10 22:31:31 -07:00
return CORD_all ( " ({ Table_t *_fallback = ( " , compile_to_pointer_depth ( env , f - > fielded , 0 , false ) , " ).fallback ; _fallback ? * _fallback : NULL_TABLE ; } ) " );
2024-02-25 11:35:25 -08:00
}
code_err ( ast , " There is no '%s' field on tables " , f - > field ) ;
}
2024-03-19 11:22:03 -07:00
case ModuleType : {
const char * name = Match ( value_t , ModuleType ) - > name ;
2024-03-29 09:54:31 -07:00
env_t * module_env = Table $ str_get ( * env - > imports , name ) ;
2024-03-19 11:22:03 -07:00
return compile ( module_env , WrapAST ( ast , Var , f - > field ) ) ;
}
2024-02-20 10:06:03 -08:00
default :
2024-03-19 11:22:03 -07:00
code_err ( ast , " Field accesses are not supported on %T values " , fielded_t ) ;
2024-02-17 17:47:43 -08:00
}
2024-02-11 22:46:32 -08:00
}
2024-02-18 11:28:35 -08:00
case Index : {
auto indexing = Match ( ast , Index ) ;
2024-02-25 09:20:25 -08:00
type_t * indexed_type = get_type ( env , indexing - > indexed ) ;
2024-08-12 14:44:05 -07:00
if ( ! indexing - > index ) {
if ( indexed_type - > tag ! = PointerType )
code_err ( ast , " Only pointers can use the '[]' operator to dereference the entire value. " ) ;
2024-02-29 10:00:28 -08:00
auto ptr = Match ( indexed_type , PointerType ) ;
if ( ptr - > pointed - > tag = = ArrayType ) {
2024-09-05 11:56:37 -07:00
return CORD_all ( " ({ Array_t *arr = " , compile ( env , indexing - > indexed ) , " ; ARRAY_INCREF ( * arr ) ; * arr ; } ) " );
2024-08-12 14:44:05 -07:00
} else if ( ptr - > pointed - > tag = = TableType | | ptr - > pointed - > tag = = SetType ) {
2024-09-05 11:57:31 -07:00
return CORD_all ( " ({ Table_t *t = " , compile ( env , indexing - > indexed ) , " ; TABLE_INCREF ( * t ) ; * t ; } ) " );
2024-02-29 10:00:28 -08:00
} else {
return CORD_all ( " *( " , compile ( env , indexing - > indexed ) , " ) " ) ;
}
2024-02-25 09:20:25 -08:00
}
2024-08-12 14:44:05 -07:00
2024-02-25 09:20:25 -08:00
type_t * container_t = value_type ( indexed_type ) ;
2024-02-18 11:28:35 -08:00
type_t * index_t = get_type ( env , indexing - > index ) ;
2024-08-10 13:03:41 -07:00
if ( container_t - > tag = = ArrayType ) {
2024-11-03 22:12:37 -08:00
if ( index_t - > tag ! = IntType & & index_t - > tag ! = BigIntType & & index_t - > tag ! = ByteType )
2024-02-18 11:28:35 -08:00
code_err ( indexing - > index , " Arrays can only be indexed by integers, not %T " , index_t ) ;
type_t * item_type = Match ( container_t , ArrayType ) - > item_type ;
2024-08-20 13:20:41 -07:00
CORD arr = compile_to_pointer_depth ( env , indexing - > indexed , 0 , false ) ;
2024-02-18 11:53:52 -08:00
file_t * f = indexing - > index - > file ;
if ( indexing - > unchecked )
2024-08-18 12:51:19 -07:00
return CORD_all ( " Array_get_unchecked( " , compile_type ( item_type ) , " , " , arr , " , " ,
compile_int_to_type ( env , indexing - > index , Type ( IntType , . bits = TYPE_IBITS64 ) ) , " ) " ) ;
2024-02-18 11:53:52 -08:00
else
2024-08-18 12:51:19 -07:00
return CORD_all ( " Array_get( " , compile_type ( item_type ) , " , " , arr , " , " ,
compile_int_to_type ( env , indexing - > index , Type ( IntType , . bits = TYPE_IBITS64 ) ) , " , " ,
2024-09-02 15:47:39 -07:00
CORD_asprintf ( " %ld " , ( int64_t ) ( indexing - > index - > start - f - > text ) ) , " , " ,
2024-02-18 11:53:52 -08:00
CORD_asprintf ( " %ld " , ( int64_t ) ( indexing - > index - > end - f - > text ) ) ,
" ) " ) ;
2024-08-10 13:03:41 -07:00
} else {
code_err ( ast , " Indexing is not supported for type: %T " , container_t ) ;
2024-02-18 11:28:35 -08:00
}
}
2024-05-18 13:31:34 -07:00
case InlineCCode : {
type_t * t = get_type ( env , ast ) ;
if ( t - > tag = = VoidType )
return CORD_all ( " { \n " , Match ( ast , InlineCCode ) - > code , " \n } " ) ;
else
2024-09-15 14:34:34 -07:00
return CORD_all ( " ({ " , Match ( ast , InlineCCode ) - > code , " ; } ) " );
2024-05-18 13:31:34 -07:00
}
2024-05-27 14:45:22 -07:00
case Use : code_err ( ast , " Compiling 'use' as expression! " ) ;
2024-07-04 15:00:01 -07:00
case Defer : code_err ( ast , " Compiling 'defer' as expression! " ) ;
2024-05-18 11:38:41 -07:00
case Extern : code_err ( ast , " Externs are not supported as expressions " ) ;
2024-02-29 10:49:24 -08:00
case TableEntry : code_err ( ast , " Table entries should not be compiled directly " ) ;
2024-11-03 09:20:53 -08:00
case Declare : case Assign : case UpdateAssign : case For : case While : case Repeat : case StructDef : case LangDef :
2024-07-04 13:23:05 -07:00
case EnumDef : case FunctionDef : case Skip : case Stop : case Pass : case Return : case DocTest : case PrintStatement :
2024-03-14 10:28:30 -07:00
code_err ( ast , " This is not a valid expression " ) ;
2024-09-08 14:17:15 -07:00
default : case Unknown : code_err ( ast , " Unknown AST " ) ;
2024-02-04 12:23:59 -08:00
}
}
2024-03-06 09:41:18 -08:00
void compile_namespace ( env_t * env , const char * ns_name , ast_t * block )
{
env_t * ns_env = namespace_env ( env , ns_name ) ;
2024-09-24 11:54:22 -07:00
CORD prefix = namespace_prefix ( ns_env , ns_env - > namespace ) ;
2024-08-17 11:41:31 -07:00
// First prepare variable initializers to prevent unitialized access:
for ( ast_list_t * stmt = block ? Match ( block , Block ) - > statements : NULL ; stmt ; stmt = stmt - > next ) {
if ( stmt - > ast - > tag = = Declare ) {
auto decl = Match ( stmt - > ast , Declare ) ;
type_t * t = get_type ( ns_env , decl - > value ) ;
if ( t - > tag = = AbortType | | t - > tag = = VoidType | | t - > tag = = ReturnType )
code_err ( stmt - > ast , " You can't declare a variable with a %T value " , t ) ;
CORD name_code = CORD_all ( prefix , Match ( decl - > var , Var ) - > name ) ;
if ( ! is_constant ( env , decl - > value ) ) {
env - > code - > variable_initializers = CORD_all (
env - > code - > variable_initializers ,
name_code , " = " , compile_maybe_incref ( ns_env , decl - > value ) , " , \n " ,
name_code , " $initialized = true; \n " ) ;
CORD checked_access = CORD_all ( " check_initialized( " , name_code , " , \" " , Match ( decl - > var , Var ) - > name , " \" ) " ) ;
set_binding ( ns_env , Match ( decl - > var , Var ) - > name , new ( binding_t , . type = t , . code = checked_access ) ) ;
}
}
}
2024-03-06 09:41:18 -08:00
for ( ast_list_t * stmt = block ? Match ( block , Block ) - > statements : NULL ; stmt ; stmt = stmt - > next ) {
ast_t * ast = stmt - > ast ;
switch ( ast - > tag ) {
2024-03-09 20:21:44 -08:00
case FunctionDef : {
2024-03-06 09:41:18 -08:00
CORD code = compile_statement ( ns_env , ast ) ;
env - > code - > funcs = CORD_cat ( env - > code - > funcs , code ) ;
break ;
2024-03-09 20:21:44 -08:00
}
2024-03-06 09:41:18 -08:00
case Declare : {
auto decl = Match ( ast , Declare ) ;
type_t * t = get_type ( ns_env , decl - > value ) ;
2024-08-20 21:48:08 -07:00
if ( t - > tag = = FunctionType )
t = Type ( ClosureType , t ) ;
2024-08-17 18:03:04 -07:00
bool is_private = ( Match ( decl - > var , Var ) - > name [ 0 ] = = ' _ ' ) ;
2024-08-17 11:41:31 -07:00
CORD name_code = CORD_all ( prefix , Match ( decl - > var , Var ) - > name ) ;
if ( ! is_constant ( env , decl - > value ) ) {
2024-08-20 21:48:08 -07:00
if ( t - > tag = = FunctionType )
t = Type ( ClosureType , t ) ;
2024-08-17 11:41:31 -07:00
env - > code - > staticdefs = CORD_all (
env - > code - > staticdefs ,
" static bool " , name_code , " $initialized = false; \n " ,
2024-08-17 18:03:04 -07:00
is_private ? " static " : CORD_EMPTY ,
2024-08-17 11:41:31 -07:00
compile_declaration ( t , name_code ) , " ; \n " ) ;
} else {
2024-08-20 21:48:08 -07:00
CORD val_code = compile_maybe_incref ( ns_env , decl - > value ) ;
if ( t - > tag = = FunctionType ) {
assert ( promote ( env , & val_code , t , Type ( ClosureType , t ) ) ) ;
t = Type ( ClosureType , t ) ;
}
2024-08-17 11:41:31 -07:00
env - > code - > staticdefs = CORD_all (
env - > code - > staticdefs ,
2024-08-17 18:03:04 -07:00
is_private ? " static " : CORD_EMPTY ,
2024-08-20 21:48:08 -07:00
compile_declaration ( t , name_code ) , " = " , val_code , " ; \n " ) ;
2024-08-17 11:41:31 -07:00
}
2024-03-06 09:41:18 -08:00
break ;
}
default : {
CORD code = compile_statement ( ns_env , ast ) ;
2024-04-12 10:09:31 -07:00
assert ( ! code ) ;
2024-03-06 09:41:18 -08:00
break ;
}
2024-06-06 13:28:53 -07:00
}
2024-03-06 09:41:18 -08:00
}
2024-06-06 13:28:53 -07:00
}
2024-02-17 15:38:29 -08:00
CORD compile_type_info ( env_t * env , type_t * t )
{
2024-09-24 11:54:22 -07:00
if ( t = = THREAD_TYPE ) return " &Thread$info " ;
else if ( t = = RANGE_TYPE ) return " &Range$info " ;
2024-11-09 13:27:54 -08:00
else if ( t = = MATCH_TYPE ) return " &Match$info " ;
2024-11-03 19:37:48 -08:00
else if ( t = = RNG_TYPE ) return " &RNG$info " ;
2024-09-24 11:54:22 -07:00
2024-02-17 15:38:29 -08:00
switch ( t - > tag ) {
2024-11-17 11:49:03 -08:00
case BoolType : case ByteType : case IntType : case BigIntType : case NumType : case CStringType : case MomentType :
2024-09-05 12:31:54 -07:00
return CORD_all ( " & " , type_to_cord ( t ) , " $info " ) ;
2024-03-21 22:52:00 -07:00
case TextType : {
auto text = Match ( t , TextType ) ;
2024-09-05 12:31:54 -07:00
if ( ! text - > lang )
return " &Text$info " ;
else if ( streq ( text - > lang , " Pattern " ) )
return " &Pattern$info " ;
2024-09-06 08:37:33 -07:00
else if ( streq ( text - > lang , " Shell " ) )
return " &Shell$info " ;
2024-09-08 21:22:12 -07:00
else if ( streq ( text - > lang , " Path " ) )
return " &Path$info " ;
2024-09-24 11:54:22 -07:00
return CORD_all ( " (& " , namespace_prefix ( text - > env , text - > env - > namespace - > parent ) , text - > lang , " ) " ) ;
2024-03-21 22:52:00 -07:00
}
case StructType : {
auto s = Match ( t , StructType ) ;
2024-09-24 11:54:22 -07:00
return CORD_all ( " (& " , namespace_prefix ( s - > env , s - > env - > namespace - > parent ) , s - > name , " ) " ) ;
2024-03-21 22:52:00 -07:00
}
case EnumType : {
auto e = Match ( t , EnumType ) ;
2024-09-24 11:54:22 -07:00
return CORD_all ( " (& " , namespace_prefix ( e - > env , e - > env - > namespace - > parent ) , e - > name , " ) " ) ;
2024-03-21 22:52:00 -07:00
}
2024-02-17 15:38:29 -08:00
case ArrayType : {
type_t * item_t = Match ( t , ArrayType ) - > item_type ;
2024-09-05 12:31:54 -07:00
return CORD_all ( " Array$info( " , compile_type_info ( env , item_t ) , " ) " ) ;
2024-08-10 12:15:38 -07:00
}
case SetType : {
type_t * item_type = Match ( t , SetType ) - > item_type ;
2024-09-05 12:31:54 -07:00
return CORD_all ( " Set$info( " , compile_type_info ( env , item_type ) , " ) " ) ;
2024-02-17 15:38:29 -08:00
}
2024-08-11 11:47:34 -07:00
case ChannelType : {
type_t * item_t = Match ( t , ChannelType ) - > item_type ;
2024-09-05 12:31:54 -07:00
return CORD_asprintf ( " Channel$info(%r) " , compile_type_info(env, item_t)) ;
2024-08-11 11:47:34 -07:00
}
2024-02-17 15:38:29 -08:00
case TableType : {
type_t * key_type = Match ( t , TableType ) - > key_type ;
type_t * value_type = Match ( t , TableType ) - > value_type ;
2024-09-05 12:31:54 -07:00
return CORD_all ( " Table$info( " , compile_type_info ( env , key_type ) , " , " , compile_type_info ( env , value_type ) , " ) " ) ;
2024-02-17 15:38:29 -08:00
}
case PointerType : {
auto ptr = Match ( t , PointerType ) ;
2024-10-27 16:54:48 -07:00
CORD sigil = ptr - > is_view ? " & " : " @ " ;
2024-09-10 22:31:31 -07:00
return CORD_asprintf ( " Pointer$info(%r, %r) " ,
2024-09-02 15:47:39 -07:00
CORD_quoted ( sigil ) ,
2024-09-10 22:31:31 -07:00
compile_type_info ( env , ptr - > pointed ) ) ;
2024-02-17 15:38:29 -08:00
}
case FunctionType : {
2024-09-05 12:31:54 -07:00
return CORD_asprintf ( " Function$info(%r) " , CORD_quoted(type_to_cord(t))) ;
2024-02-17 15:38:29 -08:00
}
case ClosureType : {
2024-09-05 12:31:54 -07:00
return CORD_asprintf ( " Closure$info(%r) " , CORD_quoted(type_to_cord(t))) ;
2024-02-17 15:38:29 -08:00
}
2024-09-10 22:31:31 -07:00
case OptionalType : {
2024-10-08 10:35:18 -07:00
type_t * non_optional = Match ( t , OptionalType ) - > type ;
return CORD_asprintf ( " Optional$info(%zu, %zu, %r) " , type_size(t), type_align(t), compile_type_info(env, non_optional)) ;
2024-09-10 22:31:31 -07:00
}
2024-09-30 11:39:30 -07:00
case TypeInfoType : return CORD_all ( " TypeInfo$info( " , CORD_quoted ( type_to_cord ( Match ( t , TypeInfoType ) - > type ) ) , " ) " ) ;
2024-09-05 12:31:54 -07:00
case MemoryType : return " &Memory$info " ;
case VoidType : return " &Void$info " ;
2024-02-26 20:10:19 -08:00
default :
compiler_err ( NULL , 0 , 0 , " I couldn't convert to a type info: %T " , t ) ;
2024-02-17 15:38:29 -08:00
}
}
2024-02-17 14:29:56 -08:00
2024-09-04 18:33:06 -07:00
static CORD get_flag_options ( type_t * t , CORD separator )
{
if ( t - > tag = = BoolType ) {
return " yes|no " ;
} else if ( t - > tag = = EnumType ) {
CORD options = CORD_EMPTY ;
for ( tag_t * tag = Match ( t , EnumType ) - > tags ; tag ; tag = tag - > next ) {
options = CORD_all ( options , tag - > name ) ;
if ( tag - > next ) options = CORD_all ( options , separator ) ;
}
return options ;
} else if ( t - > tag = = IntType | | t - > tag = = NumType | | t - > tag = = BigIntType ) {
return " N " ;
} else {
return " ... " ;
}
}
2024-04-20 11:55:27 -07:00
CORD compile_cli_arg_call ( env_t * env , CORD fn_name , type_t * fn_type )
2024-04-12 10:09:31 -07:00
{
2024-04-20 11:55:27 -07:00
auto fn_info = Match ( fn_type , FunctionType ) ;
2024-09-05 11:38:37 -07:00
2024-09-27 10:56:56 -07:00
env_t * main_env = fresh_scope ( env ) ;
CORD code = CORD_EMPTY ;
2024-09-06 21:26:30 -07:00
binding_t * usage_binding = get_binding ( env , " _USAGE " ) ;
2024-09-05 11:38:37 -07:00
CORD usage_code = usage_binding ? usage_binding - > code : " usage " ;
2024-09-06 21:26:30 -07:00
binding_t * help_binding = get_binding ( env , " _HELP " ) ;
2024-09-05 11:38:37 -07:00
CORD help_code = help_binding ? help_binding - > code : usage_code ;
2024-09-27 10:56:56 -07:00
if ( ! usage_binding ) {
bool explicit_help_flag = false ;
for ( arg_t * arg = fn_info - > args ; arg ; arg = arg - > next ) {
if ( streq ( arg - > name , " help " ) ) {
explicit_help_flag = true ;
break ;
}
2024-04-12 10:09:31 -07:00
}
2024-09-04 11:34:24 -07:00
2024-09-05 11:38:37 -07:00
CORD usage = explicit_help_flag ? CORD_EMPTY : " [--help] " ;
for ( arg_t * arg = fn_info - > args ; arg ; arg = arg - > next ) {
usage = CORD_cat ( usage , " " ) ;
type_t * t = get_arg_type ( main_env , arg ) ;
CORD flag = CORD_replace ( arg - > name , " _ " , " - " ) ;
2024-09-27 10:56:56 -07:00
if ( arg - > default_val | | arg - > type - > tag = = OptionalType ) {
if ( strlen ( arg - > name ) = = 1 ) {
if ( t - > tag = = BoolType | | ( t - > tag = = OptionalType & & Match ( t , OptionalType ) - > type - > tag = = BoolType ) )
usage = CORD_all ( usage , " [- " , flag , " ] " ) ;
else
usage = CORD_all ( usage , " [- " , flag , " " , get_flag_options ( t , " | " ) , " ] " ) ;
} else {
if ( t - > tag = = BoolType | | ( t - > tag = = OptionalType & & Match ( t , OptionalType ) - > type - > tag = = BoolType ) )
usage = CORD_all ( usage , " [-- " , flag , " ] " ) ;
else
usage = CORD_all ( usage , " [-- " , flag , " = " , get_flag_options ( t , " | " ) , " ] " ) ;
}
2024-09-05 11:38:37 -07:00
} else {
if ( t - > tag = = BoolType )
2024-09-27 10:56:56 -07:00
usage = CORD_all ( usage , " <-- " , flag , " |--no- " , flag , " > " ) ;
2024-09-05 11:38:37 -07:00
else if ( t - > tag = = EnumType )
usage = CORD_all ( usage , get_flag_options ( t , " | " ) ) ;
else if ( t - > tag = = ArrayType )
usage = CORD_all ( usage , " < " , flag , " ...> " ) ;
else
usage = CORD_all ( usage , " < " , flag , " > " ) ;
}
}
2024-09-04 12:23:23 -07:00
code = CORD_all ( code , " Text_t usage = Texts(Text( \" Usage: \" ), Text$from_str(argv[0]) " ,
usage = = CORD_EMPTY ? CORD_EMPTY : CORD_all ( " , Text( " , CORD_quoted ( usage ) , " ) " ) , " ); \n " ) ;
2024-09-05 11:38:37 -07:00
}
2024-04-12 10:09:31 -07:00
2024-09-04 11:34:24 -07:00
2024-09-27 10:56:56 -07:00
int num_args = 0 ;
2024-04-12 10:09:31 -07:00
for ( arg_t * arg = fn_info - > args ; arg ; arg = arg - > next ) {
2024-09-27 10:56:56 -07:00
type_t * opt_type = arg - > type - > tag = = OptionalType ? arg - > type : Type ( OptionalType , . type = arg - > type ) ;
code = CORD_all ( code , compile_declaration ( opt_type , CORD_all ( " $ " , arg - > name ) ) ) ;
if ( arg - > default_val ) {
CORD default_val = compile ( env , arg - > default_val ) ;
if ( arg - > type - > tag ! = OptionalType )
default_val = promote_to_optional ( arg - > type , default_val ) ;
code = CORD_all ( code , " = " , default_val ) ;
} else {
code = CORD_all ( code , " = " , compile_null ( arg - > type ) ) ;
2024-04-12 10:09:31 -07:00
}
2024-09-27 10:56:56 -07:00
code = CORD_all ( code , " ; \n " ) ;
num_args + = 1 ;
2024-04-12 10:09:31 -07:00
}
2024-09-28 12:17:09 -07:00
code = CORD_all ( code , " tomo_parse_args(argc, argv, " , usage_code , " , " , help_code ) ;
2024-04-12 10:09:31 -07:00
for ( arg_t * arg = fn_info - > args ; arg ; arg = arg - > next ) {
2024-09-28 12:17:09 -07:00
code = CORD_all ( code , " , \n { " , CORD_quoted ( CORD_replace ( arg - > name , " _ " , " - " ) ) , " , " ,
2024-09-27 10:56:56 -07:00
( arg - > default_val | | arg - > type - > tag = = OptionalType ) ? " false " : " true " , " , " ,
compile_type_info ( env , arg - > type ) ,
2024-09-28 12:17:09 -07:00
" , & " , CORD_all ( " $ " , arg - > name ) , " } " ) ;
2024-04-12 10:09:31 -07:00
}
2024-09-28 12:17:09 -07:00
code = CORD_all ( code , " ); \n " ) ;
2024-04-12 10:09:31 -07:00
2024-04-20 11:55:27 -07:00
code = CORD_all ( code , fn_name , " ( " ) ;
2024-04-12 10:09:31 -07:00
for ( arg_t * arg = fn_info - > args ; arg ; arg = arg - > next ) {
2024-09-11 21:13:53 -07:00
CORD arg_code = CORD_all ( " $ " , arg - > name ) ;
if ( arg - > type - > tag ! = OptionalType )
arg_code = optional_into_nonnull ( arg - > type , arg_code ) ;
code = CORD_all ( code , arg_code ) ;
2024-04-12 10:09:31 -07:00
if ( arg - > next ) code = CORD_all ( code , " , " ) ;
}
2024-04-20 11:55:27 -07:00
code = CORD_all ( code , " ); \n " ) ;
2024-04-12 10:09:31 -07:00
return code ;
}
2024-06-06 13:28:53 -07:00
CORD compile_file ( env_t * env , ast_t * ast )
2024-02-17 23:17:44 -08:00
{
2024-08-17 11:41:31 -07:00
// First prepare variable initializers to prevent unitialized access:
2024-04-10 08:49:43 -07:00
for ( ast_list_t * stmt = Match ( ast , Block ) - > statements ; stmt ; stmt = stmt - > next ) {
if ( stmt - > ast - > tag = = Declare ) {
auto decl = Match ( stmt - > ast , Declare ) ;
2024-04-17 10:44:01 -07:00
const char * decl_name = Match ( decl - > var , Var ) - > name ;
2024-09-24 11:54:22 -07:00
CORD full_name = CORD_all ( namespace_prefix ( env , env - > namespace ) , decl_name ) ;
2024-04-10 08:49:43 -07:00
type_t * t = get_type ( env , decl - > value ) ;
2024-07-13 15:43:50 -07:00
if ( t - > tag = = AbortType | | t - > tag = = VoidType | | t - > tag = = ReturnType )
2024-04-10 08:49:43 -07:00
code_err ( stmt - > ast , " You can't declare a variable with a %T value " , t ) ;
2024-09-27 11:56:24 -07:00
if ( ! is_constant ( env , decl - > value ) ) {
2024-08-20 21:48:08 -07:00
CORD val_code = compile_maybe_incref ( env , decl - > value ) ;
if ( t - > tag = = FunctionType ) {
assert ( promote ( env , & val_code , t , Type ( ClosureType , t ) ) ) ;
t = Type ( ClosureType , t ) ;
}
2024-08-17 11:41:31 -07:00
env - > code - > variable_initializers = CORD_all (
env - > code - > variable_initializers ,
2024-08-20 21:48:08 -07:00
full_name , " = " , val_code , " , \n " ,
2024-08-17 11:41:31 -07:00
full_name , " $initialized = true; \n " ) ;
CORD checked_access = CORD_all ( " check_initialized( " , full_name , " , \" " , decl_name , " \" ) " ) ;
set_binding ( env , decl_name , new ( binding_t , . type = t , . code = checked_access ) ) ;
}
}
}
2024-04-17 10:44:01 -07:00
2024-08-17 11:41:31 -07:00
for ( ast_list_t * stmt = Match ( ast , Block ) - > statements ; stmt ; stmt = stmt - > next ) {
if ( stmt - > ast - > tag = = Declare ) {
auto decl = Match ( stmt - > ast , Declare ) ;
const char * decl_name = Match ( decl - > var , Var ) - > name ;
2024-09-24 11:54:22 -07:00
CORD full_name = CORD_all ( namespace_prefix ( env , env - > namespace ) , decl_name ) ;
2024-08-17 11:41:31 -07:00
bool is_private = ( decl_name [ 0 ] = = ' _ ' ) ;
type_t * t = get_type ( env , decl - > value ) ;
2024-09-27 11:56:24 -07:00
if ( ! is_constant ( env , decl - > value ) ) {
2024-04-17 10:44:01 -07:00
env - > code - > staticdefs = CORD_all (
env - > code - > staticdefs ,
2024-08-17 11:41:31 -07:00
" static bool " , full_name , " $initialized = false; \n " ,
is_private ? " static " : CORD_EMPTY ,
compile_declaration ( t , full_name ) , " ; \n " ) ;
2024-04-17 10:44:01 -07:00
} else {
2024-08-20 21:48:08 -07:00
CORD val_code = compile_maybe_incref ( env , decl - > value ) ;
if ( t - > tag = = FunctionType ) {
assert ( promote ( env , & val_code , t , Type ( ClosureType , t ) ) ) ;
t = Type ( ClosureType , t ) ;
}
2024-04-17 10:44:01 -07:00
env - > code - > staticdefs = CORD_all (
env - > code - > staticdefs ,
2024-08-17 11:41:31 -07:00
is_private ? " static " : CORD_EMPTY ,
2024-08-20 21:48:08 -07:00
compile_declaration ( t , full_name ) , " = " , val_code , " ; \n " ) ;
2024-04-17 10:44:01 -07:00
}
2024-05-18 17:53:03 -07:00
} else if ( stmt - > ast - > tag = = InlineCCode ) {
CORD code = compile_statement ( env , stmt - > ast ) ;
2024-05-28 16:02:00 -07:00
env - > code - > staticdefs = CORD_all ( env - > code - > staticdefs , code , " \n " ) ;
2024-09-18 12:39:22 -07:00
} else if ( stmt - > ast - > tag = = Use ) {
CORD code = compile_statement ( env , stmt - > ast ) ;
if ( code )
env - > code - > staticdefs = CORD_all ( env - > code - > staticdefs , code ) ;
2024-04-10 08:49:43 -07:00
} else {
CORD code = compile_statement ( env , stmt - > ast ) ;
2024-04-12 10:09:31 -07:00
assert ( ! code ) ;
2024-04-10 08:49:43 -07:00
}
2024-02-17 23:17:44 -08:00
}
2024-06-06 13:28:53 -07:00
const char * name = file_base_name ( ast - > file - > filename ) ;
return CORD_all (
2024-09-03 20:32:02 -07:00
// "#line 1 ", CORD_quoted(ast->file->filename), "\n",
" #define __SOURCE_FILE__ " , CORD_quoted ( ast - > file - > filename ) , " \n " ,
2024-06-06 13:28:53 -07:00
" #include <tomo/tomo.h> \n "
" #include \" " , name , " .tm.h \" \n \n " ,
env - > code - > local_typedefs , " \n " ,
2024-10-08 10:35:18 -07:00
env - > code - > typeinfos , " \n " ,
2024-06-06 13:28:53 -07:00
env - > code - > staticdefs , " \n " ,
2024-09-12 20:41:32 -07:00
env - > code - > funcs , " \n " ,
2024-09-24 11:54:22 -07:00
" public void $ " , env - > namespace - > name , " $$initialize(void) { \n " ,
2024-08-18 21:52:35 -07:00
" static bool initialized = false; \n " ,
" if (initialized) return; \n " ,
" initialized = true; \n " ,
2024-08-17 11:41:31 -07:00
env - > code - > variable_initializers ,
2024-09-12 20:41:32 -07:00
env - > code - > function_naming ,
" } \n " ) ;
2024-06-06 13:28:53 -07:00
}
2024-10-30 11:02:34 -07:00
CORD compile_statement_type_header ( env_t * env , ast_t * ast )
2024-06-06 13:28:53 -07:00
{
switch ( ast - > tag ) {
2024-07-23 16:46:42 -07:00
case Use : {
2024-08-22 11:02:48 -07:00
auto use = Match ( ast , Use ) ;
switch ( use - > what ) {
2024-09-15 18:18:42 -07:00
case USE_MODULE : {
2024-09-22 12:07:44 -07:00
return CORD_all ( " #include < " , use - > path , " / " , use - > path , " .h> \n " ) ;
2024-09-15 18:18:42 -07:00
}
2024-08-22 11:02:48 -07:00
case USE_LOCAL :
2024-09-04 10:48:26 -07:00
return CORD_all ( " #include \" " , use - > path , " .h \" \n " ) ;
2024-08-22 11:02:48 -07:00
case USE_HEADER :
2024-09-18 12:39:22 -07:00
if ( use - > path [ 0 ] = = ' < ' )
return CORD_all ( " #include " , use - > path , " \n " ) ;
else
return CORD_all ( " #include \" " , use - > path , " \" \n " ) ;
2024-08-22 11:02:48 -07:00
default :
2024-07-23 16:46:42 -07:00
return CORD_EMPTY ;
2024-08-22 11:02:48 -07:00
}
2024-07-23 16:46:42 -07:00
}
case StructDef : {
2024-09-17 22:01:15 -07:00
return compile_struct_header ( env , ast ) ;
2024-07-23 16:46:42 -07:00
}
case EnumDef : {
2024-09-17 21:58:41 -07:00
return compile_enum_header ( env , ast ) ;
2024-07-23 16:46:42 -07:00
}
case LangDef : {
auto def = Match ( ast , LangDef ) ;
2024-09-24 11:54:22 -07:00
CORD full_name = CORD_cat ( namespace_prefix ( env , env - > namespace ) , def - > name ) ;
2024-09-03 21:18:03 -07:00
return CORD_all (
2024-09-24 11:54:22 -07:00
" typedef Text_t " , namespace_prefix ( env , env - > namespace ) , def - > name , " _t; \n "
2024-09-03 21:18:03 -07:00
// Constructor macro:
2024-09-24 11:54:22 -07:00
" #define " , namespace_prefix ( env , env - > namespace ) , def - > name ,
" (text) (( " , namespace_prefix ( env , env - > namespace ) , def - > name , " _t){.length=sizeof(text)-1, .tag=TEXT_ASCII, .ascii= \" \" text}) \n "
" #define " , namespace_prefix ( env , env - > namespace ) , def - > name ,
" s(...) (( " , namespace_prefix ( env , env - > namespace ) , def - > name , " _t)Texts(__VA_ARGS__)) \n "
2024-10-30 11:02:34 -07:00
" extern const TypeInfo_t " , full_name , " ; \n "
2024-09-17 13:20:30 -07:00
) ;
2024-06-06 13:28:53 -07:00
}
default :
return CORD_EMPTY ;
}
}
2024-10-30 11:02:34 -07:00
CORD compile_statement_namespace_header ( env_t * env , ast_t * ast )
{
const char * ns_name = NULL ;
ast_t * block = NULL ;
switch ( ast - > tag ) {
case LangDef : {
auto def = Match ( ast , LangDef ) ;
ns_name = def - > name ;
block = def - > namespace ;
break ;
}
case StructDef : {
auto def = Match ( ast , StructDef ) ;
ns_name = def - > name ;
block = def - > namespace ;
break ;
}
case EnumDef : {
auto def = Match ( ast , EnumDef ) ;
ns_name = def - > name ;
block = def - > namespace ;
break ;
}
2024-10-30 11:28:20 -07:00
case Extern : {
auto ext = Match ( ast , Extern ) ;
type_t * t = parse_type_ast ( env , ext - > type ) ;
CORD decl ;
if ( t - > tag = = ClosureType ) {
t = Match ( t , ClosureType ) - > fn ;
auto fn = Match ( t , FunctionType ) ;
decl = CORD_all ( compile_type ( fn - > ret ) , " " , ext - > name , " ( " ) ;
for ( arg_t * arg = fn - > args ; arg ; arg = arg - > next ) {
decl = CORD_all ( decl , compile_type ( arg - > type ) ) ;
if ( arg - > next ) decl = CORD_cat ( decl , " , " ) ;
}
decl = CORD_cat ( decl , " ) " ) ;
} else {
decl = compile_declaration ( t , ext - > name ) ;
}
return CORD_all ( " extern " , decl , " ; \n " ) ;
}
case Declare : {
auto decl = Match ( ast , Declare ) ;
const char * decl_name = Match ( decl - > var , Var ) - > name ;
bool is_private = ( decl_name [ 0 ] = = ' _ ' ) ;
if ( is_private )
return CORD_EMPTY ;
type_t * t = get_type ( env , decl - > value ) ;
if ( t - > tag = = FunctionType )
t = Type ( ClosureType , t ) ;
assert ( t - > tag ! = ModuleType ) ;
if ( t - > tag = = AbortType | | t - > tag = = VoidType | | t - > tag = = ReturnType )
code_err ( ast , " You can't declare a variable with a %T value " , t ) ;
return CORD_all (
compile_statement_type_header ( env , decl - > value ) ,
" extern " , compile_declaration ( t , CORD_cat ( namespace_prefix ( env , env - > namespace ) , decl_name ) ) , " ; \n " ) ;
}
2024-10-30 11:27:02 -07:00
case FunctionDef : {
auto fndef = Match ( ast , FunctionDef ) ;
const char * decl_name = Match ( fndef - > name , Var ) - > name ;
bool is_private = decl_name [ 0 ] = = ' _ ' ;
if ( is_private ) return CORD_EMPTY ;
CORD arg_signature = " ( " ;
for ( arg_ast_t * arg = fndef - > args ; arg ; arg = arg - > next ) {
type_t * arg_type = get_arg_ast_type ( env , arg ) ;
arg_signature = CORD_cat ( arg_signature , compile_declaration ( arg_type , CORD_cat ( " $ " , arg - > name ) ) ) ;
if ( arg - > next ) arg_signature = CORD_cat ( arg_signature , " , " ) ;
}
arg_signature = CORD_cat ( arg_signature , " ) " ) ;
type_t * ret_t = fndef - > ret_type ? parse_type_ast ( env , fndef - > ret_type ) : Type ( VoidType ) ;
CORD ret_type_code = compile_type ( ret_t ) ;
return CORD_all ( ret_type_code , " " , namespace_prefix ( env , env - > namespace ) , decl_name , arg_signature , " ; \ n " );
}
2024-10-30 11:02:34 -07:00
default : return CORD_EMPTY ;
}
env_t * ns_env = namespace_env ( env , ns_name ) ;
CORD header = CORD_EMPTY ;
for ( ast_list_t * stmt = block ? Match ( block , Block ) - > statements : NULL ; stmt ; stmt = stmt - > next ) {
header = CORD_all ( header , compile_statement_namespace_header ( ns_env , stmt - > ast ) ) ;
}
return header ;
}
2024-09-17 12:17:13 -07:00
typedef struct {
env_t * env ;
CORD * header ;
} compile_typedef_info_t ;
2024-09-17 13:20:30 -07:00
static void _visit_statement ( compile_typedef_info_t * info , ast_t * ast )
2024-09-17 12:17:13 -07:00
{
2024-10-30 11:02:34 -07:00
* info - > header = CORD_all ( * info - > header ,
compile_statement_type_header ( info - > env , ast ) ,
compile_statement_namespace_header ( info - > env , ast ) ) ;
2024-09-17 12:17:13 -07:00
}
2024-09-17 13:20:30 -07:00
CORD compile_file_header ( env_t * env , ast_t * ast )
2024-06-06 13:28:53 -07:00
{
2024-09-03 20:32:02 -07:00
CORD header = CORD_all (
" #pragma once \n "
// "#line 1 ", CORD_quoted(ast->file->filename), "\n",
" #include <tomo/tomo.h> \n " ) ;
2024-06-06 13:28:53 -07:00
2024-09-17 12:17:13 -07:00
compile_typedef_info_t info = { . env = env , . header = & header } ;
2024-09-17 13:20:30 -07:00
visit_topologically ( Match ( ast , Block ) - > statements , ( Closure_t ) { . fn = ( void * ) _visit_statement , & info } ) ;
2024-06-06 13:28:53 -07:00
2024-09-24 11:54:22 -07:00
header = CORD_all ( header , " void $ " , env - > namespace - > name , " $$initialize(void); \n " ) ;
2024-06-06 13:28:53 -07:00
return header ;
2024-02-17 23:17:44 -08:00
}
2024-02-04 12:23:59 -08:00
// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1,\:0