$ filename "Checksums.exe"
'uses rtl32
'uses rtl64
uses console
'RtlComputeCrc32 (DWORD dwInitial, const BYTE *pData, INT iLen) result dword
'RtlCrc32 ( void const* Buffer, size_t Size, DWORD InitialCrc ) result dword
'RtlCrc64 ( void const* Buffer, size_t Size, ULONGLONG InitialCrc ) result ulonglong
string fname="Checksums.o2bas"
string s
getfile fname, s
extern lib "ntdll.dll"
! RtlComputeCrc32
! RtlCrc32
! RtlCrc64
end extern
printl hex (RtlComputeCrc32 0, s, len s)
printl right(hex (RtlComputeCrc32 0, s, len(s), 8))
printl
printl hex (RtlCrc32 s, len(s), 0) ": Incorrect."
printl "Enter..."
waitkey
'Ported from FBSL C-code (Mike Lobanowsky)
$ filename "CheckSumCRC32.exe"
'uses rtl32
'uses rtl64
uses console
ulong crc32_table[256] ' Lookup table array
' Utility functions
function Reflect(uint ref, byte ch) as uint
' Used only by Init_CRC32_Table()
uint value = 0
' Swap bit 0 for bit 7
' bit 1 for bit 6, etc.
int i
for i = 1 to ch + 1
if (ref and 1) then value = value or 1 << (ch - i)
ref = ref >> 1
next i
return value
end function
' Call this function only once to initialize the CRC table.
sub Init_CRC32_Table()
' This is the official polynomial used by CRC-32
' in PKZip, WinZip and Ethernet.
uint ulPolynomial = 0x04c11db7
' 256 values representing ASCII character codes.
int i
for i = 0 to 0xFF
crc32_table[i] = Reflect(i, 8) << 24
int j
for j = 0 to 7
uint mask = crc32_table[i] and (1 << 31)
if mask then
crc32_table[i] = (crc32_table[i] << 1) xor ulPolynomial
else
crc32_table[i] = (crc32_table[i] << 1) xor 0
end if
next j
crc32_table[i] = Reflect(crc32_table[i], 32)
next i
end sub
' Main function
function CheckSum(string buffer) as uint
if (crc32_table[1] == 0) then Init_CRC32_Table() ' do table init once
' Be sure to use unsigned variables,
' because negative values introduce high bits
' where zero bits are required.
' Start out with all bits set high.
uint ulCRC = 0xffffffff
' Perform the algorithm on each character
' in the string, using the lookup table values.
int datalen=len(buffer)
int ix=1
while dataLen
dataLen -= 1
'older oxygen.dll
'int idx = (ulCRC and 0xFF) xor asc(mid(buffer,ix,1))
'ulCRC = (ulCRC >> 8) xor crc32_table[idx]
'newer oxygen.dll
ulCRC = (ulCRC >> 8) xor crc32_table[(ulCRC and 0xFF) xor asc(mid(buffer,ix,1))]
ix+=1
wend
' Exclusive OR the result with the beginning value.
return ulCRC xor 0xffffffff
end function
' ------ main ------
string s = "The quick brown fox jumps over the lazy dog"
printl "Text = " s
printl "The CRC-32 checksum = " right (hex(CheckSum(s)),8) '414FA339
string fname= "CheckSumCRC32.o2bas"
getfile fname, s
printl
printl "Filename = " fname
printl "The CRC-32 checksum = " right (hex(CheckSum(s)),8)
printl "Enter ..."
waitkey
'https://www.w3.org/TR/PNG/#D-CRCAppendix
$ filename "CheckCRC32.exe"
'uses rtl32
'uses rtl64
uses console
function crc32(string buf) as uint
indexbase 0
static uint crc_table[256]
static uint crc_table_computed
uint crc
int n, k
if crc_table_computed = 0 then
for n = 0 to 255
crc = n
for k = 0 to 7
if (crc and 1) then crc = (crc >> 1) xor 0xedb88320 else crc = crc >> 1
crc_table[n] = crc
next k
next n
crc_table_computed = 1
end if
crc = 0xffffffff
for n = 1 to len(buf)
'older oxygen.dll
'int idx = (crc and 0xff) xor asc(mid(buf,n,1)))
'crc = (crc >> 8) xor crc_table[idx]
'newer oxygen.dll
crc = (crc >> 8) xor crc_table[(crc and 0xff) xor asc(mid(buf,n,1))]
next
return not crc
end function
' ------ main ------
string s = "The quick brown fox jumps over the lazy dog"
printl "Text = " s
printl "The CRC-32 checksum = " right (hex(crc32(s)),8) '414FA339
string fname= "CheckCRC32.o2bas"
getfile fname, s
printl
printl "Filename = " fname
printl "The CRC-32 checksum = " right (hex(crc32(s)),8)
printl "Enter..."
waitkey
... especially when creating exe files.
'PE FILE CRC STAMP
'https://docs.microsoft.com/en-us/windows/desktop/api/imagehlp/nf-imagehlp-mapfileandchecksuma
! MapFileAndCheckSumA lib "Imagehlp.dll"
dword e,i,o
e=MapFileAndCheckSumA("oxygen.dll",@i,@o)
select e
case 2 : print "Could not map the file"
case 3 : print "Could not map a view of the file"
case 1 : print "Could not open the file"
case 4 : print "Could not convert the file name to Unicode"
case else : print hex(i) " " hex(o) " " e
end select
'MapFileAndCheckSumA
'https://docs.microsoft.com/en-us/windows/desktop/api/imagehlp/nf-imagehlp-mapfileandchecksuma
'http://www.sunshine2k.de/reversing/tuts/tut_pe.htm
'https://www.codeproject.com/Articles/19326/An-Analysis-of-the-Windows-PE-Checksum-Algorithm
'https://bytepointer.com/resources/microsoft_pe_checksum_algo_distilled.htm
'
! MapFileAndCheckSumA lib "Imagehlp.dll"
'
string fi
string s
fi="oxygen.dll"
getfile fi,s
'
int *ck 'checksum
dword b at strptr(s)+60 'PE header offset
@ck=strptr(s)+b+0x58 'checksum location
'
'CLEAR CHECKSUM
ck=0
'
'CALCULATE CHECKSUM
uint uw 'upper word
uint ls 'byte length of data
uint ca 'checksum accumulator
word *pd 'input
int j 'iterator
ls=len(s)
@pd=strptr(s)
for j=1 to ls step 2 'each word
ca+=pd
uw=ca>>16
ca and=0xffff
ca+=uw
@pd+=2
next
'final
uw=ca>>16
ca=uw+ca
ca and= 0xffff
ca+=ls 'add in byte length of data
'print hex(ca)
'
'SET CHECKSUM
ck=ca
putfile fi,s
'VERIFY
=======
dword e ' error
dword i ' previous checksum
dword o ' new checksum
'
e=MapFileAndCheckSumA("oxygen.dll",@i,@o)
select e
case 2 : print "Could not map the file"
case 3 : print "Could not map a view of the file"
case 1 : print "Could not open the file"
case 4 : print "Could not convert the file name to Unicode"
case else : print hex(i) " " hex(o) " " e '" " hex(crc32(strptr s, len s))
end select
'
endif
'
'
'CALCULATE AND SET CHECKSUM
'==========================
'
int *ck 'checksum
sys ps=strptr(dll)
dword b at ps+60 'PE header offset
@ck=ps+b+0x58 'checksum location
ck=0 'clear checksum
'
uint ls 'byte length of data
ls=len(dll)
'word *pd 'input
xor eax,eax 'uint ca 'checksum accumulator
xor edx,edx 'uint uw 'upper word '
mov rsi,ps '@pd=ps
mov ecx,ls 'int j 'iterator
( 'for j=1 to ls step 2 'each word
mov dx,[rsi] '
add eax,edx ' ca+=pd
mov edx,eax ' uw=ca>>16
shr edx,16 '
and eax,0xffff ' ca and=0xffff
add eax,edx ' ca+=uw
add rsi,2 ' @pd+=2
sub ecx,2 '
jg repeat '
) 'next
'final '
mov edx,eax 'uw=ca>>16
shr edx,16 '
add eax,edx 'ca=uw+ca
and eax,0xffff 'ca and= 0xffff
add eax,ls 'ca+=ls 'add in byte length of data
addr rsi,ck 'ck=ca 'set checksum
mov [rsi],eax '
'
http://bytepointer.com/resources/microsoft_pe_checksum_algo_distilled.htm
fatal: unable to stat 'examples/OOP/Containers/.goutputstream-CHIKWZ': No such file or directory
which required some adjustments to the include files.
typedef struct tagNMHDR {
HWND hwndFrom; '8
UINT_PTR idFrom; '8
UINT code; '4
'4 bytes padding
} NMHDR;