forked from chinshou/bcrypt-for-delphi
-
Notifications
You must be signed in to change notification settings - Fork 24
/
Copy pathBcrypt.pas
3306 lines (2696 loc) · 124 KB
/
Bcrypt.pas
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
unit Bcrypt;
(*
Sample Usage
============
//Hash a password using default cost (e.g. 11 ==> 2^11 ==> 2,048 rounds)
hash := TBCrypt.HashPassword('correct horse battery staple');
//Hash a password using custom cost factor
hash := TBCrypt.HashPassword('correct horse battery staple', 14); //14 ==> 2^14 ==> 16,384 rounds
//Check a password
var
passwordRehashNeeded: Boolean;
TBCrypt.CheckPassword(szPassword, existingHash, {out}passwordRehashNeeded);
Enhanced version
----------------
Enhanced mode pre-hashes the password with SHA2-256. This gives two benfits:
- overcomes the 72-byte limit on passwords; allowing them to be arbitrarily long
- avoids potential denial of service with really long passwords
It is essentially HashPassword(base64(HMAC_sha256(Password, salt)))
Sample usage:
// Hash password with SHA256 prehash:
hash := TBCrypt.EnhancedHashPassword('correct horse battery staple'); // $bcrypt-sha256$
// Check enhanced password the same way as regular hashes
isPasswordValid := TBCrypt.CheckPassword(szPassword, existingHash, {out}passwordRehashNeeded);
Remarks
=======
Bcrypt is an algorithm designed for hashing passwords, and only passwords.
i.e. It's not a generic, high-speed, generic hashing algorithm.
It's not a password-based key derivation function
It's computationally and memory expensive
It's limited to passwords of 71 bytes.
http://static.usenix.org/events/usenix99/provos/provos.pdf
It uses the Blowfish encryption algorithm, but with an "expensive key setup" modification,
contained in the function EksBlowfishSetup.
Version History
===============
Version 1.18 20241122
- Following the C++ example, hardening the code to be compatible with RangeChecking, Overflowchecking, and type-compatible pointers
- make TBlowfishData.Sbox be array[4] of TSBox rather than array[4][255] of UInt32 (to make types compatible)
- added typed pointer cast to convert to PLongWord when passing array to BlowfishEncryptECB (which expects PLongWord)
- changed functions to accept open arrays - now that i finally understand what they were invented for (thanks ChatGPT)
- BREAKING CHANGE: fixed the security vulnerability in enhanced (sha256 allows password shucking; replaced with hmac_sha256)
Version 1.17 20201125
- Simplified the performance timestamp to just two functions, and removed the use of GetPerformanceFrequency.
Version 1.16 20201123
- Changed minimum key length from 1 to 0. The known test vectors allow an empty password.
And if a person wants to have an empty password: that's their right.
Version 1.15 20201120
- Idera redefined the "fundamental" LongWord type to no longer be 32-bits. It is now 32-bits on 16-bit platforms, 32-bits on 32-bit platforms, and 64-bits on 64-bit platforms.
And Cardinal, the "generic" type, as been changed to be 16-bits on 16-bit platforms, 32-bits on 32-bit platforms, and 32-bits on 64-bit platforms.
This means they violated the contract of correctly written code - retroactively breaking all existing code.
- Refactored a lot of the OS-specific code into separate functions.
This way you can create your own IFDEFs to plug in your own OS-specific equivalents:
- GetPerformanceTimestamp ==> QueryPerformanceCounter (Windows)
- PerformanceTimestampToMs ==> Timestamp / QueryPerformanceFrequency (Windows)
- GenRandomBytes ==> Crypto API (Windows)
- HashBytes ==> Crypto API (Windows)
Version 1.14 20190823
- Added function to manually prehash a password using SHA-2_256
- Added EnchancedHashPassword to prehash the password and output $bcrypt-sha256$ identifier
- Improvement: unrolled main loop for performance improvement
Version 1.13 20180729
- UIntPtr isn't declared in Delphi 2010 (21.0). Maybe it first appeared in Delphi XE (22.0)?
Version 1.12 20180419
- Made compatible with Delphi 5
- Published all self tests, but put slow ones behind the -SlowUnitTests command line parameter
Version 1.11 20180120
- Bugfix: The raw version of CheckPassword forgot to time the hash operation, and set PasswordRehashNeeded out parameter approriately
Version 1.10 20161212
- Bugfix: Don't zero out password byte array when it is empty - it's a range check error and unneeded
Version 1.09 20161122
- Added: In accordance with the recommendations of NIST SP 800-63B, we now apply KC normalization to the password.
Choice was between NFKC and NFKD. SASLprep (rfc4013), like StringPrep (rfc3454) both specified NFKC.
Version 1.08 20161029
- We now burn the intermediate UTF8 form of the password.
- BCrypt key has a maximum size of 72 bytes. We burn off any excess bytes after converting the string to utf8
- Changed the handling of passwords longer than the 72 characters to match other implemntations.
No longer do we forcibly add a terminating null, but instead simply chop the utf8 array at 72 bytes.
Version 1.07 20160423
- Changed: Fixed up compiler defines so that we work on Delphi 5, Delphi 7, and Delphi XE5 (the only versions of Delphi i use)
- Added the fast SelfTests into their own DUnit test; the ones that are too slow are still kept in the public section
- Added granular abilty to disable BCrypt unit tests
Version 1.06 20151026
- Added: CheckPassword functions now take "PasswordRehashNeeded" out parameter.
The value will contain True if the password needs to be rehashed.
For for example: if the BCrypt cost needs to be increased as the hash was calculated too quickly,
or if the BCrypt standard has been updated (as OpenBSD updated their canonical output to 2b)
- Updated: We now use OS CryptGenRandom for salt generation.
If it fails, we fall back to a type 4 ("random") UUID.
Salt doesn't have to be cryptographically strong, just different.
- Fix: We now recognize version "2" and well as anything version "2"+[letter].
OpenBSD canonical version now outputs "2c".
We will continue to output "2b" for a while, at least hopefully until everyone's updated.
Previous version would fail on anything besides "2a".
In the case of a federated login, we can't have older software that only knows how to handle "2a" suddenly
start failing when it encounters "2b".
- Updated: Removed dependancy on Blowfish.pas. Copied relavent ECB function, state variable, and digits of PI
Version 1.05 20150321
- Performance improvement: Was so worried about using faster verison of Move in BlowfishEncryptECB and BlowfishDecryptECB,
that i didn't stop to notice that i shouldn't even be using Move; but instead a 32-bit assignment
- Fix: Fixed bug in EksBlowfishSetup. If you used a cost factor 31 (2,147,483,648), then the Integer loop
control variable would overflow and the expensive key setup wouldn't run at all (zero iterations)
- The original whitepaper mentions the maximum key length of 56 bytes.
This was a misunderstanding based on the Blowfish maximum recommended key size of 448 bits.
The algorithm can, and does, support up to 72 bytes (e.g. 71 ASCII characters + null terminator).
Note: Variant 2b of bcrypt also *caps* the password length at 72 (to avoid integer wraparound on passwords longer than 2 billion characters O.o)
Version 1.04 20150312
- Performance improvement: ExpandKey: Hoisted loop variable, use xor to calculate SaltHalfIndex to avoid speculative execution jump, unrolled loop to two 32-bit XORs (16% faster)
- Performance improvement (D5,D7): Now use pure pascal version of FastCode Move() (50% faster)
Version 1.03 20150319
- Fix: Defined away Modernizer (so people who are not me can use it)
- Added: If no cost factor is specified when hashing a password,
the cost factor is now a sliding factor, based on Moore's Law and when BCrypt was designed
Version 1.02 20141215
- Added support for XE2 string/UnicodeString/AnsiString
- Update: Updated code to work in 64-bit environment
Version 1.01 20130612
- New: Added HashPassword overload that lets you specify your desired cost
Version 1.0 20120504
- Initial release by Ian Boyd, Public Domain
Background
==========
bcrypt was designed for OpenBSD, where hashes in the password file have a certain format.
The convention used in BSD when generating password hash strings is to format it as:
$version$salt$hash
MD5 hash uses version "1":
"$"+"1"+"$"+salt+hash
bcrypt uses version "2a", but also encodes the cost
"$"+"2a"+"$"+rounds+"$"+salt+hash
e.g.
$2a$10$Ro0CUfOqk6cXEKf3dyaM7OhSCvnwM9s4wIX9JeLapehKK5YdLxKcm
$==$==$======================-------------------------------
The benfit of this scheme is:
- the number of rounds
- the salt used
This means that stored hashes are backwards and forwards compatible with changing the number of rounds
BCrypt variants
===============
$2$
The original specification used the prefix $2$.
This was in contrast to the other algorithm prefixes:
$1$ - MD5
$5$ - SHA-256
$6$ - SHA-512
$2a$
The original specification did not define how to handle non-ASCII character, or how to handle a null terminator.
The specification was revised to specify that when hashing strings:
- the string must be UTF-8 encoded
- the null terminator must be included
$2x$, $2y$ June 2011
A bug was discovered in crypt_blowfish, a PHP implementation of BCrypt.
It was mis-handling characters with the 8th bit set.
They suggested that system administrators update their existing password database, replacing $2a$ with $2x$,
to indicate that those hashes are bad (and need to use the old broken algorithm).
They also suggested the idea of having crypt_blowfish emit $2y$ for hashes generated by the fixed algorithm.
Nobody else, including canonical OpenBSD, adopted the idea of 2x/2y. This version marker was was limited to crypt_blowfish.
http://seclists.org/oss-sec/2011/q2/632
$2b$ February 2014
A bug was discovered in the OpenBSD implemenation of bcrypt.
They were storing the length of their strings in an unsigned char.
If a password was longer than 255 characters, it would overflow and wrap at 255.
BCrypt was created for OpenBSD. When they have a bug in *their* library, they decided its ok to bump the version.
This means that everyone else needs to follow suit if you want to remain current to "their" specification.
http://undeadly.org/cgi?action=article&sid=20140224132743
http://marc.info/?l=openbsd-misc&m=139320023202696
BCrypt strength
===============
scrypt is weaker than bcrypt for memory requirements less than 4 MB. Stick with at least 16 MB.
https://blog.ircmaxell.com/2014/03/why-i-dont-recommend-scrypt.html#Putting-It-In-Perspective
To put it in perspective, scrypt requires approximately 1000 times the memory
of bcrypt to achieve a comparable level of defense against GPU based attacks
(again, for password storage). On one hand, that's still fine, as bcrypt
uses 4KB, which means the equivalent effective scrypt protection occurs at
4MB. And considering the recommended settings are in the 16MB range, that
should be clear that scrypt is definitively stronger than bcrypt.
This proves that scrypt is demonstrably weaker than bcrypt for password
storage when using memory settings under 4mb. This is why the recommendations
are 16mb or higher. If you're using 16+mb of memory in scrypt (p=1, r=8 and
N=2^14, or p=1, r=1 and N=17), you are fine.
Argon2 is weaker than bcrypt for run times less than 1 second. (i.e. for authentication)
https://twitter.com/jmgosney/status/1111865772656246786
Actually, bcrypt is stronger than Argon2 for authentication (target runtime < 500ms.)
Argon2 does not match or surpass bcrypt's strength until >= ~1000ms runtimes (KDF.)
So "Use Argon2" is not a good one-size-fits-all answer.
*)
interface
(*
The problem is how to make "TBytes", "UIntPtr", and "UnicodeString" work in Delphi 5, Delphi 7, Delphi 2010, and XE2+
| Item | Delphi 5 | Delphi 7 | Delphi 2009 | Delphi 2010 | Delphi XE2 |
|-----------------|-----------------|-----------------------|------------------------|------------------------|------------------------|
| Product version | 5 | 7 | 12 | 14 | 16 |
| Version | VER130 | VER150 | VER200 | VER210 | VER230 |
| CompilerVersion | n/a | 15.0 | 20.0 | 21.0 | 23.0 |
| TBytes | = array of Byte | = Types.TByteDynArray | = Types.TByteDynArray? | = Types.TByteDynArray? | = SysUtils.TBytes |
| UnicodeString | = WideString | = WideString | = System.UnicodeString | = System.UnicodeString | = System.UnicodeString | Added in Delphi 2009
| UIntPtr | = Cardinal | = Cardinal | = Cardinal | = Cardinal | = System.UIntPtr |
And it wasn't until Delphi 6 (CompilerVersion >= 14.0) that conditional expressions (CONDITIONALEXPRESSIONS) were added.
*)
{$IFDEF CONDITIONALEXPRESSIONS}
{$IF CompilerVersion >= 22}
{$DEFINE COMPILER_15_UP} //Delphi XE
{$IFEND}
{$IF CompilerVersion >= 15}
{$DEFINE COMPILER_7_UP} //Delphi 7
{$IFEND}
{$ELSE}
{$DEFINE MSWINDOWS} //The MSWINDOWS define didn't work until Delphi 7
{$ENDIF}
uses
SysUtils,
Windows,
{$IFDEF COMPILER_7_UP}Types,{$ENDIF} //Types.pas didn't appear until ~Delphi 7.
ComObj, Math;
type
{$IFNDEF UNICODE}
UnicodeString = WideString; //System.UnicodeString wasn't added until Delphi 2009
{$ENDIF}
{$IFDEF COMPILER_15_UP}
//TBytes === System.TArray<System.Byte>
{$ELSE}
{$IFDEF COMPILER_7_UP} //Delphi 7
TBytes = Types.TByteDynArray; //TByteDynArray wasn't added until around Delphi 7. Sometime later it moved to SysUtils.
{$ELSE}
TBytes = array of Byte; //for old-fashioned Delphi 5, we have to do it ourselves
{$ENDIF}
{$ENDIF}
{$IFNDEF COMPILER_15_UP}
//Someone said that Delphi 2010 (Delphi 14) didn't have UIntPtr.
//So maybe it was Delphi XE (Delphi 15)
UIntPtr = Cardinal; //an unsigned, pointer sized, integer
UInt32 = Cardinal; //Idera changed the unchangeable "fundamental" LongWord type from UInt32 to UInt64. So we use the "generic" type Cardinal - which hopefully will remain UInt32 going forward
{$ENDIF}
TSBox = array[0..255] of UInt32;
PSBox = ^TSBox;
TBlowfishData= record
InitBlock: array[0..7] of Byte; { initial IV }
LastBlock: array[0..7] of Byte; { current IV }
SBox: array[0..3] of TSBox; //4 SBoxes
PBox: array[0..17] of UInt32; //18 subkeys
end;
const
CP_UTF8 = 65001;
type
TBCrypt = class(TObject)
private
class function TryParseHashString(const hashString: string;
out version: string; out Cost: Integer; out Salt: TBytes; out IsEnhanced: Boolean): Boolean;
protected
class function EksBlowfishSetup(const Cost: Integer; const Salt, Key: array of Byte): TBlowfishData;
class procedure ExpandKey(var state: TBlowfishData; const salt, key: array of Byte);
class function CryptCore(const Cost: Integer; Key: array of Byte; salt: array of Byte): TBytes;
class function FormatPasswordHashForBsd(const Version: string; const cost: Integer; const salt: array of Byte; const hash: array of Byte): string;
class function FormatEnhancedPasswordHash(const Version: string; const Cost: Integer; const Salt: array of Byte; const Hash: array of Byte): string;
class function BsdBase64Encode(const Data: array of Byte; BytesToEncode: Integer): string;
class function BsdBase64Decode(const s: string): TBytes;
class function Base64Encode(const data: array of Byte): string; //standard base64 alphabet
class function PasswordStringPrep(const Source: UnicodeString): TBytes;
class function StringToBytes(const Source: UnicodeString; const CodePage: Word=CP_UTF8): TBytes;
class function GenRandomBytes(len: Integer; const data: Pointer): HRESULT; //Ask the operating system for len random bytes
// Applies sha-256 preshashing to a password. The returned string is suitable to pass to HashPassword.
class function Prehash256(const password: UnicodeString; const Salt: array of Byte): string;
class function SelfTestA: Boolean; //known test vectors
class function SelfTestB: Boolean; //BSD's base64 encoder/decoder
class function SelfTestC: Boolean; //unicode strings in UTF8
class function SelfTestD: Boolean; //different length passwords
class function SelfTestE: Boolean; //salt rng
class function SelfTestF: Boolean; //correctbatteryhorsestapler
class function SelfTestG: Boolean; //check that we support up to 72 characters
class function SelfTestH: Boolean; //check that we don't limit our passwords to 256 characters (as OpenBSD did)
class function SelfTestI: Boolean; //check that we use unicode compatible composition (NFKC) on passwords
class function SelfTestJ: Boolean; //check that composed and decomposed strings both validate to the same
class function SelfTestK: Boolean; //SASLprep rules for passwords
class function SelfTestL: Boolean; //Test prehashing a password (sha256 -> base64)
class function GetModernCost(SampleCost: Integer; SampleHashDurationMS: Real): Integer;
class function GetModernCost_Benchmark: Integer;
class function TimingSafeSameString(const Safe, User: string): Boolean;
class function PasswordRehashNeededCore(const Version: string; const Cost: Integer; SampleCost: Integer; SampleHashDurationMS: Real): Boolean;
class function HmacSha256(const Data: array of Byte; const Key: array of Byte): TBytes;
// If you want to handle the cost, salt, and encoding yourself, you can do that.
class function HashPassword( const password: UnicodeString; const salt: array of Byte; const cost: Integer): TBytes; overload;
class function CheckPassword(const password: UnicodeString; const salt, hash: array of Byte; const Cost: Integer; out PasswordRehashNeeded: Boolean): Boolean; overload;
class function GenerateSalt: TBytes;
class function EnhancedHashPassword(const Password: UnicodeString; const Salt: array of Byte; Cost: Integer): TBytes; overload;
public
// Hashes a password into the OpenBSD password-file format (non-standard base-64 encoding). Also validate that BSD style string
class function HashPassword( const Password: UnicodeString): string; overload;
class function HashPassword( const Password: UnicodeString; Cost: Integer): string; overload;
class function CheckPassword(const Password: UnicodeString; const ExpectedHashString: string; out PasswordRehashNeeded: Boolean): Boolean; overload;
// Performs sha-256 pre-hashing on the password (to allow unlimited password lengths, and avoid DoS attacks). Emits '$bcrypt-sha256$' format.
class function EnhancedHashPassword(const password: UnicodeString): string; overload;
class function EnhancedHashPassword(const password: UnicodeString; Cost: Integer): string; overload;
// Evaluate the cost (or version) of a hash string, and figure out if it needs to be rehashed
class function PasswordRehashNeeded(const HashString: string): Boolean;
// Run self-test diagnostic.
class function SelfTest: Boolean;
end;
EBCryptException = class(Exception);
implementation
{$IFDEF Strict}
{$DEFINE Sqm}
{$ENDIF}
{$IFDEF NoSqm}
{$UNDEF Sqm}
{$ENDIF}
{$IFDEF UnitTests}
{$DEFINE BCryptUnitTests}
{$ENDIF}
{$IFDEF NoBCryptUnitTests}
{$UNDEF BCryptUnitTests}
{$ENDIF}
uses
{$IFDEF Sqm}SqmApi,{$ENDIF}
{$IFDEF BCryptUnitTests}TestFramework,{$ENDIF}
ActiveX;
const
BCRYPT_COST = 11; //cost determintes the number of rounds. 11 = 2^11 rounds (2,048)
{
| Cost | Iterations | E6300 | E5-2620 | i5-2500 | i7-2700K |
| | 2006-Q3 | 2012-Q1 | 2011-Q1 | 2011-Q4 |
| | 1.86 GHz | 2 GHz | 3.3 GHz | 3.5 GHz |
|------|-------------------|--------------|-------------|------------|-------------|
| 8 | 256 iterations | 61.65 ms | 48.8 ms | 21.7 ms | 20.8 ms | <-- minimum allowed by BCrypt
| 9 | 512 iterations | 126.09 ms | 77.7 ms | 43.3 ms | 41.5 ms |
| 10 | 1,024 iterations | 249.10 ms | 128.8 ms | 85.5 ms | 83.2 ms |
| 11 | 2,048 iterations | 449.23 ms | 250.1 ms | 173.3 ms | 166.8 ms | <-- current default (BCRYPT_COST=11)
| 12 | 4,096 iterations | 1,007.05 ms | 498.8 ms | 345.6 ms | 333.4 ms |
| 13 | 8,192 iterations | 1,995.48 ms | 999.1 ms | 694.3 ms | 667.9 ms |
| 14 | 16,384 iterations | 4,006.78 ms | 1,997.6 ms | 1,390.5 ms | 1,336.5 ms |
| 15 | 32,768 iterations | 8,027.05 ms | 3,999.9 ms | 2,781.4 ms | 2,670.5 ms |
| 16 | 65,536 iterations | 15,982.14 ms | 8,008.2 ms | 5,564.9 ms | 5,342.8 ms |
At the time of deployment in 1976, crypt could hash fewer than 4 passwords per second. (250 ms per password)
In 1977, on a VAX-11/780, crypt (MD5) could be evaluated about 3.6 times per second. (277 ms per password)
If 277 ms per hash was our target, it would mean a range of 180 ms..360 ms.
At the time of publication of BCrypt (1999) the default costs were:
- Normal User: 6
- the Superuser: 8
"Of course, whatever cost people choose should be reevaluated from time to time."
We want to target between 250-500ms per hash.
}
BCRYPT_SALT_LEN = 16; //bcrypt uses 128-bit (16-byte) salt (This isn't an adjustable parameter, just a name for a constant)
BCRYPT_MaxKeyLen = 72; //72 bytes ==> 71 ansi charcters + null terminator
BsdBase64EncodeTable: array[0..63] of Char =
{ 0:} './'+
{ 2:} 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'+
{28:} 'abcdefghijklmnopqrstuvwxyz'+
{54:} '0123456789';
//the traditional base64 encode table:
//'ABCDEFGHIJKLMNOPQRSTUVWXYZ' +
//'abcdefghijklmnopqrstuvwxyz' +
//'0123456789+/';
BsdBase64DecodeTable: array[#0..#127] of Integer = (
{ 0:} -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // ________________
{ 16:} -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // ________________
{ 32:} -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 0, 1, // ______________./
{ 48:} 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, -1, -1, -1, -1, -1, -1, // 0123456789______
{ 64:} -1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, // _ABCDEFGHIJKLMNO
{ 80:} 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, -1, -1, -1, -1, -1, // PQRSTUVWXYZ_____
{ 96:} -1, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, // _abcdefghijklmno
{113:} 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, -1, -1, -1, -1, -1); // pqrstuvwxyz_____
TestVectors: array[1..20, 1..3] of string = (
('', '$2a$06$DCq7YPn5Rq63x1Lad4cll.', '$2a$06$DCq7YPn5Rq63x1Lad4cll.TV4S6ytwfsfvkgY8jIucDrjc8deX1s.'),
('', '$2a$08$HqWuK6/Ng6sg9gQzbLrgb.', '$2a$08$HqWuK6/Ng6sg9gQzbLrgb.Tl.ZHfXLhvt/SgVyWhQqgqcZ7ZuUtye'),
('', '$2a$10$k1wbIrmNyFAPwPVPSVa/ze', '$2a$10$k1wbIrmNyFAPwPVPSVa/zecw2BCEnBwVS2GbrmgzxFUOqW9dk4TCW'),
('', '$2a$12$k42ZFHFWqBp3vWli.nIn8u', '$2a$12$k42ZFHFWqBp3vWli.nIn8uYyIkbvYRvodzbfbK18SSsY.CsIQPlxO'),
('a', '$2a$06$m0CrhHm10qJ3lXRY.5zDGO', '$2a$06$m0CrhHm10qJ3lXRY.5zDGO3rS2KdeeWLuGmsfGlMfOxih58VYVfxe'),
('a', '$2a$08$cfcvVd2aQ8CMvoMpP2EBfe', '$2a$08$cfcvVd2aQ8CMvoMpP2EBfeodLEkkFJ9umNEfPD18.hUF62qqlC/V.'),
('a', '$2a$10$k87L/MF28Q673VKh8/cPi.', '$2a$10$k87L/MF28Q673VKh8/cPi.SUl7MU/rWuSiIDDFayrKk/1tBsSQu4u'),
('a', '$2a$12$8NJH3LsPrANStV6XtBakCe', '$2a$12$8NJH3LsPrANStV6XtBakCez0cKHXVxmvxIlcz785vxAIZrihHZpeS'),
('abc', '$2a$06$If6bvum7DFjUnE9p2uDeDu', '$2a$06$If6bvum7DFjUnE9p2uDeDu0YHzrHM6tf.iqN8.yx.jNN1ILEf7h0i'),
('abc', '$2a$08$Ro0CUfOqk6cXEKf3dyaM7O', '$2a$08$Ro0CUfOqk6cXEKf3dyaM7OhSCvnwM9s4wIX9JeLapehKK5YdLxKcm'),
('abc', '$2a$10$WvvTPHKwdBJ3uk0Z37EMR.', '$2a$10$WvvTPHKwdBJ3uk0Z37EMR.hLA2W6N9AEBhEgrAOljy2Ae5MtaSIUi'),
('abc', '$2a$12$EXRkfkdmXn2gzds2SSitu.', '$2a$12$EXRkfkdmXn2gzds2SSitu.MW9.gAVqa9eLS1//RYtYCmB1eLHg.9q'),
('abcdefghijklmnopqrstuvwxyz', '$2a$06$.rCVZVOThsIa97pEDOxvGu', '$2a$06$.rCVZVOThsIa97pEDOxvGuRRgzG64bvtJ0938xuqzv18d3ZpQhstC'),
('abcdefghijklmnopqrstuvwxyz', '$2a$08$aTsUwsyowQuzRrDqFflhge', '$2a$08$aTsUwsyowQuzRrDqFflhgekJ8d9/7Z3GV3UcgvzQW3J5zMyrTvlz.'),
('abcdefghijklmnopqrstuvwxyz', '$2a$10$fVH8e28OQRj9tqiDXs1e1u', '$2a$10$fVH8e28OQRj9tqiDXs1e1uxpsjN0c7II7YPKXua2NAKYvM6iQk7dq'),
('abcdefghijklmnopqrstuvwxyz', '$2a$12$D4G5f18o7aMMfwasBL7Gpu', '$2a$12$D4G5f18o7aMMfwasBL7GpuQWuP3pkrZrOAnqP.bmezbMng.QwJ/pG'),
('~!@#$%^&*() ~!@#$%^&*()PNBFRD', '$2a$06$fPIsBO8qRqkjj273rfaOI.', '$2a$06$fPIsBO8qRqkjj273rfaOI.HtSV9jLDpTbZn782DC6/t7qT67P6FfO'),
('~!@#$%^&*() ~!@#$%^&*()PNBFRD', '$2a$08$Eq2r4G/76Wv39MzSX262hu', '$2a$08$Eq2r4G/76Wv39MzSX262huzPz612MZiYHVUJe/OcOql2jo4.9UxTW'),
('~!@#$%^&*() ~!@#$%^&*()PNBFRD', '$2a$10$LgfYWkbzEvQ4JakH7rOvHe', '$2a$10$LgfYWkbzEvQ4JakH7rOvHe0y8pHKF9OaFgwUZ2q7W2FFZmZzJYlfS'),
('~!@#$%^&*() ~!@#$%^&*()PNBFRD', '$2a$12$WApznUOJfkEGSmYRfnkrPO', '$2a$12$WApznUOJfkEGSmYRfnkrPOr466oFDCaj4b6HY3EXGvfxm43seyhgC')
);
SInvalidHashString = 'Invalid base64 hash string';
SBcryptCostRangeError = 'BCrypt cost factor must be between 4..31 (%d)';
SKeyRangeError = 'Key must be between 0 and 72 bytes long (%d)';
SSaltLengthError = 'Salt must be 16 bytes';
SInvalidLength = 'Invalid length';
{
TODO: bcrypt with SHA256 pre-hashing
passlib.hash.bcrypt_sha256 - BCrypt+SHA256¶
https://passlib.readthedocs.io/en/stable/lib/passlib.hash.bcrypt_sha256.html
BCrypt was developed to replace md5_crypt for BSD systems. It uses a modified version of the Blowfish stream cipher.
It does, however, truncate passwords to 72 bytes, and some other minor quirks (see BCrypt Password Truncation for details).
This class works around that issue by first running the password through SHA2-256.
See also: https://blogs.dropbox.com/tech/2016/09/how-dropbox-securely-stores-your-passwords/
who use sha-512 and truncate, rather than sha256.
Format
--------
Bcrypt-SHA256 is compatible with the Modular Crypt Format, and uses $bcrypt-sha256$ as the identifying prefix for all
its strings. An example hash (of password) is:
$bcrypt-sha256$2a,12$LrmaIX5x4TRtAwEfwJZa1.$2ehnw6LvuIUTM0iz4iz9hTxv21B6KFO
Bcrypt-SHA256 hashes have the format
$bcrypt-sha256$[variant],[rounds]$[salt]$[checksum]
where:
- **variant**: is the BCrypt variant in use (usually, as in this case, `2a`).
- **rounds**: is a cost parameter, encoded as decimal integer, which determines the number of iterations used
via `iterations=2**rounds` (rounds is 12 in the example).
- **salt** is a 22 character salt string, using the characters in the regexp range [./A-Za-z0-9]
(LrmaIX5x4TRtAwEfwJZa1. in the example).
- **checksum** is a 31 character checksum, using the same characters as the salt (2ehnw6LvuIUTM0iz4iz9hTxv21B6KFO in the example).
Algorithm
---------
The algorithm this hash uses is as follows:
- first the password is encoded to UTF-8 if not already encoded.
- then it's run through SHA2-256 to generate a 32 byte digest.
- this is encoded using base64, resulting in a 44-byte result (including the trailing padding =).
For the example "password", the output from this stage would be "XohImNooBHFR0OVvjcYpJ3NgPQ1qq73WKhHvch0VQtg=".
- this base64 string is then passed on to the underlying bcrypt algorithm as the new password to be hashed.
}
function GetPerformanceTimestamp: Int64;
begin
{$IFDEF MSWINDOWS}
if not QueryPerformanceCounter({var}Result) then
Result := 0;
{$ELSE}
Result := 0;
{$ENDIF}
end;
{$IFDEF MSWINDOWS}
var
_freq: Int64 = 0;
{$ENDIF}
function PerformanceTimestampToMs(const Timestamp: Int64): Real;
begin
if _freq = 0 then
begin
{$IFDEF MSWINDOWS}
if not QueryPerformanceFrequency({var}_freq) then
_freq := -1;
{$ELSE}
_freq := -1;
{$ENDIF}
end;
Result := Timestamp / _freq * 1000;
end;
procedure DebugOutput(const AText: string);
begin
{
This function is the generic "cross platform" version of OutputDebugString.
It is the same name as the Indy function in IdGlobal that exists for the same reason.
In reality their version detects Kylix and DotNet, and does whatever those things have to have.
That's too much work for me.
I understand your pain, GeoffSmith82. But i need it to remain compatible with Delphi 5.
Yes, you heard me. Delphi 5.
}
{$IFDEF MSWINDOWS}
OutputDebugString(PChar(AText));
{$ENDIF}
end;
{$IFDEF BCryptUnitTests}
type
TBCryptTests = class(TTestCase)
public
procedure SpeedTests;
function GetCompilerOptions: string;
public
//These are just too darn slow (as they should be) for continuous testing
procedure SelfTest;
published
procedure SelfTestA_KnownTestVectors; //known test vectors
procedure SelfTestB_Base64EncoderDecoder; //BSD's base64 encoder/decoder
procedure SelfTestC_UnicodeStrings; //unicode strings in UTF8
procedure SelfTestD_VariableLengthPasswords; //different length passwords
procedure SelfTestE_SaltRNG; //salt rng
procedure SelfTestF_CorrectBattery; //correctbatteryhorsestapler
procedure SelfTestG_PasswordLength; //check that we support up to 72 characters
procedure SelfTestH_OpenBSDLengthBug; //check that we don't limit our passwords to 256 characters (as OpenBSD did)
procedure SelfTestI_UnicodeCompatibleComposition; //check that we apply KC normalization (NIST SP 800-63B)
procedure SelfTestJ_NormalizedPasswordsMatch; //
procedure SelfTestK_SASLprep; //
procedure SelfTestL_Prehash;
procedure Test_ParseHashString; //How well we handle past, present, and future versioning strings
procedure TestEnhancedHash;
procedure TestParseEnhancedHash;
procedure Benchmark;
procedure Test_ManualSystem;
end;
{$ENDIF}
procedure BurnString(var s: UnicodeString); overload;
begin
if Length(s) > 0 then
begin
{$IFDEF UNICODE}
{
In Delphi 5 (and really anything before XE2), UnicodeString is an alias for WideString.
WideString does not have, or does it need, an RTL UniqueString function.
}
UniqueString({var}s); //We can't FillChar the string if it's shared, or its in the constant data page
{$ENDIF}
FillChar(s[1], Length(s)*SizeOf(WideChar), 0);
s := '';
end;
end;
procedure BurnString(var s: AnsiString); overload;
begin
{
If the string is actually constant (reference count of -1), then any attempt to burn it will be
an access violation; as the memory is sitting in a read-only data page.
But Delphi provides no supported way to get the reference count of a string.
It's also an issue if someone else is currently using the string (i.e. Reference Count > 1).
If the string were only referenced by the caller (with a reference count of 1), then
our function here, which received the string through a var reference would also have teh string with
a reference count of one.
Either way, we can only burn the string if there's no other reference.
The use of UniqueString, while counter-intuitiave, is the best approach.
If you pass an unencrypted password to BurnString as a var parameter, and there were another reference,
the string would still contain the password on exit. You can argue that what's the point of making a *copy*
of a string only to burn the copy. Two things:
- if you're debugging it, the string you passed will now be burned (i.e. your local variable will be empty)
- most of the time the RefCount will be 1. When RefCount is one, UniqueString does nothing, so we *are* burning
the only string
}
if Length(s) > 0 then
begin
{
By not calling UniqueString, we only save on a memory allocation and wipe if RefCnt <> 1
It's an unsafe micro-optimization because we're using undocumented offsets to reference counts.
And i'm really uncomfortable using it because it really is undocumented.
It is absolutely a given that it won't change. And we'd have stopped using Delphi long before
it changes. But i just can't do it.
//if PLongInt(PByte(S) - 8)^ = 1 then //RefCnt=1
// ZeroMemory(Pointer(s), System.Length(s)*SizeOf(WideChar));
}
UniqueString({var}s); //ensure the passed in string has a reference count of one
FillChar(s[1], Length(s)*SizeOf(AnsiChar), 0);
s := ''; //We want the callee to see their passed string come back as empty (even if it was shared with other variables)
end;
end;
{ TBCrypt }
class function TBCrypt.HashPassword(const Password: UnicodeString): string;
var
cost: Integer;
begin
{
Generate a hash for the specified password using the default cost.
Sample Usage:
hash := TBCrypt.HashPassword('correct horse battery stample');
Rather than using a fixed default cost, use a self-adjusting cost.
We give ourselves two methods:
- Moore's Law sliding constant
- Benchmark
The problem with using Moore's Law is that it's falling behind for single-core performance.
Since 2004, single-core performance is only going up 21% per year, rather than the 26% of Moore's Law.
26%/year ==> doubles every 18 months
21%/year ==> doubles every 44 months
So i could use a more practical variation of Moore's Law. Knowing that it is now doubling every 44 months,
and that i want the target speed to be between 500-750ms, i could use the new value.
The alternative is to run a quick benchmark. It only takes 1.8ms to do a cost=4 hash. Use it benchmark the computer.
The 3rd alternative would be to run the hash as normal, and time it. If it takes less than 500ms to calculate, then
do it again with a cost of BCRYPT_COST+1.
}
cost := TBCrypt.GetModernCost_Benchmark;
if cost < BCRYPT_COST then
cost := BCRYPT_COST;
Result := TBCrypt.HashPassword(password, cost);
end;
class function TBCrypt.HashPassword(const password: UnicodeString; Cost: Integer): string;
var
salt: TBytes;
hash: TBytes;
begin
{
Generate a hash for the supplied password using the specified cost.
Sample usage:
hash := TBCrypt.HashPassword('Correct battery Horse staple', 13); //Cost factor 13
}
salt := GenerateSalt();
hash := TBCrypt.HashPassword(password, salt, cost);
//20151010 I don't want to emit 2b just yet. The previous bcrypt would fail on anything besides 2a.
//This version handles any single letter suffix. But if we have cross system authentication, and an older system
//tries to validate a 2b password it will fail.
//Wait a year or so until everyone has the new bcrypt
Result := FormatPasswordHashForBsd('2a', cost, salt, hash);
end;
function CoCreateGuid(out guid: TGUID): HResult; stdcall; external 'ole32.dll' name 'CoCreateGuid';
function CreateGUID: TGUID;
begin
//Creates a random (i.e. Type-4) UUID
{$IFDEF COMPILER_7_UP}
OleCheck(SysUtils.CreateGuid({out}Result));
{$ELSE}
OleCheck(CoCreateGuid({out}Result));
{$ENDIF}
end;
class function TBCrypt.GenerateSalt: TBytes;
var
type4Uuid: TGUID;
salt: TBytes;
begin
//Salt is a 128-bit (16 byte) random value
SetLength(salt, BCRYPT_SALT_LEN);
//20150309 Use real random data. Fallback to random guid if it fails
if Failed(Self.GenRandomBytes(BCRYPT_SALT_LEN, {out}@salt[0])) then
begin
//Type 4 UUID (RFC 4122) is a handy source of (almost) 128-bits of random data (actually 120 bits)
//But the security doesn't come from the salt being secret, it comes from the salt being different each time
type4Uuid := CreateGuid;
Move(type4Uuid.D1, salt[0], BCRYPT_SALT_LEN); //16 bytes
end;
Result := salt;
end;
const
advapi32 = 'advapi32.dll';
function CryptAcquireContext(out phProv: THandle; pszContainer: PWideChar; pszProvider: PWideChar; dwProvType: DWORD; dwFlags: DWORD): BOOL; stdcall; external advapi32 name 'CryptAcquireContextW';
function CryptCreateHash(hProv: THandle; Algid: LongWord; hKey: THandle; dwFlags: DWORD; out hHash: THandle): BOOL; stdcall; external advapi32;
function CryptHashData(hHash: THandle; pbData: PByte; dwDataLen: DWORD; dwFlags: DWORD): BOOL; stdcall; external advapi32;
function CryptGetHashParam(hHash: THandle; dwParam: DWORD; pbData: PByte; var dwDataLen: DWORD; dwFlags: DWORD): BOOL; stdcall; external advapi32;
function CryptDestroyHash(hHash: THandle): BOOL; stdcall; external advapi32;
function CryptAcquireContextW(out phProv: THandle; pszContainer: PWideChar; pszProvider: PWideChar; dwProvType: DWORD; dwFlags: DWORD): BOOL; stdcall; external advapi32;
function CryptReleaseContext(hProv: THandle; dwFlags: DWORD): BOOL; stdcall; external advapi32;
function CryptGenRandom(hProv: THandle; dwLen: DWORD; pbBuffer: Pointer): BOOL; stdcall; external advapi32;
{$IFNDEF COMPILER_7_UP}
procedure RaiseLastOSError;
begin
RaiseLastWin32Error;
end;
{$ENDIF}
type
BCRYPT_HANDLE = type THandle;
BCRYPT_ALG_HANDLE = type BCRYPT_HANDLE;
BCRYPT_KEY_HANDLE = type BCRYPT_HANDLE;
BCRYPT_HASH_HANDLE = type BCRYPT_HANDLE;
NTSTATUS = type Cardinal;
function BCryptOpenAlgorithmProvider(out hAlgorithm: BCRYPT_ALG_HANDLE; pszAlgId, pszImplementation: PWideChar; dwFlags: Cardinal): NTSTATUS; stdcall; external 'BCrypt.dll';
function BCryptGetProperty(hObject: BCRYPT_HANDLE; pszProperty: PWideChar; {out}pbOutput: Pointer; cbOutput: Cardinal; out cbResult: Cardinal; dwFlags: Cardinal): NTSTATUS; stdcall; external 'BCrypt.dll';
function BCryptCreateHash(hAlgorithm: BCRYPT_ALG_HANDLE; out hHash: BCRYPT_HASH_HANDLE; pbHashObject: Pointer; cbHashObject: Cardinal; pbSecret: Pointer; cbSecret: Cardinal; dwFlags: DWORD): NTSTATUS; stdcall; external 'BCrypt.dll';
function BCryptHashData(hHash: BCRYPT_HASH_HANDLE; pbInput: Pointer; cbInput: Cardinal; dwFlags: Cardinal): NTSTATUS; stdcall; external 'BCrypt.dll';
function BCryptFinishHash(hHash: BCRYPT_HASH_HANDLE; pbOutput: Pointer; cbOutput: Cardinal; dwFlags: Cardinal): NTSTATUS; stdcall; external 'BCrypt.dll';
function BCryptDestroyHash(hHash: BCRYPT_HASH_HANDLE): NTSTATUS; stdcall; external 'BCrypt.dll';
function BCryptCloseAlgorithmProvider(hAlgorithm: BCRYPT_ALG_HANDLE; dwFlags: Cardinal): NTSTATUS; stdcall; external 'BCrypt.dll';
function NT_SUCCESS(const Status: NTSTATUS): Boolean;
begin
{
Ntdef.h
NT_SUCCESS(Status)
00: success <--
01: information <--
10: warning
11: error
Evaluates to TRUE if the return value specified by
- Status is a success type (0 - 0x3FFFFFFF) 00xx
- or an informational type (0x40000000 - 0x7FFFFFFF). 01xx
}
Result := ((Status and $80000000) = 0);
end;
function FormatNTStatusMessage(const NTStatusMessage: NTSTATUS): string;
var
buffer: PChar;
len: Integer;
Hand: HMODULE;
begin
{
KB259693: How to translate NTSTATUS error codes to message strings
Obtain the formatted message for the given Win32 ErrorCode
Let the OS initialize the Buffer variable. Need to LocalFree it afterward.
}
Hand := SafeLoadLibrary('ntdll.dll');
buffer := nil;
len := FormatMessage(
FORMAT_MESSAGE_ALLOCATE_BUFFER or
FORMAT_MESSAGE_FROM_SYSTEM or
// FORMAT_MESSAGE_IGNORE_INSERTS or
// FORMAT_MESSAGE_ARGUMENT_ARRAY or
FORMAT_MESSAGE_FROM_HMODULE,
Pointer(Hand),
NTStatusMessage,
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
buffer, 0, nil);
try
//Remove the undesired line breaks and '.' char
while (len > 0) and (CharInSet(buffer[len - 1], [#0..#32, '.'])) do Dec(len);
//Convert to Delphi string
SetString(Result, buffer, len);
finally
//Free the OS allocated memory block
LocalFree(HLOCAL(buffer));
end;
FreeLibrary(Hand);
end;
procedure NTStatusCheck(Status: NTSTATUS);
const
SNTError = 'NT Error 0x%.8x: %s';
begin
//Throw on warning or error
if not NT_SUCCESS(Status) then
begin
raise EOleSysError.CreateFmt(SNTError, [
HResultFromNT(Status),
FormatNTStatusMessage(Status)
]);
end;
end;
class function TBCrypt.HmacSha256(const Data: array of Byte; const Key: array of Byte): TBytes;
const
BCRYPT_SHA256_ALGORITHM = 'SHA256'; // OpenAlgorithmProvider.AlgorithmID
BCRYPT_ALG_HANDLE_HMAC_FLAG = $00000008; // OpenAlgorithmProvider.dwFlags
BCRYPT_HASH_LENGTH = 'HashDigestLength';
var
nts: NTSTATUS;
algorithm: BCRYPT_ALG_HANDLE;
bytesReceived: Cardinal;
digestSize: Cardinal;
hash: BCRYPT_HASH_HANDLE;
digest: TBytes;
begin
{
BCrypt hash algorithm identifiers:
- 'md2'
- 'md4'
- 'md5'
- 'sha1'
- 'sha256'
- 'sha384'
- 'sha512'
}
SetLength(Result, 0);
nts := BCryptOpenAlgorithmProvider({out}algorithm, PWideChar(BCRYPT_SHA256_ALGORITHM), nil, BCRYPT_ALG_HANDLE_HMAC_FLAG);
NTStatusCheck(nts);
try
//Get the size of the SHA256 digest. (We already know its 32 bytes, but its never nice to assume)
nts := BCryptGetProperty(Algorithm, BCRYPT_HASH_LENGTH, @digestSize, SizeOf(digestSize), {out}bytesReceived, 0);
NTStatusCheck(nts);
if digestSize <> 32 then
raise Exception.CreateFmt('Digest size of BCRYPT_SHA512_ALGORITHM is not 32 (%d)', [digestSize]);
//Create the hash using our key
nts := BCryptCreateHash(algorithm, {out}hash, nil, 0, Pointer(@Key[0]), Length(Key), 0);
NTStatusCheck(nts);
try
//Hash the data
nts := BCryptHashData(hash, Pointer(@Data[0]), Length(Data), 0);
NTStatusCheck(nts);
//Get the final digest
SetLength(digest, digestSize);
nts := BCryptFinishHash(Hash, @digest[0], Length(digest), 0);
NTStatusCheck(nts);
finally
BCryptDestroyHash(hash);
end;
finally
BCryptCloseAlgorithmProvider(algorithm, 0);
end;
Result := digest;
end;
class function TBCrypt.HashPassword(const password: UnicodeString; const salt: array of Byte; const cost: Integer): TBytes;
var
key: TBytes;
begin
{
The canonical BSD algorithm expects a null-terminated UTF8 key.
If the key is longer than 72 bytes, they truncate the array of bytes to 72.
Yes, this does mean that can can lose the null terminator, and we can chop a multi-byte utf8 code point into an invalid character.
}
//Pseudo-standard dictates that unicode strings are converted to UTF8 (rather than UTF16, UTF32, UTF16LE, ISO-8859-1, Windows-1252, etc)
key := TBCrypt.PasswordStringPrep(password+#0); // bcrypt calls for the string to include a null terminator
try
//Truncate if its longer than 72 bytes (BCRYPT_MaxKeyLen), and burn the excess
if Length(key) > BCRYPT_MaxKeyLen then
begin
ZeroMemory(@key[BCRYPT_MaxKeyLen], Length(key)-BCRYPT_MaxKeyLen);
SetLength(key,BCRYPT_MaxKeyLen);
end;
Result := TBCrypt.CryptCore(cost, key, salt);
finally
if Length(key) > 0 then
begin
ZeroMemory(@key[0], Length(key));
SetLength(key, 0);
end;
end;
end;
{$OVERFLOWCHECKS OFF}
{$RANGECHECKS OFF}
{
Encrypt a single 64-bit block encoded as two 32-bit halves.
InData: Pointer to two
}
procedure BlowfishEncryptECB(const Data: TBlowfishData; InData: PLongWord; OutData: PLongWord);
var
xL, xR: LongWord;
S0, S1, S2, S3: PSBox;
begin
xL := PLongWord(InData)^;
xR := PLongWord(UIntPtr(InData)+4)^;
xL := (xL shr 24) or ((xL shr 8) and $FF00) or ((xL shl 8) and $FF0000) or (xL shl 24);