| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186 | #! /usr/bin/env perl# Copyright 1999-2018 The OpenSSL Project Authors. All Rights Reserved.## Licensed under the Apache License 2.0 (the "License").  You may not use# this file except in compliance with the License.  You can obtain a copy# in the file LICENSE in the source distribution or at# https://www.openssl.org/source/license.htmlpackage x86nasm;*out=\@::out;$::lbdecor="L\$";		# local label decoration$nmdecor="_";			# external name decoration$drdecor=$::mwerks?".":"";	# directive decoration$initseg="";sub ::generic{ my $opcode=shift;  my $tmp;    if (!$::mwerks)    {   if    ($opcode =~ m/^j/o && $#_==0) # optimize jumps	{   $_[0] = "NEAR $_[0]";   	}	elsif ($opcode eq "lea" && $#_==1)  # wipe storage qualifier from lea	{   $_[1] =~ s/^[^\[]*\[/\[/o;	}	elsif ($opcode eq "clflush" && $#_==0)	{   $_[0] =~ s/^[^\[]*\[/\[/o;	}    }    &::emit($opcode,@_);  1;}## opcodes not covered by ::generic above, mostly inconsistent namings...#sub ::call	{ &::emit("call",(&::islabel($_[0]) or "$nmdecor$_[0]")); }sub ::call_ptr	{ &::emit("call",@_);	}sub ::jmp_ptr	{ &::emit("jmp",@_);	}sub get_mem{ my($size,$addr,$reg1,$reg2,$idx)=@_;  my($post,$ret);    if (!defined($idx) && 1*$reg2) { $idx=$reg2; $reg2=$reg1; undef $reg1; }    if ($size ne "")    {	$ret .= "$size";	$ret .= " PTR" if ($::mwerks);	$ret .= " ";    }    $ret .= "[";    $addr =~ s/^\s+//;    # prepend global references with optional underscore    $addr =~ s/^([^\+\-0-9][^\+\-]*)/::islabel($1) or "$nmdecor$1"/ige;    # put address arithmetic expression in parenthesis    $addr="($addr)" if ($addr =~ /^.+[\-\+].+$/);    if (($addr ne "") && ($addr ne 0))    {	if ($addr !~ /^-/)	{ $ret .= "$addr+"; }	else			{ $post=$addr;      }    }    if ($reg2 ne "")    {	$idx!=0 or $idx=1;	$ret .= "$reg2*$idx";	$ret .= "+$reg1" if ($reg1 ne "");    }    else    {	$ret .= "$reg1";   }    $ret .= "$post]";    $ret =~ s/\+\]/]/; # in case $addr was the only argument  $ret;}sub ::BP	{ &get_mem("BYTE",@_);  }sub ::DWP	{ &get_mem("DWORD",@_); }sub ::WP	{ &get_mem("WORD",@_);	}sub ::QWP	{ &get_mem("",@_);      }sub ::BC	{ (($::mwerks)?"":"BYTE ")."@_";  }sub ::DWC	{ (($::mwerks)?"":"DWORD ")."@_"; }sub ::file{   if ($::mwerks)	{ push(@out,".section\t.text,64\n"); }    else    { my $tmp=<<___;%ifidn __OUTPUT_FORMAT__,objsection	code	use32 class=code align=64%elifidn __OUTPUT_FORMAT__,win32\$\@feat.00 equ 1section	.text	code align=64%elsesection	.text	code%endif___	push(@out,$tmp);    }}sub ::function_begin_B{ my $func=shift;  my $global=($func !~ /^_/);  my $begin="${::lbdecor}_${func}_begin";    $begin =~ s/^\@/./ if ($::mwerks);	# the torture never stops    &::LABEL($func,$global?"$begin":"$nmdecor$func");    $func=$nmdecor.$func;    push(@out,"${drdecor}global	$func\n")	if ($global);    push(@out,"${drdecor}align	16\n");    push(@out,"$func:\n");    push(@out,"$begin:\n")			if ($global);    $::stack=4;}sub ::function_end_B{   $::stack=0;    &::wipe_labels();}sub ::file_end{   if (grep {/\b${nmdecor}OPENSSL_ia32cap_P\b/i} @out)    {	my $comm=<<___;${drdecor}segment	.bss${drdecor}common	${nmdecor}OPENSSL_ia32cap_P 16___	# comment out OPENSSL_ia32cap_P declarations	grep {s/(^extern\s+${nmdecor}OPENSSL_ia32cap_P)/\;$1/} @out;	push (@out,$comm)    }    push (@out,$initseg) if ($initseg);}sub ::comment {   foreach (@_) { push(@out,"\t; $_\n"); }   }sub ::external_label{   foreach(@_)    {	push(@out,"${drdecor}extern\t".&::LABEL($_,$nmdecor.$_)."\n");   }}sub ::public_label{   push(@out,"${drdecor}global\t".&::LABEL($_[0],$nmdecor.$_[0])."\n");  }sub ::data_byte{   push(@out,(($::mwerks)?".byte\t":"db\t").join(',',@_)."\n");	}sub ::data_short{   push(@out,(($::mwerks)?".word\t":"dw\t").join(',',@_)."\n");	}sub ::data_word{   push(@out,(($::mwerks)?".long\t":"dd\t").join(',',@_)."\n");	}sub ::align{   push(@out,"${drdecor}align\t$_[0]\n");	}sub ::picmeup{ my($dst,$sym)=@_;    &::lea($dst,&::DWP($sym));}sub ::initseg{ my $f=$nmdecor.shift;    if ($::win32)    {	$initseg=<<___;segment	.CRT\$XCU data align=4extern	$fdd	$f___    }}sub ::dataseg{   if ($mwerks)	{ push(@out,".section\t.data,4\n");   }    else		{ push(@out,"section\t.data align=4\n"); }}sub ::safeseh{ my $nm=shift;    push(@out,"%if	__NASM_VERSION_ID__ >= 0x02030000\n");    push(@out,"safeseh	".&::LABEL($nm,$nmdecor.$nm)."\n");    push(@out,"%endif\n");}1;
 |