.file "asma.c" .text .globl asma .type asma, @function asma: .LFB2: /* unsigned char *asma(unsigned char *s) { int i; for (i=0; i<16; i++) { unsigned char c=s[i]; c += (c>=’A’ && c<=’Z’) ? ’a’-’A’ : 0; s[i] = c; } return s; } Schreiben Sie diese Funktion in Assembler unter Verwendung von pcmpgtb. Dabei ist folgende Äquivalenz hilfreich: (c>=’A’ && c<=’Z’) ? ’a’-’A’ : 0; ist (bei Verwendung von Überlauf-Arithmetik) äquivalent zu min(’Z’+1+min_t-’A’ > c+min_t-’A’ ? 0xff : 0, ’a’-’A’) wobei min_t der minimale Wert des Datentyps ist, den der Vergleich behandelt (bei pcmpgtb also -128). Zusätzlich zu dem oben genannten dürften die Befehle pminub, paddb, und psubb nützlich sein. */ pxor %xmm1, %xmm1 pxor %xmm2, %xmm2 pxor %xmm2, %xmm3 pxor %xmm2, %xmm4 pxor %xmm2, %xmm5 #init %xmm2 mit "'Z' + 1 + min_t - 'A'" fuer jedes byte #'Z' + 1 + min_t - 'A' = 90 + 1 - 128 - 65 = -102 #102 = 01100110 #~102 = 10011001 #(~102)+1 = 10011010 = 0x9A mov $0x9a9a9a9a9a9a9a9a, %rbx movq %rbx, %xmm2 pslldq $8, %xmm2 movq %rbx, %xmm2 #init %xmm4 mit "'a'-'A'= 97-65 = 32 = 0x20 mov $0x2020202020202020, %rbx movq %rbx, %xmm4 pslldq $8, %xmm4 movq %rbx, %xmm4 #speicheradresse des pointers zeigt auf 16*8 feld = 128bit movdqu (%rdi), %xmm1 movdqu (%rdi), %xmm3 #addiere in %xmm1 "min_t-'A'" # = -128 - 65 = 63 = 0x3f mov $0x3f3f3f3f3f3f3f3f, %rbx movq %rbx, %xmm5 pslldq $8, %xmm5 movq %rbx, %xmm5 paddb %xmm5, %xmm1 #"Packed COMpare Greater Than (Byte)" pcmpgtb %xmm2, %xmm1 pminub %xmm4, %xmm1 paddb %xmm1, %xmm3 movq %xmm3, (%rax) ret .LFE2: .size asma, .-asma .section .eh_frame,"a",@progbits .Lframe1: .long .LECIE1-.LSCIE1 .LSCIE1: .long 0x0 .byte 0x1 .string "zR" .uleb128 0x1 .sleb128 -8 .byte 0x10 .uleb128 0x1 .byte 0x3 .byte 0xc .uleb128 0x7 .uleb128 0x8 .byte 0x90 .uleb128 0x1 .align 8 .LECIE1: .LSFDE1: .long .LEFDE1-.LASFDE1 .LASFDE1: .long .LASFDE1-.Lframe1 .long .LFB2 .long .LFE2-.LFB2 .uleb128 0x0 .align 8 .LEFDE1: .ident "GCC: (Debian 4.3.2-1.1) 4.3.2" .section .note.GNU-stack,"",@progbits