/* @(#)root/base:$Id: Byteswap.h 47381 2012-11-16 20:50:59Z rdm $ */ /************************************************************************* * Copyright (C) 1995-2000, Rene Brun and Fons Rademakers. * * All rights reserved. * * * * For the licensing terms see $ROOTSYS/LICENSE. * * For the list of contributors see $ROOTSYS/README/CREDITS. * *************************************************************************/ #ifndef ROOT_Byteswap #define ROOT_Byteswap /* Copyright (C) 1997 Free Software Foundation, Inc. This file is part of the GNU C Library. The GNU C Library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. The GNU C Library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with the GNU C Library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #if (defined(__linux) || defined(__APPLE__)) && \ (defined(__i386__) || defined(__x86_64__)) && \ (defined(__GNUC__) && __GNUC__ >= 2) #ifndef R__USEASMSWAP #define R__USEASMSWAP #endif #endif /* Get the machine specific, optimized definitions. */ /* The following is copied from (only from RH6.0 and above) */ /* Swap bytes in 16 bit value. */ #define R__bswap_constant_16(x) \ ((((x) >> 8) & 0xff) | (((x) & 0xff) << 8)) #if defined R__USEASMSWAP # define R__bswap_16(x) \ (__extension__ \ ({ register unsigned short int __v; \ if (__builtin_constant_p (x)) \ __v = R__bswap_constant_16 (x); \ else \ __asm__ __volatile__ ("rorw $8, %w0" \ : "=r" (__v) \ : "0" ((unsigned short int) (x)) \ : "cc"); \ __v; })) #else /* This is better than nothing. */ # define R__bswap_16(x) R__bswap_constant_16 (x) #endif /* Swap bytes in 32 bit value. */ #define R__bswap_constant_32(x) \ ((((x) & 0xff000000) >> 24) | (((x) & 0x00ff0000) >> 8) | \ (((x) & 0x0000ff00) << 8) | (((x) & 0x000000ff) << 24)) #if defined R__USEASMSWAP /* To swap the bytes in a word the i486 processors and up provide the `bswap' opcode. On i386 we have to use three instructions. */ # if !defined __i486__ && !defined __pentium__ && !defined __pentiumpro__ && \ !defined __pentium4__ && !defined __x86_64__ # define R__bswap_32(x) \ (__extension__ \ ({ register unsigned int __v; \ if (__builtin_constant_p (x)) \ __v = R__bswap_constant_32 (x); \ else \ __asm__ __volatile__ ("rorw $8, %w0;" \ "rorl $16, %0;" \ "rorw $8, %w0" \ : "=r" (__v) \ : "0" ((unsigned int) (x)) \ : "cc"); \ __v; })) # else # define R__bswap_32(x) \ (__extension__ \ ({ register unsigned int __v; \ if (__builtin_constant_p (x)) \ __v = R__bswap_constant_32 (x); \ else \ __asm__ __volatile__ ("bswap %0" \ : "=r" (__v) \ : "0" ((unsigned int) (x))); \ __v; })) # endif #else # define R__bswap_32(x) R__bswap_constant_32 (x) #endif #if defined __GNUC__ && __GNUC__ >= 2 /* Swap bytes in 64 bit value. */ # define R__bswap_64(x) \ (__extension__ \ ({ union { __extension__ unsigned long long int __ll; \ UInt_t __l[2]; } __w, __r; \ __w.__ll = (x); \ __r.__l[0] = R__bswap_32 (__w.__l[1]); \ __r.__l[1] = R__bswap_32 (__w.__l[0]); \ __r.__ll; })) #endif /* bits/byteswap.h */ /* The following definitions must all be macros since otherwise some of the possible optimizations are not possible. */ /* Return a value with all bytes in the 16 bit argument swapped. */ #define Rbswap_16(x) R__bswap_16 (x) /* Return a value with all bytes in the 32 bit argument swapped. */ #define Rbswap_32(x) R__bswap_32 (x) #if defined __GNUC__ && __GNUC__ >= 2 /* Return a value with all bytes in the 64 bit argument swapped. */ # define Rbswap_64(x) R__bswap_64 (x) #endif #endif /* Byteswap.h */