minv.c 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. /* minv.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. #include "ccmath.h"
  10. int minv(double *a, int n)
  11. {
  12. int lc, *le;
  13. double s, t, tq = 0., zr = 1.e-15;
  14. double *pa, *pd, *ps, *p, *q, *q0;
  15. int i, j, k, m;
  16. le = (int *)malloc(n * sizeof(int));
  17. q0 = (double *)malloc(n * sizeof(double));
  18. for (j = 0, pa = pd = a; j < n; ++j, ++pa, pd += n + 1) {
  19. if (j > 0) {
  20. for (i = 0, q = q0, p = pa; i < n; ++i, p += n)
  21. *q++ = *p;
  22. for (i = 1; i < n; ++i) {
  23. lc = i < j ? i : j;
  24. for (k = 0, p = pa + i * n - j, q = q0, t = 0.; k < lc; ++k)
  25. t += *p++ * *q++;
  26. q0[i] -= t;
  27. }
  28. for (i = 0, q = q0, p = pa; i < n; ++i, p += n)
  29. *p = *q++;
  30. }
  31. s = fabs(*pd);
  32. lc = j;
  33. for (k = j + 1, ps = pd; k < n; ++k) {
  34. if ((t = fabs(*(ps += n))) > s) {
  35. s = t;
  36. lc = k;
  37. }
  38. }
  39. tq = tq > s ? tq : s;
  40. if (s < zr * tq) {
  41. free(le - j);
  42. free(q0);
  43. return -1;
  44. }
  45. *le++ = lc;
  46. if (lc != j) {
  47. for (k = 0, p = a + n * j, q = a + n * lc; k < n; ++k) {
  48. t = *p;
  49. *p++ = *q;
  50. *q++ = t;
  51. }
  52. }
  53. for (k = j + 1, ps = pd, t = 1. / *pd; k < n; ++k)
  54. *(ps += n) *= t;
  55. *pd = t;
  56. }
  57. for (j = 1, pd = ps = a; j < n; ++j) {
  58. for (k = 0, pd += n + 1, q = ++ps; k < j; ++k, q += n)
  59. *q *= *pd;
  60. }
  61. for (j = 1, pa = a; j < n; ++j) {
  62. ++pa;
  63. for (i = 0, q = q0, p = pa; i < j; ++i, p += n)
  64. *q++ = *p;
  65. for (k = 0; k < j; ++k) {
  66. t = 0.;
  67. for (i = k, p = pa + k * n + k - j, q = q0 + k; i < j; ++i)
  68. t -= *p++ * *q++;
  69. q0[k] = t;
  70. }
  71. for (i = 0, q = q0, p = pa; i < j; ++i, p += n)
  72. *p = *q++;
  73. }
  74. for (j = n - 2, pd = pa = a + n * n - 1; j >= 0; --j) {
  75. --pa;
  76. pd -= n + 1;
  77. for (i = 0, m = n - j - 1, q = q0, p = pd + n; i < m; ++i, p += n)
  78. *q++ = *p;
  79. for (k = n - 1, ps = pa; k > j; --k, ps -= n) {
  80. t = -(*ps);
  81. for (i = j + 1, p = ps, q = q0; i < k; ++i)
  82. t -= *++p * *q++;
  83. q0[--m] = t;
  84. }
  85. for (i = 0, m = n - j - 1, q = q0, p = pd + n; i < m; ++i, p += n)
  86. *p = *q++;
  87. }
  88. for (k = 0, pa = a; k < n - 1; ++k, ++pa) {
  89. for (i = 0, q = q0, p = pa; i < n; ++i, p += n)
  90. *q++ = *p;
  91. for (j = 0, ps = a; j < n; ++j, ps += n) {
  92. if (j > k) {
  93. t = 0.;
  94. p = ps + j;
  95. i = j;
  96. }
  97. else {
  98. t = q0[j];
  99. p = ps + k + 1;
  100. i = k + 1;
  101. }
  102. for (; i < n;)
  103. t += *p++ * q0[i++];
  104. q0[j] = t;
  105. }
  106. for (i = 0, q = q0, p = pa; i < n; ++i, p += n)
  107. *p = *q++;
  108. }
  109. for (j = n - 2, le--; j >= 0; --j) {
  110. for (k = 0, p = a + j, q = a + *(--le); k < n; ++k, p += n, q += n) {
  111. t = *p;
  112. *p = *q;
  113. *q = t;
  114. }
  115. }
  116. free(le);
  117. free(q0);
  118. return 0;
  119. }