139 lines
3.8 KiB
C
139 lines
3.8 KiB
C
|
/* go-convert-interface.c -- convert interfaces for Go.
|
||
|
|
||
|
Copyright 2009 The Go Authors. All rights reserved.
|
||
|
Use of this source code is governed by a BSD-style
|
||
|
license that can be found in the LICENSE file. */
|
||
|
|
||
|
#include "go-alloc.h"
|
||
|
#include "go-assert.h"
|
||
|
#include "go-panic.h"
|
||
|
#include "interface.h"
|
||
|
|
||
|
/* This is called when converting one interface type into another
|
||
|
interface type. LHS_DESCRIPTOR is the type descriptor of the
|
||
|
resulting interface. RHS_DESCRIPTOR is the type descriptor of the
|
||
|
object being converted. This builds and returns a new interface
|
||
|
method table. If any method in the LHS_DESCRIPTOR interface is not
|
||
|
implemented by the object, the conversion fails. If the conversion
|
||
|
fails, then if MAY_FAIL is true this returns NULL; otherwise, it
|
||
|
panics. */
|
||
|
|
||
|
void *
|
||
|
__go_convert_interface_2 (const struct __go_type_descriptor *lhs_descriptor,
|
||
|
const struct __go_type_descriptor *rhs_descriptor,
|
||
|
_Bool may_fail)
|
||
|
{
|
||
|
const struct __go_interface_type *lhs_interface;
|
||
|
int lhs_method_count;
|
||
|
const struct __go_interface_method* lhs_methods;
|
||
|
const void **methods;
|
||
|
const struct __go_uncommon_type *rhs_uncommon;
|
||
|
int rhs_method_count;
|
||
|
const struct __go_method *p_rhs_method;
|
||
|
int i;
|
||
|
|
||
|
if (rhs_descriptor == NULL)
|
||
|
{
|
||
|
/* A nil value always converts to nil. */
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
__go_assert (lhs_descriptor->__code == GO_INTERFACE);
|
||
|
lhs_interface = (const struct __go_interface_type *) lhs_descriptor;
|
||
|
lhs_method_count = lhs_interface->__methods.__count;
|
||
|
lhs_methods = ((const struct __go_interface_method *)
|
||
|
lhs_interface->__methods.__values);
|
||
|
|
||
|
/* This should not be called for an empty interface. */
|
||
|
__go_assert (lhs_method_count > 0);
|
||
|
|
||
|
rhs_uncommon = rhs_descriptor->__uncommon;
|
||
|
if (rhs_uncommon == NULL || rhs_uncommon->__methods.__count == 0)
|
||
|
{
|
||
|
struct __go_empty_interface panic_arg;
|
||
|
|
||
|
if (may_fail)
|
||
|
return NULL;
|
||
|
|
||
|
newTypeAssertionError (NULL,
|
||
|
rhs_descriptor,
|
||
|
lhs_descriptor,
|
||
|
NULL,
|
||
|
rhs_descriptor->__reflection,
|
||
|
lhs_descriptor->__reflection,
|
||
|
lhs_methods[0].__name,
|
||
|
&panic_arg);
|
||
|
__go_panic (panic_arg);
|
||
|
}
|
||
|
|
||
|
rhs_method_count = rhs_uncommon->__methods.__count;
|
||
|
p_rhs_method = ((const struct __go_method *)
|
||
|
rhs_uncommon->__methods.__values);
|
||
|
|
||
|
methods = NULL;
|
||
|
|
||
|
for (i = 0; i < lhs_method_count; ++i)
|
||
|
{
|
||
|
const struct __go_interface_method *p_lhs_method;
|
||
|
|
||
|
p_lhs_method = &lhs_methods[i];
|
||
|
|
||
|
while (rhs_method_count > 0
|
||
|
&& (!__go_ptr_strings_equal (p_lhs_method->__name,
|
||
|
p_rhs_method->__name)
|
||
|
|| !__go_ptr_strings_equal (p_lhs_method->__pkg_path,
|
||
|
p_rhs_method->__pkg_path)))
|
||
|
{
|
||
|
++p_rhs_method;
|
||
|
--rhs_method_count;
|
||
|
}
|
||
|
|
||
|
if (rhs_method_count == 0
|
||
|
|| !__go_type_descriptors_equal (p_lhs_method->__type,
|
||
|
p_rhs_method->__mtype))
|
||
|
{
|
||
|
struct __go_empty_interface panic_arg;
|
||
|
|
||
|
if (methods != NULL)
|
||
|
__go_free (methods);
|
||
|
|
||
|
if (may_fail)
|
||
|
return NULL;
|
||
|
|
||
|
newTypeAssertionError (NULL,
|
||
|
rhs_descriptor,
|
||
|
lhs_descriptor,
|
||
|
NULL,
|
||
|
rhs_descriptor->__reflection,
|
||
|
lhs_descriptor->__reflection,
|
||
|
p_lhs_method->__name,
|
||
|
&panic_arg);
|
||
|
__go_panic (panic_arg);
|
||
|
}
|
||
|
|
||
|
if (methods == NULL)
|
||
|
{
|
||
|
methods = (const void **) __go_alloc ((lhs_method_count + 1)
|
||
|
* sizeof (void *));
|
||
|
|
||
|
/* The first field in the method table is always the type of
|
||
|
the object. */
|
||
|
methods[0] = rhs_descriptor;
|
||
|
}
|
||
|
|
||
|
methods[i + 1] = p_rhs_method->__function;
|
||
|
}
|
||
|
|
||
|
return methods;
|
||
|
}
|
||
|
|
||
|
/* This is called by the compiler to convert a value from one
|
||
|
interface type to another. */
|
||
|
|
||
|
void *
|
||
|
__go_convert_interface (const struct __go_type_descriptor *lhs_descriptor,
|
||
|
const struct __go_type_descriptor *rhs_descriptor)
|
||
|
{
|
||
|
return __go_convert_interface_2 (lhs_descriptor, rhs_descriptor, 0);
|
||
|
}
|