Blame testcases/misc_tests/fork.c

Packit 8681c6
/*
Packit 8681c6
 * COPYRIGHT (c) International Business Machines Corp. 2020
Packit 8681c6
 *
Packit 8681c6
 * This program is provided under the terms of the Common Public License,
Packit 8681c6
 * version 1.0 (CPL-1.0). Any use, reproduction or distribution for this
Packit 8681c6
 * software constitutes recipient's acceptance of CPL-1.0 terms which can be
Packit 8681c6
 * found in the file LICENSE file or at
Packit 8681c6
 * https://opensource.org/licenses/cpl1.0.php
Packit 8681c6
 */
Packit 8681c6
Packit 8681c6
/* File: fork.c
Packit 8681c6
 *
Packit 8681c6
 * Test driver.  In-depth regression test for PKCS #11
Packit 8681c6
 */
Packit 8681c6
Packit 8681c6
#include <stdio.h>
Packit 8681c6
#include <stdlib.h>
Packit 8681c6
#include <string.h>
Packit 8681c6
#include <memory.h>
Packit 8681c6
#include <unistd.h>
Packit 8681c6
Packit 8681c6
#include <dlfcn.h>
Packit 8681c6
#include <sys/types.h>
Packit 8681c6
#include <sys/wait.h>
Packit 8681c6
Packit 8681c6
#include "pkcs11types.h"
Packit 8681c6
#include "regress.h"
Packit 8681c6
Packit 8681c6
CK_BYTE user_pin[128];
Packit 8681c6
CK_ULONG user_pin_len;
Packit 8681c6
CK_SLOT_ID slot_id = 0;
Packit 8681c6
Packit 8681c6
CK_RV do_GenerateTokenRSAKeyPair(CK_SESSION_HANDLE sess, CK_BYTE *label,
Packit 8681c6
                                 CK_ULONG bits, CK_OBJECT_HANDLE *hPubKey,
Packit 8681c6
                                 CK_OBJECT_HANDLE *hPrivKey)
Packit 8681c6
{
Packit 8681c6
    CK_MECHANISM mech;
Packit 8681c6
    CK_RV rv;
Packit 8681c6
    CK_MECHANISM_INFO rsakeygeninfo;
Packit 8681c6
    CK_BBOOL false = 0;
Packit 8681c6
    CK_BYTE pub_exp[] = { 0x1, 0x0, 0x1 };
Packit 8681c6
    CK_ATTRIBUTE pub_tmpl[] = {
Packit 8681c6
        {CKA_MODULUS_BITS, &bits, sizeof(bits)},
Packit 8681c6
        {CKA_PUBLIC_EXPONENT, &pub_exp, sizeof(pub_exp)},
Packit 8681c6
        {CKA_LABEL, label, (CK_ULONG) strlen((char *) label) + 1},
Packit 8681c6
        {CKA_TOKEN, &false, sizeof(CK_BBOOL)}
Packit 8681c6
    };
Packit 8681c6
    CK_ATTRIBUTE priv_tmpl[] = {
Packit 8681c6
        {CKA_LABEL, label, (CK_ULONG) strlen((char *) label) + 1},
Packit 8681c6
        {CKA_TOKEN, &false, sizeof(CK_BBOOL)}
Packit 8681c6
    };
Packit 8681c6
Packit 8681c6
    rv = funcs->C_GetMechanismInfo(slot_id, CKM_RSA_PKCS_KEY_PAIR_GEN,
Packit 8681c6
                                   &rsakeygeninfo);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_GetMechanismInfo(CKM_RSA_PKCS_KEY_PAIR_GEN) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        return rv;;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    mech.mechanism = CKM_RSA_PKCS_KEY_PAIR_GEN;
Packit 8681c6
    mech.ulParameterLen = 0;
Packit 8681c6
    mech.pParameter = NULL;
Packit 8681c6
Packit 8681c6
    rv = funcs->C_GenerateKeyPair(sess, &mech, pub_tmpl, 4, priv_tmpl, 2,
Packit 8681c6
                                  hPubKey, hPrivKey);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_GenerateKeyPair rc = %s", p11_get_ckr(rv));
Packit 8681c6
        return rv;
Packit 8681c6
    }
Packit 8681c6
    return CKR_OK;
Packit 8681c6
}
Packit 8681c6
Packit 8681c6
CK_RV do_fork(CK_SESSION_HANDLE parent_session, CK_OBJECT_HANDLE parent_object)
Packit 8681c6
{
Packit 8681c6
    pid_t child_pid;
Packit 8681c6
    int status = 1;
Packit 8681c6
    CK_FLAGS flags;
Packit 8681c6
    CK_SESSION_HANDLE session;
Packit 8681c6
    CK_RV rv;
Packit 8681c6
    CK_C_INITIALIZE_ARGS cinit_args;
Packit 8681c6
    CK_OBJECT_HANDLE hPubKey, hPrivKey;
Packit 8681c6
Packit 8681c6
    child_pid = fork();
Packit 8681c6
    if (child_pid != 0) {
Packit 8681c6
        // parent process: wait until child exits
Packit 8681c6
        waitpid(child_pid, &status, 0);
Packit 8681c6
        return status;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // child process flows here
Packit 8681c6
    testcase_setup(0);
Packit 8681c6
    t_ran = 0;
Packit 8681c6
    t_passed = 0;
Packit 8681c6
    t_skipped = 0;
Packit 8681c6
    t_failed = 0;
Packit 8681c6
    testcase_begin(".. in client process: %u", getpid());
Packit 8681c6
Packit 8681c6
    // Ensure that OCK is not initialized in this fork now
Packit 8681c6
    testcase_new_assertion();
Packit 8681c6
    flags = CKF_SERIAL_SESSION | CKF_RW_SESSION;
Packit 8681c6
    rv = funcs->C_OpenSession(slot_id, flags, NULL, NULL, &session);
Packit 8681c6
    if (rv != CKR_CRYPTOKI_NOT_INITIALIZED) {
Packit 8681c6
        testcase_fail("C_OpenSession (client) (expected CKR_CRYPTOKI_NOT_INITIALIZED) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
    testcase_pass("C_OpenSession (client)");
Packit 8681c6
Packit 8681c6
    // Initialize
Packit 8681c6
    memset(&cinit_args, 0x0, sizeof(cinit_args));
Packit 8681c6
    cinit_args.flags = CKF_OS_LOCKING_OK;
Packit 8681c6
    if ((rv = funcs->C_Initialize(&cinit_args))) {
Packit 8681c6
        testcase_fail("C_Initialize (client) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // Check access to parent session
Packit 8681c6
    if (parent_session != CK_INVALID_HANDLE) {
Packit 8681c6
        testcase_new_assertion();
Packit 8681c6
        rv = funcs->C_CloseSession(parent_session);
Packit 8681c6
        if (rv != CKR_SESSION_HANDLE_INVALID) {
Packit 8681c6
            testcase_fail("C_CloseSession (client) (expected CKR_SESSION_HANDLE_INVALID) rc = %s", p11_get_ckr(rv));
Packit 8681c6
            goto close_session;
Packit 8681c6
        }
Packit 8681c6
        testcase_pass("C_CloseSession got expected error (client)");
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // Open a session
Packit 8681c6
    flags = CKF_SERIAL_SESSION | CKF_RW_SESSION;
Packit 8681c6
    rv = funcs->C_OpenSession(slot_id, flags, NULL, NULL, &session);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_OpenSession (client) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto finalize;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // Log in
Packit 8681c6
    rv = funcs->C_Login(session, CKU_USER, user_pin, user_pin_len);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_Login (client) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto close_session;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // Check access to parent object
Packit 8681c6
    if (parent_object != CK_INVALID_HANDLE) {
Packit 8681c6
        testcase_new_assertion();
Packit 8681c6
        rv = funcs->C_DestroyObject(session, parent_object);
Packit 8681c6
        if (rv != CKR_OBJECT_HANDLE_INVALID) {
Packit 8681c6
            testcase_fail("C_DestroyObject (client) (expected CKR_OBJECT_HANDLE_INVALID) rc = %s", p11_get_ckr(rv));
Packit 8681c6
            goto close_session;
Packit 8681c6
        }
Packit 8681c6
        testcase_pass("C_DestroyObject got expected error (client)");
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // generate a key pair
Packit 8681c6
    rv = do_GenerateTokenRSAKeyPair(session, (CK_BYTE *)"RSA-1024-CLIENT",
Packit 8681c6
                                    1024, &hPubKey, &hPrivKey);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("do_GenerateTokenRSAKeyPair (client) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto close_session;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    rv = funcs->C_DestroyObject(session, hPubKey);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_DestroyObject (client) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto close_session;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    rv = funcs->C_DestroyObject(session, hPrivKey);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_DestroyObject (client) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto close_session;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    rv = 0;
Packit 8681c6
close_session:
Packit 8681c6
    rv = funcs->C_CloseSession(session);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_CloseSession (client) rc = %s", p11_get_ckr(rv));
Packit 8681c6
    }
Packit 8681c6
finalize:
Packit 8681c6
    rv = funcs->C_Finalize(NULL);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_Finalize (client) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
out:
Packit 8681c6
Packit 8681c6
    testcase_print_result();
Packit 8681c6
    exit(rv);
Packit 8681c6
}
Packit 8681c6
Packit 8681c6
int main(int argc, char **argv)
Packit 8681c6
{
Packit 8681c6
    CK_C_INITIALIZE_ARGS cinit_args;
Packit 8681c6
    int i, ret = 1;
Packit 8681c6
    CK_RV rv;
Packit 8681c6
    CK_SESSION_HANDLE session;
Packit 8681c6
    CK_FLAGS flags;
Packit 8681c6
    CK_OBJECT_HANDLE hPubKey, hPrivKey;
Packit 8681c6
Packit 8681c6
    for (i = 1; i < argc; i++) {
Packit 8681c6
        if (strcmp(argv[i], "-slot") == 0) {
Packit 8681c6
            ++i;
Packit 8681c6
            slot_id = atoi(argv[i]);
Packit 8681c6
        }
Packit 8681c6
Packit 8681c6
        if (strcmp(argv[i], "-h") == 0) {
Packit 8681c6
            printf("usage:  %s [-slot <num>] [-h]\n\n", argv[0]);
Packit 8681c6
            printf("By default, Slot #1 is used\n\n");
Packit 8681c6
            return -1;
Packit 8681c6
        }
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    if (get_user_pin(user_pin))
Packit 8681c6
        return CKR_FUNCTION_FAILED;
Packit 8681c6
    user_pin_len = (CK_ULONG) strlen((char *) user_pin);
Packit 8681c6
Packit 8681c6
    printf("Using slot #%lu...\n\n", slot_id);
Packit 8681c6
Packit 8681c6
    rv = do_GetFunctionList();
Packit 8681c6
    if (rv != TRUE) {
Packit 8681c6
        testcase_fail("do_GetFunctionList() rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    testcase_setup(0);
Packit 8681c6
    testcase_begin("Starting...  Parent process: %u", getpid());
Packit 8681c6
Packit 8681c6
    // Test fork before C_Initialize
Packit 8681c6
    testcase_new_assertion();
Packit 8681c6
    rv = do_fork(CK_INVALID_HANDLE, CK_INVALID_HANDLE);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("do_fork() before C_Initialize rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
    testcase_pass("do_fork() before C_Initialize");
Packit 8681c6
Packit 8681c6
    // Initialize
Packit 8681c6
    memset(&cinit_args, 0x0, sizeof(cinit_args));
Packit 8681c6
    cinit_args.flags = CKF_OS_LOCKING_OK;
Packit 8681c6
Packit 8681c6
    if ((rv = funcs->C_Initialize(&cinit_args))) {
Packit 8681c6
        testcase_fail("C_Initialize (parent) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // Test fork after C_Initialize
Packit 8681c6
    testcase_new_assertion();
Packit 8681c6
    rv = do_fork(CK_INVALID_HANDLE, CK_INVALID_HANDLE);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("do_fork() after C_Initialize rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
    testcase_pass("do_fork() after C_Initialize");
Packit 8681c6
Packit 8681c6
    flags = CKF_SERIAL_SESSION | CKF_RW_SESSION;
Packit 8681c6
    rv = funcs->C_OpenSession(slot_id, flags, NULL, NULL, &session);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_OpenSession (parent) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto finalize;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    rv = funcs->C_Login(session, CKU_USER, user_pin, user_pin_len);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_Login (parent) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto close_session;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // Test fork after C_OpenSession/C_Login
Packit 8681c6
    testcase_new_assertion();
Packit 8681c6
    rv = do_fork(session, CK_INVALID_HANDLE);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("do_fork() after C_OpenSession rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
    testcase_pass("do_fork() after C_OpenSession");
Packit 8681c6
Packit 8681c6
    // generate a key pair
Packit 8681c6
    rv = do_GenerateTokenRSAKeyPair(session, (CK_BYTE *)"RSA-1024-PARENT",
Packit 8681c6
                                    1024, &hPubKey, &hPrivKey);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("do_GenerateTokenRSAKeyPair (parent) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto close_session;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // Test fork after Key Gen
Packit 8681c6
    testcase_new_assertion();
Packit 8681c6
    rv = do_fork(session, hPrivKey);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("do_fork() after KeyGen rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
    testcase_pass("do_fork() after KeyGen");
Packit 8681c6
Packit 8681c6
    rv = funcs->C_DestroyObject(session, hPubKey);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_DestroyObject (parent) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto close_session;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    rv = funcs->C_DestroyObject(session, hPrivKey);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_DestroyObject (parent) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto close_session;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    rv = funcs->C_CloseSession(session);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_CloseSession (parent) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto finalize;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // Test fork before C_Finalize
Packit 8681c6
    testcase_new_assertion();
Packit 8681c6
    rv = do_fork(CK_INVALID_HANDLE, CK_INVALID_HANDLE);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("do_fork() before C_Finalize rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
    testcase_pass("do_fork() before C_Finalize");
Packit 8681c6
Packit 8681c6
    rv = funcs->C_Finalize(NULL);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_Finalize (parent) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
Packit 8681c6
    // Test fork after C_Finalize
Packit 8681c6
    testcase_new_assertion();
Packit 8681c6
    rv = do_fork(CK_INVALID_HANDLE, CK_INVALID_HANDLE);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("do_fork() after C_Finalize rc = %s", p11_get_ckr(rv));
Packit 8681c6
        goto out;
Packit 8681c6
    }
Packit 8681c6
    testcase_pass("do_fork() after C_Finalize");
Packit 8681c6
Packit 8681c6
    ret = 0;
Packit 8681c6
    goto out;
Packit 8681c6
Packit 8681c6
close_session:
Packit 8681c6
    rv = funcs->C_CloseSession(session);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_CloseSession (parent) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        ret = 1;
Packit 8681c6
    }
Packit 8681c6
finalize:
Packit 8681c6
    rv = funcs->C_Finalize(NULL);
Packit 8681c6
    if (rv != CKR_OK) {
Packit 8681c6
        testcase_fail("C_Finalize (parent) rc = %s", p11_get_ckr(rv));
Packit 8681c6
        ret = 1;
Packit 8681c6
    }
Packit 8681c6
out:
Packit 8681c6
    testcase_print_result();
Packit 8681c6
    return ret;
Packit 8681c6
}