Newer
Older
mbed-os / components / TARGET_PSA / TARGET_MBED_SPM / COMPONENT_SPE / spm_client.c
@Oren Cohen Oren Cohen on 3 Mar 2019 10 KB Fix SPM warnings
/* Copyright (c) 2017-2018 ARM Limited
 *
 * SPDX-License-Identifier: Apache-2.0
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#include <string.h>
#include "mbed_toolchain.h"
#include "rtx_os.h"
#include "spm_client.h"
#include "spm_messages.h"
#include "spm_internal.h"
#include "spm_panic.h"
#include "handles_manager.h"

extern spm_db_t g_spm;

static inline spm_rot_service_t *rot_service_in_partition_get_by_sid(spm_partition_t *partition, uint32_t sid)
{
    for (uint32_t i = 0; i < partition->rot_services_count; ++i) {
        spm_rot_service_t *rot_service = &(partition->rot_services[i]);
        if (rot_service->sid == sid) {
            return rot_service;
        }
    }

    return NULL;
}

static inline spm_rot_service_t *rot_service_get(uint32_t sid)
{
    for (uint32_t i = 0; i < g_spm.partition_count; ++i) {
        spm_rot_service_t *rot_service = rot_service_in_partition_get_by_sid(&(g_spm.partitions[i]), sid);
        if (NULL != rot_service) {
            return rot_service;
        }
    }

    return NULL;
}

static inline void spm_validate_connection_allowed(spm_rot_service_t *target, spm_partition_t *source)
{
    if ((NULL == source) && (false == target->allow_nspe)) {
        SPM_PANIC("SID 0x%lx is not allowed to be called from NSPE\n", target->sid);
    }

    if (NULL != source) {
        if (NULL == source->extern_sids) {
            SPM_PANIC("Partition %ld did not declare extern functions\n", source->partition_id);
        }

        for (uint32_t i = 0; i < source->extern_sids_count; i++) {
            if (source->extern_sids[i] == target->sid) {
                return;
            }
        }

        SPM_PANIC("SID 0x%lx is not in partition %ld extern functions list\n", target->sid, source->partition_id);
    }
}

static inline psa_handle_t create_channel_handle(void *handle_mem, int32_t friend_pid)
{
    return psa_hndl_mgr_handle_create(&(g_spm.channels_handle_mgr), handle_mem, friend_pid);
}

static inline spm_ipc_channel_t *get_channel_from_handle(psa_handle_t handle)
{
    return (spm_ipc_channel_t *)psa_hndl_mgr_handle_get_mem(&(g_spm.channels_handle_mgr), handle);
}

static void spm_rot_service_queue_enqueue(spm_rot_service_t *rot_service, spm_ipc_channel_t *item)
{
    osStatus_t os_status = osMutexAcquire(rot_service->partition->mutex, osWaitForever);
    SPM_ASSERT(osOK == os_status);
    PSA_UNUSED(os_status);

    if (rot_service->queue.tail == NULL) {
        rot_service->queue.head = item;
    } else {
        rot_service->queue.tail->next = item;
    }

    rot_service->queue.tail = item;

    uint32_t flags = osThreadFlagsSet(rot_service->partition->thread_id, rot_service->mask);
    // osThreadFlagsSet() sets the msb on failure.
    // flags is not allowed to be 0 since only dequeue operation can clear the flags,
    // and both operations (enqueue and dequeue) are mutex protected.
    SPM_ASSERT((flags & SPM_CMSIS_RTOS_ERROR_BIT_MSK) == 0);
    SPM_ASSERT(flags & rot_service->mask);
    PSA_UNUSED(flags);

    os_status = osMutexRelease(rot_service->partition->mutex);
    SPM_ASSERT(osOK == os_status);
}

void psa_connect_async(uint32_t sid, spm_pending_connect_msg_t *msg)
{
    SPM_ASSERT(msg != NULL);

    spm_rot_service_t *dst_rot_service = rot_service_get(sid);
    if (NULL == dst_rot_service) {
        SPM_PANIC("SID 0x%lx is invalid!\n", sid);
    }

    if (((dst_rot_service->min_version_policy == PSA_MINOR_VERSION_POLICY_RELAXED) && (msg->min_version > dst_rot_service->min_version)) ||
            ((dst_rot_service->min_version_policy == PSA_MINOR_VERSION_POLICY_STRICT) && (msg->min_version != dst_rot_service->min_version))) {
        SPM_PANIC("minor version %lu does not comply with sid %lu minor version %lu and minor policy %lu",
                  msg->min_version, dst_rot_service->sid, dst_rot_service->min_version, dst_rot_service->min_version_policy);
    }

    spm_partition_t *origin_partition = get_active_partition();
    spm_validate_connection_allowed(dst_rot_service, origin_partition);

    if (!is_buffer_accessible(msg, sizeof(*msg), origin_partition)) {
        SPM_PANIC("Pending connect message is inaccessible\n");
    }

    // Allocating from SPM-Core internal memory
    spm_ipc_channel_t *channel = (spm_ipc_channel_t *)osMemoryPoolAlloc(g_spm.channel_mem_pool, PSA_POLL);
    if (NULL == channel) {
        msg->rc = PSA_CONNECTION_REFUSED;
        if (origin_partition != NULL) {
            osStatus_t os_status = osSemaphoreRelease(msg->completion_sem_id);
            SPM_ASSERT(osOK == os_status);
            PSA_UNUSED(os_status);
        } else {
            nspe_done(msg->completion_sem_id);
        }

        return;
    }

    // Create the handle in the user message so we could destroy it in case of failure.
    msg->rc = (psa_status_t)create_channel_handle(channel, dst_rot_service->partition->partition_id);

    // NOTE: all struct fields must be initialized as the allocated memory is not zeroed.
    channel->state = SPM_CHANNEL_STATE_CONNECTING;
    channel->src_partition = origin_partition;
    channel->dst_rot_service = dst_rot_service;
    channel->msg_ptr = msg;
    channel->msg_type = PSA_IPC_CONNECT;
    channel->rhandle = NULL;
    channel->next = NULL;
    channel->is_dropped = FALSE;

    spm_rot_service_queue_enqueue(dst_rot_service, channel);
}

psa_handle_t psa_connect(uint32_t sid, uint32_t minor_version)
{
    osRtxSemaphore_t msg_sem_storage = {0};
    const osSemaphoreAttr_t msg_sem_attr = {
        .name = NULL,
        .attr_bits = 0,
        .cb_mem = &msg_sem_storage,
        .cb_size = sizeof(msg_sem_storage),
    };

    spm_pending_connect_msg_t msg = {
        .min_version = minor_version,
        .rc = PSA_NULL_HANDLE,
        .completion_sem_id = osSemaphoreNew(
            SPM_COMPLETION_SEM_MAX_COUNT,
            SPM_COMPLETION_SEM_INITIAL_COUNT,
            &msg_sem_attr)
    };

    if (NULL == msg.completion_sem_id) {
        SPM_PANIC("could not create a semaphore for connect message");
    }

    psa_connect_async(sid, &msg);

    osStatus_t os_status = osSemaphoreAcquire(msg.completion_sem_id, osWaitForever);
    SPM_ASSERT(osOK == os_status);

    os_status = osSemaphoreDelete(msg.completion_sem_id);
    SPM_ASSERT(osOK == os_status);

    PSA_UNUSED(os_status);

    return (psa_handle_t)msg.rc;
}

void psa_call_async(psa_handle_t handle, spm_pending_call_msg_t *msg)
{
    SPM_ASSERT(msg != NULL);
    spm_ipc_channel_t *channel = get_channel_from_handle(handle);
    SPM_ASSERT(channel != NULL);

    if (!is_buffer_accessible(msg, sizeof(*msg), channel->src_partition)) {
        SPM_PANIC("Pending call message is inaccessible\n");
    }

    if (channel->is_dropped == TRUE) {
        msg->rc = PSA_DROP_CONNECTION;

        if (channel->src_partition == NULL) {
            nspe_done(msg->completion_sem_id);
        } else {
            osStatus_t os_status = osSemaphoreRelease(msg->completion_sem_id);
            SPM_ASSERT(osOK == os_status);
            PSA_UNUSED(os_status);
        }

        return;
    }

    channel_state_switch(&channel->state,
                         SPM_CHANNEL_STATE_IDLE, SPM_CHANNEL_STATE_PENDING);

    channel->msg_ptr = msg;
    channel->msg_type = PSA_IPC_CALL;

    validate_iovec(msg->in_vec, msg->in_vec_size, msg->out_vec, msg->out_vec_size);
    spm_rot_service_queue_enqueue(channel->dst_rot_service, channel);
}

psa_status_t psa_call(
    psa_handle_t handle,
    const psa_invec *in_vec,
    size_t in_len,
    const psa_outvec *out_vec,
    size_t out_len
)
{
    osRtxSemaphore_t msg_sem_storage = {0};
    const osSemaphoreAttr_t msg_sem_attr = {
        .name = NULL,
        .attr_bits = 0,
        .cb_mem = &msg_sem_storage,
        .cb_size = sizeof(msg_sem_storage),
    };

    spm_pending_call_msg_t msg = {
        .in_vec = in_vec,
        .in_vec_size = in_len,
        .out_vec = out_vec,
        .out_vec_size = out_len,
        .rc = PSA_SUCCESS,
        .completion_sem_id = osSemaphoreNew(
            SPM_COMPLETION_SEM_MAX_COUNT,
            SPM_COMPLETION_SEM_INITIAL_COUNT,
            &msg_sem_attr)
    };

    if (NULL == msg.completion_sem_id) {
        SPM_PANIC("could not create a semaphore for connect message");
    }

    psa_call_async(handle, &msg);

    osStatus_t os_status = osSemaphoreAcquire(msg.completion_sem_id, osWaitForever);
    SPM_ASSERT(osOK == os_status);

    os_status = osSemaphoreDelete(msg.completion_sem_id);
    SPM_ASSERT(osOK == os_status);

    PSA_UNUSED(os_status);

    return (psa_status_t)msg.rc;
}

void psa_close_async(psa_handle_t handle, spm_pending_close_msg_t *msg)
{
    SPM_ASSERT(msg != NULL);

    spm_ipc_channel_t *channel = get_channel_from_handle(handle);
    SPM_ASSERT(channel != NULL);

    if (!is_buffer_accessible(msg, sizeof(*msg), channel->src_partition)) {
        SPM_PANIC("Pending close message is inaccessible\n");
    }

    channel_state_assert(&(channel->state), SPM_CHANNEL_STATE_IDLE);

    channel->msg_ptr  = msg;
    channel->msg_type = PSA_IPC_DISCONNECT;

    spm_rot_service_queue_enqueue(channel->dst_rot_service, channel);
}

void psa_close(psa_handle_t handle)
{
    if (handle == PSA_NULL_HANDLE) {
        // Invalid handles will fail inside handle manager [called from psa_close_async()]
        return;
    }

    osRtxSemaphore_t msg_sem_storage = {0};
    const osSemaphoreAttr_t msg_sem_attr = {
        .name      = NULL,
        .attr_bits = 0,
        .cb_mem    = &msg_sem_storage,
        .cb_size   = sizeof(msg_sem_storage),
    };

    spm_pending_close_msg_t msg = {
        .handle = handle,
        .completion_sem_id = osSemaphoreNew(SPM_COMPLETION_SEM_MAX_COUNT,
                                            SPM_COMPLETION_SEM_INITIAL_COUNT,
                                            &msg_sem_attr
                                           ),

    };

    if (NULL == msg.completion_sem_id) {
        SPM_PANIC("Could not create a semaphore for close message");
    }

    psa_close_async(handle, &msg);

    osStatus_t os_status = osSemaphoreAcquire(msg.completion_sem_id, osWaitForever);
    SPM_ASSERT(osOK == os_status);

    os_status = osSemaphoreDelete(msg.completion_sem_id);
    SPM_ASSERT(osOK == os_status);

    PSA_UNUSED(os_status);
}

uint32_t psa_framework_version(void)
{
    return (uint32_t)PSA_FRAMEWORK_VERSION;
}


uint32_t psa_version(uint32_t sid)
{
    uint32_t version = PSA_VERSION_NONE;
    spm_rot_service_t *service = rot_service_get(sid);
    if (service != NULL) {
        if ((get_active_partition() != NULL) || (service->allow_nspe)) {
            version = service->min_version;
        }
    }

    return version;
}