libs-base/Source/callframe.m
Richard Frith-MacDonald ddffb6eec3 include malloc.h for alloc() on platforms where it isn't in alloca.h
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@29971 72102866-910b-0410-8b05-ffd578937521
2010-03-16 06:11:00 +00:00

181 lines
4.5 KiB
Objective-C

/** callframe.m - Wrapper/Objective-C interface for ffcall function interface
Copyright (C) 2000, Free Software Foundation, Inc.
Written by: Adam Fedor <fedor@gnu.org>
Created: Nov 2000
This file is part of the GNUstep Base Library.
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library 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
Library General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free
Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02111 USA.
*/
#import "common.h"
#ifdef HAVE_MALLOC_H
#include <malloc.h>
#endif
#ifdef HAVE_ALLOCA_H
#include <alloca.h>
#endif
#include "callframe.h"
#import "Foundation/NSException.h"
#import "Foundation/NSData.h"
#import "GSInvocation.h"
#if defined(ALPHA) || (defined(MIPS) && (_MIPS_SIM == _ABIN32))
typedef long long smallret_t;
#else
typedef int smallret_t;
#endif
callframe_t *
callframe_from_signature (NSMethodSignature *info, void **retval)
{
unsigned size = sizeof(callframe_t);
unsigned align = __alignof(double);
unsigned offset = 0;
unsigned numargs = [info numberOfArguments];
void *buf;
int i;
callframe_t *cframe;
if (numargs > 0)
{
if (size % align != 0)
{
size += align - (size % align);
}
offset = size;
size += numargs * sizeof(void*);
if (size % align != 0)
{
size += (align - (size % align));
}
for (i = 0; i < numargs; i++)
{
const char *type = [info getArgumentTypeAtIndex: i];
type = objc_skip_type_qualifiers (type);
size += objc_sizeof_type (type);
if (size % align != 0)
{
size += (align - size % align);
}
}
}
/*
* If we need space allocated to store a return value,
* make room for it at the end of the callframe so we
* only need to do a single malloc.
*/
if (retval)
{
const char *type;
unsigned full = size;
unsigned pos;
unsigned ret;
type = [info methodReturnType];
type = objc_skip_type_qualifiers (type);
if (full % align != 0)
{
full += (align - full % align);
}
if (full % 8 != 0)
{
full += (8 - full % 8);
}
pos = full;
ret = MAX(objc_sizeof_type (type), sizeof(double));
/* The addition of a constant '8' is a fudge applied simply because
* some return values write beynd the end of the memory if the buffer
* is sized exactly ... don't know why.
*/
full += ret + 8;
#if GS_WITH_GC
cframe = buf = NSAllocateCollectable(full, NSScannedOption);
#else
cframe = buf = NSZoneCalloc(NSDefaultMallocZone(), full, 1);
#endif
if (cframe)
{
*retval = buf + pos;
}
}
else
{
#if GS_WITH_GC
cframe = buf = NSAllocateCollectable(size, NSScannedOption);
#else
cframe = buf = NSZoneCalloc(NSDefaultMallocZone(), size, 1);
#endif
}
if (cframe)
{
cframe->nargs = numargs;
cframe->args = buf + offset;
offset += numargs * sizeof(void*);
if (offset % align != 0)
{
offset += align - (offset % align);
}
for (i = 0; i < cframe->nargs; i++)
{
const char *type = [info getArgumentTypeAtIndex: i];
cframe->args[i] = buf + offset;
type = objc_skip_type_qualifiers (type);
offset += objc_sizeof_type (type);
if (offset % align != 0)
{
offset += (align - offset % align);
}
}
}
return cframe;
}
void
callframe_set_arg(callframe_t *cframe, int index, void *buffer, int size)
{
if (index < 0 || index >= cframe->nargs)
return;
memcpy(cframe->args[index], buffer, size);
}
void
callframe_get_arg(callframe_t *cframe, int index, void *buffer, int size)
{
if (index < 0 || index >= cframe->nargs)
return;
memcpy(buffer, cframe->args[index], size);
}
void *
callframe_arg_addr(callframe_t *cframe, int index)
{
if (index < 0 || index >= cframe->nargs)
return NULL;
return cframe->args[index];
}