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 : #include "pari.h"
16 : #include "paripriv.h"
17 :
18 : int
19 1218076 : RgM_is_ZM(GEN x)
20 : {
21 1218076 : long i, j, h, l = lg(x);
22 1218076 : if (l == 1) return 1;
23 1217411 : h = lgcols(x);
24 1217403 : if (h == 1) return 1;
25 4464789 : for (j = l-1; j > 0; j--)
26 18487695 : for (i = h-1; i > 0; i--)
27 15240148 : if (typ(gcoeff(x,i,j)) != t_INT) return 0;
28 1073940 : return 1;
29 : }
30 :
31 : int
32 203 : RgM_is_QM(GEN x)
33 : {
34 203 : long i, j, h, l = lg(x);
35 203 : if (l == 1) return 1;
36 189 : h = lgcols(x);
37 189 : if (h == 1) return 1;
38 1218 : for (j = l-1; j > 0; j--)
39 15057 : for (i = h-1; i > 0; i--)
40 14028 : if (!is_rational_t(typ(gcoeff(x,i,j)))) return 0;
41 175 : return 1;
42 : }
43 :
44 : int
45 49 : RgV_is_ZMV(GEN V)
46 : {
47 49 : long i, l = lg(V);
48 399 : for (i=1; i<l; i++)
49 350 : if (typ(gel(V,i))!=t_MAT || !RgM_is_ZM(gel(V,i)))
50 0 : return 0;
51 49 : return 1;
52 : }
53 :
54 : /********************************************************************/
55 : /** **/
56 : /** GENERIC LINEAR ALGEBRA **/
57 : /** **/
58 : /********************************************************************/
59 : /* GENERIC MULTIPLICATION involving zc/zm */
60 :
61 : /* x[i,] * y */
62 : static GEN
63 8238237 : RgMrow_zc_mul_i(GEN x, GEN y, long c, long i)
64 : {
65 8238237 : pari_sp av = avma;
66 8238237 : GEN s = NULL;
67 : long j;
68 828652461 : for (j=1; j<c; j++)
69 : {
70 820417470 : long t = y[j];
71 820417470 : if (!t) continue;
72 28408902 : if (!s) { s = gmulgs(gcoeff(x,i,j),t); continue; }
73 20383712 : switch(t)
74 : {
75 14852620 : case 1: s = gadd(s, gcoeff(x,i,j)); break;
76 239757 : case -1: s = gsub(s, gcoeff(x,i,j)); break;
77 5291335 : default: s = gadd(s, gmulgs(gcoeff(x,i,j), t)); break;
78 : }
79 : }
80 8234991 : return s? gerepileupto(av, s): gc_const(av, gen_0);
81 : }
82 : GEN
83 70028 : RgMrow_zc_mul(GEN x, GEN y, long i) { return RgMrow_zc_mul_i(x,y,lg(y),i); }
84 : /* x nonempty t_MAT, y a compatible zc (dimension > 0). */
85 : static GEN
86 237877 : RgM_zc_mul_i(GEN x, GEN y, long c, long l)
87 : {
88 237877 : GEN z = cgetg(l,t_COL);
89 : long i;
90 8406056 : for (i = 1; i < l; i++) gel(z,i) = RgMrow_zc_mul_i(x,y,c,i);
91 237876 : return z;
92 : }
93 : GEN
94 71967 : RgM_zc_mul(GEN x, GEN y) { return RgM_zc_mul_i(x,y, lg(x), lgcols(x)); }
95 : /* x t_MAT, y a compatible zm (dimension > 0). */
96 : GEN
97 56493 : RgM_zm_mul(GEN x, GEN y)
98 : {
99 56493 : long j, c, l = lg(x), ly = lg(y);
100 56493 : GEN z = cgetg(ly, t_MAT);
101 56493 : if (l == 1) return z;
102 56493 : c = lgcols(x);
103 222403 : for (j = 1; j < ly; j++) gel(z,j) = RgM_zc_mul_i(x, gel(y,j), l,c);
104 56493 : return z;
105 : }
106 :
107 : /* x[i,]*y, l = lg(y) > 1 */
108 : static GEN
109 32283711 : RgMrow_ZC_mul_i(GEN x, GEN y, long i, long l)
110 : {
111 32283711 : pari_sp av = avma;
112 32283711 : GEN t = gmul(gcoeff(x,i,1), gel(y,1)); /* l > 1 ! */
113 : long j;
114 5021006830 : for (j=2; j<l; j++)
115 4988924087 : if (signe(gel(y,j))) t = gadd(t, gmul(gcoeff(x,i,j), gel(y,j)));
116 32082743 : return gerepileupto(av,t);
117 : }
118 :
119 : /* compatible t_MAT * t_COL, lx = lg(x) = lg(y) > 1, l = lgcols(x) */
120 : static GEN
121 1924643 : RgM_ZC_mul_i(GEN x, GEN y, long lx, long l)
122 : {
123 1924643 : GEN z = cgetg(l,t_COL);
124 : long i;
125 34212402 : for (i=1; i<l; i++) gel(z,i) = RgMrow_ZC_mul_i(x,y,i,lx);
126 1921897 : return z;
127 : }
128 :
129 : GEN
130 1924676 : RgM_ZM_mul_worker(GEN y, GEN x)
131 1924676 : { return RgM_ZC_mul_i(x, y, lg(x), lgcols(x)); }
132 :
133 : /* mostly useful when y is sparse */
134 : GEN
135 354254 : RgM_ZM_mul(GEN x, GEN y)
136 : {
137 354254 : pari_sp av = avma;
138 : GEN worker;
139 354254 : if (lg(x) == 1) return cgetg(lg(y), t_MAT);
140 354254 : worker = snm_closure(is_entry("_RgM_ZM_mul_worker"),mkvec(x));
141 354260 : return gerepileupto(av, gen_parapply(worker, y));
142 : }
143 :
144 : static GEN
145 202611 : RgV_zc_mul_i(GEN x, GEN y, long l)
146 : {
147 : long i;
148 202611 : GEN z = gen_0;
149 202611 : pari_sp av = avma;
150 7603046 : for (i = 1; i < l; i++) z = gadd(z, gmulgs(gel(x,i), y[i]));
151 202611 : return gerepileupto(av, z);
152 : }
153 : GEN
154 83384 : RgV_zc_mul(GEN x, GEN y) { return RgV_zc_mul_i(x, y, lg(x)); }
155 :
156 : GEN
157 33602 : RgV_zm_mul(GEN x, GEN y)
158 : {
159 33602 : long j, l = lg(x), ly = lg(y);
160 33602 : GEN z = cgetg(ly, t_VEC);
161 152829 : for (j = 1; j < ly; j++) gel(z,j) = RgV_zc_mul_i(x, gel(y,j), l);
162 33603 : return z;
163 : }
164 :
165 : /* scalar product x.x */
166 : GEN
167 9437012 : RgV_dotsquare(GEN x)
168 : {
169 9437012 : long i, lx = lg(x);
170 9437012 : pari_sp av = avma;
171 : GEN z;
172 9437012 : if (lx == 1) return gen_0;
173 9437012 : z = gsqr(gel(x,1));
174 21142969 : for (i=2; i<lx; i++)
175 : {
176 11706201 : z = gadd(z, gsqr(gel(x,i)));
177 11706152 : if (gc_needed(av,3))
178 : {
179 0 : if(DEBUGMEM>1) pari_warn(warnmem,"RgV_dotsquare, i = %ld",i);
180 0 : z = gerepileupto(av, z);
181 : }
182 : }
183 9436768 : return gerepileupto(av,z);
184 : }
185 :
186 : /* scalar product x.y, lx = lg(x) = lg(y) */
187 : static GEN
188 12636787 : RgV_dotproduct_i(GEN x, GEN y, long lx)
189 : {
190 12636787 : pari_sp av = avma;
191 : long i;
192 : GEN z;
193 12636787 : if (lx == 1) return gen_0;
194 12636388 : z = gmul(gel(x,1),gel(y,1));
195 213846124 : for (i=2; i<lx; i++)
196 : {
197 201211921 : z = gadd(z, gmul(gel(x,i), gel(y,i)));
198 201211167 : if (gc_needed(av,3))
199 : {
200 0 : if(DEBUGMEM>1) pari_warn(warnmem,"RgV_dotproduct, i = %ld",i);
201 0 : z = gerepileupto(av, z);
202 : }
203 : }
204 12634203 : return gerepileupto(av,z);
205 : }
206 : GEN
207 1000171 : RgV_dotproduct(GEN x,GEN y)
208 : {
209 1000171 : if (x == y) return RgV_dotsquare(x);
210 1000171 : return RgV_dotproduct_i(x, y, lg(x));
211 : }
212 : /* v[1] + ... + v[lg(v)-1] */
213 : GEN
214 3031337 : RgV_sum(GEN v)
215 : {
216 : GEN p;
217 3031337 : long i, l = lg(v);
218 3031337 : if (l == 1) return gen_0;
219 7376623 : p = gel(v,1); for (i=2; i<l; i++) p = gadd(p, gel(v,i));
220 3031268 : return p;
221 : }
222 : /* v[1] + ... + v[n]. Assume lg(v) > n. */
223 : GEN
224 0 : RgV_sumpart(GEN v, long n)
225 : {
226 : GEN p;
227 : long i;
228 0 : if (!n) return gen_0;
229 0 : p = gel(v,1); for (i=2; i<=n; i++) p = gadd(p, gel(v,i));
230 0 : return p;
231 : }
232 : /* v[m] + ... + v[n]. Assume lg(v) > n, m > 0. */
233 : GEN
234 0 : RgV_sumpart2(GEN v, long m, long n)
235 : {
236 : GEN p;
237 : long i;
238 0 : if (n < m) return gen_0;
239 0 : p = gel(v,m); for (i=m+1; i<=n; i++) p = gadd(p, gel(v,i));
240 0 : return p;
241 : }
242 : GEN
243 400 : RgM_sumcol(GEN A)
244 : {
245 400 : long i,j,m,l = lg(A);
246 : GEN v;
247 :
248 400 : if (l == 1) return cgetg(1,t_MAT);
249 400 : if (l == 2) return gcopy(gel(A,1));
250 216 : m = lgcols(A);
251 216 : v = cgetg(m, t_COL);
252 718 : for (i = 1; i < m; i++)
253 : {
254 502 : pari_sp av = avma;
255 502 : GEN s = gcoeff(A,i,1);
256 1214 : for (j = 2; j < l; j++) s = gadd(s, gcoeff(A,i,j));
257 502 : gel(v, i) = gerepileupto(av, s);
258 : }
259 216 : return v;
260 : }
261 :
262 : static GEN
263 3331001 : _gmul(void *data, GEN x, GEN y)
264 3331001 : { (void)data; return gmul(x,y); }
265 :
266 : GEN
267 450511 : RgV_prod(GEN x)
268 : {
269 450511 : return gen_product(x, NULL, _gmul);
270 : }
271 :
272 : /* ADDITION SCALAR + MATRIX */
273 : /* x square matrix, y scalar; create the square matrix x + y*Id */
274 : GEN
275 7834 : RgM_Rg_add(GEN x, GEN y)
276 : {
277 7834 : long l = lg(x), i, j;
278 7834 : GEN z = cgetg(l,t_MAT);
279 :
280 7834 : if (l==1) return z;
281 7834 : if (l != lgcols(x)) pari_err_OP("+", x, y);
282 58026 : for (i=1; i<l; i++)
283 : {
284 50192 : GEN zi = cgetg(l,t_COL), xi = gel(x,i);
285 50192 : gel(z,i) = zi;
286 2058576 : for (j=1; j<l; j++)
287 2008384 : gel(zi,j) = i==j? gadd(y,gel(xi,j)): gcopy(gel(xi,j));
288 : }
289 7834 : return z;
290 : }
291 : GEN
292 0 : RgM_Rg_sub(GEN x, GEN y)
293 : {
294 0 : long l = lg(x), i, j;
295 0 : GEN z = cgetg(l,t_MAT);
296 :
297 0 : if (l==1) return z;
298 0 : if (l != lgcols(x)) pari_err_OP("-", x, y);
299 0 : for (i=1; i<l; i++)
300 : {
301 0 : GEN zi = cgetg(l,t_COL), xi = gel(x,i);
302 0 : gel(z,i) = zi;
303 0 : for (j=1; j<l; j++)
304 0 : gel(zi,j) = i==j? gsub(gel(xi,j), y): gcopy(gel(xi,j));
305 : }
306 0 : return z;
307 : }
308 : GEN
309 739 : RgM_Rg_add_shallow(GEN x, GEN y)
310 : {
311 739 : long l = lg(x), i, j;
312 739 : GEN z = cgetg(l,t_MAT);
313 :
314 739 : if (l==1) return z;
315 739 : if (l != lgcols(x)) pari_err_OP( "+", x, y);
316 2384 : for (i=1; i<l; i++)
317 : {
318 1645 : GEN zi = cgetg(l,t_COL), xi = gel(x,i);
319 1645 : gel(z,i) = zi;
320 9456 : for (j=1; j<l; j++) gel(zi,j) = gel(xi,j);
321 1645 : gel(zi,i) = gadd(gel(zi,i), y);
322 : }
323 739 : return z;
324 : }
325 : GEN
326 311362 : RgM_Rg_sub_shallow(GEN x, GEN y)
327 : {
328 311362 : long l = lg(x), i, j;
329 311362 : GEN z = cgetg(l,t_MAT);
330 :
331 311362 : if (l==1) return z;
332 311362 : if (l != lgcols(x)) pari_err_OP( "-", x, y);
333 2009026 : for (i=1; i<l; i++)
334 : {
335 1697674 : GEN zi = cgetg(l,t_COL), xi = gel(x,i);
336 1697671 : gel(z,i) = zi;
337 27111311 : for (j=1; j<l; j++) gel(zi,j) = gel(xi,j);
338 1697671 : gel(zi,i) = gsub(gel(zi,i), y);
339 : }
340 311352 : return z;
341 : }
342 :
343 : GEN
344 4728217 : RgC_Rg_add(GEN x, GEN y)
345 : {
346 4728217 : long k, lx = lg(x);
347 4728217 : GEN z = cgetg(lx, t_COL);
348 4728221 : if (lx == 1)
349 : {
350 0 : if (isintzero(y)) return z;
351 0 : pari_err_TYPE2("+",x,y);
352 : }
353 4728221 : gel(z,1) = gadd(y,gel(x,1));
354 11427782 : for (k = 2; k < lx; k++) gel(z,k) = gcopy(gel(x,k));
355 4728203 : return z;
356 : }
357 : GEN
358 41700 : RgC_Rg_sub(GEN x, GEN y)
359 : {
360 41700 : long k, lx = lg(x);
361 41700 : GEN z = cgetg(lx, t_COL);
362 41700 : if (lx == 1)
363 : {
364 0 : if (isintzero(y)) return z;
365 0 : pari_err_TYPE2("-",x,y);
366 : }
367 41700 : gel(z,1) = gsub(gel(x,1), y);
368 107735 : for (k = 2; k < lx; k++) gel(z,k) = gcopy(gel(x,k));
369 41700 : return z;
370 : }
371 : /* a - x */
372 : GEN
373 135295 : Rg_RgC_sub(GEN a, GEN x)
374 : {
375 135295 : long k, lx = lg(x);
376 135295 : GEN z = cgetg(lx,t_COL);
377 135295 : if (lx == 1)
378 : {
379 0 : if (isintzero(a)) return z;
380 0 : pari_err_TYPE2("-",a,x);
381 : }
382 135295 : gel(z,1) = gsub(a, gel(x,1));
383 305668 : for (k = 2; k < lx; k++) gel(z,k) = gneg(gel(x,k));
384 135295 : return z;
385 : }
386 :
387 : static GEN
388 18993706 : RgC_add_i(GEN x, GEN y, long lx)
389 : {
390 18993706 : GEN A = cgetg(lx, t_COL);
391 : long i;
392 134663698 : for (i=1; i<lx; i++) gel(A,i) = gadd(gel(x,i), gel(y,i));
393 18993334 : return A;
394 : }
395 : GEN
396 15728315 : RgC_add(GEN x, GEN y) { return RgC_add_i(x, y, lg(x)); }
397 : GEN
398 2541182 : RgV_add(GEN x, GEN y)
399 10428576 : { pari_APPLY_type(t_VEC, gadd(gel(x,i), gel(y,i))) }
400 :
401 : static GEN
402 5385906 : RgC_sub_i(GEN x, GEN y, long lx)
403 : {
404 : long i;
405 5385906 : GEN A = cgetg(lx, t_COL);
406 629969375 : for (i=1; i<lx; i++) gel(A,i) = gsub(gel(x,i), gel(y,i));
407 5384786 : return A;
408 : }
409 : GEN
410 4491002 : RgC_sub(GEN x, GEN y) { return RgC_sub_i(x, y, lg(x)); }
411 : GEN
412 355366 : RgV_sub(GEN x, GEN y)
413 3011394 : { pari_APPLY_type(t_VEC, gsub(gel(x,i), gel(y,i))) }
414 :
415 : GEN
416 690522 : RgM_add(GEN x, GEN y)
417 : {
418 690522 : long lx = lg(x), l, j;
419 : GEN z;
420 690522 : if (lx == 1) return cgetg(1, t_MAT);
421 690522 : z = cgetg(lx, t_MAT); l = lgcols(x);
422 3955910 : for (j = 1; j < lx; j++) gel(z,j) = RgC_add_i(gel(x,j), gel(y,j), l);
423 690522 : return z;
424 : }
425 : GEN
426 171907 : RgM_sub(GEN x, GEN y)
427 : {
428 171907 : long lx = lg(x), l, j;
429 : GEN z;
430 171907 : if (lx == 1) return cgetg(1, t_MAT);
431 171907 : z = cgetg(lx, t_MAT); l = lgcols(x);
432 1066870 : for (j = 1; j < lx; j++) gel(z,j) = RgC_sub_i(gel(x,j), gel(y,j), l);
433 171906 : return z;
434 : }
435 :
436 : static GEN
437 3972716 : RgC_neg_i(GEN x, long lx)
438 : {
439 : long i;
440 3972716 : GEN y = cgetg(lx, t_COL);
441 30145000 : for (i=1; i<lx; i++) gel(y,i) = gneg(gel(x,i));
442 3972714 : return y;
443 : }
444 : GEN
445 988792 : RgC_neg(GEN x) { return RgC_neg_i(x, lg(x)); }
446 : GEN
447 4515 : RgV_neg(GEN x)
448 60473 : { pari_APPLY_type(t_VEC, gneg(gel(x,i))) }
449 : GEN
450 541169 : RgM_neg(GEN x)
451 : {
452 541169 : long i, hx, lx = lg(x);
453 541169 : GEN y = cgetg(lx, t_MAT);
454 541169 : if (lx == 1) return y;
455 541162 : hx = lgcols(x);
456 3525084 : for (i=1; i<lx; i++) gel(y,i) = RgC_neg_i(gel(x,i), hx);
457 541162 : return y;
458 : }
459 :
460 : GEN
461 3796919 : RgV_RgC_mul(GEN x, GEN y)
462 : {
463 3796919 : long lx = lg(x);
464 3796919 : if (lx != lg(y)) pari_err_OP("operation 'RgV_RgC_mul'", x, y);
465 3796835 : return RgV_dotproduct_i(x, y, lx);
466 : }
467 : GEN
468 1687 : RgC_RgV_mul(GEN x, GEN y)
469 : {
470 1687 : long i, ly = lg(y);
471 1687 : GEN z = cgetg(ly,t_MAT);
472 5061 : for (i=1; i<ly; i++) gel(z,i) = RgC_Rg_mul(x, gel(y,i));
473 1687 : return z;
474 : }
475 : GEN
476 0 : RgC_RgM_mul(GEN x, GEN y)
477 : {
478 0 : long i, ly = lg(y);
479 0 : GEN z = cgetg(ly,t_MAT);
480 0 : if (ly != 1 && lgcols(y) != 2) pari_err_OP("operation 'RgC_RgM_mul'",x,y);
481 0 : for (i=1; i<ly; i++) gel(z,i) = RgC_Rg_mul(x, gcoeff(y,1,i));
482 0 : return z;
483 : }
484 : GEN
485 0 : RgM_RgV_mul(GEN x, GEN y)
486 : {
487 0 : if (lg(x) != 2) pari_err_OP("operation 'RgM_RgV_mul'", x,y);
488 0 : return RgC_RgV_mul(gel(x,1), y);
489 : }
490 :
491 : /* x[i,]*y, l = lg(y) > 1 */
492 : static GEN
493 114077496 : RgMrow_RgC_mul_i(GEN x, GEN y, long i, long l)
494 : {
495 114077496 : pari_sp av = avma;
496 114077496 : GEN t = gmul(gcoeff(x,i,1), gel(y,1)); /* l > 1 ! */
497 : long j;
498 975687041 : for (j=2; j<l; j++)
499 : {
500 861612872 : GEN c = gcoeff(x,i,j);
501 861612872 : if (!isintzero(c)) t = gadd(t, gmul(c, gel(y,j)));
502 : }
503 114074169 : return gerepileupto(av,t);
504 : }
505 : GEN
506 4641 : RgMrow_RgC_mul(GEN x, GEN y, long i)
507 4641 : { return RgMrow_RgC_mul_i(x, y, i, lg(x)); }
508 :
509 : static GEN
510 28 : RgM_RgC_mul_FpM(GEN x, GEN y, GEN p)
511 : {
512 28 : pari_sp av = avma;
513 : GEN r;
514 28 : if (lgefint(p) == 3)
515 : {
516 14 : ulong pp = uel(p, 2);
517 14 : r = Flm_Flc_mul(RgM_to_Flm(x, pp), RgV_to_Flv(y, pp), pp);
518 14 : r = Flc_to_ZC_inplace(r);
519 : }
520 : else
521 14 : r = FpM_FpC_mul(RgM_to_FpM(x, p), RgC_to_FpC(y, p), p);
522 28 : return gerepileupto(av, FpC_to_mod(r, p));
523 : }
524 :
525 : static GEN
526 28 : RgM_RgC_mul_FqM(GEN x, GEN y, GEN pol, GEN p)
527 : {
528 28 : pari_sp av = avma;
529 28 : GEN b, T = RgX_to_FpX(pol, p);
530 28 : if (signe(T) == 0) pari_err_OP("*", x, y);
531 28 : b = FqM_FqC_mul(RgM_to_FqM(x, T, p), RgC_to_FqC(y, T, p), T, p);
532 28 : return gerepileupto(av, FqC_to_mod(b, T, p));
533 : }
534 :
535 : #define code(t1,t2) ((t1 << 6) | t2)
536 : static GEN
537 29952745 : RgM_RgC_mul_fast(GEN x, GEN y)
538 : {
539 : GEN p, pol;
540 : long pa;
541 29952745 : long t = RgM_RgC_type(x,y, &p,&pol,&pa);
542 29953025 : switch(t)
543 : {
544 8050518 : case t_INT: return ZM_ZC_mul(x,y);
545 142068 : case t_FRAC: return QM_QC_mul(x,y);
546 91 : case t_FFELT: return FFM_FFC_mul(x, y, pol);
547 28 : case t_INTMOD: return RgM_RgC_mul_FpM(x, y, p);
548 29 : case code(t_POLMOD, t_INTMOD):
549 29 : return RgM_RgC_mul_FqM(x, y, pol, p);
550 21760291 : default: return NULL;
551 : }
552 : }
553 : #undef code
554 :
555 : /* compatible t_MAT * t_COL, lx = lg(x) = lg(y) > 1, l = lgcols(x) */
556 : static GEN
557 33517417 : RgM_RgC_mul_i(GEN x, GEN y, long lx, long l)
558 : {
559 33517417 : GEN z = cgetg(l,t_COL);
560 : long i;
561 147585152 : for (i=1; i<l; i++) gel(z,i) = RgMrow_RgC_mul_i(x,y,i,lx);
562 33512819 : return z;
563 : }
564 :
565 : GEN
566 29952686 : RgM_RgC_mul(GEN x, GEN y)
567 : {
568 29952686 : long lx = lg(x);
569 : GEN z;
570 29952686 : if (lx != lg(y)) pari_err_OP("operation 'RgM_RgC_mul'", x,y);
571 29952737 : if (lx == 1) return cgetg(1,t_COL);
572 29952737 : z = RgM_RgC_mul_fast(x, y);
573 29952988 : if (z) return z;
574 21760256 : return RgM_RgC_mul_i(x, y, lx, lgcols(x));
575 : }
576 :
577 : GEN
578 293751 : RgV_RgM_mul(GEN x, GEN y)
579 : {
580 293751 : long i, lx, ly = lg(y);
581 : GEN z;
582 293751 : if (ly == 1) return cgetg(1,t_VEC);
583 293744 : lx = lg(x);
584 293744 : if (lx != lgcols(y)) pari_err_OP("operation 'RgV_RgM_mul'", x,y);
585 293737 : z = cgetg(ly, t_VEC);
586 3397346 : for (i=1; i<ly; i++) gel(z,i) = RgV_dotproduct_i(x, gel(y,i), lx);
587 293660 : return z;
588 : }
589 :
590 : static GEN
591 56238 : RgM_mul_FpM_i(GEN x, GEN y, GEN p)
592 : {
593 56238 : if (lgefint(p) == 3)
594 : {
595 56166 : ulong pp = uel(p, 2);
596 56166 : if (pp==2) return F2m_to_mod(F2m_mul(RgM_to_F2m(x), RgM_to_F2m(y)));
597 56117 : if (pp==3) return F3m_to_mod(F3m_mul(RgM_to_F3m(x), RgM_to_F3m(y)));
598 56103 : return Flm_to_mod(Flm_mul(RgM_to_Flm(x, pp), RgM_to_Flm(y, pp), pp), pp);
599 : }
600 72 : return FpM_to_mod(FpM_mul(RgM_to_FpM(x,p), RgM_to_FpM(y,p), p), p);
601 : }
602 : static GEN
603 56238 : RgM_mul_FpM(GEN x, GEN y, GEN p)
604 56238 : { pari_sp av = avma; return gerepileupto(av, RgM_mul_FpM_i(x, y, p)); }
605 :
606 : static GEN
607 26005 : RgM_mul_FqM(GEN x, GEN y, GEN pol, GEN p)
608 : {
609 26005 : pari_sp av = avma;
610 26005 : GEN b, T = RgX_to_FpX(pol, p);
611 26005 : if (signe(T) == 0) pari_err_OP("*", x, y);
612 26005 : b = FqM_mul(RgM_to_FqM(x, T, p), RgM_to_FqM(y, T, p), T, p);
613 26005 : return gerepileupto(av, FqM_to_mod(b, T, p));
614 : }
615 :
616 : static GEN
617 11774 : RgM_liftred(GEN x, GEN T)
618 11774 : { return RgXQM_red(liftpol_shallow(x), T); }
619 :
620 : static GEN
621 1421 : RgM_mul_ZXQM(GEN x, GEN y, GEN T)
622 : {
623 1421 : pari_sp av = avma;
624 1421 : GEN b = ZXQM_mul(RgM_liftred(x,T), RgM_liftred(y, T), T);
625 1421 : return gerepilecopy(av, QXQM_to_mod_shallow(b,T));
626 : }
627 :
628 : static GEN
629 133 : RgM_sqr_ZXQM(GEN x, GEN T)
630 : {
631 133 : pari_sp av = avma;
632 133 : GEN b = ZXQM_sqr(RgM_liftred(x, T), T);
633 133 : return gerepilecopy(av, QXQM_to_mod_shallow(b,T));
634 : }
635 :
636 : static GEN
637 4396 : RgM_mul_QXQM(GEN x, GEN y, GEN T)
638 : {
639 4396 : pari_sp av = avma;
640 4396 : GEN b = QXQM_mul(RgM_liftred(x, T), RgM_liftred(y, T), T);
641 4396 : return gerepilecopy(av, QXQM_to_mod_shallow(b,T));
642 : }
643 :
644 : static GEN
645 7 : RgM_sqr_QXQM(GEN x, GEN T)
646 : {
647 7 : pari_sp av = avma;
648 7 : GEN b = QXQM_sqr(RgM_liftred(x, T), T);
649 7 : return gerepilecopy(av, QXQM_to_mod_shallow(b,T));
650 : }
651 :
652 : INLINE int
653 4445 : RgX_is_monic_ZX(GEN pol)
654 4445 : { return RgX_is_ZX(pol) && ZX_is_monic(pol); }
655 :
656 : #define code(t1,t2) ((t1 << 6) | t2)
657 : static GEN
658 7537999 : RgM_mul_fast(GEN x, GEN y)
659 : {
660 : GEN p, pol;
661 : long pa;
662 7537999 : long t = RgM_type2(x,y, &p,&pol,&pa);
663 7538047 : switch(t)
664 : {
665 2667098 : case t_INT: return ZM_mul(x,y);
666 115738 : case t_FRAC: return QM_mul(x,y);
667 4200 : case t_FFELT: return FFM_mul(x, y, pol);
668 56175 : case t_INTMOD: return RgM_mul_FpM(x, y, p);
669 1428 : case code(t_POLMOD, t_INT):
670 1428 : return ZX_is_monic(pol)? RgM_mul_ZXQM(x, y, pol): NULL;
671 4417 : case code(t_POLMOD, t_FRAC):
672 4417 : return RgX_is_monic_ZX(pol)? RgM_mul_QXQM(x, y, pol): NULL;
673 26003 : case code(t_POLMOD, t_INTMOD):
674 26003 : return RgM_mul_FqM(x, y, pol, p);
675 4662988 : default: return NULL;
676 : }
677 : }
678 :
679 : static GEN
680 1302 : RgM_sqr_fast(GEN x)
681 : {
682 : GEN p, pol;
683 : long pa;
684 1302 : long t = RgM_type(x, &p,&pol,&pa);
685 1302 : switch(t)
686 : {
687 168 : case t_INT: return ZM_sqr(x);
688 700 : case t_FRAC: return QM_sqr(x);
689 105 : case t_FFELT: return FFM_mul(x, x, pol);
690 63 : case t_INTMOD: return RgM_mul_FpM(x, x, p);
691 140 : case code(t_POLMOD, t_INT):
692 140 : return ZX_is_monic(pol)? RgM_sqr_ZXQM(x, pol): NULL;
693 28 : case code(t_POLMOD, t_FRAC):
694 28 : return RgX_is_monic_ZX(pol)? RgM_sqr_QXQM(x, pol): NULL;
695 0 : case code(t_POLMOD, t_INTMOD):
696 0 : return RgM_mul_FqM(x, x, pol, p);
697 98 : default: return NULL;
698 : }
699 : }
700 :
701 : #undef code
702 :
703 : /* lx, ly > 1 */
704 : static GEN
705 4663014 : RgM_mul_i(GEN x, GEN y, long lx, long ly)
706 : {
707 4663014 : GEN z = cgetg(ly, t_MAT);
708 4663010 : long j, l = lgcols(x);
709 16419971 : for (j = 1; j < ly; j++) gel(z,j) = RgM_RgC_mul_i(x, gel(y,j), lx, l);
710 4662981 : return z;
711 : }
712 : GEN
713 7552294 : RgM_mul(GEN x, GEN y)
714 : {
715 7552294 : long lx, ly = lg(y);
716 : GEN z;
717 7552294 : if (ly == 1) return cgetg(1,t_MAT);
718 7538028 : lx = lg(x);
719 7538028 : if (lx != lgcols(y)) pari_err_OP("operation 'RgM_mul'", x,y);
720 7538028 : if (lx == 1) return zeromat(0,ly-1);
721 7538000 : z = RgM_mul_fast(x, y);
722 7538044 : if (z) return z;
723 4663011 : return RgM_mul_i(x, y, lx, ly);
724 : }
725 :
726 : GEN
727 1337 : RgM_sqr(GEN x)
728 : {
729 1337 : long j, lx = lg(x);
730 : GEN z;
731 1337 : if (lx == 1) return cgetg(1, t_MAT);
732 1302 : if (lx != lgcols(x)) pari_err_OP("operation 'RgM_mul'", x,x);
733 1302 : z = RgM_sqr_fast(x);
734 1302 : if (z) return z;
735 126 : z = cgetg(lx, t_MAT);
736 497 : for (j=1; j<lx; j++) gel(z,j) = RgM_RgC_mul_i(x, gel(x,j), lx, lx);
737 126 : return z;
738 : }
739 :
740 : /* assume result is symmetric */
741 : GEN
742 0 : RgM_multosym(GEN x, GEN y)
743 : {
744 0 : long j, lx, ly = lg(y);
745 : GEN M;
746 0 : if (ly == 1) return cgetg(1,t_MAT);
747 0 : lx = lg(x);
748 0 : if (lx != lgcols(y)) pari_err_OP("operation 'RgM_multosym'", x,y);
749 0 : if (lx == 1) return cgetg(1,t_MAT);
750 0 : if (ly != lgcols(x)) pari_err_OP("operation 'RgM_multosym'", x,y);
751 0 : M = cgetg(ly, t_MAT);
752 0 : for (j=1; j<ly; j++)
753 : {
754 0 : GEN z = cgetg(ly,t_COL), yj = gel(y,j);
755 : long i;
756 0 : for (i=1; i<j; i++) gel(z,i) = gcoeff(M,j,i);
757 0 : for (i=j; i<ly; i++)gel(z,i) = RgMrow_RgC_mul_i(x,yj,i,lx);
758 0 : gel(M,j) = z;
759 : }
760 0 : return M;
761 : }
762 : /* x~ * y, assuming result is symmetric */
763 : GEN
764 7558 : RgM_transmultosym(GEN x, GEN y)
765 : {
766 7558 : long i, j, l, ly = lg(y);
767 : GEN M;
768 7558 : if (ly == 1) return cgetg(1,t_MAT);
769 7558 : if (lg(x) != ly) pari_err_OP("operation 'RgM_transmultosym'", x,y);
770 7558 : l = lgcols(y);
771 7558 : if (lgcols(x) != l) pari_err_OP("operation 'RgM_transmultosym'", x,y);
772 7558 : M = cgetg(ly, t_MAT);
773 30400 : for (i=1; i<ly; i++)
774 : {
775 22842 : GEN xi = gel(x,i), c = cgetg(ly,t_COL);
776 22842 : gel(M,i) = c;
777 46860 : for (j=1; j<i; j++)
778 24018 : gcoeff(M,i,j) = gel(c,j) = RgV_dotproduct_i(xi,gel(y,j),l);
779 22842 : gel(c,i) = RgV_dotproduct_i(xi,gel(y,i),l);
780 : }
781 7558 : return M;
782 : }
783 : /* x~ * y */
784 : GEN
785 0 : RgM_transmul(GEN x, GEN y)
786 : {
787 0 : long i, j, l, lx, ly = lg(y);
788 : GEN M;
789 0 : if (ly == 1) return cgetg(1,t_MAT);
790 0 : lx = lg(x);
791 0 : l = lgcols(y);
792 0 : if (lgcols(x) != l) pari_err_OP("operation 'RgM_transmul'", x,y);
793 0 : M = cgetg(ly, t_MAT);
794 0 : for (i=1; i<ly; i++)
795 : {
796 0 : GEN yi = gel(y,i), c = cgetg(lx,t_COL);
797 0 : gel(M,i) = c;
798 0 : for (j=1; j<lx; j++) gel(c,j) = RgV_dotproduct_i(yi,gel(x,j),l);
799 : }
800 0 : return M;
801 : }
802 :
803 : GEN
804 4689312 : gram_matrix(GEN x)
805 : {
806 4689312 : long i,j, l, lx = lg(x);
807 : GEN M;
808 4689312 : if (!is_matvec_t(typ(x))) pari_err_TYPE("gram",x);
809 4689312 : if (lx == 1) return cgetg(1,t_MAT);
810 4689298 : l = lgcols(x);
811 4689295 : M = cgetg(lx,t_MAT);
812 14067814 : for (i=1; i<lx; i++)
813 : {
814 9378541 : GEN xi = gel(x,i), c = cgetg(lx,t_COL);
815 9378539 : gel(M,i) = c;
816 14067800 : for (j=1; j<i; j++)
817 4689268 : gcoeff(M,i,j) = gel(c,j) = RgV_dotproduct_i(xi,gel(x,j),l);
818 9378532 : gel(c,i) = RgV_dotsquare(xi);
819 : }
820 4689273 : return M;
821 : }
822 :
823 : static GEN
824 3801 : _RgM_add(void *E, GEN x, GEN y) { (void)E; return RgM_add(x, y); }
825 :
826 : static GEN
827 0 : _RgM_sub(void *E, GEN x, GEN y) { (void)E; return RgM_sub(x, y); }
828 :
829 : static GEN
830 5684 : _RgM_cmul(void *E, GEN P, long a, GEN x) { (void)E; return RgM_Rg_mul(x,gel(P,a+2)); }
831 :
832 : static GEN
833 238 : _RgM_sqr(void *E, GEN x) { (void) E; return RgM_sqr(x); }
834 :
835 : static GEN
836 749 : _RgM_mul(void *E, GEN x, GEN y) { (void) E; return RgM_mul(x, y); }
837 :
838 : static GEN
839 4046 : _RgM_one(void *E) { long *n = (long*) E; return matid(*n); }
840 :
841 : static GEN
842 0 : _RgM_zero(void *E) { long *n = (long*) E; return zeromat(*n,*n); }
843 :
844 : static GEN
845 2828 : _RgM_red(void *E, GEN x) { (void)E; return x; }
846 :
847 : static struct bb_algebra RgM_algebra = { _RgM_red, _RgM_add, _RgM_sub,
848 : _RgM_mul, _RgM_sqr, _RgM_one, _RgM_zero };
849 :
850 : /* generates the list of powers of x of degree 0,1,2,...,l*/
851 : GEN
852 168 : RgM_powers(GEN x, long l)
853 : {
854 168 : long n = lg(x)-1;
855 168 : return gen_powers(x,l,1,(void *) &n, &_RgM_sqr, &_RgM_mul, &_RgM_one);
856 : }
857 :
858 : GEN
859 490 : RgX_RgMV_eval(GEN Q, GEN x)
860 : {
861 490 : long n = lg(x)>1 ? lg(gel(x,1))-1:0;
862 490 : return gen_bkeval_powers(Q,degpol(Q),x,(void*)&n,&RgM_algebra,&_RgM_cmul);
863 : }
864 :
865 : GEN
866 1393 : RgX_RgM_eval(GEN Q, GEN x)
867 : {
868 1393 : long n = lg(x)-1;
869 1393 : return gen_bkeval(Q,degpol(Q),x,1,(void*)&n,&RgM_algebra,&_RgM_cmul);
870 : }
871 :
872 : GEN
873 4438886 : RgC_Rg_div(GEN x, GEN y)
874 19100117 : { pari_APPLY_type(t_COL, gdiv(gel(x,i),y)) }
875 :
876 : GEN
877 12402008 : RgC_Rg_mul(GEN x, GEN y)
878 63484710 : { pari_APPLY_type(t_COL, gmul(gel(x,i),y)) }
879 :
880 : GEN
881 30604 : RgV_Rg_mul(GEN x, GEN y)
882 1170785 : { pari_APPLY_type(t_VEC, gmul(gel(x,i),y)) }
883 :
884 : GEN
885 607888 : RgM_Rg_div(GEN X, GEN c) {
886 607888 : long i, j, h, l = lg(X);
887 607888 : GEN A = cgetg(l, t_MAT);
888 607888 : if (l == 1) return A;
889 607888 : h = lgcols(X);
890 2784755 : for (j=1; j<l; j++)
891 : {
892 2176916 : GEN a = cgetg(h, t_COL), x = gel(X, j);
893 17244115 : for (i = 1; i < h; i++) gel(a,i) = gdiv(gel(x,i), c);
894 2176866 : gel(A,j) = a;
895 : }
896 607839 : return A;
897 : }
898 : GEN
899 243224 : RgM_Rg_mul(GEN X, GEN c) {
900 243224 : long i, j, h, l = lg(X);
901 243224 : GEN A = cgetg(l, t_MAT);
902 243226 : if (l == 1) return A;
903 243226 : h = lgcols(X);
904 1066178 : for (j=1; j<l; j++)
905 : {
906 822957 : GEN a = cgetg(h, t_COL), x = gel(X, j);
907 6883402 : for (i = 1; i < h; i++) gel(a,i) = gmul(gel(x,i), c);
908 822951 : gel(A,j) = a;
909 : }
910 243221 : return A;
911 : }
912 :
913 : /********************************************************************/
914 : /* */
915 : /* SCALAR TO MATRIX/VECTOR */
916 : /* */
917 : /********************************************************************/
918 : /* fill the square nxn matrix equal to t*Id */
919 : static void
920 13132426 : fill_scalmat(GEN y, GEN t, long n)
921 : {
922 : long i;
923 49836061 : for (i = 1; i <= n; i++)
924 : {
925 36703746 : gel(y,i) = zerocol(n);
926 36703635 : gcoeff(y,i,i) = t;
927 : }
928 13132315 : }
929 :
930 : GEN
931 920422 : scalarmat(GEN x, long n) {
932 920422 : GEN y = cgetg(n+1, t_MAT);
933 920399 : if (!n) return y;
934 920399 : fill_scalmat(y, gcopy(x), n); return y;
935 : }
936 : GEN
937 3151950 : scalarmat_shallow(GEN x, long n) {
938 3151950 : GEN y = cgetg(n+1, t_MAT);
939 3152027 : fill_scalmat(y, x, n); return y;
940 : }
941 : GEN
942 217 : scalarmat_s(long x, long n) {
943 217 : GEN y = cgetg(n+1, t_MAT);
944 217 : if (!n) return y;
945 217 : fill_scalmat(y, stoi(x), n); return y;
946 : }
947 : GEN
948 9060450 : matid(long n) {
949 : GEN y;
950 9060450 : if (n < 0) pari_err_DOMAIN("matid", "size", "<", gen_0, stoi(n));
951 9060443 : y = cgetg(n+1, t_MAT);
952 9060526 : fill_scalmat(y, gen_1, n); return y;
953 : }
954 :
955 : INLINE GEN
956 2228466 : scalarcol_i(GEN x, long n, long c)
957 : {
958 : long i;
959 2228466 : GEN y = cgetg(n+1,t_COL);
960 2228457 : if (!n) return y;
961 2228457 : gel(y,1) = c? gcopy(x): x;
962 6666952 : for (i=2; i<=n; i++) gel(y,i) = gen_0;
963 2228455 : return y;
964 : }
965 :
966 : GEN
967 751098 : scalarcol(GEN x, long n) { return scalarcol_i(x,n,1); }
968 :
969 : GEN
970 1477370 : scalarcol_shallow(GEN x, long n) { return scalarcol_i(x,n,0); }
971 :
972 : int
973 32622 : RgM_isscalar(GEN x, GEN s)
974 : {
975 32622 : long i, j, lx = lg(x);
976 :
977 32622 : if (lx == 1) return 1;
978 32622 : if (lx != lgcols(x)) return 0;
979 32622 : if (!s) s = gcoeff(x,1,1);
980 :
981 90508 : for (j=1; j<lx; j++)
982 : {
983 73753 : GEN c = gel(x,j);
984 200521 : for (i=1; i<j; )
985 140101 : if (!gequal0(gel(c,i++))) return 0;
986 : /* i = j */
987 60420 : if (!gequal(gel(c,i++),s)) return 0;
988 223499 : for ( ; i<lx; )
989 165613 : if (!gequal0(gel(c,i++))) return 0;
990 : }
991 16755 : return 1;
992 : }
993 :
994 : int
995 15175 : RgM_isidentity(GEN x)
996 : {
997 15175 : long i,j, lx = lg(x);
998 :
999 15175 : if (lx == 1) return 1;
1000 15175 : if (lx != lgcols(x)) return 0;
1001 24885 : for (j=1; j<lx; j++)
1002 : {
1003 24724 : GEN c = gel(x,j);
1004 57812 : for (i=1; i<j; )
1005 38036 : if (!gequal0(gel(c,i++))) return 0;
1006 : /* i = j */
1007 19776 : if (!gequal1(gel(c,i++))) return 0;
1008 60678 : for ( ; i<lx; )
1009 50968 : if (!gequal0(gel(c,i++))) return 0;
1010 : }
1011 161 : return 1;
1012 : }
1013 :
1014 : long
1015 350 : RgC_is_ei(GEN x)
1016 : {
1017 350 : long i, j = 0, l = lg(x);
1018 2002 : for (i = 1; i < l; i++)
1019 : {
1020 1652 : GEN c = gel(x,i);
1021 1652 : if (gequal0(c)) continue;
1022 350 : if (!gequal1(c) || j) return 0;
1023 350 : j = i;
1024 : }
1025 350 : return j;
1026 : }
1027 :
1028 : int
1029 336 : RgM_isdiagonal(GEN x)
1030 : {
1031 336 : long i,j, lx = lg(x);
1032 336 : if (lx == 1) return 1;
1033 336 : if (lx != lgcols(x)) return 0;
1034 :
1035 3220 : for (j=1; j<lx; j++)
1036 : {
1037 2891 : GEN c = gel(x,j);
1038 18452 : for (i=1; i<j; i++)
1039 15561 : if (!gequal0(gel(c,i))) return 0;
1040 18452 : for (i++; i<lx; i++)
1041 15568 : if (!gequal0(gel(c,i))) return 0;
1042 : }
1043 329 : return 1;
1044 : }
1045 : int
1046 315 : isdiagonal(GEN x) { return (typ(x)==t_MAT) && RgM_isdiagonal(x); }
1047 :
1048 : GEN
1049 21071 : RgM_det_triangular(GEN mat)
1050 : {
1051 21071 : long i,l = lg(mat);
1052 : pari_sp av;
1053 : GEN s;
1054 :
1055 21071 : if (l<3) return l<2? gen_1: gcopy(gcoeff(mat,1,1));
1056 19545 : av = avma; s = gcoeff(mat,1,1);
1057 119481 : for (i=2; i<l; i++) s = gmul(s,gcoeff(mat,i,i));
1058 19545 : return av==avma? gcopy(s): gerepileupto(av,s);
1059 : }
1060 :
1061 : GEN
1062 478034 : RgV_kill0(GEN v)
1063 : {
1064 : long i, l;
1065 478034 : GEN w = cgetg_copy(v, &l);
1066 130072809 : for (i = 1; i < l; i++)
1067 : {
1068 129594750 : GEN a = gel(v,i);
1069 129594750 : gel(w,i) = gequal0(a) ? NULL: a;
1070 : }
1071 478059 : return w;
1072 : }
|