<html><head><meta name="color-scheme" content="light dark"></head><body><pre style="word-wrap: break-word; white-space: pre-wrap;">#!/usr/local/bin/perl

# void des_ncbc_encrypt(input, output, length, schedule, ivec, enc)
# des_cblock (*input);
# des_cblock (*output);
# long length;
# des_key_schedule schedule;
# des_cblock (*ivec);
# int enc;
#
# calls 
# des_encrypt((DES_LONG *)tin,schedule,DES_ENCRYPT);
#

#&amp;cbc("des_ncbc_encrypt","des_encrypt",0);
#&amp;cbc("BF_cbc_encrypt","BF_encrypt","BF_encrypt",
#	1,4,5,3,5,-1);
#&amp;cbc("des_ncbc_encrypt","des_encrypt","des_encrypt",
#	0,4,5,3,5,-1);
#&amp;cbc("des_ede3_cbc_encrypt","des_encrypt3","des_decrypt3",
#	0,6,7,3,4,5);
#
# When doing a cipher that needs bigendian order,
# for encrypt, the iv is kept in bigendian form,
# while for decrypt, it is kept in little endian.
sub cbc
	{
	local($name,$enc_func,$dec_func,$swap,$iv_off,$enc_off,$p1,$p2,$p3)=@_;
	# name is the function name
	# enc_func and dec_func and the functions to call for encrypt/decrypt
	# swap is true if byte order needs to be reversed
	# iv_off is parameter number for the iv 
	# enc_off is parameter number for the encrypt/decrypt flag
	# p1,p2,p3 are the offsets for parameters to be passed to the
	# underlying calls.

	&amp;function_begin_B($name,"");
	&amp;comment("");

	$in="esi";
	$out="edi";
	$count="ebp";

	&amp;push("ebp");
	&amp;push("ebx");
	&amp;push("esi");
	&amp;push("edi");

	$data_off=4;
	$data_off+=4 if ($p1 &gt; 0);
	$data_off+=4 if ($p2 &gt; 0);
	$data_off+=4 if ($p3 &gt; 0);

	&amp;mov($count,	&amp;wparam(2));	# length

	&amp;comment("getting iv ptr from parameter $iv_off");
	&amp;mov("ebx",	&amp;wparam($iv_off));	# Get iv ptr

	&amp;mov($in,	&amp;DWP(0,"ebx","",0));#	iv[0]
	&amp;mov($out,	&amp;DWP(4,"ebx","",0));#	iv[1]

	&amp;push($out);
	&amp;push($in);
	&amp;push($out);	# used in decrypt for iv[1]
	&amp;push($in);	# used in decrypt for iv[0]

	&amp;mov("ebx",	"esp");		# This is the address of tin[2]

	&amp;mov($in,	&amp;wparam(0));	# in
	&amp;mov($out,	&amp;wparam(1));	# out

	# We have loaded them all, how lets push things
	&amp;comment("getting encrypt flag from parameter $enc_off");
	&amp;mov("ecx",	&amp;wparam($enc_off));	# Get enc flag
	if ($p3 &gt; 0)
		{
		&amp;comment("get and push parameter $p3");
		if ($enc_off != $p3)
			{ &amp;mov("eax",	&amp;wparam($p3)); &amp;push("eax"); }
		else	{ &amp;push("ecx"); }
		}
	if ($p2 &gt; 0)
		{
		&amp;comment("get and push parameter $p2");
		if ($enc_off != $p2)
			{ &amp;mov("eax",	&amp;wparam($p2)); &amp;push("eax"); }
		else	{ &amp;push("ecx"); }
		}
	if ($p1 &gt; 0)
		{
		&amp;comment("get and push parameter $p1");
		if ($enc_off != $p1)
			{ &amp;mov("eax",	&amp;wparam($p1)); &amp;push("eax"); }
		else	{ &amp;push("ecx"); }
		}
	&amp;push("ebx");		# push data/iv

	&amp;cmp("ecx",0);
	&amp;jz(&amp;label("decrypt"));

	&amp;and($count,0xfffffff8);
	&amp;mov("eax",	&amp;DWP($data_off,"esp","",0));	# load iv[0]
	&amp;mov("ebx",	&amp;DWP($data_off+4,"esp","",0));	# load iv[1]

	&amp;jz(&amp;label("encrypt_finish"));

	#############################################################

	&amp;set_label("encrypt_loop");
	# encrypt start 
	# "eax" and "ebx" hold iv (or the last cipher text)

	&amp;mov("ecx",	&amp;DWP(0,$in,"",0));	# load first 4 bytes
	&amp;mov("edx",	&amp;DWP(4,$in,"",0));	# second 4 bytes

	&amp;xor("eax",	"ecx");
	&amp;xor("ebx",	"edx");

	&amp;bswap("eax")	if $swap;
	&amp;bswap("ebx")	if $swap;

	&amp;mov(&amp;DWP($data_off,"esp","",0),	"eax");	# put in array for call
	&amp;mov(&amp;DWP($data_off+4,"esp","",0),	"ebx");	#

	&amp;call($enc_func);

	&amp;mov("eax",	&amp;DWP($data_off,"esp","",0));
	&amp;mov("ebx",	&amp;DWP($data_off+4,"esp","",0));

	&amp;bswap("eax")	if $swap;
	&amp;bswap("ebx")	if $swap;

	&amp;mov(&amp;DWP(0,$out,"",0),"eax");
	&amp;mov(&amp;DWP(4,$out,"",0),"ebx");

	# eax and ebx are the next iv.

	&amp;add($in,	8);
	&amp;add($out,	8);

	&amp;sub($count,	8);
	&amp;jnz(&amp;label("encrypt_loop"));

###################################################################3
	&amp;set_label("encrypt_finish");
	&amp;mov($count,	&amp;wparam(2));	# length
	&amp;and($count,	7);
	&amp;jz(&amp;label("finish"));
	&amp;call(&amp;label("PIC_point"));
&amp;set_label("PIC_point");
	&amp;blindpop("edx");
	&amp;lea("ecx",&amp;DWP(&amp;label("cbc_enc_jmp_table")."-".&amp;label("PIC_point"),"edx"));
	&amp;mov($count,&amp;DWP(0,"ecx",$count,4))
	&amp;add($count,"edx");
	&amp;xor("ecx","ecx");
	&amp;xor("edx","edx");
	#&amp;mov($count,&amp;DWP(&amp;label("cbc_enc_jmp_table"),"",$count,4));
	&amp;jmp_ptr($count);

&amp;set_label("ej7");
	&amp;xor("edx",		"edx") if $ppro; # ppro friendly
	&amp;movb(&amp;HB("edx"),	&amp;BP(6,$in,"",0));
	&amp;shl("edx",8);
&amp;set_label("ej6");
	&amp;movb(&amp;HB("edx"),	&amp;BP(5,$in,"",0));
&amp;set_label("ej5");
	&amp;movb(&amp;LB("edx"),	&amp;BP(4,$in,"",0));
&amp;set_label("ej4");
	&amp;mov("ecx",		&amp;DWP(0,$in,"",0));
	&amp;jmp(&amp;label("ejend"));
&amp;set_label("ej3");
	&amp;movb(&amp;HB("ecx"),	&amp;BP(2,$in,"",0));
	&amp;xor("ecx",		"ecx") if $ppro; # ppro friendly
	&amp;shl("ecx",8);
&amp;set_label("ej2");
	&amp;movb(&amp;HB("ecx"),	&amp;BP(1,$in,"",0));
&amp;set_label("ej1");
	&amp;movb(&amp;LB("ecx"),	&amp;BP(0,$in,"",0));
&amp;set_label("ejend");

	&amp;xor("eax",	"ecx");
	&amp;xor("ebx",	"edx");

	&amp;bswap("eax")	if $swap;
	&amp;bswap("ebx")	if $swap;

	&amp;mov(&amp;DWP($data_off,"esp","",0),	"eax");	# put in array for call
	&amp;mov(&amp;DWP($data_off+4,"esp","",0),	"ebx");	#

	&amp;call($enc_func);

	&amp;mov("eax",	&amp;DWP($data_off,"esp","",0));
	&amp;mov("ebx",	&amp;DWP($data_off+4,"esp","",0));

	&amp;bswap("eax")	if $swap;
	&amp;bswap("ebx")	if $swap;

	&amp;mov(&amp;DWP(0,$out,"",0),"eax");
	&amp;mov(&amp;DWP(4,$out,"",0),"ebx");

	&amp;jmp(&amp;label("finish"));

	#############################################################
	#############################################################
	&amp;set_label("decrypt",1);
	# decrypt start 
	&amp;and($count,0xfffffff8);
	# The next 2 instructions are only for if the jz is taken
	&amp;mov("eax",	&amp;DWP($data_off+8,"esp","",0));	# get iv[0]
	&amp;mov("ebx",	&amp;DWP($data_off+12,"esp","",0));	# get iv[1]
	&amp;jz(&amp;label("decrypt_finish"));

	&amp;set_label("decrypt_loop");
	&amp;mov("eax",	&amp;DWP(0,$in,"",0));	# load first 4 bytes
	&amp;mov("ebx",	&amp;DWP(4,$in,"",0));	# second 4 bytes

	&amp;bswap("eax")	if $swap;
	&amp;bswap("ebx")	if $swap;

	&amp;mov(&amp;DWP($data_off,"esp","",0),	"eax");	# put back
	&amp;mov(&amp;DWP($data_off+4,"esp","",0),	"ebx");	#

	&amp;call($dec_func);

	&amp;mov("eax",	&amp;DWP($data_off,"esp","",0));	# get return
	&amp;mov("ebx",	&amp;DWP($data_off+4,"esp","",0));	#

	&amp;bswap("eax")	if $swap;
	&amp;bswap("ebx")	if $swap;

	&amp;mov("ecx",	&amp;DWP($data_off+8,"esp","",0));	# get iv[0]
	&amp;mov("edx",	&amp;DWP($data_off+12,"esp","",0));	# get iv[1]

	&amp;xor("ecx",	"eax");
	&amp;xor("edx",	"ebx");

	&amp;mov("eax",	&amp;DWP(0,$in,"",0));	# get old cipher text,
	&amp;mov("ebx",	&amp;DWP(4,$in,"",0));	# next iv actually

	&amp;mov(&amp;DWP(0,$out,"",0),"ecx");
	&amp;mov(&amp;DWP(4,$out,"",0),"edx");

	&amp;mov(&amp;DWP($data_off+8,"esp","",0),	"eax");	# save iv
	&amp;mov(&amp;DWP($data_off+12,"esp","",0),	"ebx");	#

	&amp;add($in,	8);
	&amp;add($out,	8);

	&amp;sub($count,	8);
	&amp;jnz(&amp;label("decrypt_loop"));
############################ ENDIT #######################3
	&amp;set_label("decrypt_finish");
	&amp;mov($count,	&amp;wparam(2));	# length
	&amp;and($count,	7);
	&amp;jz(&amp;label("finish"));

	&amp;mov("eax",	&amp;DWP(0,$in,"",0));	# load first 4 bytes
	&amp;mov("ebx",	&amp;DWP(4,$in,"",0));	# second 4 bytes

	&amp;bswap("eax")	if $swap;
	&amp;bswap("ebx")	if $swap;

	&amp;mov(&amp;DWP($data_off,"esp","",0),	"eax");	# put back
	&amp;mov(&amp;DWP($data_off+4,"esp","",0),	"ebx");	#

	&amp;call($dec_func);

	&amp;mov("eax",	&amp;DWP($data_off,"esp","",0));	# get return
	&amp;mov("ebx",	&amp;DWP($data_off+4,"esp","",0));	#

	&amp;bswap("eax")	if $swap;
	&amp;bswap("ebx")	if $swap;

	&amp;mov("ecx",	&amp;DWP($data_off+8,"esp","",0));	# get iv[0]
	&amp;mov("edx",	&amp;DWP($data_off+12,"esp","",0));	# get iv[1]

	&amp;xor("ecx",	"eax");
	&amp;xor("edx",	"ebx");

	# this is for when we exit
	&amp;mov("eax",	&amp;DWP(0,$in,"",0));	# get old cipher text,
	&amp;mov("ebx",	&amp;DWP(4,$in,"",0));	# next iv actually

&amp;set_label("dj7");
	&amp;rotr("edx",	16);
	&amp;movb(&amp;BP(6,$out,"",0),	&amp;LB("edx"));
	&amp;shr("edx",16);
&amp;set_label("dj6");
	&amp;movb(&amp;BP(5,$out,"",0),	&amp;HB("edx"));
&amp;set_label("dj5");
	&amp;movb(&amp;BP(4,$out,"",0),	&amp;LB("edx"));
&amp;set_label("dj4");
	&amp;mov(&amp;DWP(0,$out,"",0),	"ecx");
	&amp;jmp(&amp;label("djend"));
&amp;set_label("dj3");
	&amp;rotr("ecx",	16);
	&amp;movb(&amp;BP(2,$out,"",0),	&amp;LB("ecx"));
	&amp;shl("ecx",16);
&amp;set_label("dj2");
	&amp;movb(&amp;BP(1,$in,"",0),	&amp;HB("ecx"));
&amp;set_label("dj1");
	&amp;movb(&amp;BP(0,$in,"",0),	&amp;LB("ecx"));
&amp;set_label("djend");

	# final iv is still in eax:ebx
	&amp;jmp(&amp;label("finish"));


############################ FINISH #######################3
	&amp;set_label("finish",1);
	&amp;mov("ecx",	&amp;wparam($iv_off));	# Get iv ptr

	#################################################
	$total=16+4;
	$total+=4 if ($p1 &gt; 0);
	$total+=4 if ($p2 &gt; 0);
	$total+=4 if ($p3 &gt; 0);
	&amp;add("esp",$total);

	&amp;mov(&amp;DWP(0,"ecx","",0),	"eax");	# save iv
	&amp;mov(&amp;DWP(4,"ecx","",0),	"ebx");	# save iv

	&amp;function_end_A($name);

	&amp;set_label("cbc_enc_jmp_table",1);
	&amp;data_word("0");
	&amp;data_word(&amp;label("ej1")."-".&amp;label("PIC_point"));
	&amp;data_word(&amp;label("ej2")."-".&amp;label("PIC_point"));
	&amp;data_word(&amp;label("ej3")."-".&amp;label("PIC_point"));
	&amp;data_word(&amp;label("ej4")."-".&amp;label("PIC_point"));
	&amp;data_word(&amp;label("ej5")."-".&amp;label("PIC_point"));
	&amp;data_word(&amp;label("ej6")."-".&amp;label("PIC_point"));
	&amp;data_word(&amp;label("ej7")."-".&amp;label("PIC_point"));
	# not used
	#&amp;set_label("cbc_dec_jmp_table",1);
	#&amp;data_word("0");
	#&amp;data_word(&amp;label("dj1")."-".&amp;label("PIC_point"));
	#&amp;data_word(&amp;label("dj2")."-".&amp;label("PIC_point"));
	#&amp;data_word(&amp;label("dj3")."-".&amp;label("PIC_point"));
	#&amp;data_word(&amp;label("dj4")."-".&amp;label("PIC_point"));
	#&amp;data_word(&amp;label("dj5")."-".&amp;label("PIC_point"));
	#&amp;data_word(&amp;label("dj6")."-".&amp;label("PIC_point"));
	#&amp;data_word(&amp;label("dj7")."-".&amp;label("PIC_point"));

	&amp;function_end_B($name);
	
	}

1;
</pre></body></html>