/* * This file was generated automatically by ExtUtils::ParseXS version 3.40 from the * contents of Base64.xs. Do not edit this file, edit Base64.xs instead. * * ANY CHANGES MADE HERE WILL BE LOST! * */ #line 1 "Base64.xs" /* Copyright 1997-2004 Gisle Aas This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself. The tables and some of the code that used to be here was borrowed from metamail, which comes with this message: Copyright (c) 1991 Bell Communications Research, Inc. (Bellcore) Permission to use, copy, modify, and distribute this material for any purpose and without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies, and that the name of Bellcore not be used in advertising or publicity pertaining to this material without the specific, prior written permission of an authorized representative of Bellcore. BELLCORE MAKES NO REPRESENTATIONS ABOUT THE ACCURACY OR SUITABILITY OF THIS MATERIAL FOR ANY PURPOSE. IT IS PROVIDED "AS IS", WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES. */ #ifdef __cplusplus extern "C" { #endif #define PERL_NO_GET_CONTEXT /* we want efficiency */ #include "EXTERN.h" #include "perl.h" #include "XSUB.h" #ifdef __cplusplus } #endif #define MAX_LINE 76 /* size of encoded lines */ static const char basis_64[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; #define XX 255 /* illegal base64 char */ #define EQ 254 /* padding */ #define INVALID XX static const unsigned char index_64[256] = { XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,62, XX,XX,XX,63, 52,53,54,55, 56,57,58,59, 60,61,XX,XX, XX,EQ,XX,XX, XX, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9,10, 11,12,13,14, 15,16,17,18, 19,20,21,22, 23,24,25,XX, XX,XX,XX,XX, XX,26,27,28, 29,30,31,32, 33,34,35,36, 37,38,39,40, 41,42,43,44, 45,46,47,48, 49,50,51,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, XX,XX,XX,XX, }; #ifdef SvPVbyte # if PERL_REVISION == 5 && PERL_VERSION < 7 /* SvPVbyte does not work in perl-5.6.1, borrowed version for 5.7.3 */ # undef SvPVbyte # define SvPVbyte(sv, lp) \ ((SvFLAGS(sv) & (SVf_POK|SVf_UTF8)) == (SVf_POK) \ ? ((lp = SvCUR(sv)), SvPVX(sv)) : my_sv_2pvbyte(aTHX_ sv, &lp)) static char * my_sv_2pvbyte(pTHX_ register SV *sv, STRLEN *lp) { sv_utf8_downgrade(sv,0); return SvPV(sv,*lp); } # endif #else # define SvPVbyte SvPV #endif #ifndef isXDIGIT # define isXDIGIT isxdigit #endif #ifndef NATIVE_TO_ASCII # define NATIVE_TO_ASCII(ch) (ch) #endif #line 104 "Base64.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(const CV *const cv, const char *const params); STATIC void S_croak_xs_usage(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_nocontext("Usage: %s::%s(%s)", hvname, gvname, params); else Perl_croak_nocontext("Usage: %s(%s)", gvname, params); } else { /* Pants. I don't think that it should be possible to get here. */ Perl_croak_nocontext("Usage: CODE(0x%" UVxf ")(%s)", PTR2UV(cv), params); } } #undef PERL_ARGS_ASSERT_CROAK_XS_USAGE #define croak_xs_usage S_croak_xs_usage #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) */ #if PERL_VERSION_LE(5, 21, 5) # define newXS_deffile(a,b) Perl_newXS(aTHX_ a,b,file) #else # define newXS_deffile(a,b) Perl_newXS_deffile(aTHX_ a,b) #endif #line 248 "Base64.c" XS_EUPXS(XS_MIME__Base64_encode_base64); /* prototype to pass -Wmissing-prototypes */ XS_EUPXS(XS_MIME__Base64_encode_base64) { dVAR; dXSARGS; if (items < 1) croak_xs_usage(cv, "sv, ..."); { SV* sv = ST(0) ; #line 102 "Base64.xs" char *str; /* string to encode */ SSize_t len; /* length of the string */ const char*eol;/* the end-of-line sequence to use */ STRLEN eollen; /* length of the EOL sequence */ char *r; /* result string */ STRLEN rlen; /* length of result string */ unsigned char c1, c2, c3; int chunk; U32 had_utf8; #line 270 "Base64.c" SV * RETVAL; #line 113 "Base64.xs" #if PERL_REVISION == 5 && PERL_VERSION >= 6 had_utf8 = SvUTF8(sv); sv_utf8_downgrade(sv, FALSE); #endif str = SvPV(sv, rlen); /* SvPV(sv, len) gives warning for signed len */ len = (SSize_t)rlen; /* set up EOL from the second argument if present, default to "\n" */ if (items > 1 && SvOK(ST(1))) { eol = SvPV(ST(1), eollen); } else { eol = "\n"; eollen = 1; } /* calculate the length of the result */ rlen = (len+2) / 3 * 4; /* encoded bytes */ if (rlen) { /* add space for EOL */ rlen += ((rlen-1) / MAX_LINE + 1) * eollen; } /* allocate a result buffer */ RETVAL = newSV(rlen ? rlen : 1); SvPOK_on(RETVAL); SvCUR_set(RETVAL, rlen); r = SvPVX(RETVAL); /* encode */ for (chunk=0; len > 0; len -= 3, chunk++) { if (chunk == (MAX_LINE/4)) { const char *c = eol; const char *e = eol + eollen; while (c < e) *r++ = *c++; chunk = 0; } c1 = *str++; c2 = len > 1 ? *str++ : '\0'; *r++ = basis_64[c1>>2]; *r++ = basis_64[((c1 & 0x3)<< 4) | ((c2 & 0xF0) >> 4)]; if (len > 2) { c3 = *str++; *r++ = basis_64[((c2 & 0xF) << 2) | ((c3 & 0xC0) >>6)]; *r++ = basis_64[c3 & 0x3F]; } else if (len == 2) { *r++ = basis_64[(c2 & 0xF) << 2]; *r++ = '='; } else { /* len == 1 */ *r++ = '='; *r++ = '='; } } if (rlen) { /* append eol to the result string */ const char *c = eol; const char *e = eol + eollen; while (c < e) *r++ = *c++; } *r = '\0'; /* every SV in perl should be NUL-terminated */ #if PERL_REVISION == 5 && PERL_VERSION >= 6 if (had_utf8) sv_utf8_upgrade(sv); #endif #line 339 "Base64.c" RETVAL = sv_2mortal(RETVAL); ST(0) = RETVAL; } XSRETURN(1); } XS_EUPXS(XS_MIME__Base64_decode_base64); /* prototype to pass -Wmissing-prototypes */ XS_EUPXS(XS_MIME__Base64_decode_base64) { dVAR; dXSARGS; if (items != 1) croak_xs_usage(cv, "sv"); { SV* sv = ST(0) ; #line 188 "Base64.xs" STRLEN len; register unsigned char *str = (unsigned char*)SvPV(sv, len); unsigned char const* end = str + len; char *r; unsigned char c[4]; #line 363 "Base64.c" SV * RETVAL; #line 195 "Base64.xs" { /* always enough, but might be too much */ STRLEN rlen = len * 3 / 4; RETVAL = newSV(rlen ? rlen : 1); } SvPOK_on(RETVAL); r = SvPVX(RETVAL); while (str < end) { int i = 0; do { unsigned char uc = index_64[NATIVE_TO_ASCII(*str++)]; if (uc != INVALID) c[i++] = uc; if (str == end) { if (i < 4) { if (i < 2) goto thats_it; if (i == 2) c[2] = EQ; c[3] = EQ; } break; } } while (i < 4); if (c[0] == EQ || c[1] == EQ) { break; } /* printf("c0=%d,c1=%d,c2=%d,c3=%d\n", c[0],c[1],c[2],c[3]);*/ *r++ = (c[0] << 2) | ((c[1] & 0x30) >> 4); if (c[2] == EQ) break; *r++ = ((c[1] & 0x0F) << 4) | ((c[2] & 0x3C) >> 2); if (c[3] == EQ) break; *r++ = ((c[2] & 0x03) << 6) | c[3]; } thats_it: SvCUR_set(RETVAL, r - SvPVX(RETVAL)); *r = '\0'; #line 411 "Base64.c" RETVAL = sv_2mortal(RETVAL); ST(0) = RETVAL; } XSRETURN(1); } XS_EUPXS(XS_MIME__Base64_encoded_base64_length); /* prototype to pass -Wmissing-prototypes */ XS_EUPXS(XS_MIME__Base64_encoded_base64_length) { dVAR; dXSARGS; if (items < 1) croak_xs_usage(cv, "sv, ..."); { SV* sv = ST(0) ; #line 249 "Base64.xs" SSize_t len; /* length of the string */ STRLEN eollen; /* length of the EOL sequence */ U32 had_utf8; #line 433 "Base64.c" int RETVAL; dXSTARG; #line 254 "Base64.xs" #if PERL_REVISION == 5 && PERL_VERSION >= 6 had_utf8 = SvUTF8(sv); sv_utf8_downgrade(sv, FALSE); #endif len = SvCUR(sv); #if PERL_REVISION == 5 && PERL_VERSION >= 6 if (had_utf8) sv_utf8_upgrade(sv); #endif if (items > 1 && SvOK(ST(1))) { eollen = SvCUR(ST(1)); } else { eollen = 1; } RETVAL = (len+2) / 3 * 4; /* encoded bytes */ if (RETVAL) { RETVAL += ((RETVAL-1) / MAX_LINE + 1) * eollen; } #line 458 "Base64.c" XSprePUSH; PUSHi((IV)RETVAL); } XSRETURN(1); } XS_EUPXS(XS_MIME__Base64_decoded_base64_length); /* prototype to pass -Wmissing-prototypes */ XS_EUPXS(XS_MIME__Base64_decoded_base64_length) { dVAR; dXSARGS; if (items != 1) croak_xs_usage(cv, "sv"); { SV* sv = ST(0) ; #line 284 "Base64.xs" STRLEN len; register unsigned char *str = (unsigned char*)SvPV(sv, len); unsigned char const* end = str + len; int i = 0; #line 480 "Base64.c" int RETVAL; dXSTARG; #line 290 "Base64.xs" RETVAL = 0; while (str < end) { unsigned char uc = index_64[NATIVE_TO_ASCII(*str++)]; if (uc == INVALID) continue; if (uc == EQ) break; if (i++) { RETVAL++; if (i == 4) i = 0; } } #line 498 "Base64.c" XSprePUSH; PUSHi((IV)RETVAL); } XSRETURN(1); } #ifdef EBCDIC #define qp_isplain(c) ((c) == '\t' || ((!isprint(c) && (c) != '='))) #else #define qp_isplain(c) ((c) == '\t' || (((c) >= ' ' && (c) <= '~') && (c) != '=')) #endif XS_EUPXS(XS_MIME__QuotedPrint_encode_qp); /* prototype to pass -Wmissing-prototypes */ XS_EUPXS(XS_MIME__QuotedPrint_encode_qp) { dVAR; dXSARGS; if (items < 1) croak_xs_usage(cv, "sv, ..."); { SV* sv = ST(0) ; #line 322 "Base64.xs" const char *eol; STRLEN eol_len; int binary; STRLEN sv_len; STRLEN linelen; char *beg; char *end; char *p; char *p_beg; STRLEN p_len; U32 had_utf8; #line 532 "Base64.c" SV * RETVAL; #line 335 "Base64.xs" #if PERL_REVISION == 5 && PERL_VERSION >= 6 had_utf8 = SvUTF8(sv); sv_utf8_downgrade(sv, FALSE); #endif /* set up EOL from the second argument if present, default to "\n" */ if (items > 1 && SvOK(ST(1))) { eol = SvPV(ST(1), eol_len); } else { eol = "\n"; eol_len = 1; } binary = (items > 2 && SvTRUE(ST(2))); beg = SvPV(sv, sv_len); end = beg + sv_len; RETVAL = newSV(sv_len + 1); sv_setpv(RETVAL, ""); linelen = 0; p = beg; while (1) { p_beg = p; /* skip past as much plain text as possible */ while (p < end && qp_isplain(*p)) { p++; } if (p == end || *p == '\n') { /* whitespace at end of line must be encoded */ while (p > p_beg && (*(p - 1) == '\t' || *(p - 1) == ' ')) p--; } p_len = p - p_beg; if (p_len) { /* output plain text (with line breaks) */ if (eol_len) { while (p_len > MAX_LINE - 1 - linelen) { STRLEN len = MAX_LINE - 1 - linelen; sv_catpvn(RETVAL, p_beg, len); p_beg += len; p_len -= len; sv_catpvn(RETVAL, "=", 1); sv_catpvn(RETVAL, eol, eol_len); linelen = 0; } } if (p_len) { sv_catpvn(RETVAL, p_beg, p_len); linelen += p_len; } } if (p == end) { break; } else if (*p == '\n' && eol_len && !binary) { if (linelen == 1 && SvCUR(RETVAL) > eol_len + 1 && (SvEND(RETVAL)-eol_len)[-2] == '=') { /* fixup useless soft linebreak */ (SvEND(RETVAL)-eol_len)[-2] = SvEND(RETVAL)[-1]; SvCUR_set(RETVAL, SvCUR(RETVAL) - 1); } else { sv_catpvn(RETVAL, eol, eol_len); } p++; linelen = 0; } else { /* output escaped char (with line breaks) */ assert(p < end); if (eol_len && linelen > MAX_LINE - 4 && !(linelen == MAX_LINE - 3 && p + 1 < end && p[1] == '\n' && !binary)) { sv_catpvn(RETVAL, "=", 1); sv_catpvn(RETVAL, eol, eol_len); linelen = 0; } sv_catpvf(RETVAL, "=%02X", (unsigned char)*p); p++; linelen += 3; } /* optimize reallocs a bit */ if (SvLEN(RETVAL) > 80 && SvLEN(RETVAL) - SvCUR(RETVAL) < 3) { STRLEN expected_len = (SvCUR(RETVAL) * sv_len) / (p - beg); SvGROW(RETVAL, expected_len); } } if (SvCUR(RETVAL) && eol_len && linelen) { sv_catpvn(RETVAL, "=", 1); sv_catpvn(RETVAL, eol, eol_len); } #if PERL_REVISION == 5 && PERL_VERSION >= 6 if (had_utf8) sv_utf8_upgrade(sv); #endif #line 634 "Base64.c" RETVAL = sv_2mortal(RETVAL); ST(0) = RETVAL; } XSRETURN(1); } XS_EUPXS(XS_MIME__QuotedPrint_decode_qp); /* prototype to pass -Wmissing-prototypes */ XS_EUPXS(XS_MIME__QuotedPrint_decode_qp) { dVAR; dXSARGS; if (items != 1) croak_xs_usage(cv, "sv"); { SV* sv = ST(0) ; #line 443 "Base64.xs" STRLEN len; char *str = SvPVbyte(sv, len); char const* end = str + len; char *r; char *whitespace = 0; #line 658 "Base64.c" SV * RETVAL; #line 450 "Base64.xs" RETVAL = newSV(len ? len : 1); SvPOK_on(RETVAL); r = SvPVX(RETVAL); while (str < end) { if (*str == ' ' || *str == '\t') { if (!whitespace) whitespace = str; str++; } else if (*str == '\r' && (str + 1) < end && str[1] == '\n') { str++; } else if (*str == '\n') { whitespace = 0; *r++ = *str++; } else { if (whitespace) { while (whitespace < str) { *r++ = *whitespace++; } whitespace = 0; } if (*str == '=') { if ((str + 2) < end && isXDIGIT(str[1]) && isXDIGIT(str[2])) { char buf[3]; str++; buf[0] = *str++; buf[1] = *str++; buf[2] = '\0'; *r++ = (char)strtol(buf, 0, 16); } else { /* look for soft line break */ char *p = str + 1; while (p < end && (*p == ' ' || *p == '\t')) p++; if (p < end && *p == '\n') str = p + 1; else if ((p + 1) < end && *p == '\r' && *(p + 1) == '\n') str = p + 2; else *r++ = *str++; /* give up */ } } else { *r++ = *str++; } } } if (whitespace) { while (whitespace < str) { *r++ = *whitespace++; } } *r = '\0'; SvCUR_set(RETVAL, r - SvPVX(RETVAL)); #line 719 "Base64.c" RETVAL = sv_2mortal(RETVAL); ST(0) = RETVAL; } XSRETURN(1); } #ifdef __cplusplus extern "C" #endif XS_EXTERNAL(boot_MIME__Base64); /* prototype to pass -Wmissing-prototypes */ XS_EXTERNAL(boot_MIME__Base64) { #if PERL_VERSION_LE(5, 21, 5) dVAR; dXSARGS; #else dVAR; dXSBOOTARGSXSAPIVERCHK; #endif #if (PERL_REVISION == 5 && PERL_VERSION < 9) char* file = __FILE__; #else const char* file = __FILE__; #endif PERL_UNUSED_VAR(file); PERL_UNUSED_VAR(cv); /* -W */ PERL_UNUSED_VAR(items); /* -W */ #if PERL_VERSION_LE(5, 21, 5) XS_VERSION_BOOTCHECK; # ifdef XS_APIVERSION_BOOTCHECK XS_APIVERSION_BOOTCHECK; # endif #endif (void)newXSproto_portable("MIME::Base64::encode_base64", XS_MIME__Base64_encode_base64, file, "$;$"); (void)newXSproto_portable("MIME::Base64::decode_base64", XS_MIME__Base64_decode_base64, file, "$"); (void)newXSproto_portable("MIME::Base64::encoded_base64_length", XS_MIME__Base64_encoded_base64_length, file, "$;$"); (void)newXSproto_portable("MIME::Base64::decoded_base64_length", XS_MIME__Base64_decoded_base64_length, file, "$"); (void)newXSproto_portable("MIME::QuotedPrint::encode_qp", XS_MIME__QuotedPrint_encode_qp, file, "$;$$"); (void)newXSproto_portable("MIME::QuotedPrint::decode_qp", XS_MIME__QuotedPrint_decode_qp, file, "$"); #if PERL_VERSION_LE(5, 21, 5) # if PERL_VERSION_GE(5, 9, 0) if (PL_unitcheckav) call_list(PL_scopestack_ix, PL_unitcheckav); # endif XSRETURN_YES; #else Perl_xs_boot_epilog(aTHX_ ax); #endif }