]> git.ipfire.org Git - thirdparty/glibc.git/commitdiff
Add emulation of popcntb.
authorAndreas Schwab <schwab@redhat.com>
Tue, 25 Aug 2009 16:31:56 +0000 (18:31 +0200)
committerAndreas Schwab <schwab@redhat.com>
Tue, 25 Aug 2009 17:31:17 +0000 (19:31 +0200)
fedora/power6emul.c

index f1d0d20e0f6f378c415069bccb7fa35916c7f331..1b0187bddd961ab7a5250184f739af3588a12b7f 100644 (file)
@@ -163,6 +163,25 @@ asm (".globl frip, friz, frin, frim\n.hidden frip, friz, frin, frim\n\t"
 "3:"   "mtfsf   0x01,11\n\t"
        "blr\n");
 
+#ifdef __powerpc64__
+#define m1 0x5555555555555555L
+#define m2 0x3333333333333333L
+#define m3 0x0f0f0f0f0f0f0f0fL
+#else
+#define m1 0x55555555
+#define m2 0x33333333
+#define m3 0x0f0f0f0f
+#endif
+
+static inline unsigned long
+popcntb (unsigned long n)
+{
+  n -= (n >> 1) & m1;
+  n = (n & m2) + ((n >> 2) & m2);
+  n = (n + (n >> 4)) & m3;
+  return n;
+}
+
 static void
 catch_sigill (int signal, struct sigcontext *ctx)
 {
@@ -221,6 +240,18 @@ catch_sigill (int signal, struct sigcontext *ctx)
       ctx->regs->nip += 4;
       return;
     }
+  if ((insn & 0xfc00ffff) == 0x7c0000f4) /* popcntb */
+    {
+      unsigned long *regs = (unsigned long *) ctx->regs;
+      unsigned dest = (insn >> 16) & 0x1f;
+      unsigned src = (insn >> 21) & 0x1f;
+      unsigned long res = 0;
+      int i;
+
+      regs[dest] = popcntb (regs[src]);
+      ctx->regs->nip += 4;
+      return;
+    }
 
   struct sigaction sa;
   sa.sa_handler = SIG_DFL;