38 lines
1.9 KiB
C
38 lines
1.9 KiB
C
#ifndef COSMOPOLITAN_LIBC_INTRIN_PSRLDQ_H_
|
|
#define COSMOPOLITAN_LIBC_INTRIN_PSRLDQ_H_
|
|
#include "libc/intrin/macros.h"
|
|
#if !(__ASSEMBLER__ + __LINKER__ + 0)
|
|
COSMOPOLITAN_C_START_
|
|
|
|
void psrldq(uint8_t[16], const uint8_t[16], unsigned long);
|
|
|
|
#ifndef __STRICT_ANSI__
|
|
__intrin_xmm_t __psrldqs(__intrin_xmm_t);
|
|
#define psrldq(B, A, I) \
|
|
do { \
|
|
if (likely(!IsModeDbg() && X86_NEED(SSE) && X86_HAVE(SSE2))) { \
|
|
__intrin_xmm_t *Xmm0 = (void *)(B); \
|
|
const __intrin_xmm_t *Xmm1 = (const __intrin_xmm_t *)(A); \
|
|
if (isconstant(I)) { \
|
|
if (!X86_NEED(AVX)) { \
|
|
asm("psrldq\t%1,%0" : "=x"(*Xmm0) : "i"(I), "0"(*Xmm1)); \
|
|
} else { \
|
|
asm("vpsrldq\t%2,%1,%0" : "=x"(*Xmm0) : "x"(*Xmm1), "i"(I)); \
|
|
} \
|
|
} else { \
|
|
unsigned long Vimm = (I); \
|
|
typeof(__psrldqs) *Fn; \
|
|
if (Vimm > 16) Vimm = 16; \
|
|
Fn = (typeof(__psrldqs) *)((uintptr_t)&__psrldqs + Vimm * 8); \
|
|
*Xmm0 = Fn(*Xmm1); \
|
|
} \
|
|
} else { \
|
|
psrldq(B, A, I); \
|
|
} \
|
|
} while (0)
|
|
#endif
|
|
|
|
COSMOPOLITAN_C_END_
|
|
#endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */
|
|
#endif /* COSMOPOLITAN_LIBC_INTRIN_PSRLDQ_H_ */
|