/*
 * Copyright (C) 2013 Synopsys, Inc. (www.synopsys.com)
 *
 * Licensed under the LGPL v2.1 or later, see the file COPYING.LIB in this tarball.
 */

#include <tls.h>
#include <sysdep.h>
#ifndef __ASSEMBLER__
# include <pthreadP.h>
#endif

#if !defined NOT_IN_libc || defined IS_IN_libpthread || defined IS_IN_librt

#ifdef __ASSEMBLER__

#undef ret
#define ret

# undef PSEUDO
# define PSEUDO(name, syscall_name, nargs)			\
   /* vanilla version */				`	\
   ENTRY(name##_nocancel)				`	\
      DO_CALL (__NR_##syscall_name)			`	\
      jls  [blink]					`	\
      b  __syscall_error@plt				`	\
   END(name##_nocancel)					`	\
   /* thread cancellation variant */			`	\
   ENTRY(name)			    			`	\
      SINGLE_THREAD_P					`	\
      bz name##_nocancel				`	\
      DOCARGS_##nargs  /* stash syscall args */		`	\
      CENABLE          /* call enable_asynccancel */	`	\
      mov r9, r0       /* Safe-keep mask */ 		`	\
      UNDOCARGS_##nargs	/* restore syscall args */	`	\
      DO_CALL (__NR_##syscall_name)			`	\
      push  r0         /* save syscall return value */	`	\
      mov   r0, r9     /* prep mask for disable_asynccancel */  `	\
      CDISABLE	`	\
      pop  r0           /* get syscall ret value back */  ` \
      cmp  r0, -1024	`	\
      jls  [blink]					`	\
      b  __syscall_error@plt				`	\
   END(name)

#undef	PSEUDO_END
#define	PSEUDO_END(name)	\

# ifdef IS_IN_libpthread
#  define CENABLE	bl __pthread_enable_asynccancel
#  define CDISABLE	bl __pthread_disable_asynccancel
#  define __local_multiple_threads __pthread_multiple_threads
# elif !defined NOT_IN_libc
#  define CENABLE	bl __libc_enable_asynccancel
#  define CDISABLE	bl __libc_disable_asynccancel
#  define __local_multiple_threads __libc_multiple_threads
# elif defined IS_IN_librt
#  define CENABLE	bl __librt_enable_asynccancel
#  define CDISABLE	bl __librt_disable_asynccancel
# else
#  error Unsupported library
# endif

#define DO_CALL(num)		\
	mov	r8, num		`	\
	ARC_TRAP_INSN		`	\
	cmp r0, -1024

.macro push reg
	st.a \reg, [sp, -4]
.endm

.macro pop reg
	ld.ab \reg, [sp, 4]
.endm

#define DOCARGS_0	push blink
#define UNDOCARGS_0	pop  blink

#define DOCARGS_1	DOCARGS_0`	push r0
#define UNDOCARGS_1			pop  r0`	UNDOCARGS_0

#define DOCARGS_2	DOCARGS_1`	push r1
#define UNDOCARGS_2			pop  r1`	UNDOCARGS_1

#define DOCARGS_3	DOCARGS_2`	push r2
#define UNDOCARGS_3			pop  r2`	UNDOCARGS_2

#define DOCARGS_4	DOCARGS_3`	push r3
#define UNDOCARGS_4			pop  r3`	UNDOCARGS_3

#define DOCARGS_5	DOCARGS_4`	push r4
#define UNDOCARGS_5			pop  r4`	UNDOCARGS_4

#define DOCARGS_6	DOCARGS_5`	push r5
#define UNDOCARGS_6			pop  r5`	UNDOCARGS_5

#define DOCARGS_7	DOCARGS_6`	push r6
#define UNDOCARGS_7			pop  r6`	UNDOCARGS_6

#  define SINGLE_THREAD_P 			\
    THREAD_SELF r1   `				\
    ld	   r2, [r1, MULTIPLE_THREADS_OFFSET]`	\
    cmp    r2, 0

/*    ld	   r2, [r1, -TLS_PRE_TCB_SIZE + MULTIPLE_THREADS_OFFSET] */
#else	/* !__ASSEMBLER__ */

/* TBD: Can use @__local_multiple_threads for libc/libpthread like ARM */
#   define SINGLE_THREAD_P	\
    likely(THREAD_GETMEM (THREAD_SELF, header.multiple_threads) == 0)

#endif /* __ASSEMBLER__ */

#endif