#include <stdio.h> #include <stdlib.h> #include <assert.h> typedef unsigned int UInt; /* Given a word, do bt/bts/btr/btc on bits 0, 1, 2 and 3 of it, and also reconstruct the original bits 0, 1, 2, 3 by looking at the carry flag. Returned result has mashed bits 0-3 at the bottom and the reconstructed original bits 0-3 as 4-7. */ UInt mash_reg_L ( UInt orig ) { UInt reconstructed, mashed; __asm__ __volatile__ ( "movl %2, %%edx\n\t" "" "movl $0, %%eax\n\t" "\n\t" "btl $0, %%edx\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "btsl $1, %%edx\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "shll $1, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "btrl $2, %%edx\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "shll $2, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "btcl $3, %%edx\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "shll $3, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "movl %%eax, %0\n\t" "movl %%edx, %1" : "=r" (reconstructed), "=r" (mashed) : "r" (orig) : "eax", "ecx", "edx", "cc"); return (mashed & 0xF) | ((reconstructed & 0xF) << 4); } UInt mash_mem_L ( UInt* origp ) { UInt reconstructed, mashed; __asm__ __volatile__ ( "movl %2, %%edx\n\t" "" "movl $0, %%eax\n\t" "\n\t" "btl $0, (%%edx)\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "btsl $1, (%%edx)\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "shll $1, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "btrl $2, (%%edx)\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "shll $2, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "btcl $3, (%%edx)\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "shll $3, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "movl %%eax, %0\n\t" "movl (%%edx), %1" : "=r" (reconstructed), "=r" (mashed) : "r" (origp) : "eax", "ecx", "edx", "cc"); return (mashed & 0xF) | ((reconstructed & 0xF) << 4); } UInt mash_reg_W ( UInt orig ) { UInt reconstructed, mashed; __asm__ __volatile__ ( "movl %2, %%edx\n\t" "" "movl $0, %%eax\n\t" "\n\t" "btw $0, %%dx\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "btsw $1, %%dx\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "shll $1, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "btrw $2, %%dx\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "shll $2, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "btcw $3, %%dx\n\t" "setb %%cl\n\t" "movzbl %%cl, %%ecx\n\t" "shll $3, %%ecx\n\t" "orl %%ecx, %%eax\n\t" "\n\t" "movl %%eax, %0\n\t" "movl %%edx, %1" : "=r" (reconstructed), "=r" (mashed) : "r" (orig) : "eax", "ecx", "edx", "cc"); return (mashed & 0xF) | ((reconstructed & 0xF) << 4); } int main ( void ) { int i, ii; for (i = 0; i < 0x10; i++) { ii = i; printf("0x%x -> 0x%2x 0x%2x 0x%2x\n", i, mash_reg_L(i), mash_mem_L(&ii), mash_reg_W(i)); } return 1; }