![]() |
Delphi-Version: 11 Alexandria
from C# to delphi
Hello can someone help me porting this C# code into Delphi :
Code:
internal static ulong PQFactorize(ulong pq)
{ if (pq < 2) return 1; var random = new Random(); ulong g = 0; for (int i = 0, iter = 0; i < 3 || iter < 1000; i++) { ulong q = (ulong)random.Next(17, 32) % (pq - 1); ulong x = ((ulong)random.Next() + (ulong)random.Next() << 31) % (pq - 1) + 1; ulong y = x; int lim = 1 << (Math.Min(5, i) + 18); for (int j = 1; j < lim; j++) { iter++; ulong res = q, a = x; while (x != 0) { if ((x & 1) != 0) res = (res + a) % pq; a = (a + a) % pq; x >>= 1; } x = res; ulong z = x < y ? pq + x - y : x - y; g = gcd(z, pq); if (g != 1) break; if ((j & (j - 1)) == 0) y = x; } if (g > 1 && g < pq) break; } if (g != 0) { ulong other = pq / g; if (other < g) g = other; } return g; static ulong gcd(ulong left, ulong right) { while (right != 0) { ulong num = left % right; left = right; right = num; } return left; } } |
AW: from C# to delphi
Completely untested (written in Notepad++):
Delphi-Quellcode:
The keyword "static" implies that this is a static class function, so you have to declare it as such.
function PQFactorize(pq: LongWord): LongWord;
function gcd(left, right: LongWord): LongWord; var num: LongWord; begin while right <> 0 do begin num := left mod right; left := right; right := num; end; Result := left; end; var g, q, x, y, z, res, a, other: LongWord; i, j, iter, lim: integer; begin if pq < 2 then Exit(1); g := 0; i := 0; iter := 0; repeat q := LongWord(Random(16) + 17) mod (pq - 1); x := (LongWord(Random(MAXINT)) + LongWord(Random(MAXINT)) shl 31) mod (pq - 1) + 1; y := x; lim := 1 shl (math.Min(5, i) + 18); for j := 1 to lim - 1 do begin inc(iter); res := q; a := x; while x <> 0 do begin if (x and 1) <> 0 then res := (res + a) mod pq; a := (a + a) mod pq; x := x shr 1; end; x := res; if x < y then z := pq + x - y else z := x - y; g := gcd(z, pq); if g <> 1 then break; if (j and (j - 1)) = 0 then y := x; end; if (g > 1) and (g < pq) then break; inc(i); until (i >= 3) or (iter >= 1000); if (g <> 0) then begin other := pq div g; if (other < g) then g := other; end; Result := g; end; |
AW: from C# to delphi
Zitat:
|
AW: from C# to delphi
So many thanks , and what about this please :
Code:
Again thank you
public static bool IsProbablePrime(this BigInteger n)
{ var n_minus_one = n - BigInteger.One; if (n_minus_one.Sign <= 0) return false; int s; var d = n_minus_one; for (s = 0; d.IsEven; s++) d >>= 1; var bitLen = n.GetBitLength(); var randomBytes = new byte[bitLen / 8 + 1]; var lastByteMask = (byte)((1 << (int)(bitLen % 8)) - 1); BigInteger a; if (MillerRabinIterations < 15) return false; for (int i = 0; i < MillerRabinIterations; i++) { do { Encryption.RNG.GetBytes(randomBytes); randomBytes[^1] &= lastByteMask; a = new BigInteger(randomBytes); } while (a < 3 || a >= n_minus_one); a--; var x = BigInteger.ModPow(a, d, n); if (x.IsOne || x == n_minus_one) continue; int r; for (r = s - 1; r > 0; r--) { x = BigInteger.ModPow(x, 2, n); if (x.IsOne) return false; if (x == n_minus_one) break; } if (r == 0) return false; } return true; } |
AW: from C# to delphi
This would be difficult to translate, because AFAIK there is no builtin BigInteger-Type in Delphi. Also I don' t know if there are any RNG-implementations without external libraries.
|
AW: from C# to delphi
Zitat:
![]() |
AW: from C# to delphi
Zitat:
If yes, why isn't the built in random suitable? Because of the big number range? |
AW: from C# to delphi
Zitat:
You have many options to do this. 1) Create a webservice and call it from Delphi. 2) Create a COM-Server and use it from Delphi. With Delphi's import tlb tool you don't need to write any definitions and with dotnet RegAsm.exe you can generate a TLB file. If you don't want to register the COM Server you can also use SideBySide Configuration. |
AW: from C# to delphi
@generic: I suspect he wants a native implementation to achieve a speed increase.
@sdean: I guess you want to convert a bunch of functions. Think about automating the process! At least a bit. Like dictionary style. The other way could have been to use a decompiler of the .net assembly to decompile it into Delphi.net code or Oxygene Code. You could apply some finishing touches to this delphi like code and be done. I'm just not sure if you have a version of Delphi that supports Delphi.net...I think Delphi.net went the way of Kylix? I also don't know if Oxygen is still alive??? How big is the C# library/assembly/Project you are currently converting? If you just want to use a C# assembly in a Delphi project you might just want to use ![]() |
AW: from C# to delphi
A quick and dirty approach...my default type of approach...
Lines with comments, you have to check yourself. I assume that u use the Rudy's TBiginteger project. But I'm not familiar with it.yet I know that he implemented all the operators. Never tested or compiled the code. I kept all the tranlations in the same line. so the code blocks aren't nicely indented and line breaks are missing, but they are in the same line like the original code that they represent. I think you need Delphi 11 for Inline declarations to work.
Delphi-Quellcode:
Function TMyClassHelper.IsProbablePrime(n:TBigInteger):boolean;
Begin var n_minus_one:TBiginteger := n -1; if (n_minus_one.Sign <= 0) then Begin Result := false; Exit; end; // look for TBiginteger sign yourself var s:Integer := 0; var d:TBiginteger := n_minus_one; While d.IsEven Do Begin d.shr(1); inc(s); end;// look for TBiginteger IsEven yourself var bitLen:Integer := n.GetBitLength;// use code completion to find n.GetBitLength var randomBytes : TByteDynArray; SetLength(randomBytes, (bitlen div 8) +1); var lastByteMask:Byte := Byte( ((1 shl Integer(bitLen mod 8)) -1) ); var a:TBigInteger; if (MillerRabinIterations < 15) then Begin Result := 15; Exit; End; for var i:Integer := 0 TO MillerRabinIterations-1 do Begin repeat Encryption.RNG.GetBytes(randomBytes); //?? randomBytes[^1] &= lastByteMask; //?? a := TBigInteger.Create(randomBytes); Until ( (a < 3) or (a >= n_minus_one) ); a := a-1; var x:TBigInteger; x.ModPow(a, d, n);// Ithink TBigInteger.Pow(d,n) exists but for mod you might have to use the "mod" operator if (x = 1) or (x = n_minus_one) then continue; var r:Integer; for r := s -1 downto 1 do // looks weird "s-1 downto 1" Begin var x:TBigInteger; x.ModPow(x, 2, n);// Ithink TBigInteger.Pow(d,n) exists but for mod you might have to use the "mod" operator if x = 1 then Begin result := false; exit; end; if x = n_minus_one then break; end; if r = 0 then Begin Result := false; exit; end; end; result := true; end; |
Alle Zeitangaben in WEZ +1. Es ist jetzt 10:38 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz