/******************************************************************************
 * 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"

bits256 cards777_initcrypt(bits256 data,bits256 privkey,bits256 pubkey,int32_t invert)
{
    bits256 hash;
    hash = curve25519_shared(privkey,pubkey);
    if ( invert != 0 )
        hash = crecip_donna(hash);
    return(fmul_donna(data,hash));
}

bits256 cards777_cardpriv(bits256 playerpriv,bits256 *cardpubs,int32_t numcards,bits256 cipher)
{
    bits256 cardpriv,checkpub; int32_t i;
    for (i=0; i<numcards; i++)
    {
        cardpriv = cards777_initcrypt(cipher,playerpriv,cardpubs[i],1);
        //printf("(%llx %llx) ",(long long)cardpriv.txid,(long long)curve25519_shared(playerpriv,cardpubs[i]).txid);
        checkpub = curve25519(cardpriv,curve25519_basepoint9());
        if ( memcmp(checkpub.bytes,cardpubs[i].bytes,sizeof(bits256)) == 0 )
        {
            //printf("%d ",cardpriv.bytes[1]);
            //printf("decrypted card.%d %llx\n",cardpriv.bytes[1],(long long)cardpriv.txid);
            return(cardpriv);
        }
    }
    //printf("\nplayerpriv %llx cipher.%llx\n",(long long)playerpriv.txid,(long long)cipher.txid);
    memset(cardpriv.bytes,0,sizeof(cardpriv));
    return(cardpriv);
}

int32_t cards777_checkcard(bits256 *cardprivp,int32_t cardi,int32_t slot,int32_t destplayer,bits256 playerpriv,bits256 *cardpubs,int32_t numcards,bits256 card)
{
    bits256 cardpriv;
    cardpriv = cards777_cardpriv(playerpriv,cardpubs,numcards,card);
    if ( cardpriv.txid != 0 )
    {
        if ( slot >= 0 && destplayer != slot )
            printf(">>>>>>>>>>>> ERROR ");
        if ( Debuglevel > 2 )
            printf("slot.%d B DECODED cardi.%d destplayer.%d cardpriv.[%d]\n",slot,cardi,destplayer,cardpriv.bytes[1]);
        *cardprivp = cardpriv;
        return(cardpriv.bytes[1]);
    }
    memset(cardprivp,0,sizeof(*cardprivp));
    return(-1);
}

int32_t cards777_validate(bits256 cardpriv,bits256 final,bits256 *cardpubs,int32_t numcards,bits256 *audit,int32_t numplayers,bits256 playerpub)
{
    int32_t i; bits256 val,checkcard,ver;
    val = final;
    for (i=numplayers-1; i>0; i--)
    {
        val = fmul_donna(audit[i],val);
        //if ( memcmp(tmp.bytes,audit[i-1].bytes,sizeof(tmp)) != 0 )
        //    printf("cards777_validate: mismatched audit[%d] %llx vs %llx %llx\n",i-1,(long long)tmp.txid,(long long)audit[i-1].txid,(long long)audit[i].txid);
    }
    checkcard = val;//fcontract(val);
    if ( memcmp(checkcard.bytes,audit[0].bytes,sizeof(checkcard)) != 0 )
    {
        printf("cards777_validate: checkcard not validated %llx vs %llx numplayers.%d\n",(long long)checkcard.txid,(long long)audit[0].txid,numplayers);
        return(-1);
    }
    ver = cards777_initcrypt(cardpriv,cardpriv,playerpub,0);
    if ( memcmp(checkcard.bytes,ver.bytes,sizeof(checkcard)) != 0 )
    {
        printf("cards777_validate: ver not validated %llx vs %llx\n",(long long)checkcard.txid,(long long)ver.txid);
        return(-1);
    }
    return(cardpriv.bytes[1]);
}

int32_t cards777_shuffle(bits256 *shuffled,bits256 *cards,int32_t numcards,int32_t N)
{
    int32_t i,j,pos,nonz,permi[CARDS777_MAXCARDS],desti[CARDS777_MAXCARDS]; uint8_t x; uint64_t mask;
    memset(desti,0,sizeof(desti));
    for (i=0; i<numcards; i++)
        desti[i] = i;
    for (i=0; i<numcards; i++)
    {
        OS_randombytes(&x,1);
        pos = (x % ((numcards-1-i) + 1));
        //printf("%d ",pos);
        permi[i] = desti[pos];
        desti[pos] = desti[numcards-1 - i];
        desti[numcards-1 - i] = -1;
    }
    //printf("pos\n");
    for (mask=i=nonz=0; i<numcards; i++)
    {
        if ( 0 && Debuglevel > 2 )
            printf("%d ",permi[i]);
        mask |= (1LL << permi[i]);
        for (j=0; j<N; j++,nonz++)
            shuffled[nonz] = cards[permi[i]*N + j];//, printf("%llx ",(long long)shuffled[nonz].txid);
    }
    if ( Debuglevel > 2 )
        printf("shuffled mask.%llx err.%llx\n",(long long)mask,(long long)(mask ^ ((1LL<<numcards)-1)));
    return(0);
}

void cards777_layer(bits256 *layered,bits256 *xoverz,bits256 *incards,int32_t numcards,int32_t N)
{
    int32_t i,k,nonz = 0; bits256 z_x;
    for (i=nonz=0; i<numcards; i++)
    {
        for (k=0; k<N; k++,nonz++)
        {
            xoverz[nonz] = xoverz_donna(rand256(1));
            z_x = crecip_donna(xoverz[nonz]);
            layered[nonz] = fmul_donna(z_x,incards[nonz]);
            //printf("{%llx -> %llx}.%d ",(long long)incards[nonz].txid,(long long)layered[nonz].txid,nonz);
        }
        //printf("card.%d\n",i);
    }
}

int32_t cards777_calcmofn(struct supernet_info *myinfo,uint8_t *allshares,uint8_t *myshares[],uint8_t *sharenrs,int32_t M,bits256 *xoverz,int32_t numcards,int32_t N)
{
    int32_t size,j; uint8_t space[8192];
    size = N * sizeof(bits256) * numcards;
    calc_shares(myinfo,allshares,(void *)xoverz,size,size,M,N,sharenrs,space,sizeof(space)); // PM &allshares[playerj * size] to playerJ
    for (j=0; j<N; j++)
        myshares[j] = &allshares[j * size];
    return(size);
}

uint8_t *cards777_recover(uint8_t *shares[],uint8_t *sharenrs,int32_t M,int32_t numcards,int32_t N)
{
    void *G; int32_t i,size; uint8_t *recover,recovernrs[255],space[8192];
    size = N * sizeof(bits256) * numcards;
    if ( (recover= calloc(1,size)) == 0 )
    {
        printf("cards777_recover: unexpected out of memory error\n");
        return(0);
    }
    memset(recovernrs,0,sizeof(recovernrs));
    for (i=0; i<N; i++)
        if ( shares[i] != 0 )
            recovernrs[i] = sharenrs[i];
    G = gfshare_ctx_initdec(recovernrs,N,size,space,sizeof(space));
    for (i=0; i<N; i++)
        if ( shares[i] != 0 )
            gfshare_ctx_dec_giveshare(G,i,shares[i]);
    gfshare_ctx_dec_newshares(G,recovernrs);
    gfshare_ctx_decextract(0,0,G,recover);
    gfshare_ctx_free(G);
    return(recover);
}

bits256 cards777_pubkeys(bits256 *pubkeys,int32_t numcards,bits256 cmppubkey)
{
    int32_t i; bits256 bp,pubkey,hash,check,prod;
    memset(check.bytes,0,sizeof(check));
    memset(bp.bytes,0,sizeof(bp)), bp.bytes[0] = 9;
    prod = fmul_donna(bp,crecip_donna(bp));
    for (i=0; i<numcards; i++)
    {
        pubkey = pubkeys[i];
        vcalc_sha256(0,hash.bytes,pubkey.bytes,sizeof(pubkey));
        hash.bytes[0] &= 0xf8, hash.bytes[31] &= 0x7f, hash.bytes[31] |= 64;
        prod = fmul_donna(prod,hash);
    }
    check = prod;
    if ( cmppubkey.txid != 0 )
    {
        if ( memcmp(check.bytes,cmppubkey.bytes,sizeof(check)) != 0 )
            printf("cards777_pubkeys: mismatched pubkeys permicheck.%llx != prod.%llx\n",(long long)check.txid,(long long)pubkey.txid);
        //else printf("pubkeys matched\n");
    }
    return(check);
}

bits256 cards777_initdeck(bits256 *cards,bits256 *cardpubs,int32_t numcards,int32_t N,bits256 *playerpubs,bits256 *playerprivs)
{
    char buf[4096]; bits256 privkey,pubkey,hash, bp,prod; int32_t i,j,nonz,num = 0; uint64_t mask = 0;
    bp = curve25519_basepoint9();
    prod = crecip_donna(bp);
    prod = fmul_donna(bp,prod);
    if ( Debuglevel > 2 )
        printf("card777_initdeck unit.%llx\n",(long long)prod.txid);
    nonz = 0;
    while ( mask != (1LL << numcards)-1 )
    {
        privkey = curve25519_keypair(&pubkey);
        buf[0] = 0;
        if ( (i=privkey.bytes[1]) < numcards && ((1LL << i) & mask) == 0 )
        {
            mask |= (1LL << i);
            cardpubs[num] = pubkey;
            if ( playerprivs != 0 )
                sprintf(buf+strlen(buf),"%llx.",(long long)privkey.txid);
            for (j=0; j<N; j++,nonz++)
            {
                cards[nonz] = cards777_initcrypt(privkey,privkey,playerpubs[j],0);
                if ( playerprivs != 0 )
                    sprintf(buf+strlen(buf),"[%llx * %llx -> %llx] ",(long long)cards[nonz].txid,(long long)curve25519_shared(playerprivs[j],pubkey).txid,(long long)cards777_initcrypt(cards[nonz],playerprivs[j],pubkey,1).txid);
            }
            vcalc_sha256(0,hash.bytes,pubkey.bytes,sizeof(pubkey));
            hash.bytes[0] &= 0xf8, hash.bytes[31] &= 0x7f, hash.bytes[31] |= 64;
            prod = fmul_donna(prod,hash);
            //printf("(%s) num.%d [%llx] %d prod.%llx\n",buf,num,(long long)mask ^ ((1LL << numcards)-1),i,(long long)prod.txid);
            num++;
        }
    }
    if ( playerprivs != 0 )
        printf("\n%llx %llx playerprivs\n",(long long)playerprivs[0].txid,(long long)playerprivs[1].txid);
    //if ( 0 && Debuglevel > 2 )
    {
        for (i=0; i<numcards; i++)
            printf("%d ",cards[i*N].bytes[1]);
        printf("init order %llx (%llx %llx)\n",(long long)prod.txid,(long long)playerpubs[0].txid,(long long)playerpubs[1].txid);
    }
    return(prod);
}

uint8_t *cards777_encode(struct supernet_info *myinfo,bits256 *encoded,bits256 *xoverz,uint8_t *allshares,uint8_t *myshares[],uint8_t sharenrs[255],int32_t M,bits256 *ciphers,int32_t numcards,int32_t N)
{
    bits256 shuffled[CARDS777_MAXCARDS * CARDS777_MAXPLAYERS];
    cards777_shuffle(shuffled,ciphers,numcards,N);
    cards777_layer(encoded,xoverz,shuffled,numcards,N);
    memset(sharenrs,0,255);
    init_sharenrs(sharenrs,0,N,N);
    cards777_calcmofn(myinfo,allshares,myshares,sharenrs,M,xoverz,numcards,N);
    memcpy(ciphers,shuffled,numcards * N * sizeof(bits256));
    if ( (0) )
    {
        /*{
            init_hexbytes_noT(nrs,dp->hand.sharenrs,dp->N);
            if ( (nrs= jstr(json,"sharenrs")) != 0 )
                decode_hex(dp->hand.sharenrs,(int32_t)strlen(nrs)>>1,nrs);
        }*/
        int32_t i,j,m,size; uint8_t *recover,*testshares[CARDS777_MAXPLAYERS],testnrs[255];
        size = N * sizeof(bits256) * numcards;
        for (j=0; j<1; j++)
        {
            memset(testnrs,0,sizeof(testnrs));
            memset(testshares,0,sizeof(testshares));
            m = (rand() % N) + 1;
            if ( m < M )
                m = M;
            if ( init_sharenrs(testnrs,sharenrs,m,N) < 0 )
            {
                printf("iter.%d error init_sharenrs(m.%d of n.%d)\n",j,m,N);
                return(0);
            }
            for (i=0; i<N; i++)
                if ( testnrs[i] == sharenrs[i] )
                    testshares[i] = myshares[i];
            if ( (recover= cards777_recover(testshares,sharenrs,M,numcards,N)) != 0 )
            {
                if ( memcmp(xoverz,recover,size) != 0 )
                    fprintf(stderr,"(ERROR m.%d M.%d N.%d)\n",m,M,N);
                else fprintf(stderr,"reconstructed with m.%d M.%d N.%d\n",m,M,N);
                free(recover);
            } else printf("nullptr from cards777_recover\n");
        }
    }
    return(allshares);
}

bits256 cards777_decode(bits256 *seedp,bits256 *xoverz,int32_t destplayer,bits256 cipher,bits256 *outcards,int32_t numcards,int32_t N)
{
    int32_t i,ind;
    memset(seedp->bytes,0,sizeof(*seedp));
    for (i=0; i<numcards; i++)
    {
        ind = i*N + destplayer;
        //printf("[%llx] ",(long long)outcards[ind].txid);
        if ( memcmp(outcards[ind].bytes,cipher.bytes,32) == 0 )
        {
            *seedp = xoverz[ind];
            cipher = fmul_donna(xoverz[ind],cipher);
            //printf("matched %d -> %llx\n",i,(long long)cipher.txid);
            return(cipher);
        }
    }
    if ( i == numcards )
    {
        printf("decryption error %llx: destplayer.%d no match\n",(long long)cipher.txid,destplayer);
        memset(cipher.bytes,0,sizeof(cipher));
        //cipher = cards777_cardpriv(playerpriv,cardpubs,numcards,cipher);
    }
    return(cipher);
}

#ifdef notyet
struct cards777_privdata *cards777_allocpriv(int32_t numcards,int32_t N)
{
    struct cards777_privdata *priv;
    if ( (priv= calloc(1,sizeof(*priv) + sizeof(bits256) * (2*((N * numcards * N) + (N * numcards))))) == 0 )
    {
        printf("cards777_allocpriv: unexpected out of memory error\n");
        return(0);
    }
    priv->audits = &priv->data[0];
    priv->outcards = &priv->audits[N * numcards * N];
    priv->xoverz = &priv->outcards[N * numcards];
    priv->allshares = (void *)&priv->xoverz[N * numcards]; // N*numcards*N
    return(priv);
}

struct cards777_pubdata *cards777_allocpub(int32_t M,int32_t numcards,int32_t N)
{
    struct cards777_pubdata *dp;
    if ( (dp= calloc(1,sizeof(*dp) + sizeof(bits256) * ((numcards + 1) + (N * numcards)))) == 0 )
    {
        printf("cards777_allocpub: unexpected out of memory error\n");
        return(0);
    }
    dp->M = M, dp->N = N, dp->numcards = numcards;
    dp->hand.cardpubs = &dp->data[0];
    dp->hand.final = &dp->hand.cardpubs[numcards + 1];
    return(dp);
}

int32_t cards777_testinit(struct hostnet777_server *srv,int32_t M,struct hostnet777_client **clients,int32_t N,int32_t numcards)
{
    //static int64_t balances[9];
    int32_t i; uint8_t sharenrs[255]; //,destplayer,cardibits256 *ciphers,cardpriv,card; uint64_t mask = 0;
    struct cards777_pubdata *dp; //struct cards777_privdata *priv; struct pangea_info *sp;
    if ( srv->num != N )
    {
        printf("srv->num.%d != N.%d\n",srv->num,N);
        return(-1);
    }
    memset(sharenrs,0,sizeof(sharenrs));
    init_sharenrs(sharenrs,0,N,N); // this needs to be done to start a hand
    for (i=0; i<N; i++)
    {
        dp = srv->clients[i].pubdata = cards777_allocpub(M,numcards,N);
        //sp = dp->table;
        memcpy(dp->hand.sharenrs,sharenrs,dp->N);
        /*for (j=0; j<N; j++)
            sp->playerpubs[j] = srv->clients[j].pubkey;
        for (j=0; j<N; j++)
        {
            balances[j] = 100;
            dp->balances[j] = &balances[j];
        }*/
        printf("deprecated, need to init sp->\n");
        //priv = srv->clients[i].privdata = cards777_allocpriv(numcards,N);
        //priv->privkey = (i == 0) ? srv->H.privkey : clients[i]->H.privkey;
        /*if ( i == 0 )
            dp->checkprod = cards777_initdeck(priv->outcards,dp->cardpubs,numcards,N,dp->playerpubs), refdp = dp;
        else memcpy(dp->cardpubs,refdp->cardpubs,sizeof(*dp->cardpubs) * numcards);*/
    }
    return(0);
    /*priv = srv->clients[0].privdata;
    ciphers = priv->outcards;
    for (i=1; i<N; i++)
    {
        dp = srv->clients[i].pubdata;
        priv = srv->clients[i].privdata;
        cards777_encode(priv->outcards,priv->xoverz,priv->allshares,priv->myshares,dp->sharenrs,dp->M,ciphers,dp->numcards,dp->N);
        ciphers = priv->outcards;
    }
    for (cardi=0; cardi<dp->numcards; cardi++)
    {
        for (destplayer=0; destplayer<dp->N; destplayer++)
        {
            priv = srv->clients[dp->N - 1].privdata;
            card = priv->outcards[cardi*dp->N + destplayer];
            for (i=N-1; i>=0; i--)
            {
                j = (i > 0) ? i : destplayer;
                //printf("cardi.%d destplayer.%d i.%d j.%d\n",cardi,destplayer,i,j);
                dp = srv->clients[j].pubdata;
                priv = srv->clients[j].privdata;
                cardpriv = cards777_cardpriv(priv->privkey,dp->cardpubs,dp->numcards,card);
                if ( cardpriv.txid != 0 )
                {
                    mask |= (1LL << cardpriv.bytes[1]);
                    if ( destplayer != j )
                        printf(">>>>>>>>>>>> ERROR ");
                    printf("i.%d j.%d A DECODED cardi.%d destplayer.%d cardpriv.[%d] mask.%llx\n",i,j,cardi,destplayer,cardpriv.bytes[1],(long long)mask);
                    break;
                }
                card = cards777_decode(priv->xoverz,destplayer,card,priv->outcards,dp->numcards,dp->N);
                cardpriv = cards777_cardpriv(priv->privkey,dp->cardpubs,dp->numcards,card);
                if ( cardpriv.txid != 0 )
                {
                    mask |= (1LL << cardpriv.bytes[1]);
                    if ( destplayer != j )
                        printf(">>>>>>>>>>>> ERROR ");
                    printf("i.%d j.%d B DECODED cardi.%d destplayer.%d cardpriv.[%d] mask.%llx\n",i,j,cardi,destplayer,cardpriv.bytes[1],(long long)mask);
                    break;
                }
            }
        }
        printf("cardi.%d\n\n",cardi);
    }*/
    return(0);
}

void cards777_initid(struct hostnet777_id *id,bits256 pubkey,struct cards777_pubdata *dp,struct cards777_privdata *priv)
{
    id->pubkey = pubkey;
    id->nxt64bits = acct777_nxt64bits(pubkey);
    id->pubdata = dp;
    id->privdata = priv;
    id->pmsock = -1;
}

void cards777_test()
{
    int32_t i,j,vals[52][52]; bits256 keypairs[52][2],otherpairs[52][2],matrix[52][52]; char buf[512];
    FILE *fp;
    if ( (fp= fopen("/persistent/test","rb")) != 0 )
    {
        if ( fread(buf,6,1,fp) <= 0 )
            printf("read error for /persistent/test\n");
        buf[6] = 0;
        printf("test exists (%s)\n",buf);
        fclose(fp);
    } else printf("testfile not present\n");
    for (i=0; i<52; i++)
        keypairs[i][0] = curve25519_keypair(&keypairs[i][1]);
    for (j=0; j<52; j++)
        otherpairs[j][0] = curve25519_keypair(&otherpairs[j][1]);
    bits256 zmone;zmone = crecip_donna(keypairs[0][0]);
    printf("DEBUG.%d %llx vs %llx | %llx -> %llx/%llx\n",Debuglevel,(long long)keypairs[0][0].txid,(long long)fcontract(fexpand(keypairs[0][0])).txid,(long long)zmone.txid,(long long)fexpand(fmul_donna(keypairs[0][0],zmone)).txid,(long long)fmul(fexpand(keypairs[0][0]),fexpand(zmone)).txid);
    for (i=0; i<52; i++)
    {break;
        buf[0] = 0;
        for (j=0; j<52; j++)
        {
            matrix[i][j] = fmul_donna(keypairs[j][1],otherpairs[i][1]);
            vals[i][j] = matrix[i][j].bytes[1] % 52;
            sprintf(buf+strlen(buf),"%d ",vals[i][j]);
        }
        printf("%s\n",buf);
    }
    struct hostnet777_server *srv;  int32_t M,N = 9; //struct hostnet777_client **clients;
    struct cards777_pubdata *dp; struct cards777_privdata *priv;
    bits256 checkprod,cards[52],playerpubs[9],playerprivs[9];
    //clients = calloc(N+1,sizeof(*clients));
    if ( (srv= hostnet777_server(keypairs[0][0],keypairs[0][1],0,0,0,N)) == 0 )
    {
        printf("cant create hostnet777 server\n");
        return;
    }
    M = (N >> 1) + 1;
    for (i=0; i<N; i++)
    {
        cards777_initid(&srv->clients[i],keypairs[i][1],cards777_allocpub(M,52,N),cards777_allocpriv(52,N));
        playerprivs[i] = keypairs[i][0];
        playerpubs[i] = keypairs[i][1];
        if ( i == 0 )
        {
            srv->H.privkey = keypairs[i][0];
            srv->H.pubkey = keypairs[i][1];
        }
        else
        {
        }
    }
    dp = srv->clients[0].pubdata;
    dp->N = N; dp->M = M; dp->numcards = 52;
    checkprod = cards777_initdeck(cards,dp->hand.cardpubs,52,N,playerpubs,0);
    printf("deck initialzed %llx\n",(long long)checkprod.txid);
    uint8_t sharenrs[255]; uint64_t mask = 0; int32_t cardi,destplayer;
    bits256 card,cardpriv,seed,*ciphers = cards;
    for (i=1; i<N; i++)
    {
        dp = srv->clients[i].pubdata;
        dp->N = N; dp->M = M; dp->numcards = 52;
        priv = srv->clients[i].privdata;
        cards777_encode(priv->outcards,priv->xoverz,priv->allshares,priv->myshares,sharenrs,dp->M,ciphers,dp->numcards,dp->N);
        ciphers = priv->outcards;
    }
    printf("deck encrypted\n");
    for (cardi=0; cardi<dp->numcards; cardi++)
    {
        for (destplayer=0; destplayer<dp->N; destplayer++)
        {
            priv = srv->clients[dp->N - 1].privdata;
            card = priv->outcards[cardi*dp->N + destplayer];
            for (i=N-1; i>=0; i--)
            {
                j = (i > 0) ? i : destplayer;
                //printf("cardi.%d destplayer.%d i.%d j.%d\n",cardi,destplayer,i,j);
                dp = srv->clients[j].pubdata;
                priv = srv->clients[j].privdata;
                cardpriv = cards777_cardpriv(keypairs[j][0],dp->hand.cardpubs,dp->numcards,card);
                if ( cardpriv.txid != 0 )
                {
                    mask |= (1LL << cardpriv.bytes[1]);
                    if ( destplayer != j )
                        printf(">>>>>>>>>>>> ERROR ");
                    printf("i.%d j.%d A DECODED cardi.%d destplayer.%d cardpriv.[%d] mask.%llx\n",i,j,cardi,destplayer,cardpriv.bytes[1],(long long)mask);
                    break;
                }
                card = cards777_decode(&seed,priv->xoverz,destplayer,card,priv->outcards,dp->numcards,dp->N);
                cardpriv = cards777_cardpriv(keypairs[j][0],dp->hand.cardpubs,dp->numcards,card);
                if ( cardpriv.txid != 0 )
                {
                    mask |= (1LL << cardpriv.bytes[1]);
                    if ( destplayer != j )
                        printf(">>>>>>>>>>>> ERROR ");
                    printf("i.%d j.%d B DECODED cardi.%d destplayer.%d cardpriv.[%d] mask.%llx\n",i,j,cardi,destplayer,cardpriv.bytes[1],(long long)mask);
                    break;
                }
            }
        }
        printf("cardi.%d\n\n",cardi);
        break;
    }
}

#endif