package Parrot::Pmc2c::PMC::Object; # Copyright (C) 2007-2010, Parrot Foundation. use base 'Parrot::Pmc2c::PMC'; use strict; use warnings; =head1 NAME Parrot::Pmc2c::PMC::Object =head1 DESCRIPTION PMC to C Methods =head1 METHODS =head2 C Returns the C code for the method body. Overrides the default implementation to direct all unknown methods to first check if there is an implementation of the vtable function in the vtable function hash of this class of any others, and delegates up to any PMCs in the MRO. =cut sub pre_method_gen { my ($self) = @_; # vtables foreach my $method ( @{ $self->vtable->methods } ) { my $vt_method_name = $method->name; next unless $self->normal_unimplemented_vtable($vt_method_name); my $new_default_method = $method->clone( { parent_name => $self->name, type => Parrot::Pmc2c::Method::VTABLE, } ); my ( $return_prefix, $ret_suffix, $args, $sig, $return_type_char, $null_return ) = $new_default_method->signature; my ( $pcc_sig, $pcc_args, $pcc_result_decl, $pcc_return_stmt ) = $new_default_method->pcc_signature; my $void_return = $return_type_char eq 'v' ? 'return;' : ''; my $return = $return_type_char eq 'v' ? '' : $return_prefix; my $superargs = $args; $superargs =~ s/^,//; my $method_body_text = <<"EOC"; Parrot_Object_attributes * const obj = PARROT_OBJECT(_self); Parrot_Class_attributes * const _class = PARROT_CLASS(obj->_class); STRING * const meth_name = CONST_STRING_GEN(interp, "$vt_method_name"); EOC # Walk and search for the vtable. $method_body_text .= <<"EOC"; const int num_classes = VTABLE_elements(interp, _class->all_parents); int i; for (i = 0; i < num_classes; i++) { EOC # Get the class. $method_body_text .= <<"EOC"; PMC * const cur_class = VTABLE_get_pmc_keyed_int(interp, _class->all_parents, i); PMC * const meth = Parrot_oo_find_vtable_override_for_class(interp, cur_class, meth_name); if (!PMC_IS_NULL(meth)) { EOC $method_body_text .= " $pcc_result_decl\n" if $pcc_result_decl ne ''; # Dereference any Key contents $method_body_text .= " key = clone_key_arg(interp, key);\n" if $vt_method_name =~ /_keyed$/; $method_body_text .= <<"EOC"; Parrot_ext_call(interp, meth, "Pi$pcc_sig", _self$pcc_args); $pcc_return_stmt } EOC # Multiply dispatched math opcodes shouldn't be invoked on a proxy object. unless ($self->vtable_method_does_multi($vt_method_name)) { $method_body_text .= <<"EOC"; if (cur_class->vtable->base_type == enum_class_PMCProxy) { EOC # Get the PMC instance and call the vtable on that. $method_body_text .= <<"EOC"; STRING * const proxy = CONST_STRING_GEN(interp, "proxy"); PMC * const del_object = VTABLE_get_attr_str(interp, SELF, proxy); if (!PMC_IS_NULL(del_object)) { ${return}VTABLE_$vt_method_name(interp, del_object$args); $void_return } } EOC } $method_body_text .= <<"EOC"; } ${return}SUPER($superargs); $void_return EOC $new_default_method->body( Parrot::Pmc2c::Emitter->text($method_body_text) ); $self->add_method($new_default_method); } return 1; } 1; # Local Variables: # mode: cperl # cperl-indent-level: 4 # fill-column: 100 # End: # vim: expandtab shiftwidth=4: