// from http://www.azillionmonkeys.com/qed/poker.zip // shamelessly copied from azillionmonkeys. getting millions of evals per second! #include <stdio.h> #include <stdlib.h> #include <string.h> #include <stdint.h> #include <time.h> #define CLUB_SUIT (1) #define DIAMOND_SUIT (2) #define HEART_SUIT (4) #define SPADE_SUIT (8) #define RANK_SHL (27) #define SUBR_SHL (13) #define SUBR_SHLMASK ((1<<SUBR_SHL)-1) #define STRAIGHT_FLUSH_SCORE (8 << RANK_SHL) #define FOUR_KIND_SCORE (7 << RANK_SHL) #define FULL_HOUSE_SCORE (6 << RANK_SHL) #define FLUSH_SCORE (5 << RANK_SHL) #define STRAIGHT_SCORE (4 << RANK_SHL) #define THREE_KIND_SCORE (3 << RANK_SHL) #define TWO_PAIR_SCORE (2 << RANK_SHL) #define TWO_KIND_SCORE (1 << RANK_SHL) #define ONE_PAIR_SCORE (TWO_KIND_SCORE) typedef uint8_t u8; typedef int8_t s8; typedef uint32_t u32; typedef int32_t s32; typedef struct { int32_t len; u8 entry[52]; } CardPileType; static char *handstrs[16] = { "high card", "one pair", "two pair", "three of a kind", "straight", "flush", "full house", "four of a kind", "straight flush", "err", "err", "err", "err", "err", "err", "err" }; static char *kickerstrs[16] = { "", "kickers", "kicker", "kickers", "high", "high", "full of", "kicker", "high", "err", "err", "err", "err", "err", "err", "err" }; static u32 CardValue[52] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12 }; static u32 CardSuit[52] = { 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8 }; int32_t cardstr(char *cardstr,uint8_t card) { int32_t suit; char *cardc = "A234567890JQK",suitc[4] = { 'C', 'D', 'H', 'S' }; suit = card / 13; card %= 13; if ( card == 9 ) sprintf(cardstr,"10%c",suitc[suit]); else sprintf(cardstr,"%c%c",cardc[card],suitc[suit]); return(card); } static u32 CardSuitIdx[52] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }; static u32 CardMask[52] = { 1, 2, 4, 8, 0x10, 0x20, 0x40, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000, 1, 2, 4, 8, 0x10, 0x20, 0x40, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000, 1, 2, 4, 8, 0x10, 0x20, 0x40, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000, 1, 2, 4, 8, 0x10, 0x20, 0x40, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000 }; static u32 FiveCardDrawScoreFast (u32 c0, u32 c1, u32 c2, u32 c3, u32 c4, u32 u) { u32 y,z,m1,m2,m3,m4; // Test for single suitedness u = u & (u - 1); // Build masks of 1, 2, 3, and 4 of a kind. m1 = c0 | c1; m2 = c1 & c0; m2 |= c2 & m1; m1 |= c2; m2 |= c3 & m1; m1 |= c3; m2 |= c4 & m1; m1 |= c4; if (m2 == 0) { // No pairs? // Is the mask a sequence of 1 bits? z = m1 & (m1 - 1); z ^= m1; y = (z << 5) - z; // Deal with the bicycle/wheel 5,4,3,2,Ace straight if (m1 == 0x100F) { if (u != 0) return STRAIGHT_SCORE + 0xF; return STRAIGHT_FLUSH_SCORE + 0xF; } if (y == m1) { if (u != 0) return STRAIGHT_SCORE + m1; return STRAIGHT_FLUSH_SCORE + m1; } if (u != 0) return m1; // Nothing return FLUSH_SCORE + m1; } /* // m1 = c0 | ... | c4 // m2 = (c0 & c1) | ((c0|c1) & c2) | ((c0|c1|c2) & c3) | ((c0|c1|c2|c3) & c4) // m3 = mask of 3 or 4 of a kind. // m4 = mask of 4 of a kind. */ m1 = c0 | c1; m2 = c1 & c0; m3 = c2 & m2; m2 |= c2 & m1; m1 |= c2; m4 = c3 & m3; m3 |= c3 & m2; m2 |= c3 & m1; m1 |= c3; m4 |= c4 & m3; m3 |= c4 & m2; m2 |= c4 & m1; m1 |= c4; m1 &= ~m2; if (m3 == 0) { if ((m2 & (m2 - 1)) == 0) return TWO_KIND_SCORE + (m2 << SUBR_SHL) + m1; return TWO_PAIR_SCORE + (m2 << SUBR_SHL) + m1; } m2 &= ~m3; if (m4 == 0) { if (m2 == 0) return THREE_KIND_SCORE + (m3 << SUBR_SHL) + m1; return FULL_HOUSE_SCORE + (m3 << SUBR_SHL) + m2; } return FOUR_KIND_SCORE + (m4 << SUBR_SHL) + m1; } static u32 FiveCardDrawScore (const u8 * h) { u32 c0, c1, c2, c3, c4,u; // Make suits powers of two. u = CardSuit[h[0]]; u |= CardSuit[h[1]]; u |= CardSuit[h[2]]; u |= CardSuit[h[3]]; u |= CardSuit[h[4]]; // Make cards powers of two. c0 = CardMask[h[0]]; c1 = CardMask[h[1]]; c2 = CardMask[h[2]]; c3 = CardMask[h[3]]; c4 = CardMask[h[4]]; return FiveCardDrawScoreFast (c0, c1, c2, c3, c4, u); } static u32 SevenCardDrawFlush (const u8 * h, const u32 c[7]) { static s32 LG2TAB[9] = {-1, 0, 1, -1, 2, -1, -1, -1, 3}; u32 i,t,f[4]; s32 cc[4],s; // Make suits powers of two. // There is at most one flush in a 7 card hand. So up to 4 masks arecreated, each being the set of cards in one particular suit. // As themasks are created a maximum countdown from 4 is tracked. #if 0 s = (((((u32 *) h)[1])) * 5) & 0x00C0C0C0; i = (((((u32 *) h)[0])) * 5) & 0xC0C0C0C0; s = ((s * (1 + 64 + 4096)) >> 8) & 0x3F00; i = ((i * (1 + 64 + 4096 + 262144)) >> 24) | s; #endif f[0] = f[1] = f[2] = f[3] = 0; cc[0] = cc[1] = cc[2] = cc[3] = 4; i = CardSuitIdx[h[0]]; f[i] = c[0]; cc[i]--; i = CardSuitIdx[h[1]]; f[i] |= c[1]; cc[i]--; i = CardSuitIdx[h[2]]; f[i] |= c[2]; cc[i]--; i = CardSuitIdx[h[3]]; f[i] |= c[3]; cc[i]--; i = CardSuitIdx[h[4]]; f[i] |= c[4]; cc[i]--; i = CardSuitIdx[h[5]]; f[i] |= c[5]; cc[i]--; i = CardSuitIdx[h[6]]; f[i] |= c[6]; cc[i]--; // If any of the counts goes below 0, then that suit has at least 5 cards in it. if (((cc[0] | cc[1]) | (cc[2] | cc[3])) < 0) { // Map the negative count (of which there is at most 1) to the suit index. s = (((cc[0] & 8) | (cc[1] & 16)) | ((cc[2] & 32) | (cc[3] & 64))) >> 3; s = LG2TAB[s]; // Work out the top 5 cards, so that the proper flush is also known ... t = f[s]; while (cc[s] < -1) { t &= t - 1; // Rip off last bit. cc[s]++; } // ... but also keep the low part so that all straight flushes can still be tested for. return f[s] | (t << SUBR_SHL); } // 0 -> indicates there is no flush in the hand. return 0; } #define SevenCardDrawFlushScore(f) (FLUSH_SCORE + ((f) >> SUBR_SHL)) u32 SevenCardDrawScore (const u8 * h) { u32 c[7],f, t, s, u, v,m1, m2, m3, m4; c[0] = CardMask[h[0]]; c[1] = CardMask[h[1]]; c[2] = CardMask[h[2]]; c[3] = CardMask[h[3]]; c[4] = CardMask[h[4]]; c[5] = CardMask[h[5]]; c[6] = CardMask[h[6]]; // Work out flush mask always, since its only redundant in the case of a full house or 4 of a kind. f = SevenCardDrawFlush (h, c); // t is the full card mask t = ((c[0] | c[1]) | (c[2] | c[3])) | (c[4] | c[5] | c[6]); // If 7 cards are found in a row ... s = t & (t - 1); v = s ^ t; u = (v << 7) - 1; u &= ~(u >> 7); // ... then work out the possible straight flush/bicycle, straight, and flush possibilities. if ((u & t) == u) { if (f) { // Intersect the flush mask with the straight mask to try to figure out if we have a straight flush. s = f & u; u = s & (s - 1); u ^= s; u = (u << 7) - 1; u &= ~(u >> 5); t = u >> 1; v = u >> 2; if ((u & s) == u) return STRAIGHT_FLUSH_SCORE + u; if ((t & s) == t) return STRAIGHT_FLUSH_SCORE + t; if ((v & s) == v) return STRAIGHT_FLUSH_SCORE + v; // Deal with possible bicycle/wheel 5,4,3,2,Ace straight flush if ((0x100F & f) == 0x100F) return STRAIGHT_FLUSH_SCORE + 0xF; return SevenCardDrawFlushScore (f); } u &= ~(u >> 5); return STRAIGHT_SCORE + u; } // If at most 6 cards are found in a row ... u = (v << 6) - 1; u &= ~(u >> 6); if ((u & t) != u) { u = s & (s - 1); u ^= s; u = (u << 6) - 1; u &= ~(u >> 6); } // ... then work out the possible straight flush/bicycle, straight, and flush possibilities. if ((u & t) == u) { if (f) { s = f & u; u = s & (s - 1); u ^= s; u = (u << 6) - 1; u &= ~(u >> 5); t = u >> 1; if ((u & s) == u) return STRAIGHT_FLUSH_SCORE + u; if ((t & s) == t) return STRAIGHT_FLUSH_SCORE + t; // Deal with possible bicycle/wheel 5,4,3,2,Ace straight flush if ((0x100F & f) == 0x100F) return STRAIGHT_FLUSH_SCORE + 0xF; return SevenCardDrawFlushScore (f); } u &= ~(u >> 5); return STRAIGHT_SCORE + u; } // If at most 5 cards are found in a row ... u = (v << 5) - 1; u &= ~(u >> 5); if ((u & t) != u) { v = s; s = s & (s - 1); v ^= s; u = (v << 5) - 1; u &= ~(u >> 5); if ((u & t) != u) { u = s & (s - 1); s ^= u; u = (s << 5) - 1; u &= ~(u >> 5); } } // ... then work out the possible straight flush/bicycle, straight, and flush possibilities. if ((u & t) == u) { if (f) { s = f & u; u = s & (s - 1); u ^= s; u = (u << 5) - 1; u &= ~(u >> 5); if ((u & s) == u) { return STRAIGHT_FLUSH_SCORE + u; } // Deal with possible bicycle/wheel 5,4,3,2,Ace straight flush if ((0x100F & f) == 0x100F) return STRAIGHT_FLUSH_SCORE + 0xF; return SevenCardDrawFlushScore (f); } u &= ~(u >> 5); return STRAIGHT_SCORE + u; } // Deal with the bicycle/wheel 5,4,3,2,Ace straight if ((0x100F & t) == 0x100F) { if (f) { if ((0x100F & f) == 0x100F) return STRAIGHT_FLUSH_SCORE + 0xF; return SevenCardDrawFlushScore (f); } return STRAIGHT_SCORE + 0xF; } /* // m1 = c0 | ... | c4 // m2 = (c0 & c1) | ((c0|c1) & c2) | ((c0|c1|c2) & c3) | ((c0|c1|c2|c3) & c4) // m3 = mask of 3 or 4 of a kind. // m4 = mask of 4 of a kind. */ m1 = c[0] | c[1]; m2 = c[1] & c[0]; m3 = c[2] & m2; m2 |= c[2] & m1; m1 |= c[2]; m4 = c[3] & m3; m3 |= c[3] & m2; m2 |= c[3] & m1; m1 |= c[3]; m4 |= c[4] & m3; m3 |= c[4] & m2; m2 |= c[4] & m1; m1 |= c[4]; m4 |= c[5] & m3; m3 |= c[5] & m2; m2 |= c[5] & m1; m1 |= c[5]; m4 |= c[6] & m3; m3 |= c[6] & m2; m2 |= c[6] & m1; m1 |= c[6]; // Make sure the m1 is just the mask of singleton cards. m1 &= ~m2; // No 3 or 4 of a kinds. if (m3 == 0) { if (f) { return SevenCardDrawFlushScore (f); } if (m2) { s = m2 & (m2 - 1); if (s == 0) { t = m1 & (m1 - 1); t &= (t - 1); return TWO_KIND_SCORE + (m2 << SUBR_SHL) + t; } v = m1; t = s & (s - 1); if (t) { v |= m2 ^ s; m2 = s; } else { v &= (v - 1); } v &= (v - 1); return TWO_PAIR_SCORE + (m2 << SUBR_SHL) + v; } // Nothing -- just remove the two low cards. m1 &= (m1 - 1); m1 &= (m1 - 1); return m1; } // 4 of a kind. if (m4) { m1 |= (m2 & ~m4); while ((m2 = (m1 & (m1 - 1))) != 0) { m1 = m2; } return FOUR_KIND_SCORE + (m4 << SUBR_SHL) + m1; } // 3 of a kind, but no other pair in the hand if ((m2 & ~m3) == 0) { t = m3 & (m3 - 1); // Two 3 of a kinds => Full House if (t) return FULL_HOUSE_SCORE + (t << SUBR_SHL) + (m3 ^ t); if (f) return SevenCardDrawFlushScore (f); // Just remove the two low cards, and score this as 3 of a kind. m1 &= (m1 - 1); m1 &= (m1 - 1); return THREE_KIND_SCORE + (m3 << SUBR_SHL) + m1; } // 3 of a kind and a seperate 2 of a kind (full house.) m2 &= ~m3; t = m2 & (m2 - 1); // If there are two 2 pairs in the hand, then pick the higher pair if (!t) t = m2; return FULL_HOUSE_SCORE + (m3 << SUBR_SHL) + t; } // Just a brute force 7 card draw ranking by picking each of the possible two cards to be skipped, and finding the score which is highest amongst the 21 possible 5 card sub-hands. u32 SevenCardDrawScoreSlow (const u8 * h) { s32 i, j, k; u32 r, m; u8 h2[5]; m = 0; for (i=0; i < 6; i++) { for (k=0; k < i; k++) { h2[k] = h[k]; } for (j=i+1; j < 7; j++) { for (k=i+1; k < j; k++) { h2[k-1] = h[k]; } for (k=j+1; k < 7; k++) { h2[k-2] = h[k]; } r = FiveCardDrawScore (h2); if (r > m) m = r; } } return m; } // A correctly distributed (ignoring the problem of using "rand() % x") reshuffling of your cards. static void Shuffle (CardPileType * c) { int32_t i, j, k; u8 t; for (i = 0; i < (c->len - 1); i++) { k = (RAND_MAX / (c->len - i)) * (c->len - i); do { j = rand (); } while (j >= k); j = i + (j % (c->len - i)); t = c->entry[i]; c->entry[i] = c->entry[j]; c->entry[j] = t; } } static int32_t Deal (CardPileType * h, CardPileType * d, int32_t n) { int32_t i; for (i=0; i < n && d->len > 0; i++) { d->len--; h->entry[ h->len ] = d->entry[ d->len ]; h->len++; } return i; } static void InitDeck (CardPileType * deck) { u8 i; deck->len = 52; for (i=0; i < 52; i++) deck->entry[i] = i; } static void DisplayCard (u8 c, char out[]) { // char suitdisp[9] = { 0, 5, 4, 0, 3, 0, 0, 0, 6 }; static char suitdisp[9] = { 0, 'c', 'd', 0, 'h', 0, 0, 0, 's' }; char s[4]; s[0] = " 1 "[CardValue[c]]; s[1] = "234567890JQKA"[CardValue[c]]; s[2] = suitdisp[CardSuit[c]]; s[3] = '\0'; strcat (out, " "); strcat (out, s); } void DisplayHand5 (const CardPileType * h) { char out[128]; int i; out[0] = '\0'; for (i=0; i < 5; i++) DisplayCard (h->entry[i], out); sprintf (out + strlen (out), " => %08X\n", (int)FiveCardDrawScore (&h->entry[0])); printf ("%s", out); } void set_cardstr(char *cardstr,uint32_t c) { /*static char suitdisp[9] = { 0, 'c', 'd', 0, 'h', 0, 0, 0, 's' }; cardstr[0] = " 1 "[CardValue[c % 13]]; cardstr[1] = "234567890JQKA"[CardValue[c % 13]]; cardstr[2] = suitdisp[CardSuit[c]]; cardstr[3] = '\0';*/ int32_t i,j=0; c >>= 1; for (i=12; i>=0; i--) if ( ((1 << i) & c) != 0 ) { cardstr[j++] = " 1 "[i]; cardstr[j++] = "234567890JQKA"[i]; cardstr[j++] = ' '; } cardstr[j++] = 0; } uint32_t set_handstr(char *handstr,uint8_t cards[7],int32_t verbose) { char cardstr[32],cardstr2[32],*kickerstr,*str; uint32_t score,i; handstr[0] = 0; if ( cards == 0 ) { handstr[0] = 0; printf("set_handstr: null cards??\n"); return(0); } for (i=0; i<7; i++) if ( cards[i] < 0 || cards[i] >= 52 ) { //printf("illegal card[%d] %d\n",i,cards[i]); return(0); } score = SevenCardDrawScore (&cards[0]); set_cardstr(cardstr,(score>>SUBR_SHL) & SUBR_SHLMASK); set_cardstr(cardstr2,score & SUBR_SHLMASK); kickerstr = kickerstrs[(score>>RANK_SHL)&15]; str = handstrs[(score>>RANK_SHL)&15]; if ( verbose != 0 ) { if ( strcmp(kickerstr,"high") == 0 ) sprintf(handstr,"%c%c high %s",cardstr2[0],cardstr2[1],str); else if ( strcmp(str,"full house") == 0 ) sprintf(handstr,"%c%c full of %c%c",cardstr[0],cardstr[1],cardstr2[0],cardstr2[1]); else if ( strcmp(str,"three of a kind") == 0 ) sprintf(handstr,"set of %c%c with kickers %c%c %c%c",cardstr[0],cardstr[1],cardstr2[0],cardstr2[1],cardstr2[3],cardstr2[4]); else if ( strcmp(str,"two pair") == 0 ) sprintf(handstr,"two pair %c%c and %c%c with %c%c kicker",cardstr[0],cardstr[1],cardstr[3],cardstr[4],cardstr2[0],cardstr2[1]); else if ( strcmp(str,"one pair") == 0 ) sprintf(handstr,"pair of %c%c with %c%c kicker",cardstr[0],cardstr[1],cardstr2[0],cardstr2[1]); else sprintf(handstr,"%s %s %s %s",str,cardstr,kickerstr,cardstr2); } else { if ( strcmp(kickerstr,"high") == 0 ) sprintf(handstr,"%c%c high %s",cardstr2[0],cardstr2[1],str); else if ( strcmp(str,"full house") == 0 ) sprintf(handstr,"fullhouse %c%c %c%c",cardstr[0],cardstr[1],cardstr2[0],cardstr2[1]); else if ( strcmp(str,"three of a kind") == 0 ) sprintf(handstr,"trip %c%c",cardstr[0],cardstr[1]); else if ( strcmp(str,"two pair") == 0 ) sprintf(handstr,"two pairs %c%c %c%c",cardstr[0],cardstr[1],cardstr[3],cardstr[4]); else if ( strcmp(str,"one pair") == 0 ) sprintf(handstr,"pair %c%c",cardstr[0],cardstr[1]); else sprintf(handstr,"%s",cardstr2); } return(score); } void DisplayHand7(char *handstr,uint8_t *cards) { uint32_t x,y; int32_t i; char out[512]; out[0] = '\0'; for (i=0; i<7; i++) DisplayCard (cards[i], out); x = SevenCardDrawScore (cards); y = SevenCardDrawScoreSlow (cards); set_handstr(handstr,cards,1); if ( x != y ) fprintf(stderr,"Error slow score %08x vs fast score %08x???\n",y,x); sprintf (out + strlen (out), " => %08x %6d %6d -> (%s)",x,(x>>SUBR_SHL)&SUBR_SHLMASK,x&SUBR_SHLMASK,handstr); printf("%s\n",out); } void poker_test() { char *_mbstr(double n); CardPileType Deck,Hands[1000]; uint32_t c,starttime,score; uint64_t total,counter; char handstr[512]; uint8_t *cards; //srand (0); InitDeck (&Deck); Shuffle (&Deck); for (c=0; c<(sizeof(Hands)/sizeof(*Hands)); c++) { Deal (&Hands[c],&Deck,7); DisplayHand7(handstr,Hands[c].entry); Deal (&Deck,&Hands[c],7); Shuffle(&Deck); } starttime = (uint32_t)time(NULL); #ifndef _WIN32 //while ( (uint32_t)time(NULL) == starttime ) // usleep(100); total = counter = 0; while ( (uint32_t)time(NULL) < starttime+11 ) { for (c=0; c<1000; c++,counter++) { cards = Hands[c % (sizeof(Hands)/sizeof(*Hands))].entry; score = SevenCardDrawScore(cards); total += score; } } char *mbstr(char *str,double val); char str[65]; printf("counter.%llu %s in 10 seconds: ave score %llx\n",(long long)counter,mbstr(str,counter),(long long)(total/counter)); #endif }