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 - basemath - concat.c (source / functions) Hit Total Coverage
Test: PARI/GP v2.18.0 lcov report (development 29712-7c8a932571) Lines: 331 358 92.5 %
Date: 2024-11-15 09:08:45 Functions: 15 15 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* Copyright (C) 2000  The PARI group.
       2             : 
       3             : This file is part of the PARI/GP package.
       4             : 
       5             : PARI/GP is free software; you can redistribute it and/or modify it under the
       6             : terms of the GNU General Public License as published by the Free Software
       7             : Foundation; either version 2 of the License, or (at your option) any later
       8             : version. It is distributed in the hope that it will be useful, but WITHOUT
       9             : ANY WARRANTY WHATSOEVER.
      10             : 
      11             : Check the License for details. You should have received a copy of it, along
      12             : with the package; see the file 'COPYING'. If not, write to the Free Software
      13             : Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */
      14             : 
      15             : /*******************************************************************/
      16             : /*                                                                 */
      17             : /*                          CONCATENATION                          */
      18             : /*                                                                 */
      19             : /*******************************************************************/
      20             : #include "pari.h"
      21             : #include "paripriv.h"
      22             : 
      23             : /* assume A or B is a t_LIST */
      24             : static GEN
      25          21 : listconcat(GEN A, GEN B)
      26             : {
      27             :   long i, l1, lx;
      28             :   GEN L, z, L1, L2;
      29             : 
      30          21 :   if (typ(A) != t_LIST) {
      31           7 :     if (list_typ(B)!=t_LIST_RAW) pari_err_TYPE("listconcat",B);
      32           7 :     L2 = list_data(B);
      33           7 :     if (!L2) return mklistcopy(A);
      34           7 :     lx = lg(L2) + 1;
      35           7 :     z = mklist();
      36           7 :     list_data(z) = L = cgetg(lx, t_VEC);
      37          35 :     for (i = 2; i < lx; i++) gel(L,i) = gcopy(gel(L2,i-1));
      38           7 :     gel(L,1) = gcopy(A); return z;
      39          14 :   } else if (typ(B) != t_LIST) {
      40           7 :     if (list_typ(A)!=t_LIST_RAW) pari_err_TYPE("listconcat",A);
      41           7 :     L1 = list_data(A);
      42           7 :     if (!L1) return mklistcopy(B);
      43           7 :     lx = lg(L1) + 1;
      44           7 :     z = mklist();
      45           7 :     list_data(z) = L = cgetg(lx, t_VEC);
      46          35 :     for (i = 1; i < lx-1; i++) gel(L,i) = gcopy(gel(L1,i));
      47           7 :     gel(L,i) = gcopy(B); return z;
      48             :   }
      49             :   /* A, B both t_LISTs */
      50           7 :   if (list_typ(A)!=t_LIST_RAW) pari_err_TYPE("listconcat",A);
      51           7 :   if (list_typ(B)!=t_LIST_RAW) pari_err_TYPE("listconcat",B);
      52           7 :   L1 = list_data(A); if (!L1) return listcopy(B);
      53           7 :   L2 = list_data(B); if (!L2) return listcopy(A);
      54             : 
      55           7 :   l1 = lg(L1);
      56           7 :   lx = l1-1 + lg(L2);
      57           7 :   z = mklist();
      58           7 :   list_data(z) = L = cgetg(lx, t_VEC);
      59           7 :   L2 -= l1-1;
      60          35 :   for (i=1; i<l1; i++) gel(L,i) = gcopy(gel(L1,i));
      61          35 :   for (   ; i<lx; i++) gel(L,i) = gcopy(gel(L2,i));
      62           7 :   return z;
      63             : }
      64             : 
      65             : /* assume A or B is a t_STR */
      66             : static GEN
      67         427 : strconcat(GEN x, GEN y)
      68             : {
      69             :   size_t l, lx;
      70         427 :   char *sx = GENtostr_unquoted(x);
      71         427 :   char *sy = GENtostr_unquoted(y), *str;
      72         427 :   lx = strlen(sx);
      73         427 :   l = nchar2nlong(lx + strlen(sy) + 1);
      74         427 :   x = cgetg(l + 1, t_STR); str = GSTR(x);
      75         427 :   strcpy(str,   sx);
      76         427 :   strcpy(str+lx,sy); return x;
      77             : }
      78             : 
      79             : /* concat A and B vertically. Internal */
      80             : GEN
      81    20315207 : vconcat(GEN A, GEN B)
      82             : {
      83             :   long la, ha, hb, hc, i, j, T;
      84             :   GEN M, a, b, c;
      85             : 
      86    20315207 :   if (!A) return B;
      87    20278464 :   if (!B) return A;
      88    20278464 :   la = lg(A); if (la==1) return B;
      89    20219473 :   T = typ(gel(A,1)); /* t_COL or t_VECSMALL */
      90    20219473 :   ha = lgcols(A); M = cgetg(la,t_MAT);
      91    20220075 :   hb = lgcols(B); hc = ha+hb-1;
      92   251711858 :   for (j=1; j<la; j++)
      93             :   {
      94   231490958 :     c = cgetg(hc, T); gel(M, j) = c;
      95   231483441 :     a = gel(A,j);
      96   231483441 :     b = gel(B,j);
      97  1035541622 :     for (i=1; i<ha; i++) *++c = *++a;
      98  1029498694 :     for (i=1; i<hb; i++) *++c = *++b;
      99             :   }
     100    20220900 :   return M;
     101             : }
     102             : 
     103             : static void
     104          49 : err_cat(GEN x, GEN y) { pari_err_OP("concatenation",x,y); }
     105             : 
     106             : GEN
     107    24528839 : shallowconcat(GEN x, GEN y)
     108             : {
     109    24528839 :   long tx=typ(x),ty=typ(y),lx=lg(x),ly=lg(y),i;
     110             :   GEN z,p1;
     111             : 
     112    24528839 :   if (tx==t_STR  || ty==t_STR)  return strconcat(x,y);
     113    24528479 :   if (tx==t_LIST || ty==t_LIST) return listconcat(x,y);
     114             : 
     115    24528484 :   if (tx==t_MAT && lx==1)
     116             :   {
     117      194798 :     if (ty!=t_VEC) return gtomat(y);
     118           0 :     if (ly==1) return cgetg(1, t_MAT);
     119           0 :     err_cat(x,y);
     120             :   }
     121    24333686 :   if (ty==t_MAT && ly==1)
     122             :   {
     123      165475 :     if (tx!=t_VEC) return gtomat(x);
     124           0 :     if (lx==1) return cgetg(1, t_MAT);
     125           0 :     err_cat(x,y);
     126             :   }
     127             : 
     128    24168669 :   if (tx == ty)
     129             :   {
     130    15996540 :     if (tx == t_MAT)
     131    12707989 :     { if (lgcols(x) != lgcols(y)) err_cat(x,y); }
     132             :     else
     133     3288551 :       if (!is_matvec_t(tx) && tx != t_VECSMALL) return mkvec2(x, y);
     134    15996532 :     z=cgetg(lx+ly-1,tx);
     135   177423631 :     for (i=1; i<lx; i++) z[i]     = x[i];
     136    82330731 :     for (i=1; i<ly; i++) z[lx+i-1]= y[i];
     137    15996414 :     return z;
     138             :   }
     139             : 
     140     8172129 :   if (! is_matvec_t(tx))
     141             :   {
     142       20480 :     if (! is_matvec_t(ty)) return mkvec2(x, y);
     143       20480 :     z=cgetg(ly+1,ty);
     144       20480 :     if (ty != t_MAT) p1 = x;
     145             :     else
     146             :     {
     147           0 :       if (lgcols(y)!=2) err_cat(x,y);
     148           0 :       p1 = mkcol(x);
     149             :     }
     150       66886 :     for (i=2; i<=ly; i++) z[i] = y[i-1];
     151       20480 :     gel(z, 1) = p1; return z;
     152             :   }
     153     8151651 :   if (! is_matvec_t(ty))
     154             :   {
     155     1521216 :     z=cgetg(lx+1,tx);
     156     1521217 :     if (tx != t_MAT) p1 = y;
     157             :     else
     158             :     {
     159           0 :       if (lgcols(x)!=2) err_cat(x,y);
     160           0 :       p1 = mkcol(y);
     161             :     }
     162    39755362 :     for (i=1; i<lx; i++) z[i]=x[i];
     163     1521217 :     gel(z, lx) = p1; return z;
     164             :   }
     165             : 
     166     6630416 :   switch(tx)
     167             :   {
     168       16534 :     case t_VEC:
     169             :       switch(ty)
     170             :       {
     171       16534 :         case t_COL:
     172       16534 :           if (lx<=2) return (lx==1)? y: shallowconcat(gel(x,1),y);
     173           0 :           if (ly>=3) break;
     174           0 :           return (ly==1)? x: shallowconcat(x,gel(y,1));
     175           0 :         case t_MAT:
     176           0 :           z=cgetg(ly,t_MAT); if (lx != ly) break;
     177           0 :           for (i=1; i<ly; i++) gel(z,i) = shallowconcat(gel(x,i),gel(y,i));
     178           0 :           return z;
     179             :       }
     180           0 :       break;
     181             : 
     182      107269 :     case t_COL:
     183             :       switch(ty)
     184             :       {
     185         379 :         case t_VEC:
     186         379 :           if (lx<=2) return (lx==1)? y: shallowconcat(gel(x,1), y);
     187         176 :           if (ly>=3) break;
     188         176 :           return (ly==1)? x: shallowconcat(x, gel(y,1));
     189      106890 :         case t_MAT:
     190      106890 :           if (lx != lgcols(y)) break;
     191      106890 :           z=cgetg(ly+1,t_MAT);  gel(z,1) = x;
     192      663479 :           for (i=2; i<=ly; i++) gel(z,i) = gel(y,i-1);
     193      106890 :           return z;
     194             :       }
     195           0 :       break;
     196             : 
     197     6506628 :     case t_MAT:
     198             :       switch(ty)
     199             :       {
     200           0 :         case t_VEC:
     201           0 :           z=cgetg(lx, t_MAT); if (ly != lx) break;
     202           0 :           for (i=1; i<lx; i++) gel(z,i) = shallowconcat(gel(x,i), gel(y,i));
     203           0 :           return z;
     204     6506631 :         case t_COL:
     205     6506631 :           if (ly != lgcols(x)) break;
     206     6506613 :           z=cgetg(lx+1,t_MAT); gel(z,lx) = y;
     207    23482755 :           for (i=1; i<lx; i++) z[i]=x[i];
     208     6506603 :           return z;
     209             :       }
     210           0 :       break;
     211             :   }
     212           0 :   err_cat(x,y);
     213             :   return NULL; /* LCOV_EXCL_LINE */
     214             : }
     215             : 
     216             : /* see catmany() */
     217             : static GEN
     218       19294 : catmanyMAT(GEN y1, GEN y2)
     219             : {
     220       19294 :   long i, h = 0, L = 1;
     221             :   GEN z, y;
     222       66949 :   for (y = y2; y >= y1; y--)
     223             :   {
     224       47655 :     GEN c = gel(y,0);
     225       47655 :     long nc = lg(c)-1;
     226       47655 :     if (nc == 0) continue;
     227       47627 :     if (h != lgcols(c))
     228             :     {
     229       19287 :       if (h) err_cat(gel(y2,0), c);
     230       19287 :       h = lgcols(c);
     231             :     }
     232       47627 :     L += nc;
     233       47627 :     z = new_chunk(nc) - 1;
     234      166654 :     for (i=1; i<=nc; i++) gel(z,i) = gel(c,i);
     235             :   }
     236       19294 :   z = new_chunk(1);
     237       19294 :   *z = evaltyp(t_MAT) | evallg(L);
     238       19294 :   return z;
     239             : }
     240             : static GEN
     241         147 : catmanySTR(GEN y1, GEN y2)
     242             : {
     243         147 :   long L = 1; /* final \0 */
     244             :   GEN z, y;
     245             :   char *s;
     246        7336 :   for (y = y1; y <= y2; y++)
     247             :   {
     248        7189 :     char *c = GSTR( gel(y,0) );
     249        7189 :     L += strlen(c);
     250             :   }
     251         147 :   z = cgetg(nchar2nlong(L)+1, t_STR);
     252         147 :   s = GSTR(z);
     253        7336 :   for (y = y1; y <= y2; y++)
     254             :   {
     255        7189 :     char *c = GSTR( gel(y,0) );
     256        7189 :     long nc = strlen(c);
     257        7189 :     if (nc) { (void)memcpy(s, c, nc); s += nc; }
     258             :   }
     259         147 :   *s = 0; return z;
     260             : }
     261             : 
     262             : /* all entries in y have the same type t = t_VEC, COL, MAT or VECSMALL
     263             :  * concatenate y[k1..k2], with yi = y + ki, k1 <= k2 */
     264             : static GEN
     265     2120155 : catmany(GEN y1, GEN y2, long t)
     266             : {
     267             :   long i, L;
     268             :   GEN z, y;
     269     2120155 :   if (y1 == y2) return gel(y1,0);
     270     2118055 :   if (t == t_MAT) return catmanyMAT(y1, y2);
     271     2098761 :   if (t == t_STR) return catmanySTR(y1, y2);
     272     2098614 :   L = 1;
     273     7849323 :   for (y = y2; y >= y1; y--)
     274             :   {
     275     5750708 :     GEN c = gel(y,0);
     276     5750708 :     long nc = lg(c)-1;
     277     5750708 :     if (nc == 0) continue;
     278     5189001 :     L += nc;
     279     5189001 :     z = new_chunk(nc) - 1;
     280    17055420 :     for (i=1; i<=nc; i++) gel(z,i) = gel(c,i);
     281             :   }
     282     2098615 :   z = new_chunk(1);
     283     2098613 :   *z = evaltyp(t) | evallg(L);
     284     2098612 :   return z;
     285             : }
     286             : 
     287             : GEN
     288     7770112 : shallowconcat1(GEN x)
     289             : {
     290     7770112 :   pari_sp av = avma;
     291             :   long lx, t, i;
     292             :   GEN z;
     293     7770112 :   switch(typ(x))
     294             :   {
     295     7770090 :     case t_VEC: case t_COL:
     296     7770090 :       lx = lg(x);
     297     7770090 :       break;
     298          21 :     case t_LIST:
     299          21 :       if (list_typ(x)!=t_LIST_RAW) pari_err_TYPE("concat",x);
     300          21 :       if (!list_data(x)) pari_err_DOMAIN("concat","vector","=",x,x);
     301           7 :       x = list_data(x); lx = lg(x);
     302           7 :       break;
     303           1 :     default:
     304           1 :       pari_err_TYPE("concat",x);
     305             :       return NULL; /* LCOV_EXCL_LINE */
     306             :   }
     307     7770097 :   if (lx==1) pari_err_DOMAIN("concat","vector","=",x,x);
     308     7770135 :   if (lx==2) return gel(x,1);
     309     2120173 :   z = gel(x,1); t = typ(z); i = 2;
     310     2120173 :   if (is_matvec_t(t) || t == t_VECSMALL || t == t_STR)
     311             :   { /* detect a "homogeneous" object: catmany is faster */
     312     5807664 :     for (; i<lx; i++)
     313     3689834 :       if (typ(gel(x,i)) != t) break;
     314     2120154 :     z = catmany(x + 1, x + i-1, t);
     315             :   }
     316     2123121 :   for (; i<lx; i++) {
     317        2954 :     z = shallowconcat(z, gel(x,i));
     318        2954 :     if (gc_needed(av,3))
     319             :     {
     320           0 :       if (DEBUGMEM>1) pari_warn(warnmem,"concat: i = %ld", i);
     321           0 :       z = gerepilecopy(av, z);
     322             :     }
     323             :   }
     324     2120167 :   return z;
     325             : }
     326             : 
     327             : GEN
     328         840 : gconcat1(GEN x)
     329             : {
     330         840 :   pari_sp av = avma;
     331         840 :   return gerepilecopy(av, shallowconcat1(x));
     332             : }
     333             : 
     334             : /* fill M[xoff+i, yoff+j] with the contents of c ( c * Id_n if scalar ) */
     335             : static void
     336    16627202 : matfill(GEN M, GEN c, long xoff, long yoff, long n)
     337             : {
     338             :   long i, j, h, l;
     339    16627202 :   l = lg(c); if (l == 1) return;
     340    16600941 :   switch(typ(c))
     341             :   {
     342       44023 :     case t_VEC:
     343      954527 :       for (i = 1; i < l; i++)
     344      910504 :         gcoeff(M,xoff+1,yoff+i) = gel(c,i);
     345       44023 :       break;
     346      161545 :     case t_COL:
     347     1187673 :       for (i = 1; i < l; i++)
     348     1026128 :         gcoeff(M,xoff+i,yoff+1) = gel(c,i);
     349      161545 :       break;
     350     9338607 :     case t_MAT:
     351     9338607 :       h = lgcols(c);
     352    27701849 :       for (j = 1; j < l; j++)
     353    74201003 :         for (i = 1; i < h; i++) gcoeff(M,xoff+i,yoff+j) = gcoeff(c,i,j);
     354     9338603 :       break;
     355     7056766 :     default:
     356    15930043 :       for (i = 1; i <= n; i++)
     357     8873277 :         gcoeff(M, xoff+i, yoff+i) = c;
     358     7056766 :       break;
     359             :   }
     360             : }
     361             : 
     362             : static GEN
     363    18739763 : _matsize(GEN x)
     364             : {
     365    18739763 :   long t = typ(x), L = lg(x) - 1;
     366    18739763 :   switch(t)
     367             :   { /* matsize */
     368       88060 :     case t_VEC: return mkvecsmall2(1, L);
     369      323034 :     case t_COL: return mkvecsmall2(L, 1);
     370    11270861 :     case t_MAT: return mkvecsmall2(L? nbrows(x): 0, L);
     371     7057808 :     default:
     372     7057808 :       if (is_noncalc_t(t)) pari_err_TYPE("_matsize", x);
     373     7057815 :       return mkvecsmall2(1, 1);
     374             :   }
     375             : }
     376             : 
     377             : GEN
     378     3682343 : shallowmatconcat(GEN v)
     379             : {
     380     3682343 :   long i, j, h, l = lg(v), L = 0, H = 0;
     381             :   GEN M, maxh, maxl;
     382     3682343 :   if (l == 1) return cgetg(1,t_MAT);
     383     3678180 :   switch(typ(v))
     384             :   {
     385      995489 :     case t_VEC:
     386     3062295 :       for (i = 1; i < l; i++)
     387             :       {
     388     2066810 :         GEN c = gel(v,i);
     389     2066810 :         GEN s = _matsize(c);
     390     2066810 :         H = maxss(H, s[1]);
     391     2066806 :         L += s[2];
     392             :       }
     393      995485 :       M = zeromatcopy(H, L);
     394      995501 :       L = 0;
     395     3062332 :       for (i = 1; i < l; i++)
     396             :       {
     397     2066837 :         GEN c = gel(v,i);
     398     2066837 :         GEN s = _matsize(c);
     399     2066835 :         matfill(M, c, 0, L, 1);
     400     2066831 :         L += s[2];
     401             :       }
     402      995495 :       return M;
     403             : 
     404        3584 :     case t_COL:
     405       49406 :       for (i = 1; i < l; i++)
     406             :       {
     407       45822 :         GEN c = gel(v,i);
     408       45822 :         GEN s = _matsize(c);
     409       45822 :         H += s[1];
     410       45822 :         L = maxss(L, s[2]);
     411             :       }
     412        3584 :       M = zeromatcopy(H, L);
     413        3584 :       H = 0;
     414       49406 :       for (i = 1; i < l; i++)
     415             :       {
     416       45822 :         GEN c = gel(v,i);
     417       45822 :         GEN s = _matsize(c);
     418       45822 :         matfill(M, c, H, 0, 1);
     419       45822 :         H += s[1];
     420             :       }
     421        3584 :       return M;
     422     2679107 :     case t_MAT:
     423     2679107 :       h = lgcols(v);
     424     2679107 :       maxh = zero_zv(h-1);
     425     2679106 :       maxl = zero_zv(l-1);
     426     8796915 :       for (j = 1; j < l; j++)
     427    20632335 :         for (i = 1; i < h; i++)
     428             :         {
     429    14514527 :           GEN c = gcoeff(v,i,j);
     430    14514527 :           GEN s = _matsize(c);
     431    14514526 :           if (s[1] > maxh[i]) maxh[i] = s[1];
     432    14514526 :           if (s[2] > maxl[j]) maxl[j] = s[2];
     433             :         }
     434     8796924 :       for (i = 1, H = 0; i < h; i++) H += maxh[i];
     435     8796917 :       for (j = 1, L = 0; j < l; j++) L += maxl[j];
     436     2679106 :       M = zeromatcopy(H, L);
     437     8796920 :       for (j = 1, L = 0; j < l; j++)
     438             :       {
     439    20632362 :         for (i = 1, H = 0; i < h; i++)
     440             :         {
     441    14514549 :           GEN c = gcoeff(v,i,j);
     442    14514549 :           matfill(M, c, H, L, minss(maxh[i], maxl[j]));
     443    14514549 :           H += maxh[i];
     444             :         }
     445     6117813 :         L += maxl[j];
     446             :       }
     447     2679107 :       return M;
     448           0 :     default:
     449           0 :       pari_err_TYPE("shallowmatconcat", v);
     450             :       return NULL;/*LCOV_EXCL_LINE*/
     451             :   }
     452             : }
     453             : GEN
     454       76580 : matconcat(GEN v)
     455             : {
     456       76580 :   pari_sp av = avma;
     457       76580 :   return gerepilecopy(av, shallowmatconcat(v));
     458             : }
     459             : 
     460             : GEN
     461    19425273 : gconcat(GEN x, GEN y)
     462             : {
     463             :   long tx, lx,ty,ly,i;
     464             :   GEN z,p1;
     465             : 
     466    19425273 :   if (!y) return gconcat1(x);
     467    19424433 :   tx = typ(x);
     468    19424433 :   ty = typ(y);
     469    19424433 :   if (tx==t_STR  || ty==t_STR)
     470             :   {
     471           7 :     pari_sp av = avma;
     472           7 :     return gerepileuptoleaf(av, strconcat(x,y));
     473             :   }
     474    19424426 :   if (tx==t_LIST || ty==t_LIST) return listconcat(x,y);
     475    19424405 :   lx=lg(x); ly=lg(y);
     476             : 
     477    19424405 :   if (tx==t_MAT && lx==1)
     478             :   {
     479         588 :     if (ty!=t_VEC) return gtomat(y);
     480          21 :     if (ly==1) return cgetg(1, t_MAT);
     481           7 :     err_cat(x,y);
     482             :   }
     483    19423817 :   if (ty==t_MAT && ly==1)
     484             :   {
     485          21 :     if (tx!=t_VEC) return gtomat(x);
     486          14 :     if (lx==1) return cgetg(1, t_MAT);
     487           7 :     err_cat(x,y);
     488             :   }
     489             : 
     490    19423796 :   if (tx == ty)
     491             :   {
     492      653489 :     if (tx == t_MAT && lgcols(x) != lgcols(y)) err_cat(x,y);
     493      653489 :     if (!is_matvec_t(tx))
     494             :     {
     495         210 :       if (tx != t_VECSMALL) return mkvec2copy(x, y);
     496         203 :       z = cgetg(lx+ly-1,t_VECSMALL);
     497         301 :       for (i=1; i<lx; i++) z[i]     = x[i];
     498         294 :       for (i=1; i<ly; i++) z[lx+i-1]= y[i];
     499         203 :       return z;
     500             :     }
     501      653279 :     z=cgetg(lx+ly-1,tx);
     502     3304140 :     for (i=1; i<lx; i++) gel(z,i)     = gcopy(gel(x,i));
     503     2529548 :     for (i=1; i<ly; i++) gel(z,lx+i-1)= gcopy(gel(y,i));
     504      653279 :     return z;
     505             :   }
     506             : 
     507    18770307 :   if (! is_matvec_t(tx))
     508             :   {
     509         168 :     if (! is_matvec_t(ty)) return mkvec2copy(x, y);
     510         168 :     z=cgetg(ly+1,ty);
     511         168 :     if (ty != t_MAT) p1 = gcopy(x);
     512             :     else
     513             :     {
     514          14 :       if (lgcols(y)!=2) err_cat(x,y);
     515           7 :       p1 = mkcolcopy(x);
     516             :     }
     517        3514 :     for (i=2; i<=ly; i++) gel(z,i) = gcopy(gel(y,i-1));
     518         161 :     gel(z,1) = p1; return z;
     519             :   }
     520    18770139 :   if (! is_matvec_t(ty))
     521             :   {
     522    18770041 :     z=cgetg(lx+1,tx);
     523    18770041 :     if (tx != t_MAT) p1 = gcopy(y);
     524             :     else
     525             :     {
     526          14 :       if (lgcols(x)!=2) err_cat(x,y);
     527           7 :       p1 = mkcolcopy(y);
     528             :     }
     529  2547662349 :     for (i=1; i<lx; i++) gel(z,i) = gcopy(gel(x,i));
     530    18770034 :     gel(z,lx) = p1; return z;
     531             :   }
     532             : 
     533          98 :   switch(tx)
     534             :   {
     535          35 :     case t_VEC:
     536             :       switch(ty)
     537             :       {
     538          28 :         case t_COL:
     539          28 :           if (lx<=2) return (lx==1)? gcopy(y): gconcat(gel(x,1),y);
     540          28 :           if (ly>=3) break;
     541          14 :           return (ly==1)? gcopy(x): gconcat(x,gel(y,1));
     542           7 :         case t_MAT:
     543           7 :           z=cgetg(ly,t_MAT); if (lx != ly) break;
     544          21 :           for (i=1; i<ly; i++) gel(z,i) = gconcat(gel(x,i),gel(y,i));
     545           7 :           return z;
     546             :       }
     547          14 :       break;
     548             : 
     549          49 :     case t_COL:
     550             :       switch(ty)
     551             :       {
     552          42 :         case t_VEC:
     553          42 :           if (lx<=2) return (lx==1)? gcopy(y): gconcat(gel(x,1),y);
     554          28 :           if (ly>=3) break;
     555          21 :           return (ly==1)? gcopy(x): gconcat(x,gel(y,1));
     556           7 :         case t_MAT:
     557           7 :           if (lx != lgcols(y)) break;
     558           7 :           z=cgetg(ly+1,t_MAT); gel(z,1) = gcopy(x);
     559          14 :           for (i=2; i<=ly; i++) gel(z,i) = gcopy(gel(y,i-1));
     560           7 :           return z;
     561             :       }
     562           7 :       break;
     563             : 
     564          14 :     case t_MAT:
     565             :       switch(ty)
     566             :       {
     567           7 :         case t_VEC:
     568           7 :           z=cgetg(lx,t_MAT); if (ly != lx) break;
     569          21 :           for (i=1; i<lx; i++) gel(z,i) = gconcat(gel(x,i),gel(y,i));
     570           7 :           return z;
     571           7 :         case t_COL:
     572           7 :           if (ly != lgcols(x)) break;
     573           7 :           z=cgetg(lx+1,t_MAT); gel(z,lx) = gcopy(y);
     574          14 :           for (i=1; i<lx; i++) gel(z,i) = gcopy(gel(x,i));
     575           7 :           return z;
     576             :       }
     577           0 :       break;
     578             :   }
     579          21 :   err_cat(x,y);
     580             :   return NULL; /* LCOV_EXCL_LINE */
     581             : }

Generated by: LCOV version 1.16