/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

/*
 * find.c
 *
 * This file implements the nssCKFWFindObjects type and methods.
 */

#ifndef CK_H
#include "ck.h"
#endif /* CK_H */

/*
 * NSSCKFWFindObjects
 *
 *  -- create/destroy --
 *  nssCKFWFindObjects_Create
 *  nssCKFWFindObjects_Destroy
 *
 *  -- public accessors --
 *  NSSCKFWFindObjects_GetMDFindObjects
 *
 *  -- implement public accessors --
 *  nssCKFWFindObjects_GetMDFindObjects
 *
 *  -- private accessors --
 *
 *  -- module fronts --
 *  nssCKFWFindObjects_Next
 */

struct NSSCKFWFindObjectsStr {
    NSSCKFWMutex *mutex; /* merely to serialise the MDObject calls */
    NSSCKMDFindObjects *mdfo1;
    NSSCKMDFindObjects *mdfo2;
    NSSCKFWSession *fwSession;
    NSSCKMDSession *mdSession;
    NSSCKFWToken *fwToken;
    NSSCKMDToken *mdToken;
    NSSCKFWInstance *fwInstance;
    NSSCKMDInstance *mdInstance;

    NSSCKMDFindObjects *mdFindObjects; /* varies */
};

#ifdef DEBUG
/*
 * But first, the pointer-tracking stuff.
 *
 * NOTE: the pointer-tracking support in NSS/base currently relies
 * upon NSPR's CallOnce support.  That, however, relies upon NSPR's
 * locking, which is tied into the runtime.  We need a pointer-tracker
 * implementation that uses the locks supplied through C_Initialize.
 * That support, however, can be filled in later.  So for now, I'll
 * just do these routines as no-ops.
 */

static CK_RV
findObjects_add_pointer(
    const NSSCKFWFindObjects *fwFindObjects)
{
    return CKR_OK;
}

static CK_RV
findObjects_remove_pointer(
    const NSSCKFWFindObjects *fwFindObjects)
{
    return CKR_OK;
}

NSS_IMPLEMENT CK_RV
nssCKFWFindObjects_verifyPointer(
    const NSSCKFWFindObjects *fwFindObjects)
{
    return CKR_OK;
}

#endif /* DEBUG */

/*
 * nssCKFWFindObjects_Create
 *
 */
NSS_EXTERN NSSCKFWFindObjects *
nssCKFWFindObjects_Create(
    NSSCKFWSession *fwSession,
    NSSCKFWToken *fwToken,
    NSSCKFWInstance *fwInstance,
    NSSCKMDFindObjects *mdFindObjects1,
    NSSCKMDFindObjects *mdFindObjects2,
    CK_RV *pError)
{
    NSSCKFWFindObjects *fwFindObjects = NULL;
    NSSCKMDSession *mdSession;
    NSSCKMDToken *mdToken;
    NSSCKMDInstance *mdInstance;

    mdSession = nssCKFWSession_GetMDSession(fwSession);
    mdToken = nssCKFWToken_GetMDToken(fwToken);
    mdInstance = nssCKFWInstance_GetMDInstance(fwInstance);

    fwFindObjects = nss_ZNEW(NULL, NSSCKFWFindObjects);
    if (!fwFindObjects) {
        *pError = CKR_HOST_MEMORY;
        goto loser;
    }

    fwFindObjects->mdfo1 = mdFindObjects1;
    fwFindObjects->mdfo2 = mdFindObjects2;
    fwFindObjects->fwSession = fwSession;
    fwFindObjects->mdSession = mdSession;
    fwFindObjects->fwToken = fwToken;
    fwFindObjects->mdToken = mdToken;
    fwFindObjects->fwInstance = fwInstance;
    fwFindObjects->mdInstance = mdInstance;

    fwFindObjects->mutex = nssCKFWInstance_CreateMutex(fwInstance, NULL, pError);
    if (!fwFindObjects->mutex) {
        goto loser;
    }

#ifdef DEBUG
    *pError = findObjects_add_pointer(fwFindObjects);
    if (CKR_OK != *pError) {
        goto loser;
    }
#endif /* DEBUG */

    return fwFindObjects;

loser:
    if (fwFindObjects) {
        if (NULL != mdFindObjects1) {
            if (NULL != mdFindObjects1->Final) {
                fwFindObjects->mdFindObjects = mdFindObjects1;
                mdFindObjects1->Final(mdFindObjects1, fwFindObjects, mdSession,
                                      fwSession, mdToken, fwToken, mdInstance, fwInstance);
            }
        }

        if (NULL != mdFindObjects2) {
            if (NULL != mdFindObjects2->Final) {
                fwFindObjects->mdFindObjects = mdFindObjects2;
                mdFindObjects2->Final(mdFindObjects2, fwFindObjects, mdSession,
                                      fwSession, mdToken, fwToken, mdInstance, fwInstance);
            }
        }

        nss_ZFreeIf(fwFindObjects);
    }

    if (CKR_OK == *pError) {
        *pError = CKR_GENERAL_ERROR;
    }

    return (NSSCKFWFindObjects *)NULL;
}

/*
 * nssCKFWFindObjects_Destroy
 *
 */
NSS_EXTERN void
nssCKFWFindObjects_Destroy(
    NSSCKFWFindObjects *fwFindObjects)
{
#ifdef NSSDEBUG
    if (CKR_OK != nssCKFWFindObjects_verifyPointer(fwFindObjects)) {
        return;
    }
#endif /* NSSDEBUG */

    (void)nssCKFWMutex_Destroy(fwFindObjects->mutex);

    if (fwFindObjects->mdfo1) {
        if (fwFindObjects->mdfo1->Final) {
            fwFindObjects->mdFindObjects = fwFindObjects->mdfo1;
            fwFindObjects->mdfo1->Final(fwFindObjects->mdfo1, fwFindObjects,
                                        fwFindObjects->mdSession, fwFindObjects->fwSession,
                                        fwFindObjects->mdToken, fwFindObjects->fwToken,
                                        fwFindObjects->mdInstance, fwFindObjects->fwInstance);
        }
    }

    if (fwFindObjects->mdfo2) {
        if (fwFindObjects->mdfo2->Final) {
            fwFindObjects->mdFindObjects = fwFindObjects->mdfo2;
            fwFindObjects->mdfo2->Final(fwFindObjects->mdfo2, fwFindObjects,
                                        fwFindObjects->mdSession, fwFindObjects->fwSession,
                                        fwFindObjects->mdToken, fwFindObjects->fwToken,
                                        fwFindObjects->mdInstance, fwFindObjects->fwInstance);
        }
    }

    nss_ZFreeIf(fwFindObjects);

#ifdef DEBUG
    (void)findObjects_remove_pointer(fwFindObjects);
#endif /* DEBUG */

    return;
}

/*
 * nssCKFWFindObjects_GetMDFindObjects
 *
 */
NSS_EXTERN NSSCKMDFindObjects *
nssCKFWFindObjects_GetMDFindObjects(
    NSSCKFWFindObjects *fwFindObjects)
{
#ifdef NSSDEBUG
    if (CKR_OK != nssCKFWFindObjects_verifyPointer(fwFindObjects)) {
        return (NSSCKMDFindObjects *)NULL;
    }
#endif /* NSSDEBUG */

    return fwFindObjects->mdFindObjects;
}

/*
 * nssCKFWFindObjects_Next
 *
 */
NSS_EXTERN NSSCKFWObject *
nssCKFWFindObjects_Next(
    NSSCKFWFindObjects *fwFindObjects,
    NSSArena *arenaOpt,
    CK_RV *pError)
{
    NSSCKMDObject *mdObject;
    NSSCKFWObject *fwObject = (NSSCKFWObject *)NULL;
    NSSArena *objArena;

#ifdef NSSDEBUG
    if (!pError) {
        return (NSSCKFWObject *)NULL;
    }

    *pError = nssCKFWFindObjects_verifyPointer(fwFindObjects);
    if (CKR_OK != *pError) {
        return (NSSCKFWObject *)NULL;
    }
#endif /* NSSDEBUG */

    *pError = nssCKFWMutex_Lock(fwFindObjects->mutex);
    if (CKR_OK != *pError) {
        return (NSSCKFWObject *)NULL;
    }

    if (fwFindObjects->mdfo1) {
        if (fwFindObjects->mdfo1->Next) {
            fwFindObjects->mdFindObjects = fwFindObjects->mdfo1;
            mdObject = fwFindObjects->mdfo1->Next(fwFindObjects->mdfo1,
                                                  fwFindObjects, fwFindObjects->mdSession, fwFindObjects->fwSession,
                                                  fwFindObjects->mdToken, fwFindObjects->fwToken,
                                                  fwFindObjects->mdInstance, fwFindObjects->fwInstance,
                                                  arenaOpt, pError);
            if (!mdObject) {
                if (CKR_OK != *pError) {
                    goto done;
                }

                /* All done. */
                fwFindObjects->mdfo1->Final(fwFindObjects->mdfo1, fwFindObjects,
                                            fwFindObjects->mdSession, fwFindObjects->fwSession,
                                            fwFindObjects->mdToken, fwFindObjects->fwToken,
                                            fwFindObjects->mdInstance, fwFindObjects->fwInstance);
                fwFindObjects->mdfo1 = (NSSCKMDFindObjects *)NULL;
            } else {
                goto wrap;
            }
        }
    }

    if (fwFindObjects->mdfo2) {
        if (fwFindObjects->mdfo2->Next) {
            fwFindObjects->mdFindObjects = fwFindObjects->mdfo2;
            mdObject = fwFindObjects->mdfo2->Next(fwFindObjects->mdfo2,
                                                  fwFindObjects, fwFindObjects->mdSession, fwFindObjects->fwSession,
                                                  fwFindObjects->mdToken, fwFindObjects->fwToken,
                                                  fwFindObjects->mdInstance, fwFindObjects->fwInstance,
                                                  arenaOpt, pError);
            if (!mdObject) {
                if (CKR_OK != *pError) {
                    goto done;
                }

                /* All done. */
                fwFindObjects->mdfo2->Final(fwFindObjects->mdfo2, fwFindObjects,
                                            fwFindObjects->mdSession, fwFindObjects->fwSession,
                                            fwFindObjects->mdToken, fwFindObjects->fwToken,
                                            fwFindObjects->mdInstance, fwFindObjects->fwInstance);
                fwFindObjects->mdfo2 = (NSSCKMDFindObjects *)NULL;
            } else {
                goto wrap;
            }
        }
    }

    /* No more objects */
    *pError = CKR_OK;
    goto done;

wrap:
    /*
   * This seems is less than ideal-- we should determine if it's a token
   * object or a session object, and use the appropriate arena.
   * But that duplicates logic in nssCKFWObject_IsTokenObject.
   * Also we should lookup the real session the object was created on
   * if the object was a session object... however this code is actually
   * correct because nssCKFWObject_Create will return a cached version of
   * the object from it's hash. This is necessary because 1) we don't want
   * to create an arena style leak (where our arena grows with every search),
   * and 2) we want the same object to always have the same ID. This means
   * the only case the nssCKFWObject_Create() will need the objArena and the
   * Session is in the case of token objects (session objects should already
   * exist in the cache from their initial creation). So this code is correct,
   * but it depends on nssCKFWObject_Create caching all objects.
   */
    objArena = nssCKFWToken_GetArena(fwFindObjects->fwToken, pError);
    if (!objArena) {
        if (CKR_OK == *pError) {
            *pError = CKR_HOST_MEMORY;
        }
        goto done;
    }

    fwObject = nssCKFWObject_Create(objArena, mdObject,
                                    NULL, fwFindObjects->fwToken,
                                    fwFindObjects->fwInstance, pError);
    if (!fwObject) {
        if (CKR_OK == *pError) {
            *pError = CKR_GENERAL_ERROR;
        }
    }

done:
    (void)nssCKFWMutex_Unlock(fwFindObjects->mutex);
    return fwObject;
}

/*
 * NSSCKFWFindObjects_GetMDFindObjects
 *
 */

NSS_EXTERN NSSCKMDFindObjects *
NSSCKFWFindObjects_GetMDFindObjects(
    NSSCKFWFindObjects *fwFindObjects)
{
#ifdef DEBUG
    if (CKR_OK != nssCKFWFindObjects_verifyPointer(fwFindObjects)) {
        return (NSSCKMDFindObjects *)NULL;
    }
#endif /* DEBUG */

    return nssCKFWFindObjects_GetMDFindObjects(fwFindObjects);
}