LCOV - code coverage report
Current view: top level - libpkg - pkg_version.c (source / functions) Hit Total Coverage
Test: plop Lines: 107 126 84.9 %
Date: 2024-12-30 07:09:03 Functions: 4 4 100.0 %
Legend: Lines: hit not hit | Branches: + taken - not taken # not executed Branches: 86 120 71.7 %

           Branch data     Line data    Source code
       1                 :            : /*-
       2                 :            :  * Copyright (c) 2011 Philippe Pepiot <phil@philpep.org>
       3                 :            :  * Copyright (c) 2014 Vsevolod Stakhov <vsevolod@FreeBSD.org>
       4                 :            :  * All rights reserved.
       5                 :            :  * 
       6                 :            :  * Redistribution and use in source and binary forms, with or without
       7                 :            :  * modification, are permitted provided that the following conditions
       8                 :            :  * are met:
       9                 :            :  * 1. Redistributions of source code must retain the above copyright
      10                 :            :  *    notice, this list of conditions and the following disclaimer
      11                 :            :  *    in this position and unchanged.
      12                 :            :  * 2. Redistributions in binary form must reproduce the above copyright
      13                 :            :  *    notice, this list of conditions and the following disclaimer in the
      14                 :            :  *    documentation and/or other materials provided with the distribution.
      15                 :            :  * 
      16                 :            :  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR
      17                 :            :  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
      18                 :            :  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
      19                 :            :  * IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT, INDIRECT,
      20                 :            :  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
      21                 :            :  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
      22                 :            :  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
      23                 :            :  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
      24                 :            :  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
      25                 :            :  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
      26                 :            :  */
      27                 :            : 
      28                 :            : 
      29                 :            : #include <assert.h>
      30                 :            : #include <string.h>
      31                 :            : #include <stdlib.h>
      32                 :            : #include <ctype.h>
      33                 :            : 
      34                 :            : #include "pkg.h"
      35                 :            : #include "private/event.h"
      36                 :            : #include "private/pkg.h"
      37                 :            : 
      38                 :            : /*
      39                 :            :  * split_version(pkgname, endname, epoch, revision) returns a pointer to
      40                 :            :  * the version portion of a package name and the two special components.
      41                 :            :  *
      42                 :            :  * Syntax is:  ${PORTNAME}-${PORTVERSION}[_${PORTREVISION}][,${PORTEPOCH}]
      43                 :            :  *
      44                 :            :  * Written by Oliver Eikemeier
      45                 :            :  * Based on work of Jeremy D. Lea.
      46                 :            :  */
      47                 :            : static const char *
      48                 :        554 : split_version(const char *pkgname, const char **endname,
      49                 :            :     unsigned long *epoch, unsigned long *revision)
      50                 :            : {
      51                 :            :         char *ch;
      52                 :            :         const char *versionstr;
      53                 :            :         const char *endversionstr;
      54                 :            : 
      55         [ +  - ]:        554 :         if (pkgname == NULL) {
      56                 :          0 :                 pkg_emit_error("%s: Passed NULL pkgname.", __func__);
      57                 :          0 :                 return (NULL);
      58                 :            :         }
      59                 :            : 
      60                 :            :         /* Look for the last '-' the the pkgname */
      61                 :        554 :         ch = strrchr(pkgname, '-');
      62                 :            :         /* Cheat if we are just passed a version, not a valid package name */
      63         [ -  + ]:        554 :         versionstr = ch ? ch + 1 : pkgname;
      64                 :            : 
      65                 :            :         /*
      66                 :            :          * Look for the last '_' in the version string, advancing the
      67                 :            :          * end pointer.
      68                 :            :          */
      69                 :        554 :         ch = strrchr(versionstr, '_');
      70                 :            : 
      71         [ -  + ]:        554 :         if (revision != NULL)
      72         [ +  + ]:        554 :                 *revision = ch ? strtoul(ch + 1, NULL, 10) : 0;
      73                 :            : 
      74                 :        554 :         endversionstr = ch;
      75                 :            : 
      76                 :            :         /* Look for the last ',' in the remaining version string */
      77         [ +  + ]:        554 :         ch = strrchr(endversionstr ? endversionstr + 1 : versionstr, ',');
      78                 :            : 
      79         [ -  + ]:        554 :         if (epoch != NULL)
      80         [ +  + ]:        554 :                 *epoch = ch ? strtoul(ch + 1, NULL, 10) : 0;
      81                 :            : 
      82   [ +  +  -  + ]:        554 :         if (ch && !endversionstr)
      83                 :          1 :                 endversionstr = ch;
      84                 :            : 
      85                 :            :         /*
      86                 :            :          * set the pointer behind the last character of the version without
      87                 :            :          * revision or epoch
      88                 :            :          */
      89         [ -  + ]:        554 :         if (endname)
      90         [ +  + ]:        554 :                 *endname = endversionstr ? endversionstr :
      91                 :        479 :                                            strrchr(versionstr, '\0');
      92                 :            : 
      93                 :        554 :         return versionstr;
      94                 :        554 : }
      95                 :            : 
      96                 :            : /*
      97                 :            :  * PORTVERSIONs are composed of components separated by dots. A component
      98                 :            :  * consists of a version number, a letter and a patchlevel number. This does
      99                 :            :  * not conform to the porter's handbook, but let us formulate rules that
     100                 :            :  * fit the current practice and are far simpler than to make decisions
     101                 :            :  * based on the order of netters and lumbers. Besides, people use versions
     102                 :            :  * like 10b2 in the ports...
     103                 :            :  */
     104                 :            : 
     105                 :            : typedef struct {
     106                 :            : #ifdef __LONG_LONG_SUPPORTED
     107                 :            :         long long n;
     108                 :            :         long long pl;
     109                 :            : #else
     110                 :            :         long n;
     111                 :            :         long pl;
     112                 :            : #endif
     113                 :            :         int a;
     114                 :            : } version_component;
     115                 :            : 
     116                 :            : /*
     117                 :            :  * get_component(position, component) gets the value of the next component
     118                 :            :  * (number - letter - number triple) and returns a pointer to the next character
     119                 :            :  * after any leading separators
     120                 :            :  *
     121                 :            :  * - components are separated by dots
     122                 :            :  * - characters !~ [a-zA-Z0-9.+*] are treated as separators
     123                 :            :  *   (1.0:2003.09.16 = 1.0.2003.09.16), this may not be what you expect:
     124                 :            :  *   1.0.1:2003.09.16 < 1.0:2003.09.16
     125                 :            :  * - consecutive separators are collapsed (10..1 = 10.1)
     126                 :            :  * - missing separators are inserted, essentially
     127                 :            :  *   letter number letter => letter number . letter (10a1b2 = 10a1.b2)
     128                 :            :  * - missing components are assumed to be equal to 0 (10 = 10.0 = 10.0.0)
     129                 :            :  * - the letter sort order is: [none], a, b, ..., z; numbers without letters
     130                 :            :  *   sort first (10 < 10a < 10b)
     131                 :            :  * - missing version numbers (in components starting with a letter) sort as -1
     132                 :            :  *   (a < 0, 10.a < 10)
     133                 :            :  * - a separator is inserted before the special strings "pl", "alpha", "beta",
     134                 :            :  *   "pre" and "rc".
     135                 :            :  * - "pl" sorts before every other letter, "alpha", "beta", "pre" and "rc"
     136                 :            :  *   sort as a, b, p and r. (10alpha = 10.a < 10, but 10 < 10a; pl11 < alpha3
     137                 :            :  *   < 0.1beta2 = 0.1.b2 < 0.1)
     138                 :            :  * - other strings use only the first letter for sorting, case is ignored
     139                 :            :  *   (1.d2 = 1.dev2 = 1.Development2)
     140                 :            :  * - The special component `*' is guaranteed to be the smallest possible
     141                 :            :  *   component (2.* < 2pl1 < 2alpha3 < 2.9f7 < 3.*)
     142                 :            :  * - components separated by `+' are handled by version_cmp below
     143                 :            :  *
     144                 :            :  * Oliver Eikemeier
     145                 :            :  */
     146                 :            : 
     147                 :            : static const struct stage {
     148                 :            :         const char *name;
     149                 :            :         size_t namelen;
     150                 :            :         int value;
     151                 :            : } stages[] = {
     152                 :            :         { "pl",    2,  0            },
     153                 :            :         { "snap",  6,  1            },
     154                 :            : #define ABASE   2       /* Last special early-sorted prefix + 1 */
     155                 :            :         { "alpha", 5, 'a'-'a'+ABASE },
     156                 :            :         { "beta",  4, 'b'-'a'+ABASE },
     157                 :            :         { "pre",   3, 'p'-'a'+ABASE },
     158                 :            :         { "rc",    2, 'r'-'a'+ABASE },
     159                 :            : };
     160                 :            : 
     161                 :            : static const char *
     162                 :        452 : get_component(const char *position, version_component *component)
     163                 :            : {
     164                 :        452 :         const char *pos = position;
     165                 :        452 :         int hasstage = 0, haspatchlevel = 0;
     166                 :            : 
     167         [ +  - ]:        452 :         if (!pos) {
     168                 :          0 :                 pkg_emit_error("%s: Passed NULL position.", __func__);
     169                 :          0 :                 return (NULL);
     170                 :            :         }
     171                 :            : 
     172                 :            :         /* handle version number */
     173         [ +  + ]:        452 :         if (isdigit(*pos)) {
     174                 :            :                 char *endptr;
     175                 :            : #ifdef __LONG_LONG_SUPPORTED
     176                 :        439 :                 component->n = strtoll(pos, &endptr, 10);
     177                 :            : #else
     178                 :            :                 component->n = strtol(pos, &endptr, 10);
     179                 :            : #endif
     180                 :            :                 /* should we test for errno == ERANGE? */
     181                 :        439 :                 pos = endptr;
     182         [ -  + ]:        452 :         } else if (*pos == '*') {
     183                 :          0 :                 component->n = -2;
     184                 :          0 :                 do {
     185                 :          0 :                         pos++;
     186   [ #  #  #  #  :          0 :                 } while(*pos && *pos != '+');
                   #  # ]
     187                 :          0 :         } else {
     188                 :         13 :                 component->n = -1;
     189                 :         13 :                 hasstage = 1;
     190                 :            :         }
     191                 :            : 
     192                 :            :         /* handle letter */
     193         [ +  + ]:        452 :         if (isalpha(*pos)) {
     194                 :         13 :                 int c = tolower(*pos);
     195                 :         13 :                 haspatchlevel = 1;
     196                 :            :                 /* handle special suffixes */
     197         [ -  + ]:         13 :                 if (isalpha(pos[1])) {
     198                 :            :                         unsigned int i;
     199         [ +  + ]:         55 :                         for (i = 0; i < NELEM(stages); i++) {
     200                 :         53 :                                 const struct stage *stage = &stages[i];
     201                 :         53 :                                 size_t len = stage->namelen;
     202   [ +  +  -  + ]:         53 :                                 if (strncasecmp(pos, stage->name, len) == 0 &&
     203                 :         11 :                                     !isalpha(pos[stage->namelen])) {
     204         [ +  - ]:         11 :                                         if (hasstage) {
     205                 :            :                                                 /* stage to value */
     206                 :         11 :                                                 component->a = stage->value;
     207                 :         11 :                                                 pos += stage->namelen;
     208                 :         11 :                                         } else {
     209                 :            :                                                 /* insert dot */
     210                 :          0 :                                                 component->a = 0;
     211                 :          0 :                                                 haspatchlevel = 0;
     212                 :            :                                         }
     213                 :         11 :                                         c = 0;
     214                 :         11 :                                         break;
     215                 :            :                                 }
     216                 :         42 :                         }
     217                 :         13 :                 }
     218                 :            :                 /* unhandled above */
     219         [ +  + ]:         13 :                 if (c) {
     220                 :            :                         /* use the first letter and skip following */
     221                 :          2 :                         component->a = c - 'a' + ABASE;
     222                 :          8 :                         do {
     223                 :          8 :                                 ++pos;
     224         [ +  + ]:          8 :                         } while (isalpha(*pos));
     225                 :          2 :                 }
     226                 :         13 :         } else {
     227                 :        439 :                 component->a = 0;
     228                 :        439 :                 haspatchlevel = 0;
     229                 :            :         }
     230                 :            : 
     231         [ +  + ]:        452 :         if (haspatchlevel) {
     232                 :            :                 /* handle patch number */
     233         [ +  - ]:         13 :                 if (isdigit(*pos)) {
     234                 :            :                         char *endptr;
     235                 :            : #ifdef __LONG_LONG_SUPPORTED
     236                 :         13 :                         component->pl = strtoll(pos, &endptr, 10);
     237                 :            : #else
     238                 :            :                         component->pl = strtol(pos, &endptr, 10);
     239                 :            : #endif
     240                 :            :                         /* should we test for errno == ERANGE? */
     241                 :         13 :                         pos = endptr;
     242                 :         13 :                 } else {
     243                 :          0 :                         component->pl = -1;
     244                 :            :                 }
     245                 :         13 :         } else {
     246                 :        439 :                 component->pl = 0;
     247                 :            :         }
     248                 :            : 
     249                 :            :         /* skip trailing separators */
     250   [ +  +  +  +  :        806 :         while (*pos && !isdigit(*pos) && !isalpha(*pos)
                   +  + ]
     251   [ +  +  -  +  :        190 :             && *pos != '+' && *pos != '*')
                   -  + ]
     252                 :        177 :                 pos++;
     253                 :            : 
     254                 :        452 :         return pos;
     255                 :        452 : }
     256                 :            : 
     257                 :            : /*
     258                 :            :  * version_cmp(pkg1, pkg2) returns -1, 0 or 1 depending on if the version
     259                 :            :  * components of pkg1 is less than, equal to or greater than pkg2. No
     260                 :            :  * comparison of the basenames is done.
     261                 :            :  *
     262                 :            :  * The port version is defined by:
     263                 :            :  * ${PORTVERSION}[_${PORTREVISION}][,${PORTEPOCH}]
     264                 :            :  * ${PORTEPOCH} supersedes ${PORTVERSION} supersedes ${PORTREVISION}.
     265                 :            :  * See the commit log for revision 1.349 of ports/Mk/bsd.port.mk
     266                 :            :  * for more information.
     267                 :            :  *
     268                 :            :  * The epoch and revision are defined to be a single number, while the rest
     269                 :            :  * of the version should conform to the porting guidelines. It can contain
     270                 :            :  * multiple components, separated by a period, including letters.
     271                 :            :  *
     272                 :            :  * The tests allow for significantly more latitude in the version numbers
     273                 :            :  * than is allowed in the guidelines. No point in enforcing them here.
     274                 :            :  * That's what portlint is for.
     275                 :            :  *
     276                 :            :  * Jeremy D. Lea.
     277                 :            :  * reimplemented by Oliver Eikemeier
     278                 :            :  */
     279                 :            : int
     280                 :        277 : pkg_version_cmp(const char * const pkg1, const char * const pkg2)
     281                 :            : {
     282                 :            :         const char *v1, *v2, *ve1, *ve2;
     283                 :            :         unsigned long e1, e2, r1, r2;
     284                 :        277 :         int result = 0;
     285                 :            : 
     286                 :        277 :         v1 = split_version(pkg1, &ve1, &e1, &r1);
     287                 :        277 :         v2 = split_version(pkg2, &ve2, &e2, &r2);
     288                 :            : 
     289         [ +  - ]:        277 :         assert (v1 != NULL && v2 != NULL);
     290                 :            : 
     291                 :            :         /* Check epoch, port version, and port revision, in that order. */
     292         [ +  + ]:        277 :         if (e1 != e2)
     293                 :          1 :                 result = (e1 < e2 ? -1 : 1);
     294                 :            : 
     295                 :            :         /* Shortcut check for equality before invoking the parsing routines. */
     296   [ +  +  +  + ]:        488 :         if (result == 0 &&
     297         [ +  + ]:        276 :             (ve1 - v1 != ve2 - v2 || strncasecmp(v1, v2, ve1 - v1) != 0)) {
     298                 :            :                 /*
     299                 :            :                  * Loop over different components (the parts separated by dots).
     300                 :            :                  * If any component differs, there is an inequality.
     301                 :            :                  */
     302   [ +  +  +  +  :        662 :                 while (result == 0 && (v1 < ve1 || v2 < ve2)) {
             +  -  +  + ]
     303                 :        249 :                         int block_v1 = 0;
     304                 :        249 :                         int block_v2 = 0;
     305                 :        249 :                         version_component vc1 = {0, 0, 0};
     306                 :        249 :                         version_component vc2 = {0, 0, 0};
     307   [ +  +  -  + ]:        249 :                         if (v1 < ve1 && *v1 != '+') {
     308                 :        226 :                                 v1 = get_component(v1, &vc1);
     309         [ -  + ]:        226 :                                 assert (v1 != NULL);
     310                 :        226 :                         } else {
     311                 :         23 :                                 block_v1 = 1;
     312                 :            :                         }
     313   [ +  +  -  + ]:        249 :                         if (v2 < ve2 && *v2 != '+') {
     314                 :        226 :                                 v2 = get_component(v2, &vc2);
     315         [ -  + ]:        226 :                                 assert (v2 != NULL);
     316                 :        226 :                         } else {
     317                 :         23 :                                 block_v2 = 1;
     318                 :            :                         }
     319   [ +  +  +  - ]:        249 :                         if (block_v1 && block_v2) {
     320         [ #  # ]:          0 :                                 if (v1 < ve1)
     321                 :          0 :                                         v1++;
     322         [ #  # ]:          0 :                                 if (v2 < ve2)
     323                 :          0 :                                         v2++;
     324         [ +  + ]:        249 :                         } else if (vc1.n != vc2.n) {
     325                 :        158 :                                 result = (vc1.n < vc2.n ? -1 : 1);
     326         [ +  + ]:        249 :                         } else if (vc1.a != vc2.a) {
     327                 :          6 :                                 result = (vc1.a < vc2.a ? -1 : 1);
     328         [ +  - ]:         91 :                         } else if (vc1.pl != vc2.pl) {
     329                 :          0 :                                 result = (vc1.pl < vc2.pl ? -1 : 1);
     330                 :          0 :                         }
     331                 :            :                 }
     332                 :        164 :         }
     333                 :            : 
     334                 :            :         /* Compare FreeBSD revision numbers. */
     335   [ +  +  +  + ]:        277 :         if (result == 0 && r1 != r2)
     336                 :         24 :                 result = (r1 < r2 ? -1 : 1);
     337                 :            : 
     338                 :        277 :         return (result);
     339                 :            : }
     340                 :            : 
     341                 :            : pkg_change_t
     342                 :        178 : pkg_version_change_between(const struct pkg * pkg1, const struct pkg *pkg2)
     343                 :            : {
     344         [ +  - ]:        178 :         if (pkg2 == NULL)
     345                 :          0 :                 return PKG_REINSTALL;
     346                 :            : 
     347   [ +  -  +  + ]:        178 :         switch (pkg_version_cmp(pkg2->version, pkg1->version)) {
     348                 :            :         case -1:
     349                 :        142 :                 return (PKG_UPGRADE);
     350                 :            :         default:                /* placate the compiler */
     351                 :            :         case 0:
     352                 :         58 :                 return (PKG_REINSTALL);
     353                 :            :         case 1:
     354                 :         36 :                 return (PKG_DOWNGRADE);
     355                 :            :         }
     356                 :        178 : }

Generated by: LCOV version 1.15