/* -*- mode: objc; coding: utf-8 -*- */ /* Toilet Lisp, a Common Lisp subset for the Étoilé runtime. * Copyright (C) 2008 Matthias Andreas Benkard. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or (at * your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #import "MLKCompiledClosure.h" #import "functions.h" #import "globals.h" #import "runtime-compatibility.h" #import "util.h" #import #import #ifdef __OBJC_GC__ #import #endif #import @implementation MLKCompiledClosure -(id) initWithCode:(void *)code data:(id *)data length:(intptr_t)dataLength { int i; _dataLength = dataLength; _code = code; #ifdef __OBJC_GC__ _data = NSAllocateCollectable (dataLength * sizeof(id), NSScannedOption); #else _data = malloc (dataLength * sizeof(id)); #endif for (i = 0; i < _dataLength; i++) { _data[i] = LRETAIN (data[i]); } return self; } +(id) closureWithCode:(void *)code data:(id *)data length:(intptr_t)dataLength { return LAUTORELEASE ([[self alloc] initWithCode:code data:data length:dataLength]); } -(NSArray *) applyToArray:(NSArray *)arguments { int argc = ([arguments count] + 2); ffi_cif cif; ffi_type *arg_types[argc]; ffi_status status; void *argv[argc]; id argpointers[argc - 1]; ffi_arg return_value; int i; arg_types[0] = &ffi_type_pointer; argv[0] = &_data; for (i = 1; i < argc - 1; i++) { arg_types[i] = &ffi_type_pointer; argpointers[i-1] = denullify([arguments objectAtIndex:(i-1)]); argv[i] = &argpointers[i-1]; } arg_types[argc - 1] = &ffi_type_pointer; argv[argc - 1] = &MLKEndOfArgumentsMarker; status = ffi_prep_cif (&cif, FFI_DEFAULT_ABI, argc, &ffi_type_pointer, arg_types); if (status != FFI_OK) { [NSException raise:@"MLKInvalidFFITypeException" format:@"FFI type is invalid (this is probably a bug)."]; } // NSLog (@"Calling %p (argc = %d)", _code, argc); // for (i = 0; i < argc; i++) // { // NSLog (@"Argument %d: %p", i, *((void**)argv[i])); // } ffi_call (&cif, FFI_FN (_code), &return_value, (void**)argv); // return_value = ((id (*)(void *, ...))_code) (_data, argpointers[0], argpointers[1], MLKEndOfArgumentsMarker); // FIXME: multiple values return [NSArray arrayWithObject:nullify((id)return_value)]; } -(NSString *) description { return MLKPrintToString (self); } -(NSString *) descriptionForLisp { return [NSString stringWithFormat:@"", self]; } -(id (*)()) code { return _code; } -(void *) closureData { return _data; } -(void) dealloc { int i; [super dealloc]; // FIXME: Decrease refcount of _code. for (i = 0; i < _dataLength; i++) { LRELEASE (_data[i]); } free (_data); } @end