123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124 |
- The perl scripts in this directory are my 'hack' to generate
- multiple different assembler formats via the one original script.
- The way to use this library is to start with adding the path to this directory
- and then include it.
- push(@INC,"perlasm","../../perlasm");
- require "x86asm.pl";
- The first thing we do is setup the file and type of assembler
- &asm_init($ARGV[0]);
- The first argument is the 'type'. Currently
- 'cpp', 'sol', 'a.out', 'elf' or 'win32'.
- Argument 2 is the file name.
- The reciprocal function is
- &asm_finish() which should be called at the end.
- There are 2 main 'packages'. x86ms.pl, which is the Microsoft assembler,
- and x86unix.pl which is the unix (gas) version.
- Functions of interest are:
- &external_label("des_SPtrans"); declare and external variable
- &LB(reg); Low byte for a register
- &HB(reg); High byte for a register
- &BP(off,base,index,scale) Byte pointer addressing
- &DWP(off,base,index,scale) Word pointer addressing
- &stack_push(num) Basically a 'sub esp, num*4' with extra
- &stack_pop(num) inverse of stack_push
- &function_begin(name,extra) Start a function with pushing of
- edi, esi, ebx and ebp. extra is extra win32
- external info that may be required.
- &function_begin_B(name,extra) Same as normal function_begin but no pushing.
- &function_end(name) Call at end of function.
- &function_end_A(name) Standard pop and ret, for use inside functions
- &function_end_B(name) Call at end but with pop or ret.
- &swtmp(num) Address on stack temp word.
- &wparam(num) Parameter number num, that was push
- in C convention. This all works over pushes
- and pops.
- &comment("hello there") Put in a comment.
- &label("loop") Refer to a label, normally a jmp target.
- &set_label("loop") Set a label at this point.
- &data_word(word) Put in a word of data.
- So how does this all hold together? Given
- int calc(int len, int *data)
- {
- int i,j=0;
- for (i=0; i<len; i++)
- {
- j+=other(data[i]);
- }
- }
- So a very simple version of this function could be coded as
- push(@INC,"perlasm","../../perlasm");
- require "x86asm.pl";
-
- &asm_init($ARGV[0]);
- &external_label("other");
- $tmp1= "eax";
- $j= "edi";
- $data= "esi";
- $i= "ebp";
- &comment("a simple function");
- &function_begin("calc");
- &mov( $data, &wparam(1)); # data
- &xor( $j, $j);
- &xor( $i, $i);
- &set_label("loop");
- &cmp( $i, &wparam(0));
- &jge( &label("end"));
- &mov( $tmp1, &DWP(0,$data,$i,4));
- &push( $tmp1);
- &call( "other");
- &add( $j, "eax");
- &pop( $tmp1);
- &inc( $i);
- &jmp( &label("loop"));
- &set_label("end");
- &mov( "eax", $j);
- &function_end("calc");
- &asm_finish();
- The above example is very very unoptimised but gives an idea of how
- things work.
- There is also a cbc mode function generator in cbc.pl
- &cbc( $name,
- $encrypt_function_name,
- $decrypt_function_name,
- $true_if_byte_swap_needed,
- $parameter_number_for_iv,
- $parameter_number_for_encrypt_flag,
- $first_parameter_to_pass,
- $second_parameter_to_pass,
- $third_parameter_to_pass);
- So for example, given
- void BF_encrypt(BF_LONG *data,BF_KEY *key);
- void BF_decrypt(BF_LONG *data,BF_KEY *key);
- void BF_cbc_encrypt(unsigned char *in, unsigned char *out, long length,
- BF_KEY *ks, unsigned char *iv, int enc);
- &cbc("BF_cbc_encrypt","BF_encrypt","BF_encrypt",1,4,5,3,-1,-1);
- &cbc("des_ncbc_encrypt","des_encrypt","des_encrypt",0,4,5,3,5,-1);
- &cbc("des_ede3_cbc_encrypt","des_encrypt3","des_decrypt3",0,6,7,3,4,5);
|