semiprime_equationization_C_generator.pl 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. #!/usr/bin/perl
  2. # Author: Daniel "Trizen" Șuteu
  3. # License: GPLv3
  4. # Date: 10 July 2015
  5. # Website: https://github.com/trizen
  6. # Generate a C program to compute the prime factors of a semiprime number.
  7. use 5.016;
  8. use strict;
  9. use integer;
  10. use warnings;
  11. sub semiprime_equationization {
  12. my ($semiprime, $xlen, $ylen) = @_;
  13. $xlen -= 1;
  14. $ylen -= 1;
  15. my @map;
  16. my @result;
  17. my $mem = '0';
  18. my %x_loops;
  19. foreach my $i (0 .. $xlen) {
  20. my $start = $i == $xlen ? 1 : 0;
  21. $x_loops{"x$i"} = "for (unsigned int x$i = $start; x$i < 10; ++x$i) {";
  22. }
  23. my %y_loops;
  24. foreach my $i (0 .. $ylen) {
  25. my $start = $i == $ylen ? 1 : 0;
  26. $y_loops{"y$i"} = "for (unsigned int y$i = $start; y$i < 10; ++y$i) {";
  27. }
  28. my %vars;
  29. foreach my $j (0 .. $ylen) {
  30. foreach my $i (0 .. $xlen) {
  31. my $expr = '(' . join(' + ', "(x$i * y$j)", grep { $_ ne '0' } $mem) . ')';
  32. $vars{"xy$i$j"} = $expr;
  33. my $n = "xy$i$j";
  34. if ($i == $xlen) {
  35. push @{$map[$j]}, "($n % 10)", "($n / 10)";
  36. $mem = '0';
  37. }
  38. else {
  39. push @{$map[$j]}, "($n % 10)";
  40. $mem = "($n / 10)";
  41. }
  42. }
  43. my $n = $ylen - $j;
  44. if ($n > 0) {
  45. push @{$map[$j]}, ((0) x $n);
  46. }
  47. my $m = $ylen - $n;
  48. if ($m > 0) {
  49. unshift @{$map[$j]}, ((0) x $m);
  50. }
  51. }
  52. my @number = reverse split //, $semiprime;
  53. my @mrange = (0 .. $#map);
  54. my $end = $xlen + $ylen + 1;
  55. my %seen;
  56. my $loop_init = sub {
  57. my ($str) = @_;
  58. while ($str =~ /\b(y\d+)/g) {
  59. if (not $seen{$1}++) {
  60. my $init = $y_loops{$1};
  61. push @result, $init;
  62. }
  63. }
  64. while ($str =~ /\b(x\d+)/g) {
  65. if (not $seen{$1}++) {
  66. my $init = $x_loops{$1};
  67. push @result, $init;
  68. }
  69. }
  70. };
  71. my $initializer = sub {
  72. my ($str) = @_;
  73. $loop_init->($str);
  74. while ($str =~ /\b(xy\d+)/g) {
  75. if (not $seen{$1}++) {
  76. my $init = "const unsigned int $1 = $vars{$1};";
  77. __SUB__->($init);
  78. push @result, $init;
  79. }
  80. }
  81. };
  82. foreach my $i (0 .. $#number) {
  83. my $expr = '(' . join(' + ', grep { $_ ne '0' } (map { $map[$_][$i] } @mrange), $mem) . ')';
  84. $initializer->($expr);
  85. push @result, "const unsigned int n$i = $expr;";
  86. my $n = "n$i";
  87. if ($i == $#number) {
  88. push @result,
  89. qq/if ($number[$i] == $n) { printf("Cracked: /
  90. . ("%d" x ($xlen + 1))
  91. . (" * ")
  92. . ("%d" x ($ylen + 1))
  93. . qq/\\n", /
  94. . join(", ", (map { "x$_" } reverse(0 .. $xlen)), (map { "y$_" } reverse(0 .. $ylen)))
  95. . qq/); return 0; }/;
  96. }
  97. elsif ($i == 0) {
  98. push @result, "if ($number[$i] != $n) { continue; }";
  99. $mem = '0';
  100. }
  101. else {
  102. push @result, "if ($number[$i] != ($n % 10)) { continue; }";
  103. $mem = "($n / 10)";
  104. }
  105. }
  106. unshift @result, "#include <stdio.h>", "int main() {";
  107. push @result, "}" x (1 + $xlen + 1 + $ylen + 1);
  108. return @result;
  109. }
  110. # 71 * 43
  111. #say for semiprime_equationization('3053', 2, 2);
  112. # 251 * 197
  113. #say for semiprime_equationization('49447', 3, 3);
  114. # 7907 * 4999
  115. say for semiprime_equationization('39527093', 4, 4);
  116. # 472882049 * 472882049
  117. #say for semiprime_equationization('223617432266438401', 9, 9);
  118. # 37975227936943673922808872755445627854565536638199 * 40094690950920881030683735292761468389214899724061
  119. #say for semiprime_equationization('1522605027922533360535618378132637429718068114961380688657908494580122963258952897654000350692006139', 50, 50);