NAME
    XS::Object::Magic - Opaque, extensible XS pointer backed objects using
    "sv_magic"

SYNOPSIS
            package MyObject;

            use XS::Object::Magic;

            sub new {
                    my $class = shift;

                    # create any object representation you like
                    my $self = bless {}, $class;

                    $self->build_struct;

                    return $self;
            }


            # or using Moose

            package MyObject;
            use Moose;

            sub BUILD {
                    shift->build_struct;
            }


            # then in XS

            MODULE = MyObject  PACKAGE = MyObject

            void build_struct (SV *self)
                    PREINIT:
                            my_struct_t *thingy;
                    CODE:
                            thingy = create_whatever();

                            /* note that we dereference self first. This
                             * can be done using an XS typemap of course */
                            xs_object_magic_attach_struct(aTHX_ SvRV(self), thingy);


            void foo (SV *self)
                    PREINIT:
                            my_struct_t *thingy;
                    INIT:
                            thingy = xs_object_magic_get_struct_rv(aTHX_ self);
                    CODE:
                            my_struct_foo(thingy); /* delegate to C api */


            /* using typemap */
            void foo (my_struct_t *thingy)
                    CODE:
                            my_struct_foo(thingy);

            /* or better yet */
            PREFIX = my_struct_

            void
            my_struct_foo (thingy)
                    my_struct_t *thingy;


            /* don't forget a destructor */
            void
            DESTROY (my_struct_t *thingy)
                    CODE:
                            Safefree(thingy);

                            /* note that xs_object_magic_get_struct() will
                             * still return a pointe which is now invalid */

DESCRPTION
    This way of associating structs with Perl space objects is designed to
    supercede Perl's builtin "T_PTROBJ" with something that is designed to
    be:

    Extensible
        The association of the pointer using "sv_magicext" can be done on
        any data type, so you can associate C structs with any
        representation type.

        This means that you can add pointers to any object (hand coded,
        Moose or otherwise), while still having instance data in regular
        hashes.

    Opaque
        The C pointer is neither visible nor modifiable from Perl space.

        This prevents accidental corruption which could lead to segfaults
        using "T_PTROBJ" (e.g. "$$ptr_obj = 0").

C API
    void *xs_object_magic_get_struct_rv(aTHX_ SV *sv)
        When called on the object reference it will check that the "sv" is a
        reference, dereference it and return the associated pointer using
        "xs_object_magic_get_struct".

        Basically the same as "xs_object_magic_get_struct(aTHX_ SvRV(sv)"
        but croaks if no magic was found.

        Note that storing a "NULL" pointer will not cause an error.

    void *xs_object_magic_get_struct(aTHX_ SV *sv)
        Fetches the pointer associated with "sv".

        Returns "NULL" if no pointer is found. There is no way to
        distinguish this from having a "NULL" pointer.

    MAGIC *xs_object_magic_get_mg (aTHX_ SV *sv)
        Fetches the appropriate "MAGIC" entry for the struct pointer storage
        from "sv".

        This lets you manipulate "mg-"mg_ptr> if you need to.

    void xs_object_magic_attach_struct(aTHX_ SV *sv, void *ptr)
        Associates "ptr" with "sv" by adding a magic entry to "sv".

    SV *xs_object_magic_create(aTHX_ void *ptr, HV *stash)
        Convenience function that creates a hash object blessed to "stash"
        and associates it with "ptr".

        Can be used to easily create a constructor:

                SV *
                new(char *class)
                        CODE:
                                RETVAL = xs_object_magic_create(
                                        (void *)test_new(),
                                        gv_stashpv(class, 0)
                                );
                        OUTPUT: RETVAL

    int xs_object_magic_has_struct(aTHX_ SV *sv)
        Returns 1 if the SV has XS::Object::Magic magic, 0 otherwise.

    int xs_object_magic_has_struct_rv(aTHX_ SV *self)
        Returns 1 if the SV references an SV that has XS::Object::Magic
        magic, 0 otherwise.

        This lets you write a quick predicate method, like:

            void
            my_struct_has_struct (self)
                    SV *self;
                    PPCODE:
                            EXTEND(SP, 1);
                            if(xs_object_magic_has_struct_rv(aTHX_ self))
                                    PUSHs(&PL_sv_yes);
                            else
                                    PUSHs(&PL_sv_no);

        Then you can check for the existence of your struct from the Perl
        side:

            if( $object->has_struct ) { ... }

    int xs_object_magic_detach_struct(aTHX_ SV *sv)
        Removes the XS::Object::Magic magic from the given SV. Returns 1 if
        something is removed, 0 otherwise.

    int xs_object_magic_detach_struct_rv(aTHX_ SV *self)
        Likes "xs_object_magic_detach_struct", but takes a reference to the
        magic-containing SV instead of the SV itself. The reference to the
        SV is typically $self.

        Returns 0 if the SV is not a reference, otherwise returns whatever
        "xs_object_magic_detach_struct" returns.

TYPEMAP
    The included typemap provides a "T_PTROBJ_MG" entry which only supports
    the "INPUT" conversion.

    This typemap entry lets you declare methods that are invoked directly on
    the associated pointer. In your own typemap add an entry:

            TYPEMAP
            my_pointer_t *  T_PTROBJ_MG

    and then you can use "my_pointer_t" as the argument type of the
    invocant:

            I32
            method (self)
                    my_pointer_t *self;
                    CODE:
                            ...

    Note that there is no "OUTPUT" conversion. In order to return your
    object you need to use ST(0) or some other means of getting the
    invocant.

VERSION CONTROL
    <http://github.com/nothingmuch/xs-object-magic>

AUTHOR
    Florian Ragwitz, Yuval Kogman

COPYRIGHT & LICENSE
            Copyright (c) 2009 Florian Ragwitz, Yuval Kogman. All rights reserved
            This program is free software; you can redistribute
            it and/or modify it under the same terms as Perl itself.