ldumat.c 1.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758
  1. /* ldumat.c CCMATH mathematics library source code.
  2. *
  3. * Copyright (C) 2000 Daniel A. Atkinson All rights reserved.
  4. * This code may be redistributed under the terms of the GNU library
  5. * public license (LGPL). ( See the lgpl.license file for details.)
  6. * ------------------------------------------------------------------------
  7. */
  8. #include <stdlib.h>
  9. void ldumat(double *a, double *u, int m, int n)
  10. {
  11. double *p0, *q0, *p, *q, *w;
  12. int i, j, k, mm;
  13. double s, h;
  14. w = (double *)calloc(m, sizeof(double));
  15. for (i = 0, mm = m * m, q = u; i < mm; ++i)
  16. *q++ = 0.;
  17. p0 = a + n * n - 1;
  18. q0 = u + m * m - 1;
  19. mm = m - n;
  20. i = n - 1;
  21. for (j = 0; j < mm; ++j, q0 -= m + 1)
  22. *q0 = 1.;
  23. if (mm == 0) {
  24. p0 -= n + 1;
  25. *q0 = 1.;
  26. q0 -= m + 1;
  27. --i;
  28. ++mm;
  29. }
  30. for (; i >= 0; --i, ++mm, p0 -= n + 1, q0 -= m + 1) {
  31. if (*p0 != 0.) {
  32. for (j = 0, p = p0 + n, h = 1.; j < mm; p += n)
  33. w[j++] = *p;
  34. h = *p0;
  35. *q0 = 1. - h;
  36. for (j = 0, q = q0 + m; j < mm; q += m)
  37. *q = -h * w[j++];
  38. for (k = i + 1, q = q0 + 1; k < m; ++k) {
  39. for (j = 0, p = q + m, s = 0.; j < mm; p += m)
  40. s += w[j++] * *p;
  41. s *= h;
  42. for (j = 0, p = q + m; j < mm; p += m)
  43. *p -= s * w[j++];
  44. *q++ = -s;
  45. }
  46. }
  47. else {
  48. *q0 = 1.;
  49. for (j = 0, p = q0 + 1, q = q0 + m; j < mm; ++j, q += m)
  50. *q = *p++ = 0.;
  51. }
  52. }
  53. free(w);
  54. }