PageRenderTime 42ms CodeModel.GetById 7ms RepoModel.GetById 0ms app.codeStats 0ms

/cln-1.3.2/src/base/digitseq/cl_DS_mul_kara_sqr.h

#
C Header | 81 lines | 47 code | 0 blank | 34 comment | 11 complexity | d0b362b3b8b0cfde4623958bcc935e03 MD5 | raw file
Possible License(s): GPL-2.0
  1. // Eine vereinfachte Version von mulu_karatsuba für den Fall
  2. // sourceptr1 == sourceptr2 && len1 == len2.
  3. // Weniger Variablen, eine Additionsschleife weniger, eine Kopierschleife
  4. // weniger, und bei rekursiven Aufrufen ist wieder
  5. // sourceptr1 == sourceptr2 && len1 == len2.
  6. static void mulu_karatsuba_square (const uintD* sourceptr, uintC len,
  7. uintD* destptr)
  8. { // Es ist 2 <= len.
  9. CL_SMALL_ALLOCA_STACK;
  10. var uintC prod_len = 2*len;
  11. var uintD* prod_LSDptr = destptr;
  12. var uintC k_hi = floor(len,2); // Länge der High-Teile: floor(len/2) >0
  13. var uintC k_lo = len - k_hi; // Länge der Low-Teile: ceiling(len/2) >0
  14. // Es gilt k_hi <= k_lo <= len, k_lo + k_hi = len.
  15. // Summe x1+x0 berechnen:
  16. var uintD* sum_MSDptr;
  17. var uintC sum_len = k_lo; // = max(k_lo,k_hi)
  18. var uintD* sum_LSDptr;
  19. num_stack_small_alloc_1(sum_len,sum_MSDptr=,sum_LSDptr=);
  20. {var uintD carry = // Hauptteile von x1 und x0 addieren:
  21. add_loop_lsp(sourceptr lspop k_lo,sourceptr,sum_LSDptr,k_hi);
  22. if (!(k_lo==k_hi))
  23. // noch k_lo-k_hi = 1 Digits abzulegen
  24. { mspref(sum_MSDptr,0) = lspref(sourceptr,k_lo-1); // = lspref(sourceptr,k_hi)
  25. if (!(carry==0)) { if (++(mspref(sum_MSDptr,0)) == 0) carry=1; else carry=0; }
  26. }
  27. if (carry) { lsprefnext(sum_MSDptr) = 1; sum_len++; }
  28. }
  29. // Platz für Produkte x0*x0, x1*x1:
  30. { var uintC prodhi_len = 2*k_hi;
  31. var uintD* prodhi_LSDptr = prod_LSDptr lspop 2*k_lo;
  32. // prod_MSDptr/2*len/prod_LSDptr wird zuerst die beiden
  33. // Produkte x1*x1 in prod_MSDptr/2*k_hi/prodhi_LSDptr
  34. // und x0*x0 in prodhi_LSDptr/2*k_lo/prod_LSDptr,
  35. // dann das Produkt (b^k*x1+x0)*(b^k*x1+x0) enthalten.
  36. // Platz fürs Produkt (x1+x0)*(x1+x0) belegen:
  37. {var uintD* prodmid_MSDptr;
  38. var uintC prodmid_len = 2*sum_len;
  39. var uintD* prodmid_LSDptr;
  40. num_stack_small_alloc(prodmid_len,prodmid_MSDptr=,prodmid_LSDptr=);
  41. // Produkt (x1+x0)*(x1+x0) berechnen:
  42. cl_UDS_mul_square(sum_LSDptr,sum_len,prodmid_LSDptr);
  43. // Das Produkt beansprucht 2*k_lo + (0 oder 1) <= 2*sum_len = prodmid_len Digits.
  44. // Produkt x0*x0 berechnen:
  45. cl_UDS_mul_square(sourceptr,k_lo,prod_LSDptr);
  46. // Produkt x1*x1 berechnen:
  47. cl_UDS_mul_square(sourceptr lspop k_lo,k_hi,prodhi_LSDptr);
  48. // Und x1*x1 abziehen:
  49. {var uintD carry =
  50. subfrom_loop_lsp(prodhi_LSDptr,prodmid_LSDptr,prodhi_len);
  51. // Carry um maximal prodmid_len-prodhi_len Digits weitertragen:
  52. if (!(carry==0))
  53. { dec_loop_lsp(prodmid_LSDptr lspop prodhi_len,prodmid_len-prodhi_len); }
  54. }
  55. // Und x0*x0 abziehen:
  56. {var uintD carry =
  57. subfrom_loop_lsp(prod_LSDptr,prodmid_LSDptr,2*k_lo);
  58. // Falls Carry: Produkt beansprucht 2*k_lo+1 Digits.
  59. // Carry um maximal 1 Digit weitertragen:
  60. if (!(carry==0)) { lspref(prodmid_LSDptr,2*k_lo) -= 1; }
  61. }
  62. // prodmid_LSDptr[-prodmid_len..-1] enthält nun 2*x0*x1.
  63. // Dies ist < 2 * b^k_lo * b^k_hi = 2 * b^len,
  64. // paßt also in len+1 Digits.
  65. // prodmid_len, wenn möglich, um maximal 2 verkleinern:
  66. // (benutzt prodmid_len >= 2*k_lo >= len >= 2)
  67. if (mspref(prodmid_MSDptr,0)==0)
  68. { prodmid_len--;
  69. if (mspref(prodmid_MSDptr,1)==0) { prodmid_len--; }
  70. }
  71. // Nun ist k_lo+prodmid_len <= 2*len .
  72. // (Denn es war prodmid_len = 2*sum_len <= 2*(k_lo+1)
  73. // <= len+3, und nach 2-maliger Verkleinerung jedenfalls
  74. // prodmid_len <= len+1. Wegen k_lo < len also
  75. // k_lo + prodmid_len <= (len-1)+(len+1) = 2*len.)
  76. // prodmid*b^k = 2*x0*x1*b^k zu prod = x1*x1*b^(2*k) + x0*x0 addieren:
  77. {var uintD carry =
  78. addto_loop_lsp(prodmid_LSDptr,prod_LSDptr lspop k_lo,prodmid_len);
  79. if (!(carry==0))
  80. { inc_loop_lsp(prod_LSDptr lspop (k_lo+prodmid_len),prod_len-(k_lo+prodmid_len)); }
  81. } }}}