405 lines
16 KiB
405 lines
16 KiB
/******************************************************************************
|
|
* Copyright © 2014-2016 The SuperNET Developers. *
|
|
* *
|
|
* See the AUTHORS, DEVELOPER-AGREEMENT and LICENSE files at *
|
|
* the top-level directory of this distribution for the individual copyright *
|
|
* holder information and the developer policies on copyright and licensing. *
|
|
* *
|
|
* Unless otherwise agreed in a custom licensing agreement, no part of the *
|
|
* SuperNET software, including this file may be copied, modified, propagated *
|
|
* or distributed except according to the terms contained in the LICENSE file *
|
|
* *
|
|
* Removal or modification of this copyright notice is prohibited. *
|
|
* *
|
|
******************************************************************************/
|
|
|
|
#include "pangea777.h"
|
|
|
|
int64_t pangea_chipsvalue(struct supernet_info *myinfo,struct table_info *tp,int32_t numchips)
|
|
{
|
|
return((tp->G.bigblind >> 1) * numchips);
|
|
}
|
|
|
|
void pangea_playerfold(struct supernet_info *myinfo,struct table_info *tp,struct player_info *player)
|
|
{
|
|
uint8_t tmp;
|
|
//printf("player.%d folded\n",player); //getchar();
|
|
tp->hand.handmask |= (1 << player->ind);
|
|
player->betstatus = CARDS777_FOLD;
|
|
player->action = CARDS777_FOLD;
|
|
tmp = player->ind;
|
|
pangea_summaryadd(myinfo,tp,CARDS777_FOLD,&tmp,sizeof(tmp),(void *)&player->bets,sizeof(player->bets));
|
|
}
|
|
|
|
int32_t pangea_playerbet(struct supernet_info *myinfo,int64_t *actualbetp,struct table_info *tp,struct player_info *player,int64_t bet,int32_t action)
|
|
{
|
|
uint64_t sum; uint8_t tmp; struct hand_info *hand = &tp->hand;
|
|
if ( bet == 0 ) // autobet
|
|
{
|
|
if ( action == CARDS777_CALL )
|
|
bet = hand->betsize;
|
|
else if ( action == CARDS777_BET )
|
|
bet = hand->betsize + hand->lastraise;
|
|
else if ( action == CARDS777_RAISE )
|
|
bet = hand->betsize + hand->lastraise * 2;
|
|
bet -= player->bets;
|
|
}
|
|
*actualbetp = 0;
|
|
if ( Debuglevel > 2 )
|
|
printf("PANGEA_BET[%d] <- %.8f\n",player->ind,dstr(bet));
|
|
if ( player->betstatus == CARDS777_ALLIN )
|
|
return(CARDS777_ALLIN);
|
|
else if ( player->betstatus == CARDS777_FOLD )
|
|
return(CARDS777_FOLD);
|
|
else if ( action == CARDS777_ALLIN )
|
|
bet = player->balance;
|
|
if ( bet > 0 && bet >= player->balance )
|
|
{
|
|
bet = player->balance;
|
|
player->betstatus = action = CARDS777_ALLIN;
|
|
}
|
|
else
|
|
{
|
|
if ( bet > hand->betsize && bet > hand->lastraise && bet < (hand->lastraise<<1) )
|
|
{
|
|
printf("pangea_bet %.8f not double %.8f, clip to lastraise\n",dstr(bet),dstr(hand->lastraise));
|
|
bet = hand->lastraise;
|
|
action = CARDS777_RAISE;
|
|
}
|
|
}
|
|
sum = player->bets;
|
|
if ( sum+bet < hand->betsize && action != CARDS777_ALLIN )
|
|
{
|
|
pangea_playerfold(myinfo,tp,player);
|
|
action = CARDS777_FOLD;
|
|
if ( Debuglevel > 2 )
|
|
printf("player.%d betsize %.8f < hand.betsize %.8f FOLD\n",player->ind,dstr(bet),dstr(hand->betsize));
|
|
return(action);
|
|
}
|
|
else if ( bet >= 2*hand->lastraise )
|
|
{
|
|
hand->lastraise = bet;
|
|
hand->numactions = 0;
|
|
if ( action == CARDS777_CHECK )
|
|
{
|
|
action = CARDS777_FULLRAISE; // allows all players to check/bet again
|
|
if ( Debuglevel > 2 )
|
|
printf("FULLRAISE by player.%d\n",player->ind);
|
|
}
|
|
}
|
|
sum += bet;
|
|
if ( sum > hand->betsize )
|
|
{
|
|
hand->numactions = 0;
|
|
hand->betsize = sum, hand->lastbettor = player->ind;
|
|
if ( sum > hand->lastraise && action == CARDS777_ALLIN )
|
|
hand->lastraise = sum;
|
|
else if ( action == CARDS777_CHECK )
|
|
action = CARDS777_BET;
|
|
}
|
|
if ( bet > 0 && action == CARDS777_CHECK )
|
|
action = CARDS777_CALL;
|
|
tmp = player->ind;
|
|
pangea_summaryadd(myinfo,tp,action,&tmp,sizeof(tmp),(void *)&bet,sizeof(bet));
|
|
player->balance -= bet, player->bets += bet;
|
|
//if ( Debuglevel > 2 )
|
|
printf("player.%d: player.%d BET %f -> balances %f bets %f\n",tp->priv.myind,player->ind,dstr(bet),dstr(player->balance),dstr(player->bets));
|
|
*actualbetp = bet;
|
|
return(action);
|
|
}
|
|
|
|
void pangea_antes(struct supernet_info *myinfo,struct table_info *tp)
|
|
{
|
|
int32_t i,n,N; struct player_info *p; uint64_t threshold; int32_t handid; int64_t actualbet;
|
|
N = tp->G.numactive;
|
|
for (i=0; i<tp->G.N; i++)
|
|
{
|
|
tp->G.P[i].ind = i;
|
|
if ( (tp->snapshot[i]= tp->G.P[i].balance) <= 0 )
|
|
pangea_playerfold(myinfo,tp,&tp->G.P[i]);
|
|
}
|
|
handid = tp->numhands - 1;
|
|
pangea_summaryadd(myinfo,tp,CARDS777_SNAPSHOT,(void *)&handid,sizeof(handid),(void *)tp->snapshot,sizeof(uint64_t)*CARDS777_MAXPLAYERS);
|
|
if ( tp->G.ante != 0 )
|
|
{
|
|
for (i=0; i<N; i++)
|
|
{
|
|
if ( (p= tp->active[i]) != 0 )
|
|
{
|
|
if ( p->balance < tp->G.ante )
|
|
pangea_playerfold(myinfo,tp,p);
|
|
else pangea_playerbet(myinfo,&actualbet,tp,p,tp->G.ante,CARDS777_ANTE);
|
|
} else printf("unexpected null player ptr\n");
|
|
}
|
|
}
|
|
for (i=n=0; i<N; i++)
|
|
{
|
|
if ( i == 0 )
|
|
threshold = (tp->G.bigblind >> 1) - 1;
|
|
else if ( i == 1 )
|
|
threshold = tp->G.bigblind - 1;
|
|
else threshold = 0;
|
|
if ( (p= tp->active[i]) != 0 && p->balance < threshold )
|
|
pangea_playerfold(myinfo,tp,p);
|
|
else n++;
|
|
}
|
|
if ( n < 2 )
|
|
printf("pangea_antes not enough players n.%d\n",n);
|
|
else
|
|
{
|
|
pangea_playerbet(myinfo,&actualbet,tp,tp->active[0],(tp->G.bigblind>>1),CARDS777_SMALLBLIND);
|
|
pangea_playerbet(myinfo,&actualbet,tp,tp->active[1],tp->G.bigblind,CARDS777_BIGBLIND);
|
|
}
|
|
}
|
|
|
|
void pangea_checkantes(struct supernet_info *myinfo,struct table_info *tp)
|
|
{
|
|
int64_t bets[CARDS777_MAXPLAYERS+1]; int32_t i,N = tp->G.numactive; struct hand_info *hand = &tp->hand;
|
|
pangea_snapshot(tp,bets);
|
|
for (i=0; i<N; i++)
|
|
{
|
|
//printf("%.8f ",dstr(dp->balances[i]));
|
|
if ( bets[i] != 0 )
|
|
break;
|
|
}
|
|
if ( i == N && hand->checkprod.txid != 0 )
|
|
{
|
|
for (i=0; i<N; i++)
|
|
if ( bets[i] != 0 )
|
|
break;
|
|
if ( i == N )
|
|
{
|
|
//printf("i.%d vs N.%d call antes\n",i,N);
|
|
pangea_antes(myinfo,tp);
|
|
} else printf("bets i.%d\n",i);
|
|
}
|
|
}
|
|
|
|
uint64_t pangea_winnings(int32_t player,uint64_t *pangearakep,uint64_t *hostrakep,uint64_t total,int32_t numwinners,int32_t rakemillis,uint64_t maxrake)
|
|
{
|
|
uint64_t split,pangearake,rake;
|
|
if ( numwinners > 0 )
|
|
{
|
|
split = (total * (1000 - rakemillis)) / (1000 * numwinners);
|
|
pangearake = (total - split*numwinners);
|
|
if ( pangearake > maxrake )
|
|
{
|
|
pangearake = maxrake;
|
|
split = (total - pangearake) / numwinners;
|
|
pangearake = (total - split*numwinners);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
split = 0;
|
|
pangearake = total;
|
|
}
|
|
if ( rakemillis > PANGEA_MINRAKE_MILLIS )
|
|
{
|
|
rake = (pangearake * (rakemillis - PANGEA_MINRAKE_MILLIS)) / rakemillis;
|
|
pangearake -= rake;
|
|
}
|
|
else rake = 0;
|
|
*hostrakep = rake;
|
|
*pangearakep = pangearake;
|
|
printf("\nP%d: rakemillis.%d total %.8f split %.8f rake %.8f pangearake %.8f\n",player,rakemillis,dstr(total),dstr(split),dstr(rake),dstr(pangearake));
|
|
return(split);
|
|
}
|
|
|
|
int32_t pangea_sidepots(struct supernet_info *myinfo,struct table_info *tp,int32_t dispflag,int64_t sidepots[CARDS777_MAXPLAYERS][CARDS777_MAXPLAYERS],int64_t *bets)
|
|
{
|
|
int32_t i,j,nonz,N,n = 0; uint64_t bet,minbet = 0;
|
|
memset(sidepots,0,sizeof(uint64_t)*CARDS777_MAXPLAYERS*CARDS777_MAXPLAYERS);
|
|
N = tp->G.numactive;
|
|
for (j=0; j<N; j++)
|
|
sidepots[0][j] = bets[j];
|
|
nonz = 1;
|
|
while ( nonz > 0 )
|
|
{
|
|
for (minbet=j=0; j<N; j++)
|
|
{
|
|
if ( (bet= sidepots[n][j]) != 0 )
|
|
{
|
|
if ( tp->active[j] != 0 && tp->active[j]->betstatus != CARDS777_FOLD )
|
|
{
|
|
if ( minbet == 0 || bet < minbet )
|
|
minbet = bet;
|
|
}
|
|
}
|
|
}
|
|
for (j=nonz=0; j<N; j++)
|
|
{
|
|
if ( sidepots[n][j] > minbet && tp->active[j] != 0 && tp->active[j]->betstatus != CARDS777_FOLD )
|
|
nonz++;
|
|
}
|
|
if ( nonz > 0 )
|
|
{
|
|
for (j=0; j<N; j++)
|
|
{
|
|
if ( sidepots[n][j] > minbet )
|
|
{
|
|
sidepots[n+1][j] = (sidepots[n][j] - minbet);
|
|
sidepots[n][j] = minbet;
|
|
}
|
|
}
|
|
}
|
|
if ( ++n >= N )
|
|
break;
|
|
}
|
|
if ( dispflag != 0 )
|
|
{
|
|
for (i=0; i<n; i++)
|
|
{
|
|
for (j=0; j<N; j++)
|
|
printf("%.8f ",dstr(sidepots[i][j]));
|
|
printf("sidepot.%d of %d\n",i,n);
|
|
}
|
|
}
|
|
return(n);
|
|
}
|
|
|
|
int64_t pangea_splitpot(struct supernet_info *myinfo,struct table_info *tp,uint64_t *pangearakep,int64_t sidepot[CARDS777_MAXPLAYERS],int32_t rakemillis)
|
|
{
|
|
struct player_info *winners[CARDS777_MAXPLAYERS];
|
|
int32_t j,n,N,numwinners = 0; uint32_t bestrank,rank; uint8_t tmp; struct player_info *p;
|
|
uint64_t total = 0,bet,split,maxrake,rake=0,pangearake=0; char handstr[128],besthandstr[128];
|
|
N = tp->G.numactive;
|
|
bestrank = 0;
|
|
besthandstr[0] = 0;
|
|
for (j=n=0; j<N; j++)
|
|
{
|
|
if ( (bet= sidepot[j]) != 0 )
|
|
{
|
|
total += bet;
|
|
if ( (p= tp->active[j]) != 0 && p->betstatus != CARDS777_FOLD )
|
|
{
|
|
if ( p->handrank > bestrank )
|
|
{
|
|
bestrank = p->handrank;
|
|
set_handstr(besthandstr,p->hand,0);
|
|
//printf("set besthandstr.(%s)\n",besthandstr);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
for (j=0; j<N; j++)
|
|
{
|
|
if ( (p= tp->active[j]) != 0 && p->betstatus != CARDS777_FOLD && sidepot[j] > 0 )
|
|
{
|
|
if ( p->handrank == bestrank )
|
|
winners[numwinners++] = p;
|
|
rank = set_handstr(handstr,p->hand,0);
|
|
if ( handstr[strlen(handstr)-1] == ' ' )
|
|
handstr[strlen(handstr)-1] = 0;
|
|
//if ( hn->server->H.slot == 0 )
|
|
printf("(p%d %14s)",j,handstr[0]!=' '?handstr:handstr+1);
|
|
//printf("(%2d %2d).%d ",dp->hands[j][5],dp->hands[j][6],(int32_t)dp->balances[j]);
|
|
}
|
|
}
|
|
if ( numwinners == 0 )
|
|
printf("pangea_splitpot error: numwinners.0\n");
|
|
else
|
|
{
|
|
uint64_t maxrakes[CARDS777_MAXPLAYERS+1] = { 0, 0, 1, 2, 2, 3, 3, 3, 3, 3 }; // 2players 1BB, 3-4players, 2BB, 5+players 3BB
|
|
for (j=n=0; j<N; j++)
|
|
if ( (p= tp->active[j]) != 0 && p->bets > 0 )
|
|
n++;
|
|
if ( (maxrake= maxrakes[n] * tp->G.bigblind) > tp->G.maxrake )
|
|
{
|
|
maxrake = tp->G.maxrake;
|
|
//if ( strcmp(dp->coinstr,"BTC") == 0 && maxrake < PANGEA_BTCMAXRAKE )
|
|
// maxrake = PANGEA_BTCMAXRAKE;
|
|
//else if ( maxrake < PANGEA_MAXRAKE )
|
|
maxrake = PANGEA_MAXRAKE;
|
|
}
|
|
split = pangea_winnings(tp->priv.myind,&pangearake,&rake,total,numwinners,rakemillis,maxrake);
|
|
(*pangearakep) += pangearake;
|
|
for (j=0; j<numwinners; j++)
|
|
{
|
|
tmp = winners[j]->ind;
|
|
pangea_summaryadd(myinfo,tp,CARDS777_WINNINGS,&tmp,sizeof(tmp),(void *)&split,sizeof(split));
|
|
winners[j]->balance += split;
|
|
winners[j]->won += split;
|
|
}
|
|
if ( split*numwinners + rake + pangearake != total )
|
|
printf("pangea_split total error %.8f != split %.8f numwinners %d rake %.8f pangearake %.8f\n",dstr(total),dstr(split),numwinners,dstr(rake),dstr(pangearake));
|
|
//if ( hn->server->H.slot == 0 )
|
|
{
|
|
printf(" total %.8f split %.8f rake %.8f Prake %.8f hand.(%s) N%d winners ",dstr(total),dstr(split),dstr(rake),dstr(pangearake),besthandstr,tp->numhands);
|
|
for (j=0; j<numwinners; j++)
|
|
printf("%d ",winners[j]->ind);
|
|
printf("\n");
|
|
}
|
|
}
|
|
return(rake);
|
|
}
|
|
|
|
/*char *pangea_input(uint64_t my64bits,uint64_t tableid,cJSON *json)
|
|
{
|
|
char *actionstr; uint64_t sum,amount=0; int32_t action=0,num,threadid; struct table_info *sp; struct cards777_pubdata *dp; char hex[4096];
|
|
threadid = juint(json,"threadid");
|
|
if ( (sp= pangea_threadtables(&num,threadid,tableid)) == 0 )
|
|
return(clonestr("{\"error\":\"you are not playing on any tables\"}"));
|
|
if ( 0 && num != 1 )
|
|
return(clonestr("{\"error\":\"more than one active table\"}"));
|
|
else if ( (dp= sp->dp) == 0 )
|
|
return(clonestr("{\"error\":\"no pubdata ptr for table\"}"));
|
|
else if ( dp->hand.undergun != pangea_ind(sp,sp->myslot) || dp->hand.betsize == 0 )
|
|
{
|
|
printf("undergun.%d threadid.%d myind.%d\n",dp->hand.undergun,sp->tp->threadid,pangea_ind(sp,sp->myslot));
|
|
return(clonestr("{\"error\":\"not your turn\"}"));
|
|
}
|
|
else if ( (actionstr= jstr(json,"action")) == 0 )
|
|
return(clonestr("{\"error\":\"on action specified\"}"));
|
|
else
|
|
{
|
|
if ( strcmp(actionstr,"check") == 0 || strcmp(actionstr,"call") == 0 || strcmp(actionstr,"bet") == 0 || strcmp(actionstr,"raise") == 0 || strcmp(actionstr,"allin") == 0 || strcmp(actionstr,"fold") == 0 )
|
|
{
|
|
sum = dp->hand.bets[pangea_ind(sp,sp->myslot)];
|
|
if ( strcmp(actionstr,"allin") == 0 )
|
|
amount = sp->balances[sp->myslot], action = CARDS777_ALLIN;
|
|
else if ( strcmp(actionstr,"bet") == 0 )
|
|
amount = j64bits(json,"amount"), action = 1;
|
|
else
|
|
{
|
|
if ( dp->hand.betsize == sum )
|
|
{
|
|
if ( strcmp(actionstr,"check") == 0 || strcmp(actionstr,"call") == 0 )
|
|
action = 0;
|
|
else if ( strcmp(actionstr,"raise") == 0 )
|
|
{
|
|
action = 1;
|
|
if ( (amount= dp->hand.lastraise) < j64bits(json,"amount") )
|
|
amount = j64bits(json,"amount");
|
|
}
|
|
else printf("unsupported userinput command.(%s)\n",actionstr);
|
|
}
|
|
else
|
|
{
|
|
if ( strcmp(actionstr,"check") == 0 || strcmp(actionstr,"call") == 0 )
|
|
action = 1, amount = (dp->hand.betsize - sum);
|
|
else if ( strcmp(actionstr,"raise") == 0 )
|
|
{
|
|
action = 2;
|
|
amount = (dp->hand.betsize - sum);
|
|
if ( amount < dp->hand.lastraise )
|
|
amount = dp->hand.lastraise;
|
|
if ( j64bits(json,"amount") > amount )
|
|
amount = j64bits(json,"amount");
|
|
}
|
|
else if ( strcmp(actionstr,"fold") == 0 )
|
|
action = 0;
|
|
else printf("unsupported userinput command.(%s)\n",actionstr);
|
|
}
|
|
}
|
|
if ( amount > sp->balances[sp->myslot] )
|
|
amount = sp->balances[sp->myslot], action = CARDS777_ALLIN;
|
|
pangea_sendcmd(hex,&sp->tp->hn,"action",-1,(void *)&amount,sizeof(amount),dp->hand.cardi,action);
|
|
printf("ACTION.(%s)\n",hex);
|
|
return(clonestr("{\"result\":\"action submitted\"}"));
|
|
}
|
|
else return(clonestr("{\"error\":\"illegal action specified, must be: check, call, bet, raise, fold or allin\"}"));
|
|
}
|
|
}*/
|
|
|
|
|
|
|