Blame sysdeps/unix/sysv/linux/powerpc/htm.h

Packit 6c4009
/* Shared HTM header.  Emulate transactional execution facility intrinsics for
Packit 6c4009
   compilers and assemblers that do not support the intrinsics and instructions
Packit 6c4009
   yet.
Packit 6c4009
Packit 6c4009
   Copyright (C) 2015-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#ifndef _HTM_H
Packit 6c4009
#define _HTM_H 1
Packit 6c4009
Packit 6c4009
#ifdef __ASSEMBLER__
Packit 6c4009
Packit 6c4009
/* tbegin.  */
Packit 6c4009
.macro TBEGIN
Packit 6c4009
	.long 0x7c00051d
Packit 6c4009
.endm
Packit 6c4009
Packit 6c4009
/* tend. 0  */
Packit 6c4009
.macro TEND
Packit 6c4009
	.long 0x7c00055d
Packit 6c4009
.endm
Packit 6c4009
Packit 6c4009
/* tabort. code  */
Packit 6c4009
.macro TABORT code
Packit 6c4009
	.byte 0x7c
Packit 6c4009
	.byte \code
Packit 6c4009
	.byte 0x07
Packit 6c4009
	.byte 0x1d
Packit 6c4009
.endm
Packit 6c4009
Packit 6c4009
/*"TEXASR - Transaction EXception And Summary Register"
Packit 6c4009
   mfspr %dst,130  */
Packit 6c4009
.macro TEXASR dst
Packit 6c4009
	mfspr \dst,130
Packit 6c4009
.endm
Packit 6c4009
Packit 6c4009
#else
Packit 6c4009
Packit 6c4009
#include <endian.h>
Packit 6c4009
Packit 6c4009
/* Official HTM intrinsics interface matching GCC, but works
Packit 6c4009
   on older GCC compatible compilers and binutils.
Packit 6c4009
   We should somehow detect if the compiler supports it, because
Packit 6c4009
   it may be able to generate slightly better code.  */
Packit 6c4009
Packit 6c4009
#define TBEGIN ".long 0x7c00051d"
Packit 6c4009
#define TEND   ".long 0x7c00055d"
Packit 6c4009
#if __BYTE_ORDER == __LITTLE_ENDIAN
Packit 6c4009
# define TABORT ".byte 0x1d,0x07,%1,0x7c"
Packit 6c4009
#else
Packit 6c4009
# define TABORT ".byte 0x7c,%1,0x07,0x1d"
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#define __force_inline        inline __attribute__((__always_inline__))
Packit 6c4009
Packit 6c4009
#ifndef __HTM__
Packit 6c4009
Packit 6c4009
#define _TEXASRU_EXTRACT_BITS(TEXASR,BITNUM,SIZE) \
Packit 6c4009
  (((TEXASR) >> (31-(BITNUM))) & ((1<<(SIZE))-1))
Packit 6c4009
#define _TEXASRU_FAILURE_PERSISTENT(TEXASRU) \
Packit 6c4009
  _TEXASRU_EXTRACT_BITS(TEXASRU, 7, 1)
Packit 6c4009
Packit 6c4009
#define _tbegin()			\
Packit 6c4009
  ({ unsigned int __ret;		\
Packit 6c4009
     asm volatile (			\
Packit 6c4009
       TBEGIN "\t\n"			\
Packit 6c4009
       "mfcr   %0\t\n"			\
Packit 6c4009
       "rlwinm %0,%0,3,1\t\n"		\
Packit 6c4009
       "xori %0,%0,1\t\n"		\
Packit 6c4009
       : "=r" (__ret) :			\
Packit 6c4009
       : "cr0", "memory");		\
Packit 6c4009
     __ret;				\
Packit 6c4009
  })
Packit 6c4009
Packit 6c4009
#define _tend()				\
Packit 6c4009
  ({ unsigned int __ret;		\
Packit 6c4009
     asm volatile (			\
Packit 6c4009
       TEND "\t\n"			\
Packit 6c4009
       "mfcr   %0\t\n"			\
Packit 6c4009
       "rlwinm %0,%0,3,1\t\n"		\
Packit 6c4009
       "xori %0,%0,1\t\n"		\
Packit 6c4009
       : "=r" (__ret) :			\
Packit 6c4009
       : "cr0", "memory");		\
Packit 6c4009
     __ret;				\
Packit 6c4009
  })
Packit 6c4009
Packit 6c4009
#define _tabort(__code)			\
Packit 6c4009
  ({ unsigned int __ret;		\
Packit 6c4009
     asm volatile (			\
Packit 6c4009
       TABORT "\t\n"			\
Packit 6c4009
       "mfcr   %0\t\n"			\
Packit 6c4009
       "rlwinm %0,%0,3,1\t\n"		\
Packit 6c4009
       "xori %0,%0,1\t\n"		\
Packit 6c4009
       : "=r" (__ret) : "r" (__code)	\
Packit 6c4009
       : "cr0", "memory");		\
Packit 6c4009
     __ret;				\
Packit 6c4009
  })
Packit 6c4009
Packit 6c4009
#define _texasru()			\
Packit 6c4009
  ({ unsigned long __ret;		\
Packit 6c4009
     asm volatile (			\
Packit 6c4009
       "mfspr %0,131\t\n"		\
Packit 6c4009
       : "=r" (__ret));			\
Packit 6c4009
     __ret;				\
Packit 6c4009
  })
Packit 6c4009
Packit 6c4009
#define __libc_tbegin(tdb)       _tbegin ()
Packit 6c4009
#define __libc_tend(nested)      _tend ()
Packit 6c4009
#define __libc_tabort(abortcode) _tabort (abortcode)
Packit 6c4009
#define __builtin_get_texasru()  _texasru ()
Packit 6c4009
Packit 6c4009
#else
Packit 6c4009
# include <htmintrin.h>
Packit 6c4009
Packit 6c4009
# ifdef __TM_FENCE__
Packit 6c4009
   /* New GCC behavior.  */
Packit 6c4009
#  define __libc_tbegin(R)  __builtin_tbegin (R)
Packit 6c4009
#  define __libc_tend(R)    __builtin_tend (R)
Packit 6c4009
#  define __libc_tabort(R)  __builtin_tabort (R)
Packit 6c4009
# else
Packit 6c4009
   /* Workaround an old GCC behavior. Earlier releases of GCC 4.9 and 5.0,
Packit 6c4009
      didn't use to treat __builtin_tbegin, __builtin_tend and
Packit 6c4009
      __builtin_tabort as compiler barriers, moving instructions into and
Packit 6c4009
      out the transaction.
Packit 6c4009
      Remove this when glibc drops support for GCC 5.0.  */
Packit 6c4009
#  define __libc_tbegin(R)			\
Packit 6c4009
   ({ __asm__ volatile("" ::: "memory");	\
Packit 6c4009
     unsigned int __ret = __builtin_tbegin (R);	\
Packit 6c4009
     __asm__ volatile("" ::: "memory");		\
Packit 6c4009
     __ret;					\
Packit 6c4009
   })
Packit 6c4009
#  define __libc_tabort(R)			\
Packit 6c4009
  ({ __asm__ volatile("" ::: "memory");		\
Packit 6c4009
    unsigned int __ret = __builtin_tabort (R);	\
Packit 6c4009
    __asm__ volatile("" ::: "memory");		\
Packit 6c4009
    __ret;					\
Packit 6c4009
  })
Packit 6c4009
#  define __libc_tend(R)			\
Packit 6c4009
   ({ __asm__ volatile("" ::: "memory");	\
Packit 6c4009
     unsigned int __ret = __builtin_tend (R);	\
Packit 6c4009
     __asm__ volatile("" ::: "memory");		\
Packit 6c4009
     __ret;					\
Packit 6c4009
   })
Packit 6c4009
# endif /* __TM_FENCE__  */
Packit 6c4009
#endif /* __HTM__  */
Packit 6c4009
Packit 6c4009
#endif /* __ASSEMBLER__ */
Packit 6c4009
Packit 6c4009
/* Definitions used for TEXASR Failure code (bits 0:7).  If the failure
Packit 6c4009
   should be persistent, the abort code must be odd.  0xd0 through 0xff
Packit 6c4009
   are reserved for the kernel and potential hypervisor.  */
Packit 6c4009
#define _ABORT_PERSISTENT      0x01   /* An unspecified persistent abort.  */
Packit 6c4009
#define _ABORT_LOCK_BUSY       0x34   /* Busy lock, not persistent.  */
Packit 6c4009
#define _ABORT_NESTED_TRYLOCK  (0x32 | _ABORT_PERSISTENT)
Packit 6c4009
#define _ABORT_SYSCALL         (0x30 | _ABORT_PERSISTENT)
Packit 6c4009
Packit 6c4009
#endif