Code coverage tests

This page documents the degree to which the PARI/GP source code is tested by our public test suite, distributed with the source distribution in directory src/test/. This is measured by the gcov utility; we then process gcov output using the lcov frond-end.

We test a few variants depending on Configure flags on the pari.math.u-bordeaux.fr machine (x86_64 architecture), and agregate them in the final report:

The target is to exceed 90% coverage for all mathematical modules (given that branches depending on DEBUGLEVEL or DEBUGMEM are not covered). This script is run to produce the results below.

LCOV - code coverage report
Current view: top level - kernel/gmp - gcdext.c (source / functions) Hit Total Coverage
Test: PARI/GP v2.18.0 lcov report (development 29841-6055bbdca6) Lines: 76 76 100.0 %
Date: 2025-01-02 09:12:31 Functions: 2 2 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : #line 2 "../src/kernel/gmp/gcdext.c"
       2             : /* Copyright (C) 2000-2003  The PARI group.
       3             : 
       4             : This file is part of the PARI/GP package.
       5             : 
       6             : PARI/GP is free software; you can redistribute it and/or modify it under the
       7             : terms of the GNU General Public License as published by the Free Software
       8             : Foundation; either version 2 of the License, or (at your option) any later
       9             : version. It is distributed in the hope that it will be useful, but WITHOUT
      10             : ANY WARRANTY WHATSOEVER.
      11             : 
      12             : Check the License for details. You should have received a copy of it, along
      13             : with the package; see the file 'COPYING'. If not, write to the Free Software
      14             : Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */
      15             : 
      16             : /*==================================
      17             :  * invmod(a,b,res)
      18             :  *==================================
      19             :  *    If a is invertible, return 1, and set res  = a^{ -1 }
      20             :  *    Otherwise, return 0, and set res = gcd(a,b)
      21             :  */
      22             : int
      23    21989362 : invmod(GEN a, GEN b, GEN *res)
      24             : {
      25    21989362 :   if (!signe(b)) { *res=absi(a); return 0; }
      26    21989362 :   if (NLIMBS(b) < INVMOD_GMP_LIMIT)
      27    19840657 :     return invmod_pari(a,b,res);
      28             :   { /* General case: use gcdext(a+b, b) since mpn_gcdext require S1>=S2 */
      29     2148705 :     pari_sp av = avma;
      30             :     GEN  ca, cb, u, d;
      31     2148705 :     long l, su, sa = signe(a), lb,lna;
      32             :     mp_size_t lu;
      33             :     GEN na;
      34     2148705 :     if (!sa) { set_avma(av); *res = absi(b); return 0; }
      35     2148701 :     if (signe(b) < 0) b = negi(b);
      36     2148701 :     if (abscmpii(a, b) < 0)
      37     2141951 :       na = sa > 0? addii(a, b): subii(a, b);
      38             :     else
      39        6792 :       na = a;
      40             :     /* Copy serves two purposes:
      41             :      * 1) mpn_gcdext destroys its input and needs an extra limb
      42             :      * 2) allows us to use icopy instead of gerepile later. */
      43     2148740 :     lb = lgefint(b); lna = lgefint(na);
      44     2148740 :     ca = icopy_ef(na,lna+1);
      45     2148739 :     cb = icopy_ef( b,lb+1);
      46             :     /* must create u first else final icopy could fail. */
      47     2148741 :     u = cgeti(lna+1);
      48     2148741 :     d = cgeti(lna+1);
      49             :     /* na >= b */
      50     2148741 :     l = mpn_gcdext(LIMBS(d), LIMBS(u), &lu, LIMBS(ca), NLIMBS(ca), LIMBS(cb), NLIMBS(cb));
      51     2148742 :     d[1] = evalsigne(1)|evallgefint(l+2);
      52     2148742 :     if (!is_pm1(d)) {set_avma(av); *res=icopy(d); return 0;}
      53     2148596 :     su = lu?((sa ^ lu) < 0)? -1: 1: 0;
      54     2148596 :     u[1] = evalsigne(su) | evallgefint(labs(lu)+2);
      55     2148596 :     if (su < 0) u = addii(u, b);
      56     2148595 :     set_avma(av); *res=icopy(u); return 1;
      57             :   }
      58             : }
      59             : 
      60             : /*==================================
      61             :  * bezout(a,b,pu,pv)
      62             :  *==================================
      63             :  *    Return g = gcd(a,b) >= 0, and assign GENs u,v through pointers pu,pv
      64             :  *    such that g = u*a + v*b.
      65             :  * Special cases:
      66             :  *    a == b == 0 ==> pick u=1, v=0
      67             :  *    a != 0 == b ==> keep v=0
      68             :  *    a == 0 != b ==> keep u=0
      69             :  *    |a| == |b| != 0 ==> keep u=0, set v=+-1
      70             :  * Assignments through pu,pv will be suppressed when the corresponding
      71             :  * pointer is NULL  (but the computations will happen nonetheless).
      72             :  */
      73             : 
      74             : GEN
      75    84816930 : bezout(GEN a, GEN b, GEN *pu, GEN *pv)
      76             : {
      77             :   long s, sa, sb;
      78             :   ulong g;
      79             :   ulong xu,xu1,xv,xv1;                /* Lehmer stage recurrence matrix */
      80             : 
      81    84816930 :   s = abscmpii(a,b);
      82    84818747 :   if (s < 0) { swap(a,b); pswap(pu,pv); }
      83             :   /* now |a| >= |b| */
      84             : 
      85    84818747 :   sa = signe(a); sb = signe(b);
      86    84818747 :   if (!sb)
      87             :   {
      88     2998273 :     if (pv) *pv = gen_0;
      89     2998273 :     switch(sa)
      90             :     {
      91           4 :     case  0: if (pu) *pu = gen_0;  return gen_0;
      92     2994513 :     case  1: if (pu) *pu = gen_1;  return icopy(a);
      93        3756 :     case -1: if (pu) *pu = gen_m1; return negi(a);
      94             :     }
      95             :   }
      96    81820474 :   if (s == 0)                        /* |a| == |b| != 0 */
      97             :   {
      98    10540297 :     if (pu) *pu = gen_0;
      99    10540297 :     if (sb > 0)
     100    10034878 :     { if (pv) *pv = gen_1;  return icopy(b); }
     101             :     else
     102      505419 :     { if (pv) *pv = gen_m1; return negi(b); }
     103             :   }
     104             :   /* now |a| > |b| > 0 */
     105             : 
     106    71280177 :   if (lgefint(a) == 3)                /* single-word affair */
     107             :   {
     108    68111970 :     g = xxgcduu((ulong)a[2], (ulong)b[2], 0, &xu, &xu1, &xv, &xv1, &s);
     109    68361851 :     sa = s > 0 ? sa : -sa;
     110    68361851 :     sb = s > 0 ? -sb : sb;
     111    68361851 :     if (pu)
     112             :     {
     113    34966909 :       if (xu == 0) *pu = gen_0; /* can happen when b divides a */
     114    12256891 :       else if (xu == 1) *pu = sa < 0 ? gen_m1 : gen_1;
     115     6905365 :       else if (xu == 2) *pu = sa < 0 ? gen_m2 : gen_2;
     116             :       else
     117             :       {
     118     6136967 :         *pu = cgeti(3);
     119     6137541 :         (*pu)[1] = evalsigne(sa)|evallgefint(3);
     120     6137541 :         (*pu)[2] = xu;
     121             :       }
     122             :     }
     123    68362425 :     if (pv)
     124             :     {
     125    62771848 :       if (xv == 1) *pv = sb < 0 ? gen_m1 : gen_1;
     126    26366339 :       else if (xv == 2) *pv = sb < 0 ? gen_m2 : gen_2;
     127             :       else
     128             :       {
     129    23874603 :         *pv = cgeti(3);
     130    23838334 :         (*pv)[1] = evalsigne(sb)|evallgefint(3);
     131    23838334 :         (*pv)[2] = xv;
     132             :       }
     133             :     }
     134    68326156 :     if (g == 1) return gen_1;
     135    23090400 :     else if (g == 2) return gen_2;
     136    16071781 :     else return utoipos(g);
     137             :   }
     138             :   else
     139             :   { /* general case */
     140     3168207 :     pari_sp av = avma;
     141             :     /*Copy serves two purposes:
     142             :      * 1) mpn_gcdext destroys its input and needs an extra limb
     143             :      * 2) allows us to use icopy instead of gerepile later.
     144             :      * NOTE: we must put u before d else the final icopy could fail. */
     145     3168207 :     GEN ca = icopy_ef(a,lgefint(a)+1);
     146     3230422 :     GEN cb = icopy_ef(b,lgefint(b)+1);
     147     3230421 :     GEN u = cgeti(lgefint(a)+1), v = NULL;
     148     3230416 :     GEN d = cgeti(lgefint(a)+1);
     149             :     long su,l;
     150             :     mp_size_t lu;
     151     3230418 :     l = mpn_gcdext(LIMBS(d), LIMBS(u), &lu, LIMBS(ca), NLIMBS(ca), LIMBS(cb), NLIMBS(cb));
     152     3230426 :     if (lu<=0)
     153             :     {
     154     2721999 :       if (lu==0) su=0;
     155      395988 :       else {su=-1;lu=-lu;}
     156             :     }
     157             :     else
     158      508427 :       su=1;
     159     3230426 :     if (sa<0) su=-su;
     160     3230426 :     d[1] = evalsigne(1)|evallgefint(l+2);
     161     3230426 :     u[1] = evalsigne(su)|evallgefint(lu+2);
     162     3230426 :     if (pv) v=diviiexact(subii(d,mulii(u,a)),b);
     163     3230424 :     set_avma(av);
     164     3230424 :     if (pu) *pu=icopy(u);
     165     3230423 :     if (pv) *pv=icopy(v);
     166     3230423 :     return icopy(d);
     167             :   }
     168             : }

Generated by: LCOV version 1.16