flat assembler
Message board for the users of flat assembler.
Index
> Windows > How to generate random numbers? Goto page 1, 2, 3 Next |
Author |
|
revolution 24 May 2022, 10:02
If you want a good cryptographic quality PRNG then the C runtime is not going to do it for you.
Instead you can try this: Code: PROV_RSA_FULL = 1 CRYPT_VERIFYCONTEXT = 0f0000000h proc rand uses ebx,store,length local context:DWORD invoke CryptAcquireContext,addr context,NULL,NULL,PROV_RSA_FULL,CRYPT_VERIFYCONTEXT test eax,eax jz .done invoke CryptGenRandom,[context],[length],[store] test eax,eax setz bl invoke CryptReleaseContext,[context],0 test eax,eax setz al or al,bl movzx eax,al dec eax .done: ret endp cinvoke func is the same as ccall [func]. Either works for calling MSVCRT. |
|||
24 May 2022, 10:02 |
|
FlierMate 24 May 2022, 11:03
I have issue with local context:DWORD (I am not sure how to handle "proc" macro) because FASMW complained : "Error: Invalid value" by pointing to "pushd addr ..var?TQ" . To workaround that, I declare "context" in data section as "dw".
Now everything compile and run, but it exits silently after call rand and did not execute the next instruction in my code to display a message box. This is how I reference to the APIs: Code: import advapi,\ CryptAcquireContext, 'CryptAcquireContextA',\ CryptGenRandom, 'CryptGenRandom',\ CryptReleaseContext, 'CryptReleaseContext' Microsoft Docs states: Quote: Important This API is deprecated. New and existing software should start using Cryptography Next Generation APIs. Microsoft may remove this API in future releases. And I am using Windows 11, is this the reason it quits silently on my PC? I include the full source below: Code: format PE GUI 4.0 entry start include 'win32a.inc' section '.text' code readable executable proc rand uses ebx,store,length invoke CryptAcquireContext,context,0,0,1,0xF0000000 test eax,eax jz .done invoke CryptGenRandom,dword [context],[length],[store] test eax,eax setz bl invoke CryptReleaseContext,dword [context],0 test eax,eax setz al or al,bl movzx eax,al dec eax .done: ret endp start: ;ccall [srand] ;ccall [rand] call rand cinvoke wsprintf,msg,fmt,eax invoke MessageBox,0,msg,title,0x40 invoke ExitProcess,0 section '.data' data readable writeable msg rb 32 fmt db '%d',0 title db 'rand',0 context dw ? section '.idata' import readable writeable library kernel,'KERNEL32.DLL',\ user,'USER32.DLL',\ advapi,'ADVAPI32.DLL',\ msvcrt,'msvcrt.dll' import kernel,\ ExitProcess,'ExitProcess' import user,\ wsprintf,'wsprintfA',\ MessageBox,'MessageBoxA' import advapi,\ CryptAcquireContext, 'CryptAcquireContextA',\ CryptGenRandom, 'CryptGenRandom',\ CryptReleaseContext, 'CryptReleaseContext' ; import msvcrt,\ ; rand,'rand',\ ; srand,'srand' |
|||
24 May 2022, 11:03 |
|
revolution 24 May 2022, 11:10
If you use win32ax.inc then the other stuff you had trouble with should be okay.
And it is a stdcall function, not a normal call. So call it like this: Code: stdcall rand, buffer, length |
|||
24 May 2022, 11:10 |
|
revolution 24 May 2022, 11:13
context is not the random numbers, it is a private pointer for the API. The generated numbers are stored in the buffer you declare in the call.
|
|||
24 May 2022, 11:13 |
|
FlierMate 24 May 2022, 11:25
revolution wrote: If you use win32ax.inc then the other stuff you had trouble with should be okay. Thank you for bearing with me, I am still poor in many things related to FASM. Now it works: Code: stdcall rand, buffer, 32 Code: buffer rb 32 db 0 EDIT: The "eax" returned by "rand" is -1. But it is still the same number I get each time I run the program, i.e. 4202536. Is this how it should work? |
|||
24 May 2022, 11:25 |
|
revolution 24 May 2022, 11:31
eax is the indicator of success or failure. It can only be zero or negative one. Zero mean failure, negative one means success.
The buffer is filled with the result. In your case 32 bytes of data. Print the contents of the buffer, not the address. 4202536 looks like the address. |
|||
24 May 2022, 11:31 |
|
FlierMate 24 May 2022, 11:40
Excellent! You're right.
It really is random data, i.e. different set of data each time I run the program. Now I need to turn the random data to random number. Thanks @revolution for your zeal in helping me and others.
|
|||||||||||||||||||
24 May 2022, 11:40 |
|
revolution 24 May 2022, 12:22
FlierMate wrote: It really is random data, i.e. different set of data each time I run the program. |
|||
24 May 2022, 12:22 |
|
FlierMate 24 May 2022, 13:29
I don't know about the term PRNG, I think bitRAKE posted some random data generator recently.
I use your method to show quote of the day.
Last edited by FlierMate on 28 May 2022, 10:36; edited 2 times in total |
|||||||||||||||||||||
24 May 2022, 13:29 |
|
AsmGuru62 24 May 2022, 13:33
I use this generator (WELL512) -- it passes all the tests.
Just ignore some of the functions in the file -- it is for a game I was coding.
|
|||||||||||||||||||||
24 May 2022, 13:33 |
|
macomics 24 May 2022, 15:29
Code: // calculate a 't' value that will linearly interpolate from 0 to 1 and back every 20 seconds DWORD currentTime = GetTickCount(); if ( m_startTime == 0 ) { m_startTime = currentTime; } float t = 2 * (( currentTime - m_startTime) % 20000) / 20000.0f; if (t > 1.0f) { t = 2 - t; } |
|||
24 May 2022, 15:29 |
|
pabloreda 25 May 2022, 00:35
I use this code (in forth/r3)
Code: ##seed $a3b195354a39b70d ::rand | -- rand seed $da942042e4dd58b5 * 1 + dup 'seed ! ; in ASM is like Code: seed dq $a3b195354a39b70d rand: ; call rand ; rax a rand number mov rax,[seed] imul rax,$da942042e4dd58b5 add rax,1 mov [seed],rax ret |
|||
25 May 2022, 00:35 |
|
revolution 25 May 2022, 01:16
Those LCG class generators were one of the first types to be used. Simple and easy.
WP has an article on some other types of simple generators. https://en.wikipedia.org/wiki/Xorshift Check out this page for some stats on them https://prng.di.unimi.it/ None of those are cryptographic though, so the potential applications are limited by that. And the seeding is the hardest bit. So all generators, no matter how fantastic, are terrible if the seeding is bad. |
|||
25 May 2022, 01:16 |
|
Overclick 28 May 2022, 14:08
One more example, with no call to anything.
Code: rdtsc Code: rdtsc mov bl,10 mul bl mov bl,0xFF div bl |
|||
28 May 2022, 14:08 |
|
Furs 28 May 2022, 14:55
Code: rdrand |
|||
28 May 2022, 14:55 |
|
Overclick 28 May 2022, 15:08
Quote:
Nice one begins from IvyBridge and Ryzen |
|||
28 May 2022, 15:08 |
|
Overclick 28 May 2022, 16:05
fixed example to show 0-10:
Code: rdtsc mov bl,11 mul bl xor dx,dx mov bx,0x0100 div bx |
|||
28 May 2022, 16:05 |
|
revolution 28 May 2022, 16:33
Nobody trusts rdrand anyway. It is a blackbox function with no way to verify it has no backdoors.
rdtsc can be good for simple apps, but I would prefer to use it as part of the generator seeding, rather than directly as the generator itself. |
|||
28 May 2022, 16:33 |
|
Overclick 28 May 2022, 17:01
Quote: Nobody trusts rdrand anyway. It is a blackbox function with no way to verify it has no backdoors. Don't you trust CPU instructions? How can you trust to rest of them? |
|||
28 May 2022, 17:01 |
|
Goto page 1, 2, 3 Next < Last Thread | Next Thread > |
Forum Rules:
|
Copyright © 1999-2025, Tomasz Grysztar. Also on GitHub, YouTube.
Website powered by rwasa.