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/none - gcdext.c (source / functions) Hit Total Coverage
Test: PARI/GP v2.18.1 lcov report (development 29875-1c62f24b5e) Lines: 122 125 97.6 %
Date: 2025-01-17 09:09:20 Functions: 4 4 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : #line 2 "../src/kernel/none/gcdext.c"
       2             : /* Copyright (C) 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             :  * bezout(a,b,pu,pv)
      18             :  *==================================
      19             :  *    Return g = gcd(a,b) >= 0, and assign GENs u,v through pointers pu,pv
      20             :  *    such that g = u*a + v*b.
      21             :  * Special cases:
      22             :  *    a == b == 0 ==> pick u=1, v=0
      23             :  *    a != 0 == b ==> keep v=0
      24             :  *    a == 0 != b ==> keep u=0
      25             :  *    |a| == |b| != 0 ==> keep u=0, set v=+-1
      26             :  * Assignments through pu,pv will be suppressed when the corresponding
      27             :  * pointer is NULL  (but the computations will happen nonetheless).
      28             :  */
      29             : 
      30             : static GEN
      31    64369371 : bezout_lehmer(GEN a, GEN b, GEN *pu, GEN *pv)
      32             : {
      33             :   GEN t,u,u1,v,v1,d,d1,q,r;
      34             :   GEN *pt;
      35             :   pari_sp av, av1;
      36             :   long s, sa, sb;
      37             :   ulong g;
      38             :   ulong xu,xu1,xv,xv1;                /* Lehmer stage recurrence matrix */
      39             :   int lhmres;                        /* Lehmer stage return value */
      40             : 
      41    64369371 :   s = abscmpii(a,b);
      42    64369371 :   if (s < 0)
      43             :   {
      44    41385957 :     t=b; b=a; a=t;
      45    41385957 :     pt=pu; pu=pv; pv=pt;
      46             :   }
      47             :   /* now |a| >= |b| */
      48             : 
      49    64369371 :   sa = signe(a); sb = signe(b);
      50    64369371 :   if (!sb)
      51             :   {
      52     2277357 :     if (pv) *pv = gen_0;
      53     2277357 :     switch(sa)
      54             :     {
      55           3 :     case  0: if (pu) *pu = gen_0; return gen_0;
      56     2274537 :     case  1: if (pu) *pu = gen_1; return icopy(a);
      57        2817 :     case -1: if (pu) *pu = gen_m1; return(negi(a));
      58             :     }
      59             :   }
      60    62092014 :   if (s == 0)                        /* |a| == |b| != 0 */
      61             :   {
      62     7950174 :     if (pu) *pu = gen_0;
      63     7950174 :     if (sb > 0)
      64     7573971 :     { if (pv) *pv = gen_1; return icopy(b); }
      65             :     else
      66      376203 :     { if (pv) *pv = gen_m1; return(negi(b)); }
      67             :   }
      68             :   /* now |a| > |b| > 0 */
      69             : 
      70    54141840 :   if (lgefint(a) == 3)                /* single-word affair */
      71             :   {
      72    52158717 :     g = xxgcduu(uel(a,2), uel(b,2), 0, &xu, &xu1, &xv, &xv1, &s);
      73    52158717 :     sa = s > 0 ? sa : -sa;
      74    52158717 :     sb = s > 0 ? -sb : sb;
      75    52158717 :     if (pu)
      76             :     {
      77    26778840 :       if (xu == 0) *pu = gen_0; /* can happen when b divides a */
      78     9376710 :       else if (xu == 1) *pu = sa < 0 ? gen_m1 : gen_1;
      79     5281827 :       else if (xu == 2) *pu = sa < 0 ? gen_m2 : gen_2;
      80             :       else
      81             :       {
      82     4699371 :         *pu = cgeti(3);
      83     4699371 :         (*pu)[1] = evalsigne(sa)|evallgefint(3);
      84     4699371 :         (*pu)[2] = xu;
      85             :       }
      86             :     }
      87    52158717 :     if (pv)
      88             :     {
      89    47929818 :       if (xv == 1) *pv = sb < 0 ? gen_m1 : gen_1;
      90    20090112 :       else if (xv == 2) *pv = sb < 0 ? gen_m2 : gen_2;
      91             :       else
      92             :       {
      93    18196923 :         *pv = cgeti(3);
      94    18196923 :         (*pv)[1] = evalsigne(sb)|evallgefint(3);
      95    18196923 :         (*pv)[2] = xv;
      96             :       }
      97             :     }
      98    52158717 :     if      (g == 1) return gen_1;
      99    17753853 :     else if (g == 2) return gen_2;
     100    12371082 :     else return utoipos(g);
     101             :   }
     102             : 
     103             :   /* general case */
     104     1983123 :   av = avma;
     105     1983123 :   (void)new_chunk(lgefint(b) + (lgefint(a)<<1)); /* room for u,v,gcd */
     106             :   /* if a is significantly larger than b, calling lgcdii() is not the best
     107             :    * way to start -- reduce a mod b first
     108             :    */
     109     1983123 :   if (lgefint(a) > lgefint(b))
     110             :   {
     111     1498866 :     d = absi_shallow(b);
     112     1498866 :     q = dvmdii(absi_shallow(a), d, &d1);
     113     1498866 :     if (!signe(d1))                /* a == qb */
     114             :     {
     115     1273728 :       set_avma(av);
     116     1273728 :       if (pu) *pu = gen_0;
     117     1273728 :       if (pv) *pv = sb < 0 ? gen_m1 : gen_1;
     118     1273728 :       return icopy(d);
     119             :     }
     120             :     else
     121             :     {
     122      225138 :       u = gen_0;
     123      225138 :       u1 = v = gen_1;
     124      225138 :       v1 = negi(q);
     125             :     }
     126             :     /* if this results in lgefint(d) == 3, will fall past main loop */
     127             :   }
     128             :   else
     129             :   {
     130      484257 :     d = absi_shallow(a);
     131      484257 :     d1 = absi_shallow(b);
     132      484257 :     u = v1 = gen_1; u1 = v = gen_0;
     133             :   }
     134      709395 :   av1 = avma;
     135             : 
     136             :   /* main loop is almost identical to that of invmod() */
     137     2234103 :   while (lgefint(d) > 3 && signe(d1))
     138             :   {
     139     1524708 :     lhmres = lgcdii((ulong *)d, (ulong *)d1, &xu, &xu1, &xv, &xv1, ULONG_MAX);
     140     1524708 :     if (lhmres != 0)                /* check progress */
     141             :     {                                /* apply matrix */
     142     1149243 :       if ((lhmres == 1) || (lhmres == -1))
     143             :       {
     144       65571 :         if (xv1 == 1)
     145             :         {
     146       31716 :           r = subii(d,d1); d=d1; d1=r;
     147       31716 :           a = subii(u,u1); u=u1; u1=a;
     148       31716 :           a = subii(v,v1); v=v1; v1=a;
     149             :         }
     150             :         else
     151             :         {
     152       33855 :           r = subii(d, mului(xv1,d1)); d=d1; d1=r;
     153       33855 :           a = subii(u, mului(xv1,u1)); u=u1; u1=a;
     154       33855 :           a = subii(v, mului(xv1,v1)); v=v1; v1=a;
     155             :         }
     156             :       }
     157             :       else
     158             :       {
     159     1083672 :         r  = subii(muliu(d,xu),  muliu(d1,xv));
     160     1083672 :         d1 = subii(muliu(d,xu1), muliu(d1,xv1)); d = r;
     161     1083672 :         a  = subii(muliu(u,xu),  muliu(u1,xv));
     162     1083672 :         u1 = subii(muliu(u,xu1), muliu(u1,xv1)); u = a;
     163     1083672 :         a  = subii(muliu(v,xu),  muliu(v1,xv));
     164     1083672 :         v1 = subii(muliu(v,xu1), muliu(v1,xv1)); v = a;
     165     1083672 :         if (lhmres&1) { togglesign(d);  togglesign(u);  togglesign(v); }
     166      550872 :         else          { togglesign(d1); togglesign(u1); togglesign(v1); }
     167             :       }
     168             :     }
     169     1524708 :     if (lhmres <= 0 && signe(d1))
     170             :     {
     171      404340 :       q = dvmdii(d,d1,&r);
     172      404340 :       a = subii(u,mulii(q,u1));
     173      404340 :       u=u1; u1=a;
     174      404340 :       a = subii(v,mulii(q,v1));
     175      404340 :       v=v1; v1=a;
     176      404340 :       d=d1; d1=r;
     177             :     }
     178     1524708 :     if (gc_needed(av,1))
     179             :     {
     180           0 :       if(DEBUGMEM>1) pari_warn(warnmem,"bezout");
     181           0 :       gerepileall(av1,6, &d,&d1,&u,&u1,&v,&v1);
     182             :     }
     183             :   } /* end while */
     184             : 
     185             :   /* Postprocessing - final sprint */
     186      709395 :   if (signe(d1))
     187             :   {
     188             :     /* Assertions: lgefint(d)==lgefint(d1)==3, and
     189             :      * gcd(d,d1) is nonzero and fits into one word
     190             :      */
     191      332130 :     g = xxgcduu(uel(d,2), uel(d1,2), 0, &xu, &xu1, &xv, &xv1, &s);
     192      332130 :     u = subii(muliu(u,xu), muliu(u1, xv));
     193      332130 :     v = subii(muliu(v,xu), muliu(v1, xv));
     194      332130 :     if (s < 0) { sa = -sa; sb = -sb; }
     195      332130 :     set_avma(av);
     196      332130 :     if (pu) *pu = sa < 0 ? negi(u) : icopy(u);
     197      332130 :     if (pv) *pv = sb < 0 ? negi(v) : icopy(v);
     198      332130 :     if (g == 1) return gen_1;
     199      262347 :     else if (g == 2) return gen_2;
     200      245409 :     else return utoipos(g);
     201             :   }
     202             :   /* get here when the final sprint was skipped (d1 was zero already).
     203             :    * Now the matrix is final, and d contains the gcd.
     204             :    */
     205      377265 :   set_avma(av);
     206      377265 :   if (pu) *pu = sa < 0 ? negi(u) : icopy(u);
     207      377265 :   if (pv) *pv = sb < 0 ? negi(v) : icopy(v);
     208      377265 :   return icopy(d);
     209             : }
     210             : 
     211             : static GEN
     212         453 : addmulmul(GEN u, GEN v, GEN x, GEN y)
     213         453 : { return addii(mulii(u, x),mulii(v, y)); }
     214             : 
     215             : static GEN
     216         261 : bezout_halfgcd(GEN x, GEN y, GEN *ptu, GEN *ptv)
     217             : {
     218         261 :   pari_sp av=avma;
     219         261 :   GEN u, v, R = matid2();
     220         642 :   while (lgefint(y)-2 >= EXTGCD_HALFGCD_LIMIT)
     221             :   {
     222         381 :     GEN M = HGCD0(x,y);
     223         381 :     R = ZM2_mul(R, gel(M, 1));
     224         381 :     x = gel(M,2); y = gel(M,3);
     225         381 :     if (signe(y) && expi(y)<magic_threshold(x))
     226             :     {
     227         153 :       GEN r, q = dvmdii(x, y, &r);
     228         153 :       x = y; y = r;
     229         153 :       R = mulq(R, q);
     230             :     }
     231         381 :     if (gc_needed(av, 1))
     232           0 :       gerepileall(av,3,&x,&y,&R);
     233             :   }
     234         261 :   y = bezout_lehmer(x,y,&u,&v);
     235         261 :   R = ZM_inv2(R);
     236         261 :   if (ptu) *ptu = addmulmul(u,v,gcoeff(R,1,1),gcoeff(R,2,1));
     237         261 :   if (ptv) *ptv = addmulmul(u,v,gcoeff(R,1,2),gcoeff(R,2,2));
     238         261 :   return y;
     239             : }
     240             : 
     241             : GEN
     242    64369371 : bezout(GEN x, GEN y, GEN *ptu, GEN *ptv)
     243             : {
     244    64369371 :   pari_sp av = avma;
     245             :   GEN d;
     246    64369371 :   if (lgefint(y)-2 >= EXTGCD_HALFGCD_LIMIT)
     247         261 :     d = bezout_halfgcd(x, y, ptu, ptv);
     248             :   else
     249    64369110 :     d = bezout_lehmer(x, y, ptu, ptv);
     250    64369371 :   if (ptv) return gc_all(av,ptu?3:2, &d, ptv, ptu);
     251    33867918 :   return gc_all(av, ptu?2:1, &d, ptu);
     252             : }

Generated by: LCOV version 1.16