/* This file is part of the "version" CPAN distribution. Please avoid editing it in the perl core. */ #ifdef PERL_CORE # include "vutil.h" #endif #define VERSION_MAX 0x7FFFFFFF /* =for apidoc prescan_version Validate that a given string can be parsed as a version object, but doesn't actually perform the parsing. Can use either strict or lax validation rules. Can optionally set a number of hint variables to save the parsing code some time when tokenizing. =cut */ const char * #ifdef VUTIL_REPLACE_CORE Perl_prescan_version2(pTHX_ const char *s, bool strict, #else Perl_prescan_version(pTHX_ const char *s, bool strict, #endif const char **errstr, bool *sqv, int *ssaw_decimal, int *swidth, bool *salpha) { bool qv = (sqv ? *sqv : FALSE); int width = 3; int saw_decimal = 0; bool alpha = FALSE; const char *d = s; PERL_ARGS_ASSERT_PRESCAN_VERSION; PERL_UNUSED_CONTEXT; if (qv && isDIGIT(*d)) goto dotted_decimal_version; if (*d == 'v') { /* explicit v-string */ d++; if (isDIGIT(*d)) { qv = TRUE; } else { /* degenerate v-string */ /* requires v1.2.3 */ BADVERSION(s,errstr,"Invalid version format (dotted-decimal versions require at least three parts)"); } dotted_decimal_version: if (strict && d[0] == '0' && isDIGIT(d[1])) { /* no leading zeros allowed */ BADVERSION(s,errstr,"Invalid version format (no leading zeros)"); } while (isDIGIT(*d)) /* integer part */ d++; if (*d == '.') { saw_decimal++; d++; /* decimal point */ } else { if (strict) { /* require v1.2.3 */ BADVERSION(s,errstr,"Invalid version format (dotted-decimal versions require at least three parts)"); } else { goto version_prescan_finish; } } { int i = 0; int j = 0; while (isDIGIT(*d)) { /* just keep reading */ i++; while (isDIGIT(*d)) { d++; j++; /* maximum 3 digits between decimal */ if (strict && j > 3) { BADVERSION(s,errstr,"Invalid version format (maximum 3 digits between decimals)"); } } if (*d == '_') { if (strict) { BADVERSION(s,errstr,"Invalid version format (no underscores)"); } if ( alpha ) { BADVERSION(s,errstr,"Invalid version format (multiple underscores)"); } d++; alpha = TRUE; } else if (*d == '.') { if (alpha) { BADVERSION(s,errstr,"Invalid version format (underscores before decimal)"); } saw_decimal++; d++; } else if (!isDIGIT(*d)) { break; } j = 0; } if (strict && i < 2) { /* requires v1.2.3 */ BADVERSION(s,errstr,"Invalid version format (dotted-decimal versions require at least three parts)"); } } } /* end if dotted-decimal */ else { /* decimal versions */ int j = 0; /* may need this later */ /* special strict case for leading '.' or '0' */ if (strict) { if (*d == '.') { BADVERSION(s,errstr,"Invalid version format (0 before decimal required)"); } if (*d == '0' && isDIGIT(d[1])) { BADVERSION(s,errstr,"Invalid version format (no leading zeros)"); } } /* and we never support negative versions */ if ( *d == '-') { BADVERSION(s,errstr,"Invalid version format (negative version number)"); } /* consume all of the integer part */ while (isDIGIT(*d)) d++; /* look for a fractional part */ if (*d == '.') { /* we found it, so consume it */ saw_decimal++; d++; } else if (!*d || *d == ';' || isSPACE(*d) || *d == '{' || *d == '}') { if ( d == s ) { /* found nothing */ BADVERSION(s,errstr,"Invalid version format (version required)"); } /* found just an integer */ goto version_prescan_finish; } else if ( d == s ) { /* didn't find either integer or period */ BADVERSION(s,errstr,"Invalid version format (non-numeric data)"); } else if (*d == '_') { /* underscore can't come after integer part */ if (strict) { BADVERSION(s,errstr,"Invalid version format (no underscores)"); } else if (isDIGIT(d[1])) { BADVERSION(s,errstr,"Invalid version format (alpha without decimal)"); } else { BADVERSION(s,errstr,"Invalid version format (misplaced underscore)"); } } else { /* anything else after integer part is just invalid data */ BADVERSION(s,errstr,"Invalid version format (non-numeric data)"); } /* scan the fractional part after the decimal point*/ if (!isDIGIT(*d) && (strict || ! (!*d || *d == ';' || isSPACE(*d) || *d == '{' || *d == '}') )) { /* strict or lax-but-not-the-end */ BADVERSION(s,errstr,"Invalid version format (fractional part required)"); } while (isDIGIT(*d)) { d++; j++; if (*d == '.' && isDIGIT(d[-1])) { if (alpha) { BADVERSION(s,errstr,"Invalid version format (underscores before decimal)"); } if (strict) { BADVERSION(s,errstr,"Invalid version format (dotted-decimal versions must begin with 'v')"); } d = (char *)s; /* start all over again */ qv = TRUE; goto dotted_decimal_version; } if (*d == '_') { if (strict) { BADVERSION(s,errstr,"Invalid version format (no underscores)"); } if ( alpha ) { BADVERSION(s,errstr,"Invalid version format (multiple underscores)"); } if ( ! isDIGIT(d[1]) ) { BADVERSION(s,errstr,"Invalid version format (misplaced underscore)"); } width = j; d++; alpha = TRUE; } } } version_prescan_finish: while (isSPACE(*d)) d++; if (!isDIGIT(*d) && (! (!*d || *d == ';' || *d == '{' || *d == '}') )) { /* trailing non-numeric data */ BADVERSION(s,errstr,"Invalid version format (non-numeric data)"); } if (saw_decimal > 1 && d[-1] == '.') { /* no trailing period allowed */ BADVERSION(s,errstr,"Invalid version format (trailing decimal)"); } if (sqv) *sqv = qv; if (swidth) *swidth = width; if (ssaw_decimal) *ssaw_decimal = saw_decimal; if (salpha) *salpha = alpha; return d; } /* =for apidoc scan_version Returns a pointer to the next character after the parsed version string, as well as upgrading the passed in SV to an RV. Function must be called with an already existing SV like sv = newSV(0); s = scan_version(s, SV *sv, bool qv); Performs some preprocessing to the string to ensure that it has the correct characteristics of a version. Flags the object if it contains an underscore (which denotes this is an alpha version). The boolean qv denotes that the version should be interpreted as if it had multiple decimals, even if it doesn't. =cut */ const char * #ifdef VUTIL_REPLACE_CORE Perl_scan_version2(pTHX_ const char *s, SV *rv, bool qv) #else Perl_scan_version(pTHX_ const char *s, SV *rv, bool qv) #endif { const char *start = s; const char *pos; const char *last; const char *errstr = NULL; int saw_decimal = 0; int width = 3; bool alpha = FALSE; bool vinf = FALSE; AV * av; SV * hv; PERL_ARGS_ASSERT_SCAN_VERSION; while (isSPACE(*s)) /* leading whitespace is OK */ s++; last = PRESCAN_VERSION(s, FALSE, &errstr, &qv, &saw_decimal, &width, &alpha); if (errstr) { /* "undef" is a special case and not an error */ if ( ! ( *s == 'u' && strEQ(s+1,"ndef")) ) { Perl_croak(aTHX_ "%s", errstr); } } start = s; if (*s == 'v') s++; pos = s; /* Now that we are through the prescan, start creating the object */ av = newAV(); hv = newSVrv(rv, "version"); /* create an SV and upgrade the RV */ (void)sv_upgrade(hv, SVt_PVHV); /* needs to be an HV type */ #ifndef NODEFAULT_SHAREKEYS HvSHAREKEYS_on(hv); /* key-sharing on by default */ #endif if ( qv ) (void)hv_stores(MUTABLE_HV(hv), "qv", newSViv(qv)); if ( alpha ) (void)hv_stores(MUTABLE_HV(hv), "alpha", newSViv(alpha)); if ( !qv && width < 3 ) (void)hv_stores(MUTABLE_HV(hv), "width", newSViv(width)); while (isDIGIT(*pos) || *pos == '_') pos++; if (!isALPHA(*pos)) { I32 rev; for (;;) { rev = 0; { /* this is atoi() that delimits on underscores */ const char *end = pos; I32 mult = 1; I32 orev; /* the following if() will only be true after the decimal * point of a version originally created with a bare * floating point number, i.e. not quoted in any way */ if ( !qv && s > start && saw_decimal == 1 ) { mult *= 100; while ( s < end ) { if (*s == '_') continue; orev = rev; rev += (*s - '0') * mult; mult /= 10; if ( (PERL_ABS(orev) > PERL_ABS(rev)) || (PERL_ABS(rev) > VERSION_MAX )) { Perl_ck_warner(aTHX_ packWARN(WARN_OVERFLOW), "Integer overflow in version %d",VERSION_MAX); s = end - 1; rev = VERSION_MAX; vinf = 1; } s++; if ( *s == '_' ) s++; } } else { while (--end >= s) { int i; if (*end == '_') continue; i = (*end - '0'); if ( (mult == VERSION_MAX) || (i > VERSION_MAX / mult) || (i * mult > VERSION_MAX - rev)) { Perl_ck_warner(aTHX_ packWARN(WARN_OVERFLOW), "Integer overflow in version"); end = s - 1; rev = VERSION_MAX; vinf = 1; } else rev += i * mult; if (mult > VERSION_MAX / 10) mult = VERSION_MAX; else mult *= 10; } } } /* Append revision */ av_push(av, newSViv(rev)); if ( vinf ) { s = last; break; } else if ( *pos == '.' ) { pos++; if (qv) { while (*pos == '0') ++pos; } s = pos; } else if ( *pos == '_' && isDIGIT(pos[1]) ) s = ++pos; else if ( *pos == ',' && isDIGIT(pos[1]) ) s = ++pos; else if ( isDIGIT(*pos) ) s = pos; else { s = pos; break; } if ( qv ) { while ( isDIGIT(*pos) || *pos == '_') pos++; } else { int digits = 0; while ( ( isDIGIT(*pos) || *pos == '_' ) && digits < 3 ) { if ( *pos != '_' ) digits++; pos++; } } } } if ( qv ) { /* quoted versions always get at least three terms*/ SSize_t len = AvFILLp(av); /* This for loop appears to trigger a compiler bug on OS X, as it loops infinitely. Yes, len is negative. No, it makes no sense. Compiler in question is: gcc version 3.3 20030304 (Apple Computer, Inc. build 1640) for ( len = 2 - len; len > 0; len-- ) av_push(MUTABLE_AV(sv), newSViv(0)); */ len = 2 - len; while (len-- > 0) av_push(av, newSViv(0)); } /* need to save off the current version string for later */ if ( vinf ) { SV * orig = newSVpvn("v.Inf", sizeof("v.Inf")-1); (void)hv_stores(MUTABLE_HV(hv), "original", orig); (void)hv_stores(MUTABLE_HV(hv), "vinf", newSViv(1)); } else if ( s > start ) { SV * orig = newSVpvn(start,s-start); if ( qv && saw_decimal == 1 && *start != 'v' ) { /* need to insert a v to be consistent */ sv_insert(orig, 0, 0, "v", 1); } (void)hv_stores(MUTABLE_HV(hv), "original", orig); } else { (void)hv_stores(MUTABLE_HV(hv), "original", newSVpvs("0")); av_push(av, newSViv(0)); } /* And finally, store the AV in the hash */ (void)hv_stores(MUTABLE_HV(hv), "version", newRV_noinc(MUTABLE_SV(av))); /* fix RT#19517 - special case 'undef' as string */ if ( *s == 'u' && strEQ(s+1,"ndef") ) { s += 5; } return s; } /* =for apidoc new_version Returns a new version object based on the passed in SV: SV *sv = new_version(SV *ver); Does not alter the passed in ver SV. See "upg_version" if you want to upgrade the SV. =cut */ SV * #ifdef VUTIL_REPLACE_CORE Perl_new_version2(pTHX_ SV *ver) #else Perl_new_version(pTHX_ SV *ver) #endif { SV * const rv = newSV(0); PERL_ARGS_ASSERT_NEW_VERSION; if ( ISA_VERSION_OBJ(ver) ) /* can just copy directly */ { SSize_t key; AV * const av = newAV(); AV *sav; /* This will get reblessed later if a derived class*/ SV * const hv = newSVrv(rv, "version"); (void)sv_upgrade(hv, SVt_PVHV); /* needs to be an HV type */ #ifndef NODEFAULT_SHAREKEYS HvSHAREKEYS_on(hv); /* key-sharing on by default */ #endif if ( SvROK(ver) ) ver = SvRV(ver); /* Begin copying all of the elements */ if ( hv_exists(MUTABLE_HV(ver), "qv", 2) ) (void)hv_stores(MUTABLE_HV(hv), "qv", newSViv(1)); if ( hv_exists(MUTABLE_HV(ver), "alpha", 5) ) (void)hv_stores(MUTABLE_HV(hv), "alpha", newSViv(1)); { SV ** svp = hv_fetchs(MUTABLE_HV(ver), "width", FALSE); if(svp) { const I32 width = SvIV(*svp); (void)hv_stores(MUTABLE_HV(hv), "width", newSViv(width)); } } { SV ** svp = hv_fetchs(MUTABLE_HV(ver), "original", FALSE); if(svp) (void)hv_stores(MUTABLE_HV(hv), "original", newSVsv(*svp)); } sav = MUTABLE_AV(SvRV(*hv_fetchs(MUTABLE_HV(ver), "version", FALSE))); /* This will get reblessed later if a derived class*/ for ( key = 0; key <= av_len(sav); key++ ) { SV * const sv = *av_fetch(sav, key, FALSE); const I32 rev = SvIV(sv); av_push(av, newSViv(rev)); } (void)hv_stores(MUTABLE_HV(hv), "version", newRV_noinc(MUTABLE_SV(av))); return rv; } #ifdef SvVOK { const MAGIC* const mg = SvVSTRING_mg(ver); if ( mg ) { /* already a v-string */ const STRLEN len = mg->mg_len; const char * const version = (const char*)mg->mg_ptr; char *raw, *under; static const char underscore[] = "_"; sv_setpvn(rv,version,len); raw = SvPV_nolen(rv); under = ninstr(raw, raw+len, underscore, underscore + 1); if (under) { Move(under + 1, under, raw + len - under - 1, char); SvCUR(rv)--; *SvEND(rv) = '\0'; } /* this is for consistency with the pure Perl class */ if ( isDIGIT(*version) ) sv_insert(rv, 0, 0, "v", 1); } else { #endif SvSetSV_nosteal(rv, ver); /* make a duplicate */ #ifdef SvVOK } } #endif sv_2mortal(rv); /* in case upg_version croaks before it returns */ return SvREFCNT_inc_NN(UPG_VERSION(rv, FALSE)); } /* =for apidoc upg_version In-place upgrade of the supplied SV to a version object. SV *sv = upg_version(SV *sv, bool qv); Returns a pointer to the upgraded SV. Set the boolean qv if you want to force this SV to be interpreted as an "extended" version. =cut */ SV * #ifdef VUTIL_REPLACE_CORE Perl_upg_version2(pTHX_ SV *ver, bool qv) #else Perl_upg_version(pTHX_ SV *ver, bool qv) #endif { const char *version, *s; #ifdef SvVOK const MAGIC *mg; #endif #if PERL_VERSION_LT(5,19,8) && defined(USE_ITHREADS) ENTER; #endif PERL_ARGS_ASSERT_UPG_VERSION; if ( (SvUOK(ver) && SvUVX(ver) > VERSION_MAX) || (SvIOK(ver) && SvIVX(ver) > VERSION_MAX) ) { /* out of bounds [unsigned] integer */ STRLEN len; char tbuf[64]; len = my_snprintf(tbuf, sizeof(tbuf), "%d", VERSION_MAX); version = savepvn(tbuf, len); SAVEFREEPV(version); Perl_ck_warner(aTHX_ packWARN(WARN_OVERFLOW), "Integer overflow in version %d",VERSION_MAX); } else if ( SvUOK(ver) || SvIOK(ver)) #if PERL_VERSION_LT(5,17,2) VER_IV: #endif { version = savesvpv(ver); SAVEFREEPV(version); } else if (SvNOK(ver) && !( SvPOK(ver) && SvCUR(ver) == 3 ) ) #if PERL_VERSION_LT(5,17,2) VER_NV: #endif { STRLEN len; /* may get too much accuracy */ char tbuf[64]; #ifdef __vax__ SV *sv = SvNVX(ver) > 10e37 ? newSV(64) : 0; #else SV *sv = SvNVX(ver) > 10e50 ? newSV(64) : 0; #endif char *buf; #if PERL_VERSION_GE(5,19,0) if (SvPOK(ver)) { /* dualvar? */ goto VER_PV; } #endif #ifdef USE_LOCALE_NUMERIC { /* This may or may not be called from code that has switched * locales without letting perl know, therefore we have to find it * from first principals. See [perl #121930]. */ /* In windows, or not threaded, or not thread-safe, if it isn't C, * set it to C. */ # ifndef USE_POSIX_2008_LOCALE const char * locale_name_on_entry; LC_NUMERIC_LOCK(0); /* Start critical section */ locale_name_on_entry = setlocale(LC_NUMERIC, NULL); if ( strNE(locale_name_on_entry, "C") && strNE(locale_name_on_entry, "POSIX")) { setlocale(LC_NUMERIC, "C"); } else { /* This value indicates to the restore code that we didn't change the locale */ locale_name_on_entry = NULL; } # else const locale_t locale_obj_on_entry = uselocale((locale_t) 0); const char * locale_name_on_entry = NULL; DECLARATION_FOR_LC_NUMERIC_MANIPULATION; if (locale_obj_on_entry == LC_GLOBAL_LOCALE) { /* in the global locale, we can call system setlocale and if it * isn't C, set it to C. */ LC_NUMERIC_LOCK(0); locale_name_on_entry = setlocale(LC_NUMERIC, NULL); if ( strNE(locale_name_on_entry, "C") && strNE(locale_name_on_entry, "POSIX")) { setlocale(LC_NUMERIC, "C"); } else { /* This value indicates to the restore code that we didn't change the locale */ locale_name_on_entry = NULL; } } else if (locale_obj_on_entry == PL_underlying_numeric_obj) { /* Here, the locale appears to have been changed to use the * program's underlying locale. Just use our mechanisms to * switch back to C. It might be possible for this pointer to * actually refer to something else if it got released and * reused somehow. But it doesn't matter, our mechanisms will * work even so */ STORE_LC_NUMERIC_SET_STANDARD(); } else if (locale_obj_on_entry != PL_C_locale_obj) { /* The C object should be unchanged during a program's * execution, so it should be safe to assume it means what it * says, so if we are in it, no locale change is required. * Otherwise, simply use the thread-safe operation. */ uselocale(PL_C_locale_obj); } # endif /* Prevent recursed calls from trying to change back */ LOCK_LC_NUMERIC_STANDARD(); #endif if (sv) { Perl_sv_catpvf(aTHX_ sv, "%.9" NVff, SvNVX(ver)); len = SvCUR(sv); buf = SvPVX(sv); } else { len = my_snprintf(tbuf, sizeof(tbuf), "%.9" NVff, SvNVX(ver)); buf = tbuf; } #ifdef USE_LOCALE_NUMERIC UNLOCK_LC_NUMERIC_STANDARD(); # ifndef USE_POSIX_2008_LOCALE if (locale_name_on_entry) { setlocale(LC_NUMERIC, locale_name_on_entry); } LC_NUMERIC_UNLOCK; /* End critical section */ # else if (locale_name_on_entry) { setlocale(LC_NUMERIC, locale_name_on_entry); LC_NUMERIC_UNLOCK; } else if (locale_obj_on_entry == PL_underlying_numeric_obj) { RESTORE_LC_NUMERIC(); } else if (locale_obj_on_entry != PL_C_locale_obj) { uselocale(locale_obj_on_entry); } # endif } #endif /* USE_LOCALE_NUMERIC */ while (buf[len-1] == '0' && len > 0) len--; if ( buf[len-1] == '.' ) len--; /* eat the trailing decimal */ version = savepvn(buf, len); SAVEFREEPV(version); SvREFCNT_dec(sv); } #ifdef SvVOK else if ( (mg = SvVSTRING_mg(ver)) ) { /* already a v-string */ version = savepvn( (const char*)mg->mg_ptr,mg->mg_len ); SAVEFREEPV(version); qv = TRUE; } #endif else if ( SvPOK(ver))/* must be a string or something like a string */ VER_PV: { STRLEN len; version = savepvn(SvPV(ver,len), SvCUR(ver)); SAVEFREEPV(version); #ifndef SvVOK # if PERL_VERSION > 5 /* This will only be executed for 5.6.0 - 5.8.0 inclusive */ if ( len >= 3 && !instr(version,".") && !instr(version,"_")) { /* may be a v-string */ char *testv = (char *)version; STRLEN tlen = len; for (tlen=0; tlen < len; tlen++, testv++) { /* if one of the characters is non-text assume v-string */ if (testv[0] < ' ') { SV * const nsv = sv_newmortal(); const char *nver; const char *pos; int saw_decimal = 0; sv_setpvf(nsv,"v%vd",ver); pos = nver = savepv(SvPV_nolen(nsv)); SAVEFREEPV(pos); /* scan the resulting formatted string */ pos++; /* skip the leading 'v' */ while ( *pos == '.' || isDIGIT(*pos) ) { if ( *pos == '.' ) saw_decimal++ ; pos++; } /* is definitely a v-string */ if ( saw_decimal >= 2 ) { version = nver; } break; } } } # endif #endif } #if PERL_VERSION_LT(5,17,2) else if (SvIOKp(ver)) { goto VER_IV; } else if (SvNOKp(ver)) { goto VER_NV; } else if (SvPOKp(ver)) { goto VER_PV; } #endif else { /* no idea what this is */ Perl_croak(aTHX_ "Invalid version format (non-numeric data)"); } s = SCAN_VERSION(version, ver, qv); if ( *s != '\0' ) Perl_ck_warner(aTHX_ packWARN(WARN_MISC), "Version string '%s' contains invalid data; " "ignoring: '%s'", version, s); #if PERL_VERSION_LT(5,19,8) && defined(USE_ITHREADS) LEAVE; #endif return ver; } /* =for apidoc vverify Validates that the SV contains valid internal structure for a version object. It may be passed either the version object (RV) or the hash itself (HV). If the structure is valid, it returns the HV. If the structure is invalid, it returns NULL. SV *hv = vverify(sv); Note that it only confirms the bare minimum structure (so as not to get confused by derived classes which may contain additional hash entries): =over 4 =item * The SV is an HV or a reference to an HV =item * The hash contains a "version" key =item * The "version" key has a reference to an AV as its value =back =cut */ SV * #ifdef VUTIL_REPLACE_CORE Perl_vverify2(pTHX_ SV *vs) #else Perl_vverify(pTHX_ SV *vs) #endif { SV *sv; SV **svp; PERL_ARGS_ASSERT_VVERIFY; if ( SvROK(vs) ) vs = SvRV(vs); /* see if the appropriate elements exist */ if ( SvTYPE(vs) == SVt_PVHV && (svp = hv_fetchs(MUTABLE_HV(vs), "version", FALSE)) && (sv = SvRV(*svp)) && SvTYPE(sv) == SVt_PVAV ) return vs; else return NULL; } /* =for apidoc vnumify Accepts a version object and returns the normalized floating point representation. Call like: sv = vnumify(rv); NOTE: you can pass either the object directly or the SV contained within the RV. The SV returned has a refcount of 1. =cut */ SV * #ifdef VUTIL_REPLACE_CORE Perl_vnumify2(pTHX_ SV *vs) #else Perl_vnumify(pTHX_ SV *vs) #endif { SSize_t i, len; I32 digit; bool alpha = FALSE; SV *sv; AV *av; PERL_ARGS_ASSERT_VNUMIFY; /* extract the HV from the object */ vs = VVERIFY(vs); if ( ! vs ) Perl_croak(aTHX_ "Invalid version object"); /* see if various flags exist */ if ( hv_exists(MUTABLE_HV(vs), "alpha", 5 ) ) alpha = TRUE; if (alpha) { Perl_ck_warner(aTHX_ packWARN(WARN_NUMERIC), "alpha->numify() is lossy"); } /* attempt to retrieve the version array */ if ( !(av = MUTABLE_AV(SvRV(*hv_fetchs(MUTABLE_HV(vs), "version", FALSE))) ) ) { return newSVpvs("0"); } len = av_len(av); if ( len == -1 ) { return newSVpvs("0"); } { SV * tsv = *av_fetch(av, 0, 0); digit = SvIV(tsv); } sv = Perl_newSVpvf(aTHX_ "%d.", (int)PERL_ABS(digit)); for ( i = 1 ; i <= len ; i++ ) { SV * tsv = *av_fetch(av, i, 0); digit = SvIV(tsv); Perl_sv_catpvf(aTHX_ sv, "%03d", (int)digit); } if ( len == 0 ) { sv_catpvs(sv, "000"); } return sv; } /* =for apidoc vnormal Accepts a version object and returns the normalized string representation. Call like: sv = vnormal(rv); NOTE: you can pass either the object directly or the SV contained within the RV. The SV returned has a refcount of 1. =cut */ SV * #ifdef VUTIL_REPLACE_CORE Perl_vnormal2(pTHX_ SV *vs) #else Perl_vnormal(pTHX_ SV *vs) #endif { I32 i, len, digit; SV *sv; AV *av; PERL_ARGS_ASSERT_VNORMAL; /* extract the HV from the object */ vs = VVERIFY(vs); if ( ! vs ) Perl_croak(aTHX_ "Invalid version object"); av = MUTABLE_AV(SvRV(*hv_fetchs(MUTABLE_HV(vs), "version", FALSE))); len = av_len(av); if ( len == -1 ) { return newSVpvs(""); } { SV * tsv = *av_fetch(av, 0, 0); digit = SvIV(tsv); } sv = Perl_newSVpvf(aTHX_ "v%" IVdf, (IV)digit); for ( i = 1 ; i <= len ; i++ ) { SV * tsv = *av_fetch(av, i, 0); digit = SvIV(tsv); Perl_sv_catpvf(aTHX_ sv, ".%" IVdf, (IV)digit); } if ( len <= 2 ) { /* short version, must be at least three */ for ( len = 2 - len; len != 0; len-- ) sv_catpvs(sv,".0"); } return sv; } /* =for apidoc vstringify In order to maintain maximum compatibility with earlier versions of Perl, this function will return either the floating point notation or the multiple dotted notation, depending on whether the original version contained 1 or more dots, respectively. The SV returned has a refcount of 1. =cut */ SV * #ifdef VUTIL_REPLACE_CORE Perl_vstringify2(pTHX_ SV *vs) #else Perl_vstringify(pTHX_ SV *vs) #endif { SV ** svp; PERL_ARGS_ASSERT_VSTRINGIFY; /* extract the HV from the object */ vs = VVERIFY(vs); if ( ! vs ) Perl_croak(aTHX_ "Invalid version object"); svp = hv_fetchs(MUTABLE_HV(vs), "original", FALSE); if (svp) { SV *pv; pv = *svp; if ( SvPOK(pv) #if PERL_VERSION_LT(5,17,2) || SvPOKp(pv) #endif ) return newSVsv(pv); else return &PL_sv_undef; } else { if ( hv_exists(MUTABLE_HV(vs), "qv", 2) ) return VNORMAL(vs); else return VNUMIFY(vs); } } /* =for apidoc vcmp Version object aware cmp. Both operands must already have been converted into version objects. =cut */ int #ifdef VUTIL_REPLACE_CORE Perl_vcmp2(pTHX_ SV *lhv, SV *rhv) #else Perl_vcmp(pTHX_ SV *lhv, SV *rhv) #endif { SSize_t i,l,m,r; I32 retval; I32 left = 0; I32 right = 0; AV *lav, *rav; PERL_ARGS_ASSERT_VCMP; /* extract the HVs from the objects */ lhv = VVERIFY(lhv); rhv = VVERIFY(rhv); if ( ! ( lhv && rhv ) ) Perl_croak(aTHX_ "Invalid version object"); /* get the left hand term */ lav = MUTABLE_AV(SvRV(*hv_fetchs(MUTABLE_HV(lhv), "version", FALSE))); /* and the right hand term */ rav = MUTABLE_AV(SvRV(*hv_fetchs(MUTABLE_HV(rhv), "version", FALSE))); l = av_len(lav); r = av_len(rav); m = l < r ? l : r; retval = 0; i = 0; while ( i <= m && retval == 0 ) { SV * const lsv = *av_fetch(lav,i,0); SV * rsv; left = SvIV(lsv); rsv = *av_fetch(rav,i,0); right = SvIV(rsv); if ( left < right ) retval = -1; if ( left > right ) retval = +1; i++; } if ( l != r && retval == 0 ) /* possible match except for trailing 0's */ { if ( l < r ) { while ( i <= r && retval == 0 ) { SV * const rsv = *av_fetch(rav,i,0); if ( SvIV(rsv) != 0 ) retval = -1; /* not a match after all */ i++; } } else { while ( i <= l && retval == 0 ) { SV * const lsv = *av_fetch(lav,i,0); if ( SvIV(lsv) != 0 ) retval = +1; /* not a match after all */ i++; } } } return retval; } /* ex: set ro: */