Blame tests/fork-change-slot.c

Packit Service 2ea82d
/* libp11 test code: fork-change-slot.c
Packit Service 2ea82d
 *
Packit Service 2ea82d
 * This program loads a key pair using the engine pkcs11, forks to create
Packit Service 2ea82d
 * a new process, and waits for a SIGUSR1 signal before trying to sign/verify
Packit Service 2ea82d
 * random data in both parent and child processes.
Packit Service 2ea82d
 *
Packit Service 2ea82d
 * The intention of the signal waiting is to allow the user to add/remove
Packit Service 2ea82d
 * devices before continuing to the signature/verifying test.
Packit Service 2ea82d
 *
Packit Service 2ea82d
 * Adding or removing devices can lead to a change in the list of slot IDs
Packit Service 2ea82d
 * obtained from the PKCS#11 module. If the engine does not handle the
Packit Service 2ea82d
 * slot ID referenced by the previously loaded key properly, then the key in
Packit Service 2ea82d
 * the child process can reference to the wrong slot ID after forking.
Packit Service 2ea82d
 * This would lead to an error, since the engine will try to sign the data
Packit Service 2ea82d
 * using the key in the wrong slot.
Packit Service 2ea82d
 */
Packit Service 2ea82d
Packit Service 2ea82d
#include <sys/types.h>
Packit Service 2ea82d
#include <sys/stat.h>
Packit Service 2ea82d
#include <fcntl.h>
Packit Service 2ea82d
#include <termios.h>
Packit Service 2ea82d
#include <stdio.h>
Packit Service 2ea82d
#include <string.h>
Packit Service 2ea82d
#include <sys/types.h>
Packit Service 2ea82d
#include <sys/wait.h>
Packit Service 2ea82d
#include <unistd.h>
Packit Service 2ea82d
#include <signal.h>
Packit Service 2ea82d
Packit Service 2ea82d
#include <execinfo.h>
Packit Service 2ea82d
Packit Service 2ea82d
#include <openssl/evp.h>
Packit Service 2ea82d
#include <openssl/conf.h>
Packit Service 2ea82d
#include <openssl/engine.h>
Packit Service 2ea82d
#include <openssl/rand.h>
Packit Service 2ea82d
#include <openssl/err.h>
Packit Service 2ea82d
Packit Service 2ea82d
#define RANDOM_SIZE 20
Packit Service 2ea82d
#define MAX_SIGSIZE 1024
Packit Service 2ea82d
Packit Service 2ea82d
#if OPENSSL_VERSION_NUMBER < 0x10100003L
Packit Service 2ea82d
#define EVP_PKEY_get0_RSA(key) ((key)->pkey.rsa)
Packit Service 2ea82d
#endif
Packit Service 2ea82d
Packit Service 2ea82d
static int do_wait(pid_t pids[], int num)
Packit Service 2ea82d
{
Packit Service 2ea82d
    int i;
Packit Service 2ea82d
    int status = 0;
Packit Service 2ea82d
Packit Service 2ea82d
    for (i = 0; i < num; i++) {
Packit Service 2ea82d
        waitpid(pids[i], &status, 0);
Packit Service 2ea82d
        if (WIFEXITED(status)) {
Packit Service 2ea82d
            printf("child %d exited with status %d\n", pids[i], WEXITSTATUS(status));
Packit Service 2ea82d
            return (WEXITSTATUS(status));
Packit Service 2ea82d
        }
Packit Service 2ea82d
        if (WIFSIGNALED(status)) {
Packit Service 2ea82d
            fprintf(stderr, "Child %d terminated by signal #%d\n", pids[i],
Packit Service 2ea82d
                    WTERMSIG(status));
Packit Service 2ea82d
            return (WTERMSIG(status));
Packit Service 2ea82d
        }
Packit Service 2ea82d
        else {
Packit Service 2ea82d
            perror("waitpid");
Packit Service 2ea82d
        }
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    return 0;
Packit Service 2ea82d
}
Packit Service 2ea82d
Packit Service 2ea82d
static int spawn_processes(int num)
Packit Service 2ea82d
{
Packit Service 2ea82d
    int i;
Packit Service 2ea82d
    int chld_ret = 0;
Packit Service 2ea82d
    pid_t *pids;
Packit Service 2ea82d
    pid_t pid;
Packit Service 2ea82d
Packit Service 2ea82d
    sigset_t set, oldset;
Packit Service 2ea82d
    int signal;
Packit Service 2ea82d
Packit Service 2ea82d
    sigemptyset(&set);
Packit Service 2ea82d
    sigaddset(&set, SIGUSR1);
Packit Service 2ea82d
Packit Service 2ea82d
    /* If only 1 process was requested, no more processes are required */
Packit Service 2ea82d
    if (num <= 1) {
Packit Service 2ea82d
        return 0;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    pids = (pid_t *)malloc(num * sizeof(pid_t));
Packit Service 2ea82d
    if (pids == NULL) {
Packit Service 2ea82d
        exit(12);
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    /* Spawn (num - 1) new processes to get a total of num processes */
Packit Service 2ea82d
    for (i = 0; i < (num - 1); i++) {
Packit Service 2ea82d
        pid = fork();
Packit Service 2ea82d
        switch (pid) {
Packit Service 2ea82d
            case -1: /* failed */
Packit Service 2ea82d
                perror("fork");
Packit Service 2ea82d
                do_wait(pids, i);
Packit Service 2ea82d
                free(pids);
Packit Service 2ea82d
                exit(5);
Packit Service 2ea82d
            case 0: /* child */
Packit Service 2ea82d
                printf("Remove or add a device to try to cause an error\n");
Packit Service 2ea82d
                printf("Waiting for signal SIGUSR1\n");
Packit Service 2ea82d
                sigprocmask(SIG_BLOCK, &set, &oldset);
Packit Service 2ea82d
                sigwait(&set, &signal);
Packit Service 2ea82d
                sigprocmask(SIG_SETMASK, &oldset, NULL);
Packit Service 2ea82d
                free(pids);
Packit Service 2ea82d
                return 0;
Packit Service 2ea82d
            default: /* parent */
Packit Service 2ea82d
                pids[i] = pid;
Packit Service 2ea82d
                printf("spawned %d\n", pid);
Packit Service 2ea82d
        }
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    /* Wait for the created processes */
Packit Service 2ea82d
    chld_ret = do_wait(pids, (num - 1));
Packit Service 2ea82d
Packit Service 2ea82d
    free(pids);
Packit Service 2ea82d
Packit Service 2ea82d
    return chld_ret;
Packit Service 2ea82d
}
Packit Service 2ea82d
Packit Service 2ea82d
static void error_queue(const char *name, int pid)
Packit Service 2ea82d
{
Packit Service 2ea82d
    if (ERR_peek_last_error()) {
Packit Service 2ea82d
        fprintf(stderr, "pid %d: %s generated errors:\n", pid, name);
Packit Service 2ea82d
        ERR_print_errors_fp(stderr);
Packit Service 2ea82d
    }
Packit Service 2ea82d
}
Packit Service 2ea82d
Packit Service 2ea82d
static void usage(char *arg)
Packit Service 2ea82d
{
Packit Service 2ea82d
    printf("usage: %s (Key PKCS#11 URL) [opt: PKCS#11 module path]\n",
Packit Service 2ea82d
            arg);
Packit Service 2ea82d
}
Packit Service 2ea82d
Packit Service 2ea82d
int main(int argc, char *argv[])
Packit Service 2ea82d
{
Packit Service 2ea82d
    const EVP_MD *digest_algo = NULL;
Packit Service 2ea82d
    EVP_PKEY *pkey = NULL;
Packit Service 2ea82d
    EVP_MD_CTX *md_ctx = NULL;
Packit Service 2ea82d
    ENGINE *engine = NULL;
Packit Service 2ea82d
    unsigned char random[RANDOM_SIZE], signature[MAX_SIGSIZE];
Packit Service 2ea82d
    unsigned int siglen = MAX_SIGSIZE;
Packit Service 2ea82d
Packit Service 2ea82d
    int ret, num_processes = 2;
Packit Service 2ea82d
    pid_t pid;
Packit Service 2ea82d
Packit Service 2ea82d
    int rv = 1;
Packit Service 2ea82d
Packit Service 2ea82d
    /* Check arguments */
Packit Service 2ea82d
    if (argc < 2) {
Packit Service 2ea82d
        fprintf(stderr, "Missing required arguments\n");
Packit Service 2ea82d
        usage(argv[0]);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    if (argc > 4) {
Packit Service 2ea82d
        fprintf(stderr, "Too many arguments\n");
Packit Service 2ea82d
        usage(argv[0]);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    /* Check PKCS#11 URL */
Packit Service 2ea82d
    if (strncmp(argv[1], "pkcs11:", 7)) {
Packit Service 2ea82d
        fprintf(stderr, "fatal: invalid PKCS#11 URL\n");
Packit Service 2ea82d
        usage(argv[0]);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    pid = getpid();
Packit Service 2ea82d
    printf("pid %d is the parent\n", pid);
Packit Service 2ea82d
Packit Service 2ea82d
    /* Load configuration file, if provided */
Packit Service 2ea82d
    if (argc >= 3) {
Packit Service 2ea82d
        ret = CONF_modules_load_file(argv[2], "engines", 0);
Packit Service 2ea82d
        if (ret <= 0) {
Packit Service 2ea82d
            fprintf(stderr, "cannot load %s\n", argv[2]);
Packit Service 2ea82d
            error_queue("CONF_modules_load_file", pid);
Packit Service 2ea82d
            goto failed;
Packit Service 2ea82d
        }
Packit Service 2ea82d
        ENGINE_add_conf_module();
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    ENGINE_add_conf_module();
Packit Service 2ea82d
#if OPENSSL_VERSION_NUMBER>=0x10100000
Packit Service 2ea82d
	OPENSSL_init_crypto(OPENSSL_INIT_ADD_ALL_CIPHERS \
Packit Service 2ea82d
		| OPENSSL_INIT_ADD_ALL_DIGESTS \
Packit Service 2ea82d
		| OPENSSL_INIT_LOAD_CONFIG, NULL);
Packit Service 2ea82d
#else
Packit Service 2ea82d
    OpenSSL_add_all_algorithms();
Packit Service 2ea82d
    ERR_load_crypto_strings();
Packit Service 2ea82d
#endif
Packit Service 2ea82d
    ERR_clear_error();
Packit Service 2ea82d
    ENGINE_load_builtin_engines();
Packit Service 2ea82d
Packit Service 2ea82d
    /* Get structural reference */
Packit Service 2ea82d
    engine = ENGINE_by_id("pkcs11");
Packit Service 2ea82d
    if (engine == NULL) {
Packit Service 2ea82d
        fprintf(stderr, "fatal: engine \"pkcs11\" not available\n");
Packit Service 2ea82d
        error_queue("ENGINE_by_id", pid);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    /* Set the used  */
Packit Service 2ea82d
    if (argc >= 4) {
Packit Service 2ea82d
        ENGINE_ctrl_cmd(engine, "MODULE_PATH", 0, argv[3], NULL, 1);
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    /* Initialize to get the engine functional reference */
Packit Service 2ea82d
    if (ENGINE_init(engine)) {
Packit Service 2ea82d
        pkey = ENGINE_load_private_key(engine, argv[1], 0, 0);
Packit Service 2ea82d
        if (pkey == NULL) {
Packit Service 2ea82d
            error_queue("ENGINE_load_private_key", pid);
Packit Service 2ea82d
            goto failed;
Packit Service 2ea82d
        }
Packit Service 2ea82d
Packit Service 2ea82d
        ENGINE_free(engine);
Packit Service 2ea82d
        engine = NULL;
Packit Service 2ea82d
    }
Packit Service 2ea82d
    else {
Packit Service 2ea82d
        error_queue("ENGINE_init", pid);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    /* Spawn processes and check child return */
Packit Service 2ea82d
    if (spawn_processes(num_processes)) {
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
    pid = getpid();
Packit Service 2ea82d
Packit Service 2ea82d
    /* Generate random data */
Packit Service 2ea82d
    if (!RAND_bytes(random, RANDOM_SIZE)){
Packit Service 2ea82d
        error_queue("RAND_bytes", pid);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    /* Create context to sign the random data */
Packit Service 2ea82d
    digest_algo = EVP_get_digestbyname("sha256");
Packit Service 2ea82d
    md_ctx = EVP_MD_CTX_create();
Packit Service 2ea82d
    if (EVP_DigestInit(md_ctx, digest_algo) <= 0) {
Packit Service 2ea82d
        error_queue("EVP_DigestInit", pid);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    EVP_SignInit(md_ctx, digest_algo);
Packit Service 2ea82d
    if (EVP_SignUpdate(md_ctx, random, RANDOM_SIZE) <= 0) {
Packit Service 2ea82d
        error_queue("EVP_SignUpdate", pid);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    if (EVP_SignFinal(md_ctx, signature, &siglen, pkey) <= 0) {
Packit Service 2ea82d
        error_queue("EVP_SignFinal", pid);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
    EVP_MD_CTX_destroy(md_ctx);
Packit Service 2ea82d
Packit Service 2ea82d
    printf("pid %d: %u-byte signature created\n", pid, siglen);
Packit Service 2ea82d
Packit Service 2ea82d
    /* Now verify the result */
Packit Service 2ea82d
    md_ctx = EVP_MD_CTX_create();
Packit Service 2ea82d
    if (EVP_DigestInit(md_ctx, digest_algo) <= 0) {
Packit Service 2ea82d
        error_queue("EVP_DigestInit", pid);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    EVP_VerifyInit(md_ctx, digest_algo);
Packit Service 2ea82d
    if (EVP_VerifyUpdate(md_ctx, random, RANDOM_SIZE) <= 0) {
Packit Service 2ea82d
        error_queue("EVP_VerifyUpdate", pid);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
Packit Service 2ea82d
    if (EVP_VerifyFinal(md_ctx, signature, siglen, pkey) <= 0) {
Packit Service 2ea82d
        error_queue("EVP_VerifyFinal", pid);
Packit Service 2ea82d
        goto failed;
Packit Service 2ea82d
    }
Packit Service 2ea82d
    printf("pid %d: Signature matched\n", pid);
Packit Service 2ea82d
Packit Service 2ea82d
    rv = 0;
Packit Service 2ea82d
Packit Service 2ea82d
failed:
Packit Service 2ea82d
    if (md_ctx != NULL)
Packit Service 2ea82d
        EVP_MD_CTX_destroy(md_ctx);
Packit Service 2ea82d
    if (pkey != NULL)
Packit Service 2ea82d
        EVP_PKEY_free(pkey);
Packit Service 2ea82d
    if (engine != NULL)
Packit Service 2ea82d
        ENGINE_free(engine);
Packit Service 2ea82d
Packit Service 2ea82d
    return rv;
Packit Service 2ea82d
}