/*
 * This file was generated automatically by ExtUtils::ParseXS version 3.18 from the
 * contents of Hesiod.xs. Do not edit this file, edit Hesiod.xs instead.
 *
 *    ANY CHANGES MADE HERE WILL BE LOST!
 *
 */

#line 1 "Hesiod.xs"
#include "EXTERN.h"
#include "perl.h"
#include "XSUB.h"

/*The Makefile.PL should -I the directory with hesiod.h in it */
#include <hesiod.h> 
#include <string.h>
#include <pwd.h>
#include <netdb.h>

static int
not_here(char *s)
{
    croak("%s not implemented on this architecture", s);
    return -1;
}

static double
constant(char *name, int len, int arg)
{
    errno = EINVAL;
    return 0;
}

#line 35 "Hesiod.c"
#ifndef PERL_UNUSED_VAR
#  define PERL_UNUSED_VAR(var) if (0) var = var
#endif

#ifndef dVAR
#  define dVAR		dNOOP
#endif


/* This stuff is not part of the API! You have been warned. */
#ifndef PERL_VERSION_DECIMAL
#  define PERL_VERSION_DECIMAL(r,v,s) (r*1000000 + v*1000 + s)
#endif
#ifndef PERL_DECIMAL_VERSION
#  define PERL_DECIMAL_VERSION \
	  PERL_VERSION_DECIMAL(PERL_REVISION,PERL_VERSION,PERL_SUBVERSION)
#endif
#ifndef PERL_VERSION_GE
#  define PERL_VERSION_GE(r,v,s) \
	  (PERL_DECIMAL_VERSION >= PERL_VERSION_DECIMAL(r,v,s))
#endif
#ifndef PERL_VERSION_LE
#  define PERL_VERSION_LE(r,v,s) \
	  (PERL_DECIMAL_VERSION <= PERL_VERSION_DECIMAL(r,v,s))
#endif

/* XS_INTERNAL is the explicit static-linkage variant of the default
 * XS macro.
 *
 * XS_EXTERNAL is the same as XS_INTERNAL except it does not include
 * "STATIC", ie. it exports XSUB symbols. You probably don't want that
 * for anything but the BOOT XSUB.
 *
 * See XSUB.h in core!
 */


/* TODO: This might be compatible further back than 5.10.0. */
#if PERL_VERSION_GE(5, 10, 0) && PERL_VERSION_LE(5, 15, 1)
#  undef XS_EXTERNAL
#  undef XS_INTERNAL
#  if defined(__CYGWIN__) && defined(USE_DYNAMIC_LOADING)
#    define XS_EXTERNAL(name) __declspec(dllexport) XSPROTO(name)
#    define XS_INTERNAL(name) STATIC XSPROTO(name)
#  endif
#  if defined(__SYMBIAN32__)
#    define XS_EXTERNAL(name) EXPORT_C XSPROTO(name)
#    define XS_INTERNAL(name) EXPORT_C STATIC XSPROTO(name)
#  endif
#  ifndef XS_EXTERNAL
#    if defined(HASATTRIBUTE_UNUSED) && !defined(__cplusplus)
#      define XS_EXTERNAL(name) void name(pTHX_ CV* cv __attribute__unused__)
#      define XS_INTERNAL(name) STATIC void name(pTHX_ CV* cv __attribute__unused__)
#    else
#      ifdef __cplusplus
#        define XS_EXTERNAL(name) extern "C" XSPROTO(name)
#        define XS_INTERNAL(name) static XSPROTO(name)
#      else
#        define XS_EXTERNAL(name) XSPROTO(name)
#        define XS_INTERNAL(name) STATIC XSPROTO(name)
#      endif
#    endif
#  endif
#endif

/* perl >= 5.10.0 && perl <= 5.15.1 */


/* The XS_EXTERNAL macro is used for functions that must not be static
 * like the boot XSUB of a module. If perl didn't have an XS_EXTERNAL
 * macro defined, the best we can do is assume XS is the same.
 * Dito for XS_INTERNAL.
 */
#ifndef XS_EXTERNAL
#  define XS_EXTERNAL(name) XS(name)
#endif
#ifndef XS_INTERNAL
#  define XS_INTERNAL(name) XS(name)
#endif

/* Now, finally, after all this mess, we want an ExtUtils::ParseXS
 * internal macro that we're free to redefine for varying linkage due
 * to the EXPORT_XSUB_SYMBOLS XS keyword. This is internal, use
 * XS_EXTERNAL(name) or XS_INTERNAL(name) in your code if you need to!
 */

#undef XS_EUPXS
#if defined(PERL_EUPXS_ALWAYS_EXPORT)
#  define XS_EUPXS(name) XS_EXTERNAL(name)
#else
   /* default to internal */
#  define XS_EUPXS(name) XS_INTERNAL(name)
#endif

#ifndef PERL_ARGS_ASSERT_CROAK_XS_USAGE
#define PERL_ARGS_ASSERT_CROAK_XS_USAGE assert(cv); assert(params)

/* prototype to pass -Wmissing-prototypes */
STATIC void
S_croak_xs_usage(pTHX_ const CV *const cv, const char *const params);

STATIC void
S_croak_xs_usage(pTHX_ const CV *const cv, const char *const params)
{
    const GV *const gv = CvGV(cv);

    PERL_ARGS_ASSERT_CROAK_XS_USAGE;

    if (gv) {
        const char *const gvname = GvNAME(gv);
        const HV *const stash = GvSTASH(gv);
        const char *const hvname = stash ? HvNAME(stash) : NULL;

        if (hvname)
            Perl_croak(aTHX_ "Usage: %s::%s(%s)", hvname, gvname, params);
        else
            Perl_croak(aTHX_ "Usage: %s(%s)", gvname, params);
    } else {
        /* Pants. I don't think that it should be possible to get here. */
        Perl_croak(aTHX_ "Usage: CODE(0x%"UVxf")(%s)", PTR2UV(cv), params);
    }
}
#undef  PERL_ARGS_ASSERT_CROAK_XS_USAGE

#ifdef PERL_IMPLICIT_CONTEXT
#define croak_xs_usage(a,b)    S_croak_xs_usage(aTHX_ a,b)
#else
#define croak_xs_usage        S_croak_xs_usage
#endif

#endif

/* NOTE: the prototype of newXSproto() is different in versions of perls,
 * so we define a portable version of newXSproto()
 */
#ifdef newXS_flags
#define newXSproto_portable(name, c_impl, file, proto) newXS_flags(name, c_impl, file, proto, 0)
#else
#define newXSproto_portable(name, c_impl, file, proto) (PL_Sv=(SV*)newXS(name, c_impl, file), sv_setpv(PL_Sv, proto), (CV*)PL_Sv)
#endif /* !defined(newXS_flags) */

#line 177 "Hesiod.c"

XS_EUPXS(XS_Net__Hesiod_constant); /* prototype to pass -Wmissing-prototypes */
XS_EUPXS(XS_Net__Hesiod_constant)
{
    dVAR; dXSARGS;
    if (items != 2)
       croak_xs_usage(cv,  "sv, arg");
    {
#line 31 "Hesiod.xs"
	STRLEN		len;
#line 188 "Hesiod.c"
	SV *	sv = ST(0)
;
	char *	s = SvPV(sv, len);
	int	arg = (int)SvIV(ST(1))
;
	double	RETVAL;
	dXSTARG;
#line 37 "Hesiod.xs"
	RETVAL = constant(s,len,arg);
#line 198 "Hesiod.c"
	XSprePUSH; PUSHn((double)RETVAL);
    }
    XSRETURN(1);
}


XS_EUPXS(XS_Net__Hesiod_hesiod_init); /* prototype to pass -Wmissing-prototypes */
XS_EUPXS(XS_Net__Hesiod_hesiod_init)
{
    dVAR; dXSARGS;
    if (items != 1)
       croak_xs_usage(cv,  "context");
    {
	void *	context;
	int	RETVAL;
	dXSTARG;
#line 49 "Hesiod.xs"
		RETVAL = hesiod_init(&context);
#line 217 "Hesiod.c"
	sv_setref_pv(ST(0), Nullch, (void*)context);
	SvSETMAGIC(ST(0));
	XSprePUSH; PUSHi((IV)RETVAL);
    }
    XSRETURN(1);
}


XS_EUPXS(XS_Net__Hesiod_hesiod_end); /* prototype to pass -Wmissing-prototypes */
XS_EUPXS(XS_Net__Hesiod_hesiod_end)
{
    dVAR; dXSARGS;
    if (items != 1)
       croak_xs_usage(cv,  "context");
    {
	void *	context;

	if (SvROK(ST(0))) {
	    IV tmp = SvIV((SV*)SvRV(ST(0)));
	    context = INT2PTR(void *,tmp);
	}
	else
	    Perl_croak(aTHX_ "%s: %s is not a reference",
			"Net::Hesiod::hesiod_end",
			"context")
;

	hesiod_end(context);
    }
    XSRETURN_EMPTY;
}


XS_EUPXS(XS_Net__Hesiod_hesiod_to_bind); /* prototype to pass -Wmissing-prototypes */
XS_EUPXS(XS_Net__Hesiod_hesiod_to_bind)
{
    dVAR; dXSARGS;
    if (items != 3)
       croak_xs_usage(cv,  "context, name, type");
    {
	void *	context;
	const char *	name = (const char *)SvPV_nolen(ST(1))
;
	const char *	type = (const char *)SvPV_nolen(ST(2))
;
	char *	RETVAL;
	dXSTARG;

	if (SvROK(ST(0))) {
	    IV tmp = SvIV((SV*)SvRV(ST(0)));
	    context = INT2PTR(void *,tmp);
	}
	else
	    Perl_croak(aTHX_ "%s: %s is not a reference",
			"Net::Hesiod::hesiod_to_bind",
			"context")
;

	RETVAL = hesiod_to_bind(context, name, type);
	sv_setpv(TARG, RETVAL); XSprePUSH; PUSHTARG;
    }
    XSRETURN(1);
}


XS_EUPXS(XS_Net__Hesiod_hesiod_resolve); /* prototype to pass -Wmissing-prototypes */
XS_EUPXS(XS_Net__Hesiod_hesiod_resolve)
{
    dVAR; dXSARGS;
    if (items != 3)
       croak_xs_usage(cv,  "context, name, type");
    PERL_UNUSED_VAR(ax); /* -Wall */
    SP -= items;
    {
	void *	context;
	const char *	name = (const char *)SvPV_nolen(ST(1))
;
	const char *	type = (const char *)SvPV_nolen(ST(2))
;
#line 70 "Hesiod.xs"
		char ** results;
		char ** res;
		int i;
		dTARGET;
		SV *tmp;
#line 303 "Hesiod.c"

	if (SvROK(ST(0))) {
	    IV tmp = SvIV((SV*)SvRV(ST(0)));
	    context = INT2PTR(void *,tmp);
	}
	else
	    Perl_croak(aTHX_ "%s: %s is not a reference",
			"Net::Hesiod::hesiod_resolve",
			"context")
;
#line 76 "Hesiod.xs"
		results = hesiod_resolve(context,name,type);
		/* Handle an invalid query */
		if ( ! results ) { XSRETURN_EMPTY; }
		res = results;
		i =0;
		while ( *res )
		{	tmp = sv_newmortal();
			sv_setpv(tmp,*res);
			XPUSHs(tmp);
			i++;
			res++;
		}
		hesiod_free_list(context,results);
		XSRETURN(i);
#line 329 "Hesiod.c"
	PUTBACK;
	return;
    }
}


XS_EUPXS(XS_Net__Hesiod_hesiod_getpwnam); /* prototype to pass -Wmissing-prototypes */
XS_EUPXS(XS_Net__Hesiod_hesiod_getpwnam)
{
    dVAR; dXSARGS;
    if (items != 2)
       croak_xs_usage(cv,  "context, name");
    PERL_UNUSED_VAR(ax); /* -Wall */
    SP -= items;
    {
	void *	context;
	const char *	name = (const char *)SvPV_nolen(ST(1))
;
#line 97 "Hesiod.xs"
		struct passwd *pw;
		dTARGET;
		SV *tmp;
#line 352 "Hesiod.c"

	if (SvROK(ST(0))) {
	    IV tmp = SvIV((SV*)SvRV(ST(0)));
	    context = INT2PTR(void *,tmp);
	}
	else
	    Perl_croak(aTHX_ "%s: %s is not a reference",
			"Net::Hesiod::hesiod_getpwnam",
			"context")
;
#line 101 "Hesiod.xs"
		pw = hesiod_getpwnam(context,name);
		/* Handle errors by returning empty */
		if ( ! pw ) { XSRETURN_EMPTY; }
		/* Presumably, have a valid response */
		tmp = sv_newmortal(); sv_setpv(tmp,pw->pw_name);
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setpv(tmp,pw->pw_passwd);
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setiv(tmp,pw->pw_uid);
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setiv(tmp,pw->pw_gid);
		XPUSHs(tmp);
		tmp = sv_newmortal();
		/* Not all systems have a pw_quota or pw_comment elements. 
		** Solaris lacks pw_quota, having pw_age instead
		** Linux lacks both
			If missing, leave undef the return value 
		*/
#ifndef DONT_HAVE_PW_QUOTA
		sv_setiv(tmp,pw->pw_quota);
#else
		sv_setsv(tmp,&PL_sv_undef);
#endif
		XPUSHs(tmp);
		tmp = sv_newmortal(); 
#ifndef DONT_HAVE_PW_COMMENT
		sv_setpv(tmp,pw->pw_comment);
#else
		sv_setsv(tmp,&PL_sv_undef);
#endif
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setpv(tmp,pw->pw_gecos);
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setpv(tmp,pw->pw_dir);
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setpv(tmp,pw->pw_shell);
		XPUSHs(tmp);
		hesiod_free_passwd(context,pw);
		XSRETURN(9);
#line 403 "Hesiod.c"
	PUTBACK;
	return;
    }
}


XS_EUPXS(XS_Net__Hesiod_hesiod_getpwuid); /* prototype to pass -Wmissing-prototypes */
XS_EUPXS(XS_Net__Hesiod_hesiod_getpwuid)
{
    dVAR; dXSARGS;
    if (items != 2)
       croak_xs_usage(cv,  "context, uid");
    PERL_UNUSED_VAR(ax); /* -Wall */
    SP -= items;
    {
	void *	context;
	uid_t	uid = (uid_t)SvIV(ST(1))
;
#line 146 "Hesiod.xs"
		struct passwd *pw;
		dTARGET;
		SV *tmp;
#line 426 "Hesiod.c"

	if (SvROK(ST(0))) {
	    IV tmp = SvIV((SV*)SvRV(ST(0)));
	    context = INT2PTR(void *,tmp);
	}
	else
	    Perl_croak(aTHX_ "%s: %s is not a reference",
			"Net::Hesiod::hesiod_getpwuid",
			"context")
;
#line 150 "Hesiod.xs"
		pw = hesiod_getpwuid(context,uid);
		/* Handle errors by returning empty */
		if ( ! pw ) { XSRETURN_EMPTY; }
		/* Presumably, have a valid response */
		tmp = sv_newmortal(); sv_setpv(tmp,pw->pw_name);
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setpv(tmp,pw->pw_passwd);
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setiv(tmp,pw->pw_uid);
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setiv(tmp,pw->pw_gid);
		XPUSHs(tmp);
		tmp = sv_newmortal();
		/* Leave undef if lacking pw_quota or pw_comment */
#ifndef DONT_HAVE_PW_QUOTA
		sv_setiv(tmp,pw->pw_quota);
#else
		sv_setsv(tmp,&PL_sv_undef);
#endif
		XPUSHs(tmp);
		tmp = sv_newmortal();
#ifndef DONT_HAVE_PW_COMMENT
		sv_setpv(tmp,pw->pw_comment);
#else
		sv_setsv(tmp,&PL_sv_undef);
#endif
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setpv(tmp,pw->pw_gecos);
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setpv(tmp,pw->pw_dir);
		XPUSHs(tmp);
		tmp = sv_newmortal(); sv_setpv(tmp,pw->pw_shell);
		XPUSHs(tmp);
		hesiod_free_passwd(context,pw);
		XSRETURN(9);
#line 473 "Hesiod.c"
	PUTBACK;
	return;
    }
}


XS_EUPXS(XS_Net__Hesiod_hesiod_getservbyname); /* prototype to pass -Wmissing-prototypes */
XS_EUPXS(XS_Net__Hesiod_hesiod_getservbyname)
{
    dVAR; dXSARGS;
    if (items != 3)
       croak_xs_usage(cv,  "context, name, proto");
    PERL_UNUSED_VAR(ax); /* -Wall */
    SP -= items;
    {
	void *	context;
	const char *	name = (const char *)SvPV_nolen(ST(1))
;
	const char *	proto = (const char *)SvPV_nolen(ST(2))
;
#line 192 "Hesiod.xs"
		struct servent *sent;
		dTARGET;
		SV *tmp;
		int port;
#line 499 "Hesiod.c"
	struct servent *	RETVAL;

	if (SvROK(ST(0))) {
	    IV tmp = SvIV((SV*)SvRV(ST(0)));
	    context = INT2PTR(void *,tmp);
	}
	else
	    Perl_croak(aTHX_ "%s: %s is not a reference",
			"Net::Hesiod::hesiod_getservbyname",
			"context")
;
#line 197 "Hesiod.xs"
		sent = hesiod_getservbyname(context,name, proto);
		/* Handle errors by returning empty */
		if ( ! sent ) { XSRETURN_EMPTY; }
		/* Presumably, have a valid response */
		/* servent struct is supposed to store port number in
			network byte order.  We return it in host order
			to be consistent with perl getservbyname */
		port = ntohs(sent->s_port);
		tmp= sv_newmortal(); sv_setpv(tmp,sent->s_name);
		XPUSHs(tmp);
		/* WE now concat the alias list */
		tmp= sv_newmortal();
		if ( *sent->s_aliases )
		{ 	sv_setpv(tmp,*sent->s_aliases);
			sent->s_aliases++;
		} else
		{	sv_setsv(tmp,&PL_sv_undef);
		}
		while ( *sent->s_aliases )
		{	sv_catpv(tmp,*sent->s_aliases);
			sent->s_aliases++;
		} 
		XPUSHs(tmp);

		tmp= sv_newmortal(); sv_setiv(tmp,port);
		XPUSHs(tmp);
		tmp= sv_newmortal(); sv_setpv(tmp,sent->s_proto);
		XPUSHs(tmp);

		hesiod_free_servent(context,sent);
		XSRETURN(4);
#line 543 "Hesiod.c"
	PUTBACK;
	return;
    }
}


XS_EUPXS(XS_Net__Hesiod_hesiod_getmailhost); /* prototype to pass -Wmissing-prototypes */
XS_EUPXS(XS_Net__Hesiod_hesiod_getmailhost)
{
    dVAR; dXSARGS;
    if (items != 2)
       croak_xs_usage(cv,  "context, user");
    PERL_UNUSED_VAR(ax); /* -Wall */
    SP -= items;
    {
	void *	context;
	char *	user = (char *)SvPV_nolen(ST(1))
;
#line 234 "Hesiod.xs"
		struct hesiod_postoffice *po;
		dTARGET;
		SV * tmp;
#line 566 "Hesiod.c"
	struct hesiod_postoffice *	RETVAL;

	if (SvROK(ST(0))) {
	    IV tmp = SvIV((SV*)SvRV(ST(0)));
	    context = INT2PTR(void *,tmp);
	}
	else
	    Perl_croak(aTHX_ "%s: %s is not a reference",
			"Net::Hesiod::hesiod_getmailhost",
			"context")
;
#line 238 "Hesiod.xs"
		po = hesiod_getmailhost(context,user);
		/* Handle errors by returning empty */
		if ( ! po ) { XSRETURN_EMPTY; }
		/* Presumably, have a valid response */
		tmp= sv_newmortal(); sv_setpv(tmp,po->hesiod_po_type);
		XPUSHs(tmp);
		tmp= sv_newmortal(); sv_setpv(tmp,po->hesiod_po_host);
		XPUSHs(tmp);
		tmp= sv_newmortal(); sv_setpv(tmp,po->hesiod_po_name);
		XPUSHs(tmp);
		hesiod_free_postoffice(context,po);
		XSRETURN(3);
#line 591 "Hesiod.c"
	PUTBACK;
	return;
    }
}

#ifdef __cplusplus
extern "C"
#endif
XS_EXTERNAL(boot_Net__Hesiod); /* prototype to pass -Wmissing-prototypes */
XS_EXTERNAL(boot_Net__Hesiod)
{
    dVAR; dXSARGS;
#if (PERL_REVISION == 5 && PERL_VERSION < 9)
    char* file = __FILE__;
#else
    const char* file = __FILE__;
#endif

    PERL_UNUSED_VAR(cv); /* -W */
    PERL_UNUSED_VAR(items); /* -W */
#ifdef XS_APIVERSION_BOOTCHECK
    XS_APIVERSION_BOOTCHECK;
#endif
    XS_VERSION_BOOTCHECK;

        newXS("Net::Hesiod::constant", XS_Net__Hesiod_constant, file);
        newXS("Net::Hesiod::hesiod_init", XS_Net__Hesiod_hesiod_init, file);
        newXS("Net::Hesiod::hesiod_end", XS_Net__Hesiod_hesiod_end, file);
        newXS("Net::Hesiod::hesiod_to_bind", XS_Net__Hesiod_hesiod_to_bind, file);
        newXS("Net::Hesiod::hesiod_resolve", XS_Net__Hesiod_hesiod_resolve, file);
        newXS("Net::Hesiod::hesiod_getpwnam", XS_Net__Hesiod_hesiod_getpwnam, file);
        newXS("Net::Hesiod::hesiod_getpwuid", XS_Net__Hesiod_hesiod_getpwuid, file);
        newXS("Net::Hesiod::hesiod_getservbyname", XS_Net__Hesiod_hesiod_getservbyname, file);
        newXS("Net::Hesiod::hesiod_getmailhost", XS_Net__Hesiod_hesiod_getmailhost, file);
#if (PERL_REVISION == 5 && PERL_VERSION >= 9)
  if (PL_unitcheckav)
       call_list(PL_scopestack_ix, PL_unitcheckav);
#endif
    XSRETURN_YES;
}

